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.
- package/lib/_chunks-cjs/LiveQueries.js +236 -163
- package/lib/_chunks-cjs/LiveQueries.js.map +1 -1
- package/lib/_chunks-cjs/PresentationToolGrantsCheck.js +3 -7
- package/lib/_chunks-cjs/PresentationToolGrantsCheck.js.map +1 -1
- package/lib/_chunks-cjs/buildAction.js +3 -0
- package/lib/_chunks-cjs/buildAction.js.map +1 -1
- package/lib/_chunks-cjs/presentation.js +3 -4
- package/lib/_chunks-cjs/presentation.js.map +1 -1
- package/lib/_chunks-cjs/version.js +1 -1
- package/lib/_chunks-es/LiveQueries.mjs +235 -162
- package/lib/_chunks-es/LiveQueries.mjs.map +1 -1
- package/lib/_chunks-es/PresentationToolGrantsCheck.mjs +2 -4
- package/lib/_chunks-es/PresentationToolGrantsCheck.mjs.map +1 -1
- package/lib/_chunks-es/presentation.mjs +3 -3
- package/lib/_chunks-es/presentation.mjs.map +1 -1
- package/lib/_chunks-es/version.mjs +1 -1
- package/lib/_internal/cli/threads/validateDocuments.js +1 -1
- package/lib/_internal/cli/threads/validateDocuments.js.map +1 -1
- package/lib/_legacy/LiveQueries.esm.js +235 -162
- package/lib/_legacy/LiveQueries.esm.js.map +1 -1
- package/lib/_legacy/PresentationToolGrantsCheck.esm.js +2 -4
- package/lib/_legacy/PresentationToolGrantsCheck.esm.js.map +1 -1
- package/lib/_legacy/presentation.esm.js +3 -3
- package/lib/_legacy/presentation.esm.js.map +1 -1
- package/lib/_legacy/version.esm.js +1 -1
- package/package.json +13 -14
- package/src/_internal/cli/server/buildVendorDependencies.ts +1 -0
- package/src/_internal/cli/threads/validateDocuments.ts +2 -2
- package/src/presentation/PresentationTool.tsx +10 -21
- package/src/presentation/constants.ts +4 -16
- package/src/presentation/loader/LiveQueries.tsx +79 -234
- package/src/presentation/loader/useLiveEvents.ts +75 -0
- package/src/presentation/loader/useLiveQueries.ts +127 -0
- package/src/presentation/loader/utils.ts +2 -125
- package/src/presentation/types.ts +1 -18
- package/lib/_chunks-cjs/LoaderQueries.js +0 -281
- package/lib/_chunks-cjs/LoaderQueries.js.map +0 -1
- package/lib/_chunks-cjs/utils.js +0 -70
- package/lib/_chunks-cjs/utils.js.map +0 -1
- package/lib/_chunks-es/LoaderQueries.mjs +0 -288
- package/lib/_chunks-es/LoaderQueries.mjs.map +0 -1
- package/lib/_chunks-es/utils.mjs +0 -74
- package/lib/_chunks-es/utils.mjs.map +0 -1
- package/lib/_legacy/LoaderQueries.esm.js +0 -288
- package/lib/_legacy/LoaderQueries.esm.js.map +0 -1
- package/lib/_legacy/utils.esm.js +0 -74
- package/lib/_legacy/utils.esm.js.map +0 -1
- 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,
|
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 {
|
13
|
-
|
14
|
-
|
15
|
-
|
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,
|
21
|
-
|
22
|
-
|
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
|
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(
|
47
|
-
data.projectId === projectId && data.dataset === dataset && onDocumentsOnPage(
|
48
|
-
|
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
|
-
|
58
|
-
|
59
|
-
|
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
|
-
}),
|
172
|
+
}), nextComlink.start();
|
69
173
|
}
|
70
|
-
|
71
|
-
|
72
|
-
|
73
|
-
|
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
|
-
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
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
|
-
|
110
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
|
115
|
-
|
116
|
-
|
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(
|
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
|
-
|
128
|
-
|
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
|
-
|
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] !==
|
157
|
-
|
158
|
-
|
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(
|
256
|
+
const $ = c(30), {
|
167
257
|
liveDocument,
|
168
258
|
client,
|
169
259
|
query,
|
170
260
|
params,
|
171
261
|
perspective,
|
172
|
-
|
173
|
-
} = props, [
|
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
|
177
|
-
$[
|
178
|
-
|
179
|
-
|
180
|
-
|
181
|
-
|
182
|
-
|
183
|
-
|
184
|
-
|
185
|
-
|
186
|
-
|
187
|
-
|
188
|
-
|
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
|
-
|
207
|
-
|
208
|
-
|
209
|
-
|
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
|
-
},
|
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 &&
|
301
|
+
if (liveDocument && resultSourceMap) {
|
229
302
|
let t52;
|
230
|
-
$[
|
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
|
-
$[
|
305
|
+
$[22] !== resultSourceMap || $[23] !== syncTags || $[24] !== t52 ? (t6 = {
|
233
306
|
result: t52,
|
234
|
-
resultSourceMap
|
235
|
-
syncTags
|
236
|
-
}, $[
|
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
|
-
$[
|
241
|
-
result
|
242
|
-
resultSourceMap
|
243
|
-
syncTags
|
244
|
-
}, $[
|
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
|
-
|
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":";;;;;;;;;;;;AAiBgBA,SAAAA,UAAQC,OAAcC,OAAyB;AAC7D,UAAQA,MAAMC,MAAI;AAAA,IAChB,KAAK;AACI,aAAA;AAAA,QACL,GAAGF;AAAAA,QACHG,UAAU,CAAC,GAAGH,MAAMG,UAAUF,KAAK;AAAA,MACrC;AAAA,IACF,KAAK;AAAA,IACL,KAAK;AACI,aAAA;AAAA,QACL,GAAGD;AAAAA,QACHG,UAAU,CAAE;AAAA,QACZC,QAAQJ,MAAMI,SAAS;AAAA,MACzB;AAAA,IACF,KAAK;AAEIJ,aAAAA;AAAAA,IACT;AACE,YAAMK,MACJ;AAAA,MAEGJ,MAAcC,IAAI,IAErB;AAAA,QAACI,OAAOL;AAAAA,MAAAA,CACV;AAAA,EAAA;AAEN;AAEO,MAAMM,iBAAsB;AAAA,EACjCJ,UAAU,CAAE;AAAA,EACZC,QAAQ;AACV;AAEO,SAAAI,cAAAC,QAAA;AAAA,QAAAC,IAAAC,EAAA,CAAA,GACLX,CAAAA,OAAAY,QAAA,IAA0BC,WAAAd,WAAAQ,cAAgC,GAC1D,CAAAO,OAAAC,QAAA,IAA0BC,aAAsB;AAAC,MAC7CF,UAAc;AAEVA,UAAAA;AAAK,MAAAG,IAAAC;AAAA,SAAAR,EAAA,CAAA,MAAAD,OAAAU,QAGHF,KAAAA,MAAA;AACRG,UAAAA,eAAqBX,OAAMU,KAAAE,OAAA;AAAA,MAAAC,eAAA;AAAA,MAAAC,KACU;AAAA,IAAsB,CAAA,EAACC,UAAA;AAAA,MAAAC,MAElDb;AAAAA,MAAQE,OAAAY,SAEZX,SACEW,eAAGrB,QACCqB,MAAGrB,IAAAA,MACO,qCAAmC;AAAA,QAAAC,OAAUoB;AAAAA,MAAAA,CAC7D,CAAA;AAAA,IAAA,CACH;AAAC,WAAA,MACSN,aAAYO,YAAa;AAAA,EACrCT,GAAAA,KAAA,CAACT,OAAMU,IAAA,GAAMT,EAAA,CAAA,IAAAD,OAAAU,MAAAT,OAAAO,IAAAP,OAAAQ,OAAAD,KAAAP,EAAA,CAAA,GAAAQ,KAAAR,EAAA,CAAA,IAbhBkB,UAAUX,IAaPC,EAAa,GAETW,iBAAiB7B,KAAK;AAAC;AC/DnB8B,MAAAA,kBAAsDA,CACjEC,cACA;AAAA,EAACC;AAAa,MACX;AACC,MAAA,OAAOA,iBAAkB,UAAU;AACrC,QAAI,OAAOD,gBAAiB;AAE1B,aAAO,GAAGA,YAAY;AAGpBE,QAAAA,MAAMC,QAAQH,YAAY,GAAG;AAC/B,UAAIA,aAAaI,WAAW;AAEnB,eAAA;AAGLJ,UAAAA,aAAaK,KAAMC,CAAS,SAAA,OAAOA,QAAS,YAAYC,oBAAoBD,IAAI,CAAC;AACnF,eAAOE,YAAYR,YAAY;AAAA,IAAA;AAAA,EAEnC;AAGKA,SAAAA;AACT;AASgBS,SAAAA,iBACdC,aACAC,OACAC,QACe;AACR,SAAA,GAAGF,WAAW,IAAIC,KAAK,IAAIE,KAAKC,UAAUF,MAAM,CAAC;AAC1D;ACHA,SAASG,GAAG9C,OAAqB;AAC3BA,MAAAA,MAAM+C,QAAQC,OAAO;AAChBhD,WAAAA;AAGHiD,QAAAA,MAAMC,KAAKD,IAAI;AAIrB,MAAI,CAHkBhB,MAAMkB,KAAKnD,MAAMoD,WAAWC,QAAQ,EAAEjB,KACzDkB,CAAAA,UAAUA,MAAMC,cAAc,MAASN,MAAMK,MAAME,aAAaF,MAAMC,SACzE;AAESvD,WAAAA;AAET,QAAMyD,iBAAqBC,oBAAAA,IACrBC,GAAAA,kCAAkBD,IAAI;AAC5B,aAAW,CAACE,KAAKN,KAAK,KAAKtD,MAAMoD,WAAWS,QAAQ;AAC9CP,UAAMC,cAAc,MAASN,MAAMK,MAAME,aAAaF,MAAMC,cAGhEE,eAAeK,IAAIF,KAAKN,KAAK,GAC7BK,YAAYG,IAAIF,KAAK5D,MAAM+C,QAAQgB,IAAIH,GAAG,CAAC;AAGtC,SAAA;AAAA,IAAC,GAAG5D;AAAAA,IAAO+C,SAASY;AAAAA,IAAaP,YAAYK;AAAAA,EAAc;AACpE;AACA,SAASO,YAAYhE,OAAc;AAAA,EAACiE;AAA0B,GAAU;AAChEL,QAAAA,MAAMpB,iBAAiByB,QAAQxB,aAAawB,QAAQvB,OAAOuB,QAAQtB,MAAM,GACzEuB,OAAO;AAAA,IAACxB,OAAOuB,QAAQvB;AAAAA,IAAOC,QAAQsB,QAAQtB;AAAAA,IAAQF,aAAawB,QAAQxB;AAAAA,EAE3EgB,GAAAA,iBAAiB,IAAIC,IAAI1D,MAAMoD,UAAU;AAC/CK,iBAAeK,IAAIF,KAAK;AAAA,IACtBJ,YAAYN,KAAKD,IAAI;AAAA,IACrBM,WAAWU,QAAQV;AAAAA,EAAAA,CACpB;AAED,MAAII,cAAc3D,MAAM+C;AAOpB,UAAA,CAAC/C,MAAM+C,QAAQoB,IAAIP,GAAG,KAAK,CAACQ,QAAQpE,MAAM+C,QAAQgB,IAAIH,GAAG,GAAGM,IAAI,OAClEP,cAAc,IAAID,IAAI1D,MAAM+C,OAAO,GACnCY,YAAYG,IAAIF,KAAKM,IAAI,IAGpB;AAAA,IAACd,YAAYK;AAAAA,IAAgBV,SAASY;AAAAA,EAAW;AAC1D;AAEgB5D,SAAAA,QAAQC,OAAcqE,QAAuB;AAC3D,UAAQA,OAAOnE,MAAI;AAAA,IACjB,KAAK;AACI8D,aAAAA,YAAYhE,OAAOqE,MAAM;AAAA,IAClC,KAAK;AACH,aAAOvB,GAAG9C,KAAK;AAAA,IACjB;AACE,YAAMK,MACJ;AAAA,MAEGgE,OAAenE,IAAI,IAEtB;AAAA,QAACI,OAAO+D;AAAAA,MAAAA,CACV;AAAA,EAAA;AAEN;AAEO,MAAM9D,eAAsB;AAAA,EACjCwC,6BAAaW,IAAI;AAAA,EACjBN,gCAAgBM,IAAI;AACtB;AAEO,SAAAY,iBAAA;AAAA5D,QAAAA,IAAAC,EAAA,CAAA,GACLX,CAAAA,OAAAY,QAAA,IAA0BC,WAAAd,SAAAQ,YAAgC;AAAC,MAAAU,IAAAC;AAAAR,IAAA,CAAA,MAAA6D,OAAAC,IAAA,2BAAA,KAEjDvD,KAAAA,MAAA;AACRwD,UAAAA,WAAiBC,YAAA,MAAkB9D,SAAQ;AAAA,MAAAV,MAAQ;AAAA,IAAK,CAAA,GAACyE,wBAA0B;AACtEC,WAAAA,MAAAA,cAAcH,QAAQ;AAAA,EAAA,GAClCvD,KAAA,CAAA,GAAER,OAAAO,IAAAP,OAAAQ,OAAAD,KAAAP,EAAA,CAAA,GAAAQ,KAAAR,EAAA,CAAA,IAHLkB,UAAUX,IAGPC,EAAE;AAEL6B,QAAAA,UAAgBlB,iBAAiB7B,MAAK+C,OAAQ;AAAC8B,MAAAA;AAAAnE,SAAAA,SAAAqC,WACxC8B,KAAA,CAAC9B,SAASnC,QAAQ,GAACF,OAAAqC,SAAArC,OAAAmE,MAAAA,KAAAnE,EAAA,CAAA,GAAnBmE;AAAmB;AClF5B,SAAeC,YAAAC,OAAA;AAAArE,QAAAA,IAAAC,EAAA,EAAA,GACb;AAAA,IAAAqE;AAAAA,IAAAvC,aAAAwC;AAAAA,IAAAC;AAAAA,IAAAC;AAAAA,EAAAA,IAA6FJ,OAE7F,CAAAK,SAAAC,UAAA,IAA8BrE,SAC9B,GAAA,CAAAsE,aAAAC,mBAAA,IAA2CjB,eAAe,GAE1DkB,YAAkBC,aAAa,GAC/BC,UAAgBC,WAAW;AAAC,MAAA1E,IAAAC;AAAAR,WAAAsE,cAAAtE,EAAAgF,CAAAA,MAAAA,WAAAhF,EAAA6E,CAAAA,MAAAA,uBAAA7E,EAAA,CAAA,MAAAyE,qBAAAzE,SAAAwE,uBAAAxE,EAAA,CAAA,MAAA8E,aAElBvE,KAAAA,MAAA;AAAA,QACJ+D,YAAU;AACZY,YAAAA,cAAoBZ,WAAUa,cAAA;AAAA,QAAAC,MAEpB;AAAA,QAAcC,WACT;AAAA,QAASxC,WAAA;AAAA,MAAA,GAGtByC,wBAA4D,EAACC,QAAA;AAAA,QAAAC,QACnDC,0BAA+C;AAAA,MAAA,CACxD,CACH;AACWP,aAAAA,WAAAA,WAAW,GAEtBA,YAAWQ,SAAUlB,mBAAmB,GAExCU,YAAWS,GAAI,oBAAkBnC,CAAA,SAAA;AAC3BA,aAAIsB,cAAeA,aAAatB,KAAIwB,YAAaA,WACnDP,kBACE,WAEAjB,KAAIzB,aACJyB,KAAIoC,SACN;AAAA,MAEH,CAAA,GAEDV,YAAWS,GAAI,uBAAqBE,CAAA,WAAA;AAAA,YAC9BrC,OAAIsB,cAAeA,aAAatB,OAAIwB,YAAaA,SAAO;AAAA,cAExD,OAAOxB,OAAIX,aAAe,YAC1BW,OAAIX,YAAAiD;AAAuD,kBAAA,IAAAnG,MAGzD,2DAAAmG,0CAAA,IAAyG;AAG1F,8BAAA;AAAA,YAAAtG,MACX;AAAA,YAAc+D,SAAA;AAAA,cAAAxB,aAELyB,OAAIzB;AAAAA,cAAAC,OACVwB,OAAIxB;AAAAA,cAAAC,QACHuB,OAAIvB;AAAAA,cAAAY,WACDW,OAAIX,aAAA;AAAA,YAAA;AAAA,UAAmB,CAErC;AAAA,QAAA;AAAA,MAAC,CAEL,GAEMqC,YAAWa,MAAO;AAAA,IAAA;AAACC,WAAAA;AAAAA,EAAA,GAG3BxF,KAAA,CAAC8D,YAAYU,SAASH,qBAAqBJ,mBAAmBD,qBAAqBM,SAAS,GAAC9E,OAAAsE,YAAAtE,OAAAgF,SAAAhF,OAAA6E,qBAAA7E,OAAAyE,mBAAAzE,OAAAwE,qBAAAxE,OAAA8E,WAAA9E,OAAAO,IAAAP,OAAAQ,OAAAD,KAAAP,EAAA,CAAA,GAAAQ,KAAAR,EAAA,CAAA,IApDhGkB,UAAUX,IAoDPC,EAA6F;AAAC2D,MAAAA;AAAAnE,IAAA,CAAA,MAAA6D,OAAAC,IAAA,2BAAA,KAElEK,KAAA;AAAA,IAAA8B,YAAAC;AAAAA,EAAAA,GAAyBlG,OAAAmE,MAAAA,KAAAnE,EAAA,CAAA;AAAxDmG,QAAAA,eAAqBC,UAAUjC,EAAyB;AAAC,MAAAkC,IAAAC;AAAAtG,WAAAmG,gBAGrDG,KAAAH,aAAYI,WAAA;AAAA,IAAAC,iBACO;AAAA,EAClB,CAAA,GAACxG,OAAAmG,cAAAnG,QAAAsG,MAAAA,KAAAtG,EAAA,EAAA,GAAAqG,KAFFC;AAFJ,QAAAvG,SAAesG;AAMd,MAAAI,IAAAC;AAAA1G,IAAA,EAAA,MAAAuE,qBAAAvE,EAAA0E,EAAAA,MAAAA,WAAA1E,EAAAgF,EAAAA,MAAAA,WAAAhF,UAAA8E,aACS2B,KAAAA,MAAA;AACJ/B,eACFA,QAAOiC,KAAM,sBAAoB;AAAA,MAAA7B;AAAAA,MAAAE;AAAAA,MAAAjD,aAGlBwC;AAAAA,IAAAA,CACd;AAAA,EAAA,GAEFmC,MAAChC,SAASH,mBAAmBO,WAAWE,OAAO,GAAChF,QAAAuE,mBAAAvE,QAAA0E,SAAA1E,QAAAgF,SAAAhF,QAAA8E,WAAA9E,QAAAyG,IAAAzG,QAAA0G,OAAAD,KAAAzG,EAAA,EAAA,GAAA0G,KAAA1G,EAAA,EAAA,IARnDkB,UAAUuF,IAQPC,EAAgD;AAKnD,QAAAE,eAAqBzF,iBAAiBkD,MAAKuC,YAAa,GAExDC,aAAmB/G,cAAcC,MAAM;AAAC+G,MAAAA;AAAA9G,YAAA4E,eAI/BkC,KAAAlC,YAAWzB,WAAUnD,QAAA4E,aAAA5E,QAAA8G,MAAAA,KAAA9G,EAAA,EAAA;AAAA+G,MAAAA;AAAA/G,YAAA8G,MAAzBC,SAAID,EAAqB,GAAC9G,QAAA8G,IAAA9G,QAAA+G,MAAAA,KAAA/G,EAAA,EAAA;AAAAgH,MAAAA;AAAA,SAAAhH,EAAA,EAAA,MAAAD,UAAAC,EAAA0E,EAAAA,MAAAA,WAAA1E,UAAAgF,WAAAhF,EAAA,EAAA,MAAA4G,gBAAA5G,EAAA6G,EAAAA,MAAAA,WAAApH,YAAAO,UAAA6G,WAAAnH,UAAAM,EAAA8E,EAAAA,MAAAA,aAAA9E,UAAA+G,MAD7BC,KACGD,oBAAAA,UAAAA,EAAAA,UAAAA,GAA0BE,IAAAC,CAAA,QAAA;AAAM,UAAA,CAAAhE,KAAAiE,GAAA,IAAAD,KAAM;AAAA,MAAAlF;AAAAA,MAAAC;AAAAA,MAAAF;AAAAA,IAAAA,IAAAoF;AAA4B,+BAChE,mBAEYrC,EAAAA,WACFE,SACIjD,aACNC,OACCC,QACCyC,SACD3E,QACM6G,cACM,oBAAAC,WAAUpH,SATzB,GAAA,GAAGoH,WAAUnH,MAAA,IAAWwD,GAAG,EAUhC;AAAA,EAAA,CACH,EAAC,CAAA,GACDlD,QAAAD,QAAAC,QAAA0E,SAAA1E,QAAAgF,SAAAhF,QAAA4G,cAAA5G,EAAA,EAAA,IAAA6G,WAAApH,UAAAO,EAAA,EAAA,IAAA6G,WAAAnH,QAAAM,QAAA8E,WAAA9E,QAAA+G,IAAA/G,QAAAgH,MAAAA,KAAAhH,EAAA,EAAA,GAfHgH;AAeG;AAxGQ,SAAAhB,QAAA;AAAA;AA4Hf,SAAAoB,2BAAA/C,OAAA;AAAArE,QAAAA,IAAAC,EAAA,EAAA,GACE;AAAA,IAAA6E;AAAAA,IAAAE;AAAAA,IAAAjD;AAAAA,IAAAC;AAAAA,IAAAjC;AAAAA,IAAA6G;AAAAA,IAAA3E;AAAAA,IAAAyC;AAAAA,IAAA2C;AAAAA,MAUIhD,OAEJ;AAAA,IAAAiD;AAAAA,IAAAd;AAAAA,IAAAe,UAAAC;AAAAA,MAIIC,qBAAA;AAAA,IAAA1H;AAAAA,IAAA6G;AAAAA,IAAA3E;AAAAA,IAAAF;AAAAA,IAAAC;AAAAA,IAAAqF;AAAAA,EAOH,CAAA,KAAO,CAAA;AAAA9G,MAAAA;AAAAP,IAAAgF,CAAAA,MAAAA,WAAAhF,SAAA8E,aAINvE,KAAAA,CAAAmH,WAAAC,eAAAC,SAAAC,UAAAC,UAAAC,mBAAAC,WAAA;AASEtD,eAAOiC,KAAO,uBAAqB;AAAA,MAAA7B;AAAAA,MAAAE;AAAAA,MAAAjD,aAGjCA;AAAAA,MAAWC,OACXA;AAAAA,MAAKC,QACLA;AAAAA,MAAMqF,QACNA;AAAAA,MAAMd,iBACNA;AAAAA,MAAegB,MACfA;AAAAA,IAAAA,CAAI;AAAA,EAEPxH,GAAAA,OAAAgF,SAAAhF,OAAA8E,WAAA9E,OAAAO,MAAAA,KAAAP,EAAA,CAAA;AApBHiI,QAAAA,oBAA0BC,eACxB3H,EAoBF;AAAC,MAAAC,IAAA2D;AAAA,SAAAnE,EAAA,CAAA,MAAA0E,WAAA1E,EAAA,CAAA,MAAAiI,qBAAAjI,EAAA,CAAA,MAAAiC,UAAAjC,EAAA+B,CAAAA,MAAAA,eAAA/B,EAAAgC,CAAAA,MAAAA,SAAAhC,EAAAsH,CAAAA,MAAAA,UAAAtH,EAAAwG,CAAAA,MAAAA,mBAAAxG,UAAAwH,QAGShH,KAAAA,MAAA;AACJgG,uBACFyB,kBAAkBvD,SAAS3C,aAAaC,OAAOC,QAAQqF,QAAQd,iBAAiBgB,IAAI;AAAA,EAGrFrD,GAAAA,KAACO,CAAAA,SAASuD,mBAAmBhG,QAAQF,aAAaC,OAAOsF,QAAQd,iBAAiBgB,IAAI,GAACxH,OAAA0E,SAAA1E,OAAAiI,mBAAAjI,OAAAiC,QAAAjC,OAAA+B,aAAA/B,OAAAgC,OAAAhC,OAAAsH,QAAAtH,OAAAwG,iBAAAxG,QAAAwH,MAAAxH,QAAAQ,IAAAR,QAAAmE,OAAA3D,KAAAR,EAAA,EAAA,GAAAmE,KAAAnE,EAAA,EAAA,IAL1FkB,UAAUV,IAKP2D,EAAuF,GAAC;AAAA;AAI7F,MAAMgE,oBAAoBC,KAAKhB,0BAA0B;AACzDe,kBAAkBE,cAAc;AAShC,SAAAZ,qBAAApD,OAAA;AAAArE,QAAAA,IAAAC,EAAA,EAAA,GACE;AAAA,IAAA2G;AAAAA,IAAA7G;AAAAA,IAAAiC;AAAAA,IAAAC;AAAAA,IAAAF;AAAAA,IAAAsF;AAAAA,EAAAA,IAA+EhD,OAC/E,CAAAiD,QAAAgB,SAAA,IAA4BhI,aAAsB,GAClD,CAAAkG,iBAAA+B,kBAAA,IAA8CjI,aAAkD,GAChG,CAAAiH,UAAAiB,WAAA,IAAgClI,SAAAmI,MAAyC;AAAClI,MAAAA;AAAAP,WAAAqH,sBAC1C9G,KAAAA,MAAA,IAAAmI,IAAcrB,mBAAkBJ,IAAA0B,MAAoB,CAAC,GAAC3I,OAAAqH,oBAAArH,OAAAO,MAAAA,KAAAP,EAAA,CAAA;AAAtF,QAAA,CAAA4I,YAAA,IAAuBtI,SAASC,EAAsD;AAACC,MAAAA;AAAAR,MAAAA,EAAAqH,CAAAA,MAAAA,sBAAArH,SAAA4I,gBAAA5I,EAAA,CAAA,MAAAuH,UAAA;AAAApD,QAAAA;AAAAnE,aAAA4I,gBACpCzE,MAAA0E,CAAA,UAAA,CAAUD,aAAYnF,IAAKqF,MAAGC,EAAG,GAAC/I,OAAA4I,cAAA5I,OAAAmE,OAAAA,MAAAnE,EAAA,CAAA;AAArFgJ,UAAAA,mBAAyB3B,mBAAkB4B,OAAQ9E,GAAkC;AAACkC,QAAAA;AAAArG,aAAAuH,YACtClB,MAAA6C,CAC9CJ,UAAAA,MAAGtB,KAAA9F,KAAAb,CAAAA,QAAoB0G,UAAQ4B,SAAWtI,GAAG,CAAC,GAACb,OAAAuH,UAAAvH,OAAAqG,OAAAA,MAAArG,EAAA,CAAA,GAD3BQ,KAAAwI,iBAAgBI,SAAU/C,GAEhD,GAACrG,OAAAqH,oBAAArH,OAAA4I,cAAA5I,OAAAuH,UAAAvH,OAAAQ;AAAAA,EAAA;AAAAA,SAAAR,EAAA,CAAA;AACDqJ,QAAAA,kBAHsB7I,IAGeuI,IAGrC,CAAA3I,OAAAC,QAAA,IAA0BC,aAAsB;AAC5CF,MAAAA;AAAaA,UAAAA;AAAK,MAAA+D,IAAAkC;AAAArG,IAAAD,EAAAA,MAAAA,UAAAC,EAAA,EAAA,MAAAqJ,mBAAArJ,EAAAiC,EAAAA,MAAAA,UAAAjC,EAAA,EAAA,MAAA+B,eAAA/B,UAAAgC,SAGZmC,KAAAA,MAAA;AACRG,UAAAA,iBAAAgF,gBAAA;AAEMC,WAAAA,OAAAA,MACGvH,OAAOC,QAAM;AAAA,MAAAoH;AAAAA,MAAAxI,KAEb;AAAA,MAAqB2I,QAClBlF,WAAUkF;AAAAA,MAAAzH;AAAAA,MAAA0H,gBAAA;AAAA,MAAAC,aAAA;AAAA,IAAA,CAInB,EAACC,KAAAC,CAAA,aAAA;AAEAC,sBAAA,MAAA;AAEEvB,kBAASwB,CAAqBpG,SAAAA,QAAQoG,MAAMF,SAAQtC,MAAO,IAAIwC,OAAOF,SAAQtC,MAAQ,GACtFiB,mBAAkBwB,CAAAA,WAChBrG,QAAQoG,QAAMF,SAAQpD,eAAgB,IAAIsD,SAAOF,SAAQpD,eAC3D,GACAgC,YAAWwB,CAAAA,WAAYtG,QAAQoG,QAAMF,SAAQrC,QAAS,IAAIuC,SAAOF,SAAQrC,QAAU;AAAA,MAAA,CACpF;AAAA,IAAA,CACF,EAAC0C,MAAAjJ,CAAA,QAAA;AAAA,OAEI,OAAOA,OAAQ,YAAYA,KAAGoE,SAAW,iBAC3C/E,SAASW,GAAG;AAAA,IAEf,CAAA,GAAC,MAAA;AAGFsD,iBAAU4F,MAAO;AAAA,IAAC;AAAA,EAAA,GAEnB7D,KAAA,CAACtG,QAAQsJ,iBAAiBpH,QAAQF,aAAaC,KAAK,GAAChC,QAAAD,QAAAC,QAAAqJ,iBAAArJ,QAAAiC,QAAAjC,QAAA+B,aAAA/B,QAAAgC,OAAAhC,QAAAmE,IAAAnE,QAAAqG,OAAAlC,KAAAnE,EAAA,EAAA,GAAAqG,KAAArG,EAAA,EAAA,IA/BxDkB,UAAUiD,IA+BPkC,EAAqD;AAACC,MAAAA;AAAA,OAAA;AAAA,QAInDM,gBAAgBJ,iBAAe;AAAAC,UAAAA;AAAAzG,QAAA,EAAA,MAAA4G,gBAAA5G,EAAA+B,EAAAA,MAAAA,eAAA/B,EAAAsH,EAAAA,MAAAA,UAAAtH,UAAAwG,mBAEvBC,MAAA0D,6BAA6BvD,cAAcU,QAAQvF,aAAayE,eAAe,GAACxG,QAAA4G,cAAA5G,QAAA+B,aAAA/B,QAAAsH,QAAAtH,QAAAwG,iBAAAxG,QAAAyG,OAAAA,MAAAzG,EAAA,EAAA;AAAA0G,UAAAA;AAAA1G,QAAAwG,EAAAA,MAAAA,mBAAAxG,UAAAuH,YAAAvH,EAAA,EAAA,MAAAyG,OADnFC,KAAA;AAAA,QAAAY,QACGb;AAAAA,QAAgFD;AAAAA,QAAAe;AAAAA,MAAAA,GAGzFvH,QAAAwG,iBAAAxG,QAAAuH,UAAAvH,QAAAyG,KAAAzG,QAAA0G,MAAAA,KAAA1G,EAAA,EAAA,GAJDsG,KAAOI;AAIN0D,YAAAA;AAAAA,IAAAA;AAAA3D,QAAAA;AAAAzG,MAAAsH,EAAAA,MAAAA,UAAAtH,UAAAwG,mBAAAxG,EAAA,EAAA,MAAAuH,YAEId,KAAA;AAAA,MAAAa;AAAAA,MAAAd;AAAAA,MAAAe;AAAAA,IAAAA,GAAmCvH,QAAAsH,QAAAtH,QAAAwG,iBAAAxG,QAAAuH,UAAAvH,QAAAyG,MAAAA,KAAAzG,EAAA,EAAA,GAA1CsG,KAAOG;AAAAA,EAAAA;AARFH,SAAAA;AAS2D;AA5DpE,SAAAqC,OAAAG,KAAA;AAAA,SAKgFA,IAAGC;AAAA;AA0D5E,SAASoB,6BACdvD,cACAU,QACAvF,aACAyE,iBACG;AACH,MAAIzE,gBAAgB;AACZ,UAAA,IAAIpC,MAAM,+DAA+D;AAE1E0K,SAAAA,qBACL/C,QACAd,iBACC8D,CAAAA;AAAAA;AAAAA,IAIG,CAACA,eAAeC,cAChB3D,cAAc4D,OACdC,eAAe7D,aAAa4D,GAAG,MAAMC,eAAeH,eAAeE,GAAG,IAElE,OAAO5D,aAAa4D,OAAQ,YAAY,OAAOF,eAAeI,SAAU,WACnE9D,eAEF;AAAA,MACL,GAAGA;AAAAA,MACH4D,KAAK5D,aAAa4D,OAAOF,eAAeE;AAAAA,MACxCE,OAAO9D,aAAa8D,SAASJ,eAAeI;AAAAA,IAAAA,IAGzC;AAAA,KAETtJ,iBACAW,WACF;AACF;"}
|