UNPKG

@apollo/client

Version:

A fully-featured caching GraphQL client.

148 lines 6.85 kB
import { __assign } from "tslib"; import * as React from "rehackt"; import { mergeOptions } from "../../utilities/index.js"; import { createMakeWatchQueryOptions, getDefaultFetchPolicy, getObsQueryOptions, toQueryResult, useQueryInternals, } from "./useQuery.js"; import { useIsomorphicLayoutEffect } from "./internal/useIsomorphicLayoutEffect.js"; // The following methods, when called will execute the query, regardless of // whether the useLazyQuery execute function was called before. var EAGER_METHODS = [ "refetch", "reobserve", "fetchMore", "updateQuery", "startPolling", "stopPolling", "subscribeToMore", ]; /** * A hook for imperatively executing queries in an Apollo application, e.g. in response to user interaction. * * > Refer to the [Queries - Manual execution with useLazyQuery](https://www.apollographql.com/docs/react/data/queries#manual-execution-with-uselazyquery) section for a more in-depth overview of `useLazyQuery`. * * @example * ```jsx * import { gql, useLazyQuery } from "@apollo/client"; * * const GET_GREETING = gql` * query GetGreeting($language: String!) { * greeting(language: $language) { * message * } * } * `; * * function Hello() { * const [loadGreeting, { called, loading, data }] = useLazyQuery( * GET_GREETING, * { variables: { language: "english" } } * ); * if (called && loading) return <p>Loading ...</p> * if (!called) { * return <button onClick={() => loadGreeting()}>Load greeting</button> * } * return <h1>Hello {data.greeting.message}!</h1>; * } * ``` * @since 3.0.0 * * @param query - A GraphQL query document parsed into an AST by `gql`. * @param options - Default options to control how the query is executed. * @returns A tuple in the form of `[execute, result]` */ export function useLazyQuery(query, options) { var _a; var execOptionsRef = React.useRef(); var optionsRef = React.useRef(); var queryRef = React.useRef(); var merged = mergeOptions(options, execOptionsRef.current || {}); var document = (_a = merged === null || merged === void 0 ? void 0 : merged.query) !== null && _a !== void 0 ? _a : query; // Use refs to track options and the used query to ensure the `execute` // function remains referentially stable between renders. optionsRef.current = options; queryRef.current = document; var queryHookOptions = __assign(__assign({}, merged), { skip: !execOptionsRef.current }); var _b = useQueryInternals(document, queryHookOptions), obsQueryFields = _b.obsQueryFields, useQueryResult = _b.result, client = _b.client, resultData = _b.resultData, observable = _b.observable, onQueryExecuted = _b.onQueryExecuted; var initialFetchPolicy = observable.options.initialFetchPolicy || getDefaultFetchPolicy(queryHookOptions.defaultOptions, client.defaultOptions); var forceUpdateState = React.useReducer(function (tick) { return tick + 1; }, 0)[1]; // We use useMemo here to make sure the eager methods have a stable identity. var eagerMethods = React.useMemo(function () { var eagerMethods = {}; var _loop_1 = function (key) { var method = obsQueryFields[key]; eagerMethods[key] = function () { if (!execOptionsRef.current) { execOptionsRef.current = Object.create(null); // Only the first time populating execOptionsRef.current matters here. forceUpdateState(); } // @ts-expect-error this is just too generic to type return method.apply(this, arguments); }; }; for (var _i = 0, EAGER_METHODS_1 = EAGER_METHODS; _i < EAGER_METHODS_1.length; _i++) { var key = EAGER_METHODS_1[_i]; _loop_1(key); } return eagerMethods; }, [forceUpdateState, obsQueryFields]); var called = !!execOptionsRef.current; var result = React.useMemo(function () { return (__assign(__assign(__assign({}, useQueryResult), eagerMethods), { called: called })); }, [useQueryResult, eagerMethods, called]); var execute = React.useCallback(function (executeOptions) { execOptionsRef.current = executeOptions ? __assign(__assign({}, executeOptions), { fetchPolicy: executeOptions.fetchPolicy || initialFetchPolicy }) : { fetchPolicy: initialFetchPolicy, }; var options = mergeOptions(optionsRef.current, __assign({ query: queryRef.current }, execOptionsRef.current)); var promise = executeQuery(resultData, observable, client, document, __assign(__assign({}, options), { skip: false }), onQueryExecuted).then(function (queryResult) { return Object.assign(queryResult, eagerMethods); }); // Because the return value of `useLazyQuery` is usually floated, we need // to catch the promise to prevent unhandled rejections. promise.catch(function () { }); return promise; }, [ client, document, eagerMethods, initialFetchPolicy, observable, resultData, onQueryExecuted, ]); var executeRef = React.useRef(execute); useIsomorphicLayoutEffect(function () { executeRef.current = execute; }); var stableExecute = React.useCallback(function () { var args = []; for (var _i = 0; _i < arguments.length; _i++) { args[_i] = arguments[_i]; } return executeRef.current.apply(executeRef, args); }, []); return [stableExecute, result]; } function executeQuery(resultData, observable, client, currentQuery, options, onQueryExecuted) { var query = options.query || currentQuery; var watchQueryOptions = createMakeWatchQueryOptions(client, query, options, false)(observable); var concast = observable.reobserveAsConcast(getObsQueryOptions(observable, client, options, watchQueryOptions)); onQueryExecuted(watchQueryOptions); return new Promise(function (resolve) { var result; // Subscribe to the concast independently of the ObservableQuery in case // the component gets unmounted before the promise resolves. This prevents // the concast from terminating early and resolving with `undefined` when // there are no more subscribers for the concast. concast.subscribe({ next: function (value) { result = value; }, error: function () { resolve(toQueryResult(observable.getCurrentResult(), resultData.previousData, observable, client)); }, complete: function () { resolve(toQueryResult(result, resultData.previousData, observable, client)); }, }); }); } //# sourceMappingURL=useLazyQuery.js.map