joplin/packages/app-mobile/components/Checkbox.tsx

82 lines
2.1 KiB
TypeScript

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> = 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 (
<TouchableHighlight
onPress={onPress}
style={styles.container}
accessibilityRole="checkbox"
accessibilityState={accessibilityState}
accessibilityLabel={props.accessibilityLabel ?? ''}
accessibilityHint={props.accessibilityHint}
// Web requires aria-checked
aria-checked={checked}
>
<Icon name={iconName} style={styles.icon} accessibilityLabel={null} />
</TouchableHighlight>
);
};
export default Checkbox;