-
-
Notifications
You must be signed in to change notification settings - Fork 89
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
2 changed files
with
213 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1 +1,212 @@ | ||
export * from '../runtime/vue'; | ||
/* eslint-disable @typescript-eslint/ban-types */ | ||
/* eslint-disable @typescript-eslint/no-explicit-any */ | ||
import { | ||
useInfiniteQuery, | ||
useMutation, | ||
useQuery, | ||
useQueryClient, | ||
type InfiniteData, | ||
type QueryKey, | ||
type UseInfiniteQueryOptions, | ||
type UseMutationOptions, | ||
type UseQueryOptions, | ||
} from '@tanstack/vue-query'; | ||
import type { ModelMeta } from '@zenstackhq/runtime/cross'; | ||
import { computed, inject, provide, toValue, type ComputedRef, type MaybeRefOrGetter } from 'vue'; | ||
import { | ||
APIContext, | ||
DEFAULT_QUERY_ENDPOINT, | ||
fetcher, | ||
getQueryKey, | ||
makeUrl, | ||
marshal, | ||
setupInvalidation, | ||
setupOptimisticUpdate, | ||
type ExtraMutationOptions, | ||
type ExtraQueryOptions, | ||
type FetchFn, | ||
} from '../runtime/common'; | ||
|
||
export { APIContext as RequestHandlerContext } from '../runtime/common'; | ||
|
||
export const VueQueryContextKey = 'zenstack-vue-query-context'; | ||
|
||
/** | ||
* Provide context for the generated TanStack Query hooks. | ||
*/ | ||
export function provideHooksContext(context: APIContext) { | ||
provide<APIContext>(VueQueryContextKey, context); | ||
} | ||
|
||
/** | ||
* Hooks context. | ||
*/ | ||
export function getHooksContext() { | ||
const { endpoint, ...rest } = inject<APIContext>(VueQueryContextKey, { | ||
endpoint: DEFAULT_QUERY_ENDPOINT, | ||
fetch: undefined, | ||
logging: false, | ||
}); | ||
return { endpoint: endpoint ?? DEFAULT_QUERY_ENDPOINT, ...rest }; | ||
} | ||
|
||
/** | ||
* Creates a vue-query query. | ||
* | ||
* @param model The name of the model under query. | ||
* @param url The request URL. | ||
* @param args The request args object, URL-encoded and appended as "?q=" parameter | ||
* @param options The vue-query options object | ||
* @param fetch The fetch function to use for sending the HTTP request | ||
* @returns useQuery hook | ||
*/ | ||
export function useModelQuery<TQueryFnData, TData, TError>( | ||
model: string, | ||
url: string, | ||
args?: MaybeRefOrGetter<unknown> | ComputedRef<unknown>, | ||
options?: | ||
| MaybeRefOrGetter<Omit<UseQueryOptions<TQueryFnData, TError, TData>, 'queryKey'> & ExtraQueryOptions> | ||
| ComputedRef<Omit<UseQueryOptions<TQueryFnData, TError, TData>, 'queryKey'> & ExtraQueryOptions>, | ||
fetch?: FetchFn | ||
) { | ||
const queryOptions = computed(() => { | ||
const optionsValue = toValue< | ||
(Omit<UseQueryOptions<TQueryFnData, TError, TData>, 'queryKey'> & ExtraQueryOptions) | undefined | ||
>(options); | ||
return { | ||
queryKey: getQueryKey(model, url, args, { | ||
infinite: false, | ||
optimisticUpdate: optionsValue?.optimisticUpdate !== false, | ||
}), | ||
queryFn: ({ queryKey }: { queryKey: QueryKey }) => { | ||
const [_prefix, _model, _op, args] = queryKey; | ||
const reqUrl = makeUrl(url, toValue(args)); | ||
return fetcher<TQueryFnData, false>(reqUrl, undefined, fetch, false); | ||
}, | ||
...optionsValue, | ||
}; | ||
}); | ||
return useQuery<TQueryFnData, TError, TData>(queryOptions); | ||
} | ||
|
||
/** | ||
* Creates a vue-query infinite query. | ||
* | ||
* @param model The name of the model under query. | ||
* @param url The request URL. | ||
* @param args The initial request args object, URL-encoded and appended as "?q=" parameter | ||
* @param options The vue-query infinite query options object | ||
* @param fetch The fetch function to use for sending the HTTP request | ||
* @returns useInfiniteQuery hook | ||
*/ | ||
export function useInfiniteModelQuery<TQueryFnData, TData, TError>( | ||
model: string, | ||
url: string, | ||
args?: MaybeRefOrGetter<unknown> | ComputedRef<unknown>, | ||
options?: | ||
| MaybeRefOrGetter< | ||
Omit<UseInfiniteQueryOptions<TQueryFnData, TError, InfiniteData<TData>>, 'queryKey' | 'initialPageParam'> | ||
> | ||
| ComputedRef< | ||
Omit<UseInfiniteQueryOptions<TQueryFnData, TError, InfiniteData<TData>>, 'queryKey' | 'initialPageParam'> | ||
>, | ||
fetch?: FetchFn | ||
) { | ||
// CHECKME: vue-query's `useInfiniteQuery`'s input typing seems wrong | ||
const queryOptions: any = computed(() => ({ | ||
queryKey: getQueryKey(model, url, args, { infinite: true, optimisticUpdate: false }), | ||
queryFn: ({ queryKey, pageParam }: { queryKey: QueryKey; pageParam?: unknown }) => { | ||
const [_prefix, _model, _op, args] = queryKey; | ||
const reqUrl = makeUrl(url, pageParam ?? toValue(args)); | ||
return fetcher<TQueryFnData, false>(reqUrl, undefined, fetch, false); | ||
}, | ||
initialPageParam: toValue(args), | ||
...toValue(options), | ||
})); | ||
|
||
return useInfiniteQuery<TQueryFnData, TError, InfiniteData<TData>>(queryOptions); | ||
} | ||
|
||
/** | ||
* Creates a mutation with vue-query. | ||
* | ||
* @param model The name of the model under mutation. | ||
* @param method The HTTP method. | ||
* @param modelMeta The model metadata. | ||
* @param url The request URL. | ||
* @param options The vue-query options. | ||
* @param fetch The fetch function to use for sending the HTTP request | ||
* @param checkReadBack Whether to check for read back errors and return undefined if found. | ||
* @returns useMutation hooks | ||
*/ | ||
export function useModelMutation< | ||
TArgs, | ||
TError, | ||
R = any, | ||
C extends boolean = boolean, | ||
Result = C extends true ? R | undefined : R | ||
>( | ||
model: string, | ||
method: 'POST' | 'PUT' | 'DELETE', | ||
url: string, | ||
modelMeta: ModelMeta, | ||
options?: | ||
| MaybeRefOrGetter< | ||
Omit<UseMutationOptions<Result, TError, TArgs, unknown>, 'mutationFn'> & ExtraMutationOptions | ||
> | ||
| ComputedRef<Omit<UseMutationOptions<Result, TError, TArgs, unknown>, 'mutationFn'> & ExtraMutationOptions>, | ||
fetch?: FetchFn, | ||
checkReadBack?: C | ||
) { | ||
const queryClient = useQueryClient(); | ||
const mutationFn = (data: any) => { | ||
const reqUrl = method === 'DELETE' ? makeUrl(url, data) : url; | ||
const fetchInit: RequestInit = { | ||
method, | ||
...(method !== 'DELETE' && { | ||
headers: { | ||
'content-type': 'application/json', | ||
}, | ||
body: marshal(data), | ||
}), | ||
}; | ||
return fetcher<R, C>(reqUrl, fetchInit, fetch, checkReadBack) as Promise<Result>; | ||
}; | ||
|
||
const optionsValue = toValue< | ||
(Omit<UseMutationOptions<Result, TError, TArgs, unknown>, 'mutationFn'> & ExtraMutationOptions) | undefined | ||
>(options); | ||
// TODO: figure out the typing problem | ||
const finalOptions: any = computed(() => ({ ...optionsValue, mutationFn })); | ||
const operation = url.split('/').pop(); | ||
const invalidateQueries = optionsValue?.invalidateQueries !== false; | ||
const optimisticUpdate = !!optionsValue?.optimisticUpdate; | ||
|
||
if (operation) { | ||
const { logging } = getHooksContext(); | ||
if (invalidateQueries) { | ||
setupInvalidation( | ||
model, | ||
operation, | ||
modelMeta, | ||
toValue(finalOptions), | ||
(predicate) => queryClient.invalidateQueries({ predicate }), | ||
logging | ||
); | ||
} | ||
|
||
if (optimisticUpdate) { | ||
setupOptimisticUpdate( | ||
model, | ||
operation, | ||
modelMeta, | ||
toValue(finalOptions), | ||
queryClient.getQueryCache().getAll(), | ||
(queryKey, data) => queryClient.setQueryData<unknown>(queryKey, data), | ||
invalidateQueries ? (predicate) => queryClient.invalidateQueries({ predicate }) : undefined, | ||
logging | ||
); | ||
} | ||
} | ||
return useMutation<Result, TError, TArgs>(finalOptions); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters