elk/composables/masto/search.ts

110 lines
3.2 KiB
TypeScript
Raw Normal View History

2023-05-02 02:58:55 +03:00
import type { MaybeRefOrGetter } from '@vueuse/core'
2023-01-08 09:21:09 +03:00
import type { Paginator, mastodon } from 'masto'
2023-01-07 17:52:58 +03:00
import type { RouteLocation } from 'vue-router'
2022-12-18 00:35:07 +03:00
2023-05-02 02:58:55 +03:00
export type UseSearchOptions = MaybeRefOrGetter<
2023-01-08 16:42:53 +03:00
Partial<Omit<mastodon.v1.SearchParams, keyof mastodon.DefaultPaginationParams | 'q'>>
>
2022-12-18 00:35:07 +03:00
2023-01-07 17:52:58 +03:00
export interface BuildSearchResult<K extends keyof any, T> {
id: string
type: K
data: T
to: RouteLocation & {
href: string
}
}
2023-01-08 09:21:09 +03:00
export type AccountSearchResult = BuildSearchResult<'account', mastodon.v1.Account>
export type HashTagSearchResult = BuildSearchResult<'hashtag', mastodon.v1.Tag>
export type StatusSearchResult = BuildSearchResult<'status', mastodon.v1.Status>
2023-01-07 17:52:58 +03:00
export type SearchResult = HashTagSearchResult | AccountSearchResult | StatusSearchResult
2023-05-02 02:58:55 +03:00
export function useSearch(query: MaybeRefOrGetter<string>, options: UseSearchOptions = {}) {
2022-12-18 00:35:07 +03:00
const done = ref(false)
2023-01-15 11:38:02 +03:00
const { client } = $(useMasto())
2022-12-18 00:35:07 +03:00
const loading = ref(false)
2023-01-07 17:52:58 +03:00
const accounts = ref<AccountSearchResult[]>([])
const hashtags = ref<HashTagSearchResult[]>([])
const statuses = ref<StatusSearchResult[]>([])
2022-12-18 00:35:07 +03:00
2023-01-09 11:36:24 +03:00
const q = $computed(() => resolveUnref(query).trim())
2023-01-08 09:21:09 +03:00
let paginator: Paginator<mastodon.v2.Search, mastodon.v2.SearchParams> | undefined
2022-12-22 13:34:35 +03:00
2023-01-08 09:21:09 +03:00
const appendResults = (results: mastodon.v2.Search, empty = false) => {
2023-01-07 17:52:58 +03:00
if (empty) {
accounts.value = []
hashtags.value = []
statuses.value = []
}
accounts.value = [...accounts.value, ...results.accounts.map<AccountSearchResult>(account => ({
type: 'account',
id: account.id,
data: account,
to: getAccountRoute(account),
}))]
hashtags.value = [...hashtags.value, ...results.hashtags.map<HashTagSearchResult>(hashtag => ({
type: 'hashtag',
id: `hashtag-${hashtag.name}`,
data: hashtag,
to: getTagRoute(hashtag.name),
}))]
statuses.value = [...statuses.value, ...results.statuses.map<StatusSearchResult>(status => ({
type: 'status',
id: status.id,
data: status,
to: getStatusRoute(status),
}))]
}
2023-01-09 11:36:24 +03:00
watch(() => resolveUnref(query), () => {
2023-01-15 11:38:02 +03:00
loading.value = !!(q && isHydrated.value)
2023-01-07 17:59:57 +03:00
})
2023-01-09 11:36:24 +03:00
debouncedWatch(() => resolveUnref(query), async () => {
2023-01-15 11:38:02 +03:00
if (!q || !isHydrated.value)
2022-12-18 00:35:07 +03:00
return
loading.value = true
/**
* Based on the source it seems like modifying the params when calling next would result in a new search,
* but that doesn't seem to be the case. So instead we just create a new paginator with the new params.
*/
2023-01-15 11:38:02 +03:00
paginator = client.v2.search({
2023-01-09 11:36:24 +03:00
q,
2023-01-08 16:42:53 +03:00
...resolveUnref(options),
2023-01-07 17:52:58 +03:00
resolve: !!currentUser.value,
})
2022-12-18 00:35:07 +03:00
const nextResults = await paginator.next()
2023-01-07 17:52:58 +03:00
done.value = !!nextResults.done
if (!nextResults.done)
appendResults(nextResults.value, true)
2022-12-18 00:35:07 +03:00
loading.value = false
2023-01-07 17:59:57 +03:00
}, { debounce: 300 })
2022-12-18 00:35:07 +03:00
const next = async () => {
2023-01-15 11:38:02 +03:00
if (!q || !isHydrated.value || !paginator)
2022-12-18 00:35:07 +03:00
return
loading.value = true
const nextResults = await paginator.next()
loading.value = false
2023-01-07 17:52:58 +03:00
done.value = !!nextResults.done
if (!nextResults.done)
appendResults(nextResults.value)
2022-12-18 00:35:07 +03:00
}
return {
accounts,
hashtags,
statuses,
loading: readonly(loading),
next,
}
}