import React from 'react'; import { Switch } from 'react-native'; import { connect } from 'react-redux'; import { StackNavigationOptions } from '@react-navigation/stack'; import { Subscription } from 'rxjs'; import I18n from '../i18n'; import { TSupportedThemes, withTheme } from '../theme'; import StatusBar from '../containers/StatusBar'; import * as List from '../containers/List'; import database from '../lib/database'; import { changePasscode, checkHasPasscode, supportedBiometryLabel } from '../utils/localAuthentication'; import { BIOMETRY_ENABLED_KEY, DEFAULT_AUTO_LOCK, themes, SWITCH_TRACK_COLOR } from '../lib/constants'; import SafeAreaView from '../containers/SafeAreaView'; import { events, logEvent } from '../utils/log'; import userPreferences from '../lib/methods/userPreferences'; import { IApplicationState, TServerModel } from '../definitions'; const DEFAULT_BIOMETRY = false; interface IItem { title: string; value: number; disabled?: boolean; } interface IScreenLockConfigViewProps { theme: TSupportedThemes; server: string; Force_Screen_Lock: boolean; Force_Screen_Lock_After: number; } interface IScreenLockConfigViewState { autoLock: boolean; autoLockTime?: number | null; biometry: boolean; biometryLabel: string | null; } class ScreenLockConfigView extends React.Component { private serverRecord?: TServerModel; private observable?: Subscription; static navigationOptions = (): StackNavigationOptions => ({ title: I18n.t('Screen_lock') }); constructor(props: IScreenLockConfigViewProps) { super(props); this.state = { autoLock: false, autoLockTime: null, biometry: DEFAULT_BIOMETRY, biometryLabel: null }; this.init(); } componentWillUnmount() { if (this.observable && this.observable.unsubscribe) { this.observable.unsubscribe(); } } defaultAutoLockOptions = [ { title: I18n.t('Local_authentication_auto_lock_60'), value: 60 }, { title: I18n.t('Local_authentication_auto_lock_300'), value: 300 }, { title: I18n.t('Local_authentication_auto_lock_900'), value: 900 }, { title: I18n.t('Local_authentication_auto_lock_1800'), value: 1800 }, { title: I18n.t('Local_authentication_auto_lock_3600'), value: 3600 } ]; init = async () => { const { server } = this.props; const serversDB = database.servers; const serversCollection = serversDB.get('servers'); const hasBiometry = userPreferences.getBool(BIOMETRY_ENABLED_KEY) ?? DEFAULT_BIOMETRY; try { this.serverRecord = await serversCollection.find(server); this.setState({ autoLock: this.serverRecord?.autoLock, autoLockTime: this.serverRecord?.autoLockTime === null ? DEFAULT_AUTO_LOCK : this.serverRecord?.autoLockTime, biometry: hasBiometry }); } catch (error) { // Do nothing } const biometryLabel = await supportedBiometryLabel(); this.setState({ biometryLabel }); }; save = async () => { logEvent(events.SLC_SAVE_SCREEN_LOCK); const { autoLock, autoLockTime } = this.state; const serversDB = database.servers; await serversDB.write(async () => { await this.serverRecord?.update(record => { record.autoLock = autoLock; record.autoLockTime = autoLockTime === null ? DEFAULT_AUTO_LOCK : autoLockTime; }); }); }; changePasscode = async ({ force }: { force: boolean }) => { logEvent(events.SLC_CHANGE_PASSCODE); await changePasscode({ force }); }; toggleAutoLock = () => { logEvent(events.SLC_TOGGLE_AUTOLOCK); this.setState( ({ autoLock }) => ({ autoLock: !autoLock, autoLockTime: DEFAULT_AUTO_LOCK }), async () => { const { autoLock } = this.state; if (autoLock) { try { await checkHasPasscode({ force: false }); } catch { this.toggleAutoLock(); } } this.save(); } ); }; toggleBiometry = () => { logEvent(events.SLC_TOGGLE_BIOMETRY); this.setState( ({ biometry }) => ({ biometry: !biometry }), () => { const { biometry } = this.state; userPreferences.setBool(BIOMETRY_ENABLED_KEY, biometry); } ); }; isSelected = (value: number) => { const { autoLockTime } = this.state; return autoLockTime === value; }; changeAutoLockTime = (autoLockTime: number) => { logEvent(events.SLC_CHANGE_AUTOLOCK_TIME); this.setState({ autoLockTime }, () => this.save()); }; renderIcon = () => { const { theme } = this.props; return ; }; renderItem = ({ item }: { item: IItem }) => { const { title, value, disabled } = item; return ( <> this.changeAutoLockTime(value)} right={() => (this.isSelected(value) ? this.renderIcon() : null)} disabled={disabled} translateTitle={false} /> ); }; renderAutoLockSwitch = () => { const { autoLock } = this.state; const { Force_Screen_Lock } = this.props; return ( ); }; renderBiometrySwitch = () => { const { biometry } = this.state; return ; }; renderAutoLockItems = () => { const { autoLock, autoLockTime } = this.state; const { Force_Screen_Lock_After, Force_Screen_Lock } = this.props; if (!autoLock) { return null; } let items: IItem[] = this.defaultAutoLockOptions; if (Force_Screen_Lock && Force_Screen_Lock_After > 0) { items = [ { title: I18n.t('After_seconds_set_by_admin', { seconds: Force_Screen_Lock_After }), value: Force_Screen_Lock_After, disabled: true } ]; // if Force_Screen_Lock is disabled and autoLockTime is a value that isn't on our defaultOptions we'll show it } else if (Force_Screen_Lock_After === autoLockTime && !items.find(item => item.value === autoLockTime)) { items.push({ title: I18n.t('After_seconds_set_by_admin', { seconds: Force_Screen_Lock_After }), value: Force_Screen_Lock_After }); } return ( <>{items.map(item => this.renderItem({ item }))} ); }; renderBiometry = () => { const { autoLock, biometryLabel } = this.state; if (!autoLock || !biometryLabel) { return null; } return ( this.renderBiometrySwitch()} translateTitle={false} /> ); }; render() { const { autoLock } = this.state; return ( this.renderAutoLockSwitch()} /> {autoLock ? ( <> ) : null} {this.renderBiometry()} {this.renderAutoLockItems()} ); } } const mapStateToProps = (state: IApplicationState) => ({ server: state.server.server, Force_Screen_Lock: state.settings.Force_Screen_Lock as boolean, Force_Screen_Lock_After: state.settings.Force_Screen_Lock_After as number }); export default connect(mapStateToProps)(withTheme(ScreenLockConfigView));