Massive/SettingsPage.tsx

203 lines
5.2 KiB
TypeScript
Raw Normal View History

2022-08-23 00:04:52 +00:00
import React, {
ReactNode,
useCallback,
useContext,
2022-08-23 00:04:52 +00:00
useEffect,
useState,
} from 'react';
import {NativeModules, StyleSheet, Text, View} from 'react-native';
2022-08-23 00:04:52 +00:00
import {Searchbar, TextInput} from 'react-native-paper';
import {DatabaseContext, SnackbarContext} from './App';
import ConfirmDialog from './ConfirmDialog';
2022-07-17 01:45:31 +00:00
import MassiveSwitch from './MassiveSwitch';
import Settings from './settings';
2022-07-08 03:45:24 +00:00
export default function SettingsPage() {
2022-08-20 04:37:59 +00:00
const [vibrate, setVibrate] = useState(true);
2022-07-03 01:50:01 +00:00
const [minutes, setMinutes] = useState<string>('');
const [maxSets, setMaxSets] = useState<string>('3');
2022-07-03 01:50:01 +00:00
const [seconds, setSeconds] = useState<string>('');
2022-08-20 04:37:59 +00:00
const [alarm, setAlarm] = useState<boolean>(false);
const [predictive, setPredictive] = useState<boolean>(false);
const [battery, setBattery] = useState(false);
2022-07-07 00:45:45 +00:00
const [ignoring, setIgnoring] = useState(false);
2022-08-23 00:04:52 +00:00
const [search, setSearch] = useState('');
const db = useContext(DatabaseContext);
const {toast} = useContext(SnackbarContext);
2022-07-03 01:50:01 +00:00
const refresh = useCallback(async () => {
const [result] = await db.executeSql(`SELECT * FROM settings LIMIT 1`);
const settings: Settings = result.rows.item(0);
console.log('SettingsPage.refresh:', {settings});
setMinutes(settings.minutes.toString());
setSeconds(settings.seconds.toString());
setAlarm(!!settings.alarm);
setPredictive(!!settings.predict);
setMaxSets(settings.sets.toString());
setVibrate(!!settings.vibrate);
2022-07-19 04:38:58 +00:00
NativeModules.AlarmModule.ignoringBattery(setIgnoring);
}, [db]);
2022-07-07 00:45:45 +00:00
useEffect(() => {
2022-07-07 00:45:45 +00:00
refresh();
}, [refresh]);
2022-07-03 01:50:01 +00:00
useEffect(() => {
db.executeSql(
`UPDATE settings SET vibrate=?,minutes=?,sets=?,seconds=?,alarm=?,predict=?`,
[vibrate, minutes, maxSets, seconds, alarm, predictive],
);
}, [vibrate, minutes, maxSets, seconds, alarm, predictive, db]);
const changeAlarmEnabled = useCallback(
(enabled: boolean) => {
2022-08-20 04:37:59 +00:00
setAlarm(enabled);
if (enabled && !ignoring) setBattery(true);
},
2022-08-20 04:37:59 +00:00
[setBattery, ignoring],
);
2022-07-10 05:53:38 +00:00
const changePredictive = useCallback(
(enabled: boolean) => {
2022-08-20 04:37:59 +00:00
setPredictive(enabled);
toast('Predictive sets guess whats next based on todays plan.', 7000);
2022-07-10 05:53:38 +00:00
},
[setPredictive, toast],
2022-07-10 05:53:38 +00:00
);
2022-08-20 04:37:59 +00:00
const changeVibrate = useCallback(
(value: boolean) => {
setVibrate(value);
},
[setVibrate],
);
2022-08-23 00:04:52 +00:00
const items: {name: string; element: ReactNode}[] = [
{
name: 'Sets per workout',
element: (
<TextInput
2022-08-25 00:35:09 +00:00
mode="outlined"
2022-08-23 00:04:52 +00:00
label="Sets per workout"
value={maxSets}
keyboardType="numeric"
onChangeText={value => {
setMaxSets(value);
}}
style={styles.text}
selectTextOnFocus
/>
),
},
{
name: 'Rest minutes',
2022-08-23 00:04:52 +00:00
element: (
<TextInput
2022-08-25 00:35:09 +00:00
mode="outlined"
label="Rest minutes"
value={minutes}
2022-08-23 00:04:52 +00:00
keyboardType="numeric"
placeholder="3"
onChangeText={text => {
setMinutes(text);
2022-08-23 00:04:52 +00:00
}}
style={styles.text}
selectTextOnFocus
/>
),
},
{
name: 'Rest seconds',
2022-08-23 00:04:52 +00:00
element: (
<TextInput
2022-08-25 00:35:09 +00:00
mode="outlined"
label="Rest seconds"
value={seconds}
2022-08-23 00:04:52 +00:00
keyboardType="numeric"
placeholder="30"
onChangeText={s => {
setSeconds(s);
2022-08-23 00:04:52 +00:00
}}
style={styles.text}
selectTextOnFocus
/>
),
},
{
name: 'Rest timers',
element: (
<>
<Text style={styles.text}>Rest timers</Text>
<MassiveSwitch
style={[styles.text, {alignSelf: 'flex-start'}]}
value={alarm}
onValueChange={changeAlarmEnabled}
/>
</>
),
},
{
name: 'Vibrate',
element: (
<>
<Text style={styles.text}>Vibrate</Text>
<MassiveSwitch
style={[styles.text, {alignSelf: 'flex-start'}]}
value={vibrate}
onValueChange={changeVibrate}
/>
</>
),
},
{
name: 'Predictive sets',
element: (
<>
<Text style={styles.text}>Predictive sets</Text>
<MassiveSwitch
style={[styles.text, {alignSelf: 'flex-start'}]}
value={predictive}
onValueChange={changePredictive}
/>
</>
),
},
];
2022-08-20 04:37:59 +00:00
return (
<View style={styles.container}>
<Searchbar
style={{marginBottom: 10}}
placeholder="Search"
value={search}
onChangeText={setSearch}
/>
2022-08-23 00:04:52 +00:00
{items
.filter(item => item.name.toLowerCase().includes(search.toLowerCase()))
.map(item => (
<React.Fragment key={item.name}>{item.element}</React.Fragment>
))}
<ConfirmDialog
title="Battery optimizations"
2022-08-20 04:37:59 +00:00
show={battery}
setShow={setBattery}
onOk={() => {
2022-07-19 04:38:58 +00:00
NativeModules.AlarmModule.openSettings();
2022-08-20 04:37:59 +00:00
setBattery(false);
}}>
Disable battery optimizations for Massive to use rest timers.
</ConfirmDialog>
2022-07-03 01:50:01 +00:00
</View>
);
}
const styles = StyleSheet.create({
container: {
padding: 10,
2022-07-05 03:33:42 +00:00
flex: 1,
2022-07-03 01:50:01 +00:00
},
text: {
marginBottom: 10,
},
2022-07-03 01:50:01 +00:00
});