diff options
Diffstat (limited to 'src/state/queries')
-rw-r--r-- | src/state/queries/post-quotes.ts | 124 |
1 files changed, 124 insertions, 0 deletions
diff --git a/src/state/queries/post-quotes.ts b/src/state/queries/post-quotes.ts new file mode 100644 index 000000000..be51eaab0 --- /dev/null +++ b/src/state/queries/post-quotes.ts @@ -0,0 +1,124 @@ +import { + AppBskyActorDefs, + AppBskyEmbedRecord, + AppBskyFeedDefs, + AppBskyFeedGetQuotes, + AtUri, +} from '@atproto/api' +import { + InfiniteData, + QueryClient, + QueryKey, + useInfiniteQuery, +} from '@tanstack/react-query' + +import {useAgent} from '#/state/session' +import { + didOrHandleUriMatches, + embedViewRecordToPostView, + getEmbeddedPost, +} from './util' + +const PAGE_SIZE = 30 +type RQPageParam = string | undefined + +const RQKEY_ROOT = 'post-quotes' +export const RQKEY = (resolvedUri: string) => [RQKEY_ROOT, resolvedUri] + +export function usePostQuotesQuery(resolvedUri: string | undefined) { + const agent = useAgent() + return useInfiniteQuery< + AppBskyFeedGetQuotes.OutputSchema, + Error, + InfiniteData<AppBskyFeedGetQuotes.OutputSchema>, + QueryKey, + RQPageParam + >({ + queryKey: RQKEY(resolvedUri || ''), + async queryFn({pageParam}: {pageParam: RQPageParam}) { + const res = await agent.api.app.bsky.feed.getQuotes({ + uri: resolvedUri || '', + limit: PAGE_SIZE, + cursor: pageParam, + }) + return res.data + }, + initialPageParam: undefined, + getNextPageParam: lastPage => lastPage.cursor, + enabled: !!resolvedUri, + select: data => { + return { + ...data, + pages: data.pages.map(page => { + return { + ...page, + posts: page.posts.filter(post => { + if (post.embed && AppBskyEmbedRecord.isView(post.embed)) { + if (AppBskyEmbedRecord.isViewDetached(post.embed.record)) { + return false + } + } + return true + }), + } + }), + } + }, + }) +} + +export function* findAllProfilesInQueryData( + queryClient: QueryClient, + did: string, +): Generator<AppBskyActorDefs.ProfileView, void> { + const queryDatas = queryClient.getQueriesData< + InfiniteData<AppBskyFeedGetQuotes.OutputSchema> + >({ + queryKey: [RQKEY_ROOT], + }) + for (const [_queryKey, queryData] of queryDatas) { + if (!queryData?.pages) { + continue + } + for (const page of queryData?.pages) { + for (const item of page.posts) { + if (item.author.did === did) { + yield item.author + } + const quotedPost = getEmbeddedPost(item.embed) + if (quotedPost?.author.did === did) { + yield quotedPost.author + } + } + } + } +} + +export function* findAllPostsInQueryData( + queryClient: QueryClient, + uri: string, +): Generator<AppBskyFeedDefs.PostView, undefined> { + const queryDatas = queryClient.getQueriesData< + InfiniteData<AppBskyFeedGetQuotes.OutputSchema> + >({ + queryKey: [RQKEY_ROOT], + }) + const atUri = new AtUri(uri) + for (const [_queryKey, queryData] of queryDatas) { + if (!queryData?.pages) { + continue + } + for (const page of queryData?.pages) { + for (const post of page.posts) { + if (didOrHandleUriMatches(atUri, post)) { + yield post + } + + const quotedPost = getEmbeddedPost(post.embed) + if (quotedPost && didOrHandleUriMatches(atUri, quotedPost)) { + yield embedViewRecordToPostView(quotedPost) + } + } + } + } +} |