@plumile/router 0.1.13 → 0.1.17
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/README.md +43 -422
- package/lib/esm/builder.d.ts +5 -5
- package/lib/esm/builder.d.ts.map +1 -1
- package/lib/esm/builder.js +1 -1
- package/lib/esm/index.d.ts +0 -4
- package/lib/esm/index.d.ts.map +1 -1
- package/lib/esm/index.js +1 -5
- package/lib/esm/routing/Link.d.ts +3 -3
- package/lib/esm/routing/Link.d.ts.map +1 -1
- package/lib/esm/routing/Link.js +3 -8
- package/lib/esm/routing/RouteComponentWrapper.d.ts +6 -5
- package/lib/esm/routing/RouteComponentWrapper.d.ts.map +1 -1
- package/lib/esm/routing/RouteComponentWrapper.js +1 -1
- package/lib/esm/routing/RouterRenderer.d.ts.map +1 -1
- package/lib/esm/routing/RouterRenderer.js +1 -1
- package/lib/esm/routing/createRouter.d.ts +0 -1
- package/lib/esm/routing/createRouter.d.ts.map +1 -1
- package/lib/esm/routing/createRouter.js +434 -399
- package/lib/esm/routing/index.d.ts +2 -6
- package/lib/esm/routing/index.d.ts.map +1 -1
- package/lib/esm/routing/index.js +3 -7
- package/lib/esm/routing/useAllQuery.d.ts +7 -0
- package/lib/esm/routing/useAllQuery.d.ts.map +1 -0
- package/lib/esm/routing/useAllQuery.js +31 -0
- package/lib/esm/routing/useFilterDiagnostics.d.ts +2 -0
- package/lib/esm/routing/useFilterDiagnostics.d.ts.map +1 -0
- package/lib/esm/routing/useFilterDiagnostics.js +11 -0
- package/lib/esm/routing/useFilters.d.ts +16 -0
- package/lib/esm/routing/useFilters.d.ts.map +1 -0
- package/lib/esm/routing/useFilters.js +83 -0
- package/lib/esm/routing/useNavigate.d.ts +2 -9
- package/lib/esm/routing/useNavigate.d.ts.map +1 -1
- package/lib/esm/routing/useNavigate.js +1 -1
- package/lib/esm/routing/useQueryState.d.ts +1 -1
- package/lib/esm/routing/useQueryState.d.ts.map +1 -1
- package/lib/esm/routing/useQueryState.js +2 -2
- package/lib/esm/routing/useTypedQuery.js +1 -1
- package/lib/esm/tools/buildCombinedSearch.d.ts +11 -0
- package/lib/esm/tools/buildCombinedSearch.d.ts.map +1 -0
- package/lib/esm/tools/buildCombinedSearch.js +28 -0
- package/lib/esm/tools/buildSearch.d.ts +2 -8
- package/lib/esm/tools/buildSearch.d.ts.map +1 -1
- package/lib/esm/tools/buildSearch.js +12 -216
- package/lib/esm/tools.d.ts +1 -1
- package/lib/esm/tools.d.ts.map +1 -1
- package/lib/esm/tools.js +1 -1
- package/lib/esm/types.d.ts +25 -37
- package/lib/esm/types.d.ts.map +1 -1
- package/lib/esm/types.js +1 -1
- package/lib/tsconfig.esm.tsbuildinfo +1 -1
- package/lib/types/builder.d.ts +5 -5
- package/lib/types/builder.d.ts.map +1 -1
- package/lib/types/index.d.ts +0 -4
- package/lib/types/index.d.ts.map +1 -1
- package/lib/types/routing/Link.d.ts +3 -3
- package/lib/types/routing/Link.d.ts.map +1 -1
- package/lib/types/routing/RouteComponentWrapper.d.ts +6 -5
- package/lib/types/routing/RouteComponentWrapper.d.ts.map +1 -1
- package/lib/types/routing/RouterRenderer.d.ts.map +1 -1
- package/lib/types/routing/createRouter.d.ts +0 -1
- package/lib/types/routing/createRouter.d.ts.map +1 -1
- package/lib/types/routing/index.d.ts +2 -6
- package/lib/types/routing/index.d.ts.map +1 -1
- package/lib/types/routing/useAllQuery.d.ts +7 -0
- package/lib/types/routing/useAllQuery.d.ts.map +1 -0
- package/lib/types/routing/useFilterDiagnostics.d.ts +2 -0
- package/lib/types/routing/useFilterDiagnostics.d.ts.map +1 -0
- package/lib/types/routing/useFilters.d.ts +16 -0
- package/lib/types/routing/useFilters.d.ts.map +1 -0
- package/lib/types/routing/useNavigate.d.ts +2 -9
- package/lib/types/routing/useNavigate.d.ts.map +1 -1
- package/lib/types/routing/useQueryState.d.ts +1 -1
- package/lib/types/routing/useQueryState.d.ts.map +1 -1
- package/lib/types/tools/buildCombinedSearch.d.ts +11 -0
- package/lib/types/tools/buildCombinedSearch.d.ts.map +1 -0
- package/lib/types/tools/buildSearch.d.ts +2 -8
- package/lib/types/tools/buildSearch.d.ts.map +1 -1
- package/lib/types/tools.d.ts +1 -1
- package/lib/types/tools.d.ts.map +1 -1
- package/lib/types/types.d.ts +25 -37
- package/lib/types/types.d.ts.map +1 -1
- package/package.json +6 -5
- package/lib/esm/routing/devtools.d.ts +0 -20
- package/lib/esm/routing/devtools.d.ts.map +0 -1
- package/lib/esm/routing/devtools.js +0 -678
- package/lib/esm/routing/filters.d.ts +0 -97
- package/lib/esm/routing/filters.d.ts.map +0 -1
- package/lib/esm/routing/filters.js +0 -557
- package/lib/esm/routing/useActiveFilters.d.ts +0 -9
- package/lib/esm/routing/useActiveFilters.d.ts.map +0 -1
- package/lib/esm/routing/useActiveFilters.js +0 -38
- package/lib/esm/routing/useFilterState.d.ts +0 -10
- package/lib/esm/routing/useFilterState.d.ts.map +0 -1
- package/lib/esm/routing/useFilterState.js +0 -14
- package/lib/esm/routing/useNavigateWithQuery.d.ts +0 -15
- package/lib/esm/routing/useNavigateWithQuery.d.ts.map +0 -1
- package/lib/esm/routing/useNavigateWithQuery.js +0 -95
- package/lib/esm/routing/useQueryObject.d.ts +0 -18
- package/lib/esm/routing/useQueryObject.d.ts.map +0 -1
- package/lib/esm/routing/useQueryObject.js +0 -107
- package/lib/esm/routing/useStableRefEquality.d.ts +0 -5
- package/lib/esm/routing/useStableRefEquality.d.ts.map +0 -1
- package/lib/esm/routing/useStableRefEquality.js +0 -47
- package/lib/types/routing/devtools.d.ts +0 -20
- package/lib/types/routing/devtools.d.ts.map +0 -1
- package/lib/types/routing/filters.d.ts +0 -97
- package/lib/types/routing/filters.d.ts.map +0 -1
- package/lib/types/routing/useActiveFilters.d.ts +0 -9
- package/lib/types/routing/useActiveFilters.d.ts.map +0 -1
- package/lib/types/routing/useFilterState.d.ts +0 -10
- package/lib/types/routing/useFilterState.d.ts.map +0 -1
- package/lib/types/routing/useNavigateWithQuery.d.ts +0 -15
- package/lib/types/routing/useNavigateWithQuery.d.ts.map +0 -1
- package/lib/types/routing/useQueryObject.d.ts +0 -18
- package/lib/types/routing/useQueryObject.d.ts.map +0 -1
- package/lib/types/routing/useStableRefEquality.d.ts +0 -5
- package/lib/types/routing/useStableRefEquality.d.ts.map +0 -1
|
@@ -4,6 +4,8 @@ import { getMatchedRoute, prepareMatch } from '../tools.js';
|
|
|
4
4
|
import { parseTypedQuery } from '../tools/query-dsl.js';
|
|
5
5
|
import { parseRawQuery } from '../tools/query.js';
|
|
6
6
|
import { buildSearch } from '../tools/buildSearch.js';
|
|
7
|
+
import buildCombinedSearch from '../tools/buildCombinedSearch.js';
|
|
8
|
+
import { parse as parseFilters } from '@plumile/filter-query';
|
|
7
9
|
export default function createRouter(routes, opts = {}) {
|
|
8
10
|
let devtoolsGlobal;
|
|
9
11
|
let devtoolsPanel = false;
|
|
@@ -46,22 +48,36 @@ export default function createRouter(routes, opts = {}) {
|
|
|
46
48
|
let initialSchema;
|
|
47
49
|
if (route != null) {
|
|
48
50
|
const collected = [];
|
|
49
|
-
const collectedFilters = [];
|
|
50
51
|
for (const r of route.route.routes) {
|
|
51
52
|
const maybe = r.query;
|
|
52
53
|
if (maybe != null) {
|
|
53
54
|
collected.push(maybe);
|
|
54
55
|
}
|
|
55
|
-
const maybeFilter = r.filterSchema;
|
|
56
|
-
if (maybeFilter != null) {
|
|
57
|
-
collectedFilters.push(maybeFilter);
|
|
58
|
-
}
|
|
59
56
|
}
|
|
60
57
|
if (collected.length > 0) {
|
|
61
58
|
initialSchema = collected.at(-1);
|
|
62
59
|
}
|
|
63
60
|
}
|
|
64
61
|
const initialTypedQuery = parseTypedQuery(initialSchema, initialRawQuery);
|
|
62
|
+
let initialFilterSchema;
|
|
63
|
+
if (route != null) {
|
|
64
|
+
const collected = [];
|
|
65
|
+
for (const r of route.route.routes) {
|
|
66
|
+
const maybe = r
|
|
67
|
+
.filterSchema;
|
|
68
|
+
if (maybe != null)
|
|
69
|
+
collected.push(maybe);
|
|
70
|
+
}
|
|
71
|
+
if (collected.length > 0)
|
|
72
|
+
initialFilterSchema = collected.at(-1);
|
|
73
|
+
}
|
|
74
|
+
let initialFilters;
|
|
75
|
+
let initialFilterDiagnostics;
|
|
76
|
+
if (initialFilterSchema != null) {
|
|
77
|
+
const parsed = parseFilters(history.location.search, initialFilterSchema);
|
|
78
|
+
initialFilters = parsed.filters;
|
|
79
|
+
initialFilterDiagnostics = parsed.diagnostics;
|
|
80
|
+
}
|
|
65
81
|
const preparedMatch = prepareMatch(route, initialTypedQuery);
|
|
66
82
|
let currentEntry = {
|
|
67
83
|
forceRerender: false,
|
|
@@ -71,6 +87,8 @@ export default function createRouter(routes, opts = {}) {
|
|
|
71
87
|
rawSearch: history.location.search,
|
|
72
88
|
query: initialRawQuery,
|
|
73
89
|
typedQuery: initialTypedQuery,
|
|
90
|
+
filters: initialFilters,
|
|
91
|
+
filterDiagnostics: initialFilterDiagnostics,
|
|
74
92
|
};
|
|
75
93
|
const initTyped = currentEntry.typedQuery;
|
|
76
94
|
let initPage;
|
|
@@ -106,24 +124,35 @@ export default function createRouter(routes, opts = {}) {
|
|
|
106
124
|
}
|
|
107
125
|
const query = parseRawQuery(location.search);
|
|
108
126
|
let schema;
|
|
127
|
+
let filterSchema;
|
|
109
128
|
if (nextRoute != null) {
|
|
110
129
|
const collected = [];
|
|
111
|
-
const
|
|
130
|
+
const filterCollected = [];
|
|
112
131
|
for (const r of nextRoute.route.routes) {
|
|
113
132
|
const maybe = r.query;
|
|
114
133
|
if (maybe != null) {
|
|
115
134
|
collected.push(maybe);
|
|
116
135
|
}
|
|
117
|
-
const
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
136
|
+
const maybeF = r
|
|
137
|
+
.filterSchema;
|
|
138
|
+
if (maybeF != null)
|
|
139
|
+
filterCollected.push(maybeF);
|
|
121
140
|
}
|
|
122
141
|
if (collected.length > 0) {
|
|
123
142
|
schema = collected.at(-1);
|
|
124
143
|
}
|
|
144
|
+
if (filterCollected.length > 0) {
|
|
145
|
+
filterSchema = filterCollected.at(-1);
|
|
146
|
+
}
|
|
125
147
|
}
|
|
126
148
|
let typedQuery = parseTypedQuery(schema, query);
|
|
149
|
+
let filters;
|
|
150
|
+
let filterDiagnostics;
|
|
151
|
+
if (filterSchema != null) {
|
|
152
|
+
const parsedF = parseFilters(location.search, filterSchema);
|
|
153
|
+
filters = parsedF.filters;
|
|
154
|
+
filterDiagnostics = parsedF.diagnostics;
|
|
155
|
+
}
|
|
127
156
|
let normalized = false;
|
|
128
157
|
const pg = typedQuery.page;
|
|
129
158
|
if (typeof pg === 'number' && pg < 1) {
|
|
@@ -145,9 +174,16 @@ export default function createRouter(routes, opts = {}) {
|
|
|
145
174
|
rawSearch: location.search,
|
|
146
175
|
query,
|
|
147
176
|
typedQuery,
|
|
177
|
+
filters,
|
|
178
|
+
filterDiagnostics,
|
|
148
179
|
};
|
|
149
180
|
if (normalized && schema != null) {
|
|
150
|
-
const normalizedSearch =
|
|
181
|
+
const normalizedSearch = buildCombinedSearch({
|
|
182
|
+
query: typedQuery,
|
|
183
|
+
querySchema: schema,
|
|
184
|
+
filters,
|
|
185
|
+
filterSchema,
|
|
186
|
+
});
|
|
151
187
|
if (normalizedSearch !== location.search) {
|
|
152
188
|
history.set({
|
|
153
189
|
pathname: location.pathname,
|
|
@@ -162,272 +198,23 @@ export default function createRouter(routes, opts = {}) {
|
|
|
162
198
|
callback(nextEntry);
|
|
163
199
|
});
|
|
164
200
|
});
|
|
165
|
-
let pendingNavs;
|
|
166
|
-
let flushScheduled = false;
|
|
167
|
-
let batchedMergedQuery;
|
|
168
|
-
let lastBatchQuery;
|
|
169
|
-
let batchedFilters;
|
|
170
|
-
let batchedFilterSchema;
|
|
171
|
-
let devtoolsOnNavEvent;
|
|
172
|
-
let batchMode;
|
|
173
|
-
function collectRouteChain(pathname) {
|
|
174
|
-
if (pathname === currentEntry.location.pathname) {
|
|
175
|
-
if (currentEntry.route != null) {
|
|
176
|
-
return currentEntry.route.route.routes;
|
|
177
|
-
}
|
|
178
|
-
return undefined;
|
|
179
|
-
}
|
|
180
|
-
const destRoute = getMatchedRoute(flatRoutes, {
|
|
181
|
-
...window.location,
|
|
182
|
-
pathname,
|
|
183
|
-
});
|
|
184
|
-
if (destRoute != null)
|
|
185
|
-
return destRoute.route.routes;
|
|
186
|
-
return undefined;
|
|
187
|
-
}
|
|
188
|
-
function mergeQuerySchemaFromChain(chain) {
|
|
189
|
-
if (chain == null)
|
|
190
|
-
return undefined;
|
|
191
|
-
const merged = {};
|
|
192
|
-
for (const node of chain) {
|
|
193
|
-
const qobj = node.query;
|
|
194
|
-
if (qobj != null) {
|
|
195
|
-
for (const [k, v] of Object.entries(qobj)) {
|
|
196
|
-
merged[k] = v;
|
|
197
|
-
}
|
|
198
|
-
}
|
|
199
|
-
}
|
|
200
|
-
if (Object.keys(merged).length === 0)
|
|
201
|
-
return undefined;
|
|
202
|
-
return merged;
|
|
203
|
-
}
|
|
204
|
-
function mergeFilterSchemaFromChain(chain) {
|
|
205
|
-
if (chain == null)
|
|
206
|
-
return undefined;
|
|
207
|
-
let merged;
|
|
208
|
-
for (const node of chain) {
|
|
209
|
-
const fs = node.filterSchema;
|
|
210
|
-
if (fs != null) {
|
|
211
|
-
merged ??= {};
|
|
212
|
-
Object.assign(merged, fs.schema);
|
|
213
|
-
}
|
|
214
|
-
}
|
|
215
|
-
if (merged != null)
|
|
216
|
-
return { __defined: true, schema: merged };
|
|
217
|
-
return undefined;
|
|
218
|
-
}
|
|
219
|
-
function resolveSchema(pathname) {
|
|
220
|
-
return mergeQuerySchemaFromChain(collectRouteChain(pathname));
|
|
221
|
-
}
|
|
222
|
-
function resolveFilterSchema(pathname) {
|
|
223
|
-
return mergeFilterSchemaFromChain(collectRouteChain(pathname));
|
|
224
|
-
}
|
|
225
|
-
function coalesceNavigations(list) {
|
|
226
|
-
let targetPathname = currentEntry.location.pathname;
|
|
227
|
-
if (targetPathname === '') {
|
|
228
|
-
targetPathname = '/';
|
|
229
|
-
}
|
|
230
|
-
for (const nav of list) {
|
|
231
|
-
if (nav.pathname !== undefined) {
|
|
232
|
-
const pn = nav.pathname;
|
|
233
|
-
if (pn === '') {
|
|
234
|
-
targetPathname = '/';
|
|
235
|
-
}
|
|
236
|
-
else {
|
|
237
|
-
targetPathname = pn;
|
|
238
|
-
}
|
|
239
|
-
}
|
|
240
|
-
}
|
|
241
|
-
let replaceFlag;
|
|
242
|
-
for (const nav of list) {
|
|
243
|
-
if (nav.replace !== undefined) {
|
|
244
|
-
replaceFlag = nav.replace;
|
|
245
|
-
}
|
|
246
|
-
}
|
|
247
|
-
let baseQuery = currentEntry.typedQuery ?? currentEntry.query ?? {};
|
|
248
|
-
if (batchedMergedQuery != null) {
|
|
249
|
-
baseQuery = batchedMergedQuery;
|
|
250
|
-
}
|
|
251
|
-
else if (lastBatchQuery != null) {
|
|
252
|
-
baseQuery = lastBatchQuery;
|
|
253
|
-
}
|
|
254
|
-
else {
|
|
255
|
-
for (let i = list.length - 1; i >= 0; i -= 1) {
|
|
256
|
-
const item = list[i];
|
|
257
|
-
if (item != null) {
|
|
258
|
-
const q = item.query;
|
|
259
|
-
if (q != null && Object.keys(q).length > 0) {
|
|
260
|
-
baseQuery = q;
|
|
261
|
-
break;
|
|
262
|
-
}
|
|
263
|
-
}
|
|
264
|
-
}
|
|
265
|
-
}
|
|
266
|
-
const batchFilters = batchedFilters;
|
|
267
|
-
const batchFilterSchema = batchedFilterSchema;
|
|
268
|
-
const schema = resolveSchema(targetPathname);
|
|
269
|
-
let buildOpts;
|
|
270
|
-
if (batchFilters !== undefined) {
|
|
271
|
-
const fs = batchFilterSchema ?? resolveFilterSchema(targetPathname);
|
|
272
|
-
if (fs != null) {
|
|
273
|
-
buildOpts = { filters: batchFilters, filterSchema: fs };
|
|
274
|
-
}
|
|
275
|
-
}
|
|
276
|
-
const search = buildSearch(baseQuery, schema, buildOpts);
|
|
277
|
-
let normalizedPath = targetPathname;
|
|
278
|
-
if (normalizedPath === '') {
|
|
279
|
-
normalizedPath = '/';
|
|
280
|
-
}
|
|
281
|
-
let nextRoute = currentEntry.route;
|
|
282
|
-
if (normalizedPath !== currentEntry.location.pathname) {
|
|
283
|
-
nextRoute = getMatchedRoute(flatRoutes, {
|
|
284
|
-
...window.location,
|
|
285
|
-
pathname: normalizedPath,
|
|
286
|
-
});
|
|
287
|
-
}
|
|
288
|
-
const effectiveSchema = schema ?? resolveSchema(normalizedPath);
|
|
289
|
-
const rawQueryObj = parseRawQuery(search);
|
|
290
|
-
const typed = parseTypedQuery(effectiveSchema, rawQueryObj);
|
|
291
|
-
const prepared = prepareMatch(nextRoute, typed);
|
|
292
|
-
const nextEntry = {
|
|
293
|
-
forceRerender: false,
|
|
294
|
-
location: { ...currentEntry.location, pathname: normalizedPath, search },
|
|
295
|
-
route: nextRoute,
|
|
296
|
-
preparedMatch: prepared,
|
|
297
|
-
rawSearch: search,
|
|
298
|
-
query: rawQueryObj,
|
|
299
|
-
typedQuery: typed,
|
|
300
|
-
};
|
|
301
|
-
return {
|
|
302
|
-
search,
|
|
303
|
-
pathname: normalizedPath,
|
|
304
|
-
replace: replaceFlag,
|
|
305
|
-
nextEntry,
|
|
306
|
-
};
|
|
307
|
-
}
|
|
308
|
-
function flushNavigations() {
|
|
309
|
-
flushScheduled = false;
|
|
310
|
-
const list = pendingNavs;
|
|
311
|
-
pendingNavs = undefined;
|
|
312
|
-
if (list == null || list.length === 0) {
|
|
313
|
-
return;
|
|
314
|
-
}
|
|
315
|
-
const { search, pathname, replace: replaceFlag, nextEntry, } = coalesceNavigations(list);
|
|
316
|
-
batchedMergedQuery = undefined;
|
|
317
|
-
lastBatchQuery = undefined;
|
|
318
|
-
batchedFilters = undefined;
|
|
319
|
-
batchedFilterSchema = undefined;
|
|
320
|
-
currentEntry = nextEntry;
|
|
321
|
-
const loc = { pathname, search, hash: '' };
|
|
322
|
-
if (replaceFlag)
|
|
323
|
-
history.set(loc);
|
|
324
|
-
else
|
|
325
|
-
history.push(loc);
|
|
326
|
-
let mode = 'manual';
|
|
327
|
-
if (batchMode === 'auto') {
|
|
328
|
-
mode = 'auto';
|
|
329
|
-
}
|
|
330
|
-
else if (batchMode === 'mixed') {
|
|
331
|
-
mode = 'mixed';
|
|
332
|
-
}
|
|
333
|
-
devtoolsOnNavEvent?.({
|
|
334
|
-
kind: 'batch-flush',
|
|
335
|
-
mode,
|
|
336
|
-
count: list.length,
|
|
337
|
-
pathname,
|
|
338
|
-
search,
|
|
339
|
-
timestamp: Date.now(),
|
|
340
|
-
});
|
|
341
|
-
batchMode = undefined;
|
|
342
|
-
}
|
|
343
|
-
function applyProvisionalBatchUpdate(pathname) {
|
|
344
|
-
try {
|
|
345
|
-
const basePath = pathname ?? currentEntry.location.pathname;
|
|
346
|
-
const schema = resolveSchema(basePath);
|
|
347
|
-
const mergedQ = batchedMergedQuery ?? lastBatchQuery ?? {};
|
|
348
|
-
let buildOpts;
|
|
349
|
-
if (batchedFilters != null && batchedFilterSchema != null) {
|
|
350
|
-
buildOpts = {
|
|
351
|
-
filters: batchedFilters,
|
|
352
|
-
filterSchema: batchedFilterSchema,
|
|
353
|
-
};
|
|
354
|
-
}
|
|
355
|
-
else if (batchedFilters != null && batchedFilterSchema == null) {
|
|
356
|
-
let merged;
|
|
357
|
-
let routeChain;
|
|
358
|
-
if (basePath === currentEntry.location.pathname) {
|
|
359
|
-
if (currentEntry.route != null) {
|
|
360
|
-
routeChain = currentEntry.route.route.routes;
|
|
361
|
-
}
|
|
362
|
-
}
|
|
363
|
-
else {
|
|
364
|
-
const destRoute = getMatchedRoute(flatRoutes, {
|
|
365
|
-
...window.location,
|
|
366
|
-
pathname: basePath,
|
|
367
|
-
});
|
|
368
|
-
if (destRoute != null) {
|
|
369
|
-
routeChain = destRoute.route.routes;
|
|
370
|
-
}
|
|
371
|
-
}
|
|
372
|
-
if (routeChain != null) {
|
|
373
|
-
for (const node of routeChain) {
|
|
374
|
-
const fs = node.filterSchema;
|
|
375
|
-
if (fs != null) {
|
|
376
|
-
merged ??= {};
|
|
377
|
-
Object.assign(merged, fs.schema);
|
|
378
|
-
}
|
|
379
|
-
}
|
|
380
|
-
}
|
|
381
|
-
if (merged != null) {
|
|
382
|
-
buildOpts = {
|
|
383
|
-
filters: batchedFilters,
|
|
384
|
-
filterSchema: { __defined: true, schema: merged },
|
|
385
|
-
};
|
|
386
|
-
}
|
|
387
|
-
}
|
|
388
|
-
const provisionalSearch = buildSearch(mergedQ, schema, buildOpts);
|
|
389
|
-
const pathChanged = basePath !== currentEntry.location.pathname;
|
|
390
|
-
if (provisionalSearch !== currentEntry.rawSearch || pathChanged) {
|
|
391
|
-
const typed = parseTypedQuery(schema, mergedQ);
|
|
392
|
-
const rawQueryObj = parseRawQuery(provisionalSearch);
|
|
393
|
-
currentEntry = {
|
|
394
|
-
...currentEntry,
|
|
395
|
-
location: {
|
|
396
|
-
...currentEntry.location,
|
|
397
|
-
pathname: basePath,
|
|
398
|
-
search: provisionalSearch,
|
|
399
|
-
},
|
|
400
|
-
rawSearch: provisionalSearch,
|
|
401
|
-
query: rawQueryObj,
|
|
402
|
-
typedQuery: typed,
|
|
403
|
-
};
|
|
404
|
-
}
|
|
405
|
-
}
|
|
406
|
-
catch {
|
|
407
|
-
}
|
|
408
|
-
}
|
|
409
|
-
const autoBatch = opts.autoBatch === true;
|
|
410
201
|
const context = {
|
|
411
202
|
history,
|
|
412
203
|
get() {
|
|
413
204
|
return currentEntry;
|
|
414
205
|
},
|
|
415
|
-
resolveFilterSchema(pathname) {
|
|
416
|
-
return resolveFilterSchema(pathname);
|
|
417
|
-
},
|
|
418
|
-
currentMergedFilterSchema() {
|
|
419
|
-
return resolveFilterSchema(currentEntry.location.pathname);
|
|
420
|
-
},
|
|
421
206
|
preloadCode(pathname) {
|
|
422
207
|
const matches = getMatchedRoute(flatRoutes, {
|
|
423
208
|
...window.location,
|
|
424
209
|
pathname,
|
|
425
210
|
});
|
|
426
|
-
if (matches == null)
|
|
211
|
+
if (matches == null) {
|
|
427
212
|
return;
|
|
213
|
+
}
|
|
428
214
|
matches.route.routes.forEach(({ resourcePage }) => {
|
|
429
|
-
if (resourcePage == null)
|
|
215
|
+
if (resourcePage == null) {
|
|
430
216
|
return;
|
|
217
|
+
}
|
|
431
218
|
resourcePage.load();
|
|
432
219
|
});
|
|
433
220
|
},
|
|
@@ -441,158 +228,406 @@ export default function createRouter(routes, opts = {}) {
|
|
|
441
228
|
subscribe(callback) {
|
|
442
229
|
nextId += 1;
|
|
443
230
|
const id = nextId;
|
|
444
|
-
|
|
231
|
+
const dispose = () => {
|
|
445
232
|
subscribers.delete(id);
|
|
446
|
-
}
|
|
233
|
+
};
|
|
447
234
|
subscribers.set(id, callback);
|
|
448
235
|
return dispose;
|
|
449
236
|
},
|
|
450
|
-
navigate({ pathname, query,
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
if (pathname !== undefined)
|
|
458
|
-
nav.pathname = pathname;
|
|
459
|
-
if (query !== undefined)
|
|
460
|
-
nav.query = query;
|
|
461
|
-
if (replace !== undefined)
|
|
462
|
-
nav.replace = replace;
|
|
463
|
-
if (filters !== undefined)
|
|
464
|
-
nav.filters = filters;
|
|
465
|
-
if (filterSchema !== undefined)
|
|
466
|
-
nav.filterSchema = filterSchema;
|
|
467
|
-
if (effectiveInherit)
|
|
468
|
-
nav.inherit = true;
|
|
469
|
-
pendingNavs ??= [];
|
|
470
|
-
pendingNavs.push(nav);
|
|
471
|
-
let thisMode;
|
|
472
|
-
if (batch === true)
|
|
473
|
-
thisMode = 'manual';
|
|
474
|
-
else
|
|
475
|
-
thisMode = 'auto';
|
|
476
|
-
if (batchMode == null)
|
|
477
|
-
batchMode = thisMode;
|
|
478
|
-
else if (batchMode !== thisMode)
|
|
479
|
-
batchMode = 'mixed';
|
|
480
|
-
if (query != null) {
|
|
481
|
-
if (effectiveInherit) {
|
|
482
|
-
const base = batchedMergedQuery ??
|
|
483
|
-
lastBatchQuery ??
|
|
484
|
-
currentEntry.typedQuery ??
|
|
485
|
-
currentEntry.query ??
|
|
486
|
-
{};
|
|
487
|
-
batchedMergedQuery = { ...base, ...query };
|
|
488
|
-
}
|
|
489
|
-
else {
|
|
490
|
-
batchedMergedQuery = { ...query };
|
|
491
|
-
}
|
|
492
|
-
lastBatchQuery = query;
|
|
493
|
-
}
|
|
494
|
-
if (filters != null) {
|
|
495
|
-
batchedFilters = filters;
|
|
496
|
-
if (filterSchema != null)
|
|
497
|
-
batchedFilterSchema = filterSchema;
|
|
498
|
-
}
|
|
499
|
-
if (!flushScheduled) {
|
|
500
|
-
flushScheduled = true;
|
|
501
|
-
Promise.resolve().then(flushNavigations);
|
|
502
|
-
}
|
|
503
|
-
applyProvisionalBatchUpdate(pathname);
|
|
504
|
-
subscribers.forEach((cb) => {
|
|
505
|
-
cb(currentEntry);
|
|
506
|
-
});
|
|
237
|
+
navigate({ pathname, query, filters: navFilters, replace, }) {
|
|
238
|
+
const current = currentEntry;
|
|
239
|
+
const targetPathname = pathname ?? current.location.pathname;
|
|
240
|
+
let schema;
|
|
241
|
+
let filterSchema;
|
|
242
|
+
function collectSchema(routesArr) {
|
|
243
|
+
if (routesArr == null) {
|
|
507
244
|
return;
|
|
508
245
|
}
|
|
509
|
-
|
|
510
|
-
|
|
511
|
-
|
|
512
|
-
|
|
513
|
-
|
|
514
|
-
|
|
515
|
-
|
|
516
|
-
|
|
517
|
-
|
|
246
|
+
const collected = [];
|
|
247
|
+
const filterCollected = [];
|
|
248
|
+
for (const r of routesArr) {
|
|
249
|
+
const maybe = r.query;
|
|
250
|
+
if (maybe != null) {
|
|
251
|
+
collected.push(maybe);
|
|
252
|
+
}
|
|
253
|
+
const maybeF = r
|
|
254
|
+
.filterSchema;
|
|
255
|
+
if (maybeF != null)
|
|
256
|
+
filterCollected.push(maybeF);
|
|
518
257
|
}
|
|
519
|
-
|
|
520
|
-
|
|
521
|
-
const schema = resolveSchema(targetPath);
|
|
522
|
-
let buildOpts;
|
|
523
|
-
if (filters != null) {
|
|
524
|
-
const fs = filterSchema ?? resolveFilterSchema(targetPath);
|
|
525
|
-
if (fs != null)
|
|
526
|
-
buildOpts = { filters, filterSchema: fs };
|
|
258
|
+
if (collected.length > 0) {
|
|
259
|
+
schema = collected.at(-1);
|
|
527
260
|
}
|
|
528
|
-
|
|
529
|
-
|
|
530
|
-
if (targetPathname === '')
|
|
531
|
-
targetPathname = '/';
|
|
532
|
-
let nextRoute = currentEntry.route;
|
|
533
|
-
if (targetPathname !== currentEntry.location.pathname) {
|
|
534
|
-
nextRoute = getMatchedRoute(flatRoutes, {
|
|
535
|
-
...window.location,
|
|
536
|
-
pathname: targetPathname,
|
|
537
|
-
});
|
|
261
|
+
if (filterCollected.length > 0) {
|
|
262
|
+
filterSchema = filterCollected.at(-1);
|
|
538
263
|
}
|
|
539
|
-
|
|
540
|
-
|
|
541
|
-
|
|
542
|
-
|
|
543
|
-
|
|
544
|
-
|
|
545
|
-
location
|
|
546
|
-
|
|
547
|
-
pathname: targetPathname,
|
|
548
|
-
search,
|
|
549
|
-
},
|
|
550
|
-
route: nextRoute,
|
|
551
|
-
preparedMatch: prepared,
|
|
552
|
-
rawSearch: search,
|
|
553
|
-
query: rawQueryObj,
|
|
554
|
-
typedQuery: typed,
|
|
555
|
-
};
|
|
556
|
-
let normImmediate = targetPathname;
|
|
557
|
-
if (normImmediate === '')
|
|
558
|
-
normImmediate = '/';
|
|
559
|
-
const locationObj = { pathname: normImmediate, search, hash: '' };
|
|
560
|
-
if (replace)
|
|
561
|
-
history.set(locationObj);
|
|
562
|
-
else
|
|
563
|
-
history.push(locationObj);
|
|
564
|
-
devtoolsOnNavEvent?.({
|
|
565
|
-
kind: 'navigate',
|
|
566
|
-
mode: 'immediate',
|
|
567
|
-
count: 1,
|
|
568
|
-
pathname: normImmediate,
|
|
569
|
-
search,
|
|
570
|
-
timestamp: Date.now(),
|
|
264
|
+
}
|
|
265
|
+
if (pathname == null || pathname === current.location.pathname) {
|
|
266
|
+
collectSchema(current.route?.route.routes);
|
|
267
|
+
}
|
|
268
|
+
else {
|
|
269
|
+
const destRoute = getMatchedRoute(flatRoutes, {
|
|
270
|
+
...window.location,
|
|
271
|
+
pathname: targetPathname,
|
|
571
272
|
});
|
|
273
|
+
collectSchema(destRoute?.route.routes);
|
|
572
274
|
}
|
|
573
|
-
|
|
275
|
+
const effectiveFilters = navFilters ?? current.filters;
|
|
276
|
+
const search = buildCombinedSearch({
|
|
277
|
+
query,
|
|
278
|
+
querySchema: schema,
|
|
279
|
+
filters: effectiveFilters,
|
|
280
|
+
filterSchema,
|
|
281
|
+
});
|
|
282
|
+
const locationObj = { pathname: targetPathname, search, hash: '' };
|
|
283
|
+
if (replace) {
|
|
284
|
+
history.set(locationObj);
|
|
285
|
+
}
|
|
286
|
+
else {
|
|
287
|
+
history.push(locationObj);
|
|
574
288
|
}
|
|
575
289
|
},
|
|
576
290
|
};
|
|
577
|
-
if (
|
|
578
|
-
|
|
579
|
-
|
|
580
|
-
|
|
581
|
-
|
|
582
|
-
|
|
583
|
-
context,
|
|
584
|
-
|
|
585
|
-
|
|
586
|
-
|
|
587
|
-
|
|
588
|
-
|
|
291
|
+
if (devtoolsGlobal) {
|
|
292
|
+
try {
|
|
293
|
+
if (typeof window !== 'undefined') {
|
|
294
|
+
window.__PLUMILE_ROUTER__ =
|
|
295
|
+
{
|
|
296
|
+
get: context.get,
|
|
297
|
+
subscribe: context.subscribe,
|
|
298
|
+
};
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
catch {
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
if (devtoolsPanel) {
|
|
305
|
+
try {
|
|
306
|
+
if (typeof window !== 'undefined' &&
|
|
307
|
+
typeof document !== 'undefined' &&
|
|
308
|
+
document.querySelector('[data-plumile-router-devtools-panel]') == null) {
|
|
309
|
+
const host = document.createElement('div');
|
|
310
|
+
host.setAttribute('data-plumile-router-devtools-panel', '');
|
|
311
|
+
host.style.position = 'fixed';
|
|
312
|
+
host.style.zIndex = '2147483646';
|
|
313
|
+
host.style.bottom = '8px';
|
|
314
|
+
host.style.right = '8px';
|
|
315
|
+
host.style.fontFamily =
|
|
316
|
+
'ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", monospace';
|
|
317
|
+
host.style.fontSize = '12px';
|
|
318
|
+
host.style.lineHeight = '1.3';
|
|
319
|
+
host.style.color = '#111';
|
|
320
|
+
host.style.background = 'transparent';
|
|
321
|
+
host.style.pointerEvents = 'none';
|
|
322
|
+
const shadow = host.attachShadow({ mode: 'open' });
|
|
323
|
+
const panel = document.createElement('div');
|
|
324
|
+
panel.style.pointerEvents = 'auto';
|
|
325
|
+
panel.style.minWidth = '260px';
|
|
326
|
+
panel.style.maxWidth = '420px';
|
|
327
|
+
panel.style.maxHeight = '50vh';
|
|
328
|
+
panel.style.overflow = 'auto';
|
|
329
|
+
panel.style.background = 'rgba(255,255,255,0.95)';
|
|
330
|
+
panel.style.border = '1px solid #ccc';
|
|
331
|
+
panel.style.borderRadius = '6px';
|
|
332
|
+
panel.style.boxShadow = '0 2px 8px rgba(0,0,0,0.15)';
|
|
333
|
+
panel.style.padding = '6px 8px 8px';
|
|
334
|
+
panel.style.backdropFilter = 'blur(4px)';
|
|
335
|
+
panel.style.display = 'flex';
|
|
336
|
+
panel.style.flexDirection = 'column';
|
|
337
|
+
panel.style.rowGap = '4px';
|
|
338
|
+
const header = document.createElement('div');
|
|
339
|
+
header.style.display = 'flex';
|
|
340
|
+
header.style.alignItems = 'center';
|
|
341
|
+
header.style.justifyContent = 'space-between';
|
|
342
|
+
header.style.gap = '8px';
|
|
343
|
+
const title = document.createElement('span');
|
|
344
|
+
title.textContent = 'Plumile Router';
|
|
345
|
+
title.style.fontWeight = '600';
|
|
346
|
+
const buttons = document.createElement('div');
|
|
347
|
+
buttons.style.display = 'flex';
|
|
348
|
+
buttons.style.gap = '4px';
|
|
349
|
+
const collapseBtn = document.createElement('button');
|
|
350
|
+
collapseBtn.textContent = '−';
|
|
351
|
+
const closeBtn = document.createElement('button');
|
|
352
|
+
closeBtn.textContent = '×';
|
|
353
|
+
const btns = [collapseBtn, closeBtn];
|
|
354
|
+
for (const b of btns) {
|
|
355
|
+
b.style.cursor = 'pointer';
|
|
356
|
+
b.style.background = '#eee';
|
|
357
|
+
b.style.border = '1px solid #ccc';
|
|
358
|
+
b.style.borderRadius = '4px';
|
|
359
|
+
b.style.fontSize = '11px';
|
|
360
|
+
b.style.lineHeight = '1';
|
|
361
|
+
b.style.padding = '2px 5px';
|
|
362
|
+
}
|
|
363
|
+
header.appendChild(title);
|
|
364
|
+
buttons.appendChild(collapseBtn);
|
|
365
|
+
buttons.appendChild(closeBtn);
|
|
366
|
+
header.appendChild(buttons);
|
|
367
|
+
const pathEl = document.createElement('div');
|
|
368
|
+
pathEl.style.whiteSpace = 'nowrap';
|
|
369
|
+
const varsEl = document.createElement('pre');
|
|
370
|
+
const queryEl = document.createElement('pre');
|
|
371
|
+
const typedEl = document.createElement('pre');
|
|
372
|
+
const filtersEl = document.createElement('pre');
|
|
373
|
+
const filterDiagEl = document.createElement('pre');
|
|
374
|
+
const timelineWrap = document.createElement('div');
|
|
375
|
+
timelineWrap.style.display = 'flex';
|
|
376
|
+
timelineWrap.style.flexDirection = 'column';
|
|
377
|
+
timelineWrap.style.gap = '2px';
|
|
378
|
+
timelineWrap.style.marginTop = '4px';
|
|
379
|
+
const timelineHeader = document.createElement('div');
|
|
380
|
+
timelineHeader.textContent = 'Timeline (query changes)';
|
|
381
|
+
timelineHeader.style.fontWeight = '600';
|
|
382
|
+
timelineHeader.style.fontSize = '11px';
|
|
383
|
+
const timeline = document.createElement('div');
|
|
384
|
+
timeline.style.display = 'flex';
|
|
385
|
+
timeline.style.flexDirection = 'column-reverse';
|
|
386
|
+
timeline.style.maxHeight = '140px';
|
|
387
|
+
timeline.style.overflowY = 'auto';
|
|
388
|
+
timeline.style.border = '1px solid #e2e2e2';
|
|
389
|
+
timeline.style.borderRadius = '4px';
|
|
390
|
+
timeline.style.background = '#fafbfc';
|
|
391
|
+
timeline.style.fontFamily = 'inherit';
|
|
392
|
+
timeline.style.fontSize = '11px';
|
|
393
|
+
timelineWrap.appendChild(timelineHeader);
|
|
394
|
+
timelineWrap.appendChild(timeline);
|
|
395
|
+
const pres = [
|
|
396
|
+
varsEl,
|
|
397
|
+
queryEl,
|
|
398
|
+
typedEl,
|
|
399
|
+
filtersEl,
|
|
400
|
+
filterDiagEl,
|
|
401
|
+
];
|
|
402
|
+
for (const p of pres) {
|
|
403
|
+
p.style.margin = '0';
|
|
404
|
+
p.style.padding = '4px';
|
|
405
|
+
p.style.background = '#f6f8fa';
|
|
406
|
+
p.style.border = '1px solid #e2e2e2';
|
|
407
|
+
p.style.borderRadius = '4px';
|
|
408
|
+
p.style.overflowX = 'auto';
|
|
409
|
+
p.style.maxHeight = '160px';
|
|
410
|
+
}
|
|
411
|
+
const small = document.createElement('div');
|
|
412
|
+
small.style.fontSize = '10px';
|
|
413
|
+
small.style.opacity = '0.7';
|
|
414
|
+
small.textContent = `Shortcut: ${devtoolsShortcut}`;
|
|
415
|
+
panel.appendChild(header);
|
|
416
|
+
panel.appendChild(pathEl);
|
|
417
|
+
panel.appendChild(varsEl);
|
|
418
|
+
panel.appendChild(queryEl);
|
|
419
|
+
panel.appendChild(typedEl);
|
|
420
|
+
panel.appendChild(timelineWrap);
|
|
421
|
+
panel.appendChild(filtersEl);
|
|
422
|
+
panel.appendChild(filterDiagEl);
|
|
423
|
+
panel.appendChild(small);
|
|
424
|
+
shadow.appendChild(panel);
|
|
425
|
+
document.body.appendChild(host);
|
|
426
|
+
let collapsed = false;
|
|
427
|
+
collapseBtn.addEventListener('click', () => {
|
|
428
|
+
collapsed = !collapsed;
|
|
429
|
+
if (collapsed) {
|
|
430
|
+
panel.style.maxHeight = '24px';
|
|
431
|
+
panel.style.overflow = 'hidden';
|
|
432
|
+
collapseBtn.textContent = '+';
|
|
433
|
+
}
|
|
434
|
+
else {
|
|
435
|
+
panel.style.maxHeight = '50vh';
|
|
436
|
+
panel.style.overflow = 'auto';
|
|
437
|
+
collapseBtn.textContent = '−';
|
|
438
|
+
}
|
|
439
|
+
});
|
|
440
|
+
let unsub;
|
|
441
|
+
closeBtn.addEventListener('click', () => {
|
|
442
|
+
host.remove();
|
|
443
|
+
if (typeof unsub === 'function') {
|
|
444
|
+
unsub();
|
|
445
|
+
}
|
|
446
|
+
});
|
|
447
|
+
let lastQuery;
|
|
448
|
+
let lastSearch;
|
|
449
|
+
function diffQueries(prev, next) {
|
|
450
|
+
const added = [];
|
|
451
|
+
const removed = [];
|
|
452
|
+
const changed = [];
|
|
453
|
+
const unchanged = [];
|
|
454
|
+
const prevKeys = new Set(Object.keys(prev ?? {}));
|
|
455
|
+
const nextKeys = new Set(Object.keys(next));
|
|
456
|
+
nextKeys.forEach((k) => {
|
|
457
|
+
if (!prevKeys.has(k)) {
|
|
458
|
+
added.push(k);
|
|
459
|
+
return;
|
|
460
|
+
}
|
|
461
|
+
let pv;
|
|
462
|
+
if (prev != null) {
|
|
463
|
+
pv = prev[k];
|
|
464
|
+
}
|
|
465
|
+
const nv = next[k];
|
|
466
|
+
let same = false;
|
|
467
|
+
if (Array.isArray(pv) && Array.isArray(nv)) {
|
|
468
|
+
if (pv.length === nv.length) {
|
|
469
|
+
same = pv.every((v, i) => {
|
|
470
|
+
return v === nv[i];
|
|
471
|
+
});
|
|
472
|
+
}
|
|
473
|
+
}
|
|
474
|
+
else {
|
|
475
|
+
same = pv === nv;
|
|
476
|
+
}
|
|
477
|
+
if (same) {
|
|
478
|
+
unchanged.push(k);
|
|
479
|
+
}
|
|
480
|
+
else {
|
|
481
|
+
changed.push(k);
|
|
482
|
+
}
|
|
483
|
+
prevKeys.delete(k);
|
|
589
484
|
});
|
|
485
|
+
prevKeys.forEach((k) => {
|
|
486
|
+
removed.push(k);
|
|
487
|
+
});
|
|
488
|
+
return { added, removed, changed, unchanged };
|
|
590
489
|
}
|
|
490
|
+
function render() {
|
|
491
|
+
try {
|
|
492
|
+
const entry = context.get();
|
|
493
|
+
pathEl.textContent = `${entry.location.pathname}${entry.location.search}`;
|
|
494
|
+
const varsJson = JSON.stringify(entry.route?.params ?? {}, null, 2);
|
|
495
|
+
varsEl.textContent = `vars: ${varsJson}`;
|
|
496
|
+
const rawJson = JSON.stringify(entry.query, null, 2);
|
|
497
|
+
queryEl.textContent = `raw: ${rawJson}`;
|
|
498
|
+
const typedJson = JSON.stringify(entry.typedQuery, null, 2);
|
|
499
|
+
typedEl.textContent = `typed:${typedJson}`;
|
|
500
|
+
const filtersJson = JSON.stringify(entry.filters, null, 2);
|
|
501
|
+
filtersEl.textContent = `filters:${filtersJson}`;
|
|
502
|
+
const diag = entry.filterDiagnostics ?? [];
|
|
503
|
+
filterDiagEl.textContent = `filterDiagnostics: ${diag.length}`;
|
|
504
|
+
if (entry.rawSearch !== lastSearch) {
|
|
505
|
+
const nowQuery = entry.query;
|
|
506
|
+
const diff = diffQueries(lastQuery, nowQuery);
|
|
507
|
+
const row = document.createElement('div');
|
|
508
|
+
row.style.display = 'flex';
|
|
509
|
+
row.style.flexDirection = 'column';
|
|
510
|
+
row.style.borderBottom = '1px solid #e9e9e9';
|
|
511
|
+
row.style.padding = '3px 4px';
|
|
512
|
+
const head = document.createElement('div');
|
|
513
|
+
head.style.display = 'flex';
|
|
514
|
+
head.style.justifyContent = 'space-between';
|
|
515
|
+
const ts = new Date().toLocaleTimeString();
|
|
516
|
+
let searchDisplay = '(empty)';
|
|
517
|
+
if (entry.location.search !== '') {
|
|
518
|
+
searchDisplay = entry.location.search;
|
|
519
|
+
}
|
|
520
|
+
head.innerHTML = `<strong>${ts}</strong> <code style="background:#eee;padding:1px 3px;border-radius:3px;">${searchDisplay}</code>`;
|
|
521
|
+
const body = document.createElement('div');
|
|
522
|
+
body.style.display = 'flex';
|
|
523
|
+
body.style.flexWrap = 'wrap';
|
|
524
|
+
body.style.gap = '4px';
|
|
525
|
+
function pill(label, kind) {
|
|
526
|
+
const span = document.createElement('span');
|
|
527
|
+
span.textContent = label;
|
|
528
|
+
span.style.padding = '1px 4px';
|
|
529
|
+
span.style.borderRadius = '10px';
|
|
530
|
+
span.style.fontSize = '10px';
|
|
531
|
+
span.style.border = '1px solid';
|
|
532
|
+
switch (kind) {
|
|
533
|
+
case 'added':
|
|
534
|
+
span.style.background = '#e6ffed';
|
|
535
|
+
span.style.borderColor = '#2cbe4e';
|
|
536
|
+
span.style.color = '#22863a';
|
|
537
|
+
break;
|
|
538
|
+
case 'removed':
|
|
539
|
+
span.style.background = '#ffeef0';
|
|
540
|
+
span.style.borderColor = '#d73a49';
|
|
541
|
+
span.style.color = '#cb2431';
|
|
542
|
+
break;
|
|
543
|
+
case 'changed':
|
|
544
|
+
span.style.background = '#fff5b1';
|
|
545
|
+
span.style.borderColor = '#ffd33d';
|
|
546
|
+
span.style.color = '#735c0f';
|
|
547
|
+
break;
|
|
548
|
+
case 'unchanged':
|
|
549
|
+
span.style.background = '#f1f8ff';
|
|
550
|
+
span.style.borderColor = '#79b8ff';
|
|
551
|
+
span.style.color = '#0366d6';
|
|
552
|
+
break;
|
|
553
|
+
default:
|
|
554
|
+
break;
|
|
555
|
+
}
|
|
556
|
+
return span;
|
|
557
|
+
}
|
|
558
|
+
if (diff.added.length > 0) {
|
|
559
|
+
body.append(...diff.added.map((k) => {
|
|
560
|
+
return pill(`+${k}`, 'added');
|
|
561
|
+
}));
|
|
562
|
+
}
|
|
563
|
+
if (diff.changed.length > 0) {
|
|
564
|
+
body.append(...diff.changed.map((k) => {
|
|
565
|
+
return pill(`~${k}`, 'changed');
|
|
566
|
+
}));
|
|
567
|
+
}
|
|
568
|
+
if (diff.removed.length > 0) {
|
|
569
|
+
body.append(...diff.removed.map((k) => {
|
|
570
|
+
return pill(`-${k}`, 'removed');
|
|
571
|
+
}));
|
|
572
|
+
}
|
|
573
|
+
const noAdds = diff.added.length === 0;
|
|
574
|
+
const noChanges = diff.changed.length === 0;
|
|
575
|
+
const noRemovals = diff.removed.length === 0;
|
|
576
|
+
if (noAdds && noChanges && noRemovals) {
|
|
577
|
+
body.append(pill('no-change', 'unchanged'));
|
|
578
|
+
}
|
|
579
|
+
row.appendChild(head);
|
|
580
|
+
row.appendChild(body);
|
|
581
|
+
timeline.appendChild(row);
|
|
582
|
+
if (timeline.childElementCount > 50) {
|
|
583
|
+
const first = timeline.firstElementChild;
|
|
584
|
+
if (first != null) {
|
|
585
|
+
timeline.removeChild(first);
|
|
586
|
+
}
|
|
587
|
+
}
|
|
588
|
+
lastQuery = nowQuery;
|
|
589
|
+
lastSearch = entry.rawSearch;
|
|
590
|
+
}
|
|
591
|
+
}
|
|
592
|
+
catch {
|
|
593
|
+
}
|
|
594
|
+
}
|
|
595
|
+
render();
|
|
596
|
+
unsub = context.subscribe(() => {
|
|
597
|
+
render();
|
|
598
|
+
});
|
|
599
|
+
window.addEventListener('keydown', (ev) => {
|
|
600
|
+
try {
|
|
601
|
+
const parts = [];
|
|
602
|
+
if (ev.altKey)
|
|
603
|
+
parts.push('Alt');
|
|
604
|
+
if (ev.shiftKey)
|
|
605
|
+
parts.push('Shift');
|
|
606
|
+
if (ev.ctrlKey)
|
|
607
|
+
parts.push('Ctrl');
|
|
608
|
+
let keyPart = ev.key;
|
|
609
|
+
if (ev.key.length === 1) {
|
|
610
|
+
keyPart = ev.key.toUpperCase();
|
|
611
|
+
}
|
|
612
|
+
parts.push(keyPart);
|
|
613
|
+
const combo = parts.join('+');
|
|
614
|
+
if (combo === devtoolsShortcut.replace(/\s+/g, '')) {
|
|
615
|
+
if (host.style.display === 'none') {
|
|
616
|
+
host.style.display = 'block';
|
|
617
|
+
}
|
|
618
|
+
else {
|
|
619
|
+
host.style.display = 'none';
|
|
620
|
+
}
|
|
621
|
+
}
|
|
622
|
+
}
|
|
623
|
+
catch {
|
|
624
|
+
}
|
|
625
|
+
});
|
|
591
626
|
}
|
|
592
|
-
|
|
593
|
-
|
|
594
|
-
}
|
|
627
|
+
}
|
|
628
|
+
catch {
|
|
629
|
+
}
|
|
595
630
|
}
|
|
596
631
|
return { cleanup, context };
|
|
597
632
|
}
|
|
598
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"createRouter.js","sourceRoot":"","sources":["../../../src/routing/createRouter.ts"],"names":[],"mappings":"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","sourcesContent":["import { buildRoutes } from '../builder.js';\nimport { BrowserHistory } from '../history/index.js';\nimport { getMatchedRoute, prepareMatch } from '../tools.js';\nimport { parseTypedQuery } from '../tools/query-dsl.js';\nimport { parseRawQuery } from '../tools/query.js';\nimport { buildSearch } from '../tools/buildSearch.js';\nimport type {\n  Route,\n  RouteEntry,\n  RoutingContextType,\n  SubscribeCallback,\n} from '../types.js';\n\n/**\n * Return type for the createRouter function.\n */\nexport type CreateRouterReturn = {\n  /** Function to clean up router listeners and resources */\n  cleanup: () => void;\n  /** Router context object for the React Context Provider */\n  context: RoutingContextType<any>;\n};\n\n/**\n * Creates a complete router system from route configurations.\n *\n * This router is built from the same primitives as react-router but with additional\n * features for data preloading and code splitting. Each route can contain both a\n * Component and a prepare() function that can preload data for the component.\n *\n * The router watches for changes to the current location via the HTML5 History API,\n * maps the location to the corresponding route entry, and then preloads the code\n * and data for the route before rendering.\n *\n * @param routes - Array of route configurations\n * @returns Object containing the router context and cleanup function\n *\n  preparedMatch: prepareMatch(route, parseRawQuery(history.location.search)),\n * ```typescript\n * const routes = [\n *   {\n *     path: '/users/:id',\n *     resourcePage: getResourcePage('UserProfile', () => import('./UserProfile')),\n *     prepare: ({ variables }) => ({ userId: variables.id })\n *   }\n * ];\n *\n * const { context, cleanup } = createRouter(routes);\n *\n * // Use in React app\n * <RoutingContext.Provider value={context}>\n *   <RouterRenderer />\n * </RoutingContext.Provider>\n * ```\n */\nexport default function createRouter(\n  routes: Route<any, any>[],\n  opts: {\n    devtools?:\n      | boolean\n      | { global?: boolean; panel?: boolean; shortcut?: string };\n    /** When true, consecutive navigate() calls in same microtask auto‑coalesce (unless immediate: true). */\n    autoBatch?: boolean;\n  } = {},\n): CreateRouterReturn {\n  // Normalize devtools options maintaining backwards compatibility\n  let devtoolsGlobal: boolean;\n  let devtoolsPanel = false;\n  let devtoolsShortcut = 'Alt+Shift+R';\n  if (typeof opts.devtools === 'object') {\n    const baseEnabled = (() => {\n      try {\n        return (\n          typeof process !== 'undefined' &&\n          process.env.NODE_ENV !== 'production'\n        );\n      } catch {\n        return false;\n      }\n    })();\n    devtoolsGlobal = opts.devtools.global ?? baseEnabled;\n    devtoolsPanel = !!opts.devtools.panel;\n    if (\n      typeof opts.devtools.shortcut === 'string' &&\n      opts.devtools.shortcut.trim() !== ''\n    ) {\n      devtoolsShortcut = opts.devtools.shortcut;\n    }\n  } else {\n    // boolean or undefined\n    const enabled = (() => {\n      if (typeof opts.devtools === 'boolean') return opts.devtools;\n      try {\n        return (\n          typeof process !== 'undefined' &&\n          process.env.NODE_ENV !== 'production'\n        );\n      } catch {\n        return false;\n      }\n    })();\n    devtoolsGlobal = enabled;\n  }\n  // Initialize browser history manager\n  const history = new BrowserHistory();\n\n  // Build a flat list of routes for efficient matching\n  const flatRoutes = buildRoutes(routes);\n\n  // Find the initial route match and prepare it for rendering\n  const route = getMatchedRoute(flatRoutes, history.location);\n  const initialRawQuery = parseRawQuery(history.location.search);\n  // Determine schema from deepest route for initial load\n  let initialSchema: any;\n  // Determine merged filter schema from root->deepest (deepest overrides per field)\n  // (Future) merged route-level filter schema for initial load (not yet consumed in initial parsing)\n  // const initialFilterSchema: { __defined: true; schema: Record<string, any> } | undefined = undefined;\n  if (route != null) {\n    const collected: any[] = [];\n    const collectedFilters: { __defined: true; schema: Record<string, any> }[] =\n      [];\n    for (const r of route.route.routes) {\n      const maybe: unknown = (r as unknown as { query?: unknown }).query;\n      if (maybe != null) {\n        collected.push(maybe);\n      }\n      const maybeFilter: unknown = (\n        r as unknown as {\n          filterSchema?: { __defined: true; schema: Record<string, any> };\n        }\n      ).filterSchema;\n      if (maybeFilter != null) {\n        collectedFilters.push(\n          maybeFilter as { __defined: true; schema: Record<string, any> },\n        );\n      }\n    }\n    if (collected.length > 0) {\n      initialSchema = collected.at(-1);\n    }\n    // Merged filter schema for initial route would be built here if needed later.\n  }\n  const initialTypedQuery = parseTypedQuery(initialSchema, initialRawQuery);\n  const preparedMatch = prepareMatch(route, initialTypedQuery);\n  // Helper to build the raw query object from a search string\n  let currentEntry: RouteEntry<any> = {\n    forceRerender: false,\n    location: history.location,\n    route,\n    preparedMatch,\n    rawSearch: history.location.search,\n    query: initialRawQuery,\n    typedQuery: initialTypedQuery,\n  };\n\n  // Initial normalization pass (e.g., clamp page)\n  const initTyped = currentEntry.typedQuery as unknown as\n    | Record<string, unknown>\n    | undefined;\n  let initPage: unknown;\n  if (initTyped != null) {\n    initPage = initTyped.page;\n  }\n  if (\n    typeof initPage === 'number' &&\n    initPage < 1 &&\n    initialSchema != null &&\n    initTyped != null\n  ) {\n    const norm = { ...initTyped, page: 1 } as Record<string, unknown>;\n    const normalizedSearch = buildSearch(norm, initialSchema);\n    if (normalizedSearch !== history.location.search) {\n      history.set({\n        pathname: history.location.pathname,\n        search: normalizedSearch,\n        hash: '',\n      });\n    }\n  }\n\n  // Maintain a set of subscribers to the active route entry\n  let nextId = 0;\n  const subscribers = new Map<number, SubscribeCallback<any>>();\n\n  // Listen for location changes, match to the route entry, prepare the entry,\n  // and notify subscribers. This pattern ensures that data-loading\n  // occurs *outside* of - and *before* - rendering.\n  const cleanup = history.subscribe((location, forceRerender) => {\n    const samePathname = location.pathname === currentEntry.location.pathname;\n    const sameSearch = location.search === currentEntry.rawSearch;\n\n    if (!forceRerender && samePathname && sameSearch) {\n      // Nothing changed that we care about\n      return;\n    }\n\n    // If only the search changed we still want to propagate the change.\n    // Keep the existing preparedMatch when pathname is identical to avoid redundant work.\n    let nextPreparedMatch = currentEntry.preparedMatch;\n    let nextRoute = currentEntry.route;\n\n    if (!samePathname) {\n      // Path changed: recompute match + prepared data including query\n      nextRoute = getMatchedRoute(flatRoutes, history.location);\n      // we will set below after computing typed query\n    }\n\n    // Build raw query object (basic aggregation) from location.search\n    const query = parseRawQuery(location.search);\n    // Determine schema from deepest matched route\n    let schema: any;\n    if (nextRoute != null) {\n      const collected: any[] = [];\n      const collectedFilters: {\n        __defined: true;\n        schema: Record<string, any>;\n      }[] = [];\n      for (const r of nextRoute.route.routes) {\n        const maybe: unknown = (r as unknown as { query?: unknown }).query;\n        if (maybe != null) {\n          collected.push(maybe);\n        }\n        const maybeFilter: unknown = (\n          r as unknown as {\n            filterSchema?: { __defined: true; schema: Record<string, any> };\n          }\n        ).filterSchema;\n        if (maybeFilter != null) {\n          collectedFilters.push(\n            maybeFilter as { __defined: true; schema: Record<string, any> },\n          );\n        }\n      }\n      if (collected.length > 0) {\n        schema = collected.at(-1);\n      }\n      // Merged filter schema for next route would be computed here when integrated.\n    }\n    let typedQuery = parseTypedQuery(schema, query);\n    // Normalization: clamp page >= 1 if numeric page present\n    let normalized = false;\n    const pg = (typedQuery as Record<string, any>).page;\n    if (typeof pg === 'number' && pg < 1) {\n      const clone = { ...(typedQuery as Record<string, any>), page: 1 };\n      typedQuery = clone;\n      normalized = true;\n    }\n\n    // If only the search changed (same pathname) we still need to re-run prepare\n    if (!samePathname) {\n      nextPreparedMatch = prepareMatch(nextRoute, typedQuery);\n    } else if (!sameSearch) {\n      nextPreparedMatch = prepareMatch(nextRoute, typedQuery);\n    }\n\n    const nextEntry: RouteEntry<any> = {\n      forceRerender: forceRerender || (samePathname && !sameSearch),\n      location,\n      route: nextRoute,\n      preparedMatch: nextPreparedMatch,\n      rawSearch: location.search,\n      query,\n      typedQuery,\n    };\n\n    // If normalization changed the typed query we trigger a replace with normalized search\n    if (normalized && schema != null) {\n      const normalizedSearch = buildSearch(typedQuery as any, schema);\n      if (normalizedSearch !== location.search) {\n        history.set({\n          pathname: location.pathname,\n          search: normalizedSearch,\n          hash: '',\n        });\n        return; // early: subsequent set will trigger rerun\n      }\n    }\n\n    // Update current entry and notify all subscribers\n    currentEntry = nextEntry;\n    subscribers.forEach((callback) => {\n      callback(nextEntry);\n    });\n  });\n\n  // The router context object that will be passed to React Context\n  // --- Optional batching support (opt-in via navigate({ batch: true })) -----\n  type PendingNav = {\n    pathname?: string;\n    query?: Record<string, any>;\n    replace?: boolean;\n    filters?: Record<string, any>;\n    filterSchema?: { __defined: true; schema: Record<string, any> };\n    /** When true, this navigation explicitly keeps prior query state (no reset) */\n    inherit?: boolean;\n  };\n  let pendingNavs: PendingNav[] | undefined;\n  let flushScheduled = false;\n  // Rolling merged query for current batch (last value per key wins)\n  let batchedMergedQuery: Record<string, any> | undefined;\n  // Keep reference of last provided query object (for diagnostics / fallback)\n  let lastBatchQuery: Record<string, any> | undefined;\n  // Last provided filters + schema in batch\n  let batchedFilters: Record<string, any> | undefined;\n  let batchedFilterSchema:\n    | { __defined: true; schema: Record<string, any> }\n    | undefined;\n  // ---- Devtools navigation event callback (set by dynamic import) ----\n  let devtoolsOnNavEvent:\n    | ((meta: {\n        kind: string; // navigate | batch-flush\n        mode: string; // immediate | manual | auto | mixed\n        count: number; // number of merged navs (1 for immediate)\n        pathname: string;\n        search: string;\n        timestamp: number;\n      }) => void)\n    | undefined;\n  // Track current batch mode to report in timeline\n  let batchMode: 'manual' | 'auto' | 'mixed' | undefined;\n  /** Collect deepest query schema including the current route node and its descendants */\n  /** Collect route chain (root->deepest) for a pathname */\n  function collectRouteChain(pathname: string): any[] | undefined {\n    if (pathname === currentEntry.location.pathname) {\n      if (currentEntry.route != null) {\n        return currentEntry.route.route.routes as any[];\n      }\n      return undefined;\n    }\n    const destRoute = getMatchedRoute(flatRoutes, {\n      ...window.location,\n      pathname,\n    });\n    if (destRoute != null) return destRoute.route.routes as any[];\n    return undefined;\n  }\n  /** Merge query descriptors from a route chain (deeper overrides) */\n  function mergeQuerySchemaFromChain(chain: any[] | undefined): any {\n    if (chain == null) return undefined;\n    const merged: Record<string, any> = {};\n    for (const node of chain) {\n      const qobj = (node as { query?: Record<string, any> }).query;\n      if (qobj != null) {\n        for (const [k, v] of Object.entries(qobj)) {\n          merged[k] = v; // deeper overrides\n        }\n      }\n    }\n    if (Object.keys(merged).length === 0) return undefined;\n    return merged;\n  }\n  /** Merge filter schema objects (shallow field override) from a route chain */\n  function mergeFilterSchemaFromChain(\n    chain: any[] | undefined,\n  ): { __defined: true; schema: Record<string, any> } | undefined {\n    if (chain == null) return undefined;\n    let merged: Record<string, any> | undefined;\n    for (const node of chain) {\n      const fs = (\n        node as {\n          filterSchema?: { __defined: true; schema: Record<string, any> };\n        }\n      ).filterSchema;\n      if (fs != null) {\n        merged ??= {};\n        Object.assign(merged, fs.schema);\n      }\n    }\n    if (merged != null) return { __defined: true, schema: merged };\n    return undefined;\n  }\n  /** Resolve merged query schema for a pathname */\n  function resolveSchema(pathname: string): any {\n    return mergeQuerySchemaFromChain(collectRouteChain(pathname));\n  }\n  /** Resolve merged filter schema for a pathname */\n  function resolveFilterSchema(\n    pathname: string,\n  ): { __defined: true; schema: Record<string, any> } | undefined {\n    return mergeFilterSchemaFromChain(collectRouteChain(pathname));\n  }\n  /** Internal: build final search & next entry from pending batch without side-effects */\n  function coalesceNavigations(list: PendingNav[]): {\n    search: string;\n    pathname: string;\n    replace?: boolean;\n    nextEntry: RouteEntry<any>;\n  } {\n    // Determine final pathname & replace flag\n    let targetPathname = currentEntry.location.pathname;\n    if (targetPathname === '') {\n      targetPathname = '/';\n    }\n    for (const nav of list) {\n      if (nav.pathname !== undefined) {\n        const pn = nav.pathname;\n        if (pn === '') {\n          targetPathname = '/';\n        } else {\n          targetPathname = pn;\n        }\n      }\n    }\n    let replaceFlag: boolean | undefined;\n    for (const nav of list) {\n      if (nav.replace !== undefined) {\n        replaceFlag = nav.replace;\n      }\n    }\n    // Base query: prefer accumulated merged; else last explicit; else scan list; else current typed\n    let baseQuery: Record<string, any> =\n      currentEntry.typedQuery ?? currentEntry.query ?? {};\n    if (batchedMergedQuery != null) {\n      baseQuery = batchedMergedQuery;\n    } else if (lastBatchQuery != null) {\n      baseQuery = lastBatchQuery;\n    } else {\n      for (let i = list.length - 1; i >= 0; i -= 1) {\n        const item = list[i];\n        if (item != null) {\n          const q = item.query;\n          if (q != null && Object.keys(q).length > 0) {\n            baseQuery = q;\n            break;\n          }\n        }\n      }\n    }\n    // Filters\n    const batchFilters = batchedFilters;\n    const batchFilterSchema = batchedFilterSchema;\n    const schema = resolveSchema(targetPathname);\n    let buildOpts:\n      | {\n          filters: Record<string, any>;\n          filterSchema: { __defined: true; schema: Record<string, any> };\n        }\n      | undefined;\n    if (batchFilters !== undefined) {\n      const fs = batchFilterSchema ?? resolveFilterSchema(targetPathname);\n      if (fs != null) {\n        buildOpts = { filters: batchFilters, filterSchema: fs };\n      }\n    }\n    const search = buildSearch(baseQuery, schema, buildOpts);\n    // Compute next entry (optimistic)\n    let normalizedPath = targetPathname;\n    if (normalizedPath === '') {\n      normalizedPath = '/';\n    }\n    let nextRoute = currentEntry.route;\n    if (normalizedPath !== currentEntry.location.pathname) {\n      nextRoute = getMatchedRoute(flatRoutes, {\n        ...window.location,\n        pathname: normalizedPath,\n      });\n    }\n    const effectiveSchema = schema ?? resolveSchema(normalizedPath);\n    const rawQueryObj = parseRawQuery(search);\n    const typed = parseTypedQuery(effectiveSchema, rawQueryObj);\n    const prepared = prepareMatch(nextRoute, typed);\n    const nextEntry: RouteEntry<any> = {\n      forceRerender: false,\n      location: { ...currentEntry.location, pathname: normalizedPath, search },\n      route: nextRoute,\n      preparedMatch: prepared,\n      rawSearch: search,\n      query: rawQueryObj,\n      typedQuery: typed,\n    } as RouteEntry<any>;\n    return {\n      search,\n      pathname: normalizedPath,\n      replace: replaceFlag,\n      nextEntry,\n    };\n  }\n  /** Coalesce queued navigations into a single history update */\n  function flushNavigations(): void {\n    flushScheduled = false;\n    const list = pendingNavs;\n    pendingNavs = undefined;\n    if (list == null || list.length === 0) {\n      return;\n    }\n    const {\n      search,\n      pathname,\n      replace: replaceFlag,\n      nextEntry,\n    } = coalesceNavigations(list);\n    // Reset batch accumulators AFTER deriving result\n    batchedMergedQuery = undefined;\n    lastBatchQuery = undefined;\n    batchedFilters = undefined;\n    batchedFilterSchema = undefined;\n    currentEntry = nextEntry;\n    const loc = { pathname, search, hash: '' };\n    if (replaceFlag) history.set(loc);\n    else history.push(loc);\n    // Timeline event for batch flush\n    let mode = 'manual';\n    if (batchMode === 'auto') {\n      mode = 'auto';\n    } else if (batchMode === 'mixed') {\n      mode = 'mixed';\n    }\n    devtoolsOnNavEvent?.({\n      kind: 'batch-flush',\n      mode,\n      count: list.length,\n      pathname,\n      search,\n      timestamp: Date.now(),\n    });\n    batchMode = undefined;\n  }\n  // After resolveFilterSchema is defined we can apply implicit schema for any queued batch (if still pending)\n  /** Apply provisional merged query+filters update (during batching)\n   * to currentEntry for synchronous reads.\n   */\n  function applyProvisionalBatchUpdate(pathname?: string): void {\n    try {\n      const basePath = pathname ?? currentEntry.location.pathname;\n      const schema = resolveSchema(basePath);\n      const mergedQ = batchedMergedQuery ?? lastBatchQuery ?? {};\n      let buildOpts:\n        | {\n            filters: Record<string, any>;\n            filterSchema: { __defined: true; schema: Record<string, any> };\n          }\n        | undefined;\n      if (batchedFilters != null && batchedFilterSchema != null) {\n        buildOpts = {\n          filters: batchedFilters,\n          filterSchema: batchedFilterSchema,\n        };\n      } else if (batchedFilters != null && batchedFilterSchema == null) {\n        // Implicit resolution for provisional view\n        let merged: Record<string, any> | undefined;\n        let routeChain:\n          | {\n              filterSchema?: { __defined: true; schema: Record<string, any> };\n            }[]\n          | undefined;\n        if (basePath === currentEntry.location.pathname) {\n          if (currentEntry.route != null) {\n            routeChain = currentEntry.route.route.routes as any;\n          }\n        } else {\n          const destRoute = getMatchedRoute(flatRoutes, {\n            ...window.location,\n            pathname: basePath,\n          });\n          if (destRoute != null) {\n            routeChain = destRoute.route.routes as any;\n          }\n        }\n        if (routeChain != null) {\n          for (const node of routeChain) {\n            const fs = (\n              node as {\n                filterSchema?: { __defined: true; schema: Record<string, any> };\n              }\n            ).filterSchema;\n            if (fs != null) {\n              merged ??= {};\n              Object.assign(merged, fs.schema);\n            }\n          }\n        }\n        if (merged != null) {\n          buildOpts = {\n            filters: batchedFilters,\n            filterSchema: { __defined: true, schema: merged },\n          };\n        }\n      }\n      const provisionalSearch = buildSearch(mergedQ, schema, buildOpts);\n      const pathChanged = basePath !== currentEntry.location.pathname;\n      if (provisionalSearch !== currentEntry.rawSearch || pathChanged) {\n        const typed = parseTypedQuery(schema, mergedQ);\n        const rawQueryObj = parseRawQuery(provisionalSearch);\n        currentEntry = {\n          ...currentEntry,\n          location: {\n            ...currentEntry.location,\n            pathname: basePath,\n            search: provisionalSearch,\n          },\n          rawSearch: provisionalSearch,\n          query: rawQueryObj,\n          typedQuery: typed,\n        } as RouteEntry<any>;\n      }\n    } catch {\n      /* ignore */\n    }\n  }\n\n  const autoBatch = opts.autoBatch === true;\n  const context: RoutingContextType<any> = {\n    history,\n    get() {\n      return currentEntry;\n    },\n    resolveFilterSchema(pathname: string) {\n      return resolveFilterSchema(pathname);\n    },\n    currentMergedFilterSchema() {\n      return resolveFilterSchema(currentEntry.location.pathname);\n    },\n    preloadCode(pathname: string) {\n      const matches = getMatchedRoute(flatRoutes, {\n        ...window.location,\n        pathname,\n      });\n      if (matches == null) return;\n      matches.route.routes.forEach(({ resourcePage }) => {\n        if (resourcePage == null) return;\n        // eslint-disable-next-line @typescript-eslint/no-floating-promises\n        resourcePage.load();\n      });\n    },\n    preload(pathname: string) {\n      const matches = getMatchedRoute(flatRoutes, {\n        ...window.location,\n        pathname,\n      });\n      prepareMatch(matches);\n    },\n    subscribe(callback: SubscribeCallback<any>) {\n      nextId += 1;\n      const id = nextId;\n      // eslint-disable-next-line jsdoc/require-jsdoc\n      function dispose(): void {\n        subscribers.delete(id);\n      }\n      subscribers.set(id, callback);\n      return dispose;\n    },\n    navigate({\n      pathname,\n      query,\n      replace,\n      filters,\n      filterSchema,\n      batch,\n      immediate,\n      inherit,\n    }: {\n      pathname?: string;\n      query?: Record<string, any>;\n      replace?: boolean;\n      filters?: Record<string, any>;\n      filterSchema?: { __defined: true; schema: Record<string, any> };\n      batch?: boolean;\n      immediate?: boolean;\n      inherit?: boolean;\n    }): void {\n      try {\n        const effectiveInherit = inherit === true;\n        const wantsBatch =\n          batch === true ||\n          (autoBatch && immediate !== true && batch !== false);\n        if (wantsBatch) {\n          const nav: PendingNav = {};\n          if (pathname !== undefined) nav.pathname = pathname;\n          if (query !== undefined) nav.query = query;\n          if (replace !== undefined) nav.replace = replace;\n          if (filters !== undefined) nav.filters = filters;\n          if (filterSchema !== undefined) nav.filterSchema = filterSchema;\n          if (effectiveInherit) nav.inherit = true;\n          pendingNavs ??= [];\n          pendingNavs.push(nav);\n          let thisMode: 'manual' | 'auto';\n          if (batch === true) thisMode = 'manual';\n          else thisMode = 'auto';\n          if (batchMode == null) batchMode = thisMode;\n          else if (batchMode !== thisMode) batchMode = 'mixed';\n          if (query != null) {\n            if (effectiveInherit) {\n              const base =\n                batchedMergedQuery ??\n                lastBatchQuery ??\n                currentEntry.typedQuery ??\n                currentEntry.query ??\n                {};\n              batchedMergedQuery = { ...base, ...query };\n            } else {\n              batchedMergedQuery = { ...query };\n            }\n            lastBatchQuery = query;\n          }\n          if (filters != null) {\n            batchedFilters = filters;\n            if (filterSchema != null) batchedFilterSchema = filterSchema;\n          }\n          if (!flushScheduled) {\n            flushScheduled = true;\n            // eslint-disable-next-line @typescript-eslint/no-floating-promises\n            Promise.resolve().then(flushNavigations);\n          }\n          applyProvisionalBatchUpdate(pathname);\n          subscribers.forEach((cb) => {\n            cb(currentEntry);\n          });\n          return;\n        }\n        // Immediate navigation\n        let base: Record<string, any> | undefined;\n        if (effectiveInherit) {\n          base = { ...(currentEntry.typedQuery ?? currentEntry.query ?? {}) };\n        }\n        if (query != null) {\n          if (base != null) Object.assign(base, query);\n          else base = query;\n        }\n        base ??= currentEntry.typedQuery ?? currentEntry.query ?? {};\n        const targetPath = pathname ?? currentEntry.location.pathname;\n        const schema = resolveSchema(targetPath);\n        let buildOpts:\n          | {\n              filters: Record<string, any>;\n              filterSchema: { __defined: true; schema: Record<string, any> };\n            }\n          | undefined;\n        if (filters != null) {\n          const fs = filterSchema ?? resolveFilterSchema(targetPath);\n          if (fs != null) buildOpts = { filters, filterSchema: fs };\n        }\n        const search = buildSearch(base, schema, buildOpts);\n        let targetPathname = targetPath;\n        if (targetPathname === '') targetPathname = '/';\n        let nextRoute = currentEntry.route;\n        if (targetPathname !== currentEntry.location.pathname) {\n          nextRoute = getMatchedRoute(flatRoutes, {\n            ...window.location,\n            pathname: targetPathname,\n          });\n        }\n        const effectiveSchema = schema ?? resolveSchema(targetPathname);\n        const rawQueryObj = parseRawQuery(search);\n        const typed = parseTypedQuery(effectiveSchema, rawQueryObj);\n        const prepared = prepareMatch(nextRoute, typed);\n        currentEntry = {\n          forceRerender: false,\n          location: {\n            ...currentEntry.location,\n            pathname: targetPathname,\n            search,\n          },\n          route: nextRoute,\n          preparedMatch: prepared,\n          rawSearch: search,\n          query: rawQueryObj,\n          typedQuery: typed,\n        } as RouteEntry<any>;\n        let normImmediate = targetPathname;\n        if (normImmediate === '') normImmediate = '/';\n        const locationObj = { pathname: normImmediate, search, hash: '' };\n        if (replace) history.set(locationObj);\n        else history.push(locationObj);\n        devtoolsOnNavEvent?.({\n          kind: 'navigate',\n          mode: 'immediate',\n          count: 1,\n          pathname: normImmediate,\n          search,\n          timestamp: Date.now(),\n        });\n      } catch {\n        /* ignore */\n      }\n    },\n  };\n\n  // Dynamic devtools load (dev only) - separate chunk for tree-shaking\n  if ((devtoolsGlobal || devtoolsPanel) && typeof window !== 'undefined') {\n    // eslint-disable-next-line @typescript-eslint/no-floating-promises\n    (async () => {\n      try {\n        const mod: any = await import('./devtools.js');\n        // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n        if (mod != null && typeof mod.initRouterDevtools === 'function') {\n          // eslint-disable-next-line @stylistic-eslint/max-len\n          // eslint-disable-next-line @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access\n          devtoolsOnNavEvent = mod.initRouterDevtools({\n            context,\n            options: {\n              global: devtoolsGlobal,\n              panel: devtoolsPanel,\n              shortcut: devtoolsShortcut,\n            },\n          });\n        }\n      } catch {\n        /* ignore */\n      }\n    })();\n  }\n\n  return { cleanup, context };\n}\n"]}
|
|
633
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"createRouter.js","sourceRoot":"","sources":["../../../src/routing/createRouter.ts"],"names":[],"mappings":"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","sourcesContent":["import { buildRoutes } from '../builder.js';\nimport { BrowserHistory } from '../history/index.js';\nimport { getMatchedRoute, prepareMatch } from '../tools.js';\nimport { parseTypedQuery } from '../tools/query-dsl.js';\nimport { parseRawQuery } from '../tools/query.js';\nimport { buildSearch } from '../tools/buildSearch.js';\nimport buildCombinedSearch from '../tools/buildCombinedSearch.js';\nimport { parse as parseFilters } from '@plumile/filter-query';\nimport type {\n  Route,\n  RouteEntry,\n  RoutingContextType,\n  SubscribeCallback,\n} from '../types.js';\n\n/**\n * Return type for the createRouter function.\n */\nexport type CreateRouterReturn = {\n  /** Function to clean up router listeners and resources */\n  cleanup: () => void;\n  /** Router context object for the React Context Provider */\n  context: RoutingContextType<any>;\n};\n\n/**\n * Creates a complete router system from route configurations.\n *\n * This router is built from the same primitives as react-router but with additional\n * features for data preloading and code splitting. Each route can contain both a\n * Component and a prepare() function that can preload data for the component.\n *\n * The router watches for changes to the current location via the HTML5 History API,\n * maps the location to the corresponding route entry, and then preloads the code\n * and data for the route before rendering.\n *\n * @param routes - Array of route configurations\n * @returns Object containing the router context and cleanup function\n *\n  preparedMatch: prepareMatch(route, parseRawQuery(history.location.search)),\n * ```typescript\n * const routes = [\n *   {\n *     path: '/users/:id',\n *     resourcePage: getResourcePage('UserProfile', () => import('./UserProfile')),\n *     prepare: ({ variables }) => ({ userId: variables.id })\n *   }\n * ];\n *\n * const { context, cleanup } = createRouter(routes);\n *\n * // Use in React app\n * <RoutingContext.Provider value={context}>\n *   <RouterRenderer />\n * </RoutingContext.Provider>\n * ```\n */\nexport default function createRouter(\n  routes: Route<any, any>[],\n  opts: {\n    devtools?:\n      | boolean\n      | { global?: boolean; panel?: boolean; shortcut?: string };\n  } = {},\n): CreateRouterReturn {\n  // Normalize devtools options maintaining backwards compatibility\n  let devtoolsGlobal: boolean;\n  let devtoolsPanel = false;\n  let devtoolsShortcut = 'Alt+Shift+R';\n  if (typeof opts.devtools === 'object') {\n    const baseEnabled = (() => {\n      try {\n        return (\n          typeof process !== 'undefined' &&\n          process.env.NODE_ENV !== 'production'\n        );\n      } catch {\n        return false;\n      }\n    })();\n    devtoolsGlobal = opts.devtools.global ?? baseEnabled;\n    devtoolsPanel = !!opts.devtools.panel;\n    if (\n      typeof opts.devtools.shortcut === 'string' &&\n      opts.devtools.shortcut.trim() !== ''\n    ) {\n      devtoolsShortcut = opts.devtools.shortcut;\n    }\n  } else {\n    // boolean or undefined\n    const enabled = (() => {\n      if (typeof opts.devtools === 'boolean') return opts.devtools;\n      try {\n        return (\n          typeof process !== 'undefined' &&\n          process.env.NODE_ENV !== 'production'\n        );\n      } catch {\n        return false;\n      }\n    })();\n    devtoolsGlobal = enabled;\n  }\n  // Initialize browser history manager\n  const history = new BrowserHistory();\n\n  // Build a flat list of routes for efficient matching\n  const flatRoutes = buildRoutes(routes);\n\n  // Find the initial route match and prepare it for rendering\n  const route = getMatchedRoute(flatRoutes, history.location);\n  const initialRawQuery = parseRawQuery(history.location.search);\n  // Determine schema from deepest route for initial load\n  let initialSchema: any;\n  if (route != null) {\n    const collected: any[] = [];\n    for (const r of route.route.routes) {\n      const maybe: unknown = (r as unknown as { query?: unknown }).query;\n      if (maybe != null) {\n        collected.push(maybe);\n      }\n    }\n    if (collected.length > 0) {\n      initialSchema = collected.at(-1);\n    }\n  }\n  const initialTypedQuery = parseTypedQuery(initialSchema, initialRawQuery);\n  // Determine filter schema from deepest route\n  let initialFilterSchema: any;\n  if (route != null) {\n    const collected: any[] = [];\n    for (const r of route.route.routes) {\n      const maybe: unknown = (r as unknown as { filterSchema?: unknown })\n        .filterSchema;\n      if (maybe != null) collected.push(maybe);\n    }\n    if (collected.length > 0) initialFilterSchema = collected.at(-1);\n  }\n  let initialFilters: Record<string, unknown> | undefined;\n  let initialFilterDiagnostics: any[] | undefined;\n  if (initialFilterSchema != null) {\n    const parsed = parseFilters(history.location.search, initialFilterSchema);\n    initialFilters = parsed.filters as unknown as Record<string, unknown>;\n    initialFilterDiagnostics = parsed.diagnostics as unknown[];\n  }\n  const preparedMatch = prepareMatch(route, initialTypedQuery);\n  // Helper to build the raw query object from a search string\n  let currentEntry: RouteEntry<any> = {\n    forceRerender: false,\n    location: history.location,\n    route,\n    preparedMatch,\n    rawSearch: history.location.search,\n    query: initialRawQuery,\n    typedQuery: initialTypedQuery,\n    filters: initialFilters,\n    filterDiagnostics: initialFilterDiagnostics,\n  };\n\n  // Initial normalization pass (e.g., clamp page)\n  const initTyped = currentEntry.typedQuery as unknown as\n    | Record<string, unknown>\n    | undefined;\n  let initPage: unknown;\n  if (initTyped != null) {\n    initPage = initTyped.page;\n  }\n  if (\n    typeof initPage === 'number' &&\n    initPage < 1 &&\n    initialSchema != null &&\n    initTyped != null\n  ) {\n    const norm = { ...initTyped, page: 1 } as Record<string, unknown>;\n    const normalizedSearch = buildSearch(norm, initialSchema);\n    if (normalizedSearch !== history.location.search) {\n      history.set({\n        pathname: history.location.pathname,\n        search: normalizedSearch,\n        hash: '',\n      });\n    }\n  }\n\n  // Maintain a set of subscribers to the active route entry\n  let nextId = 0;\n  const subscribers = new Map<number, SubscribeCallback<any>>();\n\n  // Listen for location changes, match to the route entry, prepare the entry,\n  // and notify subscribers. This pattern ensures that data-loading\n  // occurs *outside* of - and *before* - rendering.\n  const cleanup = history.subscribe((location, forceRerender) => {\n    const samePathname = location.pathname === currentEntry.location.pathname;\n    const sameSearch = location.search === currentEntry.rawSearch;\n\n    if (!forceRerender && samePathname && sameSearch) {\n      // Nothing changed that we care about\n      return;\n    }\n\n    // If only the search changed we still want to propagate the change.\n    // Keep the existing preparedMatch when pathname is identical to avoid redundant work.\n    let nextPreparedMatch = currentEntry.preparedMatch;\n    let nextRoute = currentEntry.route;\n\n    if (!samePathname) {\n      // Path changed: recompute match + prepared data including query\n      nextRoute = getMatchedRoute(flatRoutes, history.location);\n      // we will set below after computing typed query\n    }\n\n    // Build raw query object (basic aggregation) from location.search\n    const query = parseRawQuery(location.search);\n    // Determine schema from deepest matched route\n    let schema: any;\n    let filterSchema: any;\n    if (nextRoute != null) {\n      const collected: any[] = [];\n      const filterCollected: any[] = [];\n      for (const r of nextRoute.route.routes) {\n        const maybe: unknown = (r as unknown as { query?: unknown }).query;\n        if (maybe != null) {\n          collected.push(maybe);\n        }\n        const maybeF: unknown = (r as unknown as { filterSchema?: unknown })\n          .filterSchema;\n        if (maybeF != null) filterCollected.push(maybeF);\n      }\n      if (collected.length > 0) {\n        schema = collected.at(-1);\n      }\n      if (filterCollected.length > 0) {\n        filterSchema = filterCollected.at(-1);\n      }\n    }\n    let typedQuery = parseTypedQuery(schema, query);\n    // Parse filters using filter schema if present\n    let filters: Record<string, unknown> | undefined;\n    let filterDiagnostics: any[] | undefined;\n    if (filterSchema != null) {\n      const parsedF = parseFilters(location.search, filterSchema);\n      filters = parsedF.filters as unknown as Record<string, unknown>;\n      filterDiagnostics = parsedF.diagnostics as unknown[];\n    }\n    // Normalization: clamp page >= 1 if numeric page present\n    let normalized = false;\n    const pg = (typedQuery as Record<string, any>).page;\n    if (typeof pg === 'number' && pg < 1) {\n      const clone = { ...(typedQuery as Record<string, any>), page: 1 };\n      typedQuery = clone;\n      normalized = true;\n    }\n\n    // If only the search changed (same pathname) we still need to re-run prepare\n    if (!samePathname) {\n      nextPreparedMatch = prepareMatch(nextRoute, typedQuery);\n    } else if (!sameSearch) {\n      nextPreparedMatch = prepareMatch(nextRoute, typedQuery);\n    }\n\n    const nextEntry: RouteEntry<any> = {\n      forceRerender: forceRerender || (samePathname && !sameSearch),\n      location,\n      route: nextRoute,\n      preparedMatch: nextPreparedMatch,\n      rawSearch: location.search,\n      query,\n      typedQuery,\n      filters,\n      filterDiagnostics,\n    };\n\n    // If normalization changed the typed query we trigger a replace with normalized search\n    if (normalized && schema != null) {\n      const normalizedSearch = buildCombinedSearch({\n        query: typedQuery as any,\n        querySchema: schema,\n        filters,\n        filterSchema,\n      });\n      if (normalizedSearch !== location.search) {\n        history.set({\n          pathname: location.pathname,\n          search: normalizedSearch,\n          hash: '',\n        });\n        return; // early: subsequent set will trigger rerun\n      }\n    }\n\n    // Update current entry and notify all subscribers\n    currentEntry = nextEntry;\n    subscribers.forEach((callback) => {\n      callback(nextEntry);\n    });\n  });\n\n  // The router context object that will be passed to React Context\n  const context: RoutingContextType<any> = {\n    history,\n    get() {\n      return currentEntry;\n    },\n    preloadCode(pathname) {\n      // Preload just the component code for a route without storing the result\n      const matches = getMatchedRoute(flatRoutes, {\n        ...window.location,\n        pathname,\n      });\n\n      if (matches == null) {\n        return;\n      }\n\n      // Load all resource pages for the matched route\n      matches.route.routes.forEach(({ resourcePage }) => {\n        if (resourcePage == null) {\n          return;\n        }\n        // eslint-disable-next-line @typescript-eslint/no-floating-promises\n        resourcePage.load();\n      });\n    },\n    preload(pathname) {\n      // Preload both the component code and prepared data for a route\n      const matches = getMatchedRoute(flatRoutes, {\n        ...window.location,\n        pathname,\n      });\n      prepareMatch(matches);\n    },\n    subscribe(callback) {\n      // Add a new subscriber and return unsubscribe function\n      nextId += 1;\n      const id = nextId;\n      // eslint-disable-next-line func-style\n      const dispose = () => {\n        subscribers.delete(id);\n      };\n      subscribers.set(id, callback);\n      return dispose;\n    },\n    navigate({\n      pathname,\n      query,\n      filters: navFilters,\n      replace,\n    }: {\n      pathname?: string;\n      query?: Record<string, any>;\n      filters?: Record<string, any>;\n      replace?: boolean;\n    }) {\n      const current = currentEntry;\n      const targetPathname = pathname ?? current.location.pathname;\n      // Determine schema of destination (if same path and we have route we can reuse, else rematch)\n      let schema: any;\n      let filterSchema: any;\n      /** Collect deepest query schema from route chain */\n      function collectSchema(routesArr: any[] | undefined): void {\n        if (routesArr == null) {\n          return;\n        }\n        const collected: any[] = [];\n        const filterCollected: any[] = [];\n        for (const r of routesArr) {\n          const maybe: unknown = (r as unknown as { query?: unknown }).query;\n          if (maybe != null) {\n            collected.push(maybe);\n          }\n          const maybeF: unknown = (r as unknown as { filterSchema?: unknown })\n            .filterSchema;\n          if (maybeF != null) filterCollected.push(maybeF);\n        }\n        if (collected.length > 0) {\n          schema = collected.at(-1);\n        }\n        if (filterCollected.length > 0) {\n          filterSchema = filterCollected.at(-1);\n        }\n      }\n      if (pathname == null || pathname === current.location.pathname) {\n        collectSchema(current.route?.route.routes);\n      } else {\n        const destRoute = getMatchedRoute(flatRoutes, {\n          ...window.location,\n          pathname: targetPathname,\n        });\n        collectSchema(destRoute?.route.routes);\n      }\n      const effectiveFilters = navFilters ?? current.filters;\n      const search = buildCombinedSearch({\n        query,\n        querySchema: schema,\n        filters: effectiveFilters,\n        filterSchema,\n      });\n      const locationObj = { pathname: targetPathname, search, hash: '' };\n      if (replace) {\n        history.set(locationObj);\n      } else {\n        history.push(locationObj);\n      }\n    },\n  };\n\n  // Return both the context object and a cleanup function\n  // Devtools exposure (development only)\n  if (devtoolsGlobal) {\n    try {\n      if (typeof window !== 'undefined') {\n        (window as unknown as { __PLUMILE_ROUTER__?: any }).__PLUMILE_ROUTER__ =\n          {\n            get: context.get,\n            subscribe: context.subscribe,\n          };\n      }\n    } catch {\n      /* ignore */\n    }\n  }\n  // Optional in-page devtools panel (lightweight overlay)\n  if (devtoolsPanel) {\n    try {\n      if (\n        typeof window !== 'undefined' &&\n        typeof document !== 'undefined' &&\n        document.querySelector('[data-plumile-router-devtools-panel]') == null\n      ) {\n        const host = document.createElement('div');\n        host.setAttribute('data-plumile-router-devtools-panel', '');\n        host.style.position = 'fixed';\n        host.style.zIndex = '2147483646';\n        host.style.bottom = '8px';\n        host.style.right = '8px';\n        host.style.fontFamily =\n          'ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, \"Liberation Mono\", monospace';\n        host.style.fontSize = '12px';\n        host.style.lineHeight = '1.3';\n        host.style.color = '#111';\n        host.style.background = 'transparent';\n        host.style.pointerEvents = 'none';\n        const shadow = host.attachShadow({ mode: 'open' });\n        const panel = document.createElement('div');\n        panel.style.pointerEvents = 'auto';\n        panel.style.minWidth = '260px';\n        panel.style.maxWidth = '420px';\n        panel.style.maxHeight = '50vh';\n        panel.style.overflow = 'auto';\n        panel.style.background = 'rgba(255,255,255,0.95)';\n        panel.style.border = '1px solid #ccc';\n        panel.style.borderRadius = '6px';\n        panel.style.boxShadow = '0 2px 8px rgba(0,0,0,0.15)';\n        panel.style.padding = '6px 8px 8px';\n        panel.style.backdropFilter = 'blur(4px)';\n        panel.style.display = 'flex';\n        panel.style.flexDirection = 'column';\n        panel.style.rowGap = '4px';\n        const header = document.createElement('div');\n        header.style.display = 'flex';\n        header.style.alignItems = 'center';\n        header.style.justifyContent = 'space-between';\n        header.style.gap = '8px';\n        const title = document.createElement('span');\n        title.textContent = 'Plumile Router';\n        title.style.fontWeight = '600';\n        const buttons = document.createElement('div');\n        buttons.style.display = 'flex';\n        buttons.style.gap = '4px';\n        const collapseBtn = document.createElement('button');\n        collapseBtn.textContent = '−';\n        const closeBtn = document.createElement('button');\n        closeBtn.textContent = '×';\n        const btns: HTMLButtonElement[] = [collapseBtn, closeBtn];\n        for (const b of btns) {\n          b.style.cursor = 'pointer';\n          b.style.background = '#eee';\n          b.style.border = '1px solid #ccc';\n          b.style.borderRadius = '4px';\n          b.style.fontSize = '11px';\n          b.style.lineHeight = '1';\n          b.style.padding = '2px 5px';\n        }\n        header.appendChild(title);\n        buttons.appendChild(collapseBtn);\n        buttons.appendChild(closeBtn);\n        header.appendChild(buttons);\n        const pathEl = document.createElement('div');\n        pathEl.style.whiteSpace = 'nowrap';\n        const varsEl = document.createElement('pre');\n        const queryEl = document.createElement('pre');\n        const typedEl = document.createElement('pre');\n        const filtersEl = document.createElement('pre');\n        const filterDiagEl = document.createElement('pre');\n        // Timeline container\n        const timelineWrap = document.createElement('div');\n        timelineWrap.style.display = 'flex';\n        timelineWrap.style.flexDirection = 'column';\n        timelineWrap.style.gap = '2px';\n        timelineWrap.style.marginTop = '4px';\n        const timelineHeader = document.createElement('div');\n        timelineHeader.textContent = 'Timeline (query changes)';\n        timelineHeader.style.fontWeight = '600';\n        timelineHeader.style.fontSize = '11px';\n        const timeline = document.createElement('div');\n        timeline.style.display = 'flex';\n        timeline.style.flexDirection = 'column-reverse'; // newest on top visually while appending at end for cheaper ops\n        timeline.style.maxHeight = '140px';\n        timeline.style.overflowY = 'auto';\n        timeline.style.border = '1px solid #e2e2e2';\n        timeline.style.borderRadius = '4px';\n        timeline.style.background = '#fafbfc';\n        timeline.style.fontFamily = 'inherit';\n        timeline.style.fontSize = '11px';\n        timelineWrap.appendChild(timelineHeader);\n        timelineWrap.appendChild(timeline);\n        const pres: HTMLPreElement[] = [\n          varsEl,\n          queryEl,\n          typedEl,\n          filtersEl,\n          filterDiagEl,\n        ];\n        for (const p of pres) {\n          p.style.margin = '0';\n          p.style.padding = '4px';\n          p.style.background = '#f6f8fa';\n          p.style.border = '1px solid #e2e2e2';\n          p.style.borderRadius = '4px';\n          p.style.overflowX = 'auto';\n          p.style.maxHeight = '160px';\n        }\n        const small = document.createElement('div');\n        small.style.fontSize = '10px';\n        small.style.opacity = '0.7';\n        small.textContent = `Shortcut: ${devtoolsShortcut}`;\n        panel.appendChild(header);\n        panel.appendChild(pathEl);\n        panel.appendChild(varsEl);\n        panel.appendChild(queryEl);\n        panel.appendChild(typedEl);\n        panel.appendChild(timelineWrap);\n        panel.appendChild(filtersEl);\n        panel.appendChild(filterDiagEl);\n        panel.appendChild(small);\n        shadow.appendChild(panel);\n        document.body.appendChild(host);\n        let collapsed = false;\n        collapseBtn.addEventListener('click', () => {\n          collapsed = !collapsed;\n          if (collapsed) {\n            panel.style.maxHeight = '24px';\n            panel.style.overflow = 'hidden';\n            collapseBtn.textContent = '+';\n          } else {\n            panel.style.maxHeight = '50vh';\n            panel.style.overflow = 'auto';\n            collapseBtn.textContent = '−';\n          }\n        });\n        let unsub: (() => void) | undefined;\n        closeBtn.addEventListener('click', () => {\n          host.remove();\n          if (typeof unsub === 'function') {\n            unsub();\n          }\n        });\n        /** Update panel content from current route entry */\n        // Keep last query snapshot for diff\n        let lastQuery: Record<string, unknown> | undefined;\n        let lastSearch: string | undefined;\n        /**\n         * Compute diff classification between two query objects.\n         */\n        function diffQueries(\n          prev: Record<string, unknown> | undefined,\n          next: Record<string, unknown>,\n        ): {\n          added: string[];\n          removed: string[];\n          changed: string[];\n          unchanged: string[];\n        } {\n          const added: string[] = [];\n          const removed: string[] = [];\n          const changed: string[] = [];\n          const unchanged: string[] = [];\n          const prevKeys = new Set(Object.keys(prev ?? {}));\n          const nextKeys = new Set(Object.keys(next));\n          nextKeys.forEach((k) => {\n            if (!prevKeys.has(k)) {\n              added.push(k);\n              return;\n            }\n            let pv: unknown;\n            if (prev != null) {\n              // prev is a plain record; direct index (runtime safe)\n              // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n              pv = (prev as any)[k];\n            }\n            const nv = next[k];\n            let same = false;\n            if (Array.isArray(pv) && Array.isArray(nv)) {\n              if (pv.length === (nv as unknown[]).length) {\n                same = pv.every((v, i) => {\n                  return v === (nv as unknown[])[i];\n                });\n              }\n            } else {\n              same = pv === nv;\n            }\n            if (same) {\n              unchanged.push(k);\n            } else {\n              changed.push(k);\n            }\n            prevKeys.delete(k);\n          });\n          prevKeys.forEach((k) => {\n            removed.push(k);\n          });\n          return { added, removed, changed, unchanged };\n        }\n        /** Update panel content and maybe push timeline entry */\n        function render(): void {\n          try {\n            const entry = context.get();\n            pathEl.textContent = `${entry.location.pathname}${entry.location.search}`;\n            const varsJson = JSON.stringify(entry.route?.params ?? {}, null, 2);\n            varsEl.textContent = `vars: ${varsJson}`;\n            const rawJson = JSON.stringify(entry.query, null, 2);\n            queryEl.textContent = `raw:  ${rawJson}`;\n            const typedJson = JSON.stringify(entry.typedQuery, null, 2);\n            typedEl.textContent = `typed:${typedJson}`;\n            const filtersJson = JSON.stringify(entry.filters, null, 2);\n            filtersEl.textContent = `filters:${filtersJson}`;\n            const diag = entry.filterDiagnostics ?? [];\n            filterDiagEl.textContent = `filterDiagnostics: ${diag.length}`;\n            // Timeline log if query search changed\n            if (entry.rawSearch !== lastSearch) {\n              const nowQuery = entry.query as Record<string, unknown>;\n              const diff = diffQueries(lastQuery, nowQuery);\n              const row = document.createElement('div');\n              row.style.display = 'flex';\n              row.style.flexDirection = 'column';\n              row.style.borderBottom = '1px solid #e9e9e9';\n              row.style.padding = '3px 4px';\n              const head = document.createElement('div');\n              head.style.display = 'flex';\n              head.style.justifyContent = 'space-between';\n              const ts = new Date().toLocaleTimeString();\n              let searchDisplay = '(empty)';\n              if (entry.location.search !== '') {\n                searchDisplay = entry.location.search;\n              }\n              head.innerHTML = `<strong>${ts}</strong> <code style=\"background:#eee;padding:1px 3px;border-radius:3px;\">${searchDisplay}</code>`;\n              const body = document.createElement('div');\n              body.style.display = 'flex';\n              body.style.flexWrap = 'wrap';\n              body.style.gap = '4px';\n              /** Create a colored pill element */\n              function pill(\n                label: string,\n                kind: 'added' | 'removed' | 'changed' | 'unchanged',\n              ): HTMLSpanElement {\n                const span = document.createElement('span');\n                span.textContent = label;\n                span.style.padding = '1px 4px';\n                span.style.borderRadius = '10px';\n                span.style.fontSize = '10px';\n                span.style.border = '1px solid';\n                switch (kind) {\n                  case 'added':\n                    span.style.background = '#e6ffed';\n                    span.style.borderColor = '#2cbe4e';\n                    span.style.color = '#22863a';\n                    break;\n                  case 'removed':\n                    span.style.background = '#ffeef0';\n                    span.style.borderColor = '#d73a49';\n                    span.style.color = '#cb2431';\n                    break;\n                  case 'changed':\n                    span.style.background = '#fff5b1';\n                    span.style.borderColor = '#ffd33d';\n                    span.style.color = '#735c0f';\n                    break;\n                  case 'unchanged':\n                    span.style.background = '#f1f8ff';\n                    span.style.borderColor = '#79b8ff';\n                    span.style.color = '#0366d6';\n                    break;\n                  default:\n                    break; // exhaustive\n                }\n                return span;\n              }\n              if (diff.added.length > 0) {\n                body.append(\n                  ...diff.added.map((k) => {\n                    return pill(`+${k}`, 'added');\n                  }),\n                );\n              }\n              if (diff.changed.length > 0) {\n                body.append(\n                  ...diff.changed.map((k) => {\n                    return pill(`~${k}`, 'changed');\n                  }),\n                );\n              }\n              if (diff.removed.length > 0) {\n                body.append(\n                  ...diff.removed.map((k) => {\n                    return pill(`-${k}`, 'removed');\n                  }),\n                );\n              }\n              const noAdds = diff.added.length === 0;\n              const noChanges = diff.changed.length === 0;\n              const noRemovals = diff.removed.length === 0;\n              if (noAdds && noChanges && noRemovals) {\n                body.append(pill('no-change', 'unchanged'));\n              }\n              row.appendChild(head);\n              row.appendChild(body);\n              timeline.appendChild(row);\n              if (timeline.childElementCount > 50) {\n                const first = timeline.firstElementChild;\n                if (first != null) {\n                  timeline.removeChild(first);\n                }\n              }\n              lastQuery = nowQuery;\n              lastSearch = entry.rawSearch;\n            }\n          } catch {\n            /* ignore */\n          }\n        }\n        render();\n        unsub = context.subscribe(() => {\n          render();\n        });\n        window.addEventListener('keydown', (ev) => {\n          try {\n            const parts: string[] = [];\n            if (ev.altKey) parts.push('Alt');\n            if (ev.shiftKey) parts.push('Shift');\n            if (ev.ctrlKey) parts.push('Ctrl');\n            let keyPart = ev.key;\n            if (ev.key.length === 1) {\n              keyPart = ev.key.toUpperCase();\n            }\n            parts.push(keyPart);\n            const combo = parts.join('+');\n            if (combo === devtoolsShortcut.replace(/\\s+/g, '')) {\n              if (host.style.display === 'none') {\n                host.style.display = 'block';\n              } else {\n                host.style.display = 'none';\n              }\n            }\n          } catch {\n            /* ignore */\n          }\n        });\n      }\n    } catch {\n      /* ignore */\n    }\n  }\n  return { cleanup, context };\n}\n"]}
|