Add dark mode toggle
parent
53267d755b
commit
a90fd5d26f
|
@ -5,12 +5,13 @@ import {RootSiblingParent} from 'react-native-root-siblings'
|
||||||
import {GestureHandlerRootView} from 'react-native-gesture-handler'
|
import {GestureHandlerRootView} from 'react-native-gesture-handler'
|
||||||
import SplashScreen from 'react-native-splash-screen'
|
import SplashScreen from 'react-native-splash-screen'
|
||||||
import {SafeAreaProvider} from 'react-native-safe-area-context'
|
import {SafeAreaProvider} from 'react-native-safe-area-context'
|
||||||
|
import {observer} from 'mobx-react-lite'
|
||||||
import {ThemeProvider} from './view/lib/ThemeContext'
|
import {ThemeProvider} from './view/lib/ThemeContext'
|
||||||
import * as view from './view/index'
|
import * as view from './view/index'
|
||||||
import {RootStoreModel, setupState, RootStoreProvider} from './state'
|
import {RootStoreModel, setupState, RootStoreProvider} from './state'
|
||||||
import {MobileShell} from './view/shell/mobile'
|
import {MobileShell} from './view/shell/mobile'
|
||||||
|
|
||||||
function App() {
|
const App = observer(() => {
|
||||||
const [rootStore, setRootStore] = useState<RootStoreModel | undefined>(
|
const [rootStore, setRootStore] = useState<RootStoreModel | undefined>(
|
||||||
undefined,
|
undefined,
|
||||||
)
|
)
|
||||||
|
@ -41,7 +42,7 @@ function App() {
|
||||||
<GestureHandlerRootView style={{flex: 1}}>
|
<GestureHandlerRootView style={{flex: 1}}>
|
||||||
<RootSiblingParent>
|
<RootSiblingParent>
|
||||||
<RootStoreProvider value={rootStore}>
|
<RootStoreProvider value={rootStore}>
|
||||||
<ThemeProvider>
|
<ThemeProvider theme={rootStore.shell.darkMode ? 'dark' : 'light'}>
|
||||||
<SafeAreaProvider>
|
<SafeAreaProvider>
|
||||||
<MobileShell />
|
<MobileShell />
|
||||||
</SafeAreaProvider>
|
</SafeAreaProvider>
|
||||||
|
@ -50,6 +51,6 @@ function App() {
|
||||||
</RootSiblingParent>
|
</RootSiblingParent>
|
||||||
</GestureHandlerRootView>
|
</GestureHandlerRootView>
|
||||||
)
|
)
|
||||||
}
|
})
|
||||||
|
|
||||||
export default App
|
export default App
|
||||||
|
|
|
@ -67,6 +67,7 @@ export class RootStoreModel {
|
||||||
me: this.me.serialize(),
|
me: this.me.serialize(),
|
||||||
nav: this.nav.serialize(),
|
nav: this.nav.serialize(),
|
||||||
onboard: this.onboard.serialize(),
|
onboard: this.onboard.serialize(),
|
||||||
|
shell: this.shell.serialize(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -84,6 +85,9 @@ export class RootStoreModel {
|
||||||
if (hasProp(v, 'onboard')) {
|
if (hasProp(v, 'onboard')) {
|
||||||
this.onboard.hydrate(v.onboard)
|
this.onboard.hydrate(v.onboard)
|
||||||
}
|
}
|
||||||
|
if (hasProp(v, 'shell')) {
|
||||||
|
this.shell.hydrate(v.shell)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
import {makeAutoObservable} from 'mobx'
|
import {makeAutoObservable} from 'mobx'
|
||||||
import {ProfileViewModel} from './profile-view'
|
import {ProfileViewModel} from './profile-view'
|
||||||
|
import {isObj, hasProp} from '../lib/type-guards'
|
||||||
|
|
||||||
export class ConfirmModal {
|
export class ConfirmModal {
|
||||||
name = 'confirm'
|
name = 'confirm'
|
||||||
|
@ -135,6 +136,7 @@ export interface ComposerOpts {
|
||||||
}
|
}
|
||||||
|
|
||||||
export class ShellUiModel {
|
export class ShellUiModel {
|
||||||
|
darkMode = false
|
||||||
minimalShellMode = false
|
minimalShellMode = false
|
||||||
isMainMenuOpen = false
|
isMainMenuOpen = false
|
||||||
isModalActive = false
|
isModalActive = false
|
||||||
|
@ -156,7 +158,25 @@ export class ShellUiModel {
|
||||||
composerOpts: ComposerOpts | undefined
|
composerOpts: ComposerOpts | undefined
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
makeAutoObservable(this)
|
makeAutoObservable(this, {serialize: false, hydrate: false})
|
||||||
|
}
|
||||||
|
|
||||||
|
serialize(): unknown {
|
||||||
|
return {
|
||||||
|
darkMode: this.darkMode,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
hydrate(v: unknown) {
|
||||||
|
if (isObj(v)) {
|
||||||
|
if (hasProp(v, 'darkMode') && typeof v.darkMode === 'boolean') {
|
||||||
|
this.darkMode = v.darkMode
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setDarkMode(v: boolean) {
|
||||||
|
this.darkMode = v
|
||||||
}
|
}
|
||||||
|
|
||||||
setMinimalShellMode(v: boolean) {
|
setMinimalShellMode(v: boolean) {
|
||||||
|
|
|
@ -6,6 +6,7 @@ import {
|
||||||
View,
|
View,
|
||||||
ViewStyle,
|
ViewStyle,
|
||||||
} from 'react-native'
|
} from 'react-native'
|
||||||
|
import {observer} from 'mobx-react-lite'
|
||||||
import VersionNumber from 'react-native-version-number'
|
import VersionNumber from 'react-native-version-number'
|
||||||
import {s, colors} from '../../lib/styles'
|
import {s, colors} from '../../lib/styles'
|
||||||
import {useStores} from '../../../state'
|
import {useStores} from '../../../state'
|
||||||
|
@ -18,187 +19,198 @@ import {
|
||||||
} from '../../lib/icons'
|
} from '../../lib/icons'
|
||||||
import {UserAvatar} from '../../com/util/UserAvatar'
|
import {UserAvatar} from '../../com/util/UserAvatar'
|
||||||
import {Text} from '../../com/util/text/Text'
|
import {Text} from '../../com/util/text/Text'
|
||||||
|
import {ToggleButton} from '../../com/util/forms/ToggleButton'
|
||||||
import {CreateSceneModal} from '../../../state/models/shell-ui'
|
import {CreateSceneModal} from '../../../state/models/shell-ui'
|
||||||
import {usePalette} from '../../lib/hooks/usePalette'
|
import {usePalette} from '../../lib/hooks/usePalette'
|
||||||
|
|
||||||
export const Menu = ({
|
export const Menu = observer(
|
||||||
visible,
|
({visible, onClose}: {visible: boolean; onClose: () => void}) => {
|
||||||
onClose,
|
const pal = usePalette('default')
|
||||||
}: {
|
const store = useStores()
|
||||||
visible: boolean
|
|
||||||
onClose: () => void
|
|
||||||
}) => {
|
|
||||||
const pal = usePalette('default')
|
|
||||||
const store = useStores()
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (visible) {
|
if (visible) {
|
||||||
// trigger a refresh in case memberships have changed recently
|
// trigger a refresh in case memberships have changed recently
|
||||||
// TODO this impacts performance, need to find the right time to do this
|
// TODO this impacts performance, need to find the right time to do this
|
||||||
// store.me.refreshMemberships()
|
// store.me.refreshMemberships()
|
||||||
}
|
}
|
||||||
}, [store, visible])
|
}, [store, visible])
|
||||||
|
|
||||||
// events
|
// events
|
||||||
// =
|
// =
|
||||||
|
|
||||||
const onNavigate = (url: string) => {
|
const onNavigate = (url: string) => {
|
||||||
onClose()
|
onClose()
|
||||||
if (url === '/notifications') {
|
if (url === '/notifications') {
|
||||||
store.nav.switchTo(1, true)
|
store.nav.switchTo(1, true)
|
||||||
} else {
|
} else {
|
||||||
store.nav.switchTo(0, true)
|
store.nav.switchTo(0, true)
|
||||||
if (url !== '/') {
|
if (url !== '/') {
|
||||||
store.nav.navigate(url)
|
store.nav.navigate(url)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
const onPressCreateScene = () => {
|
||||||
const onPressCreateScene = () => {
|
onClose()
|
||||||
onClose()
|
store.shell.openModal(new CreateSceneModal())
|
||||||
store.shell.openModal(new CreateSceneModal())
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// rendering
|
// rendering
|
||||||
// =
|
// =
|
||||||
|
|
||||||
const MenuItem = ({
|
const MenuItem = ({
|
||||||
icon,
|
icon,
|
||||||
label,
|
label,
|
||||||
count,
|
count,
|
||||||
url,
|
url,
|
||||||
bold,
|
bold,
|
||||||
onPress,
|
onPress,
|
||||||
}: {
|
}: {
|
||||||
icon: JSX.Element
|
icon: JSX.Element
|
||||||
label: string
|
label: string
|
||||||
count?: number
|
count?: number
|
||||||
url?: string
|
url?: string
|
||||||
bold?: boolean
|
bold?: boolean
|
||||||
onPress?: () => void
|
onPress?: () => void
|
||||||
}) => (
|
}) => (
|
||||||
<TouchableOpacity
|
|
||||||
style={styles.menuItem}
|
|
||||||
onPress={onPress ? onPress : () => onNavigate(url || '/')}>
|
|
||||||
<View style={[styles.menuItemIconWrapper]}>
|
|
||||||
{icon}
|
|
||||||
{count ? (
|
|
||||||
<View style={styles.menuItemCount}>
|
|
||||||
<Text style={styles.menuItemCountLabel}>{count}</Text>
|
|
||||||
</View>
|
|
||||||
) : undefined}
|
|
||||||
</View>
|
|
||||||
<Text
|
|
||||||
type="h4"
|
|
||||||
style={[
|
|
||||||
pal.text,
|
|
||||||
bold ? styles.menuItemLabelBold : styles.menuItemLabel,
|
|
||||||
]}
|
|
||||||
numberOfLines={1}>
|
|
||||||
{label}
|
|
||||||
</Text>
|
|
||||||
</TouchableOpacity>
|
|
||||||
)
|
|
||||||
|
|
||||||
return (
|
|
||||||
<View style={[styles.view, pal.view]}>
|
|
||||||
<TouchableOpacity
|
<TouchableOpacity
|
||||||
onPress={() => onNavigate(`/profile/${store.me.handle}`)}
|
style={styles.menuItem}
|
||||||
style={styles.profileCard}>
|
onPress={onPress ? onPress : () => onNavigate(url || '/')}>
|
||||||
<UserAvatar
|
<View style={[styles.menuItemIconWrapper]}>
|
||||||
size={60}
|
{icon}
|
||||||
displayName={store.me.displayName}
|
{count ? (
|
||||||
handle={store.me.handle}
|
<View style={styles.menuItemCount}>
|
||||||
avatar={store.me.avatar}
|
<Text style={styles.menuItemCountLabel}>{count}</Text>
|
||||||
/>
|
</View>
|
||||||
<View style={s.flex1}>
|
) : undefined}
|
||||||
<Text
|
</View>
|
||||||
type="h3"
|
<Text
|
||||||
style={[pal.text, styles.profileCardDisplayName]}
|
type="h4"
|
||||||
numberOfLines={1}>
|
style={[
|
||||||
{store.me.displayName || store.me.handle}
|
pal.text,
|
||||||
|
bold ? styles.menuItemLabelBold : styles.menuItemLabel,
|
||||||
|
]}
|
||||||
|
numberOfLines={1}>
|
||||||
|
{label}
|
||||||
|
</Text>
|
||||||
|
</TouchableOpacity>
|
||||||
|
)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<View style={[styles.view, pal.view]}>
|
||||||
|
<TouchableOpacity
|
||||||
|
onPress={() => onNavigate(`/profile/${store.me.handle}`)}
|
||||||
|
style={styles.profileCard}>
|
||||||
|
<UserAvatar
|
||||||
|
size={60}
|
||||||
|
displayName={store.me.displayName}
|
||||||
|
handle={store.me.handle}
|
||||||
|
avatar={store.me.avatar}
|
||||||
|
/>
|
||||||
|
<View style={s.flex1}>
|
||||||
|
<Text
|
||||||
|
type="h3"
|
||||||
|
style={[pal.text, styles.profileCardDisplayName]}
|
||||||
|
numberOfLines={1}>
|
||||||
|
{store.me.displayName || store.me.handle}
|
||||||
|
</Text>
|
||||||
|
<Text
|
||||||
|
style={[pal.textLight, styles.profileCardHandle]}
|
||||||
|
numberOfLines={1}>
|
||||||
|
@{store.me.handle}
|
||||||
|
</Text>
|
||||||
|
</View>
|
||||||
|
</TouchableOpacity>
|
||||||
|
<TouchableOpacity
|
||||||
|
style={[styles.searchBtn, pal.btn]}
|
||||||
|
onPress={() => onNavigate('/search')}>
|
||||||
|
<MagnifyingGlassIcon
|
||||||
|
style={pal.text as StyleProp<ViewStyle>}
|
||||||
|
size={25}
|
||||||
|
/>
|
||||||
|
<Text type="h4" style={[pal.text, styles.searchBtnLabel]}>
|
||||||
|
Search
|
||||||
</Text>
|
</Text>
|
||||||
<Text
|
</TouchableOpacity>
|
||||||
style={[pal.textLight, styles.profileCardHandle]}
|
<View style={[styles.section, pal.border]}>
|
||||||
numberOfLines={1}>
|
<MenuItem
|
||||||
@{store.me.handle}
|
icon={
|
||||||
|
<HomeIcon style={pal.text as StyleProp<ViewStyle>} size="26" />
|
||||||
|
}
|
||||||
|
label="Home"
|
||||||
|
url="/"
|
||||||
|
/>
|
||||||
|
<MenuItem
|
||||||
|
icon={
|
||||||
|
<BellIcon style={pal.text as StyleProp<ViewStyle>} size="28" />
|
||||||
|
}
|
||||||
|
label="Notifications"
|
||||||
|
url="/notifications"
|
||||||
|
count={store.me.notificationCount}
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
<View style={[styles.section, pal.border]}>
|
||||||
|
<Text type="h5" style={[pal.text, styles.heading]}>
|
||||||
|
Scenes
|
||||||
|
</Text>
|
||||||
|
{store.me.memberships
|
||||||
|
? store.me.memberships.memberships.map((membership, i) => (
|
||||||
|
<MenuItem
|
||||||
|
key={i}
|
||||||
|
icon={
|
||||||
|
<UserAvatar
|
||||||
|
size={34}
|
||||||
|
displayName={membership.displayName}
|
||||||
|
handle={membership.handle}
|
||||||
|
avatar={membership.avatar}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
label={membership.displayName || membership.handle}
|
||||||
|
url={`/profile/${membership.handle}`}
|
||||||
|
/>
|
||||||
|
))
|
||||||
|
: undefined}
|
||||||
|
</View>
|
||||||
|
<View style={[styles.section, pal.border]}>
|
||||||
|
<MenuItem
|
||||||
|
icon={
|
||||||
|
<UserGroupIcon
|
||||||
|
style={pal.text as StyleProp<ViewStyle>}
|
||||||
|
size="30"
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
label="Create a scene"
|
||||||
|
onPress={onPressCreateScene}
|
||||||
|
/>
|
||||||
|
<MenuItem
|
||||||
|
icon={
|
||||||
|
<CogIcon
|
||||||
|
style={pal.text as StyleProp<ViewStyle>}
|
||||||
|
size="30"
|
||||||
|
strokeWidth={2}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
label="Settings"
|
||||||
|
url="/settings"
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
<View style={[styles.section, pal.border]}>
|
||||||
|
<ToggleButton
|
||||||
|
label="Dark mode"
|
||||||
|
isSelected={store.shell.darkMode}
|
||||||
|
onPress={() => store.shell.setDarkMode(!store.shell.darkMode)}
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
<View style={styles.footer}>
|
||||||
|
<Text style={[pal.textLight]}>
|
||||||
|
Build version {VersionNumber.appVersion} (
|
||||||
|
{VersionNumber.buildVersion})
|
||||||
</Text>
|
</Text>
|
||||||
</View>
|
</View>
|
||||||
</TouchableOpacity>
|
|
||||||
<TouchableOpacity
|
|
||||||
style={[styles.searchBtn, pal.btn]}
|
|
||||||
onPress={() => onNavigate('/search')}>
|
|
||||||
<MagnifyingGlassIcon
|
|
||||||
style={pal.text as StyleProp<ViewStyle>}
|
|
||||||
size={25}
|
|
||||||
/>
|
|
||||||
<Text type="h4" style={[pal.text, styles.searchBtnLabel]}>
|
|
||||||
Search
|
|
||||||
</Text>
|
|
||||||
</TouchableOpacity>
|
|
||||||
<View style={[styles.section, pal.border]}>
|
|
||||||
<MenuItem
|
|
||||||
icon={<HomeIcon style={pal.text as StyleProp<ViewStyle>} size="26" />}
|
|
||||||
label="Home"
|
|
||||||
url="/"
|
|
||||||
/>
|
|
||||||
<MenuItem
|
|
||||||
icon={<BellIcon style={pal.text as StyleProp<ViewStyle>} size="28" />}
|
|
||||||
label="Notifications"
|
|
||||||
url="/notifications"
|
|
||||||
count={store.me.notificationCount}
|
|
||||||
/>
|
|
||||||
</View>
|
</View>
|
||||||
<View style={[styles.section, pal.border]}>
|
)
|
||||||
<Text type="h5" style={[pal.text, styles.heading]}>
|
},
|
||||||
Scenes
|
)
|
||||||
</Text>
|
|
||||||
{store.me.memberships
|
|
||||||
? store.me.memberships.memberships.map((membership, i) => (
|
|
||||||
<MenuItem
|
|
||||||
key={i}
|
|
||||||
icon={
|
|
||||||
<UserAvatar
|
|
||||||
size={34}
|
|
||||||
displayName={membership.displayName}
|
|
||||||
handle={membership.handle}
|
|
||||||
avatar={membership.avatar}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
label={membership.displayName || membership.handle}
|
|
||||||
url={`/profile/${membership.handle}`}
|
|
||||||
/>
|
|
||||||
))
|
|
||||||
: undefined}
|
|
||||||
</View>
|
|
||||||
<View style={[styles.section, pal.border]}>
|
|
||||||
<MenuItem
|
|
||||||
icon={
|
|
||||||
<UserGroupIcon style={pal.text as StyleProp<ViewStyle>} size="30" />
|
|
||||||
}
|
|
||||||
label="Create a scene"
|
|
||||||
onPress={onPressCreateScene}
|
|
||||||
/>
|
|
||||||
<MenuItem
|
|
||||||
icon={
|
|
||||||
<CogIcon
|
|
||||||
style={pal.text as StyleProp<ViewStyle>}
|
|
||||||
size="30"
|
|
||||||
strokeWidth={2}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
label="Settings"
|
|
||||||
url="/settings"
|
|
||||||
/>
|
|
||||||
</View>
|
|
||||||
<View style={styles.footer}>
|
|
||||||
<Text style={[pal.textLight]}>
|
|
||||||
Build version {VersionNumber.appVersion} ({VersionNumber.buildVersion}
|
|
||||||
)
|
|
||||||
</Text>
|
|
||||||
</View>
|
|
||||||
</View>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
const styles = StyleSheet.create({
|
const styles = StyleSheet.create({
|
||||||
view: {
|
view: {
|
||||||
|
|
Loading…
Reference in New Issue