sanity 3.72.1 → 3.72.2-use-live-content-api.9

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (48) hide show
  1. package/lib/_chunks-cjs/LiveQueries.js +236 -163
  2. package/lib/_chunks-cjs/LiveQueries.js.map +1 -1
  3. package/lib/_chunks-cjs/PresentationToolGrantsCheck.js +3 -7
  4. package/lib/_chunks-cjs/PresentationToolGrantsCheck.js.map +1 -1
  5. package/lib/_chunks-cjs/buildAction.js +3 -0
  6. package/lib/_chunks-cjs/buildAction.js.map +1 -1
  7. package/lib/_chunks-cjs/presentation.js +3 -4
  8. package/lib/_chunks-cjs/presentation.js.map +1 -1
  9. package/lib/_chunks-cjs/version.js +1 -1
  10. package/lib/_chunks-es/LiveQueries.mjs +235 -162
  11. package/lib/_chunks-es/LiveQueries.mjs.map +1 -1
  12. package/lib/_chunks-es/PresentationToolGrantsCheck.mjs +2 -4
  13. package/lib/_chunks-es/PresentationToolGrantsCheck.mjs.map +1 -1
  14. package/lib/_chunks-es/presentation.mjs +3 -3
  15. package/lib/_chunks-es/presentation.mjs.map +1 -1
  16. package/lib/_chunks-es/version.mjs +1 -1
  17. package/lib/_internal/cli/threads/validateDocuments.js +1 -1
  18. package/lib/_internal/cli/threads/validateDocuments.js.map +1 -1
  19. package/lib/_legacy/LiveQueries.esm.js +235 -162
  20. package/lib/_legacy/LiveQueries.esm.js.map +1 -1
  21. package/lib/_legacy/PresentationToolGrantsCheck.esm.js +2 -4
  22. package/lib/_legacy/PresentationToolGrantsCheck.esm.js.map +1 -1
  23. package/lib/_legacy/presentation.esm.js +3 -3
  24. package/lib/_legacy/presentation.esm.js.map +1 -1
  25. package/lib/_legacy/version.esm.js +1 -1
  26. package/package.json +13 -14
  27. package/src/_internal/cli/server/buildVendorDependencies.ts +1 -0
  28. package/src/_internal/cli/threads/validateDocuments.ts +2 -2
  29. package/src/presentation/PresentationTool.tsx +10 -21
  30. package/src/presentation/constants.ts +4 -16
  31. package/src/presentation/loader/LiveQueries.tsx +79 -234
  32. package/src/presentation/loader/useLiveEvents.ts +75 -0
  33. package/src/presentation/loader/useLiveQueries.ts +127 -0
  34. package/src/presentation/loader/utils.ts +2 -125
  35. package/src/presentation/types.ts +1 -18
  36. package/lib/_chunks-cjs/LoaderQueries.js +0 -281
  37. package/lib/_chunks-cjs/LoaderQueries.js.map +0 -1
  38. package/lib/_chunks-cjs/utils.js +0 -70
  39. package/lib/_chunks-cjs/utils.js.map +0 -1
  40. package/lib/_chunks-es/LoaderQueries.mjs +0 -288
  41. package/lib/_chunks-es/LoaderQueries.mjs.map +0 -1
  42. package/lib/_chunks-es/utils.mjs +0 -74
  43. package/lib/_chunks-es/utils.mjs.map +0 -1
  44. package/lib/_legacy/LoaderQueries.esm.js +0 -288
  45. package/lib/_legacy/LoaderQueries.esm.js.map +0 -1
  46. package/lib/_legacy/utils.esm.js +0 -74
  47. package/lib/_legacy/utils.esm.js.map +0 -1
  48. package/src/presentation/loader/LoaderQueries.tsx +0 -564
@@ -1,132 +1,225 @@
1
1
  import { jsx, Fragment } from "react/jsx-runtime";
2
2
  import { c } from "react-compiler-runtime";
3
- import { createClient } from "@sanity/client";
4
3
  import { applySourceDocuments, getPublishedId } from "@sanity/client/csm";
5
4
  import { createConnectionMachine } from "@sanity/comlink";
6
5
  import { createCompatibilityActors } from "@sanity/presentation-comlink";
7
6
  import isEqual from "fast-deep-equal";
8
- import { useState, useEffect, useMemo, useDeferredValue, memo } from "react";
7
+ import { useReducer, useState, useEffect, useDeferredValue, memo, startTransition } from "react";
9
8
  import { useProjectId, useDataset, useClient } from "sanity";
10
9
  import { useEffectEvent } from "use-effect-event";
11
- import { MIN_LOADER_QUERY_LISTEN_HEARTBEAT_INTERVAL } from "./presentation.mjs";
12
- import { mapChangedValue, useQueryParams, useRevalidate } from "./utils.mjs";
13
- function LoaderQueries(props) {
14
- const {
15
- liveDocument: _liveDocument,
10
+ import { LOADER_QUERY_GC_INTERVAL, MIN_LOADER_QUERY_LISTEN_HEARTBEAT_INTERVAL, API_VERSION } from "./presentation.mjs";
11
+ import { toPlainText } from "@portabletext/react";
12
+ import { isPortableTextBlock } from "@portabletext/toolkit";
13
+ function reducer$1(state, event) {
14
+ switch (event.type) {
15
+ case "message":
16
+ return {
17
+ ...state,
18
+ messages: [...state.messages, event]
19
+ };
20
+ case "reconnect":
21
+ case "restart":
22
+ return {
23
+ ...state,
24
+ messages: [],
25
+ resets: state.resets + 1
26
+ };
27
+ case "welcome":
28
+ return state;
29
+ default:
30
+ throw Error(`Unknown event: ${// eslint-disable-next-line @typescript-eslint/no-explicit-any
31
+ event.type}`, {
32
+ cause: event
33
+ });
34
+ }
35
+ }
36
+ const initialState$1 = {
37
+ messages: [],
38
+ resets: 0
39
+ };
40
+ function useLiveEvents(client) {
41
+ const $ = c(3), [state, dispatch] = useReducer(reducer$1, initialState$1), [error, setError] = useState(null);
42
+ if (error !== null)
43
+ throw error;
44
+ let t0, t1;
45
+ return $[0] !== client.live ? (t0 = () => {
46
+ const subscription = client.live.events({
47
+ includeDrafts: !0,
48
+ tag: "presentation-loader"
49
+ }).subscribe({
50
+ next: dispatch,
51
+ error: (err) => setError(err instanceof Error ? err : new Error("Unexpected error in useLiveEvents", {
52
+ cause: err
53
+ }))
54
+ });
55
+ return () => subscription.unsubscribe();
56
+ }, t1 = [client.live], $[0] = client.live, $[1] = t0, $[2] = t1) : (t0 = $[1], t1 = $[2]), useEffect(t0, t1), useDeferredValue(state);
57
+ }
58
+ const mapChangedValue = (changedValue, {
59
+ previousValue
60
+ }) => {
61
+ if (typeof previousValue == "string") {
62
+ if (typeof changedValue == "number")
63
+ return `${changedValue}`;
64
+ if (Array.isArray(changedValue)) {
65
+ if (changedValue.length === 0)
66
+ return "";
67
+ if (changedValue.some((node) => typeof node == "object" && isPortableTextBlock(node)))
68
+ return toPlainText(changedValue);
69
+ }
70
+ }
71
+ return changedValue;
72
+ };
73
+ function getQueryCacheKey(perspective, query, params) {
74
+ return `${perspective}:${query}:${JSON.stringify(params)}`;
75
+ }
76
+ function gc(state) {
77
+ if (state.queries.size < 1)
78
+ return state;
79
+ const now = Date.now();
80
+ if (!Array.from(state.heartbeats.values()).some((entry) => entry.heartbeat !== !1 && now > entry.receivedAt + entry.heartbeat))
81
+ return state;
82
+ const nextHeartbeats = /* @__PURE__ */ new Map(), nextQueries = /* @__PURE__ */ new Map();
83
+ for (const [key, entry] of state.heartbeats.entries())
84
+ entry.heartbeat !== !1 && now > entry.receivedAt + entry.heartbeat || (nextHeartbeats.set(key, entry), nextQueries.set(key, state.queries.get(key)));
85
+ return {
86
+ ...state,
87
+ queries: nextQueries,
88
+ heartbeats: nextHeartbeats
89
+ };
90
+ }
91
+ function queryListen(state, {
92
+ payload
93
+ }) {
94
+ const key = getQueryCacheKey(payload.perspective, payload.query, payload.params), data = {
95
+ query: payload.query,
96
+ params: payload.params,
97
+ perspective: payload.perspective
98
+ }, nextHeartbeats = new Map(state.heartbeats);
99
+ nextHeartbeats.set(key, {
100
+ receivedAt: Date.now(),
101
+ heartbeat: payload.heartbeat
102
+ });
103
+ let nextQueries = state.queries;
104
+ return (!state.queries.has(key) || !isEqual(state.queries.get(key), data)) && (nextQueries = new Map(state.queries), nextQueries.set(key, data)), {
105
+ heartbeats: nextHeartbeats,
106
+ queries: nextQueries
107
+ };
108
+ }
109
+ function reducer(state, action) {
110
+ switch (action.type) {
111
+ case "query-listen":
112
+ return queryListen(state, action);
113
+ case "gc":
114
+ return gc(state);
115
+ default:
116
+ throw Error(`Unknown action: ${// eslint-disable-next-line @typescript-eslint/no-explicit-any
117
+ action.type}`, {
118
+ cause: action
119
+ });
120
+ }
121
+ }
122
+ const initialState = {
123
+ queries: /* @__PURE__ */ new Map(),
124
+ heartbeats: /* @__PURE__ */ new Map()
125
+ };
126
+ function useLiveQueries() {
127
+ const $ = c(4), [state, dispatch] = useReducer(reducer, initialState);
128
+ let t0, t1;
129
+ $[0] === Symbol.for("react.memo_cache_sentinel") ? (t0 = () => {
130
+ const interval = setInterval(() => dispatch({
131
+ type: "gc"
132
+ }), LOADER_QUERY_GC_INTERVAL);
133
+ return () => clearInterval(interval);
134
+ }, t1 = [], $[0] = t0, $[1] = t1) : (t0 = $[0], t1 = $[1]), useEffect(t0, t1);
135
+ const queries = useDeferredValue(state.queries);
136
+ let t2;
137
+ return $[2] !== queries ? (t2 = [queries, dispatch], $[2] = queries, $[3] = t2) : t2 = $[3], t2;
138
+ }
139
+ function LiveQueries(props) {
140
+ const $ = c(30), {
16
141
  controller,
17
142
  perspective: activePerspective,
18
143
  onLoadersConnection,
19
144
  onDocumentsOnPage
20
- } = props, [comlink, setComlink] = useState(), [liveQueries, setLiveQueries] = useState({}), projectId = useProjectId(), dataset = useDataset();
21
- useEffect(() => {
22
- const interval = setInterval(() => setLiveQueries((liveQueries_0) => {
23
- if (Object.keys(liveQueries_0).length < 1)
24
- return liveQueries_0;
25
- const now = Date.now();
26
- if (!Object.values(liveQueries_0).some(
27
- // eslint-disable-next-line max-nested-callbacks
28
- (liveQuery) => liveQuery.heartbeat !== !1 && now > liveQuery.receivedAt + liveQuery.heartbeat
29
- ))
30
- return liveQueries_0;
31
- const next = {};
32
- for (const [key, value] of Object.entries(liveQueries_0))
33
- value.heartbeat !== !1 && now > value.receivedAt + value.heartbeat || (next[key] = value);
34
- return next;
35
- }), MIN_LOADER_QUERY_LISTEN_HEARTBEAT_INTERVAL);
36
- return () => clearInterval(interval);
37
- }, []), useEffect(() => {
145
+ } = props, [comlink, setComlink] = useState(), [liveQueries, liveQueriesDispatch] = useLiveQueries(), projectId = useProjectId(), dataset = useDataset();
146
+ let t0, t1;
147
+ $[0] !== controller || $[1] !== dataset || $[2] !== liveQueriesDispatch || $[3] !== onDocumentsOnPage || $[4] !== onLoadersConnection || $[5] !== projectId ? (t0 = () => {
38
148
  if (controller) {
39
- const comlink_0 = controller.createChannel({
149
+ const nextComlink = controller.createChannel({
40
150
  name: "presentation",
41
151
  connectTo: "loaders",
42
152
  heartbeat: !0
43
153
  }, createConnectionMachine().provide({
44
154
  actors: createCompatibilityActors()
45
155
  }));
46
- return setComlink(comlink_0), comlink_0.onStatus(onLoadersConnection), comlink_0.on("loader/documents", (data) => {
47
- data.projectId === projectId && data.dataset === dataset && onDocumentsOnPage(
48
- "loaders",
49
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
50
- data.perspective,
51
- data.documents
52
- );
53
- }), comlink_0.on("loader/query-listen", (data_0) => {
156
+ return setComlink(nextComlink), nextComlink.onStatus(onLoadersConnection), nextComlink.on("loader/documents", (data) => {
157
+ data.projectId === projectId && data.dataset === dataset && onDocumentsOnPage("loaders", data.perspective, data.documents);
158
+ }), nextComlink.on("loader/query-listen", (data_0) => {
54
159
  if (data_0.projectId === projectId && data_0.dataset === dataset) {
55
160
  if (typeof data_0.heartbeat == "number" && data_0.heartbeat < MIN_LOADER_QUERY_LISTEN_HEARTBEAT_INTERVAL)
56
161
  throw new Error(`Loader query listen heartbeat interval must be at least ${MIN_LOADER_QUERY_LISTEN_HEARTBEAT_INTERVAL}ms`);
57
- setLiveQueries((prev) => ({
58
- ...prev,
59
- [getQueryCacheKey(data_0.query, data_0.params)]: {
162
+ liveQueriesDispatch({
163
+ type: "query-listen",
164
+ payload: {
60
165
  perspective: data_0.perspective,
61
166
  query: data_0.query,
62
167
  params: data_0.params,
63
- receivedAt: Date.now(),
64
168
  heartbeat: data_0.heartbeat ?? !1
65
169
  }
66
- }));
170
+ });
67
171
  }
68
- }), comlink_0.start();
172
+ }), nextComlink.start();
69
173
  }
70
- }, [controller, dataset, onDocumentsOnPage, onLoadersConnection, projectId]);
71
- const [syncTagsInUse] = useState(() => /* @__PURE__ */ new Set()), [lastLiveEventId, setLastLiveEventId] = useState(null), studioClient = useClient({
72
- apiVersion: "2023-10-16"
73
- }), clientConfig = useMemo(() => studioClient.config(), [studioClient]), client = useMemo(() => studioClient.withConfig({
174
+ return _temp;
175
+ }, t1 = [controller, dataset, liveQueriesDispatch, onDocumentsOnPage, onLoadersConnection, projectId], $[0] = controller, $[1] = dataset, $[2] = liveQueriesDispatch, $[3] = onDocumentsOnPage, $[4] = onLoadersConnection, $[5] = projectId, $[6] = t0, $[7] = t1) : (t0 = $[6], t1 = $[7]), useEffect(t0, t1);
176
+ let t2;
177
+ $[8] === Symbol.for("react.memo_cache_sentinel") ? (t2 = {
178
+ apiVersion: API_VERSION
179
+ }, $[8] = t2) : t2 = $[8];
180
+ const studioClient = useClient(t2);
181
+ let t3, t4;
182
+ $[9] !== studioClient ? (t4 = studioClient.withConfig({
74
183
  resultSourceMap: "withKeyArraySelector"
75
- }), [studioClient]);
76
- useEffect(() => {
77
- if (comlink) {
78
- const {
79
- projectId: projectId_0,
80
- dataset: dataset_0
81
- } = clientConfig;
82
- comlink.post("loader/perspective", {
83
- projectId: projectId_0,
84
- dataset: dataset_0,
85
- perspective: activePerspective
86
- });
87
- }
88
- }, [comlink, clientConfig, activePerspective]);
89
- const handleSyncTags = useEffectEvent((event) => {
90
- const flattenedSyncTags = Array.from(syncTagsInUse).flat();
91
- event.tags.some((tag) => flattenedSyncTags.includes(tag)) ? setLastLiveEventId(event.id) : console.log("No matching tags found", event.tags, {
92
- flattenedSyncTags
93
- });
94
- });
95
- useEffect(() => {
96
- const subscription = createClient(client.config()).withConfig({
97
- // Necessary for the live drafts to work
98
- apiVersion: "vX"
99
- }).live.events({
100
- includeDrafts: !0,
101
- tag: "presentation-loader"
102
- }).subscribe({
103
- next: (event_0) => {
104
- event_0.type === "message" ? handleSyncTags(event_0) : event_0.type === "restart" ? setLastLiveEventId(event_0.id) : event_0.type === "reconnect" && setLastLiveEventId(null);
105
- },
106
- // eslint-disable-next-line no-console
107
- error: (err) => console.error("Error validating EventSource URL:", err)
184
+ }), $[9] = studioClient, $[10] = t4) : t4 = $[10], t3 = t4;
185
+ const client = t3;
186
+ let t5, t6;
187
+ $[11] !== activePerspective || $[12] !== comlink || $[13] !== dataset || $[14] !== projectId ? (t5 = () => {
188
+ comlink && comlink.post("loader/perspective", {
189
+ projectId,
190
+ dataset,
191
+ perspective: activePerspective
108
192
  });
109
- return () => subscription.unsubscribe();
110
- }, [client, handleSyncTags]);
111
- const liveDocument = useDeferredValue(_liveDocument);
112
- return /* @__PURE__ */ jsx(Fragment, { children: Object.entries(liveQueries).map(([key_0, {
113
- query,
114
- params,
115
- perspective
116
- }]) => /* @__PURE__ */ jsx(QuerySubscription, { projectId: clientConfig.projectId, dataset: clientConfig.dataset, perspective, query, params, comlink, client, liveDocument, lastLiveEventId, syncTagsInUse }, `${key_0}${perspective}`)) });
193
+ }, t6 = [comlink, activePerspective, projectId, dataset], $[11] = activePerspective, $[12] = comlink, $[13] = dataset, $[14] = projectId, $[15] = t5, $[16] = t6) : (t5 = $[15], t6 = $[16]), useEffect(t5, t6);
194
+ const liveDocument = useDeferredValue(props.liveDocument), liveEvents = useLiveEvents(client);
195
+ let t7;
196
+ $[17] !== liveQueries ? (t7 = liveQueries.entries(), $[17] = liveQueries, $[18] = t7) : t7 = $[18];
197
+ let t8;
198
+ $[19] !== t7 ? (t8 = [...t7], $[19] = t7, $[20] = t8) : t8 = $[20];
199
+ let t9;
200
+ return $[21] !== client || $[22] !== comlink || $[23] !== dataset || $[24] !== liveDocument || $[25] !== liveEvents.messages || $[26] !== liveEvents.resets || $[27] !== projectId || $[28] !== t8 ? (t9 = /* @__PURE__ */ jsx(Fragment, { children: t8.map((t10) => {
201
+ const [key, t11] = t10, {
202
+ query,
203
+ params,
204
+ perspective
205
+ } = t11;
206
+ return /* @__PURE__ */ jsx(QuerySubscription, { projectId, dataset, perspective, query, params, comlink, client, liveDocument, liveEventsMessages: liveEvents.messages }, `${liveEvents.resets}:${key}`);
207
+ }) }), $[21] = client, $[22] = comlink, $[23] = dataset, $[24] = liveDocument, $[25] = liveEvents.messages, $[26] = liveEvents.resets, $[27] = projectId, $[28] = t8, $[29] = t9) : t9 = $[29], t9;
208
+ }
209
+ function _temp() {
117
210
  }
118
211
  function QuerySubscriptionComponent(props) {
119
- const $ = c(14), {
212
+ const $ = c(13), {
120
213
  projectId,
121
214
  dataset,
122
215
  perspective,
123
216
  query,
124
217
  client,
125
218
  liveDocument,
219
+ params,
126
220
  comlink,
127
- lastLiveEventId,
128
- syncTagsInUse
129
- } = props, params = useQueryParams(props.params), {
221
+ liveEventsMessages
222
+ } = props, {
130
223
  result,
131
224
  resultSourceMap,
132
225
  syncTags: tags
@@ -136,7 +229,7 @@ function QuerySubscriptionComponent(props) {
136
229
  params,
137
230
  perspective,
138
231
  query,
139
- lastLiveEventId
232
+ liveEventsMessages
140
233
  }) || {};
141
234
  let t0;
142
235
  $[0] !== dataset || $[1] !== projectId ? (t0 = (comlink_0, perspective_0, query_0, params_0, result_0, resultSourceMap_0, tags_0) => {
@@ -153,98 +246,81 @@ function QuerySubscriptionComponent(props) {
153
246
  }, $[0] = dataset, $[1] = projectId, $[2] = t0) : t0 = $[2];
154
247
  const handleQueryChange = useEffectEvent(t0);
155
248
  let t1, t2;
156
- return $[3] !== comlink || $[4] !== handleQueryChange || $[5] !== params || $[6] !== perspective || $[7] !== query || $[8] !== result || $[9] !== resultSourceMap || $[10] !== syncTagsInUse || $[11] !== tags ? (t1 = () => {
157
- if (resultSourceMap && handleQueryChange(comlink, perspective, query, params, result, resultSourceMap, tags), Array.isArray(tags))
158
- return syncTagsInUse.add(tags), () => {
159
- syncTagsInUse.delete(tags);
160
- };
161
- }, t2 = [comlink, handleQueryChange, params, perspective, query, result, resultSourceMap, syncTagsInUse, tags], $[3] = comlink, $[4] = handleQueryChange, $[5] = params, $[6] = perspective, $[7] = query, $[8] = result, $[9] = resultSourceMap, $[10] = syncTagsInUse, $[11] = tags, $[12] = t1, $[13] = t2) : (t1 = $[12], t2 = $[13]), useEffect(t1, t2), null;
249
+ return $[3] !== comlink || $[4] !== handleQueryChange || $[5] !== params || $[6] !== perspective || $[7] !== query || $[8] !== result || $[9] !== resultSourceMap || $[10] !== tags ? (t1 = () => {
250
+ resultSourceMap && handleQueryChange(comlink, perspective, query, params, result, resultSourceMap, tags);
251
+ }, t2 = [comlink, handleQueryChange, params, perspective, query, result, resultSourceMap, tags], $[3] = comlink, $[4] = handleQueryChange, $[5] = params, $[6] = perspective, $[7] = query, $[8] = result, $[9] = resultSourceMap, $[10] = tags, $[11] = t1, $[12] = t2) : (t1 = $[11], t2 = $[12]), useEffect(t1, t2), null;
162
252
  }
163
253
  const QuerySubscription = memo(QuerySubscriptionComponent);
164
254
  QuerySubscription.displayName = "Memo(QuerySubscription)";
165
255
  function useQuerySubscription(props) {
166
- const $ = c(25), {
256
+ const $ = c(30), {
167
257
  liveDocument,
168
258
  client,
169
259
  query,
170
260
  params,
171
261
  perspective,
172
- lastLiveEventId
173
- } = props, [snapshot, setSnapshot] = useState(null), [error, setError] = useState(null);
262
+ liveEventsMessages
263
+ } = props, [result, setResult] = useState(null), [resultSourceMap, setResultSourceMap] = useState(null), [syncTags, setSyncTags] = useState(void 0);
264
+ let t0;
265
+ $[0] !== liveEventsMessages ? (t0 = () => new Set(liveEventsMessages.map(_temp2)), $[0] = liveEventsMessages, $[1] = t0) : t0 = $[1];
266
+ const [skipEventIds] = useState(t0);
267
+ let t1;
268
+ if ($[2] !== liveEventsMessages || $[3] !== skipEventIds || $[4] !== syncTags) {
269
+ let t22;
270
+ $[6] !== skipEventIds ? (t22 = (msg_0) => !skipEventIds.has(msg_0.id), $[6] = skipEventIds, $[7] = t22) : t22 = $[7];
271
+ const recentLiveEvents = liveEventsMessages.filter(t22);
272
+ let t32;
273
+ $[8] !== syncTags ? (t32 = (msg_1) => msg_1.tags.some((tag) => syncTags?.includes(tag)), $[8] = syncTags, $[9] = t32) : t32 = $[9], t1 = recentLiveEvents.findLast(t32), $[2] = liveEventsMessages, $[3] = skipEventIds, $[4] = syncTags, $[5] = t1;
274
+ } else
275
+ t1 = $[5];
276
+ const lastLiveEventId = t1?.id, [error, setError] = useState(null);
174
277
  if (error)
175
278
  throw error;
176
- let t0;
177
- $[0] === Symbol.for("react.memo_cache_sentinel") ? (t0 = {
178
- refreshInterval: 0
179
- }, $[0] = t0) : t0 = $[0];
180
- const [revalidate, startRefresh] = useRevalidate(t0), shouldRefetch = revalidate === "refresh" || revalidate === "inflight" || lastLiveEventId !== snapshot?.lastLiveEventId;
181
- let t1, t2;
182
- $[1] !== client || $[2] !== lastLiveEventId || $[3] !== params || $[4] !== perspective || $[5] !== query || $[6] !== shouldRefetch || $[7] !== startRefresh ? (t1 = () => {
183
- if (!shouldRefetch)
184
- return;
185
- let fulfilled;
186
- fulfilled = !1;
187
- let fetching;
188
- fetching = !1;
189
- const controller = new AbortController(), effect = async function() {
190
- const {
191
- signal
192
- } = controller;
193
- fetching = !0;
194
- const {
195
- result,
196
- resultSourceMap,
197
- syncTags
198
- } = await client.fetch(query, params, {
199
- lastLiveEventId,
200
- tag: "presentation-loader",
201
- signal,
202
- perspective,
203
- filterResponse: !1,
204
- returnQuery: !1
279
+ let t2, t3;
280
+ $[10] !== client || $[11] !== lastLiveEventId || $[12] !== params || $[13] !== perspective || $[14] !== query ? (t2 = () => {
281
+ const controller = new AbortController();
282
+ return client.fetch(query, params, {
283
+ lastLiveEventId,
284
+ tag: "presentation-loader",
285
+ signal: controller.signal,
286
+ perspective,
287
+ filterResponse: !1,
288
+ returnQuery: !1
289
+ }).then((response) => {
290
+ startTransition(() => {
291
+ setResult((prev) => isEqual(prev, response.result) ? prev : response.result), setResultSourceMap((prev_0) => isEqual(prev_0, response.resultSourceMap) ? prev_0 : response.resultSourceMap), setSyncTags((prev_1) => isEqual(prev_1, response.syncTags) ? prev_1 : response.syncTags);
205
292
  });
206
- fetching = !1, signal.aborted || (setSnapshot((prev) => ({
207
- result: isEqual(prev?.result, result) ? prev?.result : result,
208
- resultSourceMap: isEqual(prev?.resultSourceMap, resultSourceMap) ? prev?.resultSourceMap : resultSourceMap,
209
- syncTags: isEqual(prev?.syncTags, syncTags) ? prev?.syncTags : syncTags,
210
- lastLiveEventId
211
- })), fulfilled = !0);
212
- }, onFinally = startRefresh();
213
- return effect().catch((error_0) => {
214
- fetching = !1, error_0.name !== "AbortError" && setError(error_0);
215
- }).finally(onFinally), () => {
216
- !fulfilled && !fetching && controller.abort();
293
+ }).catch((err) => {
294
+ (typeof err != "object" || err?.name !== "AbortError") && setError(err);
295
+ }), () => {
296
+ controller.abort();
217
297
  };
218
- }, t2 = [client, lastLiveEventId, params, perspective, query, shouldRefetch, startRefresh], $[1] = client, $[2] = lastLiveEventId, $[3] = params, $[4] = perspective, $[5] = query, $[6] = shouldRefetch, $[7] = startRefresh, $[8] = t1, $[9] = t2) : (t1 = $[8], t2 = $[9]), useEffect(t1, t2);
219
- let t3;
220
- $[10] !== snapshot ? (t3 = snapshot ?? {}, $[10] = snapshot, $[11] = t3) : t3 = $[11];
221
- const {
222
- result: result_0,
223
- resultSourceMap: resultSourceMap_0,
224
- syncTags: syncTags_0
225
- } = t3;
298
+ }, t3 = [client, lastLiveEventId, params, perspective, query], $[10] = client, $[11] = lastLiveEventId, $[12] = params, $[13] = perspective, $[14] = query, $[15] = t2, $[16] = t3) : (t2 = $[15], t3 = $[16]), useEffect(t2, t3);
226
299
  let t4;
227
300
  bb0: {
228
- if (liveDocument && resultSourceMap_0) {
301
+ if (liveDocument && resultSourceMap) {
229
302
  let t52;
230
- $[12] !== liveDocument || $[13] !== perspective || $[14] !== resultSourceMap_0 || $[15] !== result_0 ? (t52 = turboChargeResultIfSourceMap(liveDocument, result_0, perspective, resultSourceMap_0), $[12] = liveDocument, $[13] = perspective, $[14] = resultSourceMap_0, $[15] = result_0, $[16] = t52) : t52 = $[16];
303
+ $[17] !== liveDocument || $[18] !== perspective || $[19] !== result || $[20] !== resultSourceMap ? (t52 = turboChargeResultIfSourceMap(liveDocument, result, perspective, resultSourceMap), $[17] = liveDocument, $[18] = perspective, $[19] = result, $[20] = resultSourceMap, $[21] = t52) : t52 = $[21];
231
304
  let t6;
232
- $[17] !== resultSourceMap_0 || $[18] !== syncTags_0 || $[19] !== t52 ? (t6 = {
305
+ $[22] !== resultSourceMap || $[23] !== syncTags || $[24] !== t52 ? (t6 = {
233
306
  result: t52,
234
- resultSourceMap: resultSourceMap_0,
235
- syncTags: syncTags_0
236
- }, $[17] = resultSourceMap_0, $[18] = syncTags_0, $[19] = t52, $[20] = t6) : t6 = $[20], t4 = t6;
307
+ resultSourceMap,
308
+ syncTags
309
+ }, $[22] = resultSourceMap, $[23] = syncTags, $[24] = t52, $[25] = t6) : t6 = $[25], t4 = t6;
237
310
  break bb0;
238
311
  }
239
312
  let t5;
240
- $[21] !== resultSourceMap_0 || $[22] !== result_0 || $[23] !== syncTags_0 ? (t5 = {
241
- result: result_0,
242
- resultSourceMap: resultSourceMap_0,
243
- syncTags: syncTags_0
244
- }, $[21] = resultSourceMap_0, $[22] = result_0, $[23] = syncTags_0, $[24] = t5) : t5 = $[24], t4 = t5;
313
+ $[26] !== result || $[27] !== resultSourceMap || $[28] !== syncTags ? (t5 = {
314
+ result,
315
+ resultSourceMap,
316
+ syncTags
317
+ }, $[26] = result, $[27] = resultSourceMap, $[28] = syncTags, $[29] = t5) : t5 = $[29], t4 = t5;
245
318
  }
246
319
  return t4;
247
320
  }
321
+ function _temp2(msg) {
322
+ return msg.id;
323
+ }
248
324
  function turboChargeResultIfSourceMap(liveDocument, result, perspective, resultSourceMap) {
249
325
  if (perspective === "raw")
250
326
  throw new Error("turboChargeResultIfSourceMap does not support raw perspective");
@@ -257,11 +333,8 @@ function turboChargeResultIfSourceMap(liveDocument, result, perspective, resultS
257
333
  } : null
258
334
  ), mapChangedValue, perspective);
259
335
  }
260
- function getQueryCacheKey(query, params) {
261
- return `${query}-${typeof params == "string" ? params : JSON.stringify(params)}`;
262
- }
263
336
  export {
264
- LoaderQueries as default,
337
+ LiveQueries as default,
265
338
  turboChargeResultIfSourceMap
266
339
  };
267
340
  //# sourceMappingURL=LiveQueries.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"LiveQueries.mjs","sources":["../../src/presentation/loader/LiveQueries.tsx"],"sourcesContent":["/* eslint-disable @typescript-eslint/no-shadow */\nimport {\n type ClientPerspective,\n type ContentSourceMap,\n createClient,\n type LiveEventMessage,\n type QueryParams,\n type SyncTag,\n} from '@sanity/client'\nimport {applySourceDocuments, getPublishedId} from '@sanity/client/csm'\nimport {\n type ChannelInstance,\n type Controller,\n createConnectionMachine,\n type StatusEvent,\n} from '@sanity/comlink'\nimport {\n createCompatibilityActors,\n type LoaderControllerMsg,\n type LoaderNodeMsg,\n} from '@sanity/presentation-comlink'\nimport isEqual from 'fast-deep-equal'\n// import {createPreviewSecret} from '@sanity/preview-url-secret/create-secret'\nimport {memo, useDeferredValue, useEffect, useMemo, useState} from 'react'\nimport {\n type SanityClient,\n type SanityDocument,\n useClient,\n // useCurrentUser,\n useDataset,\n useProjectId,\n} from 'sanity'\nimport {useEffectEvent} from 'use-effect-event'\n\n// import {useEffectEvent} from 'use-effect-event'\nimport {MIN_LOADER_QUERY_LISTEN_HEARTBEAT_INTERVAL} from '../constants'\nimport {\n type LiveQueriesState,\n type LiveQueriesStateValue,\n type LoaderConnection,\n type PresentationPerspective,\n} from '../types'\nimport {type DocumentOnPage} from '../useDocumentsOnPage'\nimport {mapChangedValue, useQueryParams, useRevalidate} from './utils'\n\nexport interface LoaderQueriesProps {\n liveDocument: Partial<SanityDocument> | null | undefined\n controller: Controller | undefined\n perspective: ClientPerspective\n onLoadersConnection: (event: StatusEvent) => void\n onDocumentsOnPage: (\n key: string,\n perspective: PresentationPerspective,\n state: DocumentOnPage[],\n ) => void\n}\n\nexport default function LoaderQueries(props: LoaderQueriesProps): React.JSX.Element {\n const {\n liveDocument: _liveDocument,\n controller,\n perspective: activePerspective,\n onLoadersConnection,\n onDocumentsOnPage,\n } = props\n\n const [comlink, setComlink] = useState<ChannelInstance<LoaderControllerMsg, LoaderNodeMsg>>()\n const [liveQueries, setLiveQueries] = useState<LiveQueriesState>({})\n\n const projectId = useProjectId()\n const dataset = useDataset()\n\n useEffect(() => {\n const interval = setInterval(\n () =>\n setLiveQueries((liveQueries) => {\n if (Object.keys(liveQueries).length < 1) {\n return liveQueries\n }\n\n const now = Date.now()\n const hasAnyExpired = Object.values(liveQueries).some(\n // eslint-disable-next-line max-nested-callbacks\n (liveQuery) =>\n liveQuery.heartbeat !== false && now > liveQuery.receivedAt + liveQuery.heartbeat,\n )\n if (!hasAnyExpired) {\n return liveQueries\n }\n const next = {} as LiveQueriesState\n for (const [key, value] of Object.entries(liveQueries)) {\n if (value.heartbeat !== false && now > value.receivedAt + value.heartbeat) {\n continue\n }\n next[key] = value\n }\n return next\n }),\n MIN_LOADER_QUERY_LISTEN_HEARTBEAT_INTERVAL,\n )\n return () => clearInterval(interval)\n }, [])\n\n useEffect(() => {\n if (controller) {\n const comlink = controller.createChannel<LoaderControllerMsg, LoaderNodeMsg>(\n {\n name: 'presentation',\n connectTo: 'loaders',\n heartbeat: true,\n },\n createConnectionMachine<LoaderControllerMsg, LoaderNodeMsg>().provide({\n actors: createCompatibilityActors<LoaderControllerMsg>(),\n }),\n )\n setComlink(comlink)\n\n comlink.onStatus(onLoadersConnection)\n\n comlink.on('loader/documents', (data) => {\n if (data.projectId === projectId && data.dataset === dataset) {\n onDocumentsOnPage(\n 'loaders',\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n data.perspective as unknown as any,\n data.documents,\n )\n }\n })\n\n comlink.on('loader/query-listen', (data) => {\n if (data.projectId === projectId && data.dataset === dataset) {\n if (\n typeof data.heartbeat === 'number' &&\n data.heartbeat < MIN_LOADER_QUERY_LISTEN_HEARTBEAT_INTERVAL\n ) {\n throw new Error(\n `Loader query listen heartbeat interval must be at least ${MIN_LOADER_QUERY_LISTEN_HEARTBEAT_INTERVAL}ms`,\n )\n }\n setLiveQueries((prev) => ({\n ...prev,\n [getQueryCacheKey(data.query, data.params)]: {\n perspective: data.perspective,\n query: data.query,\n params: data.params,\n receivedAt: Date.now(),\n heartbeat: data.heartbeat ?? false,\n } satisfies LiveQueriesStateValue,\n }))\n }\n })\n\n return comlink.start()\n }\n return undefined\n }, [controller, dataset, onDocumentsOnPage, onLoadersConnection, projectId])\n\n // const currentUser = useCurrentUser()\n // const handleCreatePreviewUrlSecret = useEffectEvent(\n // async ({projectId, dataset}: {projectId: string; dataset: string}) => {\n // try {\n // // eslint-disable-next-line no-console\n // console.log('Creating preview URL secret for ', {projectId, dataset})\n // const {secret} = await createPreviewSecret(\n // client,\n // '@sanity/presentation',\n // typeof window === 'undefined' ? '' : location.href,\n // currentUser?.id,\n // )\n // return {secret}\n // } catch (err) {\n // // eslint-disable-next-line no-console\n // console.error('Failed to generate preview URL secret', err)\n // return {secret: null}\n // }\n // },\n // )\n // useEffect(() => {\n // return comlink?.on('loader/fetch-preview-url-secret', (data) =>\n // handleCreatePreviewUrlSecret(data),\n // )\n // }, [comlink, handleCreatePreviewUrlSecret])\n\n const [syncTagsInUse] = useState(() => new Set<SyncTag[]>())\n const [lastLiveEventId, setLastLiveEventId] = useState<string | null>(null)\n const studioClient = useClient({apiVersion: '2023-10-16'})\n const clientConfig = useMemo(() => studioClient.config(), [studioClient])\n const client = useMemo(\n () =>\n studioClient.withConfig({\n resultSourceMap: 'withKeyArraySelector',\n }),\n [studioClient],\n )\n useEffect(() => {\n if (comlink) {\n // eslint-disable-next-line @typescript-eslint/no-shadow\n const {projectId, dataset} = clientConfig\n comlink.post('loader/perspective', {\n projectId: projectId!,\n dataset: dataset!,\n perspective: activePerspective,\n })\n }\n }, [comlink, clientConfig, activePerspective])\n\n const handleSyncTags = useEffectEvent((event: LiveEventMessage) => {\n const flattenedSyncTags = Array.from(syncTagsInUse).flat()\n const hasMatchingTags = event.tags.some((tag) => flattenedSyncTags.includes(tag))\n if (hasMatchingTags) {\n setLastLiveEventId(event.id)\n } else {\n // eslint-disable-next-line no-console\n console.log('No matching tags found', event.tags, {flattenedSyncTags})\n }\n })\n useEffect(() => {\n const liveClient = createClient(client.config()).withConfig({\n // Necessary for the live drafts to work\n apiVersion: 'vX',\n })\n const subscription = liveClient.live\n .events({includeDrafts: true, tag: 'presentation-loader'})\n .subscribe({\n next: (event) => {\n if (event.type === 'message') {\n handleSyncTags(event)\n } else if (event.type === 'restart') {\n setLastLiveEventId(event.id)\n } else if (event.type === 'reconnect') {\n setLastLiveEventId(null)\n }\n },\n // eslint-disable-next-line no-console\n error: (err) => console.error('Error validating EventSource URL:', err),\n })\n return () => subscription.unsubscribe()\n }, [client, handleSyncTags])\n\n /**\n * Defer the liveDocument to avoid unnecessary rerenders on rapid edits\n */\n const liveDocument = useDeferredValue(_liveDocument)\n\n return (\n <>\n {Object.entries(liveQueries).map(([key, {query, params, perspective}]) => (\n <QuerySubscription\n key={`${key}${perspective}`}\n projectId={clientConfig.projectId!}\n dataset={clientConfig.dataset!}\n perspective={perspective}\n query={query}\n params={params}\n comlink={comlink}\n client={client}\n liveDocument={liveDocument}\n lastLiveEventId={lastLiveEventId}\n syncTagsInUse={syncTagsInUse}\n />\n ))}\n </>\n )\n}\n\ninterface SharedProps {\n /**\n * The Sanity client to use for fetching data and listening to mutations.\n */\n client: SanityClient\n}\n\ninterface QuerySubscriptionProps\n extends Pick<UseQuerySubscriptionProps, 'client' | 'liveDocument' | 'lastLiveEventId'> {\n projectId: string\n dataset: string\n perspective: ClientPerspective\n query: string\n params: QueryParams\n comlink: LoaderConnection | undefined\n syncTagsInUse: Set<SyncTag[]>\n}\nfunction QuerySubscriptionComponent(props: QuerySubscriptionProps) {\n const {\n projectId,\n dataset,\n perspective,\n query,\n client,\n liveDocument,\n comlink,\n lastLiveEventId,\n syncTagsInUse,\n } = props\n\n const params = useQueryParams(props.params)\n const {\n result,\n resultSourceMap,\n syncTags: tags,\n } = useQuerySubscription({\n client,\n liveDocument,\n params,\n perspective,\n query,\n lastLiveEventId,\n }) || {}\n\n const handleQueryChange = useEffectEvent(\n (\n comlink: LoaderConnection | undefined,\n perspective: ClientPerspective,\n query: string,\n params: QueryParams,\n result: unknown,\n resultSourceMap: ContentSourceMap | undefined,\n tags: `s1:${string}`[] | undefined,\n // eslint-disable-next-line max-params\n ) => {\n comlink?.post('loader/query-change', {\n projectId,\n dataset,\n perspective,\n query,\n params,\n result,\n resultSourceMap,\n tags,\n })\n },\n )\n useEffect(() => {\n if (resultSourceMap) {\n handleQueryChange(comlink, perspective, query, params, result, resultSourceMap, tags)\n }\n if (Array.isArray(tags)) {\n syncTagsInUse.add(tags)\n return () => {\n syncTagsInUse.delete(tags)\n }\n }\n return undefined\n }, [\n comlink,\n handleQueryChange,\n params,\n perspective,\n query,\n result,\n resultSourceMap,\n syncTagsInUse,\n tags,\n ])\n\n return null\n}\nconst QuerySubscription = memo(QuerySubscriptionComponent)\nQuerySubscription.displayName = 'Memo(QuerySubscription)'\n\ninterface UseQuerySubscriptionProps extends Required<Pick<SharedProps, 'client'>> {\n liveDocument: Partial<SanityDocument> | null | undefined\n query: string\n params: QueryParams\n perspective: ClientPerspective\n lastLiveEventId: string | null\n}\nfunction useQuerySubscription(props: UseQuerySubscriptionProps) {\n const {liveDocument, client, query, params, perspective, lastLiveEventId} = props\n const [snapshot, setSnapshot] = useState<{\n result: unknown\n resultSourceMap?: ContentSourceMap\n syncTags?: SyncTag[]\n lastLiveEventId: string | null\n } | null>(null)\n // Make sure any async errors bubble up to the nearest error boundary\n const [error, setError] = useState<unknown>(null)\n if (error) throw error\n\n const [revalidate, startRefresh] = useRevalidate({\n // Refresh interval is set to zero as we're using the Live Draft Content API to revalidate queries\n refreshInterval: 0,\n })\n const shouldRefetch =\n revalidate === 'refresh' ||\n revalidate === 'inflight' ||\n lastLiveEventId !== snapshot?.lastLiveEventId\n useEffect(() => {\n if (!shouldRefetch) {\n return undefined\n }\n\n let fulfilled = false\n let fetching = false\n const controller = new AbortController()\n // eslint-disable-next-line no-inner-declarations\n async function effect() {\n const {signal} = controller\n fetching = true\n const {result, resultSourceMap, syncTags} = await client.fetch(query, params, {\n lastLiveEventId,\n tag: 'presentation-loader',\n signal,\n perspective,\n filterResponse: false,\n returnQuery: false,\n })\n fetching = false\n\n if (!signal.aborted) {\n setSnapshot((prev) => ({\n result: isEqual(prev?.result, result) ? prev?.result : result,\n resultSourceMap: isEqual(prev?.resultSourceMap, resultSourceMap)\n ? prev?.resultSourceMap\n : resultSourceMap,\n syncTags: isEqual(prev?.syncTags, syncTags) ? prev?.syncTags : syncTags,\n lastLiveEventId,\n }))\n fulfilled = true\n }\n }\n const onFinally = startRefresh()\n effect()\n .catch((error) => {\n fetching = false\n if (error.name !== 'AbortError') {\n setError(error)\n }\n })\n .finally(onFinally)\n return () => {\n if (!fulfilled && !fetching) {\n controller.abort()\n }\n }\n }, [client, lastLiveEventId, params, perspective, query, shouldRefetch, startRefresh])\n\n const {result, resultSourceMap, syncTags} = snapshot ?? {}\n return useMemo(() => {\n if (liveDocument && resultSourceMap) {\n return {\n result: turboChargeResultIfSourceMap(liveDocument, result, perspective, resultSourceMap),\n resultSourceMap,\n syncTags,\n }\n }\n return {result, resultSourceMap, syncTags}\n }, [liveDocument, perspective, result, resultSourceMap, syncTags])\n}\n\nexport function turboChargeResultIfSourceMap<T = unknown>(\n liveDocument: Partial<SanityDocument> | null | undefined,\n result: T,\n perspective: ClientPerspective,\n resultSourceMap?: ContentSourceMap,\n): T {\n if (perspective === 'raw') {\n throw new Error('turboChargeResultIfSourceMap does not support raw perspective')\n }\n return applySourceDocuments(\n result,\n resultSourceMap,\n (sourceDocument) => {\n // If there's a displayed document, always prefer it\n if (\n // If _projectId is set, it's a cross dataset reference and we should skip it\n !sourceDocument._projectId &&\n liveDocument?._id &&\n getPublishedId(liveDocument._id) === getPublishedId(sourceDocument._id)\n ) {\n if (typeof liveDocument._id === 'string' && typeof sourceDocument._type === 'string') {\n return liveDocument as unknown as Required<Pick<SanityDocument, '_id' | '_type'>>\n }\n return {\n ...liveDocument,\n _id: liveDocument._id || sourceDocument._id,\n _type: liveDocument._type || sourceDocument._type,\n }\n }\n return null\n },\n mapChangedValue,\n perspective,\n )\n}\n\nfunction getQueryCacheKey(query: string, params: QueryParams | string): `${string}-${string}` {\n return `${query}-${typeof params === 'string' ? params : JSON.stringify(params)}`\n}\n"],"names":["LoaderQueries","props","liveDocument","_liveDocument","controller","perspective","activePerspective","onLoadersConnection","onDocumentsOnPage","comlink","setComlink","useState","liveQueries","setLiveQueries","projectId","useProjectId","dataset","useDataset","useEffect","interval","setInterval","Object","keys","length","now","Date","values","some","liveQuery","heartbeat","receivedAt","next","key","value","entries","MIN_LOADER_QUERY_LISTEN_HEARTBEAT_INTERVAL","clearInterval","createChannel","name","connectTo","createConnectionMachine","provide","actors","createCompatibilityActors","onStatus","on","data","documents","Error","prev","getQueryCacheKey","query","params","start","syncTagsInUse","Set","lastLiveEventId","setLastLiveEventId","studioClient","useClient","apiVersion","clientConfig","useMemo","config","client","withConfig","resultSourceMap","post","handleSyncTags","useEffectEvent","event","flattenedSyncTags","Array","from","flat","tags","tag","includes","id","console","log","subscription","createClient","live","events","includeDrafts","subscribe","type","error","err","unsubscribe","useDeferredValue","map","QuerySubscriptionComponent","$","_c","useQueryParams","result","syncTags","useQuerySubscription","t0","comlink_0","perspective_0","query_0","params_0","result_0","resultSourceMap_0","tags_0","handleQueryChange","t1","t2","isArray","add","delete","QuerySubscription","memo","displayName","snapshot","setSnapshot","setError","Symbol","for","refreshInterval","revalidate","startRefresh","useRevalidate","shouldRefetch","fulfilled","fetching","AbortController","effect","signal","fetch","filterResponse","returnQuery","aborted","isEqual","onFinally","catch","error_0","finally","abort","t3","syncTags_0","t4","t5","turboChargeResultIfSourceMap","t6","bb0","applySourceDocuments","sourceDocument","_projectId","_id","getPublishedId","_type","mapChangedValue","JSON","stringify"],"mappings":"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"}
1
+ {"version":3,"file":"LiveQueries.mjs","sources":["../../src/presentation/loader/useLiveEvents.ts","../../src/presentation/loader/utils.ts","../../src/presentation/loader/useLiveQueries.ts","../../src/presentation/loader/LiveQueries.tsx"],"sourcesContent":["import {type LiveEvent, type LiveEventMessage} from '@sanity/client'\nimport {useDeferredValue, useEffect, useReducer, useState} from 'react'\nimport {type SanityClient} from 'sanity'\n\ntype State = {\n /**\n * Growing list over live events with Sync Tags,\n * that can be used to refetch with Sanity Client, using the id as the lastLiveEventId parameter\n */\n messages: LiveEventMessage[]\n /**\n * If the connection experiences a reconnect, or a restart event is received, the counter is incremented.\n * This counter is suitable as a `key` on React Components as a way to reset its internal state and refetch.\n */\n resets: number\n}\n\nexport function reducer(state: State, event: LiveEvent): State {\n switch (event.type) {\n case 'message':\n return {\n ...state,\n messages: [...state.messages, event],\n }\n case 'reconnect':\n case 'restart':\n return {\n ...state,\n messages: [],\n resets: state.resets + 1,\n }\n case 'welcome':\n // no-op\n return state\n default:\n throw Error(\n `Unknown event: ${\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n (event as any).type\n }`,\n {cause: event},\n )\n }\n}\n\nexport const initialState: State = {\n messages: [],\n resets: 0,\n}\n\nexport function useLiveEvents(client: SanityClient): State {\n const [state, dispatch] = useReducer(reducer, initialState)\n const [error, setError] = useState<unknown>(null)\n if (error !== null) {\n // Push error to nearest error boundary\n throw error\n }\n\n useEffect(() => {\n const subscription = client.live\n .events({includeDrafts: true, tag: 'presentation-loader'})\n .subscribe({\n next: dispatch,\n error: (err) =>\n setError(\n err instanceof Error\n ? err\n : new Error('Unexpected error in useLiveEvents', {cause: err}),\n ),\n })\n return () => subscription.unsubscribe()\n }, [client.live])\n\n return useDeferredValue(state)\n}\n","import {toPlainText} from '@portabletext/react'\nimport {isPortableTextBlock} from '@portabletext/toolkit'\nimport {type ClientPerspective, type QueryParams} from '@sanity/client'\nimport {type ApplySourceDocumentsUpdateFunction} from '@sanity/client/csm'\nimport {type FIXME} from 'sanity'\n\n/**\n * Used by `applySourceDocuments`\n * @internal\n */\nexport const mapChangedValue: ApplySourceDocumentsUpdateFunction = (\n changedValue: FIXME,\n {previousValue},\n) => {\n if (typeof previousValue === 'string') {\n if (typeof changedValue === 'number') {\n // If the string() function was used in the query, we need to convert the source value to a string as well\n return `${changedValue}`\n }\n // If it's an array in the source, but a string in the query response, it could be pt::text\n if (Array.isArray(changedValue)) {\n if (changedValue.length === 0) {\n // If it's empty assume it's PT and return an empty string\n return ''\n }\n // If the array contains any valid block type, assume the GROQ initially used pt::text on it and do the same conversion\n if (changedValue.some((node) => typeof node === 'object' && isPortableTextBlock(node))) {\n return toPlainText(changedValue)\n }\n }\n }\n\n return changedValue\n}\n\n/**\n * @internal\n */\nexport type QueryCacheKey = `${string}:${string}:${string}`\n/**\n * @internal\n */\nexport function getQueryCacheKey(\n perspective: ClientPerspective,\n query: string,\n params: QueryParams,\n): QueryCacheKey {\n return `${perspective}:${query}:${JSON.stringify(params)}`\n}\n","import {type ClientPerspective} from '@sanity/client'\nimport isEqual from 'fast-deep-equal'\nimport {useDeferredValue, useEffect, useReducer} from 'react'\nimport {type QueryParams} from 'sanity'\n\nimport {LOADER_QUERY_GC_INTERVAL} from '../constants'\nimport {getQueryCacheKey, type QueryCacheKey} from './utils'\n\ntype LiveQueriesState = Map<\n QueryCacheKey,\n {\n query: string\n params: QueryParams\n perspective: ClientPerspective\n }\n>\n\ntype State = {\n queries: LiveQueriesState\n heartbeats: Map<\n QueryCacheKey,\n {\n receivedAt: number\n /**\n * If false it means the query can't safely be garbage collected,\n * as older versions of \\@sanity/core-loader doesn't fire listen events\n * on an interval.\n */\n heartbeat: number | false\n }\n >\n}\n\ntype QueryListenAction = {\n type: 'query-listen'\n payload: {\n perspective: ClientPerspective\n query: string\n params: QueryParams\n heartbeat: number | false\n }\n}\ntype GarbageCollectAction = {type: 'gc'}\ntype Action = QueryListenAction | GarbageCollectAction\n\nfunction gc(state: State): State {\n if (state.queries.size < 1) {\n return state\n }\n\n const now = Date.now()\n const hasAnyExpired = Array.from(state.heartbeats.values()).some(\n (entry) => entry.heartbeat !== false && now > entry.receivedAt + entry.heartbeat,\n )\n if (!hasAnyExpired) {\n return state\n }\n const nextHeartbeats = new Map()\n const nextQueries = new Map()\n for (const [key, entry] of state.heartbeats.entries()) {\n if (entry.heartbeat !== false && now > entry.receivedAt + entry.heartbeat) {\n continue\n }\n nextHeartbeats.set(key, entry)\n nextQueries.set(key, state.queries.get(key))\n }\n\n return {...state, queries: nextQueries, heartbeats: nextHeartbeats}\n}\nfunction queryListen(state: State, {payload}: QueryListenAction): State {\n const key = getQueryCacheKey(payload.perspective, payload.query, payload.params)\n const data = {query: payload.query, params: payload.params, perspective: payload.perspective}\n\n const nextHeartbeats = new Map(state.heartbeats)\n nextHeartbeats.set(key, {\n receivedAt: Date.now(),\n heartbeat: payload.heartbeat,\n })\n\n let nextQueries = state.queries\n /**\n * The data comes from a postMessage event, which uses the structured clone algorithm to serialize state (https://developer.mozilla.org/en-US/docs/Web/API/Window/postMessage#message).\n * This impacts `params`, which is an object, as it will be a new object every time even if the sender is sending the same object instance on their end.\n * It also impacts `perspective`, as it's no longer just a string, but can also be an array of strings.\n * Both cases are handled by fast-deep-equal, which is used to compare the data before deciding wether the state should be updated.\n */\n if (!state.queries.has(key) || !isEqual(state.queries.get(key), data)) {\n nextQueries = new Map(state.queries)\n nextQueries.set(key, data)\n }\n\n return {heartbeats: nextHeartbeats, queries: nextQueries}\n}\n\nexport function reducer(state: State, action: Action): State {\n switch (action.type) {\n case 'query-listen':\n return queryListen(state, action)\n case 'gc':\n return gc(state)\n default:\n throw Error(\n `Unknown action: ${\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n (action as any).type\n }`,\n {cause: action},\n )\n }\n}\n\nexport const initialState: State = {\n queries: new Map(),\n heartbeats: new Map(),\n}\n\nexport function useLiveQueries(): [LiveQueriesState, React.ActionDispatch<[action: Action]>] {\n const [state, dispatch] = useReducer(reducer, initialState)\n\n useEffect(() => {\n const interval = setInterval(() => dispatch({type: 'gc'}), LOADER_QUERY_GC_INTERVAL)\n return () => clearInterval(interval)\n }, [])\n\n const queries = useDeferredValue(state.queries)\n return [queries, dispatch]\n}\n","import {\n type ClientPerspective,\n type ContentSourceMap,\n type LiveEventMessage,\n type QueryParams,\n type SyncTag,\n} from '@sanity/client'\nimport {applySourceDocuments, getPublishedId} from '@sanity/client/csm'\nimport {\n type ChannelInstance,\n type Controller,\n createConnectionMachine,\n type StatusEvent,\n} from '@sanity/comlink'\nimport {\n createCompatibilityActors,\n type LoaderControllerMsg,\n type LoaderNodeMsg,\n} from '@sanity/presentation-comlink'\nimport isEqual from 'fast-deep-equal'\nimport {memo, startTransition, useDeferredValue, useEffect, useMemo, useState} from 'react'\nimport {type SanityClient, type SanityDocument, useClient, useDataset, useProjectId} from 'sanity'\nimport {useEffectEvent} from 'use-effect-event'\n\nimport {API_VERSION, MIN_LOADER_QUERY_LISTEN_HEARTBEAT_INTERVAL} from '../constants'\nimport {type LoaderConnection, type PresentationPerspective} from '../types'\nimport {type DocumentOnPage} from '../useDocumentsOnPage'\nimport {useLiveEvents} from './useLiveEvents'\nimport {useLiveQueries} from './useLiveQueries'\nimport {mapChangedValue} from './utils'\n\nexport interface LiveQueriesProps {\n liveDocument: Partial<SanityDocument> | null | undefined\n controller: Controller | undefined\n perspective: ClientPerspective\n onLoadersConnection: (event: StatusEvent) => void\n onDocumentsOnPage: (\n key: string,\n perspective: PresentationPerspective,\n state: DocumentOnPage[],\n ) => void\n}\n\nexport default function LiveQueries(props: LiveQueriesProps): React.JSX.Element {\n const {controller, perspective: activePerspective, onLoadersConnection, onDocumentsOnPage} = props\n\n const [comlink, setComlink] = useState<ChannelInstance<LoaderControllerMsg, LoaderNodeMsg>>()\n const [liveQueries, liveQueriesDispatch] = useLiveQueries()\n\n const projectId = useProjectId()\n const dataset = useDataset()\n\n useEffect((): (() => void) => {\n if (controller) {\n const nextComlink = controller.createChannel<LoaderControllerMsg, LoaderNodeMsg>(\n {\n name: 'presentation',\n connectTo: 'loaders',\n heartbeat: true,\n },\n createConnectionMachine<LoaderControllerMsg, LoaderNodeMsg>().provide({\n actors: createCompatibilityActors<LoaderControllerMsg>(),\n }),\n )\n setComlink(nextComlink)\n\n nextComlink.onStatus(onLoadersConnection)\n\n nextComlink.on('loader/documents', (data) => {\n if (data.projectId === projectId && data.dataset === dataset) {\n onDocumentsOnPage(\n 'loaders',\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n data.perspective as unknown as any,\n data.documents,\n )\n }\n })\n\n nextComlink.on('loader/query-listen', (data) => {\n if (data.projectId === projectId && data.dataset === dataset) {\n if (\n typeof data.heartbeat === 'number' &&\n data.heartbeat < MIN_LOADER_QUERY_LISTEN_HEARTBEAT_INTERVAL\n ) {\n throw new Error(\n `Loader query listen heartbeat interval must be at least ${MIN_LOADER_QUERY_LISTEN_HEARTBEAT_INTERVAL}ms`,\n )\n }\n liveQueriesDispatch({\n type: 'query-listen',\n payload: {\n perspective: data.perspective,\n query: data.query,\n params: data.params,\n heartbeat: data.heartbeat ?? false,\n },\n })\n }\n })\n\n return nextComlink.start()\n }\n return () => undefined\n }, [controller, dataset, liveQueriesDispatch, onDocumentsOnPage, onLoadersConnection, projectId])\n\n const studioClient = useClient({apiVersion: API_VERSION})\n const client = useMemo(\n () =>\n studioClient.withConfig({\n resultSourceMap: 'withKeyArraySelector',\n }),\n [studioClient],\n )\n useEffect(() => {\n if (comlink) {\n comlink.post('loader/perspective', {\n projectId,\n dataset,\n perspective: activePerspective,\n })\n }\n }, [comlink, activePerspective, projectId, dataset])\n\n /**\n * Defer the liveDocument to avoid unnecessary rerenders on rapid edits\n */\n const liveDocument = useDeferredValue(props.liveDocument)\n\n const liveEvents = useLiveEvents(client)\n\n return (\n <>\n {[...liveQueries.entries()].map(([key, {query, params, perspective}]) => (\n <QuerySubscription\n key={`${liveEvents.resets}:${key}`}\n projectId={projectId}\n dataset={dataset}\n perspective={perspective}\n query={query}\n params={params}\n comlink={comlink}\n client={client}\n liveDocument={liveDocument}\n liveEventsMessages={liveEvents.messages}\n />\n ))}\n </>\n )\n}\n\ninterface SharedProps {\n /**\n * The Sanity client to use for fetching data and listening to mutations.\n */\n client: SanityClient\n}\n\ninterface QuerySubscriptionProps\n extends Pick<UseQuerySubscriptionProps, 'client' | 'liveDocument' | 'liveEventsMessages'> {\n projectId: string\n dataset: string\n perspective: ClientPerspective\n query: string\n params: QueryParams\n comlink: LoaderConnection | undefined\n}\nfunction QuerySubscriptionComponent(props: QuerySubscriptionProps) {\n const {\n projectId,\n dataset,\n perspective,\n query,\n client,\n liveDocument,\n params,\n comlink,\n liveEventsMessages,\n } = props\n\n const {\n result,\n resultSourceMap,\n syncTags: tags,\n } = useQuerySubscription({\n client,\n liveDocument,\n params,\n perspective,\n query,\n liveEventsMessages,\n }) || {}\n\n /* eslint-disable @typescript-eslint/no-shadow,max-params */\n const handleQueryChange = useEffectEvent(\n (\n comlink: LoaderConnection | undefined,\n perspective: ClientPerspective,\n query: string,\n params: QueryParams,\n result: unknown,\n resultSourceMap: ContentSourceMap | undefined,\n tags: `s1:${string}`[] | undefined,\n ) => {\n comlink?.post('loader/query-change', {\n projectId,\n dataset,\n perspective,\n query,\n params,\n result,\n resultSourceMap,\n tags,\n })\n },\n )\n /* eslint-enable @typescript-eslint/no-shadow,max-params */\n\n useEffect(() => {\n if (resultSourceMap) {\n handleQueryChange(comlink, perspective, query, params, result, resultSourceMap, tags)\n }\n return undefined\n }, [comlink, handleQueryChange, params, perspective, query, result, resultSourceMap, tags])\n\n return null\n}\nconst QuerySubscription = memo(QuerySubscriptionComponent)\nQuerySubscription.displayName = 'Memo(QuerySubscription)'\n\ninterface UseQuerySubscriptionProps extends Required<Pick<SharedProps, 'client'>> {\n liveDocument: Partial<SanityDocument> | null | undefined\n query: string\n params: QueryParams\n perspective: ClientPerspective\n liveEventsMessages: LiveEventMessage[]\n}\nfunction useQuerySubscription(props: UseQuerySubscriptionProps) {\n const {liveDocument, client, query, params, perspective, liveEventsMessages} = props\n const [result, setResult] = useState<unknown>(null)\n const [resultSourceMap, setResultSourceMap] = useState<ContentSourceMap | null | undefined>(null)\n const [syncTags, setSyncTags] = useState<SyncTag[] | undefined>(undefined)\n const [skipEventIds] = useState(() => new Set(liveEventsMessages.map((msg) => msg.id)))\n const recentLiveEvents = liveEventsMessages.filter((msg) => !skipEventIds.has(msg.id))\n const lastLiveEvent = recentLiveEvents.findLast((msg) =>\n msg.tags.some((tag) => syncTags?.includes(tag)),\n )\n const lastLiveEventId = lastLiveEvent?.id\n\n // Make sure any async errors bubble up to the nearest error boundary\n const [error, setError] = useState<unknown>(null)\n if (error) throw error\n\n /* eslint-disable max-nested-callbacks */\n useEffect(() => {\n const controller = new AbortController()\n\n client\n .fetch(query, params, {\n lastLiveEventId,\n tag: 'presentation-loader',\n signal: controller.signal,\n perspective,\n filterResponse: false,\n returnQuery: false,\n })\n .then((response) => {\n startTransition(() => {\n // eslint-disable-next-line max-nested-callbacks\n setResult((prev: unknown) => (isEqual(prev, response.result) ? prev : response.result))\n setResultSourceMap((prev) =>\n isEqual(prev, response.resultSourceMap) ? prev : response.resultSourceMap,\n )\n setSyncTags((prev) => (isEqual(prev, response.syncTags) ? prev : response.syncTags))\n })\n })\n .catch((err) => {\n if (typeof err !== 'object' || err?.name !== 'AbortError') {\n setError(err)\n }\n })\n\n return () => {\n controller.abort()\n }\n }, [client, lastLiveEventId, params, perspective, query])\n /* eslint-enable max-nested-callbacks */\n\n return useMemo(() => {\n if (liveDocument && resultSourceMap) {\n return {\n result: turboChargeResultIfSourceMap(liveDocument, result, perspective, resultSourceMap),\n resultSourceMap,\n syncTags,\n }\n }\n return {result, resultSourceMap, syncTags}\n }, [liveDocument, perspective, result, resultSourceMap, syncTags])\n}\n\nexport function turboChargeResultIfSourceMap<T = unknown>(\n liveDocument: Partial<SanityDocument> | null | undefined,\n result: T,\n perspective: ClientPerspective,\n resultSourceMap?: ContentSourceMap,\n): T {\n if (perspective === 'raw') {\n throw new Error('turboChargeResultIfSourceMap does not support raw perspective')\n }\n return applySourceDocuments(\n result,\n resultSourceMap,\n (sourceDocument) => {\n // If there's a displayed document, always prefer it\n if (\n // If _projectId is set, it's a cross dataset reference and we should skip it\n !sourceDocument._projectId &&\n liveDocument?._id &&\n getPublishedId(liveDocument._id) === getPublishedId(sourceDocument._id)\n ) {\n if (typeof liveDocument._id === 'string' && typeof sourceDocument._type === 'string') {\n return liveDocument as unknown as Required<Pick<SanityDocument, '_id' | '_type'>>\n }\n return {\n ...liveDocument,\n _id: liveDocument._id || sourceDocument._id,\n _type: liveDocument._type || sourceDocument._type,\n }\n }\n return null\n },\n mapChangedValue,\n perspective,\n )\n}\n"],"names":["reducer","state","event","type","messages","resets","Error","cause","initialState","useLiveEvents","client","$","_c","dispatch","useReducer","error","setError","useState","t0","t1","live","subscription","events","includeDrafts","tag","subscribe","next","err","unsubscribe","useEffect","useDeferredValue","mapChangedValue","changedValue","previousValue","Array","isArray","length","some","node","isPortableTextBlock","toPlainText","getQueryCacheKey","perspective","query","params","JSON","stringify","gc","queries","size","now","Date","from","heartbeats","values","entry","heartbeat","receivedAt","nextHeartbeats","Map","nextQueries","key","entries","set","get","queryListen","payload","data","has","isEqual","action","useLiveQueries","Symbol","for","interval","setInterval","LOADER_QUERY_GC_INTERVAL","clearInterval","t2","LiveQueries","props","controller","activePerspective","onLoadersConnection","onDocumentsOnPage","comlink","setComlink","liveQueries","liveQueriesDispatch","projectId","useProjectId","dataset","useDataset","nextComlink","createChannel","name","connectTo","createConnectionMachine","provide","actors","createCompatibilityActors","onStatus","on","documents","data_0","MIN_LOADER_QUERY_LISTEN_HEARTBEAT_INTERVAL","start","_temp","apiVersion","API_VERSION","studioClient","useClient","t3","t4","withConfig","resultSourceMap","t5","t6","post","liveDocument","liveEvents","t7","t8","t9","map","t10","t11","QuerySubscriptionComponent","liveEventsMessages","result","syncTags","tags","useQuerySubscription","comlink_0","perspective_0","query_0","params_0","result_0","resultSourceMap_0","tags_0","handleQueryChange","useEffectEvent","QuerySubscription","memo","displayName","setResult","setResultSourceMap","setSyncTags","undefined","Set","_temp2","skipEventIds","msg_0","msg","id","recentLiveEvents","filter","msg_1","includes","findLast","lastLiveEventId","AbortController","fetch","signal","filterResponse","returnQuery","then","response","startTransition","prev","prev_0","prev_1","catch","abort","turboChargeResultIfSourceMap","bb0","applySourceDocuments","sourceDocument","_projectId","_id","getPublishedId","_type"],"mappings":"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"}