import * as React from 'react'; import { useState, useEffect, useCallback, useMemo } from 'react'; import { TouchableHighlight, StyleSheet, TextStyle, AccessibilityInfo } from 'react-native'; import Icon from './Icon'; import { _ } from '@joplin/lib/locale'; interface Props { checked: boolean; accessibilityLabel?: string; accessibilityHint?: string; onChange?: (checked: boolean)=> void; style?: TextStyle; iconStyle?: TextStyle; } const useStyles = (baseStyles: TextStyle|undefined, iconStyle: TextStyle|undefined) => { return useMemo(() => { return StyleSheet.create({ container: { ...(baseStyles ?? {}), justifyContent: 'center', alignItems: 'center', }, icon: { fontSize: 20, height: 22, color: baseStyles?.color, ...iconStyle, }, }); }, [baseStyles, iconStyle]); }; const Checkbox: React.FC = props => { const [checked, setChecked] = useState(props.checked); useEffect(() => { setChecked(props.checked); }, [props.checked]); const onPress = useCallback(() => { setChecked(checked => { const newChecked = !checked; props.onChange?.(newChecked); // An .announceForAccessibility is necessary here because VoiceOver doesn't announce this // change on its own, even though we change [accessibilityState]. if (newChecked) { AccessibilityInfo.announceForAccessibility(_('Checked')); } else { AccessibilityInfo.announceForAccessibility(_('Unchecked')); } return newChecked; }); }, [props.onChange]); const iconName = checked ? 'ionicon checkbox-outline' : 'ionicon square-outline'; const styles = useStyles(props.style, props.iconStyle); const accessibilityState = useMemo(() => ({ checked, }), [checked]); return ( ); }; export default Checkbox;