forked from NWACus/avy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Preferences.tsx
72 lines (57 loc) · 2.47 KB
/
Preferences.tsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
// Create an object that uses AsyncStorage to persist user preferences
// and provide a hook to update them.
import AsyncStorage from '@react-native-async-storage/async-storage';
import {merge} from 'lodash';
import React, {createContext, ReactNode, useContext, useEffect, useState} from 'react';
import * as Sentry from 'sentry-expo';
import {useAsyncEffect} from 'use-async-effect';
import {z} from 'zod';
import {PREFERENCES_KEY} from 'data/asyncStorageKeys';
import {avalancheCenterIDSchema} from 'types/nationalAvalancheCenter';
const preferencesSchema = z.object({
center: avalancheCenterIDSchema.default('NWAC'),
hasSeenCenterPicker: z.boolean().default(false),
secretMenuCollapsed: z.boolean().default(true),
});
export type Preferences = z.infer<typeof preferencesSchema>;
const defaultPreferences = preferencesSchema.parse({});
interface PreferencesContextType {
preferences: Preferences;
setPreferences: (preferences: Partial<Preferences>) => void;
}
const PreferencesContext = createContext<PreferencesContextType>({
preferences: defaultPreferences,
setPreferences: () => undefined,
});
interface PreferencesProviderProps {
children?: ReactNode;
}
export const PreferencesProvider: React.FC<PreferencesProviderProps> = ({children}) => {
const [preferences, setFullPreferences] = useState<Preferences>(defaultPreferences);
useAsyncEffect(async () => {
let storedPreferences = {};
try {
storedPreferences = preferencesSchema.parse(JSON.parse((await AsyncStorage.getItem(PREFERENCES_KEY)) ?? '{}'));
} catch (e) {
// Error parsing preferences, ignore as we'll fall back to defaults
await clearPreferences();
// But do log it to Sentry as it shouldn't happen
Sentry.Native.captureException(e);
}
if (storedPreferences) {
setPreferences(merge({}, defaultPreferences, storedPreferences));
}
}, []);
useEffect(() => {
void AsyncStorage.setItem(PREFERENCES_KEY, JSON.stringify(preferences));
}, [preferences]);
const setPreferences = (newPreferences: Partial<Preferences>) => {
setFullPreferences(merge({}, preferences, newPreferences));
};
return <PreferencesContext.Provider value={{preferences, setPreferences}}>{children}</PreferencesContext.Provider>;
};
export const usePreferences = () => useContext(PreferencesContext);
export const clearPreferences = async () => {
await AsyncStorage.removeItem(PREFERENCES_KEY);
await AsyncStorage.setItem(PREFERENCES_KEY, JSON.stringify(defaultPreferences));
};