Split up state for SettingsPage

This improved performance when visually
toggling an option
This commit is contained in:
Brandon Presley 2022-11-01 18:58:09 +13:00
parent 6ac84d1d32
commit 949b435853
1 changed files with 110 additions and 123 deletions

View File

@ -1,6 +1,6 @@
import {Picker} from '@react-native-picker/picker' import {Picker} from '@react-native-picker/picker'
import {useFocusEffect} from '@react-navigation/native' import {useFocusEffect} from '@react-navigation/native'
import {useCallback, useEffect, useMemo, useState} from 'react' import {useCallback, useMemo, useState} from 'react'
import {DeviceEventEmitter, NativeModules, ScrollView} from 'react-native' import {DeviceEventEmitter, NativeModules, ScrollView} from 'react-native'
import DocumentPicker from 'react-native-document-picker' import DocumentPicker from 'react-native-document-picker'
import {Button} from 'react-native-paper' import {Button} from 'react-native-paper'
@ -12,35 +12,45 @@ import DrawerHeader from './DrawerHeader'
import Input from './input' import Input from './input'
import Page from './Page' import Page from './Page'
import Select from './Select' import Select from './Select'
import Settings from './settings'
import Switch from './Switch' import Switch from './Switch'
import {toast} from './toast' import {toast} from './toast'
import {useTheme} from './use-theme'
export default function SettingsPage() { export default function SettingsPage() {
const [battery, setBattery] = useState(false) const [battery, setBattery] = useState(false)
const [ignoring, setIgnoring] = useState(false) const [ignoring, setIgnoring] = useState(false)
const [term, setTerm] = useState('') const [term, setTerm] = useState('')
const [settings, setSettings] = useState<Settings>({} as Settings) const [vibrate, setVibrate] = useState(false)
const [alarm, setAlarm] = useState(false)
useEffect(() => { const [sound, setSound] = useState('')
console.log(`${SettingsPage.name}.useEffect:`, {settings}) const [notify, setNotify] = useState(false)
}, [settings]) const [images, setImages] = useState(false)
const [showUnit, setShowUnit] = useState(false)
const [steps, setSteps] = useState(false)
const [date, setDate] = useState('%Y-%m-%d %H:%M')
const {theme, setTheme, color, setColor} = useTheme()
const [showDate, setShowDate] = useState(false)
const [showSets, setShowSets] = useState(false)
const [noSound, setNoSound] = useState(false)
useFocusEffect( useFocusEffect(
useCallback(() => { useCallback(() => {
NativeModules.AlarmModule.ignoringBattery(setIgnoring) NativeModules.AlarmModule.ignoringBattery(setIgnoring)
settingsRepo.findOne({where: {}}).then(setSettings) settingsRepo.findOne({where: {}}).then(settings => {
setAlarm(settings.alarm)
setVibrate(settings.vibrate)
setSound(settings.sound)
setNotify(settings.notify)
setImages(settings.images)
setShowUnit(settings.showUnit)
setSteps(settings.steps)
setDate(settings.date)
setShowDate(settings.showDate)
setShowSets(settings.showSets)
})
}, []), }, []),
) )
const update = useCallback(
(value: boolean, field: keyof Settings) => {
settingsRepo.update({}, {[field]: value})
setSettings({...settings, [field]: value})
},
[settings, setSettings],
)
const changeAlarmEnabled = useCallback( const changeAlarmEnabled = useCallback(
(enabled: boolean) => { (enabled: boolean) => {
if (enabled) if (enabled)
@ -50,19 +60,18 @@ export default function SettingsPage() {
}) })
else toast('Stopped timers running after each set.') else toast('Stopped timers running after each set.')
if (enabled && !ignoring) setBattery(true) if (enabled && !ignoring) setBattery(true)
update(enabled, 'alarm') setAlarm(enabled)
settingsRepo.update({}, {alarm: enabled})
}, },
[setBattery, ignoring, update], [setBattery, ignoring],
) )
const changeVibrate = useCallback( const changeVibrate = useCallback((enabled: boolean) => {
(enabled: boolean) => { if (enabled) toast('When a timer completes, vibrate your phone.')
if (enabled) toast('When a timer completes, vibrate your phone.') else toast('Stop vibrating at the end of timers.')
else toast('Stop vibrating at the end of timers.') setVibrate(enabled)
update(enabled, 'vibrate') settingsRepo.update({}, {vibrate: enabled})
}, }, [])
[update],
)
const changeSound = useCallback(async () => { const changeSound = useCallback(async () => {
const {fileCopyUri} = await DocumentPicker.pickSingle({ const {fileCopyUri} = await DocumentPicker.pickSingle({
@ -71,124 +80,96 @@ export default function SettingsPage() {
}) })
if (!fileCopyUri) return if (!fileCopyUri) return
settingsRepo.update({}, {sound: fileCopyUri}) settingsRepo.update({}, {sound: fileCopyUri})
setSettings({...settings, sound: fileCopyUri}) setSound(fileCopyUri)
toast('This song will now play after rest timers complete.') toast('This song will now play after rest timers complete.')
}, [setSettings, settings]) }, [])
const changeNotify = useCallback( const changeNotify = useCallback((enabled: boolean) => {
(enabled: boolean) => { setNotify(enabled)
update(enabled, 'notify') settingsRepo.update({}, {notify: enabled})
if (enabled) toast('Show when a set is a new record.') if (enabled) toast('Show when a set is a new record.')
else toast('Stopped showing notifications for new records.') else toast('Stopped showing notifications for new records.')
}, }, [])
[update],
)
const changeImages = useCallback( const changeImages = useCallback((enabled: boolean) => {
(enabled: boolean) => { setImages(enabled)
update(enabled, 'images') settingsRepo.update({}, {images: enabled})
if (enabled) toast('Show images for sets.') if (enabled) toast('Show images for sets.')
else toast('Stopped showing images for sets.') else toast('Stopped showing images for sets.')
}, }, [])
[update],
)
const changeUnit = useCallback( const changeUnit = useCallback((enabled: boolean) => {
(enabled: boolean) => { setShowUnit(enabled)
update(enabled, 'showUnit') settingsRepo.update({}, {showUnit: enabled})
if (enabled) toast('Show option to select unit for sets.') if (enabled) toast('Show option to select unit for sets.')
else toast('Hid unit option for sets.') else toast('Hid unit option for sets.')
}, }, [])
[update],
)
const changeSteps = useCallback( const changeSteps = useCallback((enabled: boolean) => {
(enabled: boolean) => { setSteps(enabled)
update(enabled, 'steps') settingsRepo.update({}, {steps: enabled})
if (enabled) toast('Show steps for a workout.') if (enabled) toast('Show steps for a workout.')
else toast('Stopped showing steps for workouts.') else toast('Stopped showing steps for workouts.')
}, }, [])
[update],
)
const changeShowDate = useCallback( const changeShowDate = useCallback((enabled: boolean) => {
(enabled: boolean) => { setShowDate(enabled)
update(enabled, 'showDate') settingsRepo.update({}, {showDate: enabled})
if (enabled) toast('Show date for sets by default.') if (enabled) toast('Show date for sets by default.')
else toast('Stopped showing date for sets by default.') else toast('Stopped showing date for sets by default.')
}, }, [])
[update],
)
const changeShowSets = useCallback( const changeShowSets = useCallback((enabled: boolean) => {
(enabled: boolean) => { setShowSets(enabled)
update(enabled, 'showSets') settingsRepo.update({}, {showSets: enabled})
if (enabled) toast('Show target sets for workouts.') if (enabled) toast('Show target sets for workouts.')
else toast('Stopped showing target sets for workouts.') else toast('Stopped showing target sets for workouts.')
}, }, [])
[update],
)
const changeNoSound = useCallback( const changeNoSound = useCallback((enabled: boolean) => {
(enabled: boolean) => { setNoSound(enabled)
update(enabled, 'noSound') settingsRepo.update({}, {noSound: enabled})
if (enabled) toast('Disable sound on rest timer alarms.') if (enabled) toast('Disable sound on rest timer alarms.')
else toast('Enabled sound for rest timer alarms.') else toast('Enabled sound for rest timer alarms.')
}, }, [])
[update],
)
const switches: Input<boolean>[] = [ const switches: Input<boolean>[] = [
{name: 'Rest timers', value: settings.alarm, onChange: changeAlarmEnabled}, {name: 'Rest timers', value: alarm, onChange: changeAlarmEnabled},
{name: 'Vibrate', value: settings.vibrate, onChange: changeVibrate}, {name: 'Vibrate', value: vibrate, onChange: changeVibrate},
{name: 'Disable sound', value: settings.noSound, onChange: changeNoSound}, {name: 'Disable sound', value: noSound, onChange: changeNoSound},
{name: 'Notifications', value: settings.notify, onChange: changeNotify}, {name: 'Notifications', value: notify, onChange: changeNotify},
{name: 'Show images', value: settings.images, onChange: changeImages}, {name: 'Show images', value: images, onChange: changeImages},
{name: 'Show unit', value: settings.showUnit, onChange: changeUnit}, {name: 'Show unit', value: showUnit, onChange: changeUnit},
{name: 'Show steps', value: settings.steps, onChange: changeSteps}, {name: 'Show steps', value: steps, onChange: changeSteps},
{name: 'Show date', value: settings.showDate, onChange: changeShowDate}, {name: 'Show date', value: showDate, onChange: changeShowDate},
{name: 'Show sets', value: settings.showSets, onChange: changeShowSets}, {name: 'Show sets', value: showSets, onChange: changeShowSets},
] ]
const changeTheme = useCallback( const changeTheme = useCallback(
(value: string) => { (value: string) => {
settingsRepo.update({}, {theme: value}) settingsRepo.update({}, {theme: value})
setSettings({...settings, theme: value}) setTheme(value)
}, },
[settings, setSettings], [setTheme],
) )
const changeDate = useCallback( const changeDate = useCallback((value: string) => {
(value: string) => { settingsRepo.update({}, {date: value})
settingsRepo.update({}, {date: value}) setDate(value)
setSettings({...settings, date: value as any}) }, [])
},
[settings, setSettings],
)
const sound = useMemo(() => { const soundString = useMemo(() => {
if (!settings.sound) return null if (!sound) return null
const split = settings.sound.split('/') const split = sound.split('/')
return ': ' + split.pop() return ': ' + split.pop()
}, [settings.sound]) }, [sound])
const theme = useMemo(() => {
if (!'theme'.includes(term.toLowerCase())) return null
return (
<Select value={settings.theme} onChange={changeTheme}>
<Picker.Item value="system" label="Follow system theme" />
<Picker.Item value="dark" label="Dark theme" />
<Picker.Item value="light" label="Light theme" />
</Select>
)
}, [term, changeTheme, settings.theme])
const changeColor = useCallback( const changeColor = useCallback(
(value: string) => { (value: string) => {
setSettings({...settings, color: value}) setColor(value)
settingsRepo.update({}, {color: value}) settingsRepo.update({}, {color: value})
}, },
[setSettings, settings], [setColor],
) )
return ( return (
@ -209,9 +190,15 @@ export default function SettingsPage() {
{input.name} {input.name}
</Switch> </Switch>
))} ))}
{theme} {'theme'.includes(term.toLowerCase()) && (
<Select value={theme} onChange={changeTheme}>
<Picker.Item value="system" label="Follow system theme" />
<Picker.Item value="dark" label="Dark theme" />
<Picker.Item value="light" label="Light theme" />
</Select>
)}
{'color'.includes(term.toLowerCase()) && ( {'color'.includes(term.toLowerCase()) && (
<Select value={settings.color} onChange={changeColor}> <Select value={color} onChange={changeColor}>
{lightColors.concat(darkColors).map(colorOption => ( {lightColors.concat(darkColors).map(colorOption => (
<Picker.Item <Picker.Item
key={colorOption.hex} key={colorOption.hex}
@ -223,7 +210,7 @@ export default function SettingsPage() {
</Select> </Select>
)} )}
{'date format'.includes(term.toLowerCase()) && ( {'date format'.includes(term.toLowerCase()) && (
<Select value={settings.date} onChange={changeDate}> <Select value={date} onChange={changeDate}>
<Picker.Item value="%Y-%m-%d %H:%M" label="1990-12-24 15:05" /> <Picker.Item value="%Y-%m-%d %H:%M" label="1990-12-24 15:05" />
<Picker.Item value="%Y-%m-%d" label="1990-12-24" /> <Picker.Item value="%Y-%m-%d" label="1990-12-24" />
<Picker.Item value="%d/%m" label="24/12 (dd/MM)" /> <Picker.Item value="%d/%m" label="24/12 (dd/MM)" />
@ -240,7 +227,7 @@ export default function SettingsPage() {
)} )}
{'alarm sound'.includes(term.toLowerCase()) && ( {'alarm sound'.includes(term.toLowerCase()) && (
<Button style={{alignSelf: 'flex-start'}} onPress={changeSound}> <Button style={{alignSelf: 'flex-start'}} onPress={changeSound}>
Alarm sound{sound} Alarm sound{soundString}
</Button> </Button>
)} )}
</ScrollView> </ScrollView>