Performance optimization (#1676)
* upgrade sentry to support profiling monitoring * remove console logs in production builds * feeds tab bar and bottom bar animation centralized * refactor FeedPage out of Home * add script to start in production mode * move FAB inner to reanimated * move FABInner back to `Animated` RN animation * add perf commands * add testing with Maestro and perf with Flashlight * fix merge conflicts * fix resourceClass name in eas.json * fix onEndReachedThreshold in Feed * memoize styles * go back to old styling for LoadLatestBtn * remove reanimated code from useMinimalShellMode * move shell animations to hook/reanimated for perf * fix empty state issue * make shell animation feel smoother * make shell animation more smooth * run animation with autorun * specify keys for tab bar properly * remove comments * remove already imported dep * fix lint * add testing instructions * mock sentry-expo for jest * fix jest mocks * Fix the load-latest button on desktop and tablet * Fix: don't move the FAB in tablet mode * Fix type error * Fix tabs bar positioning on tablet * Fix types --------- Co-authored-by: Paul Frazee <pfrazee@gmail.com>zio/stable
parent
9042f503c2
commit
8e9cf182c2
|
@ -100,3 +100,6 @@ ios/
|
||||||
|
|
||||||
# Firebase (Android) Google services
|
# Firebase (Android) Google services
|
||||||
google-services.json
|
google-services.json
|
||||||
|
|
||||||
|
# Performance results (Flashlight)
|
||||||
|
.perf/
|
|
@ -0,0 +1,77 @@
|
||||||
|
# flow.yaml
|
||||||
|
|
||||||
|
appId: xyz.blueskyweb.app
|
||||||
|
---
|
||||||
|
- launchApp
|
||||||
|
# Login
|
||||||
|
# - runFlow:
|
||||||
|
# when:
|
||||||
|
# - tapOn: "Sign In"
|
||||||
|
# - tapOn: "Username or email address"
|
||||||
|
# - inputText: "ansh.bsky.team"
|
||||||
|
# - tapOn: "Password"
|
||||||
|
# - inputText: "PASSWORd"
|
||||||
|
# - tapOn: "Next"
|
||||||
|
# Allow notifications if popup is visible
|
||||||
|
# - runFlow:
|
||||||
|
# when:
|
||||||
|
# visible: "Notifications"
|
||||||
|
# commands:
|
||||||
|
# - tapOn: "Allow"
|
||||||
|
# Scroll in main feed
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
# Swipe between feeds
|
||||||
|
- swipe:
|
||||||
|
direction: "LEFT"
|
||||||
|
- swipe:
|
||||||
|
direction: "LEFT"
|
||||||
|
- swipe:
|
||||||
|
direction: "LEFT"
|
||||||
|
- swipe:
|
||||||
|
direction: "RIGHT"
|
||||||
|
- swipe:
|
||||||
|
direction: "RIGHT"
|
||||||
|
- swipe:
|
||||||
|
direction: "RIGHT"
|
||||||
|
# Go to Notifications
|
||||||
|
- tapOn:
|
||||||
|
id: "viewHeaderDrawerBtn"
|
||||||
|
- tapOn: "Notifications"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- swipe:
|
||||||
|
direction: "DOWN" # Make header visible
|
||||||
|
# Go to Feeds tab
|
||||||
|
- tapOn:
|
||||||
|
id: "viewHeaderDrawerBtn"
|
||||||
|
- tapOn: "Feeds"
|
||||||
|
- scrollUntilVisible:
|
||||||
|
element: "Discover"
|
||||||
|
direction: UP
|
||||||
|
- tapOn: "Discover"
|
||||||
|
- waitForAnimationToEnd
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
# Click on post
|
||||||
|
- tapOn:
|
||||||
|
id: "postText"
|
||||||
|
index: 0
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
- "scroll"
|
||||||
|
|
|
@ -0,0 +1,10 @@
|
||||||
|
jest.mock('sentry-expo', () => ({
|
||||||
|
init: () => jest.fn(),
|
||||||
|
Native: {
|
||||||
|
ReactNativeTracing: jest.fn().mockImplementation(() => ({
|
||||||
|
start: jest.fn(),
|
||||||
|
stop: jest.fn(),
|
||||||
|
})),
|
||||||
|
ReactNavigationInstrumentation: jest.fn(),
|
||||||
|
},
|
||||||
|
}))
|
|
@ -30,5 +30,10 @@ module.exports = function (api) {
|
||||||
],
|
],
|
||||||
'react-native-reanimated/plugin', // NOTE: this plugin MUST be last
|
'react-native-reanimated/plugin', // NOTE: this plugin MUST be last
|
||||||
],
|
],
|
||||||
|
env: {
|
||||||
|
production: {
|
||||||
|
plugins: ['transform-remove-console'],
|
||||||
|
},
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,14 @@
|
||||||
|
# Testing instructions
|
||||||
|
|
||||||
|
### Using Maestro E2E tests
|
||||||
|
1. Install Maestro by following [these instuctions](https://maestro.mobile.dev/getting-started/installing-maestro). This will help us run the E2E tests.
|
||||||
|
2. You can write Maestro tests in `__e2e__/maestro` directory by creating a new `.yaml` file or by modifying an existing one.
|
||||||
|
3. You can also use [Maestro Studio](https://maestro.mobile.dev/getting-started/maestro-studio) which automatically generates commands by recording your actions on the app. Therefore, you can create realistic tests without having to manually write any code. Use the `maestro studio` command to start recording your actions.
|
||||||
|
|
||||||
|
|
||||||
|
### Using Flashlight for Performance Testing
|
||||||
|
1. Make sure Maestro is installed (optional: only for auomated testing) by following the instructions above
|
||||||
|
2. Install Flashlight by following [these instructions](https://docs.flashlight.dev/)
|
||||||
|
3. The simplest way to get started is by running `yarn perf:measure` which will run a live preview of the performance test results. You can [see a demo here](https://github.com/bamlab/flashlight/assets/4534323/4038a342-f145-4c3b-8cde-17949bf52612)
|
||||||
|
4. The `yarn perf:test:measure` will run the `scroll.yaml` test located in `__e2e__/maestro/scroll.yaml` and give the results in `.perf/results.json` which can be viewed by running `yarn:perf:results`
|
||||||
|
5. You can also run your own tests by running `yarn perf:test <path_to_test>` where `<path_to_test>` is the path to your test file. For example, `yarn perf:test __e2e__/maestro/scroll.yaml` will run the `scroll.yaml` test located in `__e2e__/maestro/scroll.yaml`.
|
8
eas.json
8
eas.json
|
@ -9,7 +9,7 @@
|
||||||
"distribution": "internal",
|
"distribution": "internal",
|
||||||
"ios": {
|
"ios": {
|
||||||
"simulator": true,
|
"simulator": true,
|
||||||
"resourceClass": "m-large"
|
"resourceClass": "large"
|
||||||
},
|
},
|
||||||
"channel": "development"
|
"channel": "development"
|
||||||
},
|
},
|
||||||
|
@ -17,20 +17,20 @@
|
||||||
"developmentClient": true,
|
"developmentClient": true,
|
||||||
"distribution": "internal",
|
"distribution": "internal",
|
||||||
"ios": {
|
"ios": {
|
||||||
"resourceClass": "m-large"
|
"resourceClass": "large"
|
||||||
},
|
},
|
||||||
"channel": "development"
|
"channel": "development"
|
||||||
},
|
},
|
||||||
"preview": {
|
"preview": {
|
||||||
"distribution": "internal",
|
"distribution": "internal",
|
||||||
"ios": {
|
"ios": {
|
||||||
"resourceClass": "m-large"
|
"resourceClass": "large"
|
||||||
},
|
},
|
||||||
"channel": "preview"
|
"channel": "preview"
|
||||||
},
|
},
|
||||||
"production": {
|
"production": {
|
||||||
"ios": {
|
"ios": {
|
||||||
"resourceClass": "m-large"
|
"resourceClass": "large"
|
||||||
},
|
},
|
||||||
"channel": "production"
|
"channel": "production"
|
||||||
},
|
},
|
||||||
|
|
|
@ -74,3 +74,14 @@ jest.mock('lande', () => ({
|
||||||
__esModule: true, // this property makes it work
|
__esModule: true, // this property makes it work
|
||||||
default: jest.fn().mockReturnValue([['eng']]),
|
default: jest.fn().mockReturnValue([['eng']]),
|
||||||
}))
|
}))
|
||||||
|
|
||||||
|
jest.mock('sentry-expo', () => ({
|
||||||
|
init: () => jest.fn(),
|
||||||
|
Native: {
|
||||||
|
ReactNativeTracing: jest.fn().mockImplementation(() => ({
|
||||||
|
start: jest.fn(),
|
||||||
|
stop: jest.fn(),
|
||||||
|
})),
|
||||||
|
ReactNavigationInstrumentation: jest.fn(),
|
||||||
|
},
|
||||||
|
}))
|
||||||
|
|
11
package.json
11
package.json
|
@ -11,6 +11,7 @@
|
||||||
"web": "expo start --web",
|
"web": "expo start --web",
|
||||||
"build-web": "expo export:web && node ./scripts/post-web-build.js && cp --verbose ./web-build/static/js/*.* ./bskyweb/static/js/",
|
"build-web": "expo export:web && node ./scripts/post-web-build.js && cp --verbose ./web-build/static/js/*.* ./bskyweb/static/js/",
|
||||||
"start": "expo start --dev-client",
|
"start": "expo start --dev-client",
|
||||||
|
"start:prod": "expo start --dev-client --no-dev --minify",
|
||||||
"clean-cache": "rm -rf node_modules/.cache/babel-loader/*",
|
"clean-cache": "rm -rf node_modules/.cache/babel-loader/*",
|
||||||
"test": "jest --forceExit --testTimeout=20000 --bail",
|
"test": "jest --forceExit --testTimeout=20000 --bail",
|
||||||
"test-watch": "jest --watchAll",
|
"test-watch": "jest --watchAll",
|
||||||
|
@ -22,6 +23,11 @@
|
||||||
"e2e:metro": "RN_SRC_EXT=e2e.ts,e2e.tsx expo run:ios",
|
"e2e:metro": "RN_SRC_EXT=e2e.ts,e2e.tsx expo run:ios",
|
||||||
"e2e:build": "detox build -c ios.sim.debug",
|
"e2e:build": "detox build -c ios.sim.debug",
|
||||||
"e2e:run": "detox test --configuration ios.sim.debug --take-screenshots all",
|
"e2e:run": "detox test --configuration ios.sim.debug --take-screenshots all",
|
||||||
|
"perf:test": "maestro test",
|
||||||
|
"perf:test:run": "maestro test __e2e__/maestro/scroll.yaml",
|
||||||
|
"perf:test:measure": "flashlight test --bundleId xyz.blueskyweb.app --testCommand 'yarn perf:test' --duration 150000 --resultsFilePath .perf/results.json",
|
||||||
|
"perf:test:results": "flashlight report .perf/results.json",
|
||||||
|
"perf:measure": "flashlight measure",
|
||||||
"build:apk": "eas build -p android --profile dev-android-apk"
|
"build:apk": "eas build -p android --profile dev-android-apk"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
@ -53,7 +59,7 @@
|
||||||
"@segment/analytics-react": "^1.0.0-rc1",
|
"@segment/analytics-react": "^1.0.0-rc1",
|
||||||
"@segment/analytics-react-native": "^2.10.1",
|
"@segment/analytics-react-native": "^2.10.1",
|
||||||
"@segment/sovran-react-native": "^0.4.5",
|
"@segment/sovran-react-native": "^0.4.5",
|
||||||
"@sentry/react-native": "5.5.0",
|
"@sentry/react-native": "5.10.0",
|
||||||
"@tanstack/react-query": "^4.33.0",
|
"@tanstack/react-query": "^4.33.0",
|
||||||
"@tiptap/core": "^2.0.0-beta.220",
|
"@tiptap/core": "^2.0.0-beta.220",
|
||||||
"@tiptap/extension-document": "^2.0.0-beta.220",
|
"@tiptap/extension-document": "^2.0.0-beta.220",
|
||||||
|
@ -71,6 +77,7 @@
|
||||||
"@zxing/text-encoding": "^0.9.0",
|
"@zxing/text-encoding": "^0.9.0",
|
||||||
"array.prototype.findlast": "^1.2.3",
|
"array.prototype.findlast": "^1.2.3",
|
||||||
"await-lock": "^2.2.2",
|
"await-lock": "^2.2.2",
|
||||||
|
"babel-plugin-transform-remove-console": "^6.9.4",
|
||||||
"base64-js": "^1.5.1",
|
"base64-js": "^1.5.1",
|
||||||
"bcp-47-match": "^2.0.3",
|
"bcp-47-match": "^2.0.3",
|
||||||
"email-validator": "^2.0.4",
|
"email-validator": "^2.0.4",
|
||||||
|
@ -148,7 +155,7 @@
|
||||||
"react-native-web-linear-gradient": "^1.1.2",
|
"react-native-web-linear-gradient": "^1.1.2",
|
||||||
"react-responsive": "^9.0.2",
|
"react-responsive": "^9.0.2",
|
||||||
"rn-fetch-blob": "^0.12.0",
|
"rn-fetch-blob": "^0.12.0",
|
||||||
"sentry-expo": "~7.0.0",
|
"sentry-expo": "~7.0.1",
|
||||||
"tippy.js": "^6.3.7",
|
"tippy.js": "^6.3.7",
|
||||||
"tlds": "^1.234.0",
|
"tlds": "^1.234.0",
|
||||||
"zeego": "^1.6.2",
|
"zeego": "^1.6.2",
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
diff --git a/node_modules/@sentry/react-native/dist/js/utils/ignorerequirecyclelogs.js b/node_modules/@sentry/react-native/dist/js/utils/ignorerequirecyclelogs.js
|
diff --git a/node_modules/@sentry/react-native/dist/js/utils/ignorerequirecyclelogs.js b/node_modules/@sentry/react-native/dist/js/utils/ignorerequirecyclelogs.js
|
||||||
index 7e0b4cd..3fd7406 100644
|
index 7e0b4cd..177454c 100644
|
||||||
--- a/node_modules/@sentry/react-native/dist/js/utils/ignorerequirecyclelogs.js
|
--- a/node_modules/@sentry/react-native/dist/js/utils/ignorerequirecyclelogs.js
|
||||||
+++ b/node_modules/@sentry/react-native/dist/js/utils/ignorerequirecyclelogs.js
|
+++ b/node_modules/@sentry/react-native/dist/js/utils/ignorerequirecyclelogs.js
|
||||||
@@ -3,6 +3,8 @@ import { LogBox } from 'react-native';
|
@@ -3,6 +3,8 @@ import { LogBox } from 'react-native';
|
||||||
|
@ -12,3 +12,4 @@ index 7e0b4cd..3fd7406 100644
|
||||||
+ } catch (e) {}
|
+ } catch (e) {}
|
||||||
}
|
}
|
||||||
//# sourceMappingURL=ignorerequirecyclelogs.js.map
|
//# sourceMappingURL=ignorerequirecyclelogs.js.map
|
||||||
|
\ No newline at end of file
|
|
@ -1,36 +1,60 @@
|
||||||
import React from 'react'
|
import React from 'react'
|
||||||
import {autorun} from 'mobx'
|
import {autorun} from 'mobx'
|
||||||
import {useStores} from 'state/index'
|
import {useStores} from 'state/index'
|
||||||
import {Animated} from 'react-native'
|
import {
|
||||||
import {useAnimatedValue} from 'lib/hooks/useAnimatedValue'
|
Easing,
|
||||||
|
interpolate,
|
||||||
|
useAnimatedStyle,
|
||||||
|
useSharedValue,
|
||||||
|
withTiming,
|
||||||
|
} from 'react-native-reanimated'
|
||||||
|
|
||||||
export function useMinimalShellMode() {
|
export function useMinimalShellMode() {
|
||||||
const store = useStores()
|
const store = useStores()
|
||||||
const minimalShellInterp = useAnimatedValue(0)
|
const minimalShellInterp = useSharedValue(0)
|
||||||
const footerMinimalShellTransform = {
|
const footerMinimalShellTransform = useAnimatedStyle(() => {
|
||||||
opacity: Animated.subtract(1, minimalShellInterp),
|
return {
|
||||||
transform: [{translateY: Animated.multiply(minimalShellInterp, 50)}],
|
opacity: interpolate(minimalShellInterp.value, [0, 1], [1, 0]),
|
||||||
|
transform: [
|
||||||
|
{translateY: interpolate(minimalShellInterp.value, [0, 1], [0, 25])},
|
||||||
|
],
|
||||||
}
|
}
|
||||||
|
})
|
||||||
|
const headerMinimalShellTransform = useAnimatedStyle(() => {
|
||||||
|
return {
|
||||||
|
opacity: interpolate(minimalShellInterp.value, [0, 1], [1, 0]),
|
||||||
|
transform: [
|
||||||
|
{translateY: interpolate(minimalShellInterp.value, [0, 1], [0, -25])},
|
||||||
|
],
|
||||||
|
}
|
||||||
|
})
|
||||||
|
const fabMinimalShellTransform = useAnimatedStyle(() => {
|
||||||
|
return {
|
||||||
|
transform: [
|
||||||
|
{translateY: interpolate(minimalShellInterp.value, [0, 1], [-44, 0])},
|
||||||
|
],
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
return autorun(() => {
|
return autorun(() => {
|
||||||
if (store.shell.minimalShellMode) {
|
if (store.shell.minimalShellMode) {
|
||||||
Animated.timing(minimalShellInterp, {
|
minimalShellInterp.value = withTiming(1, {
|
||||||
toValue: 1,
|
duration: 125,
|
||||||
duration: 150,
|
easing: Easing.bezier(0.25, 0.1, 0.25, 1),
|
||||||
useNativeDriver: true,
|
})
|
||||||
isInteraction: false,
|
|
||||||
}).start()
|
|
||||||
} else {
|
} else {
|
||||||
Animated.timing(minimalShellInterp, {
|
minimalShellInterp.value = withTiming(0, {
|
||||||
toValue: 0,
|
duration: 125,
|
||||||
duration: 150,
|
easing: Easing.bezier(0.25, 0.1, 0.25, 1),
|
||||||
useNativeDriver: true,
|
})
|
||||||
isInteraction: false,
|
|
||||||
}).start()
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}, [minimalShellInterp, store])
|
}, [minimalShellInterp, store.shell.minimalShellMode])
|
||||||
|
|
||||||
return {footerMinimalShellTransform}
|
return {
|
||||||
|
footerMinimalShellTransform,
|
||||||
|
headerMinimalShellTransform,
|
||||||
|
fabMinimalShellTransform,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,210 @@
|
||||||
|
import {
|
||||||
|
FontAwesomeIcon,
|
||||||
|
FontAwesomeIconStyle,
|
||||||
|
} from '@fortawesome/react-native-fontawesome'
|
||||||
|
import {useIsFocused} from '@react-navigation/native'
|
||||||
|
import {useAnalytics} from '@segment/analytics-react-native'
|
||||||
|
import {useOnMainScroll} from 'lib/hooks/useOnMainScroll'
|
||||||
|
import {usePalette} from 'lib/hooks/usePalette'
|
||||||
|
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
|
||||||
|
import {ComposeIcon2} from 'lib/icons'
|
||||||
|
import {colors, s} from 'lib/styles'
|
||||||
|
import {observer} from 'mobx-react-lite'
|
||||||
|
import React from 'react'
|
||||||
|
import {FlatList, View} from 'react-native'
|
||||||
|
import {useStores} from 'state/index'
|
||||||
|
import {PostsFeedModel} from 'state/models/feeds/posts'
|
||||||
|
import {useHeaderOffset, POLL_FREQ} from 'view/screens/Home'
|
||||||
|
import {Feed} from '../posts/Feed'
|
||||||
|
import {TextLink} from '../util/Link'
|
||||||
|
import {FAB} from '../util/fab/FAB'
|
||||||
|
import {LoadLatestBtn} from '../util/load-latest/LoadLatestBtn'
|
||||||
|
import useAppState from 'react-native-appstate-hook'
|
||||||
|
|
||||||
|
export const FeedPage = observer(function FeedPageImpl({
|
||||||
|
testID,
|
||||||
|
isPageFocused,
|
||||||
|
feed,
|
||||||
|
renderEmptyState,
|
||||||
|
renderEndOfFeed,
|
||||||
|
}: {
|
||||||
|
testID?: string
|
||||||
|
feed: PostsFeedModel
|
||||||
|
isPageFocused: boolean
|
||||||
|
renderEmptyState: () => JSX.Element
|
||||||
|
renderEndOfFeed?: () => JSX.Element
|
||||||
|
}) {
|
||||||
|
const store = useStores()
|
||||||
|
const pal = usePalette('default')
|
||||||
|
const {isDesktop} = useWebMediaQueries()
|
||||||
|
const [onMainScroll, isScrolledDown, resetMainScroll] = useOnMainScroll(store)
|
||||||
|
const {screen, track} = useAnalytics()
|
||||||
|
const headerOffset = useHeaderOffset()
|
||||||
|
const scrollElRef = React.useRef<FlatList>(null)
|
||||||
|
const {appState} = useAppState({
|
||||||
|
onForeground: () => doPoll(true),
|
||||||
|
})
|
||||||
|
const isScreenFocused = useIsFocused()
|
||||||
|
const hasNew = feed.hasNewLatest && !feed.isRefreshing
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
// called on first load
|
||||||
|
if (!feed.hasLoaded && isPageFocused) {
|
||||||
|
feed.setup()
|
||||||
|
}
|
||||||
|
}, [isPageFocused, feed])
|
||||||
|
|
||||||
|
const doPoll = React.useCallback(
|
||||||
|
(knownActive = false) => {
|
||||||
|
if (
|
||||||
|
(!knownActive && appState !== 'active') ||
|
||||||
|
!isScreenFocused ||
|
||||||
|
!isPageFocused
|
||||||
|
) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (feed.isLoading) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
store.log.debug('HomeScreen: Polling for new posts')
|
||||||
|
feed.checkForLatest()
|
||||||
|
},
|
||||||
|
[appState, isScreenFocused, isPageFocused, store, feed],
|
||||||
|
)
|
||||||
|
|
||||||
|
const scrollToTop = React.useCallback(() => {
|
||||||
|
scrollElRef.current?.scrollToOffset({offset: -headerOffset})
|
||||||
|
resetMainScroll()
|
||||||
|
}, [headerOffset, resetMainScroll])
|
||||||
|
|
||||||
|
const onSoftReset = React.useCallback(() => {
|
||||||
|
if (isPageFocused) {
|
||||||
|
scrollToTop()
|
||||||
|
feed.refresh()
|
||||||
|
}
|
||||||
|
}, [isPageFocused, scrollToTop, feed])
|
||||||
|
|
||||||
|
// fires when page within screen is activated/deactivated
|
||||||
|
// - check for latest
|
||||||
|
React.useEffect(() => {
|
||||||
|
if (!isPageFocused || !isScreenFocused) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
const softResetSub = store.onScreenSoftReset(onSoftReset)
|
||||||
|
const feedCleanup = feed.registerListeners()
|
||||||
|
const pollInterval = setInterval(doPoll, POLL_FREQ)
|
||||||
|
|
||||||
|
screen('Feed')
|
||||||
|
store.log.debug('HomeScreen: Updating feed')
|
||||||
|
feed.checkForLatest()
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
clearInterval(pollInterval)
|
||||||
|
softResetSub.remove()
|
||||||
|
feedCleanup()
|
||||||
|
}
|
||||||
|
}, [store, doPoll, onSoftReset, screen, feed, isPageFocused, isScreenFocused])
|
||||||
|
|
||||||
|
const onPressCompose = React.useCallback(() => {
|
||||||
|
track('HomeScreen:PressCompose')
|
||||||
|
store.shell.openComposer({})
|
||||||
|
}, [store, track])
|
||||||
|
|
||||||
|
const onPressTryAgain = React.useCallback(() => {
|
||||||
|
feed.refresh()
|
||||||
|
}, [feed])
|
||||||
|
|
||||||
|
const onPressLoadLatest = React.useCallback(() => {
|
||||||
|
scrollToTop()
|
||||||
|
feed.refresh()
|
||||||
|
}, [feed, scrollToTop])
|
||||||
|
|
||||||
|
const ListHeaderComponent = React.useCallback(() => {
|
||||||
|
if (isDesktop) {
|
||||||
|
return (
|
||||||
|
<View
|
||||||
|
style={[
|
||||||
|
pal.view,
|
||||||
|
{
|
||||||
|
flexDirection: 'row',
|
||||||
|
alignItems: 'center',
|
||||||
|
justifyContent: 'space-between',
|
||||||
|
paddingHorizontal: 18,
|
||||||
|
paddingVertical: 12,
|
||||||
|
},
|
||||||
|
]}>
|
||||||
|
<TextLink
|
||||||
|
type="title-lg"
|
||||||
|
href="/"
|
||||||
|
style={[pal.text, {fontWeight: 'bold'}]}
|
||||||
|
text={
|
||||||
|
<>
|
||||||
|
{store.session.isSandbox ? 'SANDBOX' : 'Bluesky'}{' '}
|
||||||
|
{hasNew && (
|
||||||
|
<View
|
||||||
|
style={{
|
||||||
|
top: -8,
|
||||||
|
backgroundColor: colors.blue3,
|
||||||
|
width: 8,
|
||||||
|
height: 8,
|
||||||
|
borderRadius: 4,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
onPress={() => store.emitScreenSoftReset()}
|
||||||
|
/>
|
||||||
|
<TextLink
|
||||||
|
type="title-lg"
|
||||||
|
href="/settings/home-feed"
|
||||||
|
style={{fontWeight: 'bold'}}
|
||||||
|
accessibilityLabel="Feed Preferences"
|
||||||
|
accessibilityHint=""
|
||||||
|
text={
|
||||||
|
<FontAwesomeIcon
|
||||||
|
icon="sliders"
|
||||||
|
style={pal.textLight as FontAwesomeIconStyle}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
return <></>
|
||||||
|
}, [isDesktop, pal, store, hasNew])
|
||||||
|
|
||||||
|
return (
|
||||||
|
<View testID={testID} style={s.h100pct}>
|
||||||
|
<Feed
|
||||||
|
testID={testID ? `${testID}-feed` : undefined}
|
||||||
|
key="default"
|
||||||
|
feed={feed}
|
||||||
|
scrollElRef={scrollElRef}
|
||||||
|
onPressTryAgain={onPressTryAgain}
|
||||||
|
onScroll={onMainScroll}
|
||||||
|
scrollEventThrottle={100}
|
||||||
|
renderEmptyState={renderEmptyState}
|
||||||
|
renderEndOfFeed={renderEndOfFeed}
|
||||||
|
ListHeaderComponent={ListHeaderComponent}
|
||||||
|
headerOffset={headerOffset}
|
||||||
|
/>
|
||||||
|
{(isScrolledDown || hasNew) && (
|
||||||
|
<LoadLatestBtn
|
||||||
|
onPress={onPressLoadLatest}
|
||||||
|
label="Load new posts"
|
||||||
|
showIndicator={hasNew}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<FAB
|
||||||
|
testID="composeFAB"
|
||||||
|
onPress={onPressCompose}
|
||||||
|
icon={<ComposeIcon2 strokeWidth={1.5} size={29} style={s.white} />}
|
||||||
|
accessibilityRole="button"
|
||||||
|
accessibilityLabel="New post"
|
||||||
|
accessibilityHint=""
|
||||||
|
/>
|
||||||
|
</View>
|
||||||
|
)
|
||||||
|
})
|
|
@ -1,13 +1,14 @@
|
||||||
import React, {useMemo} from 'react'
|
import React, {useMemo} from 'react'
|
||||||
import {Animated, StyleSheet} from 'react-native'
|
import {StyleSheet} from 'react-native'
|
||||||
|
import Animated from 'react-native-reanimated'
|
||||||
import {observer} from 'mobx-react-lite'
|
import {observer} from 'mobx-react-lite'
|
||||||
import {TabBar} from 'view/com/pager/TabBar'
|
import {TabBar} from 'view/com/pager/TabBar'
|
||||||
import {RenderTabBarFnProps} from 'view/com/pager/Pager'
|
import {RenderTabBarFnProps} from 'view/com/pager/Pager'
|
||||||
import {useStores} from 'state/index'
|
import {useStores} from 'state/index'
|
||||||
import {usePalette} from 'lib/hooks/usePalette'
|
import {usePalette} from 'lib/hooks/usePalette'
|
||||||
import {useAnimatedValue} from 'lib/hooks/useAnimatedValue'
|
|
||||||
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
|
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
|
||||||
import {FeedsTabBar as FeedsTabBarMobile} from './FeedsTabBarMobile'
|
import {FeedsTabBar as FeedsTabBarMobile} from './FeedsTabBarMobile'
|
||||||
|
import {useMinimalShellMode} from 'lib/hooks/useMinimalShellMode'
|
||||||
|
|
||||||
export const FeedsTabBar = observer(function FeedsTabBarImpl(
|
export const FeedsTabBar = observer(function FeedsTabBarImpl(
|
||||||
props: RenderTabBarFnProps & {testID?: string; onPressSelected: () => void},
|
props: RenderTabBarFnProps & {testID?: string; onPressSelected: () => void},
|
||||||
|
@ -31,26 +32,12 @@ const FeedsTabBarTablet = observer(function FeedsTabBarTabletImpl(
|
||||||
[store.me.savedFeeds.pinnedFeedNames],
|
[store.me.savedFeeds.pinnedFeedNames],
|
||||||
)
|
)
|
||||||
const pal = usePalette('default')
|
const pal = usePalette('default')
|
||||||
const interp = useAnimatedValue(0)
|
const {headerMinimalShellTransform} = useMinimalShellMode()
|
||||||
|
|
||||||
React.useEffect(() => {
|
|
||||||
Animated.timing(interp, {
|
|
||||||
toValue: store.shell.minimalShellMode ? 1 : 0,
|
|
||||||
duration: 100,
|
|
||||||
useNativeDriver: true,
|
|
||||||
isInteraction: false,
|
|
||||||
}).start()
|
|
||||||
}, [interp, store.shell.minimalShellMode])
|
|
||||||
const transform = {
|
|
||||||
transform: [
|
|
||||||
{translateX: '-50%'},
|
|
||||||
{translateY: Animated.multiply(interp, -100)},
|
|
||||||
],
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
// @ts-ignore the type signature for transform wrong here, translateX and translateY need to be in separate objects -prf
|
// @ts-ignore the type signature for transform wrong here, translateX and translateY need to be in separate objects -prf
|
||||||
<Animated.View style={[pal.view, styles.tabBar, transform]}>
|
<Animated.View
|
||||||
|
style={[pal.view, styles.tabBar, headerMinimalShellTransform]}>
|
||||||
<TabBar
|
<TabBar
|
||||||
key={items.join(',')}
|
key={items.join(',')}
|
||||||
{...props}
|
{...props}
|
||||||
|
@ -65,7 +52,8 @@ const styles = StyleSheet.create({
|
||||||
tabBar: {
|
tabBar: {
|
||||||
position: 'absolute',
|
position: 'absolute',
|
||||||
zIndex: 1,
|
zIndex: 1,
|
||||||
left: '50%',
|
// @ts-ignore Web only -prf
|
||||||
|
left: 'calc(50% - 299px)',
|
||||||
width: 598,
|
width: 598,
|
||||||
top: 0,
|
top: 0,
|
||||||
flexDirection: 'row',
|
flexDirection: 'row',
|
||||||
|
|
|
@ -1,12 +1,10 @@
|
||||||
import React, {useMemo} from 'react'
|
import React, {useMemo} from 'react'
|
||||||
import {Animated, StyleSheet, TouchableOpacity, View} from 'react-native'
|
import {StyleSheet, TouchableOpacity, View} from 'react-native'
|
||||||
import {observer} from 'mobx-react-lite'
|
import {observer} from 'mobx-react-lite'
|
||||||
import {autorun} from 'mobx'
|
|
||||||
import {TabBar} from 'view/com/pager/TabBar'
|
import {TabBar} from 'view/com/pager/TabBar'
|
||||||
import {RenderTabBarFnProps} from 'view/com/pager/Pager'
|
import {RenderTabBarFnProps} from 'view/com/pager/Pager'
|
||||||
import {useStores} from 'state/index'
|
import {useStores} from 'state/index'
|
||||||
import {usePalette} from 'lib/hooks/usePalette'
|
import {usePalette} from 'lib/hooks/usePalette'
|
||||||
import {useAnimatedValue} from 'lib/hooks/useAnimatedValue'
|
|
||||||
import {useColorSchemeStyle} from 'lib/hooks/useColorSchemeStyle'
|
import {useColorSchemeStyle} from 'lib/hooks/useColorSchemeStyle'
|
||||||
import {Link} from '../util/Link'
|
import {Link} from '../util/Link'
|
||||||
import {Text} from '../util/text/Text'
|
import {Text} from '../util/text/Text'
|
||||||
|
@ -14,30 +12,17 @@ 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'
|
||||||
|
|
||||||
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 store = useStores()
|
const store = useStores()
|
||||||
const pal = usePalette('default')
|
const pal = usePalette('default')
|
||||||
const interp = useAnimatedValue(0)
|
|
||||||
|
|
||||||
React.useEffect(() => {
|
|
||||||
return autorun(() => {
|
|
||||||
Animated.timing(interp, {
|
|
||||||
toValue: store.shell.minimalShellMode ? 1 : 0,
|
|
||||||
duration: 150,
|
|
||||||
useNativeDriver: true,
|
|
||||||
isInteraction: false,
|
|
||||||
}).start()
|
|
||||||
})
|
|
||||||
}, [interp, store])
|
|
||||||
const transform = {
|
|
||||||
opacity: Animated.subtract(1, interp),
|
|
||||||
transform: [{translateY: Animated.multiply(interp, -50)}],
|
|
||||||
}
|
|
||||||
|
|
||||||
const brandBlue = useColorSchemeStyle(s.brandBlue, s.blue3)
|
const brandBlue = useColorSchemeStyle(s.brandBlue, s.blue3)
|
||||||
|
const {headerMinimalShellTransform} = useMinimalShellMode()
|
||||||
|
|
||||||
const onPressAvi = React.useCallback(() => {
|
const onPressAvi = React.useCallback(() => {
|
||||||
store.shell.openDrawer()
|
store.shell.openDrawer()
|
||||||
|
@ -48,13 +33,17 @@ export const FeedsTabBar = observer(function FeedsTabBarImpl(
|
||||||
[store.me.savedFeeds.pinnedFeedNames],
|
[store.me.savedFeeds.pinnedFeedNames],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const tabBarKey = useMemo(() => {
|
||||||
|
return items.join(',')
|
||||||
|
}, [items])
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Animated.View
|
<Animated.View
|
||||||
style={[
|
style={[
|
||||||
pal.view,
|
pal.view,
|
||||||
pal.border,
|
pal.border,
|
||||||
styles.tabBar,
|
styles.tabBar,
|
||||||
transform,
|
headerMinimalShellTransform,
|
||||||
store.shell.minimalShellMode && styles.disabled,
|
store.shell.minimalShellMode && styles.disabled,
|
||||||
]}>
|
]}>
|
||||||
<View style={[pal.view, styles.topBar]}>
|
<View style={[pal.view, styles.topBar]}>
|
||||||
|
@ -92,8 +81,11 @@ export const FeedsTabBar = observer(function FeedsTabBarImpl(
|
||||||
</View>
|
</View>
|
||||||
</View>
|
</View>
|
||||||
<TabBar
|
<TabBar
|
||||||
key={items.join(',')}
|
key={tabBarKey}
|
||||||
{...props}
|
onPressSelected={props.onPressSelected}
|
||||||
|
selectedPage={props.selectedPage}
|
||||||
|
onSelect={props.onSelect}
|
||||||
|
testID={props.testID}
|
||||||
items={items}
|
items={items}
|
||||||
indicatorColor={pal.colors.link}
|
indicatorColor={pal.colors.link}
|
||||||
/>
|
/>
|
||||||
|
|
|
@ -64,6 +64,7 @@ export function TabBar({
|
||||||
)
|
)
|
||||||
|
|
||||||
const styles = isDesktop || isTablet ? desktopStyles : mobileStyles
|
const styles = isDesktop || isTablet ? desktopStyles : mobileStyles
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<View testID={testID} style={[pal.view, styles.outer]}>
|
<View testID={testID} style={[pal.view, styles.outer]}>
|
||||||
<DraggableScrollView
|
<DraggableScrollView
|
||||||
|
|
|
@ -96,7 +96,7 @@ export const Feed = observer(function Feed({
|
||||||
}, [feed, track, setIsRefreshing])
|
}, [feed, track, setIsRefreshing])
|
||||||
|
|
||||||
const onEndReached = React.useCallback(async () => {
|
const onEndReached = React.useCallback(async () => {
|
||||||
if (!feed.hasLoaded) return
|
if (!feed.hasLoaded || !feed.hasMore) return
|
||||||
|
|
||||||
track('Feed:onEndReached')
|
track('Feed:onEndReached')
|
||||||
try {
|
try {
|
||||||
|
@ -178,7 +178,7 @@ export const Feed = observer(function Feed({
|
||||||
scrollEventThrottle={scrollEventThrottle}
|
scrollEventThrottle={scrollEventThrottle}
|
||||||
indicatorStyle={theme.colorScheme === 'dark' ? 'white' : 'black'}
|
indicatorStyle={theme.colorScheme === 'dark' ? 'white' : 'black'}
|
||||||
onEndReached={onEndReached}
|
onEndReached={onEndReached}
|
||||||
onEndReachedThreshold={0.6}
|
onEndReachedThreshold={2}
|
||||||
removeClippedSubviews={true}
|
removeClippedSubviews={true}
|
||||||
contentOffset={{x: 0, y: headerOffset * -1}}
|
contentOffset={{x: 0, y: headerOffset * -1}}
|
||||||
extraData={extraData}
|
extraData={extraData}
|
||||||
|
|
|
@ -223,9 +223,9 @@ const SuggestedFollow = observer(function SuggestedFollowImpl({
|
||||||
|
|
||||||
const onPress = React.useCallback(async () => {
|
const onPress = React.useCallback(async () => {
|
||||||
try {
|
try {
|
||||||
const {following} = await toggle()
|
const {following: isFollowing} = await toggle()
|
||||||
|
|
||||||
if (following) {
|
if (isFollowing) {
|
||||||
track('ProfileHeader:SuggestedFollowFollowed')
|
track('ProfileHeader:SuggestedFollowFollowed')
|
||||||
}
|
}
|
||||||
} catch (e: any) {
|
} catch (e: any) {
|
||||||
|
|
|
@ -1,17 +1,17 @@
|
||||||
import React from 'react'
|
import React from 'react'
|
||||||
import {observer} from 'mobx-react-lite'
|
import {observer} from 'mobx-react-lite'
|
||||||
import {autorun} from 'mobx'
|
import {StyleSheet, TouchableOpacity, View} from 'react-native'
|
||||||
import {Animated, StyleSheet, TouchableOpacity, View} from 'react-native'
|
|
||||||
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 {Text} from './text/Text'
|
import {Text} from './text/Text'
|
||||||
import {useStores} from 'state/index'
|
import {useStores} from 'state/index'
|
||||||
import {usePalette} from 'lib/hooks/usePalette'
|
import {usePalette} from 'lib/hooks/usePalette'
|
||||||
import {useAnimatedValue} from 'lib/hooks/useAnimatedValue'
|
|
||||||
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 Animated from 'react-native-reanimated'
|
||||||
|
|
||||||
const BACK_HITSLOP = {left: 20, top: 20, right: 50, bottom: 20}
|
const BACK_HITSLOP = {left: 20, top: 20, right: 50, bottom: 20}
|
||||||
|
|
||||||
|
@ -150,32 +150,8 @@ const Container = observer(function ContainerImpl({
|
||||||
hideOnScroll: boolean
|
hideOnScroll: boolean
|
||||||
showBorder?: boolean
|
showBorder?: boolean
|
||||||
}) {
|
}) {
|
||||||
const store = useStores()
|
|
||||||
const pal = usePalette('default')
|
const pal = usePalette('default')
|
||||||
const interp = useAnimatedValue(0)
|
const {headerMinimalShellTransform} = useMinimalShellMode()
|
||||||
|
|
||||||
React.useEffect(() => {
|
|
||||||
return autorun(() => {
|
|
||||||
if (store.shell.minimalShellMode) {
|
|
||||||
Animated.timing(interp, {
|
|
||||||
toValue: 1,
|
|
||||||
duration: 100,
|
|
||||||
useNativeDriver: true,
|
|
||||||
isInteraction: false,
|
|
||||||
}).start()
|
|
||||||
} else {
|
|
||||||
Animated.timing(interp, {
|
|
||||||
toValue: 0,
|
|
||||||
duration: 100,
|
|
||||||
useNativeDriver: true,
|
|
||||||
isInteraction: false,
|
|
||||||
}).start()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}, [interp, store])
|
|
||||||
const transform = {
|
|
||||||
transform: [{translateY: Animated.multiply(interp, -100)}],
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!hideOnScroll) {
|
if (!hideOnScroll) {
|
||||||
return (
|
return (
|
||||||
|
@ -198,7 +174,7 @@ const Container = observer(function ContainerImpl({
|
||||||
styles.headerFloating,
|
styles.headerFloating,
|
||||||
pal.view,
|
pal.view,
|
||||||
pal.border,
|
pal.border,
|
||||||
transform,
|
headerMinimalShellTransform,
|
||||||
showBorder && styles.border,
|
showBorder && styles.border,
|
||||||
]}>
|
]}>
|
||||||
{children}
|
{children}
|
||||||
|
|
|
@ -1,14 +1,13 @@
|
||||||
import React, {ComponentProps} from 'react'
|
import React, {ComponentProps} from 'react'
|
||||||
import {observer} from 'mobx-react-lite'
|
import {observer} from 'mobx-react-lite'
|
||||||
import {autorun} from 'mobx'
|
import {StyleSheet, TouchableWithoutFeedback} from 'react-native'
|
||||||
import {Animated, StyleSheet, TouchableWithoutFeedback} from 'react-native'
|
|
||||||
import LinearGradient from 'react-native-linear-gradient'
|
import LinearGradient from 'react-native-linear-gradient'
|
||||||
import {gradients} from 'lib/styles'
|
import {gradients} from 'lib/styles'
|
||||||
import {useAnimatedValue} from 'lib/hooks/useAnimatedValue'
|
|
||||||
import {useStores} from 'state/index'
|
|
||||||
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
|
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
|
||||||
import {useSafeAreaInsets} from 'react-native-safe-area-context'
|
import {useSafeAreaInsets} from 'react-native-safe-area-context'
|
||||||
import {clamp} from 'lib/numbers'
|
import {clamp} from 'lib/numbers'
|
||||||
|
import {useMinimalShellMode} from 'lib/hooks/useMinimalShellMode'
|
||||||
|
import Animated from 'react-native-reanimated'
|
||||||
|
|
||||||
export interface FABProps
|
export interface FABProps
|
||||||
extends ComponentProps<typeof TouchableWithoutFeedback> {
|
extends ComponentProps<typeof TouchableWithoutFeedback> {
|
||||||
|
@ -22,30 +21,30 @@ export const FABInner = observer(function FABInnerImpl({
|
||||||
...props
|
...props
|
||||||
}: FABProps) {
|
}: FABProps) {
|
||||||
const insets = useSafeAreaInsets()
|
const insets = useSafeAreaInsets()
|
||||||
const {isTablet} = useWebMediaQueries()
|
const {isMobile, isTablet} = useWebMediaQueries()
|
||||||
const store = useStores()
|
const {fabMinimalShellTransform} = useMinimalShellMode()
|
||||||
const interp = useAnimatedValue(0)
|
|
||||||
React.useEffect(() => {
|
const size = React.useMemo(() => {
|
||||||
return autorun(() => {
|
return isTablet ? styles.sizeLarge : styles.sizeRegular
|
||||||
Animated.timing(interp, {
|
}, [isTablet])
|
||||||
toValue: store.shell.minimalShellMode ? 0 : 1,
|
const tabletSpacing = React.useMemo(() => {
|
||||||
duration: 100,
|
return isTablet
|
||||||
useNativeDriver: true,
|
? {right: 50, bottom: 50}
|
||||||
isInteraction: false,
|
|
||||||
}).start()
|
|
||||||
})
|
|
||||||
}, [interp, store])
|
|
||||||
const transform = isTablet
|
|
||||||
? undefined
|
|
||||||
: {
|
: {
|
||||||
transform: [{translateY: Animated.multiply(interp, -44)}],
|
right: 24,
|
||||||
|
bottom: clamp(insets.bottom, 15, 60) + 15,
|
||||||
}
|
}
|
||||||
const size = isTablet ? styles.sizeLarge : styles.sizeRegular
|
}, [insets.bottom, isTablet])
|
||||||
const right = isTablet ? 50 : 24
|
|
||||||
const bottom = isTablet ? 50 : clamp(insets.bottom, 15, 60) + 15
|
|
||||||
return (
|
return (
|
||||||
<TouchableWithoutFeedback testID={testID} {...props}>
|
<TouchableWithoutFeedback testID={testID} {...props}>
|
||||||
<Animated.View style={[styles.outer, size, {right, bottom}, transform]}>
|
<Animated.View
|
||||||
|
style={[
|
||||||
|
styles.outer,
|
||||||
|
size,
|
||||||
|
tabletSpacing,
|
||||||
|
isMobile && fabMinimalShellTransform,
|
||||||
|
]}>
|
||||||
<LinearGradient
|
<LinearGradient
|
||||||
colors={[gradients.blueLight.start, gradients.blueLight.end]}
|
colors={[gradients.blueLight.start, gradients.blueLight.end]}
|
||||||
start={{x: 0, y: 0}}
|
start={{x: 0, y: 0}}
|
||||||
|
|
|
@ -2,16 +2,12 @@ import React from 'react'
|
||||||
import {StyleSheet, TouchableOpacity, View} from 'react-native'
|
import {StyleSheet, TouchableOpacity, View} from 'react-native'
|
||||||
import {observer} from 'mobx-react-lite'
|
import {observer} from 'mobx-react-lite'
|
||||||
import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
|
import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
|
||||||
import {useSafeAreaInsets} from 'react-native-safe-area-context'
|
|
||||||
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 {colors} from 'lib/styles'
|
import {colors} from 'lib/styles'
|
||||||
import {HITSLOP_20} from 'lib/constants'
|
import {HITSLOP_20} from 'lib/constants'
|
||||||
import {isWeb} from 'platform/detection'
|
import {useMinimalShellMode} from 'lib/hooks/useMinimalShellMode'
|
||||||
import {clamp} from 'lib/numbers'
|
import Animated from 'react-native-reanimated'
|
||||||
import Animated, {useAnimatedStyle, withTiming} from 'react-native-reanimated'
|
|
||||||
|
|
||||||
const AnimatedTouchableOpacity =
|
const AnimatedTouchableOpacity =
|
||||||
Animated.createAnimatedComponent(TouchableOpacity)
|
Animated.createAnimatedComponent(TouchableOpacity)
|
||||||
|
|
||||||
|
@ -23,20 +19,11 @@ export const LoadLatestBtn = observer(function LoadLatestBtnImpl({
|
||||||
onPress: () => void
|
onPress: () => void
|
||||||
label: string
|
label: string
|
||||||
showIndicator: boolean
|
showIndicator: boolean
|
||||||
minimalShellMode?: boolean // NOTE not used on mobile -prf
|
|
||||||
}) {
|
}) {
|
||||||
const store = useStores()
|
|
||||||
const pal = usePalette('default')
|
const pal = usePalette('default')
|
||||||
const {isDesktop, isTablet} = useWebMediaQueries()
|
const {isDesktop, isTablet, isMobile} = useWebMediaQueries()
|
||||||
const safeAreaInsets = useSafeAreaInsets()
|
const {fabMinimalShellTransform} = useMinimalShellMode()
|
||||||
const minMode = store.shell.minimalShellMode
|
|
||||||
const bottom = isTablet
|
|
||||||
? 50
|
|
||||||
: (minMode || isDesktop ? 16 : 60) +
|
|
||||||
(isWeb ? 20 : clamp(safeAreaInsets.bottom, 15, 60))
|
|
||||||
const animatedStyle = useAnimatedStyle(() => ({
|
|
||||||
bottom: withTiming(bottom, {duration: 150}),
|
|
||||||
}))
|
|
||||||
return (
|
return (
|
||||||
<AnimatedTouchableOpacity
|
<AnimatedTouchableOpacity
|
||||||
style={[
|
style={[
|
||||||
|
@ -45,7 +32,7 @@ export const LoadLatestBtn = observer(function LoadLatestBtnImpl({
|
||||||
isTablet && styles.loadLatestTablet,
|
isTablet && styles.loadLatestTablet,
|
||||||
pal.borderDark,
|
pal.borderDark,
|
||||||
pal.view,
|
pal.view,
|
||||||
animatedStyle,
|
isMobile && fabMinimalShellTransform,
|
||||||
]}
|
]}
|
||||||
onPress={onPress}
|
onPress={onPress}
|
||||||
hitSlop={HITSLOP_20}
|
hitSlop={HITSLOP_20}
|
||||||
|
@ -73,13 +60,11 @@ const styles = StyleSheet.create({
|
||||||
},
|
},
|
||||||
loadLatestTablet: {
|
loadLatestTablet: {
|
||||||
// @ts-ignore web only
|
// @ts-ignore web only
|
||||||
left: '50vw',
|
left: 'calc(50vw - 282px)',
|
||||||
transform: [{translateX: -282}],
|
|
||||||
},
|
},
|
||||||
loadLatestDesktop: {
|
loadLatestDesktop: {
|
||||||
// @ts-ignore web only
|
// @ts-ignore web only
|
||||||
left: '50vw',
|
left: 'calc(50vw - 382px)',
|
||||||
transform: [{translateX: -382}],
|
|
||||||
},
|
},
|
||||||
indicator: {
|
indicator: {
|
||||||
position: 'absolute',
|
position: 'absolute',
|
||||||
|
|
|
@ -1,33 +1,22 @@
|
||||||
import React from 'react'
|
import React from 'react'
|
||||||
import {FlatList, View, useWindowDimensions} from 'react-native'
|
import {useWindowDimensions} from 'react-native'
|
||||||
import {useFocusEffect, useIsFocused} from '@react-navigation/native'
|
import {useFocusEffect} from '@react-navigation/native'
|
||||||
import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
|
|
||||||
import {FontAwesomeIconStyle} from '@fortawesome/react-native-fontawesome'
|
|
||||||
import {AppBskyFeedGetFeed as GetCustomFeed} from '@atproto/api'
|
import {AppBskyFeedGetFeed as GetCustomFeed} from '@atproto/api'
|
||||||
import {observer} from 'mobx-react-lite'
|
import {observer} from 'mobx-react-lite'
|
||||||
import useAppState from 'react-native-appstate-hook'
|
|
||||||
import isEqual from 'lodash.isequal'
|
import isEqual from 'lodash.isequal'
|
||||||
import {NativeStackScreenProps, HomeTabNavigatorParams} from 'lib/routes/types'
|
import {NativeStackScreenProps, HomeTabNavigatorParams} from 'lib/routes/types'
|
||||||
import {PostsFeedModel} from 'state/models/feeds/posts'
|
import {PostsFeedModel} from 'state/models/feeds/posts'
|
||||||
import {withAuthRequired} from 'view/com/auth/withAuthRequired'
|
import {withAuthRequired} from 'view/com/auth/withAuthRequired'
|
||||||
import {TextLink} from 'view/com/util/Link'
|
|
||||||
import {Feed} from '../com/posts/Feed'
|
|
||||||
import {FollowingEmptyState} from 'view/com/posts/FollowingEmptyState'
|
import {FollowingEmptyState} from 'view/com/posts/FollowingEmptyState'
|
||||||
import {FollowingEndOfFeed} from 'view/com/posts/FollowingEndOfFeed'
|
import {FollowingEndOfFeed} from 'view/com/posts/FollowingEndOfFeed'
|
||||||
import {CustomFeedEmptyState} from 'view/com/posts/CustomFeedEmptyState'
|
import {CustomFeedEmptyState} from 'view/com/posts/CustomFeedEmptyState'
|
||||||
import {LoadLatestBtn} from '../com/util/load-latest/LoadLatestBtn'
|
|
||||||
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 {FAB} from '../com/util/fab/FAB'
|
|
||||||
import {useStores} from 'state/index'
|
import {useStores} from 'state/index'
|
||||||
import {usePalette} from 'lib/hooks/usePalette'
|
|
||||||
import {s, colors} from 'lib/styles'
|
|
||||||
import {useOnMainScroll} from 'lib/hooks/useOnMainScroll'
|
|
||||||
import {useAnalytics} from 'lib/analytics/analytics'
|
|
||||||
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
|
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
|
||||||
import {ComposeIcon2} from 'lib/icons'
|
import {FeedPage} from 'view/com/feeds/FeedPage'
|
||||||
|
|
||||||
const POLL_FREQ = 30e3 // 30sec
|
export const POLL_FREQ = 30e3 // 30sec
|
||||||
|
|
||||||
type Props = NativeStackScreenProps<HomeTabNavigatorParams, 'Home'>
|
type Props = NativeStackScreenProps<HomeTabNavigatorParams, 'Home'>
|
||||||
export const HomeScreen = withAuthRequired(
|
export const HomeScreen = withAuthRequired(
|
||||||
|
@ -98,7 +87,9 @@ export const HomeScreen = withAuthRequired(
|
||||||
(props: RenderTabBarFnProps) => {
|
(props: RenderTabBarFnProps) => {
|
||||||
return (
|
return (
|
||||||
<FeedsTabBar
|
<FeedsTabBar
|
||||||
{...props}
|
key="FEEDS_TAB_BAR"
|
||||||
|
selectedPage={props.selectedPage}
|
||||||
|
onSelect={props.onSelect}
|
||||||
testID="homeScreenFeedTabs"
|
testID="homeScreenFeedTabs"
|
||||||
onPressSelected={onPressSelected}
|
onPressSelected={onPressSelected}
|
||||||
/>
|
/>
|
||||||
|
@ -111,10 +102,6 @@ export const HomeScreen = withAuthRequired(
|
||||||
return <FollowingEmptyState />
|
return <FollowingEmptyState />
|
||||||
}, [])
|
}, [])
|
||||||
|
|
||||||
const renderFollowingEndOfFeed = React.useCallback(() => {
|
|
||||||
return <FollowingEndOfFeed />
|
|
||||||
}, [])
|
|
||||||
|
|
||||||
const renderCustomFeedEmptyState = React.useCallback(() => {
|
const renderCustomFeedEmptyState = React.useCallback(() => {
|
||||||
return <CustomFeedEmptyState />
|
return <CustomFeedEmptyState />
|
||||||
}, [])
|
}, [])
|
||||||
|
@ -132,7 +119,7 @@ export const HomeScreen = withAuthRequired(
|
||||||
isPageFocused={selectedPage === 0}
|
isPageFocused={selectedPage === 0}
|
||||||
feed={store.me.mainFeed}
|
feed={store.me.mainFeed}
|
||||||
renderEmptyState={renderFollowingEmptyState}
|
renderEmptyState={renderFollowingEmptyState}
|
||||||
renderEndOfFeed={renderFollowingEndOfFeed}
|
renderEndOfFeed={FollowingEndOfFeed}
|
||||||
/>
|
/>
|
||||||
{customFeeds.map((f, index) => {
|
{customFeeds.map((f, index) => {
|
||||||
return (
|
return (
|
||||||
|
@ -150,196 +137,7 @@ export const HomeScreen = withAuthRequired(
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
|
|
||||||
const FeedPage = observer(function FeedPageImpl({
|
export function useHeaderOffset() {
|
||||||
testID,
|
|
||||||
isPageFocused,
|
|
||||||
feed,
|
|
||||||
renderEmptyState,
|
|
||||||
renderEndOfFeed,
|
|
||||||
}: {
|
|
||||||
testID?: string
|
|
||||||
feed: PostsFeedModel
|
|
||||||
isPageFocused: boolean
|
|
||||||
renderEmptyState: () => JSX.Element
|
|
||||||
renderEndOfFeed?: () => JSX.Element
|
|
||||||
}) {
|
|
||||||
const store = useStores()
|
|
||||||
const pal = usePalette('default')
|
|
||||||
const {isDesktop} = useWebMediaQueries()
|
|
||||||
const [onMainScroll, isScrolledDown, resetMainScroll] = useOnMainScroll(store)
|
|
||||||
const {screen, track} = useAnalytics()
|
|
||||||
const headerOffset = useHeaderOffset()
|
|
||||||
const scrollElRef = React.useRef<FlatList>(null)
|
|
||||||
const {appState} = useAppState({
|
|
||||||
onForeground: () => doPoll(true),
|
|
||||||
})
|
|
||||||
const isScreenFocused = useIsFocused()
|
|
||||||
const hasNew = feed.hasNewLatest && !feed.isRefreshing
|
|
||||||
|
|
||||||
React.useEffect(() => {
|
|
||||||
// called on first load
|
|
||||||
if (!feed.hasLoaded && isPageFocused) {
|
|
||||||
feed.setup()
|
|
||||||
}
|
|
||||||
}, [isPageFocused, feed])
|
|
||||||
|
|
||||||
const doPoll = React.useCallback(
|
|
||||||
(knownActive = false) => {
|
|
||||||
if (
|
|
||||||
(!knownActive && appState !== 'active') ||
|
|
||||||
!isScreenFocused ||
|
|
||||||
!isPageFocused
|
|
||||||
) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if (feed.isLoading) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
store.log.debug('HomeScreen: Polling for new posts')
|
|
||||||
feed.checkForLatest()
|
|
||||||
},
|
|
||||||
[appState, isScreenFocused, isPageFocused, store, feed],
|
|
||||||
)
|
|
||||||
|
|
||||||
const scrollToTop = React.useCallback(() => {
|
|
||||||
scrollElRef.current?.scrollToOffset({offset: -headerOffset})
|
|
||||||
resetMainScroll()
|
|
||||||
}, [headerOffset, resetMainScroll])
|
|
||||||
|
|
||||||
const onSoftReset = React.useCallback(() => {
|
|
||||||
if (isPageFocused) {
|
|
||||||
scrollToTop()
|
|
||||||
feed.refresh()
|
|
||||||
}
|
|
||||||
}, [isPageFocused, scrollToTop, feed])
|
|
||||||
|
|
||||||
// fires when page within screen is activated/deactivated
|
|
||||||
// - check for latest
|
|
||||||
React.useEffect(() => {
|
|
||||||
if (!isPageFocused || !isScreenFocused) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
const softResetSub = store.onScreenSoftReset(onSoftReset)
|
|
||||||
const feedCleanup = feed.registerListeners()
|
|
||||||
const pollInterval = setInterval(doPoll, POLL_FREQ)
|
|
||||||
|
|
||||||
screen('Feed')
|
|
||||||
store.log.debug('HomeScreen: Updating feed')
|
|
||||||
feed.checkForLatest()
|
|
||||||
|
|
||||||
return () => {
|
|
||||||
clearInterval(pollInterval)
|
|
||||||
softResetSub.remove()
|
|
||||||
feedCleanup()
|
|
||||||
}
|
|
||||||
}, [store, doPoll, onSoftReset, screen, feed, isPageFocused, isScreenFocused])
|
|
||||||
|
|
||||||
const onPressCompose = React.useCallback(() => {
|
|
||||||
track('HomeScreen:PressCompose')
|
|
||||||
store.shell.openComposer({})
|
|
||||||
}, [store, track])
|
|
||||||
|
|
||||||
const onPressTryAgain = React.useCallback(() => {
|
|
||||||
feed.refresh()
|
|
||||||
}, [feed])
|
|
||||||
|
|
||||||
const onPressLoadLatest = React.useCallback(() => {
|
|
||||||
scrollToTop()
|
|
||||||
feed.refresh()
|
|
||||||
}, [feed, scrollToTop])
|
|
||||||
|
|
||||||
const ListHeaderComponent = React.useCallback(() => {
|
|
||||||
if (isDesktop) {
|
|
||||||
return (
|
|
||||||
<View
|
|
||||||
style={[
|
|
||||||
pal.view,
|
|
||||||
{
|
|
||||||
flexDirection: 'row',
|
|
||||||
alignItems: 'center',
|
|
||||||
justifyContent: 'space-between',
|
|
||||||
paddingHorizontal: 18,
|
|
||||||
paddingVertical: 12,
|
|
||||||
},
|
|
||||||
]}>
|
|
||||||
<TextLink
|
|
||||||
type="title-lg"
|
|
||||||
href="/"
|
|
||||||
style={[pal.text, {fontWeight: 'bold'}]}
|
|
||||||
text={
|
|
||||||
<>
|
|
||||||
{store.session.isSandbox ? 'SANDBOX' : 'Bluesky'}{' '}
|
|
||||||
{hasNew && (
|
|
||||||
<View
|
|
||||||
style={{
|
|
||||||
top: -8,
|
|
||||||
backgroundColor: colors.blue3,
|
|
||||||
width: 8,
|
|
||||||
height: 8,
|
|
||||||
borderRadius: 4,
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</>
|
|
||||||
}
|
|
||||||
onPress={() => store.emitScreenSoftReset()}
|
|
||||||
/>
|
|
||||||
<TextLink
|
|
||||||
type="title-lg"
|
|
||||||
href="/settings/home-feed"
|
|
||||||
style={{fontWeight: 'bold'}}
|
|
||||||
accessibilityLabel="Feed Preferences"
|
|
||||||
accessibilityHint=""
|
|
||||||
text={
|
|
||||||
<FontAwesomeIcon
|
|
||||||
icon="sliders"
|
|
||||||
style={pal.textLight as FontAwesomeIconStyle}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</View>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
return <></>
|
|
||||||
}, [isDesktop, pal, store, hasNew])
|
|
||||||
|
|
||||||
return (
|
|
||||||
<View testID={testID} style={s.h100pct}>
|
|
||||||
<Feed
|
|
||||||
testID={testID ? `${testID}-feed` : undefined}
|
|
||||||
key="default"
|
|
||||||
feed={feed}
|
|
||||||
scrollElRef={scrollElRef}
|
|
||||||
onPressTryAgain={onPressTryAgain}
|
|
||||||
onScroll={onMainScroll}
|
|
||||||
scrollEventThrottle={100}
|
|
||||||
renderEmptyState={renderEmptyState}
|
|
||||||
renderEndOfFeed={renderEndOfFeed}
|
|
||||||
ListHeaderComponent={ListHeaderComponent}
|
|
||||||
headerOffset={headerOffset}
|
|
||||||
/>
|
|
||||||
{(isScrolledDown || hasNew) && (
|
|
||||||
<LoadLatestBtn
|
|
||||||
onPress={onPressLoadLatest}
|
|
||||||
label="Load new posts"
|
|
||||||
showIndicator={hasNew}
|
|
||||||
minimalShellMode={store.shell.minimalShellMode}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
<FAB
|
|
||||||
testID="composeFAB"
|
|
||||||
onPress={onPressCompose}
|
|
||||||
icon={<ComposeIcon2 strokeWidth={1.5} size={29} style={s.white} />}
|
|
||||||
accessibilityRole="button"
|
|
||||||
accessibilityLabel="New post"
|
|
||||||
accessibilityHint=""
|
|
||||||
/>
|
|
||||||
</View>
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
function useHeaderOffset() {
|
|
||||||
const {isDesktop, isTablet} = useWebMediaQueries()
|
const {isDesktop, isTablet} = useWebMediaQueries()
|
||||||
const {fontScale} = useWindowDimensions()
|
const {fontScale} = useWindowDimensions()
|
||||||
if (isDesktop) {
|
if (isDesktop) {
|
||||||
|
|
|
@ -156,7 +156,6 @@ export const NotificationsScreen = withAuthRequired(
|
||||||
onPress={onPressLoadLatest}
|
onPress={onPressLoadLatest}
|
||||||
label="Load new notifications"
|
label="Load new notifications"
|
||||||
showIndicator={hasNew}
|
showIndicator={hasNew}
|
||||||
minimalShellMode={true}
|
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</View>
|
</View>
|
||||||
|
|
|
@ -1,10 +1,6 @@
|
||||||
import React, {ComponentProps} from 'react'
|
import React, {ComponentProps} from 'react'
|
||||||
import {
|
import {GestureResponderEvent, TouchableOpacity, View} from 'react-native'
|
||||||
Animated,
|
import Animated from 'react-native-reanimated'
|
||||||
GestureResponderEvent,
|
|
||||||
TouchableOpacity,
|
|
||||||
View,
|
|
||||||
} from 'react-native'
|
|
||||||
import {StackActions} from '@react-navigation/native'
|
import {StackActions} from '@react-navigation/native'
|
||||||
import {BottomTabBarProps} from '@react-navigation/bottom-tabs'
|
import {BottomTabBarProps} from '@react-navigation/bottom-tabs'
|
||||||
import {useSafeAreaInsets} from 'react-native-safe-area-context'
|
import {useSafeAreaInsets} from 'react-native-safe-area-context'
|
||||||
|
|
|
@ -2,8 +2,8 @@ import React from 'react'
|
||||||
import {observer} from 'mobx-react-lite'
|
import {observer} from 'mobx-react-lite'
|
||||||
import {useStores} from 'state/index'
|
import {useStores} from 'state/index'
|
||||||
import {usePalette} from 'lib/hooks/usePalette'
|
import {usePalette} from 'lib/hooks/usePalette'
|
||||||
import {Animated} from 'react-native'
|
|
||||||
import {useNavigationState} from '@react-navigation/native'
|
import {useNavigationState} from '@react-navigation/native'
|
||||||
|
import Animated from 'react-native-reanimated'
|
||||||
import {useSafeAreaInsets} from 'react-native-safe-area-context'
|
import {useSafeAreaInsets} from 'react-native-safe-area-context'
|
||||||
import {getCurrentRoute, isTab} from 'lib/routes/helpers'
|
import {getCurrentRoute, isTab} from 'lib/routes/helpers'
|
||||||
import {styles} from './BottomBarStyles'
|
import {styles} from './BottomBarStyles'
|
||||||
|
|
118
yarn.lock
118
yarn.lock
|
@ -3744,6 +3744,16 @@
|
||||||
"@sentry/utils" "7.52.1"
|
"@sentry/utils" "7.52.1"
|
||||||
tslib "^1.9.3"
|
tslib "^1.9.3"
|
||||||
|
|
||||||
|
"@sentry-internal/tracing@7.69.0":
|
||||||
|
version "7.69.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/@sentry-internal/tracing/-/tracing-7.69.0.tgz#8d8eb740b72967b6ba3fdc0a5173aa55331b7d35"
|
||||||
|
integrity sha512-4BgeWZUj9MO6IgfO93C9ocP3+AdngqujF/+zB2rFdUe+y9S6koDyUC7jr9Knds/0Ta72N/0D6PwhgSCpHK8s0Q==
|
||||||
|
dependencies:
|
||||||
|
"@sentry/core" "7.69.0"
|
||||||
|
"@sentry/types" "7.69.0"
|
||||||
|
"@sentry/utils" "7.69.0"
|
||||||
|
tslib "^2.4.1 || ^1.9.3"
|
||||||
|
|
||||||
"@sentry/browser@7.52.0":
|
"@sentry/browser@7.52.0":
|
||||||
version "7.52.0"
|
version "7.52.0"
|
||||||
resolved "https://registry.yarnpkg.com/@sentry/browser/-/browser-7.52.0.tgz#55d266c89ed668389ff687e5cc885c27016ea85c"
|
resolved "https://registry.yarnpkg.com/@sentry/browser/-/browser-7.52.0.tgz#55d266c89ed668389ff687e5cc885c27016ea85c"
|
||||||
|
@ -3768,6 +3778,18 @@
|
||||||
"@sentry/utils" "7.52.1"
|
"@sentry/utils" "7.52.1"
|
||||||
tslib "^1.9.3"
|
tslib "^1.9.3"
|
||||||
|
|
||||||
|
"@sentry/browser@7.69.0":
|
||||||
|
version "7.69.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/@sentry/browser/-/browser-7.69.0.tgz#65427c90fb71c1775e2c1e38431efb7f4aec1e34"
|
||||||
|
integrity sha512-5ls+zu2PrMhHCIIhclKQsWX5u6WH0Ez5/GgrCMZTtZ1d70ukGSRUvpZG9qGf5Cw1ezS1LY+1HCc3whf8x8lyPw==
|
||||||
|
dependencies:
|
||||||
|
"@sentry-internal/tracing" "7.69.0"
|
||||||
|
"@sentry/core" "7.69.0"
|
||||||
|
"@sentry/replay" "7.69.0"
|
||||||
|
"@sentry/types" "7.69.0"
|
||||||
|
"@sentry/utils" "7.69.0"
|
||||||
|
tslib "^2.4.1 || ^1.9.3"
|
||||||
|
|
||||||
"@sentry/cli@2.17.5":
|
"@sentry/cli@2.17.5":
|
||||||
version "2.17.5"
|
version "2.17.5"
|
||||||
resolved "https://registry.yarnpkg.com/@sentry/cli/-/cli-2.17.5.tgz#d41e24893a843bcd41e14274044a7ddea9332824"
|
resolved "https://registry.yarnpkg.com/@sentry/cli/-/cli-2.17.5.tgz#d41e24893a843bcd41e14274044a7ddea9332824"
|
||||||
|
@ -3779,6 +3801,17 @@
|
||||||
proxy-from-env "^1.1.0"
|
proxy-from-env "^1.1.0"
|
||||||
which "^2.0.2"
|
which "^2.0.2"
|
||||||
|
|
||||||
|
"@sentry/cli@2.20.7":
|
||||||
|
version "2.20.7"
|
||||||
|
resolved "https://registry.yarnpkg.com/@sentry/cli/-/cli-2.20.7.tgz#8f7f3f632c330cac6bd2278d820948163f3128a6"
|
||||||
|
integrity sha512-YaHKEUdsFt59nD8yLvuEGCOZ3/ArirL8GZ/66RkZ8wcD2wbpzOFbzo08Kz4te/Eo3OD5/RdW+1dPaOBgGbrXlA==
|
||||||
|
dependencies:
|
||||||
|
https-proxy-agent "^5.0.0"
|
||||||
|
node-fetch "^2.6.7"
|
||||||
|
progress "^2.0.3"
|
||||||
|
proxy-from-env "^1.1.0"
|
||||||
|
which "^2.0.2"
|
||||||
|
|
||||||
"@sentry/core@7.52.0":
|
"@sentry/core@7.52.0":
|
||||||
version "7.52.0"
|
version "7.52.0"
|
||||||
resolved "https://registry.yarnpkg.com/@sentry/core/-/core-7.52.0.tgz#6c820ca48fe2f06bfd6b290044c96de2375f2ad4"
|
resolved "https://registry.yarnpkg.com/@sentry/core/-/core-7.52.0.tgz#6c820ca48fe2f06bfd6b290044c96de2375f2ad4"
|
||||||
|
@ -3797,6 +3830,15 @@
|
||||||
"@sentry/utils" "7.52.1"
|
"@sentry/utils" "7.52.1"
|
||||||
tslib "^1.9.3"
|
tslib "^1.9.3"
|
||||||
|
|
||||||
|
"@sentry/core@7.69.0":
|
||||||
|
version "7.69.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/@sentry/core/-/core-7.69.0.tgz#ebbe01df573f438f8613107020a4e18eb9adca4d"
|
||||||
|
integrity sha512-V6jvK2lS8bhqZDMFUtvwe2XvNstFQf5A+2LMKCNBOV/NN6eSAAd6THwEpginabjet9dHsNRmMk7WNKvrUfQhZw==
|
||||||
|
dependencies:
|
||||||
|
"@sentry/types" "7.69.0"
|
||||||
|
"@sentry/utils" "7.69.0"
|
||||||
|
tslib "^2.4.1 || ^1.9.3"
|
||||||
|
|
||||||
"@sentry/hub@7.52.0":
|
"@sentry/hub@7.52.0":
|
||||||
version "7.52.0"
|
version "7.52.0"
|
||||||
resolved "https://registry.yarnpkg.com/@sentry/hub/-/hub-7.52.0.tgz#ffc087d58c745d57108862faa0f701b15503dcc2"
|
resolved "https://registry.yarnpkg.com/@sentry/hub/-/hub-7.52.0.tgz#ffc087d58c745d57108862faa0f701b15503dcc2"
|
||||||
|
@ -3807,6 +3849,16 @@
|
||||||
"@sentry/utils" "7.52.0"
|
"@sentry/utils" "7.52.0"
|
||||||
tslib "^1.9.3"
|
tslib "^1.9.3"
|
||||||
|
|
||||||
|
"@sentry/hub@7.69.0":
|
||||||
|
version "7.69.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/@sentry/hub/-/hub-7.69.0.tgz#3ef3b98e1810b05cb4fb37a861bd700ef592a2a9"
|
||||||
|
integrity sha512-71TQ7P5de9+cdW1ETGI9wgi2VNqfyWaM3cnUvheXaSjPRBrr6mhwoaSjo+GGsiwx97Ob9DESZEIhdzcLupzkFA==
|
||||||
|
dependencies:
|
||||||
|
"@sentry/core" "7.69.0"
|
||||||
|
"@sentry/types" "7.69.0"
|
||||||
|
"@sentry/utils" "7.69.0"
|
||||||
|
tslib "^2.4.1 || ^1.9.3"
|
||||||
|
|
||||||
"@sentry/integrations@7.52.0":
|
"@sentry/integrations@7.52.0":
|
||||||
version "7.52.0"
|
version "7.52.0"
|
||||||
resolved "https://registry.yarnpkg.com/@sentry/integrations/-/integrations-7.52.0.tgz#632aa5e54bdfdab910a24057c2072634a2670409"
|
resolved "https://registry.yarnpkg.com/@sentry/integrations/-/integrations-7.52.0.tgz#632aa5e54bdfdab910a24057c2072634a2670409"
|
||||||
|
@ -3827,6 +3879,30 @@
|
||||||
localforage "^1.8.1"
|
localforage "^1.8.1"
|
||||||
tslib "^1.9.3"
|
tslib "^1.9.3"
|
||||||
|
|
||||||
|
"@sentry/integrations@7.69.0":
|
||||||
|
version "7.69.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/@sentry/integrations/-/integrations-7.69.0.tgz#04c0206d9436ec7b79971e3bde5d6e1e9194595f"
|
||||||
|
integrity sha512-FEFtFqXuCo9+L7bENZxFpEAlIODwHl6FyW/DwLfniy9jOXHU7BhP/oICLrFE5J7rh1gNY7N/8VlaiQr3hCnS/g==
|
||||||
|
dependencies:
|
||||||
|
"@sentry/types" "7.69.0"
|
||||||
|
"@sentry/utils" "7.69.0"
|
||||||
|
localforage "^1.8.1"
|
||||||
|
tslib "^2.4.1 || ^1.9.3"
|
||||||
|
|
||||||
|
"@sentry/react-native@5.10.0":
|
||||||
|
version "5.10.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/@sentry/react-native/-/react-native-5.10.0.tgz#b61861276fcb35e69dbe9c4e098ed7c88598f5d9"
|
||||||
|
integrity sha512-YuEZJ3tW5qZlFGFm2FoAZ9vw1fWnjrhMh1IHxo+nUHP3FvVgGkAd/PmSSbgPr2T3YLOIJNiyDdG031Qi7YvtGA==
|
||||||
|
dependencies:
|
||||||
|
"@sentry/browser" "7.69.0"
|
||||||
|
"@sentry/cli" "2.20.7"
|
||||||
|
"@sentry/core" "7.69.0"
|
||||||
|
"@sentry/hub" "7.69.0"
|
||||||
|
"@sentry/integrations" "7.69.0"
|
||||||
|
"@sentry/react" "7.69.0"
|
||||||
|
"@sentry/types" "7.69.0"
|
||||||
|
"@sentry/utils" "7.69.0"
|
||||||
|
|
||||||
"@sentry/react-native@5.5.0":
|
"@sentry/react-native@5.5.0":
|
||||||
version "5.5.0"
|
version "5.5.0"
|
||||||
resolved "https://registry.yarnpkg.com/@sentry/react-native/-/react-native-5.5.0.tgz#b1283f68465b1772ad6059ebba149673cef33f2d"
|
resolved "https://registry.yarnpkg.com/@sentry/react-native/-/react-native-5.5.0.tgz#b1283f68465b1772ad6059ebba149673cef33f2d"
|
||||||
|
@ -3863,6 +3939,17 @@
|
||||||
hoist-non-react-statics "^3.3.2"
|
hoist-non-react-statics "^3.3.2"
|
||||||
tslib "^1.9.3"
|
tslib "^1.9.3"
|
||||||
|
|
||||||
|
"@sentry/react@7.69.0":
|
||||||
|
version "7.69.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/@sentry/react/-/react-7.69.0.tgz#b9931ac590d8dad3390a9a03a516f1b1bd75615e"
|
||||||
|
integrity sha512-J+DciRRVuruf1nMmBOi2VeJkOLGeCb4vTOFmHzWTvRJNByZ0flyo8E/fyROL7+23kBq1YbcVY6IloUlH73hneQ==
|
||||||
|
dependencies:
|
||||||
|
"@sentry/browser" "7.69.0"
|
||||||
|
"@sentry/types" "7.69.0"
|
||||||
|
"@sentry/utils" "7.69.0"
|
||||||
|
hoist-non-react-statics "^3.3.2"
|
||||||
|
tslib "^2.4.1 || ^1.9.3"
|
||||||
|
|
||||||
"@sentry/replay@7.52.0":
|
"@sentry/replay@7.52.0":
|
||||||
version "7.52.0"
|
version "7.52.0"
|
||||||
resolved "https://registry.yarnpkg.com/@sentry/replay/-/replay-7.52.0.tgz#4d78e88282d2c1044ea4b648a68d1b22173e810d"
|
resolved "https://registry.yarnpkg.com/@sentry/replay/-/replay-7.52.0.tgz#4d78e88282d2c1044ea4b648a68d1b22173e810d"
|
||||||
|
@ -3881,6 +3968,15 @@
|
||||||
"@sentry/types" "7.52.1"
|
"@sentry/types" "7.52.1"
|
||||||
"@sentry/utils" "7.52.1"
|
"@sentry/utils" "7.52.1"
|
||||||
|
|
||||||
|
"@sentry/replay@7.69.0":
|
||||||
|
version "7.69.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/@sentry/replay/-/replay-7.69.0.tgz#d727f96292d2b7c25df022fa53764fd39910fcda"
|
||||||
|
integrity sha512-oUqWyBPFUgShdVvgJtV65EQH9pVDmoYVQMOu59JI6FHVeL3ald7R5Mvz6GaNLXsirvvhp0yAkcAd2hc5Xi6hDw==
|
||||||
|
dependencies:
|
||||||
|
"@sentry/core" "7.69.0"
|
||||||
|
"@sentry/types" "7.69.0"
|
||||||
|
"@sentry/utils" "7.69.0"
|
||||||
|
|
||||||
"@sentry/types@7.52.0":
|
"@sentry/types@7.52.0":
|
||||||
version "7.52.0"
|
version "7.52.0"
|
||||||
resolved "https://registry.yarnpkg.com/@sentry/types/-/types-7.52.0.tgz#b7d5372f17355e3991cbe818ad567f3fe277cc6b"
|
resolved "https://registry.yarnpkg.com/@sentry/types/-/types-7.52.0.tgz#b7d5372f17355e3991cbe818ad567f3fe277cc6b"
|
||||||
|
@ -3891,6 +3987,11 @@
|
||||||
resolved "https://registry.yarnpkg.com/@sentry/types/-/types-7.52.1.tgz#bcff6d0462d9b9b7b9ec31c0068fe02d44f25da2"
|
resolved "https://registry.yarnpkg.com/@sentry/types/-/types-7.52.1.tgz#bcff6d0462d9b9b7b9ec31c0068fe02d44f25da2"
|
||||||
integrity sha512-OMbGBPrJsw0iEXwZ2bJUYxewI1IEAU2e1aQGc0O6QW5+6hhCh+8HO8Xl4EymqwejjztuwStkl6G1qhK+Q0/Row==
|
integrity sha512-OMbGBPrJsw0iEXwZ2bJUYxewI1IEAU2e1aQGc0O6QW5+6hhCh+8HO8Xl4EymqwejjztuwStkl6G1qhK+Q0/Row==
|
||||||
|
|
||||||
|
"@sentry/types@7.69.0":
|
||||||
|
version "7.69.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/@sentry/types/-/types-7.69.0.tgz#012b8d90d270a473cc2a5cf58a56870542739292"
|
||||||
|
integrity sha512-zPyCox0mzitzU6SIa1KIbNoJAInYDdUpdiA+PoUmMn2hFMH1llGU/cS7f4w/mAsssTlbtlBi72RMnWUCy578bw==
|
||||||
|
|
||||||
"@sentry/utils@7.52.0":
|
"@sentry/utils@7.52.0":
|
||||||
version "7.52.0"
|
version "7.52.0"
|
||||||
resolved "https://registry.yarnpkg.com/@sentry/utils/-/utils-7.52.0.tgz#cacc36d905036ba7084c14965e964fc44239d7f0"
|
resolved "https://registry.yarnpkg.com/@sentry/utils/-/utils-7.52.0.tgz#cacc36d905036ba7084c14965e964fc44239d7f0"
|
||||||
|
@ -3907,6 +4008,14 @@
|
||||||
"@sentry/types" "7.52.1"
|
"@sentry/types" "7.52.1"
|
||||||
tslib "^1.9.3"
|
tslib "^1.9.3"
|
||||||
|
|
||||||
|
"@sentry/utils@7.69.0":
|
||||||
|
version "7.69.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/@sentry/utils/-/utils-7.69.0.tgz#b7594e4eb2a88b9b25298770b841dd3f81bd2aa4"
|
||||||
|
integrity sha512-4eBixe5Y+0EGVU95R4NxH3jkkjtkE4/CmSZD4In8SCkWGSauogePtq6hyiLsZuP1QHdpPb9Kt0+zYiBb2LouBA==
|
||||||
|
dependencies:
|
||||||
|
"@sentry/types" "7.69.0"
|
||||||
|
tslib "^2.4.1 || ^1.9.3"
|
||||||
|
|
||||||
"@sideway/address@^4.1.3":
|
"@sideway/address@^4.1.3":
|
||||||
version "4.1.4"
|
version "4.1.4"
|
||||||
resolved "https://registry.yarnpkg.com/@sideway/address/-/address-4.1.4.tgz#03dccebc6ea47fdc226f7d3d1ad512955d4783f0"
|
resolved "https://registry.yarnpkg.com/@sideway/address/-/address-4.1.4.tgz#03dccebc6ea47fdc226f7d3d1ad512955d4783f0"
|
||||||
|
@ -6532,6 +6641,11 @@ babel-plugin-transform-react-remove-prop-types@^0.4.24:
|
||||||
resolved "https://registry.yarnpkg.com/babel-plugin-transform-react-remove-prop-types/-/babel-plugin-transform-react-remove-prop-types-0.4.24.tgz#f2edaf9b4c6a5fbe5c1d678bfb531078c1555f3a"
|
resolved "https://registry.yarnpkg.com/babel-plugin-transform-react-remove-prop-types/-/babel-plugin-transform-react-remove-prop-types-0.4.24.tgz#f2edaf9b4c6a5fbe5c1d678bfb531078c1555f3a"
|
||||||
integrity sha512-eqj0hVcJUR57/Ug2zE1Yswsw4LhuqqHhD+8v120T1cl3kjg76QwtyBrdIk4WVwK+lAhBJVYCd/v+4nc4y+8JsA==
|
integrity sha512-eqj0hVcJUR57/Ug2zE1Yswsw4LhuqqHhD+8v120T1cl3kjg76QwtyBrdIk4WVwK+lAhBJVYCd/v+4nc4y+8JsA==
|
||||||
|
|
||||||
|
babel-plugin-transform-remove-console@^6.9.4:
|
||||||
|
version "6.9.4"
|
||||||
|
resolved "https://registry.yarnpkg.com/babel-plugin-transform-remove-console/-/babel-plugin-transform-remove-console-6.9.4.tgz#b980360c067384e24b357a588d807d3c83527780"
|
||||||
|
integrity sha512-88blrUrMX3SPiGkT1GnvVY8E/7A+k6oj3MNvUtTIxJflFzXTw1bHkuJ/y039ouhFMp2prRn5cQGzokViYi1dsg==
|
||||||
|
|
||||||
babel-preset-current-node-syntax@^1.0.0:
|
babel-preset-current-node-syntax@^1.0.0:
|
||||||
version "1.0.1"
|
version "1.0.1"
|
||||||
resolved "https://registry.yarnpkg.com/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.0.1.tgz#b4399239b89b2a011f9ddbe3e4f401fc40cff73b"
|
resolved "https://registry.yarnpkg.com/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.0.1.tgz#b4399239b89b2a011f9ddbe3e4f401fc40cff73b"
|
||||||
|
@ -16704,7 +16818,7 @@ send@0.18.0, send@^0.18.0:
|
||||||
range-parser "~1.2.1"
|
range-parser "~1.2.1"
|
||||||
statuses "2.0.1"
|
statuses "2.0.1"
|
||||||
|
|
||||||
sentry-expo@~7.0.0:
|
sentry-expo@~7.0.1:
|
||||||
version "7.0.1"
|
version "7.0.1"
|
||||||
resolved "https://registry.yarnpkg.com/sentry-expo/-/sentry-expo-7.0.1.tgz#025f0e90ab7f7cba1e00c892fabc027de21bc5bc"
|
resolved "https://registry.yarnpkg.com/sentry-expo/-/sentry-expo-7.0.1.tgz#025f0e90ab7f7cba1e00c892fabc027de21bc5bc"
|
||||||
integrity sha512-8vmOy4R+qM1peQA9EP8rDGUMBhgMU1D5FyuWY9kfNGatmWuvEmlZpVgaXoXaNPIhPgf2TMrvQIlbqLHtTkoeSA==
|
integrity sha512-8vmOy4R+qM1peQA9EP8rDGUMBhgMU1D5FyuWY9kfNGatmWuvEmlZpVgaXoXaNPIhPgf2TMrvQIlbqLHtTkoeSA==
|
||||||
|
@ -17890,7 +18004,7 @@ tslib@^1.8.1, tslib@^1.9.3:
|
||||||
resolved "https://registry.yarnpkg.com/tslib/-/tslib-1.14.1.tgz#cf2d38bdc34a134bcaf1091c41f6619e2f672d00"
|
resolved "https://registry.yarnpkg.com/tslib/-/tslib-1.14.1.tgz#cf2d38bdc34a134bcaf1091c41f6619e2f672d00"
|
||||||
integrity sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==
|
integrity sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==
|
||||||
|
|
||||||
tslib@^2.0.0, tslib@^2.0.1, tslib@^2.0.3, tslib@^2.1.0, tslib@^2.4.0, tslib@^2.4.1:
|
tslib@^2.0.0, tslib@^2.0.1, tslib@^2.0.3, tslib@^2.1.0, tslib@^2.4.0, tslib@^2.4.1, "tslib@^2.4.1 || ^1.9.3":
|
||||||
version "2.6.2"
|
version "2.6.2"
|
||||||
resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.6.2.tgz#703ac29425e7b37cd6fd456e92404d46d1f3e4ae"
|
resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.6.2.tgz#703ac29425e7b37cd6fd456e92404d46d1f3e4ae"
|
||||||
integrity sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==
|
integrity sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==
|
||||||
|
|
Loading…
Reference in New Issue