PWI: Refactor Shell (#1989)
* Vendor createNativeStackNavigator for further tweaks * Completely disable withAuthRequired * Render LoggedOut for protected routes * Move web shell into the navigator * Simplify the logic * Add login modal * Delete withAuthRequired * Reset app state on session change * Move TS suppression
This commit is contained in:
		
							parent
							
								
									4b59a21cac
								
							
						
					
					
						commit
						f2d164ec23
					
				
					 29 changed files with 1627 additions and 1665 deletions
				
			
		|  | @ -12,7 +12,6 @@ import {Button} from '../com/util/forms/Button' | |||
| import * as Toast from '../com/util/Toast' | ||||
| import {usePalette} from 'lib/hooks/usePalette' | ||||
| import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {NativeStackScreenProps} from '@react-navigation/native-stack' | ||||
| import {CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {useAnalytics} from 'lib/analytics/analytics' | ||||
|  | @ -32,125 +31,111 @@ import {ErrorScreen} from '../com/util/error/ErrorScreen' | |||
| import {cleanError} from '#/lib/strings/errors' | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'AppPasswords'> | ||||
| export const AppPasswords = withAuthRequired( | ||||
|   function AppPasswordsImpl({}: Props) { | ||||
|     const pal = usePalette('default') | ||||
|     const setMinimalShellMode = useSetMinimalShellMode() | ||||
|     const {screen} = useAnalytics() | ||||
|     const {isTabletOrDesktop} = useWebMediaQueries() | ||||
|     const {openModal} = useModalControls() | ||||
|     const {data: appPasswords, error} = useAppPasswordsQuery() | ||||
| export function AppPasswords({}: Props) { | ||||
|   const pal = usePalette('default') | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|   const {screen} = useAnalytics() | ||||
|   const {isTabletOrDesktop} = useWebMediaQueries() | ||||
|   const {openModal} = useModalControls() | ||||
|   const {data: appPasswords, error} = useAppPasswordsQuery() | ||||
| 
 | ||||
|     useFocusEffect( | ||||
|       React.useCallback(() => { | ||||
|         screen('AppPasswords') | ||||
|         setMinimalShellMode(false) | ||||
|       }, [screen, setMinimalShellMode]), | ||||
|   useFocusEffect( | ||||
|     React.useCallback(() => { | ||||
|       screen('AppPasswords') | ||||
|       setMinimalShellMode(false) | ||||
|     }, [screen, setMinimalShellMode]), | ||||
|   ) | ||||
| 
 | ||||
|   const onAdd = React.useCallback(async () => { | ||||
|     openModal({name: 'add-app-password'}) | ||||
|   }, [openModal]) | ||||
| 
 | ||||
|   if (error) { | ||||
|     return ( | ||||
|       <CenteredView | ||||
|         style={[ | ||||
|           styles.container, | ||||
|           isTabletOrDesktop && styles.containerDesktop, | ||||
|           pal.view, | ||||
|           pal.border, | ||||
|         ]} | ||||
|         testID="appPasswordsScreen"> | ||||
|         <ErrorScreen | ||||
|           title="Oops!" | ||||
|           message="There was an issue with fetching your app passwords" | ||||
|           details={cleanError(error)} | ||||
|         /> | ||||
|       </CenteredView> | ||||
|     ) | ||||
|   } | ||||
| 
 | ||||
|     const onAdd = React.useCallback(async () => { | ||||
|       openModal({name: 'add-app-password'}) | ||||
|     }, [openModal]) | ||||
| 
 | ||||
|     if (error) { | ||||
|       return ( | ||||
|         <CenteredView | ||||
|   // no app passwords (empty) state
 | ||||
|   if (appPasswords?.length === 0) { | ||||
|     return ( | ||||
|       <CenteredView | ||||
|         style={[ | ||||
|           styles.container, | ||||
|           isTabletOrDesktop && styles.containerDesktop, | ||||
|           pal.view, | ||||
|           pal.border, | ||||
|         ]} | ||||
|         testID="appPasswordsScreen"> | ||||
|         <AppPasswordsHeader /> | ||||
|         <View style={[styles.empty, pal.viewLight]}> | ||||
|           <Text type="lg" style={[pal.text, styles.emptyText]}> | ||||
|             <Trans> | ||||
|               You have not created any app passwords yet. You can create one by | ||||
|               pressing the button below. | ||||
|             </Trans> | ||||
|           </Text> | ||||
|         </View> | ||||
|         {!isTabletOrDesktop && <View style={styles.flex1} />} | ||||
|         <View | ||||
|           style={[ | ||||
|             styles.container, | ||||
|             isTabletOrDesktop && styles.containerDesktop, | ||||
|             pal.view, | ||||
|             pal.border, | ||||
|           ]} | ||||
|           testID="appPasswordsScreen"> | ||||
|           <ErrorScreen | ||||
|             title="Oops!" | ||||
|             message="There was an issue with fetching your app passwords" | ||||
|             details={cleanError(error)} | ||||
|             styles.btnContainer, | ||||
|             isTabletOrDesktop && styles.btnContainerDesktop, | ||||
|           ]}> | ||||
|           <Button | ||||
|             testID="appPasswordBtn" | ||||
|             type="primary" | ||||
|             label="Add App Password" | ||||
|             style={styles.btn} | ||||
|             labelStyle={styles.btnLabel} | ||||
|             onPress={onAdd} | ||||
|           /> | ||||
|         </CenteredView> | ||||
|       ) | ||||
|     } | ||||
|         </View> | ||||
|       </CenteredView> | ||||
|     ) | ||||
|   } | ||||
| 
 | ||||
|     // no app passwords (empty) state
 | ||||
|     if (appPasswords?.length === 0) { | ||||
|       return ( | ||||
|         <CenteredView | ||||
|   if (appPasswords?.length) { | ||||
|     // has app passwords
 | ||||
|     return ( | ||||
|       <CenteredView | ||||
|         style={[ | ||||
|           styles.container, | ||||
|           isTabletOrDesktop && styles.containerDesktop, | ||||
|           pal.view, | ||||
|           pal.border, | ||||
|         ]} | ||||
|         testID="appPasswordsScreen"> | ||||
|         <AppPasswordsHeader /> | ||||
|         <ScrollView | ||||
|           style={[ | ||||
|             styles.container, | ||||
|             isTabletOrDesktop && styles.containerDesktop, | ||||
|             pal.view, | ||||
|             styles.scrollContainer, | ||||
|             pal.border, | ||||
|           ]} | ||||
|           testID="appPasswordsScreen"> | ||||
|           <AppPasswordsHeader /> | ||||
|           <View style={[styles.empty, pal.viewLight]}> | ||||
|             <Text type="lg" style={[pal.text, styles.emptyText]}> | ||||
|               <Trans> | ||||
|                 You have not created any app passwords yet. You can create one | ||||
|                 by pressing the button below. | ||||
|               </Trans> | ||||
|             </Text> | ||||
|           </View> | ||||
|           {!isTabletOrDesktop && <View style={styles.flex1} />} | ||||
|           <View | ||||
|             style={[ | ||||
|               styles.btnContainer, | ||||
|               isTabletOrDesktop && styles.btnContainerDesktop, | ||||
|             ]}> | ||||
|             <Button | ||||
|               testID="appPasswordBtn" | ||||
|               type="primary" | ||||
|               label="Add App Password" | ||||
|               style={styles.btn} | ||||
|               labelStyle={styles.btnLabel} | ||||
|               onPress={onAdd} | ||||
|             !isTabletOrDesktop && styles.flex1, | ||||
|           ]}> | ||||
|           {appPasswords.map((password, i) => ( | ||||
|             <AppPassword | ||||
|               key={password.name} | ||||
|               testID={`appPassword-${i}`} | ||||
|               name={password.name} | ||||
|               createdAt={password.createdAt} | ||||
|             /> | ||||
|           </View> | ||||
|         </CenteredView> | ||||
|       ) | ||||
|     } | ||||
| 
 | ||||
|     if (appPasswords?.length) { | ||||
|       // has app passwords
 | ||||
|       return ( | ||||
|         <CenteredView | ||||
|           style={[ | ||||
|             styles.container, | ||||
|             isTabletOrDesktop && styles.containerDesktop, | ||||
|             pal.view, | ||||
|             pal.border, | ||||
|           ]} | ||||
|           testID="appPasswordsScreen"> | ||||
|           <AppPasswordsHeader /> | ||||
|           <ScrollView | ||||
|             style={[ | ||||
|               styles.scrollContainer, | ||||
|               pal.border, | ||||
|               !isTabletOrDesktop && styles.flex1, | ||||
|             ]}> | ||||
|             {appPasswords.map((password, i) => ( | ||||
|               <AppPassword | ||||
|                 key={password.name} | ||||
|                 testID={`appPassword-${i}`} | ||||
|                 name={password.name} | ||||
|                 createdAt={password.createdAt} | ||||
|               /> | ||||
|             ))} | ||||
|             {isTabletOrDesktop && ( | ||||
|               <View style={[styles.btnContainer, styles.btnContainerDesktop]}> | ||||
|                 <Button | ||||
|                   testID="appPasswordBtn" | ||||
|                   type="primary" | ||||
|                   label="Add App Password" | ||||
|                   style={styles.btn} | ||||
|                   labelStyle={styles.btnLabel} | ||||
|                   onPress={onAdd} | ||||
|                 /> | ||||
|               </View> | ||||
|             )} | ||||
|           </ScrollView> | ||||
|           {!isTabletOrDesktop && ( | ||||
|             <View style={styles.btnContainer}> | ||||
|           ))} | ||||
|           {isTabletOrDesktop && ( | ||||
|             <View style={[styles.btnContainer, styles.btnContainerDesktop]}> | ||||
|               <Button | ||||
|                 testID="appPasswordBtn" | ||||
|                 type="primary" | ||||
|  | @ -161,24 +146,36 @@ export const AppPasswords = withAuthRequired( | |||
|               /> | ||||
|             </View> | ||||
|           )} | ||||
|         </CenteredView> | ||||
|       ) | ||||
|     } | ||||
| 
 | ||||
|     return ( | ||||
|       <CenteredView | ||||
|         style={[ | ||||
|           styles.container, | ||||
|           isTabletOrDesktop && styles.containerDesktop, | ||||
|           pal.view, | ||||
|           pal.border, | ||||
|         ]} | ||||
|         testID="appPasswordsScreen"> | ||||
|         <ActivityIndicator /> | ||||
|         </ScrollView> | ||||
|         {!isTabletOrDesktop && ( | ||||
|           <View style={styles.btnContainer}> | ||||
|             <Button | ||||
|               testID="appPasswordBtn" | ||||
|               type="primary" | ||||
|               label="Add App Password" | ||||
|               style={styles.btn} | ||||
|               labelStyle={styles.btnLabel} | ||||
|               onPress={onAdd} | ||||
|             /> | ||||
|           </View> | ||||
|         )} | ||||
|       </CenteredView> | ||||
|     ) | ||||
|   }, | ||||
| ) | ||||
|   } | ||||
| 
 | ||||
|   return ( | ||||
|     <CenteredView | ||||
|       style={[ | ||||
|         styles.container, | ||||
|         isTabletOrDesktop && styles.containerDesktop, | ||||
|         pal.view, | ||||
|         pal.border, | ||||
|       ]} | ||||
|       testID="appPasswordsScreen"> | ||||
|       <ActivityIndicator /> | ||||
|     </CenteredView> | ||||
|   ) | ||||
| } | ||||
| 
 | ||||
| function AppPasswordsHeader() { | ||||
|   const {isTabletOrDesktop} = useWebMediaQueries() | ||||
|  |  | |||
|  | @ -2,7 +2,6 @@ import React from 'react' | |||
| import {ActivityIndicator, StyleSheet, View, RefreshControl} from 'react-native' | ||||
| import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome' | ||||
| import {FontAwesomeIconStyle} from '@fortawesome/react-native-fontawesome' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {ViewHeader} from 'view/com/util/ViewHeader' | ||||
| import {FAB} from 'view/com/util/fab/FAB' | ||||
| import {Link} from 'view/com/util/Link' | ||||
|  | @ -88,437 +87,432 @@ type FlatlistSlice = | |||
|       key: string | ||||
|     } | ||||
| 
 | ||||
| export const FeedsScreen = withAuthRequired( | ||||
|   function FeedsScreenImpl(_props: Props) { | ||||
|     const pal = usePalette('default') | ||||
|     const {openComposer} = useComposerControls() | ||||
|     const {isMobile, isTabletOrDesktop} = useWebMediaQueries() | ||||
|     const [query, setQuery] = React.useState('') | ||||
|     const [isPTR, setIsPTR] = React.useState(false) | ||||
|     const { | ||||
|       data: preferences, | ||||
|       isLoading: isPreferencesLoading, | ||||
|       error: preferencesError, | ||||
|     } = usePreferencesQuery() | ||||
|     const { | ||||
|       data: popularFeeds, | ||||
|       isFetching: isPopularFeedsFetching, | ||||
|       error: popularFeedsError, | ||||
|       refetch: refetchPopularFeeds, | ||||
|       fetchNextPage: fetchNextPopularFeedsPage, | ||||
|       isFetchingNextPage: isPopularFeedsFetchingNextPage, | ||||
|       hasNextPage: hasNextPopularFeedsPage, | ||||
|     } = useGetPopularFeedsQuery() | ||||
|     const {_} = useLingui() | ||||
|     const setMinimalShellMode = useSetMinimalShellMode() | ||||
|     const { | ||||
|       data: searchResults, | ||||
|       mutate: search, | ||||
|       reset: resetSearch, | ||||
|       isPending: isSearchPending, | ||||
|       error: searchError, | ||||
|     } = useSearchPopularFeedsMutation() | ||||
|     const {hasSession} = useSession() | ||||
| export function FeedsScreen(_props: Props) { | ||||
|   const pal = usePalette('default') | ||||
|   const {openComposer} = useComposerControls() | ||||
|   const {isMobile, isTabletOrDesktop} = useWebMediaQueries() | ||||
|   const [query, setQuery] = React.useState('') | ||||
|   const [isPTR, setIsPTR] = React.useState(false) | ||||
|   const { | ||||
|     data: preferences, | ||||
|     isLoading: isPreferencesLoading, | ||||
|     error: preferencesError, | ||||
|   } = usePreferencesQuery() | ||||
|   const { | ||||
|     data: popularFeeds, | ||||
|     isFetching: isPopularFeedsFetching, | ||||
|     error: popularFeedsError, | ||||
|     refetch: refetchPopularFeeds, | ||||
|     fetchNextPage: fetchNextPopularFeedsPage, | ||||
|     isFetchingNextPage: isPopularFeedsFetchingNextPage, | ||||
|     hasNextPage: hasNextPopularFeedsPage, | ||||
|   } = useGetPopularFeedsQuery() | ||||
|   const {_} = useLingui() | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|   const { | ||||
|     data: searchResults, | ||||
|     mutate: search, | ||||
|     reset: resetSearch, | ||||
|     isPending: isSearchPending, | ||||
|     error: searchError, | ||||
|   } = useSearchPopularFeedsMutation() | ||||
|   const {hasSession} = useSession() | ||||
| 
 | ||||
|     /** | ||||
|      * A search query is present. We may not have search results yet. | ||||
|      */ | ||||
|     const isUserSearching = query.length > 1 | ||||
|     const debouncedSearch = React.useMemo( | ||||
|       () => debounce(q => search(q), 500), // debounce for 500ms
 | ||||
|       [search], | ||||
|   /** | ||||
|    * A search query is present. We may not have search results yet. | ||||
|    */ | ||||
|   const isUserSearching = query.length > 1 | ||||
|   const debouncedSearch = React.useMemo( | ||||
|     () => debounce(q => search(q), 500), // debounce for 500ms
 | ||||
|     [search], | ||||
|   ) | ||||
|   const onPressCompose = React.useCallback(() => { | ||||
|     openComposer({}) | ||||
|   }, [openComposer]) | ||||
|   const onChangeQuery = React.useCallback( | ||||
|     (text: string) => { | ||||
|       setQuery(text) | ||||
|       if (text.length > 1) { | ||||
|         debouncedSearch(text) | ||||
|       } else { | ||||
|         refetchPopularFeeds() | ||||
|         resetSearch() | ||||
|       } | ||||
|     }, | ||||
|     [setQuery, refetchPopularFeeds, debouncedSearch, resetSearch], | ||||
|   ) | ||||
|   const onPressCancelSearch = React.useCallback(() => { | ||||
|     setQuery('') | ||||
|     refetchPopularFeeds() | ||||
|     resetSearch() | ||||
|   }, [refetchPopularFeeds, setQuery, resetSearch]) | ||||
|   const onSubmitQuery = React.useCallback(() => { | ||||
|     debouncedSearch(query) | ||||
|   }, [query, debouncedSearch]) | ||||
|   const onPullToRefresh = React.useCallback(async () => { | ||||
|     setIsPTR(true) | ||||
|     await refetchPopularFeeds() | ||||
|     setIsPTR(false) | ||||
|   }, [setIsPTR, refetchPopularFeeds]) | ||||
|   const onEndReached = React.useCallback(() => { | ||||
|     if ( | ||||
|       isPopularFeedsFetching || | ||||
|       isUserSearching || | ||||
|       !hasNextPopularFeedsPage || | ||||
|       popularFeedsError | ||||
|     ) | ||||
|     const onPressCompose = React.useCallback(() => { | ||||
|       openComposer({}) | ||||
|     }, [openComposer]) | ||||
|     const onChangeQuery = React.useCallback( | ||||
|       (text: string) => { | ||||
|         setQuery(text) | ||||
|         if (text.length > 1) { | ||||
|           debouncedSearch(text) | ||||
|         } else { | ||||
|           refetchPopularFeeds() | ||||
|           resetSearch() | ||||
|         } | ||||
|       }, | ||||
|       [setQuery, refetchPopularFeeds, debouncedSearch, resetSearch], | ||||
|     ) | ||||
|     const onPressCancelSearch = React.useCallback(() => { | ||||
|       setQuery('') | ||||
|       refetchPopularFeeds() | ||||
|       resetSearch() | ||||
|     }, [refetchPopularFeeds, setQuery, resetSearch]) | ||||
|     const onSubmitQuery = React.useCallback(() => { | ||||
|       debouncedSearch(query) | ||||
|     }, [query, debouncedSearch]) | ||||
|     const onPullToRefresh = React.useCallback(async () => { | ||||
|       setIsPTR(true) | ||||
|       await refetchPopularFeeds() | ||||
|       setIsPTR(false) | ||||
|     }, [setIsPTR, refetchPopularFeeds]) | ||||
|     const onEndReached = React.useCallback(() => { | ||||
|       if ( | ||||
|         isPopularFeedsFetching || | ||||
|         isUserSearching || | ||||
|         !hasNextPopularFeedsPage || | ||||
|         popularFeedsError | ||||
|       ) | ||||
|         return | ||||
|       fetchNextPopularFeedsPage() | ||||
|     }, [ | ||||
|       isPopularFeedsFetching, | ||||
|       isUserSearching, | ||||
|       popularFeedsError, | ||||
|       hasNextPopularFeedsPage, | ||||
|       fetchNextPopularFeedsPage, | ||||
|     ]) | ||||
|       return | ||||
|     fetchNextPopularFeedsPage() | ||||
|   }, [ | ||||
|     isPopularFeedsFetching, | ||||
|     isUserSearching, | ||||
|     popularFeedsError, | ||||
|     hasNextPopularFeedsPage, | ||||
|     fetchNextPopularFeedsPage, | ||||
|   ]) | ||||
| 
 | ||||
|     useFocusEffect( | ||||
|       React.useCallback(() => { | ||||
|         setMinimalShellMode(false) | ||||
|       }, [setMinimalShellMode]), | ||||
|     ) | ||||
|   useFocusEffect( | ||||
|     React.useCallback(() => { | ||||
|       setMinimalShellMode(false) | ||||
|     }, [setMinimalShellMode]), | ||||
|   ) | ||||
| 
 | ||||
|     const items = React.useMemo(() => { | ||||
|       let slices: FlatlistSlice[] = [] | ||||
|   const items = React.useMemo(() => { | ||||
|     let slices: FlatlistSlice[] = [] | ||||
| 
 | ||||
|       if (hasSession) { | ||||
|     if (hasSession) { | ||||
|       slices.push({ | ||||
|         key: 'savedFeedsHeader', | ||||
|         type: 'savedFeedsHeader', | ||||
|       }) | ||||
| 
 | ||||
|       if (preferencesError) { | ||||
|         slices.push({ | ||||
|           key: 'savedFeedsHeader', | ||||
|           type: 'savedFeedsHeader', | ||||
|           key: 'savedFeedsError', | ||||
|           type: 'error', | ||||
|           error: cleanError(preferencesError.toString()), | ||||
|         }) | ||||
| 
 | ||||
|         if (preferencesError) { | ||||
|       } else { | ||||
|         if (isPreferencesLoading || !preferences?.feeds?.saved) { | ||||
|           slices.push({ | ||||
|             key: 'savedFeedsError', | ||||
|             type: 'error', | ||||
|             error: cleanError(preferencesError.toString()), | ||||
|             key: 'savedFeedsLoading', | ||||
|             type: 'savedFeedsLoading', | ||||
|             // pendingItems: this.rootStore.preferences.savedFeeds.length || 3,
 | ||||
|           }) | ||||
|         } else { | ||||
|           if (isPreferencesLoading || !preferences?.feeds?.saved) { | ||||
|           if (preferences?.feeds?.saved.length === 0) { | ||||
|             slices.push({ | ||||
|               key: 'savedFeedsLoading', | ||||
|               type: 'savedFeedsLoading', | ||||
|               // pendingItems: this.rootStore.preferences.savedFeeds.length || 3,
 | ||||
|               key: 'savedFeedNoResults', | ||||
|               type: 'savedFeedNoResults', | ||||
|             }) | ||||
|           } else { | ||||
|             if (preferences?.feeds?.saved.length === 0) { | ||||
|               slices.push({ | ||||
|                 key: 'savedFeedNoResults', | ||||
|                 type: 'savedFeedNoResults', | ||||
|               }) | ||||
|             } else { | ||||
|               const {saved, pinned} = preferences.feeds | ||||
|             const {saved, pinned} = preferences.feeds | ||||
| 
 | ||||
|               slices = slices.concat( | ||||
|                 pinned.map(uri => ({ | ||||
|             slices = slices.concat( | ||||
|               pinned.map(uri => ({ | ||||
|                 key: `savedFeed:${uri}`, | ||||
|                 type: 'savedFeed', | ||||
|                 feedUri: uri, | ||||
|               })), | ||||
|             ) | ||||
| 
 | ||||
|             slices = slices.concat( | ||||
|               saved | ||||
|                 .filter(uri => !pinned.includes(uri)) | ||||
|                 .map(uri => ({ | ||||
|                   key: `savedFeed:${uri}`, | ||||
|                   type: 'savedFeed', | ||||
|                   feedUri: uri, | ||||
|                 })), | ||||
|               ) | ||||
|             ) | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     slices.push({ | ||||
|       key: 'popularFeedsHeader', | ||||
|       type: 'popularFeedsHeader', | ||||
|     }) | ||||
| 
 | ||||
|     if (popularFeedsError || searchError) { | ||||
|       slices.push({ | ||||
|         key: 'popularFeedsError', | ||||
|         type: 'error', | ||||
|         error: cleanError( | ||||
|           popularFeedsError?.toString() ?? searchError?.toString() ?? '', | ||||
|         ), | ||||
|       }) | ||||
|     } else { | ||||
|       if (isUserSearching) { | ||||
|         if (isSearchPending || !searchResults) { | ||||
|           slices.push({ | ||||
|             key: 'popularFeedsLoading', | ||||
|             type: 'popularFeedsLoading', | ||||
|           }) | ||||
|         } else { | ||||
|           if (!searchResults || searchResults?.length === 0) { | ||||
|             slices.push({ | ||||
|               key: 'popularFeedsNoResults', | ||||
|               type: 'popularFeedsNoResults', | ||||
|             }) | ||||
|           } else { | ||||
|             slices = slices.concat( | ||||
|               searchResults.map(feed => ({ | ||||
|                 key: `popularFeed:${feed.uri}`, | ||||
|                 type: 'popularFeed', | ||||
|                 feedUri: feed.uri, | ||||
|               })), | ||||
|             ) | ||||
|           } | ||||
|         } | ||||
|       } else { | ||||
|         if (isPopularFeedsFetching && !popularFeeds?.pages) { | ||||
|           slices.push({ | ||||
|             key: 'popularFeedsLoading', | ||||
|             type: 'popularFeedsLoading', | ||||
|           }) | ||||
|         } else { | ||||
|           if ( | ||||
|             !popularFeeds?.pages || | ||||
|             popularFeeds?.pages[0]?.feeds?.length === 0 | ||||
|           ) { | ||||
|             slices.push({ | ||||
|               key: 'popularFeedsNoResults', | ||||
|               type: 'popularFeedsNoResults', | ||||
|             }) | ||||
|           } else { | ||||
|             for (const page of popularFeeds.pages || []) { | ||||
|               slices = slices.concat( | ||||
|                 saved | ||||
|                   .filter(uri => !pinned.includes(uri)) | ||||
|                   .map(uri => ({ | ||||
|                     key: `savedFeed:${uri}`, | ||||
|                     type: 'savedFeed', | ||||
|                     feedUri: uri, | ||||
|                 page.feeds | ||||
|                   .filter(feed => !preferences?.feeds?.saved.includes(feed.uri)) | ||||
|                   .map(feed => ({ | ||||
|                     key: `popularFeed:${feed.uri}`, | ||||
|                     type: 'popularFeed', | ||||
|                     feedUri: feed.uri, | ||||
|                   })), | ||||
|               ) | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|       } | ||||
| 
 | ||||
|       slices.push({ | ||||
|         key: 'popularFeedsHeader', | ||||
|         type: 'popularFeedsHeader', | ||||
|       }) | ||||
| 
 | ||||
|       if (popularFeedsError || searchError) { | ||||
|         slices.push({ | ||||
|           key: 'popularFeedsError', | ||||
|           type: 'error', | ||||
|           error: cleanError( | ||||
|             popularFeedsError?.toString() ?? searchError?.toString() ?? '', | ||||
|           ), | ||||
|         }) | ||||
|       } else { | ||||
|         if (isUserSearching) { | ||||
|           if (isSearchPending || !searchResults) { | ||||
|             slices.push({ | ||||
|               key: 'popularFeedsLoading', | ||||
|               type: 'popularFeedsLoading', | ||||
|             }) | ||||
|           } else { | ||||
|             if (!searchResults || searchResults?.length === 0) { | ||||
|             if (isPopularFeedsFetchingNextPage) { | ||||
|               slices.push({ | ||||
|                 key: 'popularFeedsNoResults', | ||||
|                 type: 'popularFeedsNoResults', | ||||
|                 key: 'popularFeedsLoadingMore', | ||||
|                 type: 'popularFeedsLoadingMore', | ||||
|               }) | ||||
|             } else { | ||||
|               slices = slices.concat( | ||||
|                 searchResults.map(feed => ({ | ||||
|                   key: `popularFeed:${feed.uri}`, | ||||
|                   type: 'popularFeed', | ||||
|                   feedUri: feed.uri, | ||||
|                 })), | ||||
|               ) | ||||
|             } | ||||
|           } | ||||
|         } else { | ||||
|           if (isPopularFeedsFetching && !popularFeeds?.pages) { | ||||
|             slices.push({ | ||||
|               key: 'popularFeedsLoading', | ||||
|               type: 'popularFeedsLoading', | ||||
|             }) | ||||
|           } else { | ||||
|             if ( | ||||
|               !popularFeeds?.pages || | ||||
|               popularFeeds?.pages[0]?.feeds?.length === 0 | ||||
|             ) { | ||||
|               slices.push({ | ||||
|                 key: 'popularFeedsNoResults', | ||||
|                 type: 'popularFeedsNoResults', | ||||
|               }) | ||||
|             } else { | ||||
|               for (const page of popularFeeds.pages || []) { | ||||
|                 slices = slices.concat( | ||||
|                   page.feeds | ||||
|                     .filter( | ||||
|                       feed => !preferences?.feeds?.saved.includes(feed.uri), | ||||
|                     ) | ||||
|                     .map(feed => ({ | ||||
|                       key: `popularFeed:${feed.uri}`, | ||||
|                       type: 'popularFeed', | ||||
|                       feedUri: feed.uri, | ||||
|                     })), | ||||
|                 ) | ||||
|               } | ||||
| 
 | ||||
|               if (isPopularFeedsFetchingNextPage) { | ||||
|                 slices.push({ | ||||
|                   key: 'popularFeedsLoadingMore', | ||||
|                   type: 'popularFeedsLoadingMore', | ||||
|                 }) | ||||
|               } | ||||
|             } | ||||
|           } | ||||
|         } | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|       return slices | ||||
|     }, [ | ||||
|       hasSession, | ||||
|       preferences, | ||||
|       isPreferencesLoading, | ||||
|       preferencesError, | ||||
|       popularFeeds, | ||||
|       isPopularFeedsFetching, | ||||
|       popularFeedsError, | ||||
|       isPopularFeedsFetchingNextPage, | ||||
|       searchResults, | ||||
|       isSearchPending, | ||||
|       searchError, | ||||
|       isUserSearching, | ||||
|     ]) | ||||
| 
 | ||||
|     const renderHeaderBtn = React.useCallback(() => { | ||||
|       return ( | ||||
|         <Link | ||||
|           href="/settings/saved-feeds" | ||||
|           hitSlop={10} | ||||
|           accessibilityRole="button" | ||||
|           accessibilityLabel={_(msg`Edit Saved Feeds`)} | ||||
|           accessibilityHint="Opens screen to edit Saved Feeds"> | ||||
|           <CogIcon size={22} strokeWidth={2} style={pal.textLight} /> | ||||
|         </Link> | ||||
|       ) | ||||
|     }, [pal, _]) | ||||
| 
 | ||||
|     const renderItem = React.useCallback( | ||||
|       ({item}: {item: FlatlistSlice}) => { | ||||
|         if (item.type === 'error') { | ||||
|           return <ErrorMessage message={item.error} /> | ||||
|         } else if ( | ||||
|           item.type === 'popularFeedsLoadingMore' || | ||||
|           item.type === 'savedFeedsLoading' | ||||
|         ) { | ||||
|           return ( | ||||
|             <View style={s.p10}> | ||||
|               <ActivityIndicator /> | ||||
|             </View> | ||||
|           ) | ||||
|         } else if (item.type === 'savedFeedsHeader') { | ||||
|           if (!isMobile) { | ||||
|             return ( | ||||
|               <View | ||||
|                 style={[ | ||||
|                   pal.view, | ||||
|                   styles.header, | ||||
|                   pal.border, | ||||
|                   { | ||||
|                     borderBottomWidth: 1, | ||||
|                   }, | ||||
|                 ]}> | ||||
|                 <Text type="title-lg" style={[pal.text, s.bold]}> | ||||
|                   <Trans>My Feeds</Trans> | ||||
|                 </Text> | ||||
|                 <Link | ||||
|                   href="/settings/saved-feeds" | ||||
|                   accessibilityLabel={_(msg`Edit My Feeds`)} | ||||
|                   accessibilityHint=""> | ||||
|                   <CogIcon strokeWidth={1.5} style={pal.icon} size={28} /> | ||||
|                 </Link> | ||||
|               </View> | ||||
|             ) | ||||
|           } | ||||
|           return <View /> | ||||
|         } else if (item.type === 'savedFeedNoResults') { | ||||
|           return ( | ||||
|             <View | ||||
|               style={{ | ||||
|                 paddingHorizontal: 16, | ||||
|                 paddingTop: 10, | ||||
|               }}> | ||||
|               <Text type="lg" style={pal.textLight}> | ||||
|                 <Trans>You don't have any saved feeds!</Trans> | ||||
|               </Text> | ||||
|             </View> | ||||
|           ) | ||||
|         } else if (item.type === 'savedFeed') { | ||||
|           return <SavedFeed feedUri={item.feedUri} /> | ||||
|         } else if (item.type === 'popularFeedsHeader') { | ||||
|           return ( | ||||
|             <> | ||||
|               <View | ||||
|                 style={[ | ||||
|                   pal.view, | ||||
|                   styles.header, | ||||
|                   { | ||||
|                     // This is first in the flatlist without a session -esb
 | ||||
|                     marginTop: hasSession ? 16 : 0, | ||||
|                     paddingLeft: isMobile ? 12 : undefined, | ||||
|                     paddingRight: 10, | ||||
|                     paddingBottom: isMobile ? 6 : undefined, | ||||
|                   }, | ||||
|                 ]}> | ||||
|                 <Text type="title-lg" style={[pal.text, s.bold]}> | ||||
|                   <Trans>Discover new feeds</Trans> | ||||
|                 </Text> | ||||
| 
 | ||||
|                 {!isMobile && ( | ||||
|                   <SearchInput | ||||
|                     query={query} | ||||
|                     onChangeQuery={onChangeQuery} | ||||
|                     onPressCancelSearch={onPressCancelSearch} | ||||
|                     onSubmitQuery={onSubmitQuery} | ||||
|                     style={{flex: 1, maxWidth: 250}} | ||||
|                   /> | ||||
|                 )} | ||||
|               </View> | ||||
| 
 | ||||
|               {isMobile && ( | ||||
|                 <View style={{paddingHorizontal: 8, paddingBottom: 10}}> | ||||
|                   <SearchInput | ||||
|                     query={query} | ||||
|                     onChangeQuery={onChangeQuery} | ||||
|                     onPressCancelSearch={onPressCancelSearch} | ||||
|                     onSubmitQuery={onSubmitQuery} | ||||
|                   /> | ||||
|                 </View> | ||||
|               )} | ||||
|             </> | ||||
|           ) | ||||
|         } else if (item.type === 'popularFeedsLoading') { | ||||
|           return <FeedFeedLoadingPlaceholder /> | ||||
|         } else if (item.type === 'popularFeed') { | ||||
|           return ( | ||||
|             <FeedSourceCard | ||||
|               feedUri={item.feedUri} | ||||
|               showSaveBtn={hasSession} | ||||
|               showDescription | ||||
|               showLikes | ||||
|             /> | ||||
|           ) | ||||
|         } else if (item.type === 'popularFeedsNoResults') { | ||||
|           return ( | ||||
|             <View | ||||
|               style={{ | ||||
|                 paddingHorizontal: 16, | ||||
|                 paddingTop: 10, | ||||
|                 paddingBottom: '150%', | ||||
|               }}> | ||||
|               <Text type="lg" style={pal.textLight}> | ||||
|                 <Trans>No results found for "{query}"</Trans> | ||||
|               </Text> | ||||
|             </View> | ||||
|           ) | ||||
|         } | ||||
|         return null | ||||
|       }, | ||||
|       [ | ||||
|         _, | ||||
|         hasSession, | ||||
|         isMobile, | ||||
|         pal, | ||||
|         query, | ||||
|         onChangeQuery, | ||||
|         onPressCancelSearch, | ||||
|         onSubmitQuery, | ||||
|       ], | ||||
|     ) | ||||
|     return slices | ||||
|   }, [ | ||||
|     hasSession, | ||||
|     preferences, | ||||
|     isPreferencesLoading, | ||||
|     preferencesError, | ||||
|     popularFeeds, | ||||
|     isPopularFeedsFetching, | ||||
|     popularFeedsError, | ||||
|     isPopularFeedsFetchingNextPage, | ||||
|     searchResults, | ||||
|     isSearchPending, | ||||
|     searchError, | ||||
|     isUserSearching, | ||||
|   ]) | ||||
| 
 | ||||
|   const renderHeaderBtn = React.useCallback(() => { | ||||
|     return ( | ||||
|       <View style={[pal.view, styles.container]}> | ||||
|         {isMobile && ( | ||||
|           <ViewHeader | ||||
|             title={_(msg`Feeds`)} | ||||
|             canGoBack={false} | ||||
|             renderButton={renderHeaderBtn} | ||||
|             showBorder | ||||
|           /> | ||||
|         )} | ||||
| 
 | ||||
|         {preferences ? <View /> : <ActivityIndicator />} | ||||
| 
 | ||||
|         <FlatList | ||||
|           style={[!isTabletOrDesktop && s.flex1, styles.list]} | ||||
|           data={items} | ||||
|           keyExtractor={item => item.key} | ||||
|           contentContainerStyle={styles.contentContainer} | ||||
|           renderItem={renderItem} | ||||
|           refreshControl={ | ||||
|             <RefreshControl | ||||
|               refreshing={isPTR} | ||||
|               onRefresh={isUserSearching ? undefined : onPullToRefresh} | ||||
|               tintColor={pal.colors.text} | ||||
|               titleColor={pal.colors.text} | ||||
|             /> | ||||
|           } | ||||
|           initialNumToRender={10} | ||||
|           onEndReached={onEndReached} | ||||
|           // @ts-ignore our .web version only -prf
 | ||||
|           desktopFixedHeight | ||||
|         /> | ||||
| 
 | ||||
|         {hasSession && ( | ||||
|           <FAB | ||||
|             testID="composeFAB" | ||||
|             onPress={onPressCompose} | ||||
|             icon={<ComposeIcon2 strokeWidth={1.5} size={29} style={s.white} />} | ||||
|             accessibilityRole="button" | ||||
|             accessibilityLabel={_(msg`New post`)} | ||||
|             accessibilityHint="" | ||||
|           /> | ||||
|         )} | ||||
|       </View> | ||||
|       <Link | ||||
|         href="/settings/saved-feeds" | ||||
|         hitSlop={10} | ||||
|         accessibilityRole="button" | ||||
|         accessibilityLabel={_(msg`Edit Saved Feeds`)} | ||||
|         accessibilityHint="Opens screen to edit Saved Feeds"> | ||||
|         <CogIcon size={22} strokeWidth={2} style={pal.textLight} /> | ||||
|       </Link> | ||||
|     ) | ||||
|   }, | ||||
|   {isPublic: true}, | ||||
| ) | ||||
|   }, [pal, _]) | ||||
| 
 | ||||
|   const renderItem = React.useCallback( | ||||
|     ({item}: {item: FlatlistSlice}) => { | ||||
|       if (item.type === 'error') { | ||||
|         return <ErrorMessage message={item.error} /> | ||||
|       } else if ( | ||||
|         item.type === 'popularFeedsLoadingMore' || | ||||
|         item.type === 'savedFeedsLoading' | ||||
|       ) { | ||||
|         return ( | ||||
|           <View style={s.p10}> | ||||
|             <ActivityIndicator /> | ||||
|           </View> | ||||
|         ) | ||||
|       } else if (item.type === 'savedFeedsHeader') { | ||||
|         if (!isMobile) { | ||||
|           return ( | ||||
|             <View | ||||
|               style={[ | ||||
|                 pal.view, | ||||
|                 styles.header, | ||||
|                 pal.border, | ||||
|                 { | ||||
|                   borderBottomWidth: 1, | ||||
|                 }, | ||||
|               ]}> | ||||
|               <Text type="title-lg" style={[pal.text, s.bold]}> | ||||
|                 <Trans>My Feeds</Trans> | ||||
|               </Text> | ||||
|               <Link | ||||
|                 href="/settings/saved-feeds" | ||||
|                 accessibilityLabel={_(msg`Edit My Feeds`)} | ||||
|                 accessibilityHint=""> | ||||
|                 <CogIcon strokeWidth={1.5} style={pal.icon} size={28} /> | ||||
|               </Link> | ||||
|             </View> | ||||
|           ) | ||||
|         } | ||||
|         return <View /> | ||||
|       } else if (item.type === 'savedFeedNoResults') { | ||||
|         return ( | ||||
|           <View | ||||
|             style={{ | ||||
|               paddingHorizontal: 16, | ||||
|               paddingTop: 10, | ||||
|             }}> | ||||
|             <Text type="lg" style={pal.textLight}> | ||||
|               <Trans>You don't have any saved feeds!</Trans> | ||||
|             </Text> | ||||
|           </View> | ||||
|         ) | ||||
|       } else if (item.type === 'savedFeed') { | ||||
|         return <SavedFeed feedUri={item.feedUri} /> | ||||
|       } else if (item.type === 'popularFeedsHeader') { | ||||
|         return ( | ||||
|           <> | ||||
|             <View | ||||
|               style={[ | ||||
|                 pal.view, | ||||
|                 styles.header, | ||||
|                 { | ||||
|                   // This is first in the flatlist without a session -esb
 | ||||
|                   marginTop: hasSession ? 16 : 0, | ||||
|                   paddingLeft: isMobile ? 12 : undefined, | ||||
|                   paddingRight: 10, | ||||
|                   paddingBottom: isMobile ? 6 : undefined, | ||||
|                 }, | ||||
|               ]}> | ||||
|               <Text type="title-lg" style={[pal.text, s.bold]}> | ||||
|                 <Trans>Discover new feeds</Trans> | ||||
|               </Text> | ||||
| 
 | ||||
|               {!isMobile && ( | ||||
|                 <SearchInput | ||||
|                   query={query} | ||||
|                   onChangeQuery={onChangeQuery} | ||||
|                   onPressCancelSearch={onPressCancelSearch} | ||||
|                   onSubmitQuery={onSubmitQuery} | ||||
|                   style={{flex: 1, maxWidth: 250}} | ||||
|                 /> | ||||
|               )} | ||||
|             </View> | ||||
| 
 | ||||
|             {isMobile && ( | ||||
|               <View style={{paddingHorizontal: 8, paddingBottom: 10}}> | ||||
|                 <SearchInput | ||||
|                   query={query} | ||||
|                   onChangeQuery={onChangeQuery} | ||||
|                   onPressCancelSearch={onPressCancelSearch} | ||||
|                   onSubmitQuery={onSubmitQuery} | ||||
|                 /> | ||||
|               </View> | ||||
|             )} | ||||
|           </> | ||||
|         ) | ||||
|       } else if (item.type === 'popularFeedsLoading') { | ||||
|         return <FeedFeedLoadingPlaceholder /> | ||||
|       } else if (item.type === 'popularFeed') { | ||||
|         return ( | ||||
|           <FeedSourceCard | ||||
|             feedUri={item.feedUri} | ||||
|             showSaveBtn={hasSession} | ||||
|             showDescription | ||||
|             showLikes | ||||
|           /> | ||||
|         ) | ||||
|       } else if (item.type === 'popularFeedsNoResults') { | ||||
|         return ( | ||||
|           <View | ||||
|             style={{ | ||||
|               paddingHorizontal: 16, | ||||
|               paddingTop: 10, | ||||
|               paddingBottom: '150%', | ||||
|             }}> | ||||
|             <Text type="lg" style={pal.textLight}> | ||||
|               <Trans>No results found for "{query}"</Trans> | ||||
|             </Text> | ||||
|           </View> | ||||
|         ) | ||||
|       } | ||||
|       return null | ||||
|     }, | ||||
|     [ | ||||
|       _, | ||||
|       hasSession, | ||||
|       isMobile, | ||||
|       pal, | ||||
|       query, | ||||
|       onChangeQuery, | ||||
|       onPressCancelSearch, | ||||
|       onSubmitQuery, | ||||
|     ], | ||||
|   ) | ||||
| 
 | ||||
|   return ( | ||||
|     <View style={[pal.view, styles.container]}> | ||||
|       {isMobile && ( | ||||
|         <ViewHeader | ||||
|           title={_(msg`Feeds`)} | ||||
|           canGoBack={false} | ||||
|           renderButton={renderHeaderBtn} | ||||
|           showBorder | ||||
|         /> | ||||
|       )} | ||||
| 
 | ||||
|       {preferences ? <View /> : <ActivityIndicator />} | ||||
| 
 | ||||
|       <FlatList | ||||
|         style={[!isTabletOrDesktop && s.flex1, styles.list]} | ||||
|         data={items} | ||||
|         keyExtractor={item => item.key} | ||||
|         contentContainerStyle={styles.contentContainer} | ||||
|         renderItem={renderItem} | ||||
|         refreshControl={ | ||||
|           <RefreshControl | ||||
|             refreshing={isPTR} | ||||
|             onRefresh={isUserSearching ? undefined : onPullToRefresh} | ||||
|             tintColor={pal.colors.text} | ||||
|             titleColor={pal.colors.text} | ||||
|           /> | ||||
|         } | ||||
|         initialNumToRender={10} | ||||
|         onEndReached={onEndReached} | ||||
|         // @ts-ignore our .web version only -prf
 | ||||
|         desktopFixedHeight | ||||
|       /> | ||||
| 
 | ||||
|       {hasSession && ( | ||||
|         <FAB | ||||
|           testID="composeFAB" | ||||
|           onPress={onPressCompose} | ||||
|           icon={<ComposeIcon2 strokeWidth={1.5} size={29} style={s.white} />} | ||||
|           accessibilityRole="button" | ||||
|           accessibilityLabel={_(msg`New post`)} | ||||
|           accessibilityHint="" | ||||
|         /> | ||||
|       )} | ||||
|     </View> | ||||
|   ) | ||||
| } | ||||
| 
 | ||||
| function SavedFeed({feedUri}: {feedUri: string}) { | ||||
|   const pal = usePalette('default') | ||||
|  |  | |||
|  | @ -3,7 +3,6 @@ import {View, ActivityIndicator, StyleSheet} from 'react-native' | |||
| import {useFocusEffect} from '@react-navigation/native' | ||||
| import {NativeStackScreenProps, HomeTabNavigatorParams} from 'lib/routes/types' | ||||
| import {FeedDescriptor, FeedParams} from '#/state/queries/post-feed' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {FollowingEmptyState} from 'view/com/posts/FollowingEmptyState' | ||||
| import {FollowingEndOfFeed} from 'view/com/posts/FollowingEndOfFeed' | ||||
| import {CustomFeedEmptyState} from 'view/com/posts/CustomFeedEmptyState' | ||||
|  | @ -17,29 +16,24 @@ import {emitSoftReset} from '#/state/events' | |||
| import {useSession} from '#/state/session' | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<HomeTabNavigatorParams, 'Home'> | ||||
| export const HomeScreen = withAuthRequired( | ||||
|   function HomeScreenImpl(props: Props) { | ||||
|     const {hasSession} = useSession() | ||||
|     const {data: preferences} = usePreferencesQuery() | ||||
| export function HomeScreen(props: Props) { | ||||
|   const {hasSession} = useSession() | ||||
|   const {data: preferences} = usePreferencesQuery() | ||||
| 
 | ||||
|     if (!hasSession) { | ||||
|       return <HomeScreenPublic /> | ||||
|     } | ||||
|   if (!hasSession) { | ||||
|     return <HomeScreenPublic /> | ||||
|   } | ||||
| 
 | ||||
|     if (preferences) { | ||||
|       return <HomeScreenReady {...props} preferences={preferences} /> | ||||
|     } else { | ||||
|       return ( | ||||
|         <View style={styles.loading}> | ||||
|           <ActivityIndicator size="large" /> | ||||
|         </View> | ||||
|       ) | ||||
|     } | ||||
|   }, | ||||
|   { | ||||
|     isPublic: true, | ||||
|   }, | ||||
| ) | ||||
|   if (preferences) { | ||||
|     return <HomeScreenReady {...props} preferences={preferences} /> | ||||
|   } else { | ||||
|     return ( | ||||
|       <View style={styles.loading}> | ||||
|         <ActivityIndicator size="large" /> | ||||
|       </View> | ||||
|     ) | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| function HomeScreenPublic() { | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|  |  | |||
|  | @ -4,7 +4,6 @@ import {useFocusEffect, useNavigation} from '@react-navigation/native' | |||
| import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome' | ||||
| import {AtUri} from '@atproto/api' | ||||
| import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {MyLists} from '#/view/com/lists/MyLists' | ||||
| import {Text} from 'view/com/util/text/Text' | ||||
| import {Button} from 'view/com/util/forms/Button' | ||||
|  | @ -18,70 +17,68 @@ import {useModalControls} from '#/state/modals' | |||
| import {Trans} from '@lingui/macro' | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'Lists'> | ||||
| export const ListsScreen = withAuthRequired( | ||||
|   function ListsScreenImpl({}: Props) { | ||||
|     const pal = usePalette('default') | ||||
|     const setMinimalShellMode = useSetMinimalShellMode() | ||||
|     const {isMobile} = useWebMediaQueries() | ||||
|     const navigation = useNavigation<NavigationProp>() | ||||
|     const {openModal} = useModalControls() | ||||
| export function ListsScreen({}: Props) { | ||||
|   const pal = usePalette('default') | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|   const {isMobile} = useWebMediaQueries() | ||||
|   const navigation = useNavigation<NavigationProp>() | ||||
|   const {openModal} = useModalControls() | ||||
| 
 | ||||
|     useFocusEffect( | ||||
|       React.useCallback(() => { | ||||
|         setMinimalShellMode(false) | ||||
|       }, [setMinimalShellMode]), | ||||
|     ) | ||||
|   useFocusEffect( | ||||
|     React.useCallback(() => { | ||||
|       setMinimalShellMode(false) | ||||
|     }, [setMinimalShellMode]), | ||||
|   ) | ||||
| 
 | ||||
|     const onPressNewList = React.useCallback(() => { | ||||
|       openModal({ | ||||
|         name: 'create-or-edit-list', | ||||
|         purpose: 'app.bsky.graph.defs#curatelist', | ||||
|         onSave: (uri: string) => { | ||||
|           try { | ||||
|             const urip = new AtUri(uri) | ||||
|             navigation.navigate('ProfileList', { | ||||
|               name: urip.hostname, | ||||
|               rkey: urip.rkey, | ||||
|             }) | ||||
|           } catch {} | ||||
|         }, | ||||
|       }) | ||||
|     }, [openModal, navigation]) | ||||
|   const onPressNewList = React.useCallback(() => { | ||||
|     openModal({ | ||||
|       name: 'create-or-edit-list', | ||||
|       purpose: 'app.bsky.graph.defs#curatelist', | ||||
|       onSave: (uri: string) => { | ||||
|         try { | ||||
|           const urip = new AtUri(uri) | ||||
|           navigation.navigate('ProfileList', { | ||||
|             name: urip.hostname, | ||||
|             rkey: urip.rkey, | ||||
|           }) | ||||
|         } catch {} | ||||
|       }, | ||||
|     }) | ||||
|   }, [openModal, navigation]) | ||||
| 
 | ||||
|     return ( | ||||
|       <View style={s.hContentRegion} testID="listsScreen"> | ||||
|         <SimpleViewHeader | ||||
|           showBackButton={isMobile} | ||||
|           style={ | ||||
|             !isMobile && [pal.border, {borderLeftWidth: 1, borderRightWidth: 1}] | ||||
|           }> | ||||
|           <View style={{flex: 1}}> | ||||
|             <Text type="title-lg" style={[pal.text, {fontWeight: 'bold'}]}> | ||||
|               <Trans>User Lists</Trans> | ||||
|   return ( | ||||
|     <View style={s.hContentRegion} testID="listsScreen"> | ||||
|       <SimpleViewHeader | ||||
|         showBackButton={isMobile} | ||||
|         style={ | ||||
|           !isMobile && [pal.border, {borderLeftWidth: 1, borderRightWidth: 1}] | ||||
|         }> | ||||
|         <View style={{flex: 1}}> | ||||
|           <Text type="title-lg" style={[pal.text, {fontWeight: 'bold'}]}> | ||||
|             <Trans>User Lists</Trans> | ||||
|           </Text> | ||||
|           <Text style={pal.textLight}> | ||||
|             <Trans>Public, shareable lists which can drive feeds.</Trans> | ||||
|           </Text> | ||||
|         </View> | ||||
|         <View> | ||||
|           <Button | ||||
|             testID="newUserListBtn" | ||||
|             type="default" | ||||
|             onPress={onPressNewList} | ||||
|             style={{ | ||||
|               flexDirection: 'row', | ||||
|               alignItems: 'center', | ||||
|               gap: 8, | ||||
|             }}> | ||||
|             <FontAwesomeIcon icon="plus" color={pal.colors.text} /> | ||||
|             <Text type="button" style={pal.text}> | ||||
|               <Trans>New</Trans> | ||||
|             </Text> | ||||
|             <Text style={pal.textLight}> | ||||
|               <Trans>Public, shareable lists which can drive feeds.</Trans> | ||||
|             </Text> | ||||
|           </View> | ||||
|           <View> | ||||
|             <Button | ||||
|               testID="newUserListBtn" | ||||
|               type="default" | ||||
|               onPress={onPressNewList} | ||||
|               style={{ | ||||
|                 flexDirection: 'row', | ||||
|                 alignItems: 'center', | ||||
|                 gap: 8, | ||||
|               }}> | ||||
|               <FontAwesomeIcon icon="plus" color={pal.colors.text} /> | ||||
|               <Text type="button" style={pal.text}> | ||||
|                 <Trans>New</Trans> | ||||
|               </Text> | ||||
|             </Button> | ||||
|           </View> | ||||
|         </SimpleViewHeader> | ||||
|         <MyLists filter="curate" style={s.flexGrow1} /> | ||||
|       </View> | ||||
|     ) | ||||
|   }, | ||||
| ) | ||||
|           </Button> | ||||
|         </View> | ||||
|       </SimpleViewHeader> | ||||
|       <MyLists filter="curate" style={s.flexGrow1} /> | ||||
|     </View> | ||||
|   ) | ||||
| } | ||||
|  |  | |||
|  | @ -6,7 +6,6 @@ import { | |||
|   FontAwesomeIconStyle, | ||||
| } from '@fortawesome/react-native-fontawesome' | ||||
| import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {s} from 'lib/styles' | ||||
| import {CenteredView} from '../com/util/Views' | ||||
| import {ViewHeader} from '../com/util/ViewHeader' | ||||
|  | @ -21,100 +20,98 @@ import {Trans, msg} from '@lingui/macro' | |||
| import {useLingui} from '@lingui/react' | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'Moderation'> | ||||
| export const ModerationScreen = withAuthRequired( | ||||
|   function Moderation({}: Props) { | ||||
|     const pal = usePalette('default') | ||||
|     const {_} = useLingui() | ||||
|     const setMinimalShellMode = useSetMinimalShellMode() | ||||
|     const {screen, track} = useAnalytics() | ||||
|     const {isTabletOrDesktop} = useWebMediaQueries() | ||||
|     const {openModal} = useModalControls() | ||||
| export function ModerationScreen({}: Props) { | ||||
|   const pal = usePalette('default') | ||||
|   const {_} = useLingui() | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|   const {screen, track} = useAnalytics() | ||||
|   const {isTabletOrDesktop} = useWebMediaQueries() | ||||
|   const {openModal} = useModalControls() | ||||
| 
 | ||||
|     useFocusEffect( | ||||
|       React.useCallback(() => { | ||||
|         screen('Moderation') | ||||
|         setMinimalShellMode(false) | ||||
|       }, [screen, setMinimalShellMode]), | ||||
|     ) | ||||
|   useFocusEffect( | ||||
|     React.useCallback(() => { | ||||
|       screen('Moderation') | ||||
|       setMinimalShellMode(false) | ||||
|     }, [screen, setMinimalShellMode]), | ||||
|   ) | ||||
| 
 | ||||
|     const onPressContentFiltering = React.useCallback(() => { | ||||
|       track('Moderation:ContentfilteringButtonClicked') | ||||
|       openModal({name: 'content-filtering-settings'}) | ||||
|     }, [track, openModal]) | ||||
|   const onPressContentFiltering = React.useCallback(() => { | ||||
|     track('Moderation:ContentfilteringButtonClicked') | ||||
|     openModal({name: 'content-filtering-settings'}) | ||||
|   }, [track, openModal]) | ||||
| 
 | ||||
|     return ( | ||||
|       <CenteredView | ||||
|         style={[ | ||||
|           s.hContentRegion, | ||||
|           pal.border, | ||||
|           isTabletOrDesktop ? styles.desktopContainer : pal.viewLight, | ||||
|         ]} | ||||
|         testID="moderationScreen"> | ||||
|         <ViewHeader title={_(msg`Moderation`)} showOnDesktop /> | ||||
|         <View style={styles.spacer} /> | ||||
|         <TouchableOpacity | ||||
|           testID="contentFilteringBtn" | ||||
|           style={[styles.linkCard, pal.view]} | ||||
|           onPress={onPressContentFiltering} | ||||
|           accessibilityRole="tab" | ||||
|           accessibilityHint="Content filtering" | ||||
|           accessibilityLabel=""> | ||||
|           <View style={[styles.iconContainer, pal.btn]}> | ||||
|             <FontAwesomeIcon | ||||
|               icon="eye" | ||||
|               style={pal.text as FontAwesomeIconStyle} | ||||
|             /> | ||||
|           </View> | ||||
|           <Text type="lg" style={pal.text}> | ||||
|             <Trans>Content filtering</Trans> | ||||
|           </Text> | ||||
|         </TouchableOpacity> | ||||
|         <Link | ||||
|           testID="moderationlistsBtn" | ||||
|           style={[styles.linkCard, pal.view]} | ||||
|           href="/moderation/modlists"> | ||||
|           <View style={[styles.iconContainer, pal.btn]}> | ||||
|             <FontAwesomeIcon | ||||
|               icon="users-slash" | ||||
|               style={pal.text as FontAwesomeIconStyle} | ||||
|             /> | ||||
|           </View> | ||||
|           <Text type="lg" style={pal.text}> | ||||
|             <Trans>Moderation lists</Trans> | ||||
|           </Text> | ||||
|         </Link> | ||||
|         <Link | ||||
|           testID="mutedAccountsBtn" | ||||
|           style={[styles.linkCard, pal.view]} | ||||
|           href="/moderation/muted-accounts"> | ||||
|           <View style={[styles.iconContainer, pal.btn]}> | ||||
|             <FontAwesomeIcon | ||||
|               icon="user-slash" | ||||
|               style={pal.text as FontAwesomeIconStyle} | ||||
|             /> | ||||
|           </View> | ||||
|           <Text type="lg" style={pal.text}> | ||||
|             <Trans>Muted accounts</Trans> | ||||
|           </Text> | ||||
|         </Link> | ||||
|         <Link | ||||
|           testID="blockedAccountsBtn" | ||||
|           style={[styles.linkCard, pal.view]} | ||||
|           href="/moderation/blocked-accounts"> | ||||
|           <View style={[styles.iconContainer, pal.btn]}> | ||||
|             <FontAwesomeIcon | ||||
|               icon="ban" | ||||
|               style={pal.text as FontAwesomeIconStyle} | ||||
|             /> | ||||
|           </View> | ||||
|           <Text type="lg" style={pal.text}> | ||||
|             <Trans>Blocked accounts</Trans> | ||||
|           </Text> | ||||
|         </Link> | ||||
|       </CenteredView> | ||||
|     ) | ||||
|   }, | ||||
| ) | ||||
|   return ( | ||||
|     <CenteredView | ||||
|       style={[ | ||||
|         s.hContentRegion, | ||||
|         pal.border, | ||||
|         isTabletOrDesktop ? styles.desktopContainer : pal.viewLight, | ||||
|       ]} | ||||
|       testID="moderationScreen"> | ||||
|       <ViewHeader title={_(msg`Moderation`)} showOnDesktop /> | ||||
|       <View style={styles.spacer} /> | ||||
|       <TouchableOpacity | ||||
|         testID="contentFilteringBtn" | ||||
|         style={[styles.linkCard, pal.view]} | ||||
|         onPress={onPressContentFiltering} | ||||
|         accessibilityRole="tab" | ||||
|         accessibilityHint="Content filtering" | ||||
|         accessibilityLabel=""> | ||||
|         <View style={[styles.iconContainer, pal.btn]}> | ||||
|           <FontAwesomeIcon | ||||
|             icon="eye" | ||||
|             style={pal.text as FontAwesomeIconStyle} | ||||
|           /> | ||||
|         </View> | ||||
|         <Text type="lg" style={pal.text}> | ||||
|           <Trans>Content filtering</Trans> | ||||
|         </Text> | ||||
|       </TouchableOpacity> | ||||
|       <Link | ||||
|         testID="moderationlistsBtn" | ||||
|         style={[styles.linkCard, pal.view]} | ||||
|         href="/moderation/modlists"> | ||||
|         <View style={[styles.iconContainer, pal.btn]}> | ||||
|           <FontAwesomeIcon | ||||
|             icon="users-slash" | ||||
|             style={pal.text as FontAwesomeIconStyle} | ||||
|           /> | ||||
|         </View> | ||||
|         <Text type="lg" style={pal.text}> | ||||
|           <Trans>Moderation lists</Trans> | ||||
|         </Text> | ||||
|       </Link> | ||||
|       <Link | ||||
|         testID="mutedAccountsBtn" | ||||
|         style={[styles.linkCard, pal.view]} | ||||
|         href="/moderation/muted-accounts"> | ||||
|         <View style={[styles.iconContainer, pal.btn]}> | ||||
|           <FontAwesomeIcon | ||||
|             icon="user-slash" | ||||
|             style={pal.text as FontAwesomeIconStyle} | ||||
|           /> | ||||
|         </View> | ||||
|         <Text type="lg" style={pal.text}> | ||||
|           <Trans>Muted accounts</Trans> | ||||
|         </Text> | ||||
|       </Link> | ||||
|       <Link | ||||
|         testID="blockedAccountsBtn" | ||||
|         style={[styles.linkCard, pal.view]} | ||||
|         href="/moderation/blocked-accounts"> | ||||
|         <View style={[styles.iconContainer, pal.btn]}> | ||||
|           <FontAwesomeIcon | ||||
|             icon="ban" | ||||
|             style={pal.text as FontAwesomeIconStyle} | ||||
|           /> | ||||
|         </View> | ||||
|         <Text type="lg" style={pal.text}> | ||||
|           <Trans>Blocked accounts</Trans> | ||||
|         </Text> | ||||
|       </Link> | ||||
|     </CenteredView> | ||||
|   ) | ||||
| } | ||||
| 
 | ||||
| const styles = StyleSheet.create({ | ||||
|   desktopContainer: { | ||||
|  |  | |||
|  | @ -10,7 +10,6 @@ import {AppBskyActorDefs as ActorDefs} from '@atproto/api' | |||
| import {Text} from '../com/util/text/Text' | ||||
| import {usePalette} from 'lib/hooks/usePalette' | ||||
| import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {NativeStackScreenProps} from '@react-navigation/native-stack' | ||||
| import {CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {useAnalytics} from 'lib/analytics/analytics' | ||||
|  | @ -30,146 +29,144 @@ type Props = NativeStackScreenProps< | |||
|   CommonNavigatorParams, | ||||
|   'ModerationBlockedAccounts' | ||||
| > | ||||
| export const ModerationBlockedAccounts = withAuthRequired( | ||||
|   function ModerationBlockedAccountsImpl({}: Props) { | ||||
|     const pal = usePalette('default') | ||||
|     const {_} = useLingui() | ||||
|     const setMinimalShellMode = useSetMinimalShellMode() | ||||
|     const {isTabletOrDesktop} = useWebMediaQueries() | ||||
|     const {screen} = useAnalytics() | ||||
|     const [isPTRing, setIsPTRing] = React.useState(false) | ||||
|     const { | ||||
|       data, | ||||
|       isFetching, | ||||
|       isError, | ||||
|       error, | ||||
|       refetch, | ||||
|       hasNextPage, | ||||
|       fetchNextPage, | ||||
|       isFetchingNextPage, | ||||
|     } = useMyBlockedAccountsQuery() | ||||
|     const isEmpty = !isFetching && !data?.pages[0]?.blocks.length | ||||
|     const profiles = React.useMemo(() => { | ||||
|       if (data?.pages) { | ||||
|         return data.pages.flatMap(page => page.blocks) | ||||
|       } | ||||
|       return [] | ||||
|     }, [data]) | ||||
| export function ModerationBlockedAccounts({}: Props) { | ||||
|   const pal = usePalette('default') | ||||
|   const {_} = useLingui() | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|   const {isTabletOrDesktop} = useWebMediaQueries() | ||||
|   const {screen} = useAnalytics() | ||||
|   const [isPTRing, setIsPTRing] = React.useState(false) | ||||
|   const { | ||||
|     data, | ||||
|     isFetching, | ||||
|     isError, | ||||
|     error, | ||||
|     refetch, | ||||
|     hasNextPage, | ||||
|     fetchNextPage, | ||||
|     isFetchingNextPage, | ||||
|   } = useMyBlockedAccountsQuery() | ||||
|   const isEmpty = !isFetching && !data?.pages[0]?.blocks.length | ||||
|   const profiles = React.useMemo(() => { | ||||
|     if (data?.pages) { | ||||
|       return data.pages.flatMap(page => page.blocks) | ||||
|     } | ||||
|     return [] | ||||
|   }, [data]) | ||||
| 
 | ||||
|     useFocusEffect( | ||||
|       React.useCallback(() => { | ||||
|         screen('BlockedAccounts') | ||||
|         setMinimalShellMode(false) | ||||
|       }, [screen, setMinimalShellMode]), | ||||
|     ) | ||||
|   useFocusEffect( | ||||
|     React.useCallback(() => { | ||||
|       screen('BlockedAccounts') | ||||
|       setMinimalShellMode(false) | ||||
|     }, [screen, setMinimalShellMode]), | ||||
|   ) | ||||
| 
 | ||||
|     const onRefresh = React.useCallback(async () => { | ||||
|       setIsPTRing(true) | ||||
|       try { | ||||
|         await refetch() | ||||
|       } catch (err) { | ||||
|         logger.error('Failed to refresh my muted accounts', {error: err}) | ||||
|       } | ||||
|       setIsPTRing(false) | ||||
|     }, [refetch, setIsPTRing]) | ||||
|   const onRefresh = React.useCallback(async () => { | ||||
|     setIsPTRing(true) | ||||
|     try { | ||||
|       await refetch() | ||||
|     } catch (err) { | ||||
|       logger.error('Failed to refresh my muted accounts', {error: err}) | ||||
|     } | ||||
|     setIsPTRing(false) | ||||
|   }, [refetch, setIsPTRing]) | ||||
| 
 | ||||
|     const onEndReached = React.useCallback(async () => { | ||||
|       if (isFetching || !hasNextPage || isError) return | ||||
|   const onEndReached = React.useCallback(async () => { | ||||
|     if (isFetching || !hasNextPage || isError) return | ||||
| 
 | ||||
|       try { | ||||
|         await fetchNextPage() | ||||
|       } catch (err) { | ||||
|         logger.error('Failed to load more of my muted accounts', {error: err}) | ||||
|       } | ||||
|     }, [isFetching, hasNextPage, isError, fetchNextPage]) | ||||
|     try { | ||||
|       await fetchNextPage() | ||||
|     } catch (err) { | ||||
|       logger.error('Failed to load more of my muted accounts', {error: err}) | ||||
|     } | ||||
|   }, [isFetching, hasNextPage, isError, fetchNextPage]) | ||||
| 
 | ||||
|     const renderItem = ({ | ||||
|       item, | ||||
|       index, | ||||
|     }: { | ||||
|       item: ActorDefs.ProfileView | ||||
|       index: number | ||||
|     }) => ( | ||||
|       <ProfileCard | ||||
|         testID={`blockedAccount-${index}`} | ||||
|         key={item.did} | ||||
|         profile={item} | ||||
|       /> | ||||
|     ) | ||||
|     return ( | ||||
|       <CenteredView | ||||
|   const renderItem = ({ | ||||
|     item, | ||||
|     index, | ||||
|   }: { | ||||
|     item: ActorDefs.ProfileView | ||||
|     index: number | ||||
|   }) => ( | ||||
|     <ProfileCard | ||||
|       testID={`blockedAccount-${index}`} | ||||
|       key={item.did} | ||||
|       profile={item} | ||||
|     /> | ||||
|   ) | ||||
|   return ( | ||||
|     <CenteredView | ||||
|       style={[ | ||||
|         styles.container, | ||||
|         isTabletOrDesktop && styles.containerDesktop, | ||||
|         pal.view, | ||||
|         pal.border, | ||||
|       ]} | ||||
|       testID="blockedAccountsScreen"> | ||||
|       <ViewHeader title={_(msg`Blocked Accounts`)} showOnDesktop /> | ||||
|       <Text | ||||
|         type="sm" | ||||
|         style={[ | ||||
|           styles.container, | ||||
|           isTabletOrDesktop && styles.containerDesktop, | ||||
|           pal.view, | ||||
|           pal.border, | ||||
|         ]} | ||||
|         testID="blockedAccountsScreen"> | ||||
|         <ViewHeader title={_(msg`Blocked Accounts`)} showOnDesktop /> | ||||
|         <Text | ||||
|           type="sm" | ||||
|           style={[ | ||||
|             styles.description, | ||||
|             pal.text, | ||||
|             isTabletOrDesktop && styles.descriptionDesktop, | ||||
|           ]}> | ||||
|           <Trans> | ||||
|             Blocked accounts cannot reply in your threads, mention you, or | ||||
|             otherwise interact with you. You will not see their content and they | ||||
|             will be prevented from seeing yours. | ||||
|           </Trans> | ||||
|         </Text> | ||||
|         {isEmpty ? ( | ||||
|           <View style={[pal.border, !isTabletOrDesktop && styles.flex1]}> | ||||
|             {isError ? ( | ||||
|               <ErrorScreen | ||||
|                 title="Oops!" | ||||
|                 message={cleanError(error)} | ||||
|                 onPressTryAgain={refetch} | ||||
|               /> | ||||
|             ) : ( | ||||
|               <View style={[styles.empty, pal.viewLight]}> | ||||
|                 <Text type="lg" style={[pal.text, styles.emptyText]}> | ||||
|                   <Trans> | ||||
|                     You have not blocked any accounts yet. To block an account, | ||||
|                     go to their profile and selected "Block account" from the | ||||
|                     menu on their account. | ||||
|                   </Trans> | ||||
|                 </Text> | ||||
|               </View> | ||||
|             )} | ||||
|           </View> | ||||
|         ) : ( | ||||
|           <FlatList | ||||
|             style={[!isTabletOrDesktop && styles.flex1]} | ||||
|             data={profiles} | ||||
|             keyExtractor={(item: ActorDefs.ProfileView) => item.did} | ||||
|             refreshControl={ | ||||
|               <RefreshControl | ||||
|                 refreshing={isPTRing} | ||||
|                 onRefresh={onRefresh} | ||||
|                 tintColor={pal.colors.text} | ||||
|                 titleColor={pal.colors.text} | ||||
|               /> | ||||
|             } | ||||
|             onEndReached={onEndReached} | ||||
|             renderItem={renderItem} | ||||
|             initialNumToRender={15} | ||||
|             // FIXME(dan)
 | ||||
|           styles.description, | ||||
|           pal.text, | ||||
|           isTabletOrDesktop && styles.descriptionDesktop, | ||||
|         ]}> | ||||
|         <Trans> | ||||
|           Blocked accounts cannot reply in your threads, mention you, or | ||||
|           otherwise interact with you. You will not see their content and they | ||||
|           will be prevented from seeing yours. | ||||
|         </Trans> | ||||
|       </Text> | ||||
|       {isEmpty ? ( | ||||
|         <View style={[pal.border, !isTabletOrDesktop && styles.flex1]}> | ||||
|           {isError ? ( | ||||
|             <ErrorScreen | ||||
|               title="Oops!" | ||||
|               message={cleanError(error)} | ||||
|               onPressTryAgain={refetch} | ||||
|             /> | ||||
|           ) : ( | ||||
|             <View style={[styles.empty, pal.viewLight]}> | ||||
|               <Text type="lg" style={[pal.text, styles.emptyText]}> | ||||
|                 <Trans> | ||||
|                   You have not blocked any accounts yet. To block an account, go | ||||
|                   to their profile and selected "Block account" from the menu on | ||||
|                   their account. | ||||
|                 </Trans> | ||||
|               </Text> | ||||
|             </View> | ||||
|           )} | ||||
|         </View> | ||||
|       ) : ( | ||||
|         <FlatList | ||||
|           style={[!isTabletOrDesktop && styles.flex1]} | ||||
|           data={profiles} | ||||
|           keyExtractor={(item: ActorDefs.ProfileView) => item.did} | ||||
|           refreshControl={ | ||||
|             <RefreshControl | ||||
|               refreshing={isPTRing} | ||||
|               onRefresh={onRefresh} | ||||
|               tintColor={pal.colors.text} | ||||
|               titleColor={pal.colors.text} | ||||
|             /> | ||||
|           } | ||||
|           onEndReached={onEndReached} | ||||
|           renderItem={renderItem} | ||||
|           initialNumToRender={15} | ||||
|           // FIXME(dan)
 | ||||
| 
 | ||||
|             ListFooterComponent={() => ( | ||||
|               <View style={styles.footer}> | ||||
|                 {(isFetching || isFetchingNextPage) && <ActivityIndicator />} | ||||
|               </View> | ||||
|             )} | ||||
|             // @ts-ignore our .web version only -prf
 | ||||
|             desktopFixedHeight | ||||
|           /> | ||||
|         )} | ||||
|       </CenteredView> | ||||
|     ) | ||||
|   }, | ||||
| ) | ||||
|           ListFooterComponent={() => ( | ||||
|             <View style={styles.footer}> | ||||
|               {(isFetching || isFetchingNextPage) && <ActivityIndicator />} | ||||
|             </View> | ||||
|           )} | ||||
|           // @ts-ignore our .web version only -prf
 | ||||
|           desktopFixedHeight | ||||
|         /> | ||||
|       )} | ||||
|     </CenteredView> | ||||
|   ) | ||||
| } | ||||
| 
 | ||||
| const styles = StyleSheet.create({ | ||||
|   container: { | ||||
|  |  | |||
|  | @ -4,7 +4,6 @@ import {useFocusEffect, useNavigation} from '@react-navigation/native' | |||
| import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome' | ||||
| import {AtUri} from '@atproto/api' | ||||
| import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {MyLists} from '#/view/com/lists/MyLists' | ||||
| import {Text} from 'view/com/util/text/Text' | ||||
| import {Button} from 'view/com/util/forms/Button' | ||||
|  | @ -17,70 +16,68 @@ import {useSetMinimalShellMode} from '#/state/shell' | |||
| import {useModalControls} from '#/state/modals' | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'ModerationModlists'> | ||||
| export const ModerationModlistsScreen = withAuthRequired( | ||||
|   function ModerationModlistsScreenImpl({}: Props) { | ||||
|     const pal = usePalette('default') | ||||
|     const setMinimalShellMode = useSetMinimalShellMode() | ||||
|     const {isMobile} = useWebMediaQueries() | ||||
|     const navigation = useNavigation<NavigationProp>() | ||||
|     const {openModal} = useModalControls() | ||||
| export function ModerationModlistsScreen({}: Props) { | ||||
|   const pal = usePalette('default') | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|   const {isMobile} = useWebMediaQueries() | ||||
|   const navigation = useNavigation<NavigationProp>() | ||||
|   const {openModal} = useModalControls() | ||||
| 
 | ||||
|     useFocusEffect( | ||||
|       React.useCallback(() => { | ||||
|         setMinimalShellMode(false) | ||||
|       }, [setMinimalShellMode]), | ||||
|     ) | ||||
|   useFocusEffect( | ||||
|     React.useCallback(() => { | ||||
|       setMinimalShellMode(false) | ||||
|     }, [setMinimalShellMode]), | ||||
|   ) | ||||
| 
 | ||||
|     const onPressNewList = React.useCallback(() => { | ||||
|       openModal({ | ||||
|         name: 'create-or-edit-list', | ||||
|         purpose: 'app.bsky.graph.defs#modlist', | ||||
|         onSave: (uri: string) => { | ||||
|           try { | ||||
|             const urip = new AtUri(uri) | ||||
|             navigation.navigate('ProfileList', { | ||||
|               name: urip.hostname, | ||||
|               rkey: urip.rkey, | ||||
|             }) | ||||
|           } catch {} | ||||
|         }, | ||||
|       }) | ||||
|     }, [openModal, navigation]) | ||||
|   const onPressNewList = React.useCallback(() => { | ||||
|     openModal({ | ||||
|       name: 'create-or-edit-list', | ||||
|       purpose: 'app.bsky.graph.defs#modlist', | ||||
|       onSave: (uri: string) => { | ||||
|         try { | ||||
|           const urip = new AtUri(uri) | ||||
|           navigation.navigate('ProfileList', { | ||||
|             name: urip.hostname, | ||||
|             rkey: urip.rkey, | ||||
|           }) | ||||
|         } catch {} | ||||
|       }, | ||||
|     }) | ||||
|   }, [openModal, navigation]) | ||||
| 
 | ||||
|     return ( | ||||
|       <View style={s.hContentRegion} testID="moderationModlistsScreen"> | ||||
|         <SimpleViewHeader | ||||
|           showBackButton={isMobile} | ||||
|           style={ | ||||
|             !isMobile && [pal.border, {borderLeftWidth: 1, borderRightWidth: 1}] | ||||
|           }> | ||||
|           <View style={{flex: 1}}> | ||||
|             <Text type="title-lg" style={[pal.text, {fontWeight: 'bold'}]}> | ||||
|               Moderation Lists | ||||
|   return ( | ||||
|     <View style={s.hContentRegion} testID="moderationModlistsScreen"> | ||||
|       <SimpleViewHeader | ||||
|         showBackButton={isMobile} | ||||
|         style={ | ||||
|           !isMobile && [pal.border, {borderLeftWidth: 1, borderRightWidth: 1}] | ||||
|         }> | ||||
|         <View style={{flex: 1}}> | ||||
|           <Text type="title-lg" style={[pal.text, {fontWeight: 'bold'}]}> | ||||
|             Moderation Lists | ||||
|           </Text> | ||||
|           <Text style={pal.textLight}> | ||||
|             Public, shareable lists of users to mute or block in bulk. | ||||
|           </Text> | ||||
|         </View> | ||||
|         <View> | ||||
|           <Button | ||||
|             testID="newModListBtn" | ||||
|             type="default" | ||||
|             onPress={onPressNewList} | ||||
|             style={{ | ||||
|               flexDirection: 'row', | ||||
|               alignItems: 'center', | ||||
|               gap: 8, | ||||
|             }}> | ||||
|             <FontAwesomeIcon icon="plus" color={pal.colors.text} /> | ||||
|             <Text type="button" style={pal.text}> | ||||
|               New | ||||
|             </Text> | ||||
|             <Text style={pal.textLight}> | ||||
|               Public, shareable lists of users to mute or block in bulk. | ||||
|             </Text> | ||||
|           </View> | ||||
|           <View> | ||||
|             <Button | ||||
|               testID="newModListBtn" | ||||
|               type="default" | ||||
|               onPress={onPressNewList} | ||||
|               style={{ | ||||
|                 flexDirection: 'row', | ||||
|                 alignItems: 'center', | ||||
|                 gap: 8, | ||||
|               }}> | ||||
|               <FontAwesomeIcon icon="plus" color={pal.colors.text} /> | ||||
|               <Text type="button" style={pal.text}> | ||||
|                 New | ||||
|               </Text> | ||||
|             </Button> | ||||
|           </View> | ||||
|         </SimpleViewHeader> | ||||
|         <MyLists filter="mod" style={s.flexGrow1} /> | ||||
|       </View> | ||||
|     ) | ||||
|   }, | ||||
| ) | ||||
|           </Button> | ||||
|         </View> | ||||
|       </SimpleViewHeader> | ||||
|       <MyLists filter="mod" style={s.flexGrow1} /> | ||||
|     </View> | ||||
|   ) | ||||
| } | ||||
|  |  | |||
|  | @ -10,7 +10,6 @@ import {AppBskyActorDefs as ActorDefs} from '@atproto/api' | |||
| import {Text} from '../com/util/text/Text' | ||||
| import {usePalette} from 'lib/hooks/usePalette' | ||||
| import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {NativeStackScreenProps} from '@react-navigation/native-stack' | ||||
| import {CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {useAnalytics} from 'lib/analytics/analytics' | ||||
|  | @ -30,145 +29,143 @@ type Props = NativeStackScreenProps< | |||
|   CommonNavigatorParams, | ||||
|   'ModerationMutedAccounts' | ||||
| > | ||||
| export const ModerationMutedAccounts = withAuthRequired( | ||||
|   function ModerationMutedAccountsImpl({}: Props) { | ||||
|     const pal = usePalette('default') | ||||
|     const {_} = useLingui() | ||||
|     const setMinimalShellMode = useSetMinimalShellMode() | ||||
|     const {isTabletOrDesktop} = useWebMediaQueries() | ||||
|     const {screen} = useAnalytics() | ||||
|     const [isPTRing, setIsPTRing] = React.useState(false) | ||||
|     const { | ||||
|       data, | ||||
|       isFetching, | ||||
|       isError, | ||||
|       error, | ||||
|       refetch, | ||||
|       hasNextPage, | ||||
|       fetchNextPage, | ||||
|       isFetchingNextPage, | ||||
|     } = useMyMutedAccountsQuery() | ||||
|     const isEmpty = !isFetching && !data?.pages[0]?.mutes.length | ||||
|     const profiles = React.useMemo(() => { | ||||
|       if (data?.pages) { | ||||
|         return data.pages.flatMap(page => page.mutes) | ||||
|       } | ||||
|       return [] | ||||
|     }, [data]) | ||||
| export function ModerationMutedAccounts({}: Props) { | ||||
|   const pal = usePalette('default') | ||||
|   const {_} = useLingui() | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|   const {isTabletOrDesktop} = useWebMediaQueries() | ||||
|   const {screen} = useAnalytics() | ||||
|   const [isPTRing, setIsPTRing] = React.useState(false) | ||||
|   const { | ||||
|     data, | ||||
|     isFetching, | ||||
|     isError, | ||||
|     error, | ||||
|     refetch, | ||||
|     hasNextPage, | ||||
|     fetchNextPage, | ||||
|     isFetchingNextPage, | ||||
|   } = useMyMutedAccountsQuery() | ||||
|   const isEmpty = !isFetching && !data?.pages[0]?.mutes.length | ||||
|   const profiles = React.useMemo(() => { | ||||
|     if (data?.pages) { | ||||
|       return data.pages.flatMap(page => page.mutes) | ||||
|     } | ||||
|     return [] | ||||
|   }, [data]) | ||||
| 
 | ||||
|     useFocusEffect( | ||||
|       React.useCallback(() => { | ||||
|         screen('MutedAccounts') | ||||
|         setMinimalShellMode(false) | ||||
|       }, [screen, setMinimalShellMode]), | ||||
|     ) | ||||
|   useFocusEffect( | ||||
|     React.useCallback(() => { | ||||
|       screen('MutedAccounts') | ||||
|       setMinimalShellMode(false) | ||||
|     }, [screen, setMinimalShellMode]), | ||||
|   ) | ||||
| 
 | ||||
|     const onRefresh = React.useCallback(async () => { | ||||
|       setIsPTRing(true) | ||||
|       try { | ||||
|         await refetch() | ||||
|       } catch (err) { | ||||
|         logger.error('Failed to refresh my muted accounts', {error: err}) | ||||
|       } | ||||
|       setIsPTRing(false) | ||||
|     }, [refetch, setIsPTRing]) | ||||
|   const onRefresh = React.useCallback(async () => { | ||||
|     setIsPTRing(true) | ||||
|     try { | ||||
|       await refetch() | ||||
|     } catch (err) { | ||||
|       logger.error('Failed to refresh my muted accounts', {error: err}) | ||||
|     } | ||||
|     setIsPTRing(false) | ||||
|   }, [refetch, setIsPTRing]) | ||||
| 
 | ||||
|     const onEndReached = React.useCallback(async () => { | ||||
|       if (isFetching || !hasNextPage || isError) return | ||||
|   const onEndReached = React.useCallback(async () => { | ||||
|     if (isFetching || !hasNextPage || isError) return | ||||
| 
 | ||||
|       try { | ||||
|         await fetchNextPage() | ||||
|       } catch (err) { | ||||
|         logger.error('Failed to load more of my muted accounts', {error: err}) | ||||
|       } | ||||
|     }, [isFetching, hasNextPage, isError, fetchNextPage]) | ||||
|     try { | ||||
|       await fetchNextPage() | ||||
|     } catch (err) { | ||||
|       logger.error('Failed to load more of my muted accounts', {error: err}) | ||||
|     } | ||||
|   }, [isFetching, hasNextPage, isError, fetchNextPage]) | ||||
| 
 | ||||
|     const renderItem = ({ | ||||
|       item, | ||||
|       index, | ||||
|     }: { | ||||
|       item: ActorDefs.ProfileView | ||||
|       index: number | ||||
|     }) => ( | ||||
|       <ProfileCard | ||||
|         testID={`mutedAccount-${index}`} | ||||
|         key={item.did} | ||||
|         profile={item} | ||||
|       /> | ||||
|     ) | ||||
|     return ( | ||||
|       <CenteredView | ||||
|   const renderItem = ({ | ||||
|     item, | ||||
|     index, | ||||
|   }: { | ||||
|     item: ActorDefs.ProfileView | ||||
|     index: number | ||||
|   }) => ( | ||||
|     <ProfileCard | ||||
|       testID={`mutedAccount-${index}`} | ||||
|       key={item.did} | ||||
|       profile={item} | ||||
|     /> | ||||
|   ) | ||||
|   return ( | ||||
|     <CenteredView | ||||
|       style={[ | ||||
|         styles.container, | ||||
|         isTabletOrDesktop && styles.containerDesktop, | ||||
|         pal.view, | ||||
|         pal.border, | ||||
|       ]} | ||||
|       testID="mutedAccountsScreen"> | ||||
|       <ViewHeader title={_(msg`Muted Accounts`)} showOnDesktop /> | ||||
|       <Text | ||||
|         type="sm" | ||||
|         style={[ | ||||
|           styles.container, | ||||
|           isTabletOrDesktop && styles.containerDesktop, | ||||
|           pal.view, | ||||
|           pal.border, | ||||
|         ]} | ||||
|         testID="mutedAccountsScreen"> | ||||
|         <ViewHeader title={_(msg`Muted Accounts`)} showOnDesktop /> | ||||
|         <Text | ||||
|           type="sm" | ||||
|           style={[ | ||||
|             styles.description, | ||||
|             pal.text, | ||||
|             isTabletOrDesktop && styles.descriptionDesktop, | ||||
|           ]}> | ||||
|           <Trans> | ||||
|             Muted accounts have their posts removed from your feed and from your | ||||
|             notifications. Mutes are completely private. | ||||
|           </Trans> | ||||
|         </Text> | ||||
|         {isEmpty ? ( | ||||
|           <View style={[pal.border, !isTabletOrDesktop && styles.flex1]}> | ||||
|             {isError ? ( | ||||
|               <ErrorScreen | ||||
|                 title="Oops!" | ||||
|                 message={cleanError(error)} | ||||
|                 onPressTryAgain={refetch} | ||||
|               /> | ||||
|             ) : ( | ||||
|               <View style={[styles.empty, pal.viewLight]}> | ||||
|                 <Text type="lg" style={[pal.text, styles.emptyText]}> | ||||
|                   <Trans> | ||||
|                     You have not muted any accounts yet. To mute an account, go | ||||
|                     to their profile and selected "Mute account" from the menu | ||||
|                     on their account. | ||||
|                   </Trans> | ||||
|                 </Text> | ||||
|               </View> | ||||
|             )} | ||||
|           </View> | ||||
|         ) : ( | ||||
|           <FlatList | ||||
|             style={[!isTabletOrDesktop && styles.flex1]} | ||||
|             data={profiles} | ||||
|             keyExtractor={item => item.did} | ||||
|             refreshControl={ | ||||
|               <RefreshControl | ||||
|                 refreshing={isPTRing} | ||||
|                 onRefresh={onRefresh} | ||||
|                 tintColor={pal.colors.text} | ||||
|                 titleColor={pal.colors.text} | ||||
|               /> | ||||
|             } | ||||
|             onEndReached={onEndReached} | ||||
|             renderItem={renderItem} | ||||
|             initialNumToRender={15} | ||||
|             // FIXME(dan)
 | ||||
|           styles.description, | ||||
|           pal.text, | ||||
|           isTabletOrDesktop && styles.descriptionDesktop, | ||||
|         ]}> | ||||
|         <Trans> | ||||
|           Muted accounts have their posts removed from your feed and from your | ||||
|           notifications. Mutes are completely private. | ||||
|         </Trans> | ||||
|       </Text> | ||||
|       {isEmpty ? ( | ||||
|         <View style={[pal.border, !isTabletOrDesktop && styles.flex1]}> | ||||
|           {isError ? ( | ||||
|             <ErrorScreen | ||||
|               title="Oops!" | ||||
|               message={cleanError(error)} | ||||
|               onPressTryAgain={refetch} | ||||
|             /> | ||||
|           ) : ( | ||||
|             <View style={[styles.empty, pal.viewLight]}> | ||||
|               <Text type="lg" style={[pal.text, styles.emptyText]}> | ||||
|                 <Trans> | ||||
|                   You have not muted any accounts yet. To mute an account, go to | ||||
|                   their profile and selected "Mute account" from the menu on | ||||
|                   their account. | ||||
|                 </Trans> | ||||
|               </Text> | ||||
|             </View> | ||||
|           )} | ||||
|         </View> | ||||
|       ) : ( | ||||
|         <FlatList | ||||
|           style={[!isTabletOrDesktop && styles.flex1]} | ||||
|           data={profiles} | ||||
|           keyExtractor={item => item.did} | ||||
|           refreshControl={ | ||||
|             <RefreshControl | ||||
|               refreshing={isPTRing} | ||||
|               onRefresh={onRefresh} | ||||
|               tintColor={pal.colors.text} | ||||
|               titleColor={pal.colors.text} | ||||
|             /> | ||||
|           } | ||||
|           onEndReached={onEndReached} | ||||
|           renderItem={renderItem} | ||||
|           initialNumToRender={15} | ||||
|           // FIXME(dan)
 | ||||
| 
 | ||||
|             ListFooterComponent={() => ( | ||||
|               <View style={styles.footer}> | ||||
|                 {(isFetching || isFetchingNextPage) && <ActivityIndicator />} | ||||
|               </View> | ||||
|             )} | ||||
|             // @ts-ignore our .web version only -prf
 | ||||
|             desktopFixedHeight | ||||
|           /> | ||||
|         )} | ||||
|       </CenteredView> | ||||
|     ) | ||||
|   }, | ||||
| ) | ||||
|           ListFooterComponent={() => ( | ||||
|             <View style={styles.footer}> | ||||
|               {(isFetching || isFetchingNextPage) && <ActivityIndicator />} | ||||
|             </View> | ||||
|           )} | ||||
|           // @ts-ignore our .web version only -prf
 | ||||
|           desktopFixedHeight | ||||
|         /> | ||||
|       )} | ||||
|     </CenteredView> | ||||
|   ) | ||||
| } | ||||
| 
 | ||||
| const styles = StyleSheet.create({ | ||||
|   container: { | ||||
|  |  | |||
|  | @ -6,7 +6,6 @@ import { | |||
|   NativeStackScreenProps, | ||||
|   NotificationsTabNavigatorParams, | ||||
| } from 'lib/routes/types' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {ViewHeader} from '../com/util/ViewHeader' | ||||
| import {Feed} from '../com/notifications/Feed' | ||||
| import {TextLink} from 'view/com/util/Link' | ||||
|  | @ -28,102 +27,100 @@ type Props = NativeStackScreenProps< | |||
|   NotificationsTabNavigatorParams, | ||||
|   'Notifications' | ||||
| > | ||||
| export const NotificationsScreen = withAuthRequired( | ||||
|   function NotificationsScreenImpl({}: Props) { | ||||
|     const {_} = useLingui() | ||||
|     const setMinimalShellMode = useSetMinimalShellMode() | ||||
|     const [onMainScroll, isScrolledDown, resetMainScroll] = useOnMainScroll() | ||||
|     const scrollElRef = React.useRef<FlatList>(null) | ||||
|     const {screen} = useAnalytics() | ||||
|     const pal = usePalette('default') | ||||
|     const {isDesktop} = useWebMediaQueries() | ||||
|     const unreadNotifs = useUnreadNotifications() | ||||
|     const queryClient = useQueryClient() | ||||
|     const hasNew = !!unreadNotifs | ||||
| export function NotificationsScreen({}: Props) { | ||||
|   const {_} = useLingui() | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|   const [onMainScroll, isScrolledDown, resetMainScroll] = useOnMainScroll() | ||||
|   const scrollElRef = React.useRef<FlatList>(null) | ||||
|   const {screen} = useAnalytics() | ||||
|   const pal = usePalette('default') | ||||
|   const {isDesktop} = useWebMediaQueries() | ||||
|   const unreadNotifs = useUnreadNotifications() | ||||
|   const queryClient = useQueryClient() | ||||
|   const hasNew = !!unreadNotifs | ||||
| 
 | ||||
|     // event handlers
 | ||||
|     // =
 | ||||
|     const scrollToTop = React.useCallback(() => { | ||||
|       scrollElRef.current?.scrollToOffset({offset: 0}) | ||||
|       resetMainScroll() | ||||
|     }, [scrollElRef, resetMainScroll]) | ||||
|   // event handlers
 | ||||
|   // =
 | ||||
|   const scrollToTop = React.useCallback(() => { | ||||
|     scrollElRef.current?.scrollToOffset({offset: 0}) | ||||
|     resetMainScroll() | ||||
|   }, [scrollElRef, resetMainScroll]) | ||||
| 
 | ||||
|     const onPressLoadLatest = React.useCallback(() => { | ||||
|       scrollToTop() | ||||
|       queryClient.invalidateQueries({ | ||||
|         queryKey: NOTIFS_RQKEY(), | ||||
|       }) | ||||
|     }, [scrollToTop, queryClient]) | ||||
|   const onPressLoadLatest = React.useCallback(() => { | ||||
|     scrollToTop() | ||||
|     queryClient.invalidateQueries({ | ||||
|       queryKey: NOTIFS_RQKEY(), | ||||
|     }) | ||||
|   }, [scrollToTop, queryClient]) | ||||
| 
 | ||||
|     // on-visible setup
 | ||||
|     // =
 | ||||
|     useFocusEffect( | ||||
|       React.useCallback(() => { | ||||
|         setMinimalShellMode(false) | ||||
|         logger.debug('NotificationsScreen: Updating feed') | ||||
|         screen('Notifications') | ||||
|         return listenSoftReset(onPressLoadLatest) | ||||
|       }, [screen, onPressLoadLatest, setMinimalShellMode]), | ||||
|     ) | ||||
|   // on-visible setup
 | ||||
|   // =
 | ||||
|   useFocusEffect( | ||||
|     React.useCallback(() => { | ||||
|       setMinimalShellMode(false) | ||||
|       logger.debug('NotificationsScreen: Updating feed') | ||||
|       screen('Notifications') | ||||
|       return listenSoftReset(onPressLoadLatest) | ||||
|     }, [screen, onPressLoadLatest, setMinimalShellMode]), | ||||
|   ) | ||||
| 
 | ||||
|     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="/notifications" | ||||
|               style={[pal.text, {fontWeight: 'bold'}]} | ||||
|               text={ | ||||
|                 <> | ||||
|                   <Trans>Notifications</Trans>{' '} | ||||
|                   {hasNew && ( | ||||
|                     <View | ||||
|                       style={{ | ||||
|                         top: -8, | ||||
|                         backgroundColor: colors.blue3, | ||||
|                         width: 8, | ||||
|                         height: 8, | ||||
|                         borderRadius: 4, | ||||
|                       }} | ||||
|                     /> | ||||
|                   )} | ||||
|                 </> | ||||
|               } | ||||
|               onPress={emitSoftReset} | ||||
|             /> | ||||
|           </View> | ||||
|         ) | ||||
|       } | ||||
|       return <></> | ||||
|     }, [isDesktop, pal, hasNew]) | ||||
| 
 | ||||
|     return ( | ||||
|       <View testID="notificationsScreen" style={s.hContentRegion}> | ||||
|         <ViewHeader title={_(msg`Notifications`)} canGoBack={false} /> | ||||
|         <Feed | ||||
|           onScroll={onMainScroll} | ||||
|           scrollElRef={scrollElRef} | ||||
|           ListHeaderComponent={ListHeaderComponent} | ||||
|         /> | ||||
|         {(isScrolledDown || hasNew) && ( | ||||
|           <LoadLatestBtn | ||||
|             onPress={onPressLoadLatest} | ||||
|             label={_(msg`Load new notifications`)} | ||||
|             showIndicator={hasNew} | ||||
|   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="/notifications" | ||||
|             style={[pal.text, {fontWeight: 'bold'}]} | ||||
|             text={ | ||||
|               <> | ||||
|                 <Trans>Notifications</Trans>{' '} | ||||
|                 {hasNew && ( | ||||
|                   <View | ||||
|                     style={{ | ||||
|                       top: -8, | ||||
|                       backgroundColor: colors.blue3, | ||||
|                       width: 8, | ||||
|                       height: 8, | ||||
|                       borderRadius: 4, | ||||
|                     }} | ||||
|                   /> | ||||
|                 )} | ||||
|               </> | ||||
|             } | ||||
|             onPress={emitSoftReset} | ||||
|           /> | ||||
|         )} | ||||
|       </View> | ||||
|     ) | ||||
|   }, | ||||
| ) | ||||
|         </View> | ||||
|       ) | ||||
|     } | ||||
|     return <></> | ||||
|   }, [isDesktop, pal, hasNew]) | ||||
| 
 | ||||
|   return ( | ||||
|     <View testID="notificationsScreen" style={s.hContentRegion}> | ||||
|       <ViewHeader title={_(msg`Notifications`)} canGoBack={false} /> | ||||
|       <Feed | ||||
|         onScroll={onMainScroll} | ||||
|         scrollElRef={scrollElRef} | ||||
|         ListHeaderComponent={ListHeaderComponent} | ||||
|       /> | ||||
|       {(isScrolledDown || hasNew) && ( | ||||
|         <LoadLatestBtn | ||||
|           onPress={onPressLoadLatest} | ||||
|           label={_(msg`Load new notifications`)} | ||||
|           showIndicator={hasNew} | ||||
|         /> | ||||
|       )} | ||||
|     </View> | ||||
|   ) | ||||
| } | ||||
|  |  | |||
|  | @ -2,7 +2,6 @@ import React from 'react' | |||
| import {View} from 'react-native' | ||||
| import {useFocusEffect} from '@react-navigation/native' | ||||
| import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {ViewHeader} from '../com/util/ViewHeader' | ||||
| import {PostLikedBy as PostLikedByComponent} from '../com/post-thread/PostLikedBy' | ||||
| import {makeRecordUri} from 'lib/strings/url-helpers' | ||||
|  | @ -11,25 +10,22 @@ import {msg} from '@lingui/macro' | |||
| import {useLingui} from '@lingui/react' | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'PostLikedBy'> | ||||
| export const PostLikedByScreen = withAuthRequired( | ||||
|   ({route}: Props) => { | ||||
|     const setMinimalShellMode = useSetMinimalShellMode() | ||||
|     const {name, rkey} = route.params | ||||
|     const uri = makeRecordUri(name, 'app.bsky.feed.post', rkey) | ||||
|     const {_} = useLingui() | ||||
| export const PostLikedByScreen = ({route}: Props) => { | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|   const {name, rkey} = route.params | ||||
|   const uri = makeRecordUri(name, 'app.bsky.feed.post', rkey) | ||||
|   const {_} = useLingui() | ||||
| 
 | ||||
|     useFocusEffect( | ||||
|       React.useCallback(() => { | ||||
|         setMinimalShellMode(false) | ||||
|       }, [setMinimalShellMode]), | ||||
|     ) | ||||
|   useFocusEffect( | ||||
|     React.useCallback(() => { | ||||
|       setMinimalShellMode(false) | ||||
|     }, [setMinimalShellMode]), | ||||
|   ) | ||||
| 
 | ||||
|     return ( | ||||
|       <View> | ||||
|         <ViewHeader title={_(msg`Liked by`)} /> | ||||
|         <PostLikedByComponent uri={uri} /> | ||||
|       </View> | ||||
|     ) | ||||
|   }, | ||||
|   {isPublic: true}, | ||||
| ) | ||||
|   return ( | ||||
|     <View> | ||||
|       <ViewHeader title={_(msg`Liked by`)} /> | ||||
|       <PostLikedByComponent uri={uri} /> | ||||
|     </View> | ||||
|   ) | ||||
| } | ||||
|  |  | |||
|  | @ -1,7 +1,6 @@ | |||
| import React from 'react' | ||||
| import {View} from 'react-native' | ||||
| import {useFocusEffect} from '@react-navigation/native' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {ViewHeader} from '../com/util/ViewHeader' | ||||
| import {PostRepostedBy as PostRepostedByComponent} from '../com/post-thread/PostRepostedBy' | ||||
|  | @ -11,25 +10,22 @@ import {useLingui} from '@lingui/react' | |||
| import {msg} from '@lingui/macro' | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'PostRepostedBy'> | ||||
| export const PostRepostedByScreen = withAuthRequired( | ||||
|   ({route}: Props) => { | ||||
|     const {name, rkey} = route.params | ||||
|     const uri = makeRecordUri(name, 'app.bsky.feed.post', rkey) | ||||
|     const setMinimalShellMode = useSetMinimalShellMode() | ||||
|     const {_} = useLingui() | ||||
| export const PostRepostedByScreen = ({route}: Props) => { | ||||
|   const {name, rkey} = route.params | ||||
|   const uri = makeRecordUri(name, 'app.bsky.feed.post', rkey) | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|   const {_} = useLingui() | ||||
| 
 | ||||
|     useFocusEffect( | ||||
|       React.useCallback(() => { | ||||
|         setMinimalShellMode(false) | ||||
|       }, [setMinimalShellMode]), | ||||
|     ) | ||||
|   useFocusEffect( | ||||
|     React.useCallback(() => { | ||||
|       setMinimalShellMode(false) | ||||
|     }, [setMinimalShellMode]), | ||||
|   ) | ||||
| 
 | ||||
|     return ( | ||||
|       <View> | ||||
|         <ViewHeader title={_(msg`Reposted by`)} /> | ||||
|         <PostRepostedByComponent uri={uri} /> | ||||
|       </View> | ||||
|     ) | ||||
|   }, | ||||
|   {isPublic: true}, | ||||
| ) | ||||
|   return ( | ||||
|     <View> | ||||
|       <ViewHeader title={_(msg`Reposted by`)} /> | ||||
|       <PostRepostedByComponent uri={uri} /> | ||||
|     </View> | ||||
|   ) | ||||
| } | ||||
|  |  | |||
|  | @ -5,7 +5,6 @@ import {useFocusEffect} from '@react-navigation/native' | |||
| import {useQueryClient} from '@tanstack/react-query' | ||||
| import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {makeRecordUri} from 'lib/strings/url-helpers' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {ViewHeader} from '../com/util/ViewHeader' | ||||
| import {PostThread as PostThreadComponent} from '../com/post-thread/PostThread' | ||||
| import {ComposePrompt} from 'view/com/composer/Prompt' | ||||
|  | @ -27,85 +26,82 @@ import {CenteredView} from '../com/util/Views' | |||
| import {useComposerControls} from '#/state/shell/composer' | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'PostThread'> | ||||
| export const PostThreadScreen = withAuthRequired( | ||||
|   function PostThreadScreenImpl({route}: Props) { | ||||
|     const queryClient = useQueryClient() | ||||
|     const {_} = useLingui() | ||||
|     const {fabMinimalShellTransform} = useMinimalShellMode() | ||||
|     const setMinimalShellMode = useSetMinimalShellMode() | ||||
|     const {openComposer} = useComposerControls() | ||||
|     const safeAreaInsets = useSafeAreaInsets() | ||||
|     const {name, rkey} = route.params | ||||
|     const {isMobile} = useWebMediaQueries() | ||||
|     const uri = makeRecordUri(name, 'app.bsky.feed.post', rkey) | ||||
|     const {data: resolvedUri, error: uriError} = useResolveUriQuery(uri) | ||||
| export function PostThreadScreen({route}: Props) { | ||||
|   const queryClient = useQueryClient() | ||||
|   const {_} = useLingui() | ||||
|   const {fabMinimalShellTransform} = useMinimalShellMode() | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|   const {openComposer} = useComposerControls() | ||||
|   const safeAreaInsets = useSafeAreaInsets() | ||||
|   const {name, rkey} = route.params | ||||
|   const {isMobile} = useWebMediaQueries() | ||||
|   const uri = makeRecordUri(name, 'app.bsky.feed.post', rkey) | ||||
|   const {data: resolvedUri, error: uriError} = useResolveUriQuery(uri) | ||||
| 
 | ||||
|     useFocusEffect( | ||||
|       React.useCallback(() => { | ||||
|         setMinimalShellMode(false) | ||||
|       }, [setMinimalShellMode]), | ||||
|   useFocusEffect( | ||||
|     React.useCallback(() => { | ||||
|       setMinimalShellMode(false) | ||||
|     }, [setMinimalShellMode]), | ||||
|   ) | ||||
| 
 | ||||
|   const onPressReply = React.useCallback(() => { | ||||
|     if (!resolvedUri) { | ||||
|       return | ||||
|     } | ||||
|     const thread = queryClient.getQueryData<ThreadNode>( | ||||
|       POST_THREAD_RQKEY(resolvedUri.uri), | ||||
|     ) | ||||
| 
 | ||||
|     const onPressReply = React.useCallback(() => { | ||||
|       if (!resolvedUri) { | ||||
|         return | ||||
|       } | ||||
|       const thread = queryClient.getQueryData<ThreadNode>( | ||||
|         POST_THREAD_RQKEY(resolvedUri.uri), | ||||
|       ) | ||||
|       if (thread?.type !== 'post') { | ||||
|         return | ||||
|       } | ||||
|       openComposer({ | ||||
|         replyTo: { | ||||
|           uri: thread.post.uri, | ||||
|           cid: thread.post.cid, | ||||
|           text: thread.record.text, | ||||
|           author: { | ||||
|             handle: thread.post.author.handle, | ||||
|             displayName: thread.post.author.displayName, | ||||
|             avatar: thread.post.author.avatar, | ||||
|           }, | ||||
|     if (thread?.type !== 'post') { | ||||
|       return | ||||
|     } | ||||
|     openComposer({ | ||||
|       replyTo: { | ||||
|         uri: thread.post.uri, | ||||
|         cid: thread.post.cid, | ||||
|         text: thread.record.text, | ||||
|         author: { | ||||
|           handle: thread.post.author.handle, | ||||
|           displayName: thread.post.author.displayName, | ||||
|           avatar: thread.post.author.avatar, | ||||
|         }, | ||||
|         onPost: () => | ||||
|           queryClient.invalidateQueries({ | ||||
|             queryKey: POST_THREAD_RQKEY(resolvedUri.uri || ''), | ||||
|           }), | ||||
|       }) | ||||
|     }, [openComposer, queryClient, resolvedUri]) | ||||
|       }, | ||||
|       onPost: () => | ||||
|         queryClient.invalidateQueries({ | ||||
|           queryKey: POST_THREAD_RQKEY(resolvedUri.uri || ''), | ||||
|         }), | ||||
|     }) | ||||
|   }, [openComposer, queryClient, resolvedUri]) | ||||
| 
 | ||||
|     return ( | ||||
|       <View style={s.hContentRegion}> | ||||
|         {isMobile && <ViewHeader title={_(msg`Post`)} />} | ||||
|         <View style={s.flex1}> | ||||
|           {uriError ? ( | ||||
|             <CenteredView> | ||||
|               <ErrorMessage message={String(uriError)} /> | ||||
|             </CenteredView> | ||||
|           ) : ( | ||||
|             <PostThreadComponent | ||||
|               uri={resolvedUri?.uri} | ||||
|               onPressReply={onPressReply} | ||||
|             /> | ||||
|           )} | ||||
|         </View> | ||||
|         {isMobile && ( | ||||
|           <Animated.View | ||||
|             style={[ | ||||
|               styles.prompt, | ||||
|               fabMinimalShellTransform, | ||||
|               { | ||||
|                 bottom: clamp(safeAreaInsets.bottom, 15, 30), | ||||
|               }, | ||||
|             ]}> | ||||
|             <ComposePrompt onPressCompose={onPressReply} /> | ||||
|           </Animated.View> | ||||
|   return ( | ||||
|     <View style={s.hContentRegion}> | ||||
|       {isMobile && <ViewHeader title={_(msg`Post`)} />} | ||||
|       <View style={s.flex1}> | ||||
|         {uriError ? ( | ||||
|           <CenteredView> | ||||
|             <ErrorMessage message={String(uriError)} /> | ||||
|           </CenteredView> | ||||
|         ) : ( | ||||
|           <PostThreadComponent | ||||
|             uri={resolvedUri?.uri} | ||||
|             onPressReply={onPressReply} | ||||
|           /> | ||||
|         )} | ||||
|       </View> | ||||
|     ) | ||||
|   }, | ||||
|   {isPublic: true}, | ||||
| ) | ||||
|       {isMobile && ( | ||||
|         <Animated.View | ||||
|           style={[ | ||||
|             styles.prompt, | ||||
|             fabMinimalShellTransform, | ||||
|             { | ||||
|               bottom: clamp(safeAreaInsets.bottom, 15, 30), | ||||
|             }, | ||||
|           ]}> | ||||
|           <ComposePrompt onPressCompose={onPressReply} /> | ||||
|         </Animated.View> | ||||
|       )} | ||||
|     </View> | ||||
|   ) | ||||
| } | ||||
| 
 | ||||
| const styles = StyleSheet.create({ | ||||
|   prompt: { | ||||
|  |  | |||
|  | @ -5,7 +5,6 @@ import {AppBskyActorDefs, moderateProfile, ModerationOpts} from '@atproto/api' | |||
| import {msg} from '@lingui/macro' | ||||
| import {useLingui} from '@lingui/react' | ||||
| import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {ViewSelectorHandle} from '../com/util/ViewSelector' | ||||
| import {CenteredView, FlatList} from '../com/util/Views' | ||||
| import {ScreenHider} from 'view/com/util/moderation/ScreenHider' | ||||
|  | @ -43,83 +42,78 @@ interface SectionRef { | |||
| } | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'Profile'> | ||||
| export const ProfileScreen = withAuthRequired( | ||||
|   function ProfileScreenImpl({route}: Props) { | ||||
|     const {currentAccount} = useSession() | ||||
|     const name = | ||||
|       route.params.name === 'me' ? currentAccount?.did : route.params.name | ||||
|     const moderationOpts = useModerationOpts() | ||||
|     const { | ||||
|       data: resolvedDid, | ||||
|       error: resolveError, | ||||
|       refetch: refetchDid, | ||||
|       isFetching: isFetchingDid, | ||||
|     } = useResolveDidQuery(name) | ||||
|     const { | ||||
|       data: profile, | ||||
|       error: profileError, | ||||
|       refetch: refetchProfile, | ||||
|       isFetching: isFetchingProfile, | ||||
|     } = useProfileQuery({ | ||||
|       did: resolvedDid?.did, | ||||
|     }) | ||||
| export function ProfileScreen({route}: Props) { | ||||
|   const {currentAccount} = useSession() | ||||
|   const name = | ||||
|     route.params.name === 'me' ? currentAccount?.did : route.params.name | ||||
|   const moderationOpts = useModerationOpts() | ||||
|   const { | ||||
|     data: resolvedDid, | ||||
|     error: resolveError, | ||||
|     refetch: refetchDid, | ||||
|     isFetching: isFetchingDid, | ||||
|   } = useResolveDidQuery(name) | ||||
|   const { | ||||
|     data: profile, | ||||
|     error: profileError, | ||||
|     refetch: refetchProfile, | ||||
|     isFetching: isFetchingProfile, | ||||
|   } = useProfileQuery({ | ||||
|     did: resolvedDid?.did, | ||||
|   }) | ||||
| 
 | ||||
|     const onPressTryAgain = React.useCallback(() => { | ||||
|       if (resolveError) { | ||||
|         refetchDid() | ||||
|       } else { | ||||
|         refetchProfile() | ||||
|       } | ||||
|     }, [resolveError, refetchDid, refetchProfile]) | ||||
|   const onPressTryAgain = React.useCallback(() => { | ||||
|     if (resolveError) { | ||||
|       refetchDid() | ||||
|     } else { | ||||
|       refetchProfile() | ||||
|     } | ||||
|   }, [resolveError, refetchDid, refetchProfile]) | ||||
| 
 | ||||
|     if (isFetchingDid || isFetchingProfile || !moderationOpts) { | ||||
|       return ( | ||||
|         <CenteredView> | ||||
|           <ProfileHeader | ||||
|             profile={null} | ||||
|             moderation={null} | ||||
|             isProfilePreview={true} | ||||
|           /> | ||||
|         </CenteredView> | ||||
|       ) | ||||
|     } | ||||
|     if (resolveError || profileError) { | ||||
|       return ( | ||||
|         <CenteredView> | ||||
|           <ErrorScreen | ||||
|             testID="profileErrorScreen" | ||||
|             title="Oops!" | ||||
|             message={cleanError(resolveError || profileError)} | ||||
|             onPressTryAgain={onPressTryAgain} | ||||
|           /> | ||||
|         </CenteredView> | ||||
|       ) | ||||
|     } | ||||
|     if (profile && moderationOpts) { | ||||
|       return ( | ||||
|         <ProfileScreenLoaded | ||||
|           profile={profile} | ||||
|           moderationOpts={moderationOpts} | ||||
|           hideBackButton={!!route.params.hideBackButton} | ||||
|   if (isFetchingDid || isFetchingProfile || !moderationOpts) { | ||||
|     return ( | ||||
|       <CenteredView> | ||||
|         <ProfileHeader | ||||
|           profile={null} | ||||
|           moderation={null} | ||||
|           isProfilePreview={true} | ||||
|         /> | ||||
|       ) | ||||
|     } | ||||
|     // should never happen
 | ||||
|       </CenteredView> | ||||
|     ) | ||||
|   } | ||||
|   if (resolveError || profileError) { | ||||
|     return ( | ||||
|       <CenteredView> | ||||
|         <ErrorScreen | ||||
|           testID="profileErrorScreen" | ||||
|           title="Oops!" | ||||
|           message="Something went wrong and we're not sure what." | ||||
|           message={cleanError(resolveError || profileError)} | ||||
|           onPressTryAgain={onPressTryAgain} | ||||
|         /> | ||||
|       </CenteredView> | ||||
|     ) | ||||
|   }, | ||||
|   { | ||||
|     isPublic: true, | ||||
|   }, | ||||
| ) | ||||
|   } | ||||
|   if (profile && moderationOpts) { | ||||
|     return ( | ||||
|       <ProfileScreenLoaded | ||||
|         profile={profile} | ||||
|         moderationOpts={moderationOpts} | ||||
|         hideBackButton={!!route.params.hideBackButton} | ||||
|       /> | ||||
|     ) | ||||
|   } | ||||
|   // should never happen
 | ||||
|   return ( | ||||
|     <CenteredView> | ||||
|       <ErrorScreen | ||||
|         testID="profileErrorScreen" | ||||
|         title="Oops!" | ||||
|         message="Something went wrong and we're not sure what." | ||||
|         onPressTryAgain={onPressTryAgain} | ||||
|       /> | ||||
|     </CenteredView> | ||||
|   ) | ||||
| } | ||||
| 
 | ||||
| function ProfileScreenLoaded({ | ||||
|   profile: profileUnshadowed, | ||||
|  |  | |||
|  | @ -16,7 +16,6 @@ import {CommonNavigatorParams} from 'lib/routes/types' | |||
| import {makeRecordUri} from 'lib/strings/url-helpers' | ||||
| import {colors, s} from 'lib/styles' | ||||
| import {FeedDescriptor} from '#/state/queries/post-feed' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {PagerWithHeader} from 'view/com/pager/PagerWithHeader' | ||||
| import {ProfileSubpageHeader} from 'view/com/profile/ProfileSubpageHeader' | ||||
| import {Feed} from 'view/com/posts/Feed' | ||||
|  | @ -69,70 +68,65 @@ interface SectionRef { | |||
| } | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'ProfileFeed'> | ||||
| export const ProfileFeedScreen = withAuthRequired( | ||||
|   function ProfileFeedScreenImpl(props: Props) { | ||||
|     const {rkey, name: handleOrDid} = props.route.params | ||||
| export function ProfileFeedScreen(props: Props) { | ||||
|   const {rkey, name: handleOrDid} = props.route.params | ||||
| 
 | ||||
|     const pal = usePalette('default') | ||||
|     const {_} = useLingui() | ||||
|     const navigation = useNavigation<NavigationProp>() | ||||
|   const pal = usePalette('default') | ||||
|   const {_} = useLingui() | ||||
|   const navigation = useNavigation<NavigationProp>() | ||||
| 
 | ||||
|     const uri = useMemo( | ||||
|       () => makeRecordUri(handleOrDid, 'app.bsky.feed.generator', rkey), | ||||
|       [rkey, handleOrDid], | ||||
|     ) | ||||
|     const {error, data: resolvedUri} = useResolveUriQuery(uri) | ||||
|   const uri = useMemo( | ||||
|     () => makeRecordUri(handleOrDid, 'app.bsky.feed.generator', rkey), | ||||
|     [rkey, handleOrDid], | ||||
|   ) | ||||
|   const {error, data: resolvedUri} = useResolveUriQuery(uri) | ||||
| 
 | ||||
|     const onPressBack = React.useCallback(() => { | ||||
|       if (navigation.canGoBack()) { | ||||
|         navigation.goBack() | ||||
|       } else { | ||||
|         navigation.navigate('Home') | ||||
|       } | ||||
|     }, [navigation]) | ||||
| 
 | ||||
|     if (error) { | ||||
|       return ( | ||||
|         <CenteredView> | ||||
|           <View style={[pal.view, pal.border, styles.notFoundContainer]}> | ||||
|             <Text type="title-lg" style={[pal.text, s.mb10]}> | ||||
|               <Trans>Could not load feed</Trans> | ||||
|             </Text> | ||||
|             <Text type="md" style={[pal.text, s.mb20]}> | ||||
|               {error.toString()} | ||||
|             </Text> | ||||
| 
 | ||||
|             <View style={{flexDirection: 'row'}}> | ||||
|               <Button | ||||
|                 type="default" | ||||
|                 accessibilityLabel={_(msg`Go Back`)} | ||||
|                 accessibilityHint="Return to previous page" | ||||
|                 onPress={onPressBack} | ||||
|                 style={{flexShrink: 1}}> | ||||
|                 <Text type="button" style={pal.text}> | ||||
|                   <Trans>Go Back</Trans> | ||||
|                 </Text> | ||||
|               </Button> | ||||
|             </View> | ||||
|           </View> | ||||
|         </CenteredView> | ||||
|       ) | ||||
|   const onPressBack = React.useCallback(() => { | ||||
|     if (navigation.canGoBack()) { | ||||
|       navigation.goBack() | ||||
|     } else { | ||||
|       navigation.navigate('Home') | ||||
|     } | ||||
|   }, [navigation]) | ||||
| 
 | ||||
|     return resolvedUri ? ( | ||||
|       <ProfileFeedScreenIntermediate feedUri={resolvedUri.uri} /> | ||||
|     ) : ( | ||||
|   if (error) { | ||||
|     return ( | ||||
|       <CenteredView> | ||||
|         <View style={s.p20}> | ||||
|           <ActivityIndicator size="large" /> | ||||
|         <View style={[pal.view, pal.border, styles.notFoundContainer]}> | ||||
|           <Text type="title-lg" style={[pal.text, s.mb10]}> | ||||
|             <Trans>Could not load feed</Trans> | ||||
|           </Text> | ||||
|           <Text type="md" style={[pal.text, s.mb20]}> | ||||
|             {error.toString()} | ||||
|           </Text> | ||||
| 
 | ||||
|           <View style={{flexDirection: 'row'}}> | ||||
|             <Button | ||||
|               type="default" | ||||
|               accessibilityLabel={_(msg`Go Back`)} | ||||
|               accessibilityHint="Return to previous page" | ||||
|               onPress={onPressBack} | ||||
|               style={{flexShrink: 1}}> | ||||
|               <Text type="button" style={pal.text}> | ||||
|                 <Trans>Go Back</Trans> | ||||
|               </Text> | ||||
|             </Button> | ||||
|           </View> | ||||
|         </View> | ||||
|       </CenteredView> | ||||
|     ) | ||||
|   }, | ||||
|   { | ||||
|     isPublic: true, | ||||
|   }, | ||||
| ) | ||||
|   } | ||||
| 
 | ||||
|   return resolvedUri ? ( | ||||
|     <ProfileFeedScreenIntermediate feedUri={resolvedUri.uri} /> | ||||
|   ) : ( | ||||
|     <CenteredView> | ||||
|       <View style={s.p20}> | ||||
|         <ActivityIndicator size="large" /> | ||||
|       </View> | ||||
|     </CenteredView> | ||||
|   ) | ||||
| } | ||||
| 
 | ||||
| function ProfileFeedScreenIntermediate({feedUri}: {feedUri: string}) { | ||||
|   const {data: preferences} = usePreferencesQuery() | ||||
|  |  | |||
|  | @ -2,7 +2,6 @@ import React from 'react' | |||
| import {View} from 'react-native' | ||||
| import {useFocusEffect} from '@react-navigation/native' | ||||
| import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {ViewHeader} from '../com/util/ViewHeader' | ||||
| import {PostLikedBy as PostLikedByComponent} from '../com/post-thread/PostLikedBy' | ||||
| import {makeRecordUri} from 'lib/strings/url-helpers' | ||||
|  | @ -11,25 +10,22 @@ import {useLingui} from '@lingui/react' | |||
| import {msg} from '@lingui/macro' | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'ProfileFeedLikedBy'> | ||||
| export const ProfileFeedLikedByScreen = withAuthRequired( | ||||
|   ({route}: Props) => { | ||||
|     const setMinimalShellMode = useSetMinimalShellMode() | ||||
|     const {name, rkey} = route.params | ||||
|     const uri = makeRecordUri(name, 'app.bsky.feed.generator', rkey) | ||||
|     const {_} = useLingui() | ||||
| export const ProfileFeedLikedByScreen = ({route}: Props) => { | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|   const {name, rkey} = route.params | ||||
|   const uri = makeRecordUri(name, 'app.bsky.feed.generator', rkey) | ||||
|   const {_} = useLingui() | ||||
| 
 | ||||
|     useFocusEffect( | ||||
|       React.useCallback(() => { | ||||
|         setMinimalShellMode(false) | ||||
|       }, [setMinimalShellMode]), | ||||
|     ) | ||||
|   useFocusEffect( | ||||
|     React.useCallback(() => { | ||||
|       setMinimalShellMode(false) | ||||
|     }, [setMinimalShellMode]), | ||||
|   ) | ||||
| 
 | ||||
|     return ( | ||||
|       <View> | ||||
|         <ViewHeader title={_(msg`Liked by`)} /> | ||||
|         <PostLikedByComponent uri={uri} /> | ||||
|       </View> | ||||
|     ) | ||||
|   }, | ||||
|   {isPublic: true}, | ||||
| ) | ||||
|   return ( | ||||
|     <View> | ||||
|       <ViewHeader title={_(msg`Liked by`)} /> | ||||
|       <PostLikedByComponent uri={uri} /> | ||||
|     </View> | ||||
|   ) | ||||
| } | ||||
|  |  | |||
|  | @ -2,7 +2,6 @@ import React from 'react' | |||
| import {View} from 'react-native' | ||||
| import {useFocusEffect} from '@react-navigation/native' | ||||
| import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {ViewHeader} from '../com/util/ViewHeader' | ||||
| import {ProfileFollowers as ProfileFollowersComponent} from '../com/profile/ProfileFollowers' | ||||
| import {useSetMinimalShellMode} from '#/state/shell' | ||||
|  | @ -10,24 +9,21 @@ import {useLingui} from '@lingui/react' | |||
| import {msg} from '@lingui/macro' | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'ProfileFollowers'> | ||||
| export const ProfileFollowersScreen = withAuthRequired( | ||||
|   ({route}: Props) => { | ||||
|     const {name} = route.params | ||||
|     const setMinimalShellMode = useSetMinimalShellMode() | ||||
|     const {_} = useLingui() | ||||
| export const ProfileFollowersScreen = ({route}: Props) => { | ||||
|   const {name} = route.params | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|   const {_} = useLingui() | ||||
| 
 | ||||
|     useFocusEffect( | ||||
|       React.useCallback(() => { | ||||
|         setMinimalShellMode(false) | ||||
|       }, [setMinimalShellMode]), | ||||
|     ) | ||||
|   useFocusEffect( | ||||
|     React.useCallback(() => { | ||||
|       setMinimalShellMode(false) | ||||
|     }, [setMinimalShellMode]), | ||||
|   ) | ||||
| 
 | ||||
|     return ( | ||||
|       <View> | ||||
|         <ViewHeader title={_(msg`Followers`)} /> | ||||
|         <ProfileFollowersComponent name={name} /> | ||||
|       </View> | ||||
|     ) | ||||
|   }, | ||||
|   {isPublic: true}, | ||||
| ) | ||||
|   return ( | ||||
|     <View> | ||||
|       <ViewHeader title={_(msg`Followers`)} /> | ||||
|       <ProfileFollowersComponent name={name} /> | ||||
|     </View> | ||||
|   ) | ||||
| } | ||||
|  |  | |||
|  | @ -2,7 +2,6 @@ import React from 'react' | |||
| import {View} from 'react-native' | ||||
| import {useFocusEffect} from '@react-navigation/native' | ||||
| import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {ViewHeader} from '../com/util/ViewHeader' | ||||
| import {ProfileFollows as ProfileFollowsComponent} from '../com/profile/ProfileFollows' | ||||
| import {useSetMinimalShellMode} from '#/state/shell' | ||||
|  | @ -10,24 +9,21 @@ import {useLingui} from '@lingui/react' | |||
| import {msg} from '@lingui/macro' | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'ProfileFollows'> | ||||
| export const ProfileFollowsScreen = withAuthRequired( | ||||
|   ({route}: Props) => { | ||||
|     const {name} = route.params | ||||
|     const setMinimalShellMode = useSetMinimalShellMode() | ||||
|     const {_} = useLingui() | ||||
| export const ProfileFollowsScreen = ({route}: Props) => { | ||||
|   const {name} = route.params | ||||
|   const setMinimalShellMode = useSetMinimalShellMode() | ||||
|   const {_} = useLingui() | ||||
| 
 | ||||
|     useFocusEffect( | ||||
|       React.useCallback(() => { | ||||
|         setMinimalShellMode(false) | ||||
|       }, [setMinimalShellMode]), | ||||
|     ) | ||||
|   useFocusEffect( | ||||
|     React.useCallback(() => { | ||||
|       setMinimalShellMode(false) | ||||
|     }, [setMinimalShellMode]), | ||||
|   ) | ||||
| 
 | ||||
|     return ( | ||||
|       <View> | ||||
|         <ViewHeader title={_(msg`Following`)} /> | ||||
|         <ProfileFollowsComponent name={name} /> | ||||
|       </View> | ||||
|     ) | ||||
|   }, | ||||
|   {isPublic: true}, | ||||
| ) | ||||
|   return ( | ||||
|     <View> | ||||
|       <ViewHeader title={_(msg`Following`)} /> | ||||
|       <ProfileFollowsComponent name={name} /> | ||||
|     </View> | ||||
|   ) | ||||
| } | ||||
|  |  | |||
|  | @ -12,7 +12,6 @@ import {useNavigation} from '@react-navigation/native' | |||
| import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome' | ||||
| import {AppBskyGraphDefs, AtUri, RichText as RichTextAPI} from '@atproto/api' | ||||
| import {useQueryClient} from '@tanstack/react-query' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {PagerWithHeader} from 'view/com/pager/PagerWithHeader' | ||||
| import {ProfileSubpageHeader} from 'view/com/profile/ProfileSubpageHeader' | ||||
| import {Feed} from 'view/com/posts/Feed' | ||||
|  | @ -64,42 +63,40 @@ interface SectionRef { | |||
| } | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'ProfileList'> | ||||
| export const ProfileListScreen = withAuthRequired( | ||||
|   function ProfileListScreenImpl(props: Props) { | ||||
|     const {name: handleOrDid, rkey} = props.route.params | ||||
|     const {data: resolvedUri, error: resolveError} = useResolveUriQuery( | ||||
|       AtUri.make(handleOrDid, 'app.bsky.graph.list', rkey).toString(), | ||||
|     ) | ||||
|     const {data: list, error: listError} = useListQuery(resolvedUri?.uri) | ||||
| export function ProfileListScreen(props: Props) { | ||||
|   const {name: handleOrDid, rkey} = props.route.params | ||||
|   const {data: resolvedUri, error: resolveError} = useResolveUriQuery( | ||||
|     AtUri.make(handleOrDid, 'app.bsky.graph.list', rkey).toString(), | ||||
|   ) | ||||
|   const {data: list, error: listError} = useListQuery(resolvedUri?.uri) | ||||
| 
 | ||||
|     if (resolveError) { | ||||
|       return ( | ||||
|         <CenteredView> | ||||
|           <ErrorScreen | ||||
|             error={`We're sorry, but we were unable to resolve this list. If this persists, please contact the list creator, @${handleOrDid}.`} | ||||
|           /> | ||||
|         </CenteredView> | ||||
|       ) | ||||
|     } | ||||
|     if (listError) { | ||||
|       return ( | ||||
|         <CenteredView> | ||||
|           <ErrorScreen error={cleanError(listError)} /> | ||||
|         </CenteredView> | ||||
|       ) | ||||
|     } | ||||
| 
 | ||||
|     return resolvedUri && list ? ( | ||||
|       <ProfileListScreenLoaded {...props} uri={resolvedUri.uri} list={list} /> | ||||
|     ) : ( | ||||
|   if (resolveError) { | ||||
|     return ( | ||||
|       <CenteredView> | ||||
|         <View style={s.p20}> | ||||
|           <ActivityIndicator size="large" /> | ||||
|         </View> | ||||
|         <ErrorScreen | ||||
|           error={`We're sorry, but we were unable to resolve this list. If this persists, please contact the list creator, @${handleOrDid}.`} | ||||
|         /> | ||||
|       </CenteredView> | ||||
|     ) | ||||
|   }, | ||||
| ) | ||||
|   } | ||||
|   if (listError) { | ||||
|     return ( | ||||
|       <CenteredView> | ||||
|         <ErrorScreen error={cleanError(listError)} /> | ||||
|       </CenteredView> | ||||
|     ) | ||||
|   } | ||||
| 
 | ||||
|   return resolvedUri && list ? ( | ||||
|     <ProfileListScreenLoaded {...props} uri={resolvedUri.uri} list={list} /> | ||||
|   ) : ( | ||||
|     <CenteredView> | ||||
|       <View style={s.p20}> | ||||
|         <ActivityIndicator size="large" /> | ||||
|       </View> | ||||
|     </CenteredView> | ||||
|   ) | ||||
| } | ||||
| 
 | ||||
| function ProfileListScreenLoaded({ | ||||
|   route, | ||||
|  |  | |||
|  | @ -14,7 +14,6 @@ import {useAnalytics} from 'lib/analytics/analytics' | |||
| import {usePalette} from 'lib/hooks/usePalette' | ||||
| import {CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import {ViewHeader} from 'view/com/util/ViewHeader' | ||||
| import {ScrollView, CenteredView} from 'view/com/util/Views' | ||||
| import {Text} from 'view/com/util/text/Text' | ||||
|  | @ -51,7 +50,7 @@ const HITSLOP_BOTTOM = { | |||
| } | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'SavedFeeds'> | ||||
| export const SavedFeeds = withAuthRequired(function SavedFeedsImpl({}: Props) { | ||||
| export function SavedFeeds({}: Props) { | ||||
|   const pal = usePalette('default') | ||||
|   const {_} = useLingui() | ||||
|   const {isMobile, isTabletOrDesktop} = useWebMediaQueries() | ||||
|  | @ -147,7 +146,7 @@ export const SavedFeeds = withAuthRequired(function SavedFeedsImpl({}: Props) { | |||
|       </ScrollView> | ||||
|     </CenteredView> | ||||
|   ) | ||||
| }) | ||||
| } | ||||
| 
 | ||||
| function ListItem({ | ||||
|   feedUri, | ||||
|  |  | |||
|  | @ -1,6 +1,3 @@ | |||
| import {withAuthRequired} from '#/view/com/auth/withAuthRequired' | ||||
| import {SearchScreenMobile} from '#/view/screens/Search/Search' | ||||
| 
 | ||||
| export const SearchScreen = withAuthRequired(SearchScreenMobile, { | ||||
|   isPublic: true, | ||||
| }) | ||||
| export const SearchScreen = SearchScreenMobile | ||||
|  |  | |||
|  | @ -1,6 +1,3 @@ | |||
| import {withAuthRequired} from '#/view/com/auth/withAuthRequired' | ||||
| import {SearchScreenDesktop} from '#/view/screens/Search/Search' | ||||
| 
 | ||||
| export const SearchScreen = withAuthRequired(SearchScreenDesktop, { | ||||
|   isPublic: true, | ||||
| }) | ||||
| export const SearchScreen = SearchScreenDesktop | ||||
|  |  | |||
|  | @ -20,7 +20,6 @@ import { | |||
|   FontAwesomeIconStyle, | ||||
| } from '@fortawesome/react-native-fontawesome' | ||||
| import {NativeStackScreenProps, CommonNavigatorParams} from 'lib/routes/types' | ||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||
| import * as AppInfo from 'lib/app-info' | ||||
| import {s, colors} from 'lib/styles' | ||||
| import {ScrollView} from '../com/util/Views' | ||||
|  | @ -141,7 +140,7 @@ function SettingsAccountCard({account}: {account: SessionAccount}) { | |||
| } | ||||
| 
 | ||||
| type Props = NativeStackScreenProps<CommonNavigatorParams, 'Settings'> | ||||
| export const SettingsScreen = withAuthRequired(function Settings({}: Props) { | ||||
| export function SettingsScreen({}: Props) { | ||||
|   const queryClient = useQueryClient() | ||||
|   const colorMode = useColorMode() | ||||
|   const setColorMode = useSetColorMode() | ||||
|  | @ -731,7 +730,7 @@ export const SettingsScreen = withAuthRequired(function Settings({}: Props) { | |||
|       </ScrollView> | ||||
|     </View> | ||||
|   ) | ||||
| }) | ||||
| } | ||||
| 
 | ||||
| function EmailConfirmationNotice() { | ||||
|   const pal = usePalette('default') | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue