Extract shell state into separate context (#1824)

* WIP

* Add shell state

* Integrate new shell state for drawer and minimal shell mode

* Replace isDrawerSwipeDisabled

* Split shell state into separate contexts to avoid needless re-renders

* Fix typo

---------

Co-authored-by: Paul Frazee <pfrazee@gmail.com>
zio/stable
Eric Bailey 2023-11-07 13:37:47 -06:00 committed by GitHub
parent 7158157f5f
commit bfe196bac5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
51 changed files with 368 additions and 238 deletions

View File

@ -19,6 +19,7 @@ import * as analytics from 'lib/analytics/analytics'
import * as Toast from 'view/com/util/Toast' import * as Toast from 'view/com/util/Toast'
import {queryClient} from 'lib/react-query' import {queryClient} from 'lib/react-query'
import {TestCtrls} from 'view/com/testing/TestCtrls' import {TestCtrls} from 'view/com/testing/TestCtrls'
import {Provider as ShellStateProvider} from 'state/shell'
SplashScreen.preventAutoHideAsync() SplashScreen.preventAutoHideAsync()
@ -44,6 +45,7 @@ const App = observer(function AppImpl() {
return null return null
} }
return ( return (
<ShellStateProvider>
<QueryClientProvider client={queryClient}> <QueryClientProvider client={queryClient}>
<ThemeProvider theme={rootStore.shell.colorMode}> <ThemeProvider theme={rootStore.shell.colorMode}>
<RootSiblingParent> <RootSiblingParent>
@ -58,6 +60,7 @@ const App = observer(function AppImpl() {
</RootSiblingParent> </RootSiblingParent>
</ThemeProvider> </ThemeProvider>
</QueryClientProvider> </QueryClientProvider>
</ShellStateProvider>
) )
}) })

View File

