* Add the animated checkmark svg * Add progress guide list and task components * Add ProgressGuide Toast component * Implement progress-guide controller * Add 7 follows to the progress guide * Wire up action captures * Wire up progress-guide persistence * Trigger progress guide on account creation * Clear the progress guide from storage on complete * Add progress guide interstitial, put behind gate * Fix: read progress guide state from prefs * Some defensive type checks * Create separate toast for completion * List tweaks * Only show on Discover * Spacing and progress tweaks * Completely hide when complete * Capture the progress guide in local state, and only render toasts while guide is active * Fix: ensure persisted hydrates into local state * Gate --------- Co-authored-by: Eric Bailey <git@esb.lol> Co-authored-by: Dan Abramov <dan.abramov@gmail.com>
537 lines
15 KiB
TypeScript
537 lines
15 KiB
TypeScript
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<AppBskyActorDefs.ProfileViewDetailed>({
|
|
// 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<AppBskyActorDefs.ProfileViewBasic>(
|
|
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<void, Error, ProfileUpdateParams>({
|
|
mutationFn: async ({
|
|
profile,
|
|
updates,
|
|
newUserAvatar,
|
|
newUserBanner,
|
|
checkCommitted,
|
|
}) => {
|
|
let newUserAvatarPromise:
|
|
| Promise<ComAtprotoRepoUploadBlob.Response>
|
|
| undefined
|
|
if (newUserAvatar) {
|
|
newUserAvatarPromise = uploadBlob(
|
|
agent,
|
|
newUserAvatar.path,
|
|
newUserAvatar.mime,
|
|
)
|
|
}
|
|
let newUserBannerPromise:
|
|
| Promise<ComAtprotoRepoUploadBlob.Response>
|
|
| 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<AppBskyActorDefs.ProfileViewDetailed>,
|
|
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<AppBskyActorDefs.ProfileViewDetailed>,
|
|
) {
|
|
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<void, Error, {did: string; followUri: string}>({
|
|
mutationFn: async ({followUri}) => {
|
|
logEvent('profile:unfollow', {logContext})
|
|
track('Profile:Unfollow', {username: followUri})
|
|
return await agent.deleteFollow(followUri)
|
|
},
|
|
})
|
|
}
|
|
|
|
export function useProfileMuteMutationQueue(
|
|
profile: Shadow<AppBskyActorDefs.ProfileViewDetailed>,
|
|
) {
|
|
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<void, Error, {did: string}>({
|
|
mutationFn: async ({did}) => {
|
|
await agent.mute(did)
|
|
},
|
|
onSuccess() {
|
|
queryClient.invalidateQueries({queryKey: RQKEY_MY_MUTED()})
|
|
},
|
|
})
|
|
}
|
|
|
|
function useProfileUnmuteMutation() {
|
|
const queryClient = useQueryClient()
|
|
const agent = useAgent()
|
|
return useMutation<void, Error, {did: string}>({
|
|
mutationFn: async ({did}) => {
|
|
await agent.unmute(did)
|
|
},
|
|
onSuccess() {
|
|
queryClient.invalidateQueries({queryKey: RQKEY_MY_MUTED()})
|
|
},
|
|
})
|
|
}
|
|
|
|
export function useProfileBlockMutationQueue(
|
|
profile: Shadow<AppBskyActorDefs.ProfileViewDetailed>,
|
|
) {
|
|
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<void, Error, {did: string; blockUri: string}>({
|
|
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<AppBskyActorDefs.ProfileViewDetailed, void> {
|
|
const queryDatas =
|
|
queryClient.getQueriesData<AppBskyActorDefs.ProfileViewDetailed>({
|
|
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<AppBskyActorDefs.ProfileViewDetailed>(
|
|
RQKEY(did),
|
|
)
|
|
}
|