import React, { forwardRef, useImperativeHandle, useRef, useState } from 'react'; import { Col, Grid, Row } from 'react-native-easy-grid'; import range from 'lodash/range'; import { View } from 'react-native'; import * as Animatable from 'react-native-animatable'; import * as Haptics from 'expo-haptics'; import styles from './styles'; import Button from './Button'; import Dots from './Dots'; import { TYPE } from '../constants'; import { themes } from '../../../constants/colors'; import { PASSCODE_LENGTH } from '../../../constants/localAuthentication'; import { useTheme } from '../../../theme'; import LockIcon from './LockIcon'; import Title from './Title'; import Subtitle from './Subtitle'; interface IPasscodeBase { type: string; previousPasscode?: string; title: string; subtitle?: string; showBiometry?: boolean; onEndProcess: Function; onError?: Function; onBiometryPress?(): void; } export interface IBase { clearPasscode: () => void; wrongPasscode: () => void; animate: (animation: Animatable.Animation, duration?: number) => void; } const Base = forwardRef( ({ type, onEndProcess, previousPasscode, title, subtitle, onError, showBiometry, onBiometryPress }, ref) => { const { theme } = useTheme(); const rootRef = useRef(null); const dotsRef = useRef(null); const [passcode, setPasscode] = useState(''); const clearPasscode = () => setPasscode(''); const wrongPasscode = () => { clearPasscode(); dotsRef?.current?.shake?.(500); Haptics.notificationAsync(Haptics.NotificationFeedbackType.Error); }; const animate = (animation: Animatable.Animation, duration = 500) => { rootRef?.current?.[animation]?.(duration); }; const onPressNumber = (text: string) => setPasscode(p => { const currentPasscode = p + text; if (currentPasscode?.length === PASSCODE_LENGTH) { switch (type) { case TYPE.CHOOSE: onEndProcess(currentPasscode); break; case TYPE.CONFIRM: if (currentPasscode !== previousPasscode) { onError?.(); } else { onEndProcess(currentPasscode); } break; case TYPE.ENTER: onEndProcess(currentPasscode); break; default: break; } } return currentPasscode; }); const onPressDelete = () => setPasscode(p => { if (p?.length > 0) { const newPasscode = p.slice(0, -1); return newPasscode; } return ''; }); useImperativeHandle(ref, () => ({ wrongPasscode, animate, clearPasscode })); return ( {subtitle ? <Subtitle text={subtitle} /> : null} <Row style={styles.row}> <Animatable.View ref={dotsRef}> <Dots passcode={passcode} length={PASSCODE_LENGTH} /> </Animatable.View> </Row> <Row style={[styles.row, styles.buttonRow]}> {range(1, 4).map(i => ( <Col key={i} style={styles.colButton}> <Button text={i.toString()} onPress={onPressNumber} /> </Col> ))} </Row> <Row style={[styles.row, styles.buttonRow]}> {range(4, 7).map(i => ( <Col key={i} style={styles.colButton}> <Button text={i.toString()} onPress={onPressNumber} /> </Col> ))} </Row> <Row style={[styles.row, styles.buttonRow]}> {range(7, 10).map(i => ( <Col key={i} style={styles.colButton}> <Button text={i.toString()} onPress={onPressNumber} /> </Col> ))} </Row> <Row style={[styles.row, styles.buttonRow]}> {showBiometry ? ( <Col style={styles.colButton}> <Button icon='fingerprint' onPress={onBiometryPress} /> </Col> ) : ( <Col style={styles.colButton} /> )} <Col style={styles.colButton}> <Button text='0' onPress={onPressNumber} /> </Col> <Col style={styles.colButton}> <Button icon='backspace' onPress={onPressDelete} /> </Col> </Row> </Grid> </Animatable.View> ); } ); export default Base;