Improve usability of search on web (#3663)
* dont select the text on web * TODO REVERT THESE CHANGES * use `usethrottledvalue` for autocomplete * use `isFetching` from query * rm setTimeout * getting there * improve functionality of cancel button * rm todo * add comment back * encode `searchText` rather than `queryTerm` * use "back" on web in some cases * don't flash results in autocomplete * remove unnecesary usestate * rename everything to `query` temporarily * revert accidental lint * rm todo * rm comment * use `useFocusEffect` to update the query term on back navigation * `searchText` is always defined here * Fix race * remove back functionality * use `keepPreviousData` for query * rename `q` to `queryParam` * remove hack * remove `q=` on cancel * blur on submit * use `setParams` instead of `replace` * use `replace` on web still * clear the search input when we clear `q` on native * onPress dismiss attempt * Adjustments * Fix search history * Always hide autocomplete * Clear right pane search on select * `blur` on autosuggestion press * Rename to reduce diff * Fixes * Unify codepaths * Fixes * precache the autosuggestion * do the cache in the link card * Revert "precache the autosuggestion" This reverts commit 79c433e984621ba4231a2a4c4b3f4690b0516b4d. * use `throttledValue` and `keepPreviousData` in sidebar search * show spinner when fetching pt 1 * show spinner when fetching pt 2 * show spinner properly for autocomplete * Fix extra border * Position fixed * TS * Revert "TS" This reverts commit df187ea2d7a96d0f1832bc2392215f4d969a87c9. * Revert "Position fixed" This reverts commit 9c721c952b0fa4e5e4a23de38cab916ab13397e6. * Maybe fix iPad * Revert "TODO REVERT THESE CHANGES" This reverts commit 279f717f3091c9df8c73ba35f9a038e12f5a1122. * Rename var --------- Co-authored-by: Dan Abramov <dan.abramov@gmail.com>
This commit is contained in:
		
							parent
							
								
									d81a373d21
								
							
						
					
					
						commit
						5f9136479b
					
				
					 3 changed files with 155 additions and 171 deletions
				
			
		|  | @ -1,6 +1,6 @@ | ||||||
| import React from 'react' | import React from 'react' | ||||||
| import {AppBskyActorDefs, moderateProfile, ModerationOpts} from '@atproto/api' | import {AppBskyActorDefs, moderateProfile, ModerationOpts} from '@atproto/api' | ||||||
| import {useQuery, useQueryClient} from '@tanstack/react-query' | import {keepPreviousData, useQuery, useQueryClient} from '@tanstack/react-query' | ||||||
| 
 | 
 | ||||||
| import {isJustAMute} from '#/lib/moderation' | import {isJustAMute} from '#/lib/moderation' | ||||||
| import {logger} from '#/logger' | import {logger} from '#/logger' | ||||||
|  | @ -16,7 +16,10 @@ const DEFAULT_MOD_OPTS = { | ||||||
| const RQKEY_ROOT = 'actor-autocomplete' | const RQKEY_ROOT = 'actor-autocomplete' | ||||||
| export const RQKEY = (prefix: string) => [RQKEY_ROOT, prefix] | export const RQKEY = (prefix: string) => [RQKEY_ROOT, prefix] | ||||||
| 
 | 
 | ||||||
| export function useActorAutocompleteQuery(prefix: string) { | export function useActorAutocompleteQuery( | ||||||
|  |   prefix: string, | ||||||
|  |   maintainData?: boolean, | ||||||
|  | ) { | ||||||
|   const moderationOpts = useModerationOpts() |   const moderationOpts = useModerationOpts() | ||||||
|   const {getAgent} = useAgent() |   const {getAgent} = useAgent() | ||||||
| 
 | 
 | ||||||
|  | @ -40,6 +43,7 @@ export function useActorAutocompleteQuery(prefix: string) { | ||||||
|       }, |       }, | ||||||
|       [moderationOpts], |       [moderationOpts], | ||||||
|     ), |     ), | ||||||
|  |     placeholderData: maintainData ? keepPreviousData : undefined, | ||||||
|   }) |   }) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -27,7 +27,7 @@ import {s} from '#/lib/styles' | ||||||
| import {logger} from '#/logger' | import {logger} from '#/logger' | ||||||
| import {isNative, isWeb} from '#/platform/detection' | import {isNative, isWeb} from '#/platform/detection' | ||||||
| import {listenSoftReset} from '#/state/events' | import {listenSoftReset} from '#/state/events' | ||||||
| import {useActorAutocompleteFn} from '#/state/queries/actor-autocomplete' | import {useActorAutocompleteQuery} from '#/state/queries/actor-autocomplete' | ||||||
| import {useActorSearch} from '#/state/queries/actor-search' | import {useActorSearch} from '#/state/queries/actor-search' | ||||||
| import {useModerationOpts} from '#/state/queries/preferences' | import {useModerationOpts} from '#/state/queries/preferences' | ||||||
| import {useSearchPostsQuery} from '#/state/queries/search-posts' | import {useSearchPostsQuery} from '#/state/queries/search-posts' | ||||||
|  | @ -35,6 +35,7 @@ import {useSuggestedFollowsQuery} from '#/state/queries/suggested-follows' | ||||||
| import {useSession} from '#/state/session' | import {useSession} from '#/state/session' | ||||||
| import {useSetDrawerOpen} from '#/state/shell' | import {useSetDrawerOpen} from '#/state/shell' | ||||||
| import {useSetDrawerSwipeDisabled, useSetMinimalShellMode} from '#/state/shell' | import {useSetDrawerSwipeDisabled, useSetMinimalShellMode} from '#/state/shell' | ||||||
|  | import {useNonReactiveCallback} from 'lib/hooks/useNonReactiveCallback' | ||||||
| import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' | import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' | ||||||
| import { | import { | ||||||
|   NativeStackScreenProps, |   NativeStackScreenProps, | ||||||
|  | @ -308,7 +309,7 @@ function SearchScreenUserResults({ | ||||||
|   const {_} = useLingui() |   const {_} = useLingui() | ||||||
| 
 | 
 | ||||||
|   const {data: results, isFetched} = useActorSearch({ |   const {data: results, isFetched} = useActorSearch({ | ||||||
|     query, |     query: query, | ||||||
|     enabled: active, |     enabled: active, | ||||||
|   }) |   }) | ||||||
| 
 | 
 | ||||||
|  | @ -478,43 +479,25 @@ export function SearchScreen( | ||||||
|   const {track} = useAnalytics() |   const {track} = useAnalytics() | ||||||
|   const setDrawerOpen = useSetDrawerOpen() |   const setDrawerOpen = useSetDrawerOpen() | ||||||
|   const moderationOpts = useModerationOpts() |   const moderationOpts = useModerationOpts() | ||||||
|   const search = useActorAutocompleteFn() |  | ||||||
|   const setMinimalShellMode = useSetMinimalShellMode() |   const setMinimalShellMode = useSetMinimalShellMode() | ||||||
|   const {isTabletOrDesktop, isTabletOrMobile} = useWebMediaQueries() |   const {isTabletOrDesktop, isTabletOrMobile} = useWebMediaQueries() | ||||||
| 
 | 
 | ||||||
|   const searchDebounceTimeout = React.useRef<NodeJS.Timeout | undefined>( |   // Query terms
 | ||||||
|     undefined, |   const queryParam = props.route?.params?.q ?? '' | ||||||
|   ) |   const [searchText, setSearchText] = React.useState<string>(queryParam) | ||||||
|   const [isFetching, setIsFetching] = React.useState<boolean>(false) |   const {data: autocompleteData, isFetching: isAutocompleteFetching} = | ||||||
|   const [query, setQuery] = React.useState<string>(props.route?.params?.q || '') |     useActorAutocompleteQuery(searchText, true) | ||||||
|   const [searchResults, setSearchResults] = React.useState< | 
 | ||||||
|     AppBskyActorDefs.ProfileViewBasic[] |   const [showAutocomplete, setShowAutocomplete] = React.useState(false) | ||||||
|   >([]) |  | ||||||
|   const [inputIsFocused, setInputIsFocused] = React.useState(false) |  | ||||||
|   const [showAutocompleteResults, setShowAutocompleteResults] = |  | ||||||
|     React.useState(false) |  | ||||||
|   const [searchHistory, setSearchHistory] = React.useState<string[]>([]) |   const [searchHistory, setSearchHistory] = React.useState<string[]>([]) | ||||||
| 
 | 
 | ||||||
|   /** |   useFocusEffect( | ||||||
|    * The Search screen's `q` param |     useNonReactiveCallback(() => { | ||||||
|    */ |       if (isWeb) { | ||||||
|   const queryParam = props.route?.params?.q |         setSearchText(queryParam) | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * If `true`, this means we received new instructions from the router. This |  | ||||||
|    * is handled in a effect, and used to update the value of `query` locally |  | ||||||
|    * within this screen. |  | ||||||
|    */ |  | ||||||
|   const routeParamsMismatch = queryParam && queryParam !== query |  | ||||||
| 
 |  | ||||||
|   React.useEffect(() => { |  | ||||||
|     if (queryParam && routeParamsMismatch) { |  | ||||||
|       // reset immediately and let local state take over
 |  | ||||||
|       navigation.setParams({q: ''}) |  | ||||||
|       // update query for next search
 |  | ||||||
|       setQuery(queryParam) |  | ||||||
|       } |       } | ||||||
|   }, [queryParam, routeParamsMismatch, navigation]) |     }), | ||||||
|  |   ) | ||||||
| 
 | 
 | ||||||
|   React.useEffect(() => { |   React.useEffect(() => { | ||||||
|     const loadSearchHistory = async () => { |     const loadSearchHistory = async () => { | ||||||
|  | @ -536,60 +519,45 @@ export function SearchScreen( | ||||||
|     setDrawerOpen(true) |     setDrawerOpen(true) | ||||||
|   }, [track, setDrawerOpen]) |   }, [track, setDrawerOpen]) | ||||||
| 
 | 
 | ||||||
|   const onPressCancelSearch = React.useCallback(() => { |  | ||||||
|     scrollToTopWeb() |  | ||||||
|     textInput.current?.blur() |  | ||||||
|     setQuery('') |  | ||||||
|     setShowAutocompleteResults(false) |  | ||||||
|     if (searchDebounceTimeout.current) |  | ||||||
|       clearTimeout(searchDebounceTimeout.current) |  | ||||||
|   }, [textInput]) |  | ||||||
| 
 |  | ||||||
|   const onPressClearQuery = React.useCallback(() => { |   const onPressClearQuery = React.useCallback(() => { | ||||||
|     scrollToTopWeb() |     scrollToTopWeb() | ||||||
|     setQuery('') |     setSearchText('') | ||||||
|     setShowAutocompleteResults(false) |     textInput.current?.focus() | ||||||
|   }, [setQuery]) |   }, []) | ||||||
| 
 | 
 | ||||||
|   const onChangeText = React.useCallback( |   const onPressCancelSearch = React.useCallback(() => { | ||||||
|     async (text: string) => { |  | ||||||
|     scrollToTopWeb() |     scrollToTopWeb() | ||||||
| 
 | 
 | ||||||
|       setQuery(text) |     if (showAutocomplete) { | ||||||
| 
 |       textInput.current?.blur() | ||||||
|       if (text.length > 0) { |       setShowAutocomplete(false) | ||||||
|         setIsFetching(true) |       setSearchText(queryParam) | ||||||
|         setShowAutocompleteResults(true) |  | ||||||
| 
 |  | ||||||
|         if (searchDebounceTimeout.current) { |  | ||||||
|           clearTimeout(searchDebounceTimeout.current) |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         searchDebounceTimeout.current = setTimeout(async () => { |  | ||||||
|           const results = await search({query: text, limit: 30}) |  | ||||||
| 
 |  | ||||||
|           if (results) { |  | ||||||
|             setSearchResults(results) |  | ||||||
|             setIsFetching(false) |  | ||||||
|           } |  | ||||||
|         }, 300) |  | ||||||
|     } else { |     } else { | ||||||
|         if (searchDebounceTimeout.current) { |       // If we just `setParams` and set `q` to an empty string, the URL still displays `q=`, which isn't pretty.
 | ||||||
|           clearTimeout(searchDebounceTimeout.current) |       // However, `.replace()` on native has a "push" animation that we don't want. So we need to handle these
 | ||||||
|  |       // differently.
 | ||||||
|  |       if (isWeb) { | ||||||
|  |         navigation.replace('Search', {}) | ||||||
|  |       } else { | ||||||
|  |         setSearchText('') | ||||||
|  |         navigation.setParams({q: ''}) | ||||||
|       } |       } | ||||||
|         setSearchResults([]) |  | ||||||
|         setIsFetching(false) |  | ||||||
|         setShowAutocompleteResults(false) |  | ||||||
|     } |     } | ||||||
|     }, |   }, [showAutocomplete, navigation, queryParam]) | ||||||
|     [setQuery, search, setSearchResults], | 
 | ||||||
|   ) |   const onChangeText = React.useCallback(async (text: string) => { | ||||||
|  |     scrollToTopWeb() | ||||||
|  |     setSearchText(text) | ||||||
|  |   }, []) | ||||||
| 
 | 
 | ||||||
|   const updateSearchHistory = React.useCallback( |   const updateSearchHistory = React.useCallback( | ||||||
|     async (newQuery: string) => { |     async (newQuery: string) => { | ||||||
|       newQuery = newQuery.trim() |       newQuery = newQuery.trim() | ||||||
|       if (newQuery && !searchHistory.includes(newQuery)) { |       if (newQuery) { | ||||||
|         let newHistory = [newQuery, ...searchHistory] |         let newHistory = [ | ||||||
|  |           newQuery, | ||||||
|  |           ...searchHistory.filter(q => q !== newQuery), | ||||||
|  |         ] | ||||||
| 
 | 
 | ||||||
|         if (newHistory.length > 5) { |         if (newHistory.length > 5) { | ||||||
|           newHistory = newHistory.slice(0, 5) |           newHistory = newHistory.slice(0, 5) | ||||||
|  | @ -609,11 +577,30 @@ export function SearchScreen( | ||||||
|     [searchHistory, setSearchHistory], |     [searchHistory, setSearchHistory], | ||||||
|   ) |   ) | ||||||
| 
 | 
 | ||||||
|   const onSubmit = React.useCallback(() => { |   const navigateToItem = React.useCallback( | ||||||
|  |     (item: string) => { | ||||||
|       scrollToTopWeb() |       scrollToTopWeb() | ||||||
|     setShowAutocompleteResults(false) |       setShowAutocomplete(false) | ||||||
|     updateSearchHistory(query) |       updateSearchHistory(item) | ||||||
|   }, [query, setShowAutocompleteResults, updateSearchHistory]) | 
 | ||||||
|  |       if (isWeb) { | ||||||
|  |         navigation.push('Search', {q: item}) | ||||||
|  |       } else { | ||||||
|  |         textInput.current?.blur() | ||||||
|  |         navigation.setParams({q: item}) | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     [updateSearchHistory, navigation], | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  |   const onSubmit = React.useCallback(() => { | ||||||
|  |     navigateToItem(searchText) | ||||||
|  |   }, [navigateToItem, searchText]) | ||||||
|  | 
 | ||||||
|  |   const handleHistoryItemClick = (item: string) => { | ||||||
|  |     setSearchText(item) | ||||||
|  |     navigateToItem(item) | ||||||
|  |   } | ||||||
| 
 | 
 | ||||||
|   const onSoftReset = React.useCallback(() => { |   const onSoftReset = React.useCallback(() => { | ||||||
|     scrollToTopWeb() |     scrollToTopWeb() | ||||||
|  | @ -621,9 +608,9 @@ export function SearchScreen( | ||||||
|   }, [onPressCancelSearch]) |   }, [onPressCancelSearch]) | ||||||
| 
 | 
 | ||||||
|   const queryMaybeHandle = React.useMemo(() => { |   const queryMaybeHandle = React.useMemo(() => { | ||||||
|     const match = MATCH_HANDLE.exec(query) |     const match = MATCH_HANDLE.exec(queryParam) | ||||||
|     return match && match[1] |     return match && match[1] | ||||||
|   }, [query]) |   }, [queryParam]) | ||||||
| 
 | 
 | ||||||
|   useFocusEffect( |   useFocusEffect( | ||||||
|     React.useCallback(() => { |     React.useCallback(() => { | ||||||
|  | @ -632,11 +619,6 @@ export function SearchScreen( | ||||||
|     }, [onSoftReset, setMinimalShellMode]), |     }, [onSoftReset, setMinimalShellMode]), | ||||||
|   ) |   ) | ||||||
| 
 | 
 | ||||||
|   const handleHistoryItemClick = (item: React.SetStateAction<string>) => { |  | ||||||
|     setQuery(item) |  | ||||||
|     onSubmit() |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   const handleRemoveHistoryItem = (itemToRemove: string) => { |   const handleRemoveHistoryItem = (itemToRemove: string) => { | ||||||
|     const updatedHistory = searchHistory.filter(item => item !== itemToRemove) |     const updatedHistory = searchHistory.filter(item => item !== itemToRemove) | ||||||
|     setSearchHistory(updatedHistory) |     setSearchHistory(updatedHistory) | ||||||
|  | @ -688,17 +670,21 @@ export function SearchScreen( | ||||||
|             ref={textInput} |             ref={textInput} | ||||||
|             placeholder={_(msg`Search`)} |             placeholder={_(msg`Search`)} | ||||||
|             placeholderTextColor={pal.colors.textLight} |             placeholderTextColor={pal.colors.textLight} | ||||||
|             selectTextOnFocus |             selectTextOnFocus={isNative} | ||||||
|             returnKeyType="search" |             returnKeyType="search" | ||||||
|             value={query} |             value={searchText} | ||||||
|             style={[pal.text, styles.headerSearchInput]} |             style={[pal.text, styles.headerSearchInput]} | ||||||
|             keyboardAppearance={theme.colorScheme} |             keyboardAppearance={theme.colorScheme} | ||||||
|             onFocus={() => setInputIsFocused(true)} |             onFocus={() => { | ||||||
|             onBlur={() => { |               if (isWeb) { | ||||||
|               // HACK
 |                 // Prevent a jump on iPad by ensuring that
 | ||||||
|               // give 100ms to not stop click handlers in the search history
 |                 // the initial focused render has no result list.
 | ||||||
|               // -prf
 |                 requestAnimationFrame(() => { | ||||||
|               setTimeout(() => setInputIsFocused(false), 100) |                   setShowAutocomplete(true) | ||||||
|  |                 }) | ||||||
|  |               } else { | ||||||
|  |                 setShowAutocomplete(true) | ||||||
|  |               } | ||||||
|             }} |             }} | ||||||
|             onChangeText={onChangeText} |             onChangeText={onChangeText} | ||||||
|             onSubmitEditing={onSubmit} |             onSubmitEditing={onSubmit} | ||||||
|  | @ -710,7 +696,7 @@ export function SearchScreen( | ||||||
|             autoComplete="off" |             autoComplete="off" | ||||||
|             autoCapitalize="none" |             autoCapitalize="none" | ||||||
|           /> |           /> | ||||||
|           {query ? ( |           {showAutocomplete ? ( | ||||||
|             <Pressable |             <Pressable | ||||||
|               testID="searchTextInputClearBtn" |               testID="searchTextInputClearBtn" | ||||||
|               onPress={onPressClearQuery} |               onPress={onPressClearQuery} | ||||||
|  | @ -727,7 +713,7 @@ export function SearchScreen( | ||||||
|           ) : undefined} |           ) : undefined} | ||||||
|         </View> |         </View> | ||||||
| 
 | 
 | ||||||
|         {query || inputIsFocused ? ( |         {(queryParam || showAutocomplete) && ( | ||||||
|           <View style={styles.headerCancelBtn}> |           <View style={styles.headerCancelBtn}> | ||||||
|             <Pressable |             <Pressable | ||||||
|               onPress={onPressCancelSearch} |               onPress={onPressCancelSearch} | ||||||
|  | @ -738,12 +724,13 @@ export function SearchScreen( | ||||||
|               </Text> |               </Text> | ||||||
|             </Pressable> |             </Pressable> | ||||||
|           </View> |           </View> | ||||||
|         ) : undefined} |         )} | ||||||
|       </CenteredView> |       </CenteredView> | ||||||
| 
 | 
 | ||||||
|       {showAutocompleteResults ? ( |       {showAutocomplete && searchText.length > 0 ? ( | ||||||
|         <> |         <> | ||||||
|           {isFetching || !moderationOpts ? ( |           {(isAutocompleteFetching && !autocompleteData?.length) || | ||||||
|  |           !moderationOpts ? ( | ||||||
|             <Loader /> |             <Loader /> | ||||||
|           ) : ( |           ) : ( | ||||||
|             <ScrollView |             <ScrollView | ||||||
|  | @ -753,12 +740,12 @@ export function SearchScreen( | ||||||
|               keyboardShouldPersistTaps="handled" |               keyboardShouldPersistTaps="handled" | ||||||
|               keyboardDismissMode="on-drag"> |               keyboardDismissMode="on-drag"> | ||||||
|               <SearchLinkCard |               <SearchLinkCard | ||||||
|                 label={_(msg`Search for "${query}"`)} |                 label={_(msg`Search for "${searchText}"`)} | ||||||
|                 onPress={isNative ? onSubmit : undefined} |                 onPress={isNative ? onSubmit : undefined} | ||||||
|                 to={ |                 to={ | ||||||
|                   isNative |                   isNative | ||||||
|                     ? undefined |                     ? undefined | ||||||
|                     : `/search?q=${encodeURIComponent(query)}` |                     : `/search?q=${encodeURIComponent(searchText)}` | ||||||
|                 } |                 } | ||||||
|                 style={{borderBottomWidth: 1}} |                 style={{borderBottomWidth: 1}} | ||||||
|               /> |               /> | ||||||
|  | @ -770,11 +757,18 @@ export function SearchScreen( | ||||||
|                 /> |                 /> | ||||||
|               ) : null} |               ) : null} | ||||||
| 
 | 
 | ||||||
|               {searchResults.map(item => ( |               {autocompleteData?.map(item => ( | ||||||
|                 <SearchProfileCard |                 <SearchProfileCard | ||||||
|                   key={item.did} |                   key={item.did} | ||||||
|                   profile={item} |                   profile={item} | ||||||
|                   moderation={moderateProfile(item, moderationOpts)} |                   moderation={moderateProfile(item, moderationOpts)} | ||||||
|  |                   onPress={() => { | ||||||
|  |                     if (isWeb) { | ||||||
|  |                       setShowAutocomplete(false) | ||||||
|  |                     } else { | ||||||
|  |                       textInput.current?.blur() | ||||||
|  |                     } | ||||||
|  |                   }} | ||||||
|                 /> |                 /> | ||||||
|               ))} |               ))} | ||||||
| 
 | 
 | ||||||
|  | @ -782,7 +776,7 @@ export function SearchScreen( | ||||||
|             </ScrollView> |             </ScrollView> | ||||||
|           )} |           )} | ||||||
|         </> |         </> | ||||||
|       ) : !query && inputIsFocused ? ( |       ) : !queryParam && showAutocomplete ? ( | ||||||
|         <CenteredView |         <CenteredView | ||||||
|           sideBorders={isTabletOrDesktop} |           sideBorders={isTabletOrDesktop} | ||||||
|           // @ts-ignore web only -prf
 |           // @ts-ignore web only -prf
 | ||||||
|  | @ -826,10 +820,8 @@ export function SearchScreen( | ||||||
|             )} |             )} | ||||||
|           </View> |           </View> | ||||||
|         </CenteredView> |         </CenteredView> | ||||||
|       ) : routeParamsMismatch ? ( |  | ||||||
|         <ActivityIndicator /> |  | ||||||
|       ) : ( |       ) : ( | ||||||
|         <SearchScreenInner query={query} /> |         <SearchScreenInner query={queryParam} /> | ||||||
|       )} |       )} | ||||||
|     </View> |     </View> | ||||||
|   ) |   ) | ||||||
|  |  | ||||||
|  | @ -1,33 +1,35 @@ | ||||||
| import React from 'react' | import React from 'react' | ||||||
| import { | import { | ||||||
|   ViewStyle, |  | ||||||
|   TextInput, |  | ||||||
|   View, |  | ||||||
|   StyleSheet, |  | ||||||
|   TouchableOpacity, |  | ||||||
|   ActivityIndicator, |   ActivityIndicator, | ||||||
|  |   StyleSheet, | ||||||
|  |   TextInput, | ||||||
|  |   TouchableOpacity, | ||||||
|  |   View, | ||||||
|  |   ViewStyle, | ||||||
| } from 'react-native' | } from 'react-native' | ||||||
| import {useNavigation, StackActions} from '@react-navigation/native' |  | ||||||
| import { | import { | ||||||
|   AppBskyActorDefs, |   AppBskyActorDefs, | ||||||
|   moderateProfile, |   moderateProfile, | ||||||
|   ModerationDecision, |   ModerationDecision, | ||||||
| } from '@atproto/api' | } from '@atproto/api' | ||||||
| import {Trans, msg} from '@lingui/macro' | import {msg, Trans} from '@lingui/macro' | ||||||
| import {useLingui} from '@lingui/react' | import {useLingui} from '@lingui/react' | ||||||
|  | import {StackActions, useNavigation} from '@react-navigation/native' | ||||||
|  | import {useQueryClient} from '@tanstack/react-query' | ||||||
| 
 | 
 | ||||||
| import {s} from '#/lib/styles' | import {makeProfileLink} from '#/lib/routes/links' | ||||||
| import {sanitizeDisplayName} from '#/lib/strings/display-names' | import {sanitizeDisplayName} from '#/lib/strings/display-names' | ||||||
| import {sanitizeHandle} from '#/lib/strings/handles' | import {sanitizeHandle} from '#/lib/strings/handles' | ||||||
| import {makeProfileLink} from '#/lib/routes/links' | import {s} from '#/lib/styles' | ||||||
| import {Link} from '#/view/com/util/Link' | import {useActorAutocompleteQuery} from '#/state/queries/actor-autocomplete' | ||||||
|  | import {useModerationOpts} from '#/state/queries/preferences' | ||||||
| import {usePalette} from 'lib/hooks/usePalette' | import {usePalette} from 'lib/hooks/usePalette' | ||||||
| import {MagnifyingGlassIcon2} from 'lib/icons' | import {MagnifyingGlassIcon2} from 'lib/icons' | ||||||
| import {NavigationProp} from 'lib/routes/types' | import {NavigationProp} from 'lib/routes/types' | ||||||
| import {Text} from 'view/com/util/text/Text' | import {precacheProfile} from 'state/queries/profile' | ||||||
|  | import {Link} from '#/view/com/util/Link' | ||||||
| import {UserAvatar} from '#/view/com/util/UserAvatar' | import {UserAvatar} from '#/view/com/util/UserAvatar' | ||||||
| import {useActorAutocompleteFn} from '#/state/queries/actor-autocomplete' | import {Text} from 'view/com/util/text/Text' | ||||||
| import {useModerationOpts} from '#/state/queries/preferences' |  | ||||||
| 
 | 
 | ||||||
| export const MATCH_HANDLE = | export const MATCH_HANDLE = | ||||||
|   /@?([a-zA-Z0-9-]+(?:\.[a-zA-Z0-9-]+)*(?:\.[a-zA-Z]{2,}))/ |   /@?([a-zA-Z0-9-]+(?:\.[a-zA-Z0-9-]+)*(?:\.[a-zA-Z]{2,}))/ | ||||||
|  | @ -84,11 +86,19 @@ export function SearchLinkCard({ | ||||||
| export function SearchProfileCard({ | export function SearchProfileCard({ | ||||||
|   profile, |   profile, | ||||||
|   moderation, |   moderation, | ||||||
|  |   onPress: onPressInner, | ||||||
| }: { | }: { | ||||||
|   profile: AppBskyActorDefs.ProfileViewBasic |   profile: AppBskyActorDefs.ProfileViewBasic | ||||||
|   moderation: ModerationDecision |   moderation: ModerationDecision | ||||||
|  |   onPress: () => void | ||||||
| }) { | }) { | ||||||
|   const pal = usePalette('default') |   const pal = usePalette('default') | ||||||
|  |   const queryClient = useQueryClient() | ||||||
|  | 
 | ||||||
|  |   const onPress = React.useCallback(() => { | ||||||
|  |     precacheProfile(queryClient, profile) | ||||||
|  |     onPressInner() | ||||||
|  |   }, [queryClient, profile, onPressInner]) | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <Link |     <Link | ||||||
|  | @ -96,7 +106,8 @@ export function SearchProfileCard({ | ||||||
|       href={makeProfileLink(profile)} |       href={makeProfileLink(profile)} | ||||||
|       title={profile.handle} |       title={profile.handle} | ||||||
|       asAnchor |       asAnchor | ||||||
|       anchorNoUnderline> |       anchorNoUnderline | ||||||
|  |       onBeforePress={onPress}> | ||||||
|       <View |       <View | ||||||
|         style={[ |         style={[ | ||||||
|           pal.border, |           pal.border, | ||||||
|  | @ -138,63 +149,35 @@ export function DesktopSearch() { | ||||||
|   const {_} = useLingui() |   const {_} = useLingui() | ||||||
|   const pal = usePalette('default') |   const pal = usePalette('default') | ||||||
|   const navigation = useNavigation<NavigationProp>() |   const navigation = useNavigation<NavigationProp>() | ||||||
|   const searchDebounceTimeout = React.useRef<NodeJS.Timeout | undefined>( |  | ||||||
|     undefined, |  | ||||||
|   ) |  | ||||||
|   const [isActive, setIsActive] = React.useState<boolean>(false) |   const [isActive, setIsActive] = React.useState<boolean>(false) | ||||||
|   const [isFetching, setIsFetching] = React.useState<boolean>(false) |  | ||||||
|   const [query, setQuery] = React.useState<string>('') |   const [query, setQuery] = React.useState<string>('') | ||||||
|   const [searchResults, setSearchResults] = React.useState< |   const {data: autocompleteData, isFetching} = useActorAutocompleteQuery( | ||||||
|     AppBskyActorDefs.ProfileViewBasic[] |     query, | ||||||
|   >([]) |     true, | ||||||
|  |   ) | ||||||
| 
 | 
 | ||||||
|   const moderationOpts = useModerationOpts() |   const moderationOpts = useModerationOpts() | ||||||
|   const search = useActorAutocompleteFn() |  | ||||||
| 
 | 
 | ||||||
|   const onChangeText = React.useCallback( |   const onChangeText = React.useCallback((text: string) => { | ||||||
|     async (text: string) => { |  | ||||||
|     setQuery(text) |     setQuery(text) | ||||||
| 
 |     setIsActive(text.length > 0) | ||||||
|       if (text.length > 0) { |   }, []) | ||||||
|         setIsFetching(true) |  | ||||||
|         setIsActive(true) |  | ||||||
| 
 |  | ||||||
|         if (searchDebounceTimeout.current) |  | ||||||
|           clearTimeout(searchDebounceTimeout.current) |  | ||||||
| 
 |  | ||||||
|         searchDebounceTimeout.current = setTimeout(async () => { |  | ||||||
|           const results = await search({query: text}) |  | ||||||
| 
 |  | ||||||
|           if (results) { |  | ||||||
|             setSearchResults(results) |  | ||||||
|             setIsFetching(false) |  | ||||||
|           } |  | ||||||
|         }, 300) |  | ||||||
|       } else { |  | ||||||
|         if (searchDebounceTimeout.current) |  | ||||||
|           clearTimeout(searchDebounceTimeout.current) |  | ||||||
|         setSearchResults([]) |  | ||||||
|         setIsFetching(false) |  | ||||||
|         setIsActive(false) |  | ||||||
|       } |  | ||||||
|     }, |  | ||||||
|     [setQuery, search, setSearchResults], |  | ||||||
|   ) |  | ||||||
| 
 | 
 | ||||||
|   const onPressCancelSearch = React.useCallback(() => { |   const onPressCancelSearch = React.useCallback(() => { | ||||||
|     setQuery('') |     setQuery('') | ||||||
|     setIsActive(false) |     setIsActive(false) | ||||||
|     if (searchDebounceTimeout.current) |  | ||||||
|       clearTimeout(searchDebounceTimeout.current) |  | ||||||
|   }, [setQuery]) |   }, [setQuery]) | ||||||
|  | 
 | ||||||
|   const onSubmit = React.useCallback(() => { |   const onSubmit = React.useCallback(() => { | ||||||
|     setIsActive(false) |     setIsActive(false) | ||||||
|     if (!query.length) return |     if (!query.length) return | ||||||
|     setSearchResults([]) |  | ||||||
|     if (searchDebounceTimeout.current) |  | ||||||
|       clearTimeout(searchDebounceTimeout.current) |  | ||||||
|     navigation.dispatch(StackActions.push('Search', {q: query})) |     navigation.dispatch(StackActions.push('Search', {q: query})) | ||||||
|   }, [query, navigation, setSearchResults]) |   }, [query, navigation]) | ||||||
|  | 
 | ||||||
|  |   const onSearchProfileCardPress = React.useCallback(() => { | ||||||
|  |     setQuery('') | ||||||
|  |     setIsActive(false) | ||||||
|  |   }, []) | ||||||
| 
 | 
 | ||||||
|   const queryMaybeHandle = React.useMemo(() => { |   const queryMaybeHandle = React.useMemo(() => { | ||||||
|     const match = MATCH_HANDLE.exec(query) |     const match = MATCH_HANDLE.exec(query) | ||||||
|  | @ -246,7 +229,7 @@ export function DesktopSearch() { | ||||||
| 
 | 
 | ||||||
|       {query !== '' && isActive && moderationOpts && ( |       {query !== '' && isActive && moderationOpts && ( | ||||||
|         <View style={[pal.view, pal.borderDark, styles.resultsContainer]}> |         <View style={[pal.view, pal.borderDark, styles.resultsContainer]}> | ||||||
|           {isFetching ? ( |           {isFetching && !autocompleteData?.length ? ( | ||||||
|             <View style={{padding: 8}}> |             <View style={{padding: 8}}> | ||||||
|               <ActivityIndicator /> |               <ActivityIndicator /> | ||||||
|             </View> |             </View> | ||||||
|  | @ -255,7 +238,11 @@ export function DesktopSearch() { | ||||||
|               <SearchLinkCard |               <SearchLinkCard | ||||||
|                 label={_(msg`Search for "${query}"`)} |                 label={_(msg`Search for "${query}"`)} | ||||||
|                 to={`/search?q=${encodeURIComponent(query)}`} |                 to={`/search?q=${encodeURIComponent(query)}`} | ||||||
|                 style={{borderBottomWidth: 1}} |                 style={ | ||||||
|  |                   queryMaybeHandle || (autocompleteData?.length ?? 0) > 0 | ||||||
|  |                     ? {borderBottomWidth: 1} | ||||||
|  |                     : undefined | ||||||
|  |                 } | ||||||
|               /> |               /> | ||||||
| 
 | 
 | ||||||
|               {queryMaybeHandle ? ( |               {queryMaybeHandle ? ( | ||||||
|  | @ -265,11 +252,12 @@ export function DesktopSearch() { | ||||||
|                 /> |                 /> | ||||||
|               ) : null} |               ) : null} | ||||||
| 
 | 
 | ||||||
|               {searchResults.map(item => ( |               {autocompleteData?.map(item => ( | ||||||
|                 <SearchProfileCard |                 <SearchProfileCard | ||||||
|                   key={item.did} |                   key={item.did} | ||||||
|                   profile={item} |                   profile={item} | ||||||
|                   moderation={moderateProfile(item, moderationOpts)} |                   moderation={moderateProfile(item, moderationOpts)} | ||||||
|  |                   onPress={onSearchProfileCardPress} | ||||||
|                 /> |                 /> | ||||||
|               ))} |               ))} | ||||||
|             </> |             </> | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue