UNPKG

13.2 kBJavaScriptView Raw
1import { __assign, __rest } from "tslib";
2import { invariant } from "../../utilities/globals/index.js";
3import { useCallback, useContext, useMemo, useRef, useState, } from 'react';
4import { useSyncExternalStore } from "./useSyncExternalStore.js";
5import { equal } from '@wry/equality';
6import { mergeOptions } from "../../core/index.js";
7import { getApolloContext } from "../context/index.js";
8import { ApolloError } from "../../errors/index.js";
9import { NetworkStatus, } from "../../core/index.js";
10import { DocumentType, verifyDocumentType } from "../parser/index.js";
11import { useApolloClient } from "./useApolloClient.js";
12import { canUseWeakMap, canUseWeakSet, compact, isNonEmptyArray, maybeDeepFreeze } from "../../utilities/index.js";
13var hasOwnProperty = Object.prototype.hasOwnProperty;
14export function useQuery(query, options) {
15 if (options === void 0) { options = Object.create(null); }
16 return useInternalState(useApolloClient(options.client), query).useQuery(options);
17}
18export function useInternalState(client, query) {
19 var stateRef = useRef();
20 if (!stateRef.current ||
21 client !== stateRef.current.client ||
22 query !== stateRef.current.query) {
23 stateRef.current = new InternalState(client, query, stateRef.current);
24 }
25 var state = stateRef.current;
26 var _a = useState(0), _tick = _a[0], setTick = _a[1];
27 state.forceUpdate = function () {
28 setTick(function (tick) { return tick + 1; });
29 };
30 return state;
31}
32var InternalState = (function () {
33 function InternalState(client, query, previous) {
34 this.client = client;
35 this.query = query;
36 this.asyncResolveFns = new Set();
37 this.optionsToIgnoreOnce = new (canUseWeakSet ? WeakSet : Set)();
38 this.ssrDisabledResult = maybeDeepFreeze({
39 loading: true,
40 data: void 0,
41 error: void 0,
42 networkStatus: NetworkStatus.loading,
43 });
44 this.skipStandbyResult = maybeDeepFreeze({
45 loading: false,
46 data: void 0,
47 error: void 0,
48 networkStatus: NetworkStatus.ready,
49 });
50 this.toQueryResultCache = new (canUseWeakMap ? WeakMap : Map)();
51 verifyDocumentType(query, DocumentType.Query);
52 var previousResult = previous && previous.result;
53 var previousData = previousResult && previousResult.data;
54 if (previousData) {
55 this.previousData = previousData;
56 }
57 }
58 InternalState.prototype.forceUpdate = function () {
59 __DEV__ && invariant.warn("Calling default no-op implementation of InternalState#forceUpdate");
60 };
61 InternalState.prototype.asyncUpdate = function () {
62 var _this = this;
63 return new Promise(function (resolve) {
64 _this.asyncResolveFns.add(resolve);
65 _this.optionsToIgnoreOnce.add(_this.watchQueryOptions);
66 _this.forceUpdate();
67 });
68 };
69 InternalState.prototype.useQuery = function (options) {
70 var _this = this;
71 this.renderPromises = useContext(getApolloContext()).renderPromises;
72 this.useOptions(options);
73 var obsQuery = this.useObservableQuery();
74 var result = useSyncExternalStore(useCallback(function () {
75 if (_this.renderPromises) {
76 return function () { };
77 }
78 var onNext = function () {
79 var previousResult = _this.result;
80 var result = obsQuery.getCurrentResult();
81 if (previousResult &&
82 previousResult.loading === result.loading &&
83 previousResult.networkStatus === result.networkStatus &&
84 equal(previousResult.data, result.data)) {
85 return;
86 }
87 _this.setResult(result);
88 };
89 var onError = function (error) {
90 var last = obsQuery["last"];
91 subscription.unsubscribe();
92 try {
93 obsQuery.resetLastResults();
94 subscription = obsQuery.subscribe(onNext, onError);
95 }
96 finally {
97 obsQuery["last"] = last;
98 }
99 if (!hasOwnProperty.call(error, 'graphQLErrors')) {
100 throw error;
101 }
102 var previousResult = _this.result;
103 if (!previousResult ||
104 (previousResult && previousResult.loading) ||
105 !equal(error, previousResult.error)) {
106 _this.setResult({
107 data: (previousResult && previousResult.data),
108 error: error,
109 loading: false,
110 networkStatus: NetworkStatus.error,
111 });
112 }
113 };
114 var subscription = obsQuery.subscribe(onNext, onError);
115 return function () { return subscription.unsubscribe(); };
116 }, [
117 obsQuery,
118 this.renderPromises,
119 this.client.disableNetworkFetches,
120 ]), function () { return _this.getCurrentResult(); }, function () { return _this.getCurrentResult(); });
121 this.unsafeHandlePartialRefetch(result);
122 var queryResult = this.toQueryResult(result);
123 if (!queryResult.loading && this.asyncResolveFns.size) {
124 this.asyncResolveFns.forEach(function (resolve) { return resolve(queryResult); });
125 this.asyncResolveFns.clear();
126 }
127 return queryResult;
128 };
129 InternalState.prototype.useOptions = function (options) {
130 var _a;
131 var watchQueryOptions = this.createWatchQueryOptions(this.queryHookOptions = options);
132 var currentWatchQueryOptions = this.watchQueryOptions;
133 if (this.optionsToIgnoreOnce.has(currentWatchQueryOptions) ||
134 !equal(watchQueryOptions, currentWatchQueryOptions)) {
135 this.watchQueryOptions = watchQueryOptions;
136 if (currentWatchQueryOptions && this.observable) {
137 this.optionsToIgnoreOnce.delete(currentWatchQueryOptions);
138 this.observable.reobserve(this.getObsQueryOptions());
139 this.previousData = ((_a = this.result) === null || _a === void 0 ? void 0 : _a.data) || this.previousData;
140 this.result = void 0;
141 }
142 }
143 this.onCompleted = options.onCompleted || InternalState.prototype.onCompleted;
144 this.onError = options.onError || InternalState.prototype.onError;
145 if ((this.renderPromises || this.client.disableNetworkFetches) &&
146 this.queryHookOptions.ssr === false &&
147 !this.queryHookOptions.skip) {
148 this.result = this.ssrDisabledResult;
149 }
150 else if (this.queryHookOptions.skip ||
151 this.watchQueryOptions.fetchPolicy === 'standby') {
152 this.result = this.skipStandbyResult;
153 }
154 else if (this.result === this.ssrDisabledResult ||
155 this.result === this.skipStandbyResult) {
156 this.result = void 0;
157 }
158 };
159 InternalState.prototype.getObsQueryOptions = function () {
160 var toMerge = [];
161 var globalDefaults = this.client.defaultOptions.watchQuery;
162 if (globalDefaults)
163 toMerge.push(globalDefaults);
164 if (this.queryHookOptions.defaultOptions) {
165 toMerge.push(this.queryHookOptions.defaultOptions);
166 }
167 toMerge.push(compact(this.observable && this.observable.options, this.watchQueryOptions));
168 return toMerge.reduce(mergeOptions);
169 };
170 InternalState.prototype.createWatchQueryOptions = function (_a) {
171 var _b;
172 if (_a === void 0) { _a = {}; }
173 var skip = _a.skip, ssr = _a.ssr, onCompleted = _a.onCompleted, onError = _a.onError, displayName = _a.displayName, defaultOptions = _a.defaultOptions, otherOptions = __rest(_a, ["skip", "ssr", "onCompleted", "onError", "displayName", "defaultOptions"]);
174 var watchQueryOptions = Object.assign(otherOptions, { query: this.query });
175 if (this.renderPromises &&
176 (watchQueryOptions.fetchPolicy === 'network-only' ||
177 watchQueryOptions.fetchPolicy === 'cache-and-network')) {
178 watchQueryOptions.fetchPolicy = 'cache-first';
179 }
180 if (!watchQueryOptions.variables) {
181 watchQueryOptions.variables = {};
182 }
183 if (skip) {
184 var _c = watchQueryOptions.fetchPolicy, fetchPolicy = _c === void 0 ? this.getDefaultFetchPolicy() : _c, _d = watchQueryOptions.initialFetchPolicy, initialFetchPolicy = _d === void 0 ? fetchPolicy : _d;
185 Object.assign(watchQueryOptions, {
186 initialFetchPolicy: initialFetchPolicy,
187 fetchPolicy: 'standby',
188 });
189 }
190 else if (!watchQueryOptions.fetchPolicy) {
191 watchQueryOptions.fetchPolicy =
192 ((_b = this.observable) === null || _b === void 0 ? void 0 : _b.options.initialFetchPolicy) ||
193 this.getDefaultFetchPolicy();
194 }
195 return watchQueryOptions;
196 };
197 InternalState.prototype.getDefaultFetchPolicy = function () {
198 var _a, _b;
199 return (((_a = this.queryHookOptions.defaultOptions) === null || _a === void 0 ? void 0 : _a.fetchPolicy) ||
200 ((_b = this.client.defaultOptions.watchQuery) === null || _b === void 0 ? void 0 : _b.fetchPolicy) ||
201 "cache-first");
202 };
203 InternalState.prototype.onCompleted = function (data) { };
204 InternalState.prototype.onError = function (error) { };
205 InternalState.prototype.useObservableQuery = function () {
206 var obsQuery = this.observable =
207 this.renderPromises
208 && this.renderPromises.getSSRObservable(this.watchQueryOptions)
209 || this.observable
210 || this.client.watchQuery(this.getObsQueryOptions());
211 this.obsQueryFields = useMemo(function () { return ({
212 refetch: obsQuery.refetch.bind(obsQuery),
213 reobserve: obsQuery.reobserve.bind(obsQuery),
214 fetchMore: obsQuery.fetchMore.bind(obsQuery),
215 updateQuery: obsQuery.updateQuery.bind(obsQuery),
216 startPolling: obsQuery.startPolling.bind(obsQuery),
217 stopPolling: obsQuery.stopPolling.bind(obsQuery),
218 subscribeToMore: obsQuery.subscribeToMore.bind(obsQuery),
219 }); }, [obsQuery]);
220 var ssrAllowed = !(this.queryHookOptions.ssr === false ||
221 this.queryHookOptions.skip);
222 if (this.renderPromises && ssrAllowed) {
223 this.renderPromises.registerSSRObservable(obsQuery);
224 if (obsQuery.getCurrentResult().loading) {
225 this.renderPromises.addObservableQueryPromise(obsQuery);
226 }
227 }
228 return obsQuery;
229 };
230 InternalState.prototype.setResult = function (nextResult) {
231 var previousResult = this.result;
232 if (previousResult && previousResult.data) {
233 this.previousData = previousResult.data;
234 }
235 this.result = nextResult;
236 this.forceUpdate();
237 this.handleErrorOrCompleted(nextResult);
238 };
239 InternalState.prototype.handleErrorOrCompleted = function (result) {
240 if (!result.loading) {
241 if (result.error) {
242 this.onError(result.error);
243 }
244 else if (result.data) {
245 this.onCompleted(result.data);
246 }
247 }
248 };
249 InternalState.prototype.getCurrentResult = function () {
250 if (!this.result) {
251 this.handleErrorOrCompleted(this.result = this.observable.getCurrentResult());
252 }
253 return this.result;
254 };
255 InternalState.prototype.toQueryResult = function (result) {
256 var queryResult = this.toQueryResultCache.get(result);
257 if (queryResult)
258 return queryResult;
259 var data = result.data, partial = result.partial, resultWithoutPartial = __rest(result, ["data", "partial"]);
260 this.toQueryResultCache.set(result, queryResult = __assign(__assign(__assign({ data: data }, resultWithoutPartial), this.obsQueryFields), { client: this.client, observable: this.observable, variables: this.observable.variables, called: !this.queryHookOptions.skip, previousData: this.previousData }));
261 if (!queryResult.error && isNonEmptyArray(result.errors)) {
262 queryResult.error = new ApolloError({ graphQLErrors: result.errors });
263 }
264 return queryResult;
265 };
266 InternalState.prototype.unsafeHandlePartialRefetch = function (result) {
267 if (result.partial &&
268 this.queryHookOptions.partialRefetch &&
269 !result.loading &&
270 (!result.data || Object.keys(result.data).length === 0) &&
271 this.observable.options.fetchPolicy !== 'cache-only') {
272 Object.assign(result, {
273 loading: true,
274 networkStatus: NetworkStatus.refetch,
275 });
276 this.observable.refetch();
277 }
278 };
279 return InternalState;
280}());
281//# sourceMappingURL=useQuery.js.map
\No newline at end of file