import {useCallback} from 'react' import { type AppBskyActorDefs, type AppBskyActorGetProfile, type AppBskyActorGetProfiles, type AppBskyActorProfile, type AppBskyGraphGetFollows, AtUri, type BskyAgent, type ComAtprotoRepoUploadBlob, type Un$Typed, } from '@atproto/api' import { type InfiniteData, keepPreviousData, type QueryClient, useMutation, useQuery, useQueryClient, } from '@tanstack/react-query' import {uploadBlob} from '#/lib/api' import {until} from '#/lib/async/until' import {useToggleMutationQueue} from '#/lib/hooks/useToggleMutationQueue' import {updateProfileShadow} from '#/state/cache/profile-shadow' import {type Shadow} from '#/state/cache/types' import {type ImageMeta} from '#/state/gallery' import {STALE} from '#/state/queries' import {resetProfilePostsQueries} from '#/state/queries/post-feed' import {RQKEY as PROFILE_FOLLOWS_RQKEY} from '#/state/queries/profile-follows' import { unstableCacheProfileView, useUnstableProfileViewCache, } from '#/state/queries/unstable-profile-cache' import {useUpdateProfileVerificationCache} from '#/state/queries/verification/useUpdateProfileVerificationCache' import {useAgent, useSession} from '#/state/session' import * as userActionHistory from '#/state/userActionHistory' import {useAnalytics} from '#/analytics' import {type Metrics, toClout} from '#/analytics/metrics' import type * as bsky from '#/types/bsky' import { ProgressGuideAction, useProgressGuideControls, } from '../shell/progress-guide' import {RQKEY_ROOT as RQKEY_LIST_CONVOS} from './messages/list-conversations' import {RQKEY as RQKEY_MY_BLOCKED} from './my-blocked-accounts' import {RQKEY as RQKEY_MY_MUTED} from './my-muted-accounts' export * from '#/state/queries/unstable-profile-cache' /** * @deprecated use {@link unstableCacheProfileView} instead */ export const precacheProfile = unstableCacheProfileView const RQKEY_ROOT = 'profile' export const RQKEY = (did: string) => [RQKEY_ROOT, did] export type TealArtist = { artistName: string artistMbId?: string } export type TealPlayView = { trackName: string trackMbId?: string recordingMbId?: string duration?: number artists: TealArtist[] releaseName?: string releaseMbId?: string isrc?: string originUrl?: string musicServiceBaseDomain?: string submissionClientAgent?: string playedTime?: string // datetime } export type TealActorStatus = { time: string // datetime expiry?: string // datetime item: TealPlayView } export const profilesQueryKeyRoot = 'profiles' export const profilesQueryKey = (handles: string[]) => [ profilesQueryKeyRoot, handles, ] export function useProfileQuery({ did, staleTime = STALE.SECONDS.FIFTEEN, }: { did: string | undefined staleTime?: number }) { const agent = useAgent() const {getUnstableProfile} = useUnstableProfileViewCache() return useQuery< AppBskyActorDefs.ProfileViewDetailed & { tealStatus: TealActorStatus | undefined } >({ // 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 ?? ''}) try { const teal = await fetch( `https://slingshot.microcosm.blue/xrpc/com.atproto.repo.getRecord?repo=${did ?? ''}&collection=fm.teal.alpha.actor.status&rkey=self`, { method: 'GET', }, ) const tealData = await teal.json() return { ...res.data, tealStatus: tealData.value as TealActorStatus | undefined, } } catch (e) { return { ...res.data, tealStatus: undefined as TealActorStatus | undefined, } } }, placeholderData: () => { if (!did) return const profile = getUnstableProfile(did) as AppBskyActorDefs.ProfileViewDetailed return profile ? { ...profile, tealStatus: undefined, } : undefined }, enabled: !!did, }) } export function useProfilesQuery({ handles, maintainData, }: { handles: string[] maintainData?: boolean }) { const agent = useAgent() return useQuery({ staleTime: STALE.MINUTES.FIVE, queryKey: profilesQueryKey(handles), queryFn: async () => { const res = await agent.getProfiles({actors: handles}) return res.data }, placeholderData: maintainData ? keepPreviousData : undefined, }) } 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.ProfileViewDetailed updates: | Un$Typed | (( existing: Un$Typed, ) => Un$Typed) newUserAvatar?: ImageMeta | undefined | null newUserBanner?: ImageMeta | undefined | null checkCommitted?: (res: AppBskyActorGetProfile.Response) => boolean } export function useProfileUpdateMutation() { const queryClient = useQueryClient() const agent = useAgent() const updateProfileVerificationCache = useUpdateProfileVerificationCache() 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 => { let next: Un$Typed = existing || {} if (typeof updates === 'function') { next = updates(next) } else { next.displayName = updates.displayName next.description = updates.description if ('pinnedPost' in updates) { next.pinnedPost = updates.pinnedPost } } if (newUserAvatarPromise) { const res = await newUserAvatarPromise next.avatar = res.data.blob } else if (newUserAvatar === null) { next.avatar = undefined } if (newUserBannerPromise) { const res = await newUserBannerPromise next.banner = res.data.blob } else if (newUserBanner === null) { next.banner = undefined } return next }) 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 ) }), ) }, async onSuccess(_, variables) { // invalidate cache queryClient.invalidateQueries({ queryKey: RQKEY(variables.profile.did), }) queryClient.invalidateQueries({ queryKey: [profilesQueryKeyRoot, [variables.profile.did]], }) await updateProfileVerificationCache({profile: variables.profile}) }, }) } export function useProfileFollowMutationQueue( profile: Shadow, logContext: Metrics['profile:follow']['logContext'], position?: number, contextProfileDid?: string, ) { const agent = useAgent() const queryClient = useQueryClient() const {currentAccount} = useSession() const did = profile.did const initialFollowingUri = profile.viewer?.following const followMutation = useProfileFollowMutation( logContext, profile, position, contextProfileDid, ) const unfollowMutation = useProfileUnfollowMutation(logContext) const queueToggle = useToggleMutationQueue({ initialState: initialFollowingUri, runMutation: async (prevFollowingUri, shouldFollow) => { if (shouldFollow) { const {uri} = await followMutation.mutateAsync({ did, }) userActionHistory.follow([did]) return uri } else { if (prevFollowingUri) { await unfollowMutation.mutateAsync({ did, followUri: prevFollowingUri, }) userActionHistory.unfollow([did]) } return undefined } }, onSuccess(finalFollowingUri) { // finalize updateProfileShadow(queryClient, did, { followingUri: finalFollowingUri, }) // Optimistically update profile follows cache for avatar displays if (currentAccount?.did) { type FollowsQueryData = InfiniteData queryClient.setQueryData( PROFILE_FOLLOWS_RQKEY(currentAccount.did), old => { if (!old?.pages?.[0]) return old if (finalFollowingUri) { // Add the followed profile to the beginning const alreadyExists = old.pages[0].follows.some( f => f.did === profile.did, ) if (alreadyExists) return old return { ...old, pages: [ { ...old.pages[0], follows: [ profile as AppBskyActorDefs.ProfileView, ...old.pages[0].follows, ], }, ...old.pages.slice(1), ], } } else { // Remove the unfollowed profile return { ...old, pages: old.pages.map(page => ({ ...page, follows: page.follows.filter(f => f.did !== profile.did), })), } } }, ) } if (finalFollowingUri) { agent.app.bsky.graph .getSuggestedFollowsByActor({ actor: did, }) .then(res => { const dids = res.data.suggestions .filter(a => !a.viewer?.following) .map(a => a.did) .slice(0, 8) userActionHistory.followSuggestion(dids) }) } }, }) 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: Metrics['profile:follow']['logContext'], profile: Shadow, position?: number, contextProfileDid?: string, ) { const ax = useAnalytics() 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) ax.metric('profile:follow', { logContext, didBecomeMutual: profile.viewer ? Boolean(profile.viewer.followedBy) : undefined, followeeClout: 'followersCount' in profile ? toClout(profile.followersCount) : undefined, followeeDid: did, followerClout: toClout(ownProfile?.followersCount), position, contextProfileDid, }) return await agent.follow(did) }, }) } function useProfileUnfollowMutation( logContext: Metrics['profile:unfollow']['logContext'], ) { const ax = useAnalytics() const agent = useAgent() return useMutation({ mutationFn: async ({followUri}) => { ax.metric('profile:unfollow', {logContext}) 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) }, }) } 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 profileQueryDatas = queryClient.getQueriesData({ queryKey: [RQKEY_ROOT], }) for (const [_queryKey, queryData] of profileQueryDatas) { if (!queryData) { continue } if (queryData.did === did) { yield queryData } } const profilesQueryDatas = queryClient.getQueriesData({ queryKey: [profilesQueryKeyRoot], }) for (const [_queryKey, queryData] of profilesQueryDatas) { if (!queryData) { continue } for (let profile of queryData.profiles) { if (profile.did === did) { yield profile } } } } export function findProfileQueryData( queryClient: QueryClient, did: string, ): AppBskyActorDefs.ProfileViewDetailed | undefined { return queryClient.getQueryData( RQKEY(did), ) }