@ -14,6 +14,7 @@ import {Shell} from 'view/shell/index'
import {ToastContainer} from 'view/com/util/Toast.web' import {ToastContainer} from 'view/com/util/Toast.web'
import {ThemeProvider} from 'lib/ThemeContext' import {ThemeProvider} from 'lib/ThemeContext'
import {queryClient} from 'lib/react-query' import {queryClient} from 'lib/react-query'
import {Provider as ShellStateProvider} from 'state/shell'
const App = observer(function AppImpl() { const App = observer(function AppImpl() {
const [rootStore, setRootStore] = useState<RootStoreModel | undefined>( const [rootStore, setRootStore] = useState<RootStoreModel | undefined>(
@ -34,6 +35,7 @@ const App = observer(function AppImpl() {
} }
return ( return (
<ShellStateProvider>
<QueryClientProvider client={queryClient}> <QueryClientProvider client={queryClient}>
<ThemeProvider theme={rootStore.shell.colorMode}> <ThemeProvider theme={rootStore.shell.colorMode}>
<RootSiblingParent> <RootSiblingParent>
@ -48,6 +50,7 @@ const App = observer(function AppImpl() {
</RootSiblingParent> </RootSiblingParent>
</ThemeProvider> </ThemeProvider>
</QueryClientProvider> </QueryClientProvider>
</ShellStateProvider>
) )
}) })

View File

@ -6,6 +6,7 @@ import {NavigationProp} from 'lib/routes/types'
import {AccountData} from 'state/models/session' import {AccountData} from 'state/models/session'
import {reset as resetNavigation} from '../../Navigation' import {reset as resetNavigation} from '../../Navigation'
import * as Toast from 'view/com/util/Toast' import * as Toast from 'view/com/util/Toast'
import {useSetDrawerOpen} from '#/state/shell/drawer-open'
export function useAccountSwitcher(): [ export function useAccountSwitcher(): [
boolean, boolean,
@ -13,8 +14,8 @@ export function useAccountSwitcher(): [
(acct: AccountData) => Promise<void>, (acct: AccountData) => Promise<void>,
] { ] {
const {track} = useAnalytics() const {track} = useAnalytics()
const store = useStores() const store = useStores()
const setDrawerOpen = useSetDrawerOpen()
const [isSwitching, setIsSwitching] = useState(false) const [isSwitching, setIsSwitching] = useState(false)
const navigation = useNavigation<NavigationProp>() const navigation = useNavigation<NavigationProp>()
@ -23,6 +24,7 @@ export function useAccountSwitcher(): [
track('Settings:SwitchAccountButtonClicked') track('Settings:SwitchAccountButtonClicked')
setIsSwitching(true) setIsSwitching(true)
const success = await store.session.resumeSession(acct) const success = await store.session.resumeSession(acct)
setDrawerOpen(false)
store.shell.closeAllActiveElements() store.shell.closeAllActiveElements()
if (success) { if (success) {
resetNavigation() resetNavigation()
@ -34,7 +36,7 @@ export function useAccountSwitcher(): [
store.session.clear() store.session.clear()
} }
}, },
[track, setIsSwitching, navigation, store], [track, setIsSwitching, navigation, store, setDrawerOpen],
) )
return [isSwitching, setIsSwitching, onPressSwitchAccount] return [isSwitching, setIsSwitching, onPressSwitchAccount]

View File

@ -1,6 +1,5 @@
import React from 'react' import React from 'react'
import {autorun} from 'mobx' import {autorun} from 'mobx'
import {useStores} from 'state/index'
import { import {
Easing, Easing,
interpolate, interpolate,
@ -9,8 +8,10 @@ import {
withTiming, withTiming,
} from 'react-native-reanimated' } from 'react-native-reanimated'
import {useMinimalShellMode as useMinimalShellModeState} from '#/state/shell/minimal-mode'
export function useMinimalShellMode() { export function useMinimalShellMode() {
const store = useStores() const minimalShellMode = useMinimalShellModeState()
const minimalShellInterp = useSharedValue(0) const minimalShellInterp = useSharedValue(0)
const footerMinimalShellTransform = useAnimatedStyle(() => { const footerMinimalShellTransform = useAnimatedStyle(() => {
return { return {
@ -38,7 +39,7 @@ export function useMinimalShellMode() {
React.useEffect(() => { React.useEffect(() => {
return autorun(() => { return autorun(() => {
if (store.shell.minimalShellMode) { if (minimalShellMode) {
minimalShellInterp.value = withTiming(1, { minimalShellInterp.value = withTiming(1, {
duration: 125, duration: 125,
easing: Easing.bezier(0.25, 0.1, 0.25, 1), easing: Easing.bezier(0.25, 0.1, 0.25, 1),
@ -50,9 +51,10 @@ export function useMinimalShellMode() {
}) })
} }
}) })
}, [minimalShellInterp, store.shell.minimalShellMode]) }, [minimalShellInterp, minimalShellMode])
return { return {
minimalShellMode,
footerMinimalShellTransform, footerMinimalShellTransform,
headerMinimalShellTransform, headerMinimalShellTransform,
fabMinimalShellTransform, fabMinimalShellTransform,

View File

@ -1,8 +1,8 @@
import {useState, useCallback, useRef} from 'react' import {useState, useCallback, useRef} from 'react'
import {NativeSyntheticEvent, NativeScrollEvent} from 'react-native' import {NativeSyntheticEvent, NativeScrollEvent} from 'react-native'
import {RootStoreModel} from 'state/index'
import {s} from 'lib/styles' import {s} from 'lib/styles'
import {useWebMediaQueries} from './useWebMediaQueries' import {useWebMediaQueries} from './useWebMediaQueries'
import {useSetMinimalShellMode, useMinimalShellMode} from '#/state/shell'
const Y_LIMIT = 10 const Y_LIMIT = 10
@ -19,12 +19,12 @@ export type OnScrollCb = (
) => void ) => void
export type ResetCb = () => void export type ResetCb = () => void
export function useOnMainScroll( export function useOnMainScroll(): [OnScrollCb, boolean, ResetCb] {
store: RootStoreModel,
): [OnScrollCb, boolean, ResetCb] {
let lastY = useRef(0) let lastY = useRef(0)
let [isScrolledDown, setIsScrolledDown] = useState(false) let [isScrolledDown, setIsScrolledDown] = useState(false)
const {dyLimitUp, dyLimitDown} = useDeviceLimits() const {dyLimitUp, dyLimitDown} = useDeviceLimits()
const minimalShellMode = useMinimalShellMode()
const setMinimalShellMode = useSetMinimalShellMode()
return [ return [
useCallback( useCallback(
@ -33,13 +33,10 @@ export function useOnMainScroll(
const dy = y - (lastY.current || 0) const dy = y - (lastY.current || 0)
lastY.current = y lastY.current = y
if (!store.shell.minimalShellMode && dy > dyLimitDown && y > Y_LIMIT) { if (!minimalShellMode && dy > dyLimitDown && y > Y_LIMIT) {
store.shell.setMinimalShellMode(true) setMinimalShellMode(true)
} else if ( } else if (minimalShellMode && (dy < dyLimitUp * -1 || y <= Y_LIMIT)) {
store.shell.minimalShellMode && setMinimalShellMode(false)
(dy < dyLimitUp * -1 || y <= Y_LIMIT)
) {
store.shell.setMinimalShellMode(false)
} }
if ( if (
@ -54,13 +51,19 @@ export function useOnMainScroll(
setIsScrolledDown(false) setIsScrolledDown(false)
} }
}, },
[store.shell, dyLimitDown, dyLimitUp, isScrolledDown], [
dyLimitDown,
dyLimitUp,
isScrolledDown,
minimalShellMode,
setMinimalShellMode,
],
), ),
isScrolledDown, isScrolledDown,
useCallback(() => { useCallback(() => {
setIsScrolledDown(false) setIsScrolledDown(false)
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
lastY.current = 1e8 // NOTE we set this very high so that the onScroll logic works right -prf lastY.current = 1e8 // NOTE we set this very high so that the onScroll logic works right -prf
}, [store, setIsScrolledDown]), }, [setIsScrolledDown, setMinimalShellMode]),
] ]
} }

View File

@ -1,19 +0,0 @@
import {isAndroid} from 'platform/detection'
import {BackHandler} from 'react-native'
import {RootStoreModel} from 'state/index'
export function init(store: RootStoreModel) {
// only register back handler on android, otherwise it throws an error
if (isAndroid) {
const backHandler = BackHandler.addEventListener(
'hardwareBackPress',
() => {
return store.shell.closeAnyActiveElement()
},
)
return () => {
backHandler.remove()
}
}
return () => {}
}

View File

@ -266,9 +266,6 @@ export interface ComposerOpts {
export class ShellUiModel { export class ShellUiModel {
colorMode: ColorMode = 'system' colorMode: ColorMode = 'system'
minimalShellMode = false
isDrawerOpen = false
isDrawerSwipeDisabled = false
isModalActive = false isModalActive = false
activeModals: Modal[] = [] activeModals: Modal[] = []
isLightboxActive = false isLightboxActive = false
@ -313,10 +310,6 @@ export class ShellUiModel {
} }
} }
setMinimalShellMode(v: boolean) {
this.minimalShellMode = v
}
/** /**
* returns true if something was closed * returns true if something was closed
* (used by the android hardware back btn) * (used by the android hardware back btn)
@ -334,10 +327,6 @@ export class ShellUiModel {
this.closeComposer() this.closeComposer()
return true return true
} }
if (this.isDrawerOpen) {
this.closeDrawer()
return true
}
return false return false
} }
@ -354,21 +343,6 @@ export class ShellUiModel {
if (this.isComposerActive) { if (this.isComposerActive) {
this.closeComposer() this.closeComposer()
} }
if (this.isDrawerOpen) {
this.closeDrawer()
}
}
openDrawer() {
this.isDrawerOpen = true
}
closeDrawer() {
this.isDrawerOpen = false
}
setIsDrawerSwipeDisabled(v: boolean) {
this.isDrawerSwipeDisabled = v
} }
openModal(modal: Modal) { openModal(modal: Modal) {

View File

@ -0,0 +1,24 @@
import React from 'react'
type StateContext = boolean
type SetContext = (v: boolean) => void
const stateContext = React.createContext<StateContext>(false)
const setContext = React.createContext<SetContext>((_: boolean) => {})
export function Provider({children}: React.PropsWithChildren<{}>) {
const [state, setState] = React.useState(false)
return (
<stateContext.Provider value={state}>
<setContext.Provider value={setState}>{children}</setContext.Provider>
</stateContext.Provider>
)
}
export function useIsDrawerOpen() {
return React.useContext(stateContext)
}
export function useSetDrawerOpen() {
return React.useContext(setContext)
}

View File

@ -0,0 +1,24 @@
import React from 'react'
type StateContext = boolean
type SetContext = (v: boolean) => void
const stateContext = React.createContext<StateContext>(false)
const setContext = React.createContext<SetContext>((_: boolean) => {})
export function Provider({children}: React.PropsWithChildren<{}>) {
const [state, setState] = React.useState(false)
return (
<stateContext.Provider value={state}>
<setContext.Provider value={setState}>{children}</setContext.Provider>
</stateContext.Provider>
)
}
export function useIsDrawerSwipeDisabled() {
return React.useContext(stateContext)
}
export function useSetDrawerSwipeDisabled() {
return React.useContext(setContext)
}

View File

@ -0,0 +1,21 @@
import React from 'react'
import {Provider as DrawerOpenProvider} from './drawer-open'
import {Provider as DrawerSwipableProvider} from './drawer-swipe-disabled'
import {Provider as MinimalModeProvider} from './minimal-mode'
export {useIsDrawerOpen, useSetDrawerOpen} from './drawer-open'
export {
useIsDrawerSwipeDisabled,
useSetDrawerSwipeDisabled,
} from './drawer-swipe-disabled'
export {useMinimalShellMode, useSetMinimalShellMode} from './minimal-mode'
export function Provider({children}: React.PropsWithChildren<{}>) {
return (
<DrawerOpenProvider>
<DrawerSwipableProvider>
<MinimalModeProvider>{children}</MinimalModeProvider>
</DrawerSwipableProvider>
</DrawerOpenProvider>
)
}

View File

@ -0,0 +1,24 @@
import React from 'react'
type StateContext = boolean
type SetContext = (v: boolean) => void
const stateContext = React.createContext<StateContext>(false)
const setContext = React.createContext<SetContext>((_: boolean) => {})
export function Provider({children}: React.PropsWithChildren<{}>) {
const [state, setState] = React.useState(false)
return (
<stateContext.Provider value={state}>
<setContext.Provider value={setState}>{children}</setContext.Provider>
</stateContext.Provider>
)
}
export function useMinimalShellMode() {
return React.useContext(stateContext)
}
export function useSetMinimalShellMode() {
return React.useContext(setContext)
}

View File

@ -9,6 +9,7 @@ import {usePalette} from 'lib/hooks/usePalette'
import {useStores} from 'state/index' import {useStores} from 'state/index'
import {useAnalytics} from 'lib/analytics/analytics' import {useAnalytics} from 'lib/analytics/analytics'
import {SplashScreen} from './SplashScreen' import {SplashScreen} from './SplashScreen'
import {useSetMinimalShellMode} from '#/state/shell/minimal-mode'
enum ScreenState { enum ScreenState {
S_LoginOrCreateAccount, S_LoginOrCreateAccount,
@ -19,6 +20,7 @@ enum ScreenState {
export const LoggedOut = observer(function LoggedOutImpl() { export const LoggedOut = observer(function LoggedOutImpl() {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode()
const {screen} = useAnalytics() const {screen} = useAnalytics()
const [screenState, setScreenState] = React.useState<ScreenState>( const [screenState, setScreenState] = React.useState<ScreenState>(
ScreenState.S_LoginOrCreateAccount, ScreenState.S_LoginOrCreateAccount,
@ -26,8 +28,8 @@ export const LoggedOut = observer(function LoggedOutImpl() {
React.useEffect(() => { React.useEffect(() => {
screen('Login') screen('Login')
store.shell.setMinimalShellMode(true) setMinimalShellMode(true)
}, [store, screen]) }, [screen, setMinimalShellMode])
if ( if (
store.session.isResumingSession || store.session.isResumingSession ||

View File

@ -8,14 +8,16 @@ import {useStores} from 'state/index'
import {Welcome} from './onboarding/Welcome' import {Welcome} from './onboarding/Welcome'
import {RecommendedFeeds} from './onboarding/RecommendedFeeds' import {RecommendedFeeds} from './onboarding/RecommendedFeeds'
import {RecommendedFollows} from './onboarding/RecommendedFollows' import {RecommendedFollows} from './onboarding/RecommendedFollows'
import {useSetMinimalShellMode} from '#/state/shell/minimal-mode'
export const Onboarding = observer(function OnboardingImpl() { export const Onboarding = observer(function OnboardingImpl() {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode()
React.useEffect(() => { React.useEffect(() => {
store.shell.setMinimalShellMode(true) setMinimalShellMode(true)
}, [store]) }, [setMinimalShellMode])
const next = () => store.onboarding.next() const next = () => store.onboarding.next()
const skip = () => store.onboarding.skip() const skip = () => store.onboarding.skip()

View File

@ -38,7 +38,7 @@ export const FeedPage = observer(function FeedPageImpl({
const store = useStores() const store = useStores()
const pal = usePalette('default') const pal = usePalette('default')
const {isDesktop} = useWebMediaQueries() const {isDesktop} = useWebMediaQueries()
const [onMainScroll, isScrolledDown, resetMainScroll] = useOnMainScroll(store) const [onMainScroll, isScrolledDown, resetMainScroll] = useOnMainScroll()
const {screen, track} = useAnalytics() const {screen, track} = useAnalytics()
const headerOffset = useHeaderOffset() const headerOffset = useHeaderOffset()
const scrollElRef = React.useRef<FlatList>(null) const scrollElRef = React.useRef<FlatList>(null)

View File

@ -13,21 +13,23 @@ import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {FontAwesomeIconStyle} from '@fortawesome/react-native-fontawesome' import {FontAwesomeIconStyle} from '@fortawesome/react-native-fontawesome'
import {s} from 'lib/styles' import {s} from 'lib/styles'
import {HITSLOP_10} from 'lib/constants' import {HITSLOP_10} from 'lib/constants'
import {useMinimalShellMode} from 'lib/hooks/useMinimalShellMode'
import Animated from 'react-native-reanimated' import Animated from 'react-native-reanimated'
import {useMinimalShellMode} from 'lib/hooks/useMinimalShellMode'
import {useSetDrawerOpen} from '#/state/shell/drawer-open'
export const FeedsTabBar = observer(function FeedsTabBarImpl( export const FeedsTabBar = observer(function FeedsTabBarImpl(
props: RenderTabBarFnProps & {testID?: string; onPressSelected: () => void}, props: RenderTabBarFnProps & {testID?: string; onPressSelected: () => void},
) { ) {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
const setDrawerOpen = useSetDrawerOpen()
const items = useHomeTabs(store.preferences.pinnedFeeds) const items = useHomeTabs(store.preferences.pinnedFeeds)
const brandBlue = useColorSchemeStyle(s.brandBlue, s.blue3) const brandBlue = useColorSchemeStyle(s.brandBlue, s.blue3)
const {headerMinimalShellTransform} = useMinimalShellMode() const {minimalShellMode, headerMinimalShellTransform} = useMinimalShellMode()
const onPressAvi = React.useCallback(() => { const onPressAvi = React.useCallback(() => {
store.shell.openDrawer() setDrawerOpen(true)
}, [store]) }, [setDrawerOpen])
return ( return (
<Animated.View <Animated.View
@ -36,7 +38,7 @@ export const FeedsTabBar = observer(function FeedsTabBarImpl(
pal.border, pal.border,
styles.tabBar, styles.tabBar,
headerMinimalShellTransform, headerMinimalShellTransform,
store.shell.minimalShellMode && styles.disabled, minimalShellMode && styles.disabled,
]}> ]}>
<View style={[pal.view, styles.topBar]}> <View style={[pal.view, styles.topBar]}>
<View style={[pal.view]}> <View style={[pal.view]}>

View File

@ -17,6 +17,7 @@ import {NavigationProp} from 'lib/routes/types'
import {BACK_HITSLOP} from 'lib/constants' import {BACK_HITSLOP} from 'lib/constants'
import {isNative} from 'platform/detection' import {isNative} from 'platform/detection'
import {ImagesLightbox} from 'state/models/ui/shell' import {ImagesLightbox} from 'state/models/ui/shell'
import {useSetDrawerOpen} from '#/state/shell'
export const ProfileSubpageHeader = observer(function HeaderImpl({ export const ProfileSubpageHeader = observer(function HeaderImpl({
isLoading, isLoading,
@ -42,6 +43,7 @@ export const ProfileSubpageHeader = observer(function HeaderImpl({
avatarType: UserAvatarType avatarType: UserAvatarType
}>) { }>) {
const store = useStores() const store = useStores()
const setDrawerOpen = useSetDrawerOpen()
const navigation = useNavigation<NavigationProp>() const navigation = useNavigation<NavigationProp>()
const {isMobile} = useWebMediaQueries() const {isMobile} = useWebMediaQueries()
const pal = usePalette('default') const pal = usePalette('default')
@ -56,8 +58,8 @@ export const ProfileSubpageHeader = observer(function HeaderImpl({
}, [navigation]) }, [navigation])
const onPressMenu = React.useCallback(() => { const onPressMenu = React.useCallback(() => {
store.shell.openDrawer() setDrawerOpen(true)
}, [store]) }, [setDrawerOpen])
const onPressAvi = React.useCallback(() => { const onPressAvi = React.useCallback(() => {
if ( if (

View File

@ -8,10 +8,10 @@ import {Text} from 'view/com/util/text/Text'
import {MagnifyingGlassIcon} from 'lib/icons' import {MagnifyingGlassIcon} from 'lib/icons'
import {useTheme} from 'lib/ThemeContext' import {useTheme} from 'lib/ThemeContext'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import {useStores} from 'state/index'
import {useAnalytics} from 'lib/analytics/analytics' import {useAnalytics} from 'lib/analytics/analytics'
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
import {HITSLOP_10} from 'lib/constants' import {HITSLOP_10} from 'lib/constants'
import {useSetDrawerOpen} from '#/state/shell'
interface Props { interface Props {
isInputFocused: boolean isInputFocused: boolean
@ -33,7 +33,7 @@ export function HeaderWithInput({
onSubmitQuery, onSubmitQuery,
showMenu = true, showMenu = true,
}: Props) { }: Props) {
const store = useStores() const setDrawerOpen = useSetDrawerOpen()
const theme = useTheme() const theme = useTheme()
const pal = usePalette('default') const pal = usePalette('default')
const {track} = useAnalytics() const {track} = useAnalytics()
@ -42,8 +42,8 @@ export function HeaderWithInput({
const onPressMenu = React.useCallback(() => { const onPressMenu = React.useCallback(() => {
track('ViewHeader:MenuButtonClicked') track('ViewHeader:MenuButtonClicked')
store.shell.openDrawer() setDrawerOpen(true)
}, [track, store]) }, [track, setDrawerOpen])
const onPressCancelSearchInner = React.useCallback(() => { const onPressCancelSearchInner = React.useCallback(() => {
onPressCancelSearch() onPressCancelSearch()

View File

@ -10,11 +10,11 @@ import {
import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome' import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {useNavigation} from '@react-navigation/native' import {useNavigation} from '@react-navigation/native'
import {CenteredView} from './Views' import {CenteredView} from './Views'
import {useStores} from 'state/index'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
import {useAnalytics} from 'lib/analytics/analytics' import {useAnalytics} from 'lib/analytics/analytics'
import {NavigationProp} from 'lib/routes/types' import {NavigationProp} from 'lib/routes/types'
import {useSetDrawerOpen} from '#/state/shell'
const BACK_HITSLOP = {left: 20, top: 20, right: 50, bottom: 20} const BACK_HITSLOP = {left: 20, top: 20, right: 50, bottom: 20}
@ -27,7 +27,7 @@ export const SimpleViewHeader = observer(function SimpleViewHeaderImpl({
style?: StyleProp<ViewStyle> style?: StyleProp<ViewStyle>
}>) { }>) {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const setDrawerOpen = useSetDrawerOpen()
const navigation = useNavigation<NavigationProp>() const navigation = useNavigation<NavigationProp>()
const {track} = useAnalytics() const {track} = useAnalytics()
const {isMobile} = useWebMediaQueries() const {isMobile} = useWebMediaQueries()
@ -43,8 +43,8 @@ export const SimpleViewHeader = observer(function SimpleViewHeaderImpl({
const onPressMenu = React.useCallback(() => { const onPressMenu = React.useCallback(() => {
track('ViewHeader:MenuButtonClicked') track('ViewHeader:MenuButtonClicked')
store.shell.openDrawer() setDrawerOpen(true)
}, [track, store]) }, [track, setDrawerOpen])
const Container = isMobile ? View : CenteredView const Container = isMobile ? View : CenteredView
return ( return (

View File

@ -5,13 +5,13 @@ import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {useNavigation} from '@react-navigation/native' import {useNavigation} from '@react-navigation/native'
import {CenteredView} from './Views' import {CenteredView} from './Views'
import {Text} from './text/Text' import {Text} from './text/Text'
import {useStores} from 'state/index'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
import {useAnalytics} from 'lib/analytics/analytics' import {useAnalytics} from 'lib/analytics/analytics'
import {NavigationProp} from 'lib/routes/types' import {NavigationProp} from 'lib/routes/types'
import {useMinimalShellMode} from 'lib/hooks/useMinimalShellMode' import {useMinimalShellMode} from 'lib/hooks/useMinimalShellMode'
import Animated from 'react-native-reanimated' import Animated from 'react-native-reanimated'
import {useSetDrawerOpen} from '#/state/shell'
const BACK_HITSLOP = {left: 20, top: 20, right: 50, bottom: 20} const BACK_HITSLOP = {left: 20, top: 20, right: 50, bottom: 20}
@ -33,7 +33,7 @@ export const ViewHeader = observer(function ViewHeaderImpl({
renderButton?: () => JSX.Element renderButton?: () => JSX.Element
}) { }) {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const setDrawerOpen = useSetDrawerOpen()
const navigation = useNavigation<NavigationProp>() const navigation = useNavigation<NavigationProp>()
const {track} = useAnalytics() const {track} = useAnalytics()
const {isDesktop, isTablet} = useWebMediaQueries() const {isDesktop, isTablet} = useWebMediaQueries()
@ -48,8 +48,8 @@ export const ViewHeader = observer(function ViewHeaderImpl({
const onPressMenu = React.useCallback(() => { const onPressMenu = React.useCallback(() => {
track('ViewHeader:MenuButtonClicked') track('ViewHeader:MenuButtonClicked')
store.shell.openDrawer() setDrawerOpen(true)
}, [track, store]) }, [track, setDrawerOpen])
if (isDesktop) { if (isDesktop) {
if (showOnDesktop) { if (showOnDesktop) {

View File

@ -16,20 +16,22 @@ import {useAnalytics} from 'lib/analytics/analytics'
import {useFocusEffect} from '@react-navigation/native' import {useFocusEffect} from '@react-navigation/native'
import {ViewHeader} from '../com/util/ViewHeader' import {ViewHeader} from '../com/util/ViewHeader'
import {CenteredView} from 'view/com/util/Views' import {CenteredView} from 'view/com/util/Views'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'AppPasswords'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'AppPasswords'>
export const AppPasswords = withAuthRequired( export const AppPasswords = withAuthRequired(
observer(function AppPasswordsImpl({}: Props) { observer(function AppPasswordsImpl({}: Props) {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode()
const {screen} = useAnalytics() const {screen} = useAnalytics()
const {isTabletOrDesktop} = useWebMediaQueries() const {isTabletOrDesktop} = useWebMediaQueries()
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
screen('AppPasswords') screen('AppPasswords')
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [screen, store]), }, [screen, setMinimalShellMode]),
) )
const onAdd = React.useCallback(async () => { const onAdd = React.useCallback(async () => {

View File

@ -5,10 +5,10 @@ import {Text} from 'view/com/util/text/Text'
import {TextLink} from 'view/com/util/Link' import {TextLink} from 'view/com/util/Link'
import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types'
import {ViewHeader} from '../com/util/ViewHeader' import {ViewHeader} from '../com/util/ViewHeader'
import {useStores} from 'state/index'
import {ScrollView} from 'view/com/util/Views' import {ScrollView} from 'view/com/util/Views'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import {s} from 'lib/styles' import {s} from 'lib/styles'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps< type Props = NativeStackScreenProps<
CommonNavigatorParams, CommonNavigatorParams,
@ -16,12 +16,12 @@ type Props = NativeStackScreenProps<
> >
export const CommunityGuidelinesScreen = (_props: Props) => { export const CommunityGuidelinesScreen = (_props: Props) => {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const setMinimalShellMode = useSetMinimalShellMode()
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [store]), }, [setMinimalShellMode]),
) )
return ( return (

View File

@ -5,20 +5,20 @@ import {Text} from 'view/com/util/text/Text'
import {TextLink} from 'view/com/util/Link' import {TextLink} from 'view/com/util/Link'
import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types'
import {ViewHeader} from '../com/util/ViewHeader' import {ViewHeader} from '../com/util/ViewHeader'
import {useStores} from 'state/index'
import {ScrollView} from 'view/com/util/Views' import {ScrollView} from 'view/com/util/Views'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import {s} from 'lib/styles' import {s} from 'lib/styles'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'CopyrightPolicy'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'CopyrightPolicy'>
export const CopyrightPolicyScreen = (_props: Props) => { export const CopyrightPolicyScreen = (_props: Props) => {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const setMinimalShellMode = useSetMinimalShellMode()
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [store]), }, [setMinimalShellMode]),
) )
return ( return (

View File

@ -27,12 +27,14 @@ import {FeedSourceModel} from 'state/models/content/feed-source'
import {FlatList} from 'view/com/util/Views' import {FlatList} from 'view/com/util/Views'
import {useFocusEffect} from '@react-navigation/native' import {useFocusEffect} from '@react-navigation/native'
import {FeedSourceCard} from 'view/com/feeds/FeedSourceCard' import {FeedSourceCard} from 'view/com/feeds/FeedSourceCard'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<FeedsTabNavigatorParams, 'Feeds'> type Props = NativeStackScreenProps<FeedsTabNavigatorParams, 'Feeds'>
export const FeedsScreen = withAuthRequired( export const FeedsScreen = withAuthRequired(
observer<Props>(function FeedsScreenImpl({}: Props) { observer<Props>(function FeedsScreenImpl({}: Props) {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode()
const {isMobile, isTabletOrDesktop} = useWebMediaQueries() const {isMobile, isTabletOrDesktop} = useWebMediaQueries()
const myFeeds = store.me.myFeeds const myFeeds = store.me.myFeeds
const [query, setQuery] = React.useState<string>('') const [query, setQuery] = React.useState<string>('')
@ -43,14 +45,14 @@ export const FeedsScreen = withAuthRequired(
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
myFeeds.setup() myFeeds.setup()
const softResetSub = store.onScreenSoftReset(() => myFeeds.refresh()) const softResetSub = store.onScreenSoftReset(() => myFeeds.refresh())
return () => { return () => {
softResetSub.remove() softResetSub.remove()
} }
}, [store, myFeeds]), }, [store, myFeeds, setMinimalShellMode]),
) )
React.useEffect(() => { React.useEffect(() => {
// watch for changes to saved/pinned feeds // watch for changes to saved/pinned feeds

View File

@ -14,6 +14,7 @@ import {Pager, PagerRef, RenderTabBarFnProps} from 'view/com/pager/Pager'
import {useStores} from 'state/index' import {useStores} from 'state/index'
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
import {FeedPage} from 'view/com/feeds/FeedPage' import {FeedPage} from 'view/com/feeds/FeedPage'
import {useSetMinimalShellMode, useSetDrawerSwipeDisabled} from '#/state/shell'
export const POLL_FREQ = 30e3 // 30sec export const POLL_FREQ = 30e3 // 30sec
@ -21,6 +22,8 @@ type Props = NativeStackScreenProps<HomeTabNavigatorParams, 'Home'>
export const HomeScreen = withAuthRequired( export const HomeScreen = withAuthRequired(
observer(function HomeScreenImpl({}: Props) { observer(function HomeScreenImpl({}: Props) {
const store = useStores() const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode()
const setDrawerSwipeDisabled = useSetDrawerSwipeDisabled()
const pagerRef = React.useRef<PagerRef>(null) const pagerRef = React.useRef<PagerRef>(null)
const [selectedPage, setSelectedPage] = React.useState(0) const [selectedPage, setSelectedPage] = React.useState(0)
const [customFeeds, setCustomFeeds] = React.useState<PostsFeedModel[]>([]) const [customFeeds, setCustomFeeds] = React.useState<PostsFeedModel[]>([])
@ -61,21 +64,21 @@ export const HomeScreen = withAuthRequired(
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
store.shell.setIsDrawerSwipeDisabled(selectedPage > 0) setDrawerSwipeDisabled(selectedPage > 0)
return () => { return () => {
store.shell.setIsDrawerSwipeDisabled(false) setDrawerSwipeDisabled(false)
} }
}, [store, selectedPage]), }, [setDrawerSwipeDisabled, selectedPage, setMinimalShellMode]),
) )
const onPageSelected = React.useCallback( const onPageSelected = React.useCallback(
(index: number) => { (index: number) => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
setSelectedPage(index) setSelectedPage(index)
store.shell.setIsDrawerSwipeDisabled(index > 0) setDrawerSwipeDisabled(index > 0)
}, },
[store, setSelectedPage], [setDrawerSwipeDisabled, setSelectedPage, setMinimalShellMode],
) )
const onPressSelected = React.useCallback(() => { const onPressSelected = React.useCallback(() => {

View File

@ -18,6 +18,7 @@ import {useAnalytics} from 'lib/analytics/analytics'
import {useFocusEffect} from '@react-navigation/native' import {useFocusEffect} from '@react-navigation/native'
import {LANGUAGES} from 'lib/../locale/languages' import {LANGUAGES} from 'lib/../locale/languages'
import RNPickerSelect, {PickerSelectProps} from 'react-native-picker-select' import RNPickerSelect, {PickerSelectProps} from 'react-native-picker-select'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'LanguageSettings'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'LanguageSettings'>
@ -28,12 +29,13 @@ export const LanguageSettingsScreen = observer(function LanguageSettingsImpl(
const store = useStores() const store = useStores()
const {isTabletOrDesktop} = useWebMediaQueries() const {isTabletOrDesktop} = useWebMediaQueries()
const {screen, track} = useAnalytics() const {screen, track} = useAnalytics()
const setMinimalShellMode = useSetMinimalShellMode()
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
screen('Settings') screen('Settings')
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [screen, store]), }, [screen, setMinimalShellMode]),
) )
const onPressContentLanguages = React.useCallback(() => { const onPressContentLanguages = React.useCallback(() => {

View File

@ -16,12 +16,14 @@ import {usePalette} from 'lib/hooks/usePalette'
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
import {SimpleViewHeader} from 'view/com/util/SimpleViewHeader' import {SimpleViewHeader} from 'view/com/util/SimpleViewHeader'
import {s} from 'lib/styles' import {s} from 'lib/styles'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'Lists'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'Lists'>
export const ListsScreen = withAuthRequired( export const ListsScreen = withAuthRequired(
observer(function ListsScreenImpl({}: Props) { observer(function ListsScreenImpl({}: Props) {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode()
const {isMobile} = useWebMediaQueries() const {isMobile} = useWebMediaQueries()
const navigation = useNavigation<NavigationProp>() const navigation = useNavigation<NavigationProp>()
@ -32,9 +34,9 @@ export const ListsScreen = withAuthRequired(
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
listsLists.refresh() listsLists.refresh()
}, [store, listsLists]), }, [listsLists, setMinimalShellMode]),
) )
const onPressNewList = React.useCallback(() => { const onPressNewList = React.useCallback(() => {

View File

@ -5,26 +5,26 @@ import {observer} from 'mobx-react-lite'
import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome' import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types'
import {ScrollView} from '../com/util/Views' import {ScrollView} from '../com/util/Views'
import {useStores} from 'state/index'
import {s} from 'lib/styles' import {s} from 'lib/styles'
import {ViewHeader} from '../com/util/ViewHeader' import {ViewHeader} from '../com/util/ViewHeader'
import {Text} from '../com/util/text/Text' import {Text} from '../com/util/text/Text'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import {getEntries} from '#/logger/logDump' import {getEntries} from '#/logger/logDump'
import {ago} from 'lib/strings/time' import {ago} from 'lib/strings/time'
import {useSetMinimalShellMode} from '#/state/shell'
export const LogScreen = observer(function Log({}: NativeStackScreenProps< export const LogScreen = observer(function Log({}: NativeStackScreenProps<
CommonNavigatorParams, CommonNavigatorParams,
'Log' 'Log'
>) { >) {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const setMinimalShellMode = useSetMinimalShellMode()
const [expanded, setExpanded] = React.useState<string[]>([]) const [expanded, setExpanded] = React.useState<string[]>([])
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [store]), }, [setMinimalShellMode]),
) )
const toggler = (id: string) => () => { const toggler = (id: string) => () => {

View File

@ -17,20 +17,22 @@ import {Text} from '../com/util/text/Text'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import {useAnalytics} from 'lib/analytics/analytics' import {useAnalytics} from 'lib/analytics/analytics'
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'Moderation'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'Moderation'>
export const ModerationScreen = withAuthRequired( export const ModerationScreen = withAuthRequired(
observer(function Moderation({}: Props) { observer(function Moderation({}: Props) {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode()
const {screen, track} = useAnalytics() const {screen, track} = useAnalytics()
const {isTabletOrDesktop} = useWebMediaQueries() const {isTabletOrDesktop} = useWebMediaQueries()
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
screen('Moderation') screen('Moderation')
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [screen, store]), }, [screen, setMinimalShellMode]),
) )
const onPressContentFiltering = React.useCallback(() => { const onPressContentFiltering = React.useCallback(() => {

View File

@ -22,6 +22,7 @@ import {ViewHeader} from '../com/util/ViewHeader'
import {CenteredView} from 'view/com/util/Views' import {CenteredView} from 'view/com/util/Views'
import {ProfileCard} from 'view/com/profile/ProfileCard' import {ProfileCard} from 'view/com/profile/ProfileCard'
import {logger} from '#/logger' import {logger} from '#/logger'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps< type Props = NativeStackScreenProps<
CommonNavigatorParams, CommonNavigatorParams,
@ -31,6 +32,7 @@ export const ModerationBlockedAccounts = withAuthRequired(
observer(function ModerationBlockedAccountsImpl({}: Props) { observer(function ModerationBlockedAccountsImpl({}: Props) {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode()
const {isTabletOrDesktop} = useWebMediaQueries() const {isTabletOrDesktop} = useWebMediaQueries()
const {screen} = useAnalytics() const {screen} = useAnalytics()
const blockedAccounts = useMemo( const blockedAccounts = useMemo(
@ -41,9 +43,9 @@ export const ModerationBlockedAccounts = withAuthRequired(
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
screen('BlockedAccounts') screen('BlockedAccounts')
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
blockedAccounts.refresh() blockedAccounts.refresh()
}, [screen, store, blockedAccounts]), }, [screen, setMinimalShellMode, blockedAccounts]),
) )
const onRefresh = React.useCallback(() => { const onRefresh = React.useCallback(() => {

View File

@ -16,12 +16,14 @@ import {usePalette} from 'lib/hooks/usePalette'
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
import {SimpleViewHeader} from 'view/com/util/SimpleViewHeader' import {SimpleViewHeader} from 'view/com/util/SimpleViewHeader'
import {s} from 'lib/styles' import {s} from 'lib/styles'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'ModerationModlists'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'ModerationModlists'>
export const ModerationModlistsScreen = withAuthRequired( export const ModerationModlistsScreen = withAuthRequired(
observer(function ModerationModlistsScreenImpl({}: Props) { observer(function ModerationModlistsScreenImpl({}: Props) {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode()
const {isMobile} = useWebMediaQueries() const {isMobile} = useWebMediaQueries()
const navigation = useNavigation<NavigationProp>() const navigation = useNavigation<NavigationProp>()
@ -32,9 +34,9 @@ export const ModerationModlistsScreen = withAuthRequired(
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
mutelists.refresh() mutelists.refresh()
}, [store, mutelists]), }, [mutelists, setMinimalShellMode]),
) )
const onPressNewList = React.useCallback(() => { const onPressNewList = React.useCallback(() => {

View File

@ -22,6 +22,7 @@ import {ViewHeader} from '../com/util/ViewHeader'
import {CenteredView} from 'view/com/util/Views' import {CenteredView} from 'view/com/util/Views'
import {ProfileCard} from 'view/com/profile/ProfileCard' import {ProfileCard} from 'view/com/profile/ProfileCard'
import {logger} from '#/logger' import {logger} from '#/logger'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps< type Props = NativeStackScreenProps<
CommonNavigatorParams, CommonNavigatorParams,
@ -31,6 +32,7 @@ export const ModerationMutedAccounts = withAuthRequired(
observer(function ModerationMutedAccountsImpl({}: Props) { observer(function ModerationMutedAccountsImpl({}: Props) {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode()
const {isTabletOrDesktop} = useWebMediaQueries() const {isTabletOrDesktop} = useWebMediaQueries()
const {screen} = useAnalytics() const {screen} = useAnalytics()
const mutedAccounts = useMemo(() => new MutedAccountsModel(store), [store]) const mutedAccounts = useMemo(() => new MutedAccountsModel(store), [store])
@ -38,9 +40,9 @@ export const ModerationMutedAccounts = withAuthRequired(
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
screen('MutedAccounts') screen('MutedAccounts')
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
mutedAccounts.refresh() mutedAccounts.refresh()
}, [screen, store, mutedAccounts]), }, [screen, setMinimalShellMode, mutedAccounts]),
) )
const onRefresh = React.useCallback(() => { const onRefresh = React.useCallback(() => {

View File

@ -10,18 +10,18 @@ import {Text} from '../com/util/text/Text'
import {Button} from 'view/com/util/forms/Button' import {Button} from 'view/com/util/forms/Button'
import {NavigationProp} from 'lib/routes/types' import {NavigationProp} from 'lib/routes/types'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import {useStores} from 'state/index'
import {s} from 'lib/styles' import {s} from 'lib/styles'
import {useSetMinimalShellMode} from '#/state/shell'
export const NotFoundScreen = () => { export const NotFoundScreen = () => {
const pal = usePalette('default') const pal = usePalette('default')
const navigation = useNavigation<NavigationProp>() const navigation = useNavigation<NavigationProp>()
const store = useStores() const setMinimalShellMode = useSetMinimalShellMode()
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [store]), }, [setMinimalShellMode]),
) )
const canGoBack = navigation.canGoBack() const canGoBack = navigation.canGoBack()

View File

@ -21,6 +21,7 @@ import {s, colors} from 'lib/styles'
import {useAnalytics} from 'lib/analytics/analytics' import {useAnalytics} from 'lib/analytics/analytics'
import {isWeb} from 'platform/detection' import {isWeb} from 'platform/detection'
import {logger} from '#/logger' import {logger} from '#/logger'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps< type Props = NativeStackScreenProps<
NotificationsTabNavigatorParams, NotificationsTabNavigatorParams,
@ -29,8 +30,8 @@ type Props = NativeStackScreenProps<
export const NotificationsScreen = withAuthRequired( export const NotificationsScreen = withAuthRequired(
observer(function NotificationsScreenImpl({}: Props) { observer(function NotificationsScreenImpl({}: Props) {
const store = useStores() const store = useStores()
const [onMainScroll, isScrolledDown, resetMainScroll] = const setMinimalShellMode = useSetMinimalShellMode()
useOnMainScroll(store) const [onMainScroll, isScrolledDown, resetMainScroll] = useOnMainScroll()
const scrollElRef = React.useRef<FlatList>(null) const scrollElRef = React.useRef<FlatList>(null)
const {screen} = useAnalytics() const {screen} = useAnalytics()
const pal = usePalette('default') const pal = usePalette('default')
@ -60,7 +61,7 @@ export const NotificationsScreen = withAuthRequired(
// = // =
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
logger.debug('NotificationsScreen: Updating feed') logger.debug('NotificationsScreen: Updating feed')
const softResetSub = store.onScreenSoftReset(onPressLoadLatest) const softResetSub = store.onScreenSoftReset(onPressLoadLatest)
store.me.notifications.update() store.me.notifications.update()
@ -70,7 +71,7 @@ export const NotificationsScreen = withAuthRequired(
softResetSub.remove() softResetSub.remove()
store.me.notifications.markAllRead() store.me.notifications.markAllRead()
} }
}, [store, screen, onPressLoadLatest]), }, [store, screen, onPressLoadLatest, setMinimalShellMode]),
) )
useTabFocusEffect( useTabFocusEffect(

View File

@ -5,19 +5,19 @@ import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types'
import {withAuthRequired} from 'view/com/auth/withAuthRequired' import {withAuthRequired} from 'view/com/auth/withAuthRequired'
import {ViewHeader} from '../com/util/ViewHeader' import {ViewHeader} from '../com/util/ViewHeader'
import {PostLikedBy as PostLikedByComponent} from '../com/post-thread/PostLikedBy' import {PostLikedBy as PostLikedByComponent} from '../com/post-thread/PostLikedBy'
import {useStores} from 'state/index'
import {makeRecordUri} from 'lib/strings/url-helpers' import {makeRecordUri} from 'lib/strings/url-helpers'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'PostLikedBy'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'PostLikedBy'>
export const PostLikedByScreen = withAuthRequired(({route}: Props) => { export const PostLikedByScreen = withAuthRequired(({route}: Props) => {
const store = useStores() const setMinimalShellMode = useSetMinimalShellMode()
const {name, rkey} = route.params const {name, rkey} = route.params
const uri = makeRecordUri(name, 'app.bsky.feed.post', rkey) const uri = makeRecordUri(name, 'app.bsky.feed.post', rkey)
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [store]), }, [setMinimalShellMode]),
) )
return ( return (

View File

@ -5,19 +5,19 @@ import {withAuthRequired} from 'view/com/auth/withAuthRequired'
import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types'
import {ViewHeader} from '../com/util/ViewHeader' import {ViewHeader} from '../com/util/ViewHeader'
import {PostRepostedBy as PostRepostedByComponent} from '../com/post-thread/PostRepostedBy' import {PostRepostedBy as PostRepostedByComponent} from '../com/post-thread/PostRepostedBy'
import {useStores} from 'state/index'
import {makeRecordUri} from 'lib/strings/url-helpers' import {makeRecordUri} from 'lib/strings/url-helpers'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'PostRepostedBy'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'PostRepostedBy'>
export const PostRepostedByScreen = withAuthRequired(({route}: Props) => { export const PostRepostedByScreen = withAuthRequired(({route}: Props) => {
const store = useStores()
const {name, rkey} = route.params const {name, rkey} = route.params
const uri = makeRecordUri(name, 'app.bsky.feed.post', rkey) const uri = makeRecordUri(name, 'app.bsky.feed.post', rkey)
const setMinimalShellMode = useSetMinimalShellMode()
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [store]), }, [setMinimalShellMode]),
) )
return ( return (

View File

@ -15,6 +15,7 @@ import {useSafeAreaInsets} from 'react-native-safe-area-context'
import {clamp} from 'lodash' import {clamp} from 'lodash'
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
import {logger} from '#/logger' import {logger} from '#/logger'
import {useMinimalShellMode, useSetMinimalShellMode} from '#/state/shell'
const SHELL_FOOTER_HEIGHT = 44 const SHELL_FOOTER_HEIGHT = 44
@ -22,6 +23,8 @@ type Props = NativeStackScreenProps<CommonNavigatorParams, 'PostThread'>
export const PostThreadScreen = withAuthRequired( export const PostThreadScreen = withAuthRequired(
observer(function PostThreadScreenImpl({route}: Props) { observer(function PostThreadScreenImpl({route}: Props) {
const store = useStores() const store = useStores()
const minimalShellMode = useMinimalShellMode()
const setMinimalShellMode = useSetMinimalShellMode()
const safeAreaInsets = useSafeAreaInsets() const safeAreaInsets = useSafeAreaInsets()
const {name, rkey} = route.params const {name, rkey} = route.params
const uri = makeRecordUri(name, 'app.bsky.feed.post', rkey) const uri = makeRecordUri(name, 'app.bsky.feed.post', rkey)
@ -33,7 +36,7 @@ export const PostThreadScreen = withAuthRequired(
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
const threadCleanup = view.registerListeners() const threadCleanup = view.registerListeners()
InteractionManager.runAfterInteractions(() => { InteractionManager.runAfterInteractions(() => {
@ -47,7 +50,7 @@ export const PostThreadScreen = withAuthRequired(
return () => { return () => {
threadCleanup() threadCleanup()
} }
}, [store, view]), }, [view, setMinimalShellMode]),
) )
const onPressReply = React.useCallback(() => { const onPressReply = React.useCallback(() => {
@ -80,7 +83,7 @@ export const PostThreadScreen = withAuthRequired(
treeView={!!store.preferences.thread.lab_treeViewEnabled} treeView={!!store.preferences.thread.lab_treeViewEnabled}
/> />
</View> </View>
{isMobile && !store.shell.minimalShellMode && ( {isMobile && !minimalShellMode && (
<View <View
style={[ style={[
styles.prompt, styles.prompt,

View File

@ -5,20 +5,20 @@ import {Text} from 'view/com/util/text/Text'
import {TextLink} from 'view/com/util/Link' import {TextLink} from 'view/com/util/Link'
import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types'
import {ViewHeader} from '../com/util/ViewHeader' import {ViewHeader} from '../com/util/ViewHeader'
import {useStores} from 'state/index'
import {ScrollView} from 'view/com/util/Views' import {ScrollView} from 'view/com/util/Views'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import {s} from 'lib/styles' import {s} from 'lib/styles'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'PrivacyPolicy'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'PrivacyPolicy'>
export const PrivacyPolicyScreen = (_props: Props) => { export const PrivacyPolicyScreen = (_props: Props) => {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const setMinimalShellMode = useSetMinimalShellMode()
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [store]), }, [setMinimalShellMode]),
) )
return ( return (

View File

@ -30,11 +30,13 @@ import {FeedSourceModel} from 'state/models/content/feed-source'
import {useSetTitle} from 'lib/hooks/useSetTitle' import {useSetTitle} from 'lib/hooks/useSetTitle'
import {combinedDisplayName} from 'lib/strings/display-names' import {combinedDisplayName} from 'lib/strings/display-names'
import {logger} from '#/logger' import {logger} from '#/logger'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'Profile'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'Profile'>
export const ProfileScreen = withAuthRequired( export const ProfileScreen = withAuthRequired(
observer(function ProfileScreenImpl({route}: Props) { observer(function ProfileScreenImpl({route}: Props) {
const store = useStores() const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode()
const {screen, track} = useAnalytics() const {screen, track} = useAnalytics()
const viewSelectorRef = React.useRef<ViewSelectorHandle>(null) const viewSelectorRef = React.useRef<ViewSelectorHandle>(null)
const name = route.params.name === 'me' ? store.me.did : route.params.name const name = route.params.name === 'me' ? store.me.did : route.params.name
@ -69,7 +71,7 @@ export const ProfileScreen = withAuthRequired(
React.useCallback(() => { React.useCallback(() => {
const softResetSub = store.onScreenSoftReset(onSoftReset) const softResetSub = store.onScreenSoftReset(onSoftReset)
let aborted = false let aborted = false
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
const feedCleanup = uiState.feed.registerListeners() const feedCleanup = uiState.feed.registerListeners()
if (!hasSetup) { if (!hasSetup) {
uiState.setup().then(() => { uiState.setup().then(() => {
@ -84,7 +86,7 @@ export const ProfileScreen = withAuthRequired(
feedCleanup() feedCleanup()
softResetSub.remove() softResetSub.remove()
} }
}, [store, onSoftReset, uiState, hasSetup]), }, [store, onSoftReset, uiState, hasSetup, setMinimalShellMode]),
) )
// events // events

View File

@ -5,19 +5,19 @@ import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types'
import {withAuthRequired} from 'view/com/auth/withAuthRequired' import {withAuthRequired} from 'view/com/auth/withAuthRequired'
import {ViewHeader} from '../com/util/ViewHeader' import {ViewHeader} from '../com/util/ViewHeader'
import {PostLikedBy as PostLikedByComponent} from '../com/post-thread/PostLikedBy' import {PostLikedBy as PostLikedByComponent} from '../com/post-thread/PostLikedBy'
import {useStores} from 'state/index'
import {makeRecordUri} from 'lib/strings/url-helpers' import {makeRecordUri} from 'lib/strings/url-helpers'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'ProfileFeedLikedBy'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'ProfileFeedLikedBy'>
export const ProfileFeedLikedByScreen = withAuthRequired(({route}: Props) => { export const ProfileFeedLikedByScreen = withAuthRequired(({route}: Props) => {
const store = useStores() const setMinimalShellMode = useSetMinimalShellMode()
const {name, rkey} = route.params const {name, rkey} = route.params
const uri = makeRecordUri(name, 'app.bsky.feed.generator', rkey) const uri = makeRecordUri(name, 'app.bsky.feed.generator', rkey)
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [store]), }, [setMinimalShellMode]),
) )
return ( return (

View File

@ -5,17 +5,17 @@ import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types'
import {withAuthRequired} from 'view/com/auth/withAuthRequired' import {withAuthRequired} from 'view/com/auth/withAuthRequired'
import {ViewHeader} from '../com/util/ViewHeader' import {ViewHeader} from '../com/util/ViewHeader'
import {ProfileFollowers as ProfileFollowersComponent} from '../com/profile/ProfileFollowers' import {ProfileFollowers as ProfileFollowersComponent} from '../com/profile/ProfileFollowers'
import {useStores} from 'state/index' import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'ProfileFollowers'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'ProfileFollowers'>
export const ProfileFollowersScreen = withAuthRequired(({route}: Props) => { export const ProfileFollowersScreen = withAuthRequired(({route}: Props) => {
const store = useStores()
const {name} = route.params const {name} = route.params
const setMinimalShellMode = useSetMinimalShellMode()
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [store]), }, [setMinimalShellMode]),
) )
return ( return (

View File

@ -5,17 +5,17 @@ import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types'
import {withAuthRequired} from 'view/com/auth/withAuthRequired' import {withAuthRequired} from 'view/com/auth/withAuthRequired'
import {ViewHeader} from '../com/util/ViewHeader' import {ViewHeader} from '../com/util/ViewHeader'
import {ProfileFollows as ProfileFollowsComponent} from '../com/profile/ProfileFollows' import {ProfileFollows as ProfileFollowsComponent} from '../com/profile/ProfileFollows'
import {useStores} from 'state/index' import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'ProfileFollows'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'ProfileFollows'>
export const ProfileFollowsScreen = withAuthRequired(({route}: Props) => { export const ProfileFollowsScreen = withAuthRequired(({route}: Props) => {
const store = useStores()
const {name} = route.params const {name} = route.params
const setMinimalShellMode = useSetMinimalShellMode()
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [store]), }, [setMinimalShellMode]),
) )
return ( return (

View File

@ -45,6 +45,7 @@ import {makeProfileLink, makeListLink} from 'lib/routes/links'
import {ComposeIcon2} from 'lib/icons' import {ComposeIcon2} from 'lib/icons'
import {ListItems} from 'view/com/lists/ListItems' import {ListItems} from 'view/com/lists/ListItems'
import {logger} from '#/logger' import {logger} from '#/logger'
import {useSetMinimalShellMode} from '#/state/shell'
const SECTION_TITLES_CURATE = ['Posts', 'About'] const SECTION_TITLES_CURATE = ['Posts', 'About']
const SECTION_TITLES_MOD = ['About'] const SECTION_TITLES_MOD = ['About']
@ -105,6 +106,7 @@ export const ProfileListScreenInner = observer(
listOwnerDid, listOwnerDid,
}: Props & {listOwnerDid: string}) { }: Props & {listOwnerDid: string}) {
const store = useStores() const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode()
const {rkey} = route.params const {rkey} = route.params
const feedSectionRef = React.useRef<SectionRef>(null) const feedSectionRef = React.useRef<SectionRef>(null)
const aboutSectionRef = React.useRef<SectionRef>(null) const aboutSectionRef = React.useRef<SectionRef>(null)
@ -124,13 +126,13 @@ export const ProfileListScreenInner = observer(
useFocusEffect( useFocusEffect(
useCallback(() => { useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
list.loadMore(true).then(() => { list.loadMore(true).then(() => {
if (list.isCuratelist) { if (list.isCuratelist) {
feed.setup() feed.setup()
} }
}) })
}, [store, list, feed]), }, [setMinimalShellMode, list, feed]),
) )
const onPressAddUser = useCallback(() => { const onPressAddUser = useCallback(() => {

View File

@ -27,6 +27,7 @@ import * as Toast from 'view/com/util/Toast'
import {Haptics} from 'lib/haptics' import {Haptics} from 'lib/haptics'
import {TextLink} from 'view/com/util/Link' import {TextLink} from 'view/com/util/Link'
import {logger} from '#/logger' import {logger} from '#/logger'
import {useSetMinimalShellMode} from '#/state/shell'
const HITSLOP_TOP = { const HITSLOP_TOP = {
top: 20, top: 20,
@ -48,6 +49,7 @@ export const SavedFeeds = withAuthRequired(
const store = useStores() const store = useStores()
const {isMobile, isTabletOrDesktop} = useWebMediaQueries() const {isMobile, isTabletOrDesktop} = useWebMediaQueries()
const {screen} = useAnalytics() const {screen} = useAnalytics()
const setMinimalShellMode = useSetMinimalShellMode()
const savedFeeds = useMemo(() => { const savedFeeds = useMemo(() => {
const model = new SavedFeedsModel(store) const model = new SavedFeedsModel(store)
@ -57,9 +59,9 @@ export const SavedFeeds = withAuthRequired(
useFocusEffect( useFocusEffect(
useCallback(() => { useCallback(() => {
screen('SavedFeeds') screen('SavedFeeds')
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
savedFeeds.refresh() savedFeeds.refresh()
}, [screen, store, savedFeeds]), }, [screen, setMinimalShellMode, savedFeeds]),
) )
return ( return (

View File

@ -27,15 +27,18 @@ import {ProfileCard} from 'view/com/profile/ProfileCard'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import {useOnMainScroll} from 'lib/hooks/useOnMainScroll' import {useOnMainScroll} from 'lib/hooks/useOnMainScroll'
import {isAndroid, isIOS} from 'platform/detection' import {isAndroid, isIOS} from 'platform/detection'
import {useSetMinimalShellMode, useSetDrawerSwipeDisabled} from '#/state/shell'
type Props = NativeStackScreenProps<SearchTabNavigatorParams, 'Search'> type Props = NativeStackScreenProps<SearchTabNavigatorParams, 'Search'>
export const SearchScreen = withAuthRequired( export const SearchScreen = withAuthRequired(
observer<Props>(function SearchScreenImpl({}: Props) { observer<Props>(function SearchScreenImpl({}: Props) {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode()
const setIsDrawerSwipeDisabled = useSetDrawerSwipeDisabled()
const scrollViewRef = React.useRef<ScrollView>(null) const scrollViewRef = React.useRef<ScrollView>(null)
const flatListRef = React.useRef<FlatList>(null) const flatListRef = React.useRef<FlatList>(null)
const [onMainScroll] = useOnMainScroll(store) const [onMainScroll] = useOnMainScroll()
const [isInputFocused, setIsInputFocused] = React.useState<boolean>(false) const [isInputFocused, setIsInputFocused] = React.useState<boolean>(false)
const [query, setQuery] = React.useState<string>('') const [query, setQuery] = React.useState<string>('')
const autocompleteView = React.useMemo<UserAutocompleteModel>( const autocompleteView = React.useMemo<UserAutocompleteModel>(
@ -75,8 +78,8 @@ export const SearchScreen = withAuthRequired(
setQuery('') setQuery('')
autocompleteView.setActive(false) autocompleteView.setActive(false)
setSearchUIModel(undefined) setSearchUIModel(undefined)
store.shell.setIsDrawerSwipeDisabled(false) setIsDrawerSwipeDisabled(false)
}, [setQuery, autocompleteView, store]) }, [setQuery, autocompleteView, setIsDrawerSwipeDisabled])
const onSubmitQuery = React.useCallback(() => { const onSubmitQuery = React.useCallback(() => {
if (query.length === 0) { if (query.length === 0) {
@ -86,8 +89,8 @@ export const SearchScreen = withAuthRequired(
const model = new SearchUIModel(store) const model = new SearchUIModel(store)
model.fetch(query) model.fetch(query)
setSearchUIModel(model) setSearchUIModel(model)
store.shell.setIsDrawerSwipeDisabled(true) setIsDrawerSwipeDisabled(true)
}, [query, setSearchUIModel, store]) }, [query, setSearchUIModel, store, setIsDrawerSwipeDisabled])
const onSoftReset = React.useCallback(() => { const onSoftReset = React.useCallback(() => {
scrollViewRef.current?.scrollTo({x: 0, y: 0}) scrollViewRef.current?.scrollTo({x: 0, y: 0})
@ -102,7 +105,7 @@ export const SearchScreen = withAuthRequired(
softResetSub.remove() softResetSub.remove()
} }
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
autocompleteView.setup() autocompleteView.setup()
if (!foafs.hasData) { if (!foafs.hasData) {
foafs.fetch() foafs.fetch()
@ -112,7 +115,14 @@ export const SearchScreen = withAuthRequired(
} }
return cleanup return cleanup
}, [store, autocompleteView, foafs, suggestedActors, onSoftReset]), }, [
store,
autocompleteView,
foafs,
suggestedActors,
onSoftReset,
setMinimalShellMode,
]),
) )
const onPress = useCallback(() => { const onPress = useCallback(() => {

View File

@ -46,6 +46,7 @@ import Clipboard from '@react-native-clipboard/clipboard'
import {makeProfileLink} from 'lib/routes/links' import {makeProfileLink} from 'lib/routes/links'
import {AccountDropdownBtn} from 'view/com/util/AccountDropdownBtn' import {AccountDropdownBtn} from 'view/com/util/AccountDropdownBtn'
import {logger} from '#/logger' import {logger} from '#/logger'
import {useSetMinimalShellMode} from '#/state/shell'
// TEMPORARY (APP-700) // TEMPORARY (APP-700)
// remove after backend testing finishes // remove after backend testing finishes
@ -58,6 +59,7 @@ export const SettingsScreen = withAuthRequired(
observer(function Settings({}: Props) { observer(function Settings({}: Props) {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode()
const navigation = useNavigation<NavigationProp>() const navigation = useNavigation<NavigationProp>()
const {isMobile} = useWebMediaQueries() const {isMobile} = useWebMediaQueries()
const {screen, track} = useAnalytics() const {screen, track} = useAnalytics()
@ -88,8 +90,8 @@ export const SettingsScreen = withAuthRequired(
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
screen('Settings') screen('Settings')
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [screen, store]), }, [screen, setMinimalShellMode]),
) )
const onPressAddAccount = React.useCallback(() => { const onPressAddAccount = React.useCallback(() => {

View File

@ -3,23 +3,23 @@ import {View} from 'react-native'
import {useFocusEffect} from '@react-navigation/native' import {useFocusEffect} from '@react-navigation/native'
import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types'
import {ViewHeader} from '../com/util/ViewHeader' import {ViewHeader} from '../com/util/ViewHeader'
import {useStores} from 'state/index'
import {Text} from 'view/com/util/text/Text' import {Text} from 'view/com/util/text/Text'
import {TextLink} from 'view/com/util/Link' import {TextLink} from 'view/com/util/Link'
import {CenteredView} from 'view/com/util/Views' import {CenteredView} from 'view/com/util/Views'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import {s} from 'lib/styles' import {s} from 'lib/styles'
import {HELP_DESK_URL} from 'lib/constants' import {HELP_DESK_URL} from 'lib/constants'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'Support'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'Support'>
export const SupportScreen = (_props: Props) => { export const SupportScreen = (_props: Props) => {
const store = useStores()
const pal = usePalette('default') const pal = usePalette('default')
const setMinimalShellMode = useSetMinimalShellMode()
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [store]), }, [setMinimalShellMode]),
) )
return ( return (

View File

@ -5,20 +5,20 @@ import {Text} from 'view/com/util/text/Text'
import {TextLink} from 'view/com/util/Link' import {TextLink} from 'view/com/util/Link'
import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types'
import {ViewHeader} from '../com/util/ViewHeader' import {ViewHeader} from '../com/util/ViewHeader'
import {useStores} from 'state/index'
import {ScrollView} from 'view/com/util/Views' import {ScrollView} from 'view/com/util/Views'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import {s} from 'lib/styles' import {s} from 'lib/styles'
import {useSetMinimalShellMode} from '#/state/shell'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'TermsOfService'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'TermsOfService'>
export const TermsOfServiceScreen = (_props: Props) => { export const TermsOfServiceScreen = (_props: Props) => {
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const setMinimalShellMode = useSetMinimalShellMode()
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
store.shell.setMinimalShellMode(false) setMinimalShellMode(false)
}, [store]), }, [setMinimalShellMode]),
) )
return ( return (

View File

@ -43,11 +43,13 @@ import {NavigationProp} from 'lib/routes/types'
import {useNavigationTabState} from 'lib/hooks/useNavigationTabState' import {useNavigationTabState} from 'lib/hooks/useNavigationTabState'
import {isWeb} from 'platform/detection' import {isWeb} from 'platform/detection'
import {formatCount, formatCountShortOnly} from 'view/com/util/numeric/format' import {formatCount, formatCountShortOnly} from 'view/com/util/numeric/format'
import {useSetDrawerOpen} from '#/state/shell'
export const DrawerContent = observer(function DrawerContentImpl() { export const DrawerContent = observer(function DrawerContentImpl() {
const theme = useTheme() const theme = useTheme()
const pal = usePalette('default') const pal = usePalette('default')
const store = useStores() const store = useStores()
const setDrawerOpen = useSetDrawerOpen()
const navigation = useNavigation<NavigationProp>() const navigation = useNavigation<NavigationProp>()
const {track} = useAnalytics() const {track} = useAnalytics()
const {isAtHome, isAtSearch, isAtFeeds, isAtNotifications, isAtMyProfile} = const {isAtHome, isAtSearch, isAtFeeds, isAtNotifications, isAtMyProfile} =
@ -62,7 +64,7 @@ export const DrawerContent = observer(function DrawerContentImpl() {
(tab: string) => { (tab: string) => {
track('Menu:ItemClicked', {url: tab}) track('Menu:ItemClicked', {url: tab})
const state = navigation.getState() const state = navigation.getState()
store.shell.closeDrawer() setDrawerOpen(false)
if (isWeb) { if (isWeb) {
// hack because we have flat navigator for web and MyProfile does not exist on the web navigator -ansh // hack because we have flat navigator for web and MyProfile does not exist on the web navigator -ansh
if (tab === 'MyProfile') { if (tab === 'MyProfile') {
@ -83,7 +85,7 @@ export const DrawerContent = observer(function DrawerContentImpl() {
} }
} }
}, },
[store, track, navigation], [store, track, navigation, setDrawerOpen],
) )
const onPressHome = React.useCallback(() => onPressTab('Home'), [onPressTab]) const onPressHome = React.useCallback(() => onPressTab('Home'), [onPressTab])
@ -110,20 +112,20 @@ export const DrawerContent = observer(function DrawerContentImpl() {
const onPressLists = React.useCallback(() => { const onPressLists = React.useCallback(() => {
track('Menu:ItemClicked', {url: 'Lists'}) track('Menu:ItemClicked', {url: 'Lists'})
navigation.navigate('Lists') navigation.navigate('Lists')
store.shell.closeDrawer() setDrawerOpen(false)
}, [navigation, track, store.shell]) }, [navigation, track, setDrawerOpen])
const onPressModeration = React.useCallback(() => { const onPressModeration = React.useCallback(() => {
track('Menu:ItemClicked', {url: 'Moderation'}) track('Menu:ItemClicked', {url: 'Moderation'})
navigation.navigate('Moderation') navigation.navigate('Moderation')
store.shell.closeDrawer() setDrawerOpen(false)
}, [navigation, track, store.shell]) }, [navigation, track, setDrawerOpen])
const onPressSettings = React.useCallback(() => { const onPressSettings = React.useCallback(() => {
track('Menu:ItemClicked', {url: 'Settings'}) track('Menu:ItemClicked', {url: 'Settings'})
navigation.navigate('Settings') navigation.navigate('Settings')
store.shell.closeDrawer() setDrawerOpen(false)
}, [navigation, track, store.shell]) }, [navigation, track, setDrawerOpen])
const onPressFeedback = React.useCallback(() => { const onPressFeedback = React.useCallback(() => {
track('Menu:FeedbackClicked') track('Menu:FeedbackClicked')
@ -437,13 +439,14 @@ const InviteCodes = observer(function InviteCodesImpl({
}) { }) {
const {track} = useAnalytics() const {track} = useAnalytics()
const store = useStores() const store = useStores()
const setDrawerOpen = useSetDrawerOpen()
const pal = usePalette('default') const pal = usePalette('default')
const {invitesAvailable} = store.me const {invitesAvailable} = store.me
const onPress = React.useCallback(() => { const onPress = React.useCallback(() => {
track('Menu:ItemClicked', {url: '#invite-codes'}) track('Menu:ItemClicked', {url: '#invite-codes'})
store.shell.closeDrawer() setDrawerOpen(false)
store.shell.openModal({name: 'invite-codes'}) store.shell.openModal({name: 'invite-codes'})
}, [store, track]) }, [store, track, setDrawerOpen])
return ( return (
<TouchableOpacity <TouchableOpacity
testID="menuItemInviteCodes" testID="menuItemInviteCodes"

View File

@ -37,7 +37,7 @@ export const BottomBar = observer(function BottomBarImpl({
const {isAtHome, isAtSearch, isAtFeeds, isAtNotifications, isAtMyProfile} = const {isAtHome, isAtSearch, isAtFeeds, isAtNotifications, isAtMyProfile} =
useNavigationTabState() useNavigationTabState()
const {footerMinimalShellTransform} = useMinimalShellMode() const {minimalShellMode, footerMinimalShellTransform} = useMinimalShellMode()
const {notifications} = store.me const {notifications} = store.me
const onPressTab = React.useCallback( const onPressTab = React.useCallback(
@ -83,7 +83,7 @@ export const BottomBar = observer(function BottomBarImpl({
pal.border, pal.border,
{paddingBottom: clamp(safeAreaInsets.bottom, 15, 30)}, {paddingBottom: clamp(safeAreaInsets.bottom, 15, 30)},
footerMinimalShellTransform, footerMinimalShellTransform,
store.shell.minimalShellMode && styles.disabled, minimalShellMode && styles.disabled,
]}> ]}>
<Btn <Btn
testID="bottomBarHomeBtn" testID="bottomBarHomeBtn"

View File

@ -6,6 +6,7 @@ import {
StyleSheet, StyleSheet,
useWindowDimensions, useWindowDimensions,
View, View,
BackHandler,
} from 'react-native' } from 'react-native'
import {useSafeAreaInsets} from 'react-native-safe-area-context' import {useSafeAreaInsets} from 'react-native-safe-area-context'
import {Drawer} from 'react-native-drawer-layout' import {Drawer} from 'react-native-drawer-layout'
@ -18,7 +19,6 @@ import {DrawerContent} from './Drawer'
import {Composer} from './Composer' import {Composer} from './Composer'
import {useTheme} from 'lib/ThemeContext' import {useTheme} from 'lib/ThemeContext'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import * as backHandler from 'lib/routes/back-handler'
import {RoutesContainer, TabsNavigator} from '../../Navigation' import {RoutesContainer, TabsNavigator} from '../../Navigation'
import {isStateAtTabRoot} from 'lib/routes/helpers' import {isStateAtTabRoot} from 'lib/routes/helpers'
import { import {
@ -26,9 +26,18 @@ import {
initialWindowMetrics, initialWindowMetrics,
} from 'react-native-safe-area-context' } from 'react-native-safe-area-context'
import {useOTAUpdate} from 'lib/hooks/useOTAUpdate' import {useOTAUpdate} from 'lib/hooks/useOTAUpdate'
import {
useIsDrawerOpen,
useSetDrawerOpen,
useIsDrawerSwipeDisabled,
} from '#/state/shell'
import {isAndroid} from 'platform/detection'
const ShellInner = observer(function ShellInnerImpl() { const ShellInner = observer(function ShellInnerImpl() {
const store = useStores() const store = useStores()
const isDrawerOpen = useIsDrawerOpen()
const isDrawerSwipeDisabled = useIsDrawerSwipeDisabled()
const setIsDrawerOpen = useSetDrawerOpen()
useOTAUpdate() // this hook polls for OTA updates every few seconds useOTAUpdate() // this hook polls for OTA updates every few seconds
const winDim = useWindowDimensions() const winDim = useWindowDimensions()
const safeAreaInsets = useSafeAreaInsets() const safeAreaInsets = useSafeAreaInsets()
@ -38,20 +47,26 @@ const ShellInner = observer(function ShellInnerImpl() {
) )
const renderDrawerContent = React.useCallback(() => <DrawerContent />, []) const renderDrawerContent = React.useCallback(() => <DrawerContent />, [])
const onOpenDrawer = React.useCallback( const onOpenDrawer = React.useCallback(
() => store.shell.openDrawer(), () => setIsDrawerOpen(true),
[store], [setIsDrawerOpen],
) )
const onCloseDrawer = React.useCallback( const onCloseDrawer = React.useCallback(
() => store.shell.closeDrawer(), () => setIsDrawerOpen(false),
[store], [setIsDrawerOpen],
) )
const canGoBack = useNavigationState(state => !isStateAtTabRoot(state)) const canGoBack = useNavigationState(state => !isStateAtTabRoot(state))
React.useEffect(() => { React.useEffect(() => {
const listener = backHandler.init(store) let listener = {remove() {}}
return () => { if (isAndroid) {
listener() listener = BackHandler.addEventListener('hardwareBackPress', () => {
setIsDrawerOpen(false)
return store.shell.closeAnyActiveElement()
})
} }
}, [store]) return () => {
listener.remove()
}
}, [store, setIsDrawerOpen])
return ( return (
<> <>
@ -59,14 +74,12 @@ const ShellInner = observer(function ShellInnerImpl() {
<ErrorBoundary> <ErrorBoundary>
<Drawer <Drawer
renderDrawerContent={renderDrawerContent} renderDrawerContent={renderDrawerContent}
open={store.shell.isDrawerOpen} open={isDrawerOpen}
onOpen={onOpenDrawer} onOpen={onOpenDrawer}
onClose={onCloseDrawer} onClose={onCloseDrawer}
swipeEdgeWidth={winDim.width / 2} swipeEdgeWidth={winDim.width / 2}
swipeEnabled={ swipeEnabled={
!canGoBack && !canGoBack && store.session.hasSession && !isDrawerSwipeDisabled
store.session.hasSession &&
!store.shell.isDrawerSwipeDisabled
}> }>
<TabsNavigator /> <TabsNavigator />
</Drawer> </Drawer>

View File

@ -17,18 +17,22 @@ import {BottomBarWeb} from './bottom-bar/BottomBarWeb'
import {useNavigation} from '@react-navigation/native' import {useNavigation} from '@react-navigation/native'
import {NavigationProp} from 'lib/routes/types' import {NavigationProp} from 'lib/routes/types'
import {useAuxClick} from 'lib/hooks/useAuxClick' import {useAuxClick} from 'lib/hooks/useAuxClick'
import {useIsDrawerOpen, useSetDrawerOpen} from '#/state/shell'
const ShellInner = observer(function ShellInnerImpl() { const ShellInner = observer(function ShellInnerImpl() {
const store = useStores() const store = useStores()
const isDrawerOpen = useIsDrawerOpen()
const setDrawerOpen = useSetDrawerOpen()
const {isDesktop, isMobile} = useWebMediaQueries() const {isDesktop, isMobile} = useWebMediaQueries()
const navigator = useNavigation<NavigationProp>() const navigator = useNavigation<NavigationProp>()
useAuxClick() useAuxClick()
useEffect(() => { useEffect(() => {
navigator.addListener('state', () => { navigator.addListener('state', () => {
setDrawerOpen(false)
store.shell.closeAnyActiveElement() store.shell.closeAnyActiveElement()
}) })
}, [navigator, store.shell]) }, [navigator, store.shell, setDrawerOpen])
const showBottomBar = isMobile && !store.onboarding.isActive const showBottomBar = isMobile && !store.onboarding.isActive
const showSideNavs = const showSideNavs =
@ -57,9 +61,9 @@ const ShellInner = observer(function ShellInnerImpl() {
{showBottomBar && <BottomBarWeb />} {showBottomBar && <BottomBarWeb />}
<ModalsContainer /> <ModalsContainer />
<Lightbox /> <Lightbox />
{!isDesktop && store.shell.isDrawerOpen && ( {!isDesktop && isDrawerOpen && (
<TouchableOpacity <TouchableOpacity
onPress={() => store.shell.closeDrawer()} onPress={() => setDrawerOpen(false)}
style={styles.drawerMask} style={styles.drawerMask}
accessibilityLabel="Close navigation footer" accessibilityLabel="Close navigation footer"
accessibilityHint="Closes bottom navigation bar"> accessibilityHint="Closes bottom navigation bar">