Shell behaviors update (react-query refactor) (#1915)

* Move tick-every-minute into a hook/context

* Move soft-reset event out of the shell model

* Update soft-reset listener on new search page

* Implement session-loaded and session-dropped events

* Update analytics and push-notifications to use new session system
zio/stable
Paul Frazee 2023-11-15 17:17:50 -08:00 committed by GitHub
parent f23e9978d8
commit 6616b2bff0
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
20 changed files with 186 additions and 136 deletions

View File

@ -11,6 +11,8 @@ import {QueryClientProvider} from '@tanstack/react-query'
import 'view/icons' import 'view/icons'
import {init as initPersistedState} from '#/state/persisted' import {init as initPersistedState} from '#/state/persisted'
import {init as initReminders} from '#/state/shell/reminders'
import {listenSessionDropped} from './state/events'
import {useColorMode} from 'state/shell' import {useColorMode} from 'state/shell'
import {ThemeProvider} from 'lib/ThemeContext' import {ThemeProvider} from 'lib/ThemeContext'
import {s} from 'lib/styles' import {s} from 'lib/styles'
@ -53,15 +55,17 @@ const InnerApp = observer(function AppImpl() {
useEffect(() => { useEffect(() => {
setupState().then(store => { setupState().then(store => {
setRootStore(store) setRootStore(store)
analytics.init(store)
notifications.init(store, queryClient)
store.onSessionDropped(() => {
Toast.show('Sorry! Your session expired. Please log in again.')
})
}) })
}, []) }, [])
useEffect(() => { useEffect(() => {
initReminders()
analytics.init()
notifications.init(queryClient)
listenSessionDropped(() => {
Toast.show('Sorry! Your session expired. Please log in again.')
})
const account = persisted.get('session').currentAccount const account = persisted.get('session').currentAccount
resumeSession(account) resumeSession(account)
}, [resumeSession]) }, [resumeSession])

View File

@ -9,6 +9,7 @@ import {RootSiblingParent} from 'react-native-root-siblings'
import 'view/icons' import 'view/icons'
import {init as initPersistedState} from '#/state/persisted' import {init as initPersistedState} from '#/state/persisted'
import {init as initReminders} from '#/state/shell/reminders'
import {useColorMode} from 'state/shell' import {useColorMode} from 'state/shell'
import * as analytics from 'lib/analytics/analytics' import * as analytics from 'lib/analytics/analytics'
import {RootStoreModel, setupState, RootStoreProvider} from './state' import {RootStoreModel, setupState, RootStoreProvider} from './state'
@ -44,12 +45,14 @@ const InnerApp = observer(function AppImpl() {
useEffect(() => { useEffect(() => {
setupState().then(store => { setupState().then(store => {
setRootStore(store) setRootStore(store)
analytics.init(store)
}) })
dynamicActivate(defaultLocale) // async import of locale data
}, []) }, [])
useEffect(() => { useEffect(() => {
initReminders()
analytics.init()
dynamicActivate(defaultLocale) // async import of locale data
const account = persisted.get('session').currentAccount const account = persisted.get('session').currentAccount
resumeSession(account) resumeSession(account)
}, [resumeSession]) }, [resumeSession])

View File

@ -1,16 +1,18 @@
import React from 'react' import React from 'react'
import {AppState, AppStateStatus} from 'react-native' import {AppState, AppStateStatus} from 'react-native'
import AsyncStorage from '@react-native-async-storage/async-storage'
import { import {
createClient, createClient,
AnalyticsProvider, AnalyticsProvider,
useAnalytics as useAnalyticsOrig, useAnalytics as useAnalyticsOrig,
ClientMethods, ClientMethods,
} from '@segment/analytics-react-native' } from '@segment/analytics-react-native'
import {RootStoreModel, AppInfo} from 'state/models/root-store' import {AppInfo} from 'state/models/root-store'
import {useStores} from 'state/models/root-store' import {useSession} from '#/state/session'
import {sha256} from 'js-sha256' import {sha256} from 'js-sha256'
import {ScreenEvent, TrackEvent} from './types' import {ScreenEvent, TrackEvent} from './types'
import {logger} from '#/logger' import {logger} from '#/logger'
import {listenSessionLoaded} from '#/state/events'
const segmentClient = createClient({ const segmentClient = createClient({
writeKey: '8I6DsgfiSLuoONyaunGoiQM7A6y2ybdI', writeKey: '8I6DsgfiSLuoONyaunGoiQM7A6y2ybdI',
@ -21,10 +23,10 @@ const segmentClient = createClient({
export const track = segmentClient?.track?.bind?.(segmentClient) as TrackEvent export const track = segmentClient?.track?.bind?.(segmentClient) as TrackEvent
export function useAnalytics() { export function useAnalytics() {
const store = useStores() const {hasSession} = useSession()
const methods: ClientMethods = useAnalyticsOrig() const methods: ClientMethods = useAnalyticsOrig()
return React.useMemo(() => { return React.useMemo(() => {
if (store.session.hasSession) { if (hasSession) {
return { return {
screen: methods.screen as ScreenEvent, // ScreenEvents defines all the possible screen names screen: methods.screen as ScreenEvent, // ScreenEvents defines all the possible screen names
track: methods.track as TrackEvent, // TrackEvents defines all the possible track events and their properties track: methods.track as TrackEvent, // TrackEvents defines all the possible track events and their properties
@ -45,29 +47,26 @@ export function useAnalytics() {
alias: () => Promise<void>, alias: () => Promise<void>,
reset: () => Promise<void>, reset: () => Promise<void>,
} }
}, [store, methods]) }, [hasSession, methods])
} }
export function init(store: RootStoreModel) { export function init() {
store.onSessionLoaded(() => { listenSessionLoaded(account => {
const sess = store.session.currentSession if (account.did) {
if (sess) { const did_hashed = sha256(account.did)
if (sess.did) {
const did_hashed = sha256(sess.did)
segmentClient.identify(did_hashed, {did_hashed}) segmentClient.identify(did_hashed, {did_hashed})
logger.debug('Ping w/hash') logger.debug('Ping w/hash')
} else { } else {
logger.debug('Ping w/o hash') logger.debug('Ping w/o hash')
segmentClient.identify() segmentClient.identify()
} }
}
}) })
// NOTE // NOTE
// this is a copy of segment's own lifecycle event tracking // this is a copy of segment's own lifecycle event tracking
// we handle it manually to ensure that it never fires while the app is backgrounded // we handle it manually to ensure that it never fires while the app is backgrounded
// -prf // -prf
segmentClient.isReady.onChange(() => { segmentClient.isReady.onChange(async () => {
if (AppState.currentState !== 'active') { if (AppState.currentState !== 'active') {
logger.debug('Prevented a metrics ping while the app was backgrounded') logger.debug('Prevented a metrics ping while the app was backgrounded')
return return
@ -78,20 +77,17 @@ export function init(store: RootStoreModel) {
return return
} }
const oldAppInfo = store.appInfo const oldAppInfo = await readAppInfo()
const newAppInfo = context.app as AppInfo const newAppInfo = context.app as AppInfo
store.setAppInfo(newAppInfo) writeAppInfo(newAppInfo)
logger.debug('Recording app info', {new: newAppInfo, old: oldAppInfo}) logger.debug('Recording app info', {new: newAppInfo, old: oldAppInfo})
if (typeof oldAppInfo === 'undefined') { if (typeof oldAppInfo === 'undefined') {
if (store.session.hasSession) {
segmentClient.track('Application Installed', { segmentClient.track('Application Installed', {
version: newAppInfo.version, version: newAppInfo.version,
build: newAppInfo.build, build: newAppInfo.build,
}) })
}
} else if (newAppInfo.version !== oldAppInfo.version) { } else if (newAppInfo.version !== oldAppInfo.version) {
if (store.session.hasSession) {
segmentClient.track('Application Updated', { segmentClient.track('Application Updated', {
version: newAppInfo.version, version: newAppInfo.version,
build: newAppInfo.build, build: newAppInfo.build,
@ -99,14 +95,11 @@ export function init(store: RootStoreModel) {
previous_build: oldAppInfo.build, previous_build: oldAppInfo.build,
}) })
} }
}
if (store.session.hasSession) {
segmentClient.track('Application Opened', { segmentClient.track('Application Opened', {
from_background: false, from_background: false,
version: newAppInfo.version, version: newAppInfo.version,
build: newAppInfo.build, build: newAppInfo.build,
}) })
}
}) })
let lastState: AppStateStatus = AppState.currentState let lastState: AppStateStatus = AppState.currentState
@ -130,3 +123,12 @@ export function Provider({children}: React.PropsWithChildren<{}>) {
<AnalyticsProvider client={segmentClient}>{children}</AnalyticsProvider> <AnalyticsProvider client={segmentClient}>{children}</AnalyticsProvider>
) )
} }
async function writeAppInfo(value: AppInfo) {
await AsyncStorage.setItem('BSKY_APP_INFO', JSON.stringify(value))
}
async function readAppInfo(): Promise<Partial<AppInfo> | undefined> {
const rawData = await AsyncStorage.getItem('BSKY_APP_INFO')
return rawData ? JSON.parse(rawData) : undefined
}

View File

@ -1,19 +1,19 @@
import * as Notifications from 'expo-notifications' import * as Notifications from 'expo-notifications'
import {QueryClient} from '@tanstack/react-query' import {QueryClient} from '@tanstack/react-query'
import {RootStoreModel} from '../../state'
import {resetToTab} from '../../Navigation' import {resetToTab} from '../../Navigation'
import {devicePlatform, isIOS} from 'platform/detection' import {devicePlatform, isIOS} from 'platform/detection'
import {track} from 'lib/analytics/analytics' import {track} from 'lib/analytics/analytics'
import {logger} from '#/logger' import {logger} from '#/logger'
import {RQKEY as RQKEY_NOTIFS} from '#/state/queries/notifications/feed' import {RQKEY as RQKEY_NOTIFS} from '#/state/queries/notifications/feed'
import {listenSessionLoaded} from '#/state/events'
const SERVICE_DID = (serviceUrl?: string) => const SERVICE_DID = (serviceUrl?: string) =>
serviceUrl?.includes('staging') serviceUrl?.includes('staging')
? 'did:web:api.staging.bsky.dev' ? 'did:web:api.staging.bsky.dev'
: 'did:web:api.bsky.app' : 'did:web:api.bsky.app'
export function init(store: RootStoreModel, queryClient: QueryClient) { export function init(queryClient: QueryClient) {
store.onSessionLoaded(async () => { listenSessionLoaded(async (account, agent) => {
// request notifications permission once the user has logged in // request notifications permission once the user has logged in
const perms = await Notifications.getPermissionsAsync() const perms = await Notifications.getPermissionsAsync()
if (!perms.granted) { if (!perms.granted) {
@ -24,8 +24,8 @@ export function init(store: RootStoreModel, queryClient: QueryClient) {
const token = await getPushToken() const token = await getPushToken()
if (token) { if (token) {
try { try {
await store.agent.api.app.bsky.notification.registerPush({ await agent.api.app.bsky.notification.registerPush({
serviceDid: SERVICE_DID(store.session.data?.service), serviceDid: SERVICE_DID(account.service),
platform: devicePlatform, platform: devicePlatform,
token: token.data, token: token.data,
appId: 'xyz.blueskyweb.app', appId: 'xyz.blueskyweb.app',
@ -53,8 +53,8 @@ export function init(store: RootStoreModel, queryClient: QueryClient) {
) )
if (t) { if (t) {
try { try {
await store.agent.api.app.bsky.notification.registerPush({ await agent.api.app.bsky.notification.registerPush({
serviceDid: SERVICE_DID(store.session.data?.service), serviceDid: SERVICE_DID(account.service),
platform: devicePlatform, platform: devicePlatform,
token: t, token: t,
appId: 'xyz.blueskyweb.app', appId: 'xyz.blueskyweb.app',

View File

@ -0,0 +1,38 @@
import EventEmitter from 'eventemitter3'
import {BskyAgent} from '@atproto/api'
import {SessionAccount} from './session'
type UnlistenFn = () => void
const emitter = new EventEmitter()
// a "soft reset" typically means scrolling to top and loading latest
// but it can depend on the screen
export function emitSoftReset() {
emitter.emit('soft-reset')
}
export function listenSoftReset(fn: () => void): UnlistenFn {
emitter.on('soft-reset', fn)
return () => emitter.off('soft-reset', fn)
}
export function emitSessionLoaded(
sessionAccount: SessionAccount,
agent: BskyAgent,
) {
emitter.emit('session-loaded', sessionAccount, agent)
}
export function listenSessionLoaded(
fn: (sessionAccount: SessionAccount, agent: BskyAgent) => void,
): UnlistenFn {
emitter.on('session-loaded', fn)
return () => emitter.off('session-loaded', fn)
}
export function emitSessionDropped() {
emitter.emit('session-dropped')
}
export function listenSessionDropped(fn: () => void): UnlistenFn {
emitter.on('session-dropped', fn)
return () => emitter.off('session-dropped', fn)
}

View File

@ -1,6 +1,6 @@
import {AppBskyActorDefs} from '@atproto/api' import {AppBskyActorDefs} from '@atproto/api'
import {RootStoreModel} from '../root-store' import {RootStoreModel} from '../root-store'
import {makeAutoObservable, runInAction} from 'mobx' import {makeAutoObservable} from 'mobx'
import { import {
shouldRequestEmailConfirmation, shouldRequestEmailConfirmation,
setEmailConfirmationRequested, setEmailConfirmationRequested,
@ -40,14 +40,12 @@ export class ImagesLightbox implements LightboxModel {
export class ShellUiModel { export class ShellUiModel {
isLightboxActive = false isLightboxActive = false
activeLightbox: ProfileImageLightbox | ImagesLightbox | null = null activeLightbox: ProfileImageLightbox | ImagesLightbox | null = null
tickEveryMinute = Date.now()
constructor(public rootStore: RootStoreModel) { constructor(public rootStore: RootStoreModel) {
makeAutoObservable(this, { makeAutoObservable(this, {
rootStore: false, rootStore: false,
}) })
this.setupClock()
this.setupLoginModals() this.setupLoginModals()
} }
@ -83,14 +81,6 @@ export class ShellUiModel {
this.activeLightbox = null this.activeLightbox = null
} }
setupClock() {
setInterval(() => {
runInAction(() => {
this.tickEveryMinute = Date.now()
})
}, 60_000)
}
setupLoginModals() { setupLoginModals() {
this.rootStore.onSessionReady(() => { this.rootStore.onSessionReady(() => {
if (shouldRequestEmailConfirmation(this.rootStore.session)) { if (shouldRequestEmailConfirmation(this.rootStore.session)) {

View File

@ -1,5 +1,4 @@
import React from 'react' import React from 'react'
import {DeviceEventEmitter} from 'react-native'
import {BskyAgent, AtpPersistSessionHandler} from '@atproto/api' import {BskyAgent, AtpPersistSessionHandler} from '@atproto/api'
import {networkRetry} from '#/lib/async/retry' import {networkRetry} from '#/lib/async/retry'
@ -7,6 +6,7 @@ import {logger} from '#/logger'
import * as persisted from '#/state/persisted' import * as persisted from '#/state/persisted'
import {PUBLIC_BSKY_AGENT} from '#/state/queries' import {PUBLIC_BSKY_AGENT} from '#/state/queries'
import {IS_PROD} from '#/lib/constants' import {IS_PROD} from '#/lib/constants'
import {emitSessionLoaded, emitSessionDropped} from '../events'
export type SessionAccount = persisted.PersistedAccount export type SessionAccount = persisted.PersistedAccount
@ -98,7 +98,9 @@ function createPersistSessionHandler(
logger.DebugContext.session, logger.DebugContext.session,
) )
if (expired) DeviceEventEmitter.emit('session-dropped') if (expired) {
emitSessionDropped()
}
persistSessionCallback({ persistSessionCallback({
expired, expired,
@ -180,6 +182,7 @@ export function Provider({children}: React.PropsWithChildren<{}>) {
setState(s => ({...s, agent})) setState(s => ({...s, agent}))
upsertAccount(account) upsertAccount(account)
emitSessionLoaded(account, agent)
logger.debug( logger.debug(
`session: created account`, `session: created account`,
@ -230,6 +233,7 @@ export function Provider({children}: React.PropsWithChildren<{}>) {
setState(s => ({...s, agent})) setState(s => ({...s, agent}))
upsertAccount(account) upsertAccount(account)
emitSessionLoaded(account, agent)
logger.debug( logger.debug(
`session: logged in`, `session: logged in`,
@ -291,6 +295,7 @@ export function Provider({children}: React.PropsWithChildren<{}>) {
setState(s => ({...s, agent})) setState(s => ({...s, agent}))
upsertAccount(account) upsertAccount(account)
emitSessionLoaded(account, agent)
}, },
[upsertAccount], [upsertAccount],
) )

View File

@ -6,6 +6,7 @@ import {Provider as MinimalModeProvider} from './minimal-mode'
import {Provider as ColorModeProvider} from './color-mode' import {Provider as ColorModeProvider} from './color-mode'
import {Provider as OnboardingProvider} from './onboarding' import {Provider as OnboardingProvider} from './onboarding'
import {Provider as ComposerProvider} from './composer' import {Provider as ComposerProvider} from './composer'
import {Provider as TickEveryMinuteProvider} from './tick-every-minute'
export {useIsDrawerOpen, useSetDrawerOpen} from './drawer-open' export {useIsDrawerOpen, useSetDrawerOpen} from './drawer-open'
export { export {
@ -15,6 +16,8 @@ export {
export {useMinimalShellMode, useSetMinimalShellMode} from './minimal-mode' export {useMinimalShellMode, useSetMinimalShellMode} from './minimal-mode'
export {useColorMode, useSetColorMode} from './color-mode' export {useColorMode, useSetColorMode} from './color-mode'
export {useOnboardingState, useOnboardingDispatch} from './onboarding' export {useOnboardingState, useOnboardingDispatch} from './onboarding'
export {useComposerState, useComposerControls} from './composer'
export {useTickEveryMinute} from './tick-every-minute'
export function Provider({children}: React.PropsWithChildren<{}>) { export function Provider({children}: React.PropsWithChildren<{}>) {
return ( return (
@ -24,7 +27,9 @@ export function Provider({children}: React.PropsWithChildren<{}>) {
<MinimalModeProvider> <MinimalModeProvider>
<ColorModeProvider> <ColorModeProvider>
<OnboardingProvider> <OnboardingProvider>
<ComposerProvider>{children}</ComposerProvider> <ComposerProvider>
<TickEveryMinuteProvider>{children}</TickEveryMinuteProvider>
</ComposerProvider>
</OnboardingProvider> </OnboardingProvider>
</ColorModeProvider> </ColorModeProvider>
</MinimalModeProvider> </MinimalModeProvider>

View File

@ -1,14 +1,24 @@
import * as persisted from '#/state/persisted' import * as persisted from '#/state/persisted'
import {SessionModel} from '../models/session'
import {toHashCode} from 'lib/strings/helpers' import {toHashCode} from 'lib/strings/helpers'
import {isOnboardingActive} from './onboarding' import {isOnboardingActive} from './onboarding'
import {SessionAccount} from '../session'
import {listenSessionLoaded} from '../events'
import {unstable__openModal} from '../modals'
export function shouldRequestEmailConfirmation(session: SessionModel) { export function init() {
const sess = session.currentSession listenSessionLoaded(account => {
if (!sess) { if (shouldRequestEmailConfirmation(account)) {
unstable__openModal({name: 'verify-email', showReminder: true})
setEmailConfirmationRequested()
}
})
}
export function shouldRequestEmailConfirmation(account: SessionAccount) {
if (!account) {
return false return false
} }
if (sess.emailConfirmed) { if (account.emailConfirmed) {
return false return false
} }
if (isOnboardingActive()) { if (isOnboardingActive()) {
@ -22,7 +32,7 @@ export function shouldRequestEmailConfirmation(session: SessionModel) {
// shard the users into 2 day of the week buckets // shard the users into 2 day of the week buckets
// (this is to avoid a sudden influx of email updates when // (this is to avoid a sudden influx of email updates when
// this feature rolls out) // this feature rolls out)
const code = toHashCode(sess.did) % 7 const code = toHashCode(account.did) % 7
if (code !== today.getDay() && code !== (today.getDay() + 1) % 7) { if (code !== today.getDay() && code !== (today.getDay() + 1) % 7) {
return false return false
} }

View File

@ -0,0 +1,20 @@
import React from 'react'
type StateContext = number
const stateContext = React.createContext<StateContext>(0)
export function Provider({children}: React.PropsWithChildren<{}>) {
const [tick, setTick] = React.useState(Date.now())
React.useEffect(() => {
const i = setInterval(() => {
setTick(Date.now())
}, 60_000)
return () => clearInterval(i)
}, [])
return <stateContext.Provider value={tick}>{children}</stateContext.Provider>
}
export function useTickEveryMinute() {
return React.useContext(stateContext)
}

View File

@ -14,7 +14,6 @@ import {ComposeIcon2} from 'lib/icons'
import {colors, s} from 'lib/styles' import {colors, s} from 'lib/styles'
import React from 'react' import React from 'react'
import {FlatList, View, useWindowDimensions} from 'react-native' import {FlatList, View, useWindowDimensions} from 'react-native'
import {useStores} from 'state/index'
import {Feed} from '../posts/Feed' import {Feed} from '../posts/Feed'
import {TextLink} from '../util/Link' import {TextLink} from '../util/Link'
import {FAB} from '../util/fab/FAB' import {FAB} from '../util/fab/FAB'
@ -23,6 +22,7 @@ import {msg} from '@lingui/macro'
import {useLingui} from '@lingui/react' import {useLingui} from '@lingui/react'
import {useSession} from '#/state/session' import {useSession} from '#/state/session'
import {useComposerControls} from '#/state/shell/composer' import {useComposerControls} from '#/state/shell/composer'
import {listenSoftReset, emitSoftReset} from '#/state/events'
const POLL_FREQ = 30e3 // 30sec const POLL_FREQ = 30e3 // 30sec
@ -41,7 +41,6 @@ export function FeedPage({
renderEmptyState: () => JSX.Element renderEmptyState: () => JSX.Element
renderEndOfFeed?: () => JSX.Element renderEndOfFeed?: () => JSX.Element
}) { }) {
const store = useStores()
const {isSandbox} = useSession() const {isSandbox} = useSession()
const pal = usePalette('default') const pal = usePalette('default')
const {_} = useLingui() const {_} = useLingui()
@ -73,12 +72,9 @@ export function FeedPage({
if (!isPageFocused || !isScreenFocused) { if (!isPageFocused || !isScreenFocused) {
return return
} }
const softResetSub = store.onScreenSoftReset(onSoftReset)
screen('Feed') screen('Feed')
return () => { return listenSoftReset(onSoftReset)
softResetSub.remove() }, [onSoftReset, screen, isPageFocused, isScreenFocused])
}
}, [store, onSoftReset, screen, feed, isPageFocused, isScreenFocused])
const onPressCompose = React.useCallback(() => { const onPressCompose = React.useCallback(() => {
track('HomeScreen:PressCompose') track('HomeScreen:PressCompose')
@ -125,7 +121,7 @@ export function FeedPage({
)} )}
</> </>
} }
onPress={() => store.emitScreenSoftReset()} onPress={emitSoftReset}
/> />
<TextLink <TextLink
type="title-lg" type="title-lg"
@ -144,16 +140,7 @@ export function FeedPage({
) )
} }
return <></> return <></>
}, [ }, [isDesktop, pal.view, pal.text, pal.textLight, hasNew, _, isSandbox])
isDesktop,
pal.view,
pal.text,
pal.textLight,
store,
hasNew,
_,
isSandbox,
])
return ( return (
<View testID={testID} style={s.h100pct}> <View testID={testID} style={s.h100pct}>

View File

@ -20,6 +20,7 @@ import {ImagesLightbox} from 'state/models/ui/shell'
import {useLingui} from '@lingui/react' import {useLingui} from '@lingui/react'
import {msg} from '@lingui/macro' import {msg} from '@lingui/macro'
import {useSetDrawerOpen} from '#/state/shell' import {useSetDrawerOpen} from '#/state/shell'
import {emitSoftReset} from '#/state/events'
export const ProfileSubpageHeader = observer(function HeaderImpl({ export const ProfileSubpageHeader = observer(function HeaderImpl({
isLoading, isLoading,
@ -145,7 +146,7 @@ export const ProfileSubpageHeader = observer(function HeaderImpl({
href={href} href={href}
style={[pal.text, {fontWeight: 'bold'}]} style={[pal.text, {fontWeight: 'bold'}]}
text={title || ''} text={title || ''}
onPress={() => store.emitScreenSoftReset()} onPress={emitSoftReset}
numberOfLines={4} numberOfLines={4}
/> />
)} )}

View File

@ -1,24 +1,23 @@
import React from 'react' import React from 'react'
import {observer} from 'mobx-react-lite'
import {ago} from 'lib/strings/time' import {ago} from 'lib/strings/time'
import {useStores} from 'state/index' import {useTickEveryMinute} from '#/state/shell'
// FIXME(dan): Figure out why the false positives // FIXME(dan): Figure out why the false positives
/* eslint-disable react/prop-types */ /* eslint-disable react/prop-types */
export const TimeElapsed = observer(function TimeElapsed({ export function TimeElapsed({
timestamp, timestamp,
children, children,
}: { }: {
timestamp: string timestamp: string
children: ({timeElapsed}: {timeElapsed: string}) => JSX.Element children: ({timeElapsed}: {timeElapsed: string}) => JSX.Element
}) { }) {
const stores = useStores() const tick = useTickEveryMinute()
const [timeElapsed, setTimeAgo] = React.useState(ago(timestamp)) const [timeElapsed, setTimeAgo] = React.useState(ago(timestamp))
React.useEffect(() => { React.useEffect(() => {
setTimeAgo(ago(timestamp)) setTimeAgo(ago(timestamp))
}, [timestamp, setTimeAgo, stores.shell.tickEveryMinute]) }, [timestamp, setTimeAgo, tick])
return children({timeElapsed}) return children({timeElapsed})
}) }

View File

@ -9,15 +9,14 @@ import {FollowingEndOfFeed} from 'view/com/posts/FollowingEndOfFeed'
import {CustomFeedEmptyState} from 'view/com/posts/CustomFeedEmptyState' import {CustomFeedEmptyState} from 'view/com/posts/CustomFeedEmptyState'
import {FeedsTabBar} from '../com/pager/FeedsTabBar' import {FeedsTabBar} from '../com/pager/FeedsTabBar'
import {Pager, PagerRef, RenderTabBarFnProps} from 'view/com/pager/Pager' import {Pager, PagerRef, RenderTabBarFnProps} from 'view/com/pager/Pager'
import {useStores} from 'state/index'
import {FeedPage} from 'view/com/feeds/FeedPage' import {FeedPage} from 'view/com/feeds/FeedPage'
import {useSetMinimalShellMode, useSetDrawerSwipeDisabled} from '#/state/shell' import {useSetMinimalShellMode, useSetDrawerSwipeDisabled} from '#/state/shell'
import {usePreferencesQuery} from '#/state/queries/preferences' import {usePreferencesQuery} from '#/state/queries/preferences'
import {emitSoftReset} from '#/state/events'
type Props = NativeStackScreenProps<HomeTabNavigatorParams, 'Home'> 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 setMinimalShellMode = useSetMinimalShellMode() const setMinimalShellMode = useSetMinimalShellMode()
const setDrawerSwipeDisabled = useSetDrawerSwipeDisabled() const setDrawerSwipeDisabled = useSetDrawerSwipeDisabled()
const pagerRef = React.useRef<PagerRef>(null) const pagerRef = React.useRef<PagerRef>(null)
@ -74,8 +73,8 @@ export const HomeScreen = withAuthRequired(
) )
const onPressSelected = React.useCallback(() => { const onPressSelected = React.useCallback(() => {
store.emitScreenSoftReset() emitSoftReset()
}, [store]) }, [])
const onPageScrollStateChanged = React.useCallback( const onPageScrollStateChanged = React.useCallback(
(state: 'idle' | 'dragging' | 'settling') => { (state: 'idle' | 'dragging' | 'settling') => {

View File

@ -11,7 +11,6 @@ import {ViewHeader} from '../com/util/ViewHeader'
import {Feed} from '../com/notifications/Feed' import {Feed} from '../com/notifications/Feed'
import {TextLink} from 'view/com/util/Link' import {TextLink} from 'view/com/util/Link'
import {LoadLatestBtn} from 'view/com/util/load-latest/LoadLatestBtn' import {LoadLatestBtn} from 'view/com/util/load-latest/LoadLatestBtn'
import {useStores} from 'state/index'
import {useOnMainScroll} from 'lib/hooks/useOnMainScroll' import {useOnMainScroll} from 'lib/hooks/useOnMainScroll'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
@ -21,6 +20,7 @@ import {logger} from '#/logger'
import {useSetMinimalShellMode} from '#/state/shell' import {useSetMinimalShellMode} from '#/state/shell'
import {useUnreadNotifications} from '#/state/queries/notifications/unread' import {useUnreadNotifications} from '#/state/queries/notifications/unread'
import {RQKEY as NOTIFS_RQKEY} from '#/state/queries/notifications/feed' import {RQKEY as NOTIFS_RQKEY} from '#/state/queries/notifications/feed'
import {listenSoftReset, emitSoftReset} from '#/state/events'
type Props = NativeStackScreenProps< type Props = NativeStackScreenProps<
NotificationsTabNavigatorParams, NotificationsTabNavigatorParams,
@ -28,7 +28,6 @@ type Props = NativeStackScreenProps<
> >
export const NotificationsScreen = withAuthRequired( export const NotificationsScreen = withAuthRequired(
function NotificationsScreenImpl({}: Props) { function NotificationsScreenImpl({}: Props) {
const store = useStores()
const setMinimalShellMode = useSetMinimalShellMode() const setMinimalShellMode = useSetMinimalShellMode()
const [onMainScroll, isScrolledDown, resetMainScroll] = useOnMainScroll() const [onMainScroll, isScrolledDown, resetMainScroll] = useOnMainScroll()
const scrollElRef = React.useRef<FlatList>(null) const scrollElRef = React.useRef<FlatList>(null)
@ -57,13 +56,9 @@ export const NotificationsScreen = withAuthRequired(
React.useCallback(() => { React.useCallback(() => {
setMinimalShellMode(false) setMinimalShellMode(false)
logger.debug('NotificationsScreen: Updating feed') logger.debug('NotificationsScreen: Updating feed')
const softResetSub = store.onScreenSoftReset(onPressLoadLatest)
screen('Notifications') screen('Notifications')
return listenSoftReset(onPressLoadLatest)
return () => { }, [screen, onPressLoadLatest, setMinimalShellMode]),
softResetSub.remove()
}
}, [store, screen, onPressLoadLatest, setMinimalShellMode]),
) )
const ListHeaderComponent = React.useCallback(() => { const ListHeaderComponent = React.useCallback(() => {
@ -100,13 +95,13 @@ export const NotificationsScreen = withAuthRequired(
)} )}
</> </>
} }
onPress={() => store.emitScreenSoftReset()} onPress={emitSoftReset}
/> />
</View> </View>
) )
} }
return <></> return <></>
}, [isDesktop, pal, store, hasNew]) }, [isDesktop, pal, hasNew])
return ( return (
<View testID="notificationsScreen" style={s.hContentRegion}> <View testID="notificationsScreen" style={s.hContentRegion}>

View File

@ -12,7 +12,6 @@ import {ScreenHider} from 'view/com/util/moderation/ScreenHider'
import {Feed} from 'view/com/posts/Feed' import {Feed} from 'view/com/posts/Feed'
import {ProfileLists} from '../com/lists/ProfileLists' import {ProfileLists} from '../com/lists/ProfileLists'
import {ProfileFeedgens} from '../com/feeds/ProfileFeedgens' import {ProfileFeedgens} from '../com/feeds/ProfileFeedgens'
import {useStores} from 'state/index'
import {ProfileHeader} from '../com/profile/ProfileHeader' import {ProfileHeader} from '../com/profile/ProfileHeader'
import {PagerWithHeader} from 'view/com/pager/PagerWithHeader' import {PagerWithHeader} from 'view/com/pager/PagerWithHeader'
import {ErrorScreen} from '../com/util/error/ErrorScreen' import {ErrorScreen} from '../com/util/error/ErrorScreen'
@ -37,6 +36,7 @@ import {cleanError} from '#/lib/strings/errors'
import {LoadLatestBtn} from '../com/util/load-latest/LoadLatestBtn' import {LoadLatestBtn} from '../com/util/load-latest/LoadLatestBtn'
import {useQueryClient} from '@tanstack/react-query' import {useQueryClient} from '@tanstack/react-query'
import {useComposerControls} from '#/state/shell/composer' import {useComposerControls} from '#/state/shell/composer'
import {listenSoftReset} from '#/state/events'
type Props = NativeStackScreenProps<CommonNavigatorParams, 'Profile'> type Props = NativeStackScreenProps<CommonNavigatorParams, 'Profile'>
export const ProfileScreen = withAuthRequired(function ProfileScreenImpl({ export const ProfileScreen = withAuthRequired(function ProfileScreenImpl({
@ -126,7 +126,6 @@ function ProfileScreenLoaded({
hideBackButton: boolean hideBackButton: boolean
}) { }) {
const profile = useProfileShadow(profileUnshadowed, dataUpdatedAt) const profile = useProfileShadow(profileUnshadowed, dataUpdatedAt)
const store = useStores()
const {currentAccount} = useSession() const {currentAccount} = useSession()
const setMinimalShellMode = useSetMinimalShellMode() const setMinimalShellMode = useSetMinimalShellMode()
const {openComposer} = useComposerControls() const {openComposer} = useComposerControls()
@ -169,11 +168,10 @@ function ProfileScreenLoaded({
React.useCallback(() => { React.useCallback(() => {
setMinimalShellMode(false) setMinimalShellMode(false)
screen('Profile') screen('Profile')
const softResetSub = store.onScreenSoftReset(() => { return listenSoftReset(() => {
viewSelectorRef.current?.scrollToTop() viewSelectorRef.current?.scrollToTop()
}) })
return () => softResetSub.remove() }, [viewSelectorRef, setMinimalShellMode, screen]),
}, [store, viewSelectorRef, setMinimalShellMode, screen]),
) )
useFocusEffect( useFocusEffect(

View File

@ -42,8 +42,8 @@ import {MagnifyingGlassIcon} from '#/lib/icons'
import {useModerationOpts} from '#/state/queries/preferences' import {useModerationOpts} from '#/state/queries/preferences'
import {SearchResultCard} from '#/view/shell/desktop/Search' import {SearchResultCard} from '#/view/shell/desktop/Search'
import {useSetMinimalShellMode, useSetDrawerSwipeDisabled} from '#/state/shell' import {useSetMinimalShellMode, useSetDrawerSwipeDisabled} from '#/state/shell'
import {useStores} from '#/state'
import {isWeb} from '#/platform/detection' import {isWeb} from '#/platform/detection'
import {listenSoftReset} from '#/state/events'
function Loader() { function Loader() {
const pal = usePalette('default') const pal = usePalette('default')
@ -421,7 +421,6 @@ export function SearchScreenMobile(
const moderationOpts = useModerationOpts() const moderationOpts = useModerationOpts()
const search = useActorAutocompleteFn() const search = useActorAutocompleteFn()
const setMinimalShellMode = useSetMinimalShellMode() const setMinimalShellMode = useSetMinimalShellMode()
const store = useStores()
const {isTablet} = useWebMediaQueries() const {isTablet} = useWebMediaQueries()
const searchDebounceTimeout = React.useRef<NodeJS.Timeout | undefined>( const searchDebounceTimeout = React.useRef<NodeJS.Timeout | undefined>(
@ -490,14 +489,9 @@ export function SearchScreenMobile(
useFocusEffect( useFocusEffect(
React.useCallback(() => { React.useCallback(() => {
const softResetSub = store.onScreenSoftReset(onSoftReset)
setMinimalShellMode(false) setMinimalShellMode(false)
return listenSoftReset(onSoftReset)
return () => { }, [onSoftReset, setMinimalShellMode]),
softResetSub.remove()
}
}, [store, onSoftReset, setMinimalShellMode]),
) )
return ( return (

View File

@ -50,6 +50,7 @@ import {useModalControls} from '#/state/modals'
import {useSession, SessionAccount} from '#/state/session' import {useSession, SessionAccount} from '#/state/session'
import {useProfileQuery} from '#/state/queries/profile' import {useProfileQuery} from '#/state/queries/profile'
import {useUnreadNotifications} from '#/state/queries/notifications/unread' import {useUnreadNotifications} from '#/state/queries/notifications/unread'
import {emitSoftReset} from '#/state/events'
export function DrawerProfileCard({ export function DrawerProfileCard({
account, account,
@ -103,7 +104,6 @@ export function DrawerProfileCard({
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 {_} = useLingui() const {_} = useLingui()
const setDrawerOpen = useSetDrawerOpen() const setDrawerOpen = useSetDrawerOpen()
const navigation = useNavigation<NavigationProp>() const navigation = useNavigation<NavigationProp>()
@ -124,7 +124,7 @@ export const DrawerContent = observer(function DrawerContentImpl() {
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') {
navigation.navigate('Profile', {name: store.me.handle}) navigation.navigate('Profile', {name: currentAccount!.handle})
} else { } else {
// @ts-ignore must be Home, Search, Notifications, or MyProfile // @ts-ignore must be Home, Search, Notifications, or MyProfile
navigation.navigate(tab) navigation.navigate(tab)
@ -132,7 +132,7 @@ export const DrawerContent = observer(function DrawerContentImpl() {
} else { } else {
const tabState = getTabState(state, tab) const tabState = getTabState(state, tab)
if (tabState === TabState.InsideAtRoot) { if (tabState === TabState.InsideAtRoot) {
store.emitScreenSoftReset() emitSoftReset()
} else if (tabState === TabState.Inside) { } else if (tabState === TabState.Inside) {
navigation.dispatch(StackActions.popToTop()) navigation.dispatch(StackActions.popToTop())
} else { } else {
@ -141,7 +141,7 @@ export const DrawerContent = observer(function DrawerContentImpl() {
} }
} }
}, },
[store, track, navigation, setDrawerOpen], [track, navigation, setDrawerOpen, currentAccount],
) )
const onPressHome = React.useCallback(() => onPressTab('Home'), [onPressTab]) const onPressHome = React.useCallback(() => onPressTab('Home'), [onPressTab])

View File

@ -29,6 +29,7 @@ import {msg} from '@lingui/macro'
import {useModalControls} from '#/state/modals' import {useModalControls} from '#/state/modals'
import {useShellLayout} from '#/state/shell/shell-layout' import {useShellLayout} from '#/state/shell/shell-layout'
import {useUnreadNotifications} from '#/state/queries/notifications/unread' import {useUnreadNotifications} from '#/state/queries/notifications/unread'
import {emitSoftReset} from '#/state/events'
type TabOptions = 'Home' | 'Search' | 'Notifications' | 'MyProfile' | 'Feeds' type TabOptions = 'Home' | 'Search' | 'Notifications' | 'MyProfile' | 'Feeds'
@ -53,14 +54,14 @@ export const BottomBar = observer(function BottomBarImpl({
const state = navigation.getState() const state = navigation.getState()
const tabState = getTabState(state, tab) const tabState = getTabState(state, tab)
if (tabState === TabState.InsideAtRoot) { if (tabState === TabState.InsideAtRoot) {
store.emitScreenSoftReset() emitSoftReset()
} else if (tabState === TabState.Inside) { } else if (tabState === TabState.Inside) {
navigation.dispatch(StackActions.popToTop()) navigation.dispatch(StackActions.popToTop())
} else { } else {
navigation.navigate(`${tab}Tab`) navigation.navigate(`${tab}Tab`)
} }
}, },
[store, track, navigation], [track, navigation],
) )
const onPressHome = React.useCallback(() => onPressTab('Home'), [onPressTab]) const onPressHome = React.useCallback(() => onPressTab('Home'), [onPressTab])
const onPressSearch = React.useCallback( const onPressSearch = React.useCallback(

View File

@ -16,7 +16,6 @@ import {UserAvatar} from 'view/com/util/UserAvatar'
import {Link} from 'view/com/util/Link' import {Link} from 'view/com/util/Link'
import {LoadingPlaceholder} from 'view/com/util/LoadingPlaceholder' import {LoadingPlaceholder} from 'view/com/util/LoadingPlaceholder'
import {usePalette} from 'lib/hooks/usePalette' import {usePalette} from 'lib/hooks/usePalette'
import {useStores} from 'state/index'
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
import {s, colors} from 'lib/styles' import {s, colors} from 'lib/styles'
import { import {
@ -46,6 +45,7 @@ import {useSession} from '#/state/session'
import {useUnreadNotifications} from '#/state/queries/notifications/unread' import {useUnreadNotifications} from '#/state/queries/notifications/unread'
import {useComposerControls} from '#/state/shell/composer' import {useComposerControls} from '#/state/shell/composer'
import {useFetchHandle} from '#/state/queries/handle' import {useFetchHandle} from '#/state/queries/handle'
import {emitSoftReset} from '#/state/events'
const ProfileCard = observer(function ProfileCardImpl() { const ProfileCard = observer(function ProfileCardImpl() {
const {currentAccount} = useSession() const {currentAccount} = useSession()
@ -126,7 +126,6 @@ const NavItem = observer(function NavItemImpl({
}: NavItemProps) { }: NavItemProps) {
const pal = usePalette('default') const pal = usePalette('default')
const {currentAccount} = useSession() const {currentAccount} = useSession()
const store = useStores()
const {isDesktop, isTablet} = useWebMediaQueries() const {isDesktop, isTablet} = useWebMediaQueries()
const [pathName] = React.useMemo(() => router.matchPath(href), [href]) const [pathName] = React.useMemo(() => router.matchPath(href), [href])
const currentRouteInfo = useNavigationState(state => { const currentRouteInfo = useNavigationState(state => {
@ -149,12 +148,12 @@ const NavItem = observer(function NavItemImpl({
} }
e.preventDefault() e.preventDefault()
if (isCurrent) { if (isCurrent) {
store.emitScreenSoftReset() emitSoftReset()
} else { } else {
onPress() onPress()
} }
}, },
[onPress, isCurrent, store], [onPress, isCurrent],
) )
return ( return (