|
/* See plain javascript version below 👇 */ |
|
import useSWR, { |
|
ConfigInterface, |
|
responseInterface as ResponseInterface, |
|
} from 'swr' |
|
import { DocumentNode } from 'graphql' |
|
import { print } from 'graphql/language/printer' |
|
import { Variables } from 'graphql-request/dist/src/types' |
|
import { request } from 'graphql-request' |
|
|
|
const apiUrl = 'MY_API_URL' |
|
|
|
type LazyRequestTuple<R, V extends Variables = Variables> = [ |
|
(v?: V) => Promise<void>, |
|
ResponseInterface<R | null, Error>, |
|
] |
|
|
|
interface RequestArgs<R, V> { |
|
query: DocumentNode | string |
|
variables?: V |
|
options?: ConfigInterface<R> |
|
} |
|
|
|
/* Turns a graphQL query into a string. |
|
* If it is already a string, return it as is. |
|
* If it is a DocumentNode (created by gql), |
|
* use print() to turn it into a string. */ |
|
const asString = (query: string | DocumentNode): string => |
|
typeof query === 'string' ? query : print(query) |
|
|
|
export const useLazyRequest = <R, V extends Variables = Variables>( |
|
query: DocumentNode | string, |
|
variables: V = {}, |
|
options: ConfigInterface<R> = {}, |
|
): LazyRequestTuple<R, V> => { |
|
const queryString = asString(query) |
|
const response = useSWR<R | null>( |
|
/* Use the GraphQL Query string as the key for this SWR request */ |
|
[queryString, JSON.stringify(variables)], |
|
/* Our initial "fetch" will just return null as the data since this is lazy */ |
|
() => null, |
|
/* with any other SWR options we want */ |
|
options, |
|
) |
|
|
|
const executeQuery = async (variables?: V) => { |
|
/* |
|
* Later you will call executeQuery(variables) |
|
* This will send the "real" request. In this example we are using |
|
* graphql-request but you could use fetch or anything else. |
|
*/ |
|
const result = await request<R>(apiUrl, queryString, variables) |
|
|
|
/* |
|
* Once we get the result, we "mutate" the response to this request. |
|
* This will update the `response` object that was created by `useSwr` |
|
* and our component will receive the updated data. |
|
*/ |
|
response.mutate(result, false) |
|
} |
|
|
|
return [executeQuery, response] |
|
} |
Hello @good-idea, I was working on something similar but I needed to be compatible with Suspense. In my case I used conditional fetching to solve it.
Thanks, your idea helped me a lot, I would like to share mine with you:
https://gist.github.com/stivsk/44854eb08884cb7d876c8e549afaa343