import {useCallback} from 'react' import {Image as RNImage} from 'react-native-image-crop-picker' import { AppBskyActorDefs, AppBskyActorGetProfile, AppBskyActorProfile, AtUri, BskyAgent, ComAtprotoRepoUploadBlob, } from '@atproto/api' import { QueryClient, useMutation, useQuery, useQueryClient, } from '@tanstack/react-query' import {track} from '#/lib/analytics/analytics' import {uploadBlob} from '#/lib/api' import {until} from '#/lib/async/until' import {useToggleMutationQueue} from '#/lib/hooks/useToggleMutationQueue' import {logEvent, LogEvents, toClout} from '#/lib/statsig/statsig' import {Shadow} from '#/state/cache/types' import {STALE} from '#/state/queries' import {resetProfilePostsQueries} from '#/state/queries/post-feed' import {updateProfileShadow} from '../cache/profile-shadow' import {useAgent, useSession} from '../session' import { ProgressGuideAction, useProgressGuideControls, } from '../shell/progress-guide' import {RQKEY as RQKEY_LIST_CONVOS} from './messages/list-converations' import {RQKEY as RQKEY_MY_BLOCKED} from './my-blocked-accounts' import {RQKEY as RQKEY_MY_MUTED} from './my-muted-accounts' const RQKEY_ROOT = 'profile' export const RQKEY = (did: string) => [RQKEY_ROOT, did] const profilesQueryKeyRoot = 'profiles' export const profilesQueryKey = (handles: string[]) => [ profilesQueryKeyRoot, handles, ] const profileBasicQueryKeyRoot = 'profileBasic' export const profileBasicQueryKey = (didOrHandle: string) => [ profileBasicQueryKeyRoot, didOrHandle, ] export function useProfileQuery({ did, staleTime = STALE.SECONDS.FIFTEEN, }: { did: string | undefined staleTime?: number }) { const queryClient = useQueryClient() const agent = useAgent() return useQuery({ // WARNING // this staleTime is load-bearing // if you remove it, the UI infinite-loops // -prf staleTime, refetchOnWindowFocus: true, queryKey: RQKEY(did ?? ''), queryFn: async () => { const res = await agent.getProfile({actor: did ?? ''}) return res.data }, placeholderData: () => { if (!did) return return queryClient.getQueryData( profileBasicQueryKey(did), ) }, enabled: !!did, }) } export function useProfilesQuery({handles}: {handles: string[]}) { const agent = useAgent() return useQuery({ staleTime: STALE.MINUTES.FIVE, queryKey: profilesQueryKey(handles), queryFn: async () => { const res = await agent.getProfiles({actors: handles}) return res.data }, }) } export function usePrefetchProfileQuery() { const agent = useAgent() const queryClient = useQueryClient() const prefetchProfileQuery = useCallback( async (did: string) => { await queryClient.prefetchQuery({ staleTime: STALE.SECONDS.THIRTY, queryKey: RQKEY(did), queryFn: async () => { const res = await agent.getProfile({actor: did || ''}) return res.data }, }) }, [queryClient, agent], ) return prefetchProfileQuery } interface ProfileUpdateParams { profile: AppBskyActorDefs.ProfileView updates: | AppBskyActorProfile.Record | ((existing: AppBskyActorProfile.Record) => AppBskyActorProfile.Record) newUserAvatar?: RNImage | undefined | null newUserBanner?: RNImage | undefined | null checkCommitted?: (res: AppBskyActorGetProfile.Response) => boolean } export function useProfileUpdateMutation() { const queryClient = useQueryClient() const agent = useAgent() return useMutation({ mutationFn: async ({ profile, updates, newUserAvatar, newUserBanner, checkCommitted, }) => { let newUserAvatarPromise: | Promise | undefined if (newUserAvatar) { newUserAvatarPromise = uploadBlob( agent, newUserAvatar.path, newUserAvatar.mime, ) } let newUserBannerPromise: | Promise | undefined if (newUserBanner) { newUserBannerPromise = uploadBlob( agent, newUserBanner.path, newUserBanner.mime, ) } await agent.upsertProfile(async existing => { existing = existing || {} if (typeof updates === 'function') { existing = updates(existing) } else { existing.displayName = updates.displayName existing.description = updates.description } if (newUserAvatarPromise) { const res = await newUserAvatarPromise existing.avatar = res.data.blob } else if (newUserAvatar === null) { existing.avatar = undefined } if (newUserBannerPromise) { const res = await newUserBannerPromise existing.banner = res.data.blob } else if (newUserBanner === null) { existing.banner = undefined } return existing }) await whenAppViewReady( agent, profile.did, checkCommitted || (res => { if (typeof newUserAvatar !== 'undefined') { if (newUserAvatar === null && res.data.avatar) { // url hasnt cleared yet return false } else if (res.data.avatar === profile.avatar) { // url hasnt changed yet return false } } if (typeof newUserBanner !== 'undefined') { if (newUserBanner === null && res.data.banner) { // url hasnt cleared yet return false } else if (res.data.banner === profile.banner) { // url hasnt changed yet return false } } if (typeof updates === 'function') { return true } return ( res.data.displayName === updates.displayName && res.data.description === updates.description ) }), ) }, onSuccess(data, variables) { // invalidate cache queryClient.invalidateQueries({ queryKey: RQKEY(variables.profile.did), }) }, }) } export function useProfileFollowMutationQueue( profile: Shadow, logContext: LogEvents['profile:follow']['logContext'] & LogEvents['profile:unfollow']['logContext'], ) { const queryClient = useQueryClient() const did = profile.did const initialFollowingUri = profile.viewer?.following const followMutation = useProfileFollowMutation(logContext, profile) const unfollowMutation = useProfileUnfollowMutation(logContext) const queueToggle = useToggleMutationQueue({ initialState: initialFollowingUri, runMutation: async (prevFollowingUri, shouldFollow) => { if (shouldFollow) { const {uri} = await followMutation.mutateAsync({ did, }) return uri } else { if (prevFollowingUri) { await unfollowMutation.mutateAsync({ did, followUri: prevFollowingUri, }) } return undefined } }, onSuccess(finalFollowingUri) { // finalize updateProfileShadow(queryClient, did, { followingUri: finalFollowingUri, }) }, }) const queueFollow = useCallback(() => { // optimistically update updateProfileShadow(queryClient, did, { followingUri: 'pending', }) return queueToggle(true) }, [queryClient, did, queueToggle]) const queueUnfollow = useCallback(() => { // optimistically update updateProfileShadow(queryClient, did, { followingUri: undefined, }) return queueToggle(false) }, [queryClient, did, queueToggle]) return [queueFollow, queueUnfollow] } function useProfileFollowMutation( logContext: LogEvents['profile:follow']['logContext'], profile: Shadow, ) { const {currentAccount} = useSession() const agent = useAgent() const queryClient = useQueryClient() const {captureAction} = useProgressGuideControls() return useMutation<{uri: string; cid: string}, Error, {did: string}>({ mutationFn: async ({did}) => { let ownProfile: AppBskyActorDefs.ProfileViewDetailed | undefined if (currentAccount) { ownProfile = findProfileQueryData(queryClient, currentAccount.did) } captureAction(ProgressGuideAction.Follow) logEvent('profile:follow', { logContext, didBecomeMutual: profile.viewer ? Boolean(profile.viewer.followedBy) : undefined, followeeClout: toClout(profile.followersCount), followerClout: toClout(ownProfile?.followersCount), }) return await agent.follow(did) }, onSuccess(data, variables) { track('Profile:Follow', {username: variables.did}) }, }) } function useProfileUnfollowMutation( logContext: LogEvents['profile:unfollow']['logContext'], ) { const agent = useAgent() return useMutation({ mutationFn: async ({followUri}) => { logEvent('profile:unfollow', {logContext}) track('Profile:Unfollow', {username: followUri}) return await agent.deleteFollow(followUri) }, }) } export function useProfileMuteMutationQueue( profile: Shadow, ) { const queryClient = useQueryClient() const did = profile.did const initialMuted = profile.viewer?.muted const muteMutation = useProfileMuteMutation() const unmuteMutation = useProfileUnmuteMutation() const queueToggle = useToggleMutationQueue({ initialState: initialMuted, runMutation: async (_prevMuted, shouldMute) => { if (shouldMute) { await muteMutation.mutateAsync({ did, }) return true } else { await unmuteMutation.mutateAsync({ did, }) return false } }, onSuccess(finalMuted) { // finalize updateProfileShadow(queryClient, did, {muted: finalMuted}) }, }) const queueMute = useCallback(() => { // optimistically update updateProfileShadow(queryClient, did, { muted: true, }) return queueToggle(true) }, [queryClient, did, queueToggle]) const queueUnmute = useCallback(() => { // optimistically update updateProfileShadow(queryClient, did, { muted: false, }) return queueToggle(false) }, [queryClient, did, queueToggle]) return [queueMute, queueUnmute] } function useProfileMuteMutation() { const queryClient = useQueryClient() const agent = useAgent() return useMutation({ mutationFn: async ({did}) => { await agent.mute(did) }, onSuccess() { queryClient.invalidateQueries({queryKey: RQKEY_MY_MUTED()}) }, }) } function useProfileUnmuteMutation() { const queryClient = useQueryClient() const agent = useAgent() return useMutation({ mutationFn: async ({did}) => { await agent.unmute(did) }, onSuccess() { queryClient.invalidateQueries({queryKey: RQKEY_MY_MUTED()}) }, }) } export function useProfileBlockMutationQueue( profile: Shadow, ) { const queryClient = useQueryClient() const did = profile.did const initialBlockingUri = profile.viewer?.blocking const blockMutation = useProfileBlockMutation() const unblockMutation = useProfileUnblockMutation() const queueToggle = useToggleMutationQueue({ initialState: initialBlockingUri, runMutation: async (prevBlockUri, shouldFollow) => { if (shouldFollow) { const {uri} = await blockMutation.mutateAsync({ did, }) return uri } else { if (prevBlockUri) { await unblockMutation.mutateAsync({ did, blockUri: prevBlockUri, }) } return undefined } }, onSuccess(finalBlockingUri) { // finalize updateProfileShadow(queryClient, did, { blockingUri: finalBlockingUri, }) queryClient.invalidateQueries({queryKey: RQKEY_LIST_CONVOS}) }, }) const queueBlock = useCallback(() => { // optimistically update updateProfileShadow(queryClient, did, { blockingUri: 'pending', }) return queueToggle(true) }, [queryClient, did, queueToggle]) const queueUnblock = useCallback(() => { // optimistically update updateProfileShadow(queryClient, did, { blockingUri: undefined, }) return queueToggle(false) }, [queryClient, did, queueToggle]) return [queueBlock, queueUnblock] } function useProfileBlockMutation() { const {currentAccount} = useSession() const agent = useAgent() const queryClient = useQueryClient() return useMutation<{uri: string; cid: string}, Error, {did: string}>({ mutationFn: async ({did}) => { if (!currentAccount) { throw new Error('Not signed in') } return await agent.app.bsky.graph.block.create( {repo: currentAccount.did}, {subject: did, createdAt: new Date().toISOString()}, ) }, onSuccess(_, {did}) { queryClient.invalidateQueries({queryKey: RQKEY_MY_BLOCKED()}) resetProfilePostsQueries(queryClient, did, 1000) }, }) } function useProfileUnblockMutation() { const {currentAccount} = useSession() const agent = useAgent() const queryClient = useQueryClient() return useMutation({ mutationFn: async ({blockUri}) => { if (!currentAccount) { throw new Error('Not signed in') } const {rkey} = new AtUri(blockUri) await agent.app.bsky.graph.block.delete({ repo: currentAccount.did, rkey, }) }, onSuccess(_, {did}) { resetProfilePostsQueries(queryClient, did, 1000) }, }) } export function precacheProfile( queryClient: QueryClient, profile: AppBskyActorDefs.ProfileViewBasic, ) { queryClient.setQueryData(profileBasicQueryKey(profile.handle), profile) queryClient.setQueryData(profileBasicQueryKey(profile.did), profile) } async function whenAppViewReady( agent: BskyAgent, actor: string, fn: (res: AppBskyActorGetProfile.Response) => boolean, ) { await until( 5, // 5 tries 1e3, // 1s delay between tries fn, () => agent.app.bsky.actor.getProfile({actor}), ) } export function* findAllProfilesInQueryData( queryClient: QueryClient, did: string, ): Generator { const queryDatas = queryClient.getQueriesData({ queryKey: [RQKEY_ROOT], }) for (const [_queryKey, queryData] of queryDatas) { if (!queryData) { continue } if (queryData.did === did) { yield queryData } } } export function findProfileQueryData( queryClient: QueryClient, did: string, ): AppBskyActorDefs.ProfileViewDetailed | undefined { return queryClient.getQueryData( RQKEY(did), ) }