Add tags and mute words (#2968)
* Add bare minimum hashtags support (#2804) * Add bare minimum hashtags support As atproto/api already parses hashtags, this is as simple as hooking it up like link segments. This is "bare minimum" because: - Opening hashtag "#foo" is actually just a search for "foo" right now to work around #2491. - There is no integration in the composer. This hasn't stopped people from using hashtags already, and can be added later. - This change itself only had to hook things up - thank you for having already put the hashtag parsing in place. * Remove workaround for hash search not working now that it's fixed * Add RichTextTag and TagMenu * Sketch * Remove hackfix * Some cleanup * Sketch web * Mobile design * Mobile handling of tags search * Web only * Fix navigation woes * Use new callback * Hook it up * Integrate muted tags * Fix dropdown styles * Type error * Use close callback * Fix styles * Cleanup, install latest sdk * Quick muted words screen * Targets * Dir structure * Icons, list view * Move to dialog * Add removal confirmation * Swap copy * Improve checkboxees * Update matching, add tests * Moderate embeds * Create global dialogs concept again to prevent flashing * Add access from moderation screen * Highlight tags on native * Add web highlighting * Add close to web modal * Adjust close color * Rename toggles and adjust logic * Icon update * Load states * Improve regex * Improve regex * Improve regex * Revert link test * Hyphenated words * Improve matching * Enhance * Some tweaks * Muted words modal changes * Handle invalid handles, handle long tags * Remove main regex * Better test * Space/punct check drop to includes * Lowercase post text before comparison * Add better real world test case --------- Co-authored-by: Kisaragi Hiu <mail@kisaragi-hiu.com>
This commit is contained in:
		
							parent
							
								
									c8582924e2
								
							
						
					
					
						commit
						58aaad704a
					
				
					 49 changed files with 1983 additions and 39 deletions
				
			
		|  | @ -1,5 +1,6 @@ | |||
| import React from 'react' | ||||
| import {DialogControlProps} from '#/components/Dialog' | ||||
| import {Provider as GlobalDialogsProvider} from '#/components/dialogs/Context' | ||||
| 
 | ||||
| const DialogContext = React.createContext<{ | ||||
|   activeDialogs: React.MutableRefObject< | ||||
|  | @ -37,7 +38,7 @@ export function Provider({children}: React.PropsWithChildren<{}>) { | |||
|   return ( | ||||
|     <DialogContext.Provider value={context}> | ||||
|       <DialogControlContext.Provider value={controls}> | ||||
|         {children} | ||||
|         <GlobalDialogsProvider>{children}</GlobalDialogsProvider> | ||||
|       </DialogControlContext.Provider> | ||||
|     </DialogContext.Provider> | ||||
|   ) | ||||
|  |  | |||
|  | @ -49,4 +49,6 @@ export const DEFAULT_LOGGED_OUT_PREFERENCES: UsePreferencesQueryResponse = { | |||
|   threadViewPrefs: DEFAULT_THREAD_VIEW_PREFS, | ||||
|   userAge: 13, // TODO(pwi)
 | ||||
|   interests: {tags: []}, | ||||
|   mutedWords: [], | ||||
|   hiddenPosts: [], | ||||
| } | ||||
|  |  | |||
|  | @ -1,6 +1,10 @@ | |||
| import {useMemo} from 'react' | ||||
| import {useQuery, useMutation, useQueryClient} from '@tanstack/react-query' | ||||
| import {LabelPreference, BskyFeedViewPreference} from '@atproto/api' | ||||
| import { | ||||
|   LabelPreference, | ||||
|   BskyFeedViewPreference, | ||||
|   AppBskyActorDefs, | ||||
| } from '@atproto/api' | ||||
| 
 | ||||
| import {track} from '#/lib/analytics/analytics' | ||||
| import {getAge} from '#/lib/strings/time' | ||||
|  | @ -108,6 +112,7 @@ export function useModerationOpts() { | |||
|     return { | ||||
|       ...moderationOpts, | ||||
|       hiddenPosts, | ||||
|       mutedWords: prefs.data.mutedWords || [], | ||||
|     } | ||||
|   }, [currentAccount?.did, prefs.data, hiddenPosts]) | ||||
|   return opts | ||||
|  | @ -278,3 +283,45 @@ export function useUnpinFeedMutation() { | |||
|     }, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| export function useUpsertMutedWordsMutation() { | ||||
|   const queryClient = useQueryClient() | ||||
| 
 | ||||
|   return useMutation({ | ||||
|     mutationFn: async (mutedWords: AppBskyActorDefs.MutedWord[]) => { | ||||
|       await getAgent().upsertMutedWords(mutedWords) | ||||
|       // triggers a refetch
 | ||||
|       await queryClient.invalidateQueries({ | ||||
|         queryKey: preferencesQueryKey, | ||||
|       }) | ||||
|     }, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| export function useUpdateMutedWordMutation() { | ||||
|   const queryClient = useQueryClient() | ||||
| 
 | ||||
|   return useMutation({ | ||||
|     mutationFn: async (mutedWord: AppBskyActorDefs.MutedWord) => { | ||||
|       await getAgent().updateMutedWord(mutedWord) | ||||
|       // triggers a refetch
 | ||||
|       await queryClient.invalidateQueries({ | ||||
|         queryKey: preferencesQueryKey, | ||||
|       }) | ||||
|     }, | ||||
|   }) | ||||
| } | ||||
| 
 | ||||
| export function useRemoveMutedWordMutation() { | ||||
|   const queryClient = useQueryClient() | ||||
| 
 | ||||
|   return useMutation({ | ||||
|     mutationFn: async (mutedWord: AppBskyActorDefs.MutedWord) => { | ||||
|       await getAgent().removeMutedWord(mutedWord) | ||||
|       // triggers a refetch
 | ||||
|       await queryClient.invalidateQueries({ | ||||
|         queryKey: preferencesQueryKey, | ||||
|       }) | ||||
|     }, | ||||
|   }) | ||||
| } | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue