@plumile/router 0.1.12 → 0.1.13

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (68) hide show
  1. package/README.md +426 -13
  2. package/lib/esm/index.d.ts +4 -0
  3. package/lib/esm/index.d.ts.map +1 -1
  4. package/lib/esm/index.js +5 -1
  5. package/lib/esm/routing/createRouter.d.ts +1 -0
  6. package/lib/esm/routing/createRouter.d.ts.map +1 -1
  7. package/lib/esm/routing/createRouter.js +399 -359
  8. package/lib/esm/routing/devtools.d.ts +20 -0
  9. package/lib/esm/routing/devtools.d.ts.map +1 -0
  10. package/lib/esm/routing/devtools.js +678 -0
  11. package/lib/esm/routing/filters.d.ts +97 -0
  12. package/lib/esm/routing/filters.d.ts.map +1 -0
  13. package/lib/esm/routing/filters.js +557 -0
  14. package/lib/esm/routing/index.d.ts +6 -0
  15. package/lib/esm/routing/index.d.ts.map +1 -1
  16. package/lib/esm/routing/index.js +7 -1
  17. package/lib/esm/routing/useActiveFilters.d.ts +9 -0
  18. package/lib/esm/routing/useActiveFilters.d.ts.map +1 -0
  19. package/lib/esm/routing/useActiveFilters.js +38 -0
  20. package/lib/esm/routing/useFilterState.d.ts +10 -0
  21. package/lib/esm/routing/useFilterState.d.ts.map +1 -0
  22. package/lib/esm/routing/useFilterState.js +14 -0
  23. package/lib/esm/routing/useNavigate.d.ts +7 -0
  24. package/lib/esm/routing/useNavigate.d.ts.map +1 -1
  25. package/lib/esm/routing/useNavigate.js +1 -1
  26. package/lib/esm/routing/useNavigateWithQuery.d.ts +15 -0
  27. package/lib/esm/routing/useNavigateWithQuery.d.ts.map +1 -0
  28. package/lib/esm/routing/useNavigateWithQuery.js +95 -0
  29. package/lib/esm/routing/useQueryObject.d.ts +18 -0
  30. package/lib/esm/routing/useQueryObject.d.ts.map +1 -0
  31. package/lib/esm/routing/useQueryObject.js +107 -0
  32. package/lib/esm/routing/useStableRefEquality.d.ts +5 -0
  33. package/lib/esm/routing/useStableRefEquality.d.ts.map +1 -0
  34. package/lib/esm/routing/useStableRefEquality.js +47 -0
  35. package/lib/esm/tools/buildSearch.d.ts +8 -2
  36. package/lib/esm/tools/buildSearch.d.ts.map +1 -1
  37. package/lib/esm/tools/buildSearch.js +216 -12
  38. package/lib/esm/types.d.ts +19 -0
  39. package/lib/esm/types.d.ts.map +1 -1
  40. package/lib/esm/types.js +1 -1
  41. package/lib/tsconfig.esm.tsbuildinfo +1 -1
  42. package/lib/types/index.d.ts +4 -0
  43. package/lib/types/index.d.ts.map +1 -1
  44. package/lib/types/routing/createRouter.d.ts +1 -0
  45. package/lib/types/routing/createRouter.d.ts.map +1 -1
  46. package/lib/types/routing/devtools.d.ts +20 -0
  47. package/lib/types/routing/devtools.d.ts.map +1 -0
  48. package/lib/types/routing/filters.d.ts +97 -0
  49. package/lib/types/routing/filters.d.ts.map +1 -0
  50. package/lib/types/routing/index.d.ts +6 -0
  51. package/lib/types/routing/index.d.ts.map +1 -1
  52. package/lib/types/routing/useActiveFilters.d.ts +9 -0
  53. package/lib/types/routing/useActiveFilters.d.ts.map +1 -0
  54. package/lib/types/routing/useFilterState.d.ts +10 -0
  55. package/lib/types/routing/useFilterState.d.ts.map +1 -0
  56. package/lib/types/routing/useNavigate.d.ts +7 -0
  57. package/lib/types/routing/useNavigate.d.ts.map +1 -1
  58. package/lib/types/routing/useNavigateWithQuery.d.ts +15 -0
  59. package/lib/types/routing/useNavigateWithQuery.d.ts.map +1 -0
  60. package/lib/types/routing/useQueryObject.d.ts +18 -0
  61. package/lib/types/routing/useQueryObject.d.ts.map +1 -0
  62. package/lib/types/routing/useStableRefEquality.d.ts +5 -0
  63. package/lib/types/routing/useStableRefEquality.d.ts.map +1 -0
  64. package/lib/types/tools/buildSearch.d.ts +8 -2
  65. package/lib/types/tools/buildSearch.d.ts.map +1 -1
  66. package/lib/types/types.d.ts +19 -0
  67. package/lib/types/types.d.ts.map +1 -1
  68. package/package.json +3 -3
@@ -46,11 +46,16 @@ export default function createRouter(routes, opts = {}) {
46
46
  let initialSchema;
47
47
  if (route != null) {
48
48
  const collected = [];
49
+ const collectedFilters = [];
49
50
  for (const r of route.route.routes) {
50
51
  const maybe = r.query;
51
52
  if (maybe != null) {
52
53
  collected.push(maybe);
53
54
  }
55
+ const maybeFilter = r.filterSchema;
56
+ if (maybeFilter != null) {
57
+ collectedFilters.push(maybeFilter);
58
+ }
54
59
  }
55
60
  if (collected.length > 0) {
56
61
  initialSchema = collected.at(-1);
@@ -103,11 +108,16 @@ export default function createRouter(routes, opts = {}) {
103
108
  let schema;
104
109
  if (nextRoute != null) {
105
110
  const collected = [];
111
+ const collectedFilters = [];
106
112
  for (const r of nextRoute.route.routes) {
107
113
  const maybe = r.query;
108
114
  if (maybe != null) {
109
115
  collected.push(maybe);
110
116
  }
117
+ const maybeFilter = r.filterSchema;
118
+ if (maybeFilter != null) {
119
+ collectedFilters.push(maybeFilter);
120
+ }
111
121
  }
112
122
  if (collected.length > 0) {
113
123
  schema = collected.at(-1);
@@ -152,23 +162,272 @@ export default function createRouter(routes, opts = {}) {
152
162
  callback(nextEntry);
153
163
  });
154
164
  });
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;
155
410
  const context = {
156
411
  history,
157
412
  get() {
158
413
  return currentEntry;
159
414
  },
415
+ resolveFilterSchema(pathname) {
416
+ return resolveFilterSchema(pathname);
417
+ },
418
+ currentMergedFilterSchema() {
419
+ return resolveFilterSchema(currentEntry.location.pathname);
420
+ },
160
421
  preloadCode(pathname) {
161
422
  const matches = getMatchedRoute(flatRoutes, {
162
423
  ...window.location,
163
424
  pathname,
164
425
  });
165
- if (matches == null) {
426
+ if (matches == null)
166
427
  return;
167
- }
168
428
  matches.route.routes.forEach(({ resourcePage }) => {
169
- if (resourcePage == null) {
429
+ if (resourcePage == null)
170
430
  return;
171
- }
172
431
  resourcePage.load();
173
432
  });
174
433
  },
@@ -182,377 +441,158 @@ export default function createRouter(routes, opts = {}) {
182
441
  subscribe(callback) {
183
442
  nextId += 1;
184
443
  const id = nextId;
185
- const dispose = () => {
444
+ function dispose() {
186
445
  subscribers.delete(id);
187
- };
446
+ }
188
447
  subscribers.set(id, callback);
189
448
  return dispose;
190
449
  },
191
- navigate({ pathname, query, replace, }) {
192
- const current = currentEntry;
193
- const targetPathname = pathname ?? current.location.pathname;
194
- let schema;
195
- function collectSchema(routesArr) {
196
- if (routesArr == null) {
450
+ navigate({ pathname, query, replace, filters, filterSchema, batch, immediate, inherit, }) {
451
+ try {
452
+ const effectiveInherit = inherit === true;
453
+ const wantsBatch = batch === true ||
454
+ (autoBatch && immediate !== true && batch !== false);
455
+ if (wantsBatch) {
456
+ const nav = {};
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
+ });
197
507
  return;
198
508
  }
199
- const collected = [];
200
- for (const r of routesArr) {
201
- const maybe = r.query;
202
- if (maybe != null) {
203
- collected.push(maybe);
204
- }
509
+ let base;
510
+ if (effectiveInherit) {
511
+ base = { ...(currentEntry.typedQuery ?? currentEntry.query ?? {}) };
205
512
  }
206
- if (collected.length > 0) {
207
- schema = collected.at(-1);
513
+ if (query != null) {
514
+ if (base != null)
515
+ Object.assign(base, query);
516
+ else
517
+ base = query;
208
518
  }
209
- }
210
- if (pathname == null || pathname === current.location.pathname) {
211
- collectSchema(current.route?.route.routes);
212
- }
213
- else {
214
- const destRoute = getMatchedRoute(flatRoutes, {
215
- ...window.location,
216
- pathname: targetPathname,
519
+ base ??= currentEntry.typedQuery ?? currentEntry.query ?? {};
520
+ const targetPath = pathname ?? currentEntry.location.pathname;
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 };
527
+ }
528
+ const search = buildSearch(base, schema, buildOpts);
529
+ let targetPathname = targetPath;
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
+ });
538
+ }
539
+ const effectiveSchema = schema ?? resolveSchema(targetPathname);
540
+ const rawQueryObj = parseRawQuery(search);
541
+ const typed = parseTypedQuery(effectiveSchema, rawQueryObj);
542
+ const prepared = prepareMatch(nextRoute, typed);
543
+ currentEntry = {
544
+ forceRerender: false,
545
+ location: {
546
+ ...currentEntry.location,
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(),
217
571
  });
218
- collectSchema(destRoute?.route.routes);
219
572
  }
220
- const search = buildSearch(query, schema);
221
- const locationObj = { pathname: targetPathname, search, hash: '' };
222
- if (replace) {
223
- history.set(locationObj);
224
- }
225
- else {
226
- history.push(locationObj);
573
+ catch {
227
574
  }
228
575
  },
229
576
  };
230
- if (devtoolsGlobal) {
231
- try {
232
- if (typeof window !== 'undefined') {
233
- window.__PLUMILE_ROUTER__ =
234
- {
235
- get: context.get,
236
- subscribe: context.subscribe,
237
- };
238
- }
239
- }
240
- catch {
241
- }
242
- }
243
- if (devtoolsPanel) {
244
- try {
245
- if (typeof window !== 'undefined' &&
246
- typeof document !== 'undefined' &&
247
- document.querySelector('[data-plumile-router-devtools-panel]') == null) {
248
- const host = document.createElement('div');
249
- host.setAttribute('data-plumile-router-devtools-panel', '');
250
- host.style.position = 'fixed';
251
- host.style.zIndex = '2147483646';
252
- host.style.bottom = '8px';
253
- host.style.right = '8px';
254
- host.style.fontFamily =
255
- 'ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", monospace';
256
- host.style.fontSize = '12px';
257
- host.style.lineHeight = '1.3';
258
- host.style.color = '#111';
259
- host.style.background = 'transparent';
260
- host.style.pointerEvents = 'none';
261
- const shadow = host.attachShadow({ mode: 'open' });
262
- const panel = document.createElement('div');
263
- panel.style.pointerEvents = 'auto';
264
- panel.style.minWidth = '260px';
265
- panel.style.maxWidth = '420px';
266
- panel.style.maxHeight = '50vh';
267
- panel.style.overflow = 'auto';
268
- panel.style.background = 'rgba(255,255,255,0.95)';
269
- panel.style.border = '1px solid #ccc';
270
- panel.style.borderRadius = '6px';
271
- panel.style.boxShadow = '0 2px 8px rgba(0,0,0,0.15)';
272
- panel.style.padding = '6px 8px 8px';
273
- panel.style.backdropFilter = 'blur(4px)';
274
- panel.style.display = 'flex';
275
- panel.style.flexDirection = 'column';
276
- panel.style.rowGap = '4px';
277
- const header = document.createElement('div');
278
- header.style.display = 'flex';
279
- header.style.alignItems = 'center';
280
- header.style.justifyContent = 'space-between';
281
- header.style.gap = '8px';
282
- const title = document.createElement('span');
283
- title.textContent = 'Plumile Router';
284
- title.style.fontWeight = '600';
285
- const buttons = document.createElement('div');
286
- buttons.style.display = 'flex';
287
- buttons.style.gap = '4px';
288
- const collapseBtn = document.createElement('button');
289
- collapseBtn.textContent = '−';
290
- const closeBtn = document.createElement('button');
291
- closeBtn.textContent = '×';
292
- const btns = [collapseBtn, closeBtn];
293
- for (const b of btns) {
294
- b.style.cursor = 'pointer';
295
- b.style.background = '#eee';
296
- b.style.border = '1px solid #ccc';
297
- b.style.borderRadius = '4px';
298
- b.style.fontSize = '11px';
299
- b.style.lineHeight = '1';
300
- b.style.padding = '2px 5px';
301
- }
302
- header.appendChild(title);
303
- buttons.appendChild(collapseBtn);
304
- buttons.appendChild(closeBtn);
305
- header.appendChild(buttons);
306
- const pathEl = document.createElement('div');
307
- pathEl.style.whiteSpace = 'nowrap';
308
- const varsEl = document.createElement('pre');
309
- const queryEl = document.createElement('pre');
310
- const typedEl = document.createElement('pre');
311
- const timelineWrap = document.createElement('div');
312
- timelineWrap.style.display = 'flex';
313
- timelineWrap.style.flexDirection = 'column';
314
- timelineWrap.style.gap = '2px';
315
- timelineWrap.style.marginTop = '4px';
316
- const timelineHeader = document.createElement('div');
317
- timelineHeader.textContent = 'Timeline (query changes)';
318
- timelineHeader.style.fontWeight = '600';
319
- timelineHeader.style.fontSize = '11px';
320
- const timeline = document.createElement('div');
321
- timeline.style.display = 'flex';
322
- timeline.style.flexDirection = 'column-reverse';
323
- timeline.style.maxHeight = '140px';
324
- timeline.style.overflowY = 'auto';
325
- timeline.style.border = '1px solid #e2e2e2';
326
- timeline.style.borderRadius = '4px';
327
- timeline.style.background = '#fafbfc';
328
- timeline.style.fontFamily = 'inherit';
329
- timeline.style.fontSize = '11px';
330
- timelineWrap.appendChild(timelineHeader);
331
- timelineWrap.appendChild(timeline);
332
- const pres = [varsEl, queryEl, typedEl];
333
- for (const p of pres) {
334
- p.style.margin = '0';
335
- p.style.padding = '4px';
336
- p.style.background = '#f6f8fa';
337
- p.style.border = '1px solid #e2e2e2';
338
- p.style.borderRadius = '4px';
339
- p.style.overflowX = 'auto';
340
- p.style.maxHeight = '160px';
341
- }
342
- const small = document.createElement('div');
343
- small.style.fontSize = '10px';
344
- small.style.opacity = '0.7';
345
- small.textContent = `Shortcut: ${devtoolsShortcut}`;
346
- panel.appendChild(header);
347
- panel.appendChild(pathEl);
348
- panel.appendChild(varsEl);
349
- panel.appendChild(queryEl);
350
- panel.appendChild(typedEl);
351
- panel.appendChild(timelineWrap);
352
- panel.appendChild(small);
353
- shadow.appendChild(panel);
354
- document.body.appendChild(host);
355
- let collapsed = false;
356
- collapseBtn.addEventListener('click', () => {
357
- collapsed = !collapsed;
358
- if (collapsed) {
359
- panel.style.maxHeight = '24px';
360
- panel.style.overflow = 'hidden';
361
- collapseBtn.textContent = '+';
362
- }
363
- else {
364
- panel.style.maxHeight = '50vh';
365
- panel.style.overflow = 'auto';
366
- collapseBtn.textContent = '−';
367
- }
368
- });
369
- let unsub;
370
- closeBtn.addEventListener('click', () => {
371
- host.remove();
372
- if (typeof unsub === 'function') {
373
- unsub();
374
- }
375
- });
376
- let lastQuery;
377
- let lastSearch;
378
- function diffQueries(prev, next) {
379
- const added = [];
380
- const removed = [];
381
- const changed = [];
382
- const unchanged = [];
383
- const prevKeys = new Set(Object.keys(prev ?? {}));
384
- const nextKeys = new Set(Object.keys(next));
385
- nextKeys.forEach((k) => {
386
- if (!prevKeys.has(k)) {
387
- added.push(k);
388
- return;
389
- }
390
- let pv;
391
- if (prev != null) {
392
- pv = prev[k];
393
- }
394
- const nv = next[k];
395
- let same = false;
396
- if (Array.isArray(pv) && Array.isArray(nv)) {
397
- if (pv.length === nv.length) {
398
- same = pv.every((v, i) => {
399
- return v === nv[i];
400
- });
401
- }
402
- }
403
- else {
404
- same = pv === nv;
405
- }
406
- if (same) {
407
- unchanged.push(k);
408
- }
409
- else {
410
- changed.push(k);
411
- }
412
- prevKeys.delete(k);
413
- });
414
- prevKeys.forEach((k) => {
415
- removed.push(k);
577
+ if ((devtoolsGlobal || devtoolsPanel) && typeof window !== 'undefined') {
578
+ (async () => {
579
+ try {
580
+ const mod = await import('./devtools.js');
581
+ if (mod != null && typeof mod.initRouterDevtools === 'function') {
582
+ devtoolsOnNavEvent = mod.initRouterDevtools({
583
+ context,
584
+ options: {
585
+ global: devtoolsGlobal,
586
+ panel: devtoolsPanel,
587
+ shortcut: devtoolsShortcut,
588
+ },
416
589
  });
417
- return { added, removed, changed, unchanged };
418
590
  }
419
- function render() {
420
- try {
421
- const entry = context.get();
422
- pathEl.textContent = `${entry.location.pathname}${entry.location.search}`;
423
- const varsJson = JSON.stringify(entry.route?.params ?? {}, null, 2);
424
- varsEl.textContent = `vars: ${varsJson}`;
425
- const rawJson = JSON.stringify(entry.query, null, 2);
426
- queryEl.textContent = `raw: ${rawJson}`;
427
- const typedJson = JSON.stringify(entry.typedQuery, null, 2);
428
- typedEl.textContent = `typed:${typedJson}`;
429
- if (entry.rawSearch !== lastSearch) {
430
- const nowQuery = entry.query;
431
- const diff = diffQueries(lastQuery, nowQuery);
432
- const row = document.createElement('div');
433
- row.style.display = 'flex';
434
- row.style.flexDirection = 'column';
435
- row.style.borderBottom = '1px solid #e9e9e9';
436
- row.style.padding = '3px 4px';
437
- const head = document.createElement('div');
438
- head.style.display = 'flex';
439
- head.style.justifyContent = 'space-between';
440
- const ts = new Date().toLocaleTimeString();
441
- let searchDisplay = '(empty)';
442
- if (entry.location.search !== '') {
443
- searchDisplay = entry.location.search;
444
- }
445
- head.innerHTML = `<strong>${ts}</strong> <code style="background:#eee;padding:1px 3px;border-radius:3px;">${searchDisplay}</code>`;
446
- const body = document.createElement('div');
447
- body.style.display = 'flex';
448
- body.style.flexWrap = 'wrap';
449
- body.style.gap = '4px';
450
- function pill(label, kind) {
451
- const span = document.createElement('span');
452
- span.textContent = label;
453
- span.style.padding = '1px 4px';
454
- span.style.borderRadius = '10px';
455
- span.style.fontSize = '10px';
456
- span.style.border = '1px solid';
457
- switch (kind) {
458
- case 'added':
459
- span.style.background = '#e6ffed';
460
- span.style.borderColor = '#2cbe4e';
461
- span.style.color = '#22863a';
462
- break;
463
- case 'removed':
464
- span.style.background = '#ffeef0';
465
- span.style.borderColor = '#d73a49';
466
- span.style.color = '#cb2431';
467
- break;
468
- case 'changed':
469
- span.style.background = '#fff5b1';
470
- span.style.borderColor = '#ffd33d';
471
- span.style.color = '#735c0f';
472
- break;
473
- case 'unchanged':
474
- span.style.background = '#f1f8ff';
475
- span.style.borderColor = '#79b8ff';
476
- span.style.color = '#0366d6';
477
- break;
478
- default:
479
- break;
480
- }
481
- return span;
482
- }
483
- if (diff.added.length > 0) {
484
- body.append(...diff.added.map((k) => {
485
- return pill(`+${k}`, 'added');
486
- }));
487
- }
488
- if (diff.changed.length > 0) {
489
- body.append(...diff.changed.map((k) => {
490
- return pill(`~${k}`, 'changed');
491
- }));
492
- }
493
- if (diff.removed.length > 0) {
494
- body.append(...diff.removed.map((k) => {
495
- return pill(`-${k}`, 'removed');
496
- }));
497
- }
498
- const noAdds = diff.added.length === 0;
499
- const noChanges = diff.changed.length === 0;
500
- const noRemovals = diff.removed.length === 0;
501
- if (noAdds && noChanges && noRemovals) {
502
- body.append(pill('no-change', 'unchanged'));
503
- }
504
- row.appendChild(head);
505
- row.appendChild(body);
506
- timeline.appendChild(row);
507
- if (timeline.childElementCount > 50) {
508
- const first = timeline.firstElementChild;
509
- if (first != null) {
510
- timeline.removeChild(first);
511
- }
512
- }
513
- lastQuery = nowQuery;
514
- lastSearch = entry.rawSearch;
515
- }
516
- }
517
- catch {
518
- }
519
- }
520
- render();
521
- unsub = context.subscribe(() => {
522
- render();
523
- });
524
- window.addEventListener('keydown', (ev) => {
525
- try {
526
- const parts = [];
527
- if (ev.altKey)
528
- parts.push('Alt');
529
- if (ev.shiftKey)
530
- parts.push('Shift');
531
- if (ev.ctrlKey)
532
- parts.push('Ctrl');
533
- let keyPart = ev.key;
534
- if (ev.key.length === 1) {
535
- keyPart = ev.key.toUpperCase();
536
- }
537
- parts.push(keyPart);
538
- const combo = parts.join('+');
539
- if (combo === devtoolsShortcut.replace(/\s+/g, '')) {
540
- if (host.style.display === 'none') {
541
- host.style.display = 'block';
542
- }
543
- else {
544
- host.style.display = 'none';
545
- }
546
- }
547
- }
548
- catch {
549
- }
550
- });
551
591
  }
552
- }
553
- catch {
554
- }
592
+ catch {
593
+ }
594
+ })();
555
595
  }
556
596
  return { cleanup, context };
557
597
  }
558
- //# 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  } = {},\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  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      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      }\n      if (collected.length > 0) {\n        schema = collected.at(-1);\n      }\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  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      replace,\n    }: {\n      pathname?: string;\n      query?: 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      /** 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        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        }\n        if (collected.length > 0) {\n          schema = collected.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 search = buildSearch(query, schema);\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        // 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[] = [varsEl, queryEl, typedEl];\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(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            // 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"]}
598
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"createRouter.js","sourceRoot":"","sources":["../../../src/routing/createRouter.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,eAAe,CAAC;AAC5C,OAAO,EAAE,cAAc,EAAE,MAAM,qBAAqB,CAAC;AACrD,OAAO,EAAE,eAAe,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAC5D,OAAO,EAAE,eAAe,EAAE,MAAM,uBAAuB,CAAC;AACxD,OAAO,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAC;AAClD,OAAO,EAAE,WAAW,EAAE,MAAM,yBAAyB,CAAC;AAkDtD,MAAM,CAAC,OAAO,UAAU,YAAY,CAClC,MAAyB,EACzB,OAMI,EAAE;IAGN,IAAI,cAAuB,CAAC;IAC5B,IAAI,aAAa,GAAG,KAAK,CAAC;IAC1B,IAAI,gBAAgB,GAAG,aAAa,CAAC;IACrC,IAAI,OAAO,IAAI,CAAC,QAAQ,KAAK,QAAQ,EAAE,CAAC;QACtC,MAAM,WAAW,GAAG,CAAC,GAAG,EAAE;YACxB,IAAI,CAAC;gBACH,OAAO,CACL,OAAO,OAAO,KAAK,WAAW;oBAC9B,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,YAAY,CACtC,CAAC;YACJ,CAAC;YAAC,MAAM,CAAC;gBACP,OAAO,KAAK,CAAC;YACf,CAAC;QACH,CAAC,CAAC,EAAE,CAAC;QACL,cAAc,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,IAAI,WAAW,CAAC;QACrD,aAAa,GAAG,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC;QACtC,IACE,OAAO,IAAI,CAAC,QAAQ,CAAC,QAAQ,KAAK,QAAQ;YAC1C,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,EAAE,KAAK,EAAE,EACpC,CAAC;YACD,gBAAgB,GAAG,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC;QAC5C,CAAC;IACH,CAAC;SAAM,CAAC;QAEN,MAAM,OAAO,GAAG,CAAC,GAAG,EAAE;YACpB,IAAI,OAAO,IAAI,CAAC,QAAQ,KAAK,SAAS;gBAAE,OAAO,IAAI,CAAC,QAAQ,CAAC;YAC7D,IAAI,CAAC;gBACH,OAAO,CACL,OAAO,OAAO,KAAK,WAAW;oBAC9B,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,YAAY,CACtC,CAAC;YACJ,CAAC;YAAC,MAAM,CAAC;gBACP,OAAO,KAAK,CAAC;YACf,CAAC;QACH,CAAC,CAAC,EAAE,CAAC;QACL,cAAc,GAAG,OAAO,CAAC;IAC3B,CAAC;IAED,MAAM,OAAO,GAAG,IAAI,cAAc,EAAE,CAAC;IAGrC,MAAM,UAAU,GAAG,WAAW,CAAC,MAAM,CAAC,CAAC;IAGvC,MAAM,KAAK,GAAG,eAAe,CAAC,UAAU,EAAE,OAAO,CAAC,QAAQ,CAAC,CAAC;IAC5D,MAAM,eAAe,GAAG,aAAa,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;IAE/D,IAAI,aAAkB,CAAC;IAIvB,IAAI,KAAK,IAAI,IAAI,EAAE,CAAC;QAClB,MAAM,SAAS,GAAU,EAAE,CAAC;QAC5B,MAAM,gBAAgB,GACpB,EAAE,CAAC;QACL,KAAK,MAAM,CAAC,IAAI,KAAK,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;YACnC,MAAM,KAAK,GAAa,CAAoC,CAAC,KAAK,CAAC;YACnE,IAAI,KAAK,IAAI,IAAI,EAAE,CAAC;gBAClB,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACxB,CAAC;YACD,MAAM,WAAW,GACf,CAGD,CAAC,YAAY,CAAC;YACf,IAAI,WAAW,IAAI,IAAI,EAAE,CAAC;gBACxB,gBAAgB,CAAC,IAAI,CACnB,WAA+D,CAChE,CAAC;YACJ,CAAC;QACH,CAAC;QACD,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACzB,aAAa,GAAG,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;QACnC,CAAC;IAEH,CAAC;IACD,MAAM,iBAAiB,GAAG,eAAe,CAAC,aAAa,EAAE,eAAe,CAAC,CAAC;IAC1E,MAAM,aAAa,GAAG,YAAY,CAAC,KAAK,EAAE,iBAAiB,CAAC,CAAC;IAE7D,IAAI,YAAY,GAAoB;QAClC,aAAa,EAAE,KAAK;QACpB,QAAQ,EAAE,OAAO,CAAC,QAAQ;QAC1B,KAAK;QACL,aAAa;QACb,SAAS,EAAE,OAAO,CAAC,QAAQ,CAAC,MAAM;QAClC,KAAK,EAAE,eAAe;QACtB,UAAU,EAAE,iBAAiB;KAC9B,CAAC;IAGF,MAAM,SAAS,GAAG,YAAY,CAAC,UAElB,CAAC;IACd,IAAI,QAAiB,CAAC;IACtB,IAAI,SAAS,IAAI,IAAI,EAAE,CAAC;QACtB,QAAQ,GAAG,SAAS,CAAC,IAAI,CAAC;IAC5B,CAAC;IACD,IACE,OAAO,QAAQ,KAAK,QAAQ;QAC5B,QAAQ,GAAG,CAAC;QACZ,aAAa,IAAI,IAAI;QACrB,SAAS,IAAI,IAAI,EACjB,CAAC;QACD,MAAM,IAAI,GAAG,EAAE,GAAG,SAAS,EAAE,IAAI,EAAE,CAAC,EAA6B,CAAC;QAClE,MAAM,gBAAgB,GAAG,WAAW,CAAC,IAAI,EAAE,aAAa,CAAC,CAAC;QAC1D,IAAI,gBAAgB,KAAK,OAAO,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC;YACjD,OAAO,CAAC,GAAG,CAAC;gBACV,QAAQ,EAAE,OAAO,CAAC,QAAQ,CAAC,QAAQ;gBACnC,MAAM,EAAE,gBAAgB;gBACxB,IAAI,EAAE,EAAE;aACT,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAGD,IAAI,MAAM,GAAG,CAAC,CAAC;IACf,MAAM,WAAW,GAAG,IAAI,GAAG,EAAkC,CAAC;IAK9D,MAAM,OAAO,GAAG,OAAO,CAAC,SAAS,CAAC,CAAC,QAAQ,EAAE,aAAa,EAAE,EAAE;QAC5D,MAAM,YAAY,GAAG,QAAQ,CAAC,QAAQ,KAAK,YAAY,CAAC,QAAQ,CAAC,QAAQ,CAAC;QAC1E,MAAM,UAAU,GAAG,QAAQ,CAAC,MAAM,KAAK,YAAY,CAAC,SAAS,CAAC;QAE9D,IAAI,CAAC,aAAa,IAAI,YAAY,IAAI,UAAU,EAAE,CAAC;YAEjD,OAAO;QACT,CAAC;QAID,IAAI,iBAAiB,GAAG,YAAY,CAAC,aAAa,CAAC;QACnD,IAAI,SAAS,GAAG,YAAY,CAAC,KAAK,CAAC;QAEnC,IAAI,CAAC,YAAY,EAAE,CAAC;YAElB,SAAS,GAAG,eAAe,CAAC,UAAU,EAAE,OAAO,CAAC,QAAQ,CAAC,CAAC;QAE5D,CAAC;QAGD,MAAM,KAAK,GAAG,aAAa,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;QAE7C,IAAI,MAAW,CAAC;QAChB,IAAI,SAAS,IAAI,IAAI,EAAE,CAAC;YACtB,MAAM,SAAS,GAAU,EAAE,CAAC;YAC5B,MAAM,gBAAgB,GAGhB,EAAE,CAAC;YACT,KAAK,MAAM,CAAC,IAAI,SAAS,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;gBACvC,MAAM,KAAK,GAAa,CAAoC,CAAC,KAAK,CAAC;gBACnE,IAAI,KAAK,IAAI,IAAI,EAAE,CAAC;oBAClB,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBACxB,CAAC;gBACD,MAAM,WAAW,GACf,CAGD,CAAC,YAAY,CAAC;gBACf,IAAI,WAAW,IAAI,IAAI,EAAE,CAAC;oBACxB,gBAAgB,CAAC,IAAI,CACnB,WAA+D,CAChE,CAAC;gBACJ,CAAC;YACH,CAAC;YACD,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACzB,MAAM,GAAG,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;YAC5B,CAAC;QAEH,CAAC;QACD,IAAI,UAAU,GAAG,eAAe,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;QAEhD,IAAI,UAAU,GAAG,KAAK,CAAC;QACvB,MAAM,EAAE,GAAI,UAAkC,CAAC,IAAI,CAAC;QACpD,IAAI,OAAO,EAAE,KAAK,QAAQ,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC;YACrC,MAAM,KAAK,GAAG,EAAE,GAAI,UAAkC,EAAE,IAAI,EAAE,CAAC,EAAE,CAAC;YAClE,UAAU,GAAG,KAAK,CAAC;YACnB,UAAU,GAAG,IAAI,CAAC;QACpB,CAAC;QAGD,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,iBAAiB,GAAG,YAAY,CAAC,SAAS,EAAE,UAAU,CAAC,CAAC;QAC1D,CAAC;aAAM,IAAI,CAAC,UAAU,EAAE,CAAC;YACvB,iBAAiB,GAAG,YAAY,CAAC,SAAS,EAAE,UAAU,CAAC,CAAC;QAC1D,CAAC;QAED,MAAM,SAAS,GAAoB;YACjC,aAAa,EAAE,aAAa,IAAI,CAAC,YAAY,IAAI,CAAC,UAAU,CAAC;YAC7D,QAAQ;YACR,KAAK,EAAE,SAAS;YAChB,aAAa,EAAE,iBAAiB;YAChC,SAAS,EAAE,QAAQ,CAAC,MAAM;YAC1B,KAAK;YACL,UAAU;SACX,CAAC;QAGF,IAAI,UAAU,IAAI,MAAM,IAAI,IAAI,EAAE,CAAC;YACjC,MAAM,gBAAgB,GAAG,WAAW,CAAC,UAAiB,EAAE,MAAM,CAAC,CAAC;YAChE,IAAI,gBAAgB,KAAK,QAAQ,CAAC,MAAM,EAAE,CAAC;gBACzC,OAAO,CAAC,GAAG,CAAC;oBACV,QAAQ,EAAE,QAAQ,CAAC,QAAQ;oBAC3B,MAAM,EAAE,gBAAgB;oBACxB,IAAI,EAAE,EAAE;iBACT,CAAC,CAAC;gBACH,OAAO;YACT,CAAC;QACH,CAAC;QAGD,YAAY,GAAG,SAAS,CAAC;QACzB,WAAW,CAAC,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE;YAC/B,QAAQ,CAAC,SAAS,CAAC,CAAC;QACtB,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAaH,IAAI,WAAqC,CAAC;IAC1C,IAAI,cAAc,GAAG,KAAK,CAAC;IAE3B,IAAI,kBAAmD,CAAC;IAExD,IAAI,cAA+C,CAAC;IAEpD,IAAI,cAA+C,CAAC;IACpD,IAAI,mBAES,CAAC;IAEd,IAAI,kBASS,CAAC;IAEd,IAAI,SAAkD,CAAC;IAGvD,SAAS,iBAAiB,CAAC,QAAgB;QACzC,IAAI,QAAQ,KAAK,YAAY,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;YAChD,IAAI,YAAY,CAAC,KAAK,IAAI,IAAI,EAAE,CAAC;gBAC/B,OAAO,YAAY,CAAC,KAAK,CAAC,KAAK,CAAC,MAAe,CAAC;YAClD,CAAC;YACD,OAAO,SAAS,CAAC;QACnB,CAAC;QACD,MAAM,SAAS,GAAG,eAAe,CAAC,UAAU,EAAE;YAC5C,GAAG,MAAM,CAAC,QAAQ;YAClB,QAAQ;SACT,CAAC,CAAC;QACH,IAAI,SAAS,IAAI,IAAI;YAAE,OAAO,SAAS,CAAC,KAAK,CAAC,MAAe,CAAC;QAC9D,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,SAAS,yBAAyB,CAAC,KAAwB;QACzD,IAAI,KAAK,IAAI,IAAI;YAAE,OAAO,SAAS,CAAC;QACpC,MAAM,MAAM,GAAwB,EAAE,CAAC;QACvC,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;YACzB,MAAM,IAAI,GAAI,IAAwC,CAAC,KAAK,CAAC;YAC7D,IAAI,IAAI,IAAI,IAAI,EAAE,CAAC;gBACjB,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC;oBAC1C,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC;gBAChB,CAAC;YACH,CAAC;QACH,CAAC;QACD,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,SAAS,CAAC;QACvD,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,SAAS,0BAA0B,CACjC,KAAwB;QAExB,IAAI,KAAK,IAAI,IAAI;YAAE,OAAO,SAAS,CAAC;QACpC,IAAI,MAAuC,CAAC;QAC5C,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;YACzB,MAAM,EAAE,GACN,IAGD,CAAC,YAAY,CAAC;YACf,IAAI,EAAE,IAAI,IAAI,EAAE,CAAC;gBACf,MAAM,KAAK,EAAE,CAAC;gBACd,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,CAAC;YACnC,CAAC;QACH,CAAC;QACD,IAAI,MAAM,IAAI,IAAI;YAAE,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,EAAE,MAAM,EAAE,CAAC;QAC/D,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,SAAS,aAAa,CAAC,QAAgB;QACrC,OAAO,yBAAyB,CAAC,iBAAiB,CAAC,QAAQ,CAAC,CAAC,CAAC;IAChE,CAAC;IAED,SAAS,mBAAmB,CAC1B,QAAgB;QAEhB,OAAO,0BAA0B,CAAC,iBAAiB,CAAC,QAAQ,CAAC,CAAC,CAAC;IACjE,CAAC;IAED,SAAS,mBAAmB,CAAC,IAAkB;QAO7C,IAAI,cAAc,GAAG,YAAY,CAAC,QAAQ,CAAC,QAAQ,CAAC;QACpD,IAAI,cAAc,KAAK,EAAE,EAAE,CAAC;YAC1B,cAAc,GAAG,GAAG,CAAC;QACvB,CAAC;QACD,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;YACvB,IAAI,GAAG,CAAC,QAAQ,KAAK,SAAS,EAAE,CAAC;gBAC/B,MAAM,EAAE,GAAG,GAAG,CAAC,QAAQ,CAAC;gBACxB,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC;oBACd,cAAc,GAAG,GAAG,CAAC;gBACvB,CAAC;qBAAM,CAAC;oBACN,cAAc,GAAG,EAAE,CAAC;gBACtB,CAAC;YACH,CAAC;QACH,CAAC;QACD,IAAI,WAAgC,CAAC;QACrC,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE,CAAC;YACvB,IAAI,GAAG,CAAC,OAAO,KAAK,SAAS,EAAE,CAAC;gBAC9B,WAAW,GAAG,GAAG,CAAC,OAAO,CAAC;YAC5B,CAAC;QACH,CAAC;QAED,IAAI,SAAS,GACX,YAAY,CAAC,UAAU,IAAI,YAAY,CAAC,KAAK,IAAI,EAAE,CAAC;QACtD,IAAI,kBAAkB,IAAI,IAAI,EAAE,CAAC;YAC/B,SAAS,GAAG,kBAAkB,CAAC;QACjC,CAAC;aAAM,IAAI,cAAc,IAAI,IAAI,EAAE,CAAC;YAClC,SAAS,GAAG,cAAc,CAAC;QAC7B,CAAC;aAAM,CAAC;YACN,KAAK,IAAI,CAAC,GAAG,IAAI,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC;gBAC7C,MAAM,IAAI,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC;gBACrB,IAAI,IAAI,IAAI,IAAI,EAAE,CAAC;oBACjB,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC;oBACrB,IAAI,CAAC,IAAI,IAAI,IAAI,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;wBAC3C,SAAS,GAAG,CAAC,CAAC;wBACd,MAAM;oBACR,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;QAED,MAAM,YAAY,GAAG,cAAc,CAAC;QACpC,MAAM,iBAAiB,GAAG,mBAAmB,CAAC;QAC9C,MAAM,MAAM,GAAG,aAAa,CAAC,cAAc,CAAC,CAAC;QAC7C,IAAI,SAKS,CAAC;QACd,IAAI,YAAY,KAAK,SAAS,EAAE,CAAC;YAC/B,MAAM,EAAE,GAAG,iBAAiB,IAAI,mBAAmB,CAAC,cAAc,CAAC,CAAC;YACpE,IAAI,EAAE,IAAI,IAAI,EAAE,CAAC;gBACf,SAAS,GAAG,EAAE,OAAO,EAAE,YAAY,EAAE,YAAY,EAAE,EAAE,EAAE,CAAC;YAC1D,CAAC;QACH,CAAC;QACD,MAAM,MAAM,GAAG,WAAW,CAAC,SAAS,EAAE,MAAM,EAAE,SAAS,CAAC,CAAC;QAEzD,IAAI,cAAc,GAAG,cAAc,CAAC;QACpC,IAAI,cAAc,KAAK,EAAE,EAAE,CAAC;YAC1B,cAAc,GAAG,GAAG,CAAC;QACvB,CAAC;QACD,IAAI,SAAS,GAAG,YAAY,CAAC,KAAK,CAAC;QACnC,IAAI,cAAc,KAAK,YAAY,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;YACtD,SAAS,GAAG,eAAe,CAAC,UAAU,EAAE;gBACtC,GAAG,MAAM,CAAC,QAAQ;gBAClB,QAAQ,EAAE,cAAc;aACzB,CAAC,CAAC;QACL,CAAC;QACD,MAAM,eAAe,GAAG,MAAM,IAAI,aAAa,CAAC,cAAc,CAAC,CAAC;QAChE,MAAM,WAAW,GAAG,aAAa,CAAC,MAAM,CAAC,CAAC;QAC1C,MAAM,KAAK,GAAG,eAAe,CAAC,eAAe,EAAE,WAAW,CAAC,CAAC;QAC5D,MAAM,QAAQ,GAAG,YAAY,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;QAChD,MAAM,SAAS,GAAoB;YACjC,aAAa,EAAE,KAAK;YACpB,QAAQ,EAAE,EAAE,GAAG,YAAY,CAAC,QAAQ,EAAE,QAAQ,EAAE,cAAc,EAAE,MAAM,EAAE;YACxE,KAAK,EAAE,SAAS;YAChB,aAAa,EAAE,QAAQ;YACvB,SAAS,EAAE,MAAM;YACjB,KAAK,EAAE,WAAW;YAClB,UAAU,EAAE,KAAK;SACC,CAAC;QACrB,OAAO;YACL,MAAM;YACN,QAAQ,EAAE,cAAc;YACxB,OAAO,EAAE,WAAW;YACpB,SAAS;SACV,CAAC;IACJ,CAAC;IAED,SAAS,gBAAgB;QACvB,cAAc,GAAG,KAAK,CAAC;QACvB,MAAM,IAAI,GAAG,WAAW,CAAC;QACzB,WAAW,GAAG,SAAS,CAAC;QACxB,IAAI,IAAI,IAAI,IAAI,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACtC,OAAO;QACT,CAAC;QACD,MAAM,EACJ,MAAM,EACN,QAAQ,EACR,OAAO,EAAE,WAAW,EACpB,SAAS,GACV,GAAG,mBAAmB,CAAC,IAAI,CAAC,CAAC;QAE9B,kBAAkB,GAAG,SAAS,CAAC;QAC/B,cAAc,GAAG,SAAS,CAAC;QAC3B,cAAc,GAAG,SAAS,CAAC;QAC3B,mBAAmB,GAAG,SAAS,CAAC;QAChC,YAAY,GAAG,SAAS,CAAC;QACzB,MAAM,GAAG,GAAG,EAAE,QAAQ,EAAE,MAAM,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC;QAC3C,IAAI,WAAW;YAAE,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;;YAC7B,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAEvB,IAAI,IAAI,GAAG,QAAQ,CAAC;QACpB,IAAI,SAAS,KAAK,MAAM,EAAE,CAAC;YACzB,IAAI,GAAG,MAAM,CAAC;QAChB,CAAC;aAAM,IAAI,SAAS,KAAK,OAAO,EAAE,CAAC;YACjC,IAAI,GAAG,OAAO,CAAC;QACjB,CAAC;QACD,kBAAkB,EAAE,CAAC;YACnB,IAAI,EAAE,aAAa;YACnB,IAAI;YACJ,KAAK,EAAE,IAAI,CAAC,MAAM;YAClB,QAAQ;YACR,MAAM;YACN,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;SACtB,CAAC,CAAC;QACH,SAAS,GAAG,SAAS,CAAC;IACxB,CAAC;IAKD,SAAS,2BAA2B,CAAC,QAAiB;QACpD,IAAI,CAAC;YACH,MAAM,QAAQ,GAAG,QAAQ,IAAI,YAAY,CAAC,QAAQ,CAAC,QAAQ,CAAC;YAC5D,MAAM,MAAM,GAAG,aAAa,CAAC,QAAQ,CAAC,CAAC;YACvC,MAAM,OAAO,GAAG,kBAAkB,IAAI,cAAc,IAAI,EAAE,CAAC;YAC3D,IAAI,SAKS,CAAC;YACd,IAAI,cAAc,IAAI,IAAI,IAAI,mBAAmB,IAAI,IAAI,EAAE,CAAC;gBAC1D,SAAS,GAAG;oBACV,OAAO,EAAE,cAAc;oBACvB,YAAY,EAAE,mBAAmB;iBAClC,CAAC;YACJ,CAAC;iBAAM,IAAI,cAAc,IAAI,IAAI,IAAI,mBAAmB,IAAI,IAAI,EAAE,CAAC;gBAEjE,IAAI,MAAuC,CAAC;gBAC5C,IAAI,UAIS,CAAC;gBACd,IAAI,QAAQ,KAAK,YAAY,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;oBAChD,IAAI,YAAY,CAAC,KAAK,IAAI,IAAI,EAAE,CAAC;wBAC/B,UAAU,GAAG,YAAY,CAAC,KAAK,CAAC,KAAK,CAAC,MAAa,CAAC;oBACtD,CAAC;gBACH,CAAC;qBAAM,CAAC;oBACN,MAAM,SAAS,GAAG,eAAe,CAAC,UAAU,EAAE;wBAC5C,GAAG,MAAM,CAAC,QAAQ;wBAClB,QAAQ,EAAE,QAAQ;qBACnB,CAAC,CAAC;oBACH,IAAI,SAAS,IAAI,IAAI,EAAE,CAAC;wBACtB,UAAU,GAAG,SAAS,CAAC,KAAK,CAAC,MAAa,CAAC;oBAC7C,CAAC;gBACH,CAAC;gBACD,IAAI,UAAU,IAAI,IAAI,EAAE,CAAC;oBACvB,KAAK,MAAM,IAAI,IAAI,UAAU,EAAE,CAAC;wBAC9B,MAAM,EAAE,GACN,IAGD,CAAC,YAAY,CAAC;wBACf,IAAI,EAAE,IAAI,IAAI,EAAE,CAAC;4BACf,MAAM,KAAK,EAAE,CAAC;4BACd,MAAM,CAAC,MAAM,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,CAAC;wBACnC,CAAC;oBACH,CAAC;gBACH,CAAC;gBACD,IAAI,MAAM,IAAI,IAAI,EAAE,CAAC;oBACnB,SAAS,GAAG;wBACV,OAAO,EAAE,cAAc;wBACvB,YAAY,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,MAAM,EAAE,MAAM,EAAE;qBAClD,CAAC;gBACJ,CAAC;YACH,CAAC;YACD,MAAM,iBAAiB,GAAG,WAAW,CAAC,OAAO,EAAE,MAAM,EAAE,SAAS,CAAC,CAAC;YAClE,MAAM,WAAW,GAAG,QAAQ,KAAK,YAAY,CAAC,QAAQ,CAAC,QAAQ,CAAC;YAChE,IAAI,iBAAiB,KAAK,YAAY,CAAC,SAAS,IAAI,WAAW,EAAE,CAAC;gBAChE,MAAM,KAAK,GAAG,eAAe,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;gBAC/C,MAAM,WAAW,GAAG,aAAa,CAAC,iBAAiB,CAAC,CAAC;gBACrD,YAAY,GAAG;oBACb,GAAG,YAAY;oBACf,QAAQ,EAAE;wBACR,GAAG,YAAY,CAAC,QAAQ;wBACxB,QAAQ,EAAE,QAAQ;wBAClB,MAAM,EAAE,iBAAiB;qBAC1B;oBACD,SAAS,EAAE,iBAAiB;oBAC5B,KAAK,EAAE,WAAW;oBAClB,UAAU,EAAE,KAAK;iBACC,CAAC;YACvB,CAAC;QACH,CAAC;QAAC,MAAM,CAAC;QAET,CAAC;IACH,CAAC;IAED,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,KAAK,IAAI,CAAC;IAC1C,MAAM,OAAO,GAA4B;QACvC,OAAO;QACP,GAAG;YACD,OAAO,YAAY,CAAC;QACtB,CAAC;QACD,mBAAmB,CAAC,QAAgB;YAClC,OAAO,mBAAmB,CAAC,QAAQ,CAAC,CAAC;QACvC,CAAC;QACD,yBAAyB;YACvB,OAAO,mBAAmB,CAAC,YAAY,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;QAC7D,CAAC;QACD,WAAW,CAAC,QAAgB;YAC1B,MAAM,OAAO,GAAG,eAAe,CAAC,UAAU,EAAE;gBAC1C,GAAG,MAAM,CAAC,QAAQ;gBAClB,QAAQ;aACT,CAAC,CAAC;YACH,IAAI,OAAO,IAAI,IAAI;gBAAE,OAAO;YAC5B,OAAO,CAAC,KAAK,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,EAAE,YAAY,EAAE,EAAE,EAAE;gBAChD,IAAI,YAAY,IAAI,IAAI;oBAAE,OAAO;gBAEjC,YAAY,CAAC,IAAI,EAAE,CAAC;YACtB,CAAC,CAAC,CAAC;QACL,CAAC;QACD,OAAO,CAAC,QAAgB;YACtB,MAAM,OAAO,GAAG,eAAe,CAAC,UAAU,EAAE;gBAC1C,GAAG,MAAM,CAAC,QAAQ;gBAClB,QAAQ;aACT,CAAC,CAAC;YACH,YAAY,CAAC,OAAO,CAAC,CAAC;QACxB,CAAC;QACD,SAAS,CAAC,QAAgC;YACxC,MAAM,IAAI,CAAC,CAAC;YACZ,MAAM,EAAE,GAAG,MAAM,CAAC;YAElB,SAAS,OAAO;gBACd,WAAW,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;YACzB,CAAC;YACD,WAAW,CAAC,GAAG,CAAC,EAAE,EAAE,QAAQ,CAAC,CAAC;YAC9B,OAAO,OAAO,CAAC;QACjB,CAAC;QACD,QAAQ,CAAC,EACP,QAAQ,EACR,KAAK,EACL,OAAO,EACP,OAAO,EACP,YAAY,EACZ,KAAK,EACL,SAAS,EACT,OAAO,GAUR;YACC,IAAI,CAAC;gBACH,MAAM,gBAAgB,GAAG,OAAO,KAAK,IAAI,CAAC;gBAC1C,MAAM,UAAU,GACd,KAAK,KAAK,IAAI;oBACd,CAAC,SAAS,IAAI,SAAS,KAAK,IAAI,IAAI,KAAK,KAAK,KAAK,CAAC,CAAC;gBACvD,IAAI,UAAU,EAAE,CAAC;oBACf,MAAM,GAAG,GAAe,EAAE,CAAC;oBAC3B,IAAI,QAAQ,KAAK,SAAS;wBAAE,GAAG,CAAC,QAAQ,GAAG,QAAQ,CAAC;oBACpD,IAAI,KAAK,KAAK,SAAS;wBAAE,GAAG,CAAC,KAAK,GAAG,KAAK,CAAC;oBAC3C,IAAI,OAAO,KAAK,SAAS;wBAAE,GAAG,CAAC,OAAO,GAAG,OAAO,CAAC;oBACjD,IAAI,OAAO,KAAK,SAAS;wBAAE,GAAG,CAAC,OAAO,GAAG,OAAO,CAAC;oBACjD,IAAI,YAAY,KAAK,SAAS;wBAAE,GAAG,CAAC,YAAY,GAAG,YAAY,CAAC;oBAChE,IAAI,gBAAgB;wBAAE,GAAG,CAAC,OAAO,GAAG,IAAI,CAAC;oBACzC,WAAW,KAAK,EAAE,CAAC;oBACnB,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;oBACtB,IAAI,QAA2B,CAAC;oBAChC,IAAI,KAAK,KAAK,IAAI;wBAAE,QAAQ,GAAG,QAAQ,CAAC;;wBACnC,QAAQ,GAAG,MAAM,CAAC;oBACvB,IAAI,SAAS,IAAI,IAAI;wBAAE,SAAS,GAAG,QAAQ,CAAC;yBACvC,IAAI,SAAS,KAAK,QAAQ;wBAAE,SAAS,GAAG,OAAO,CAAC;oBACrD,IAAI,KAAK,IAAI,IAAI,EAAE,CAAC;wBAClB,IAAI,gBAAgB,EAAE,CAAC;4BACrB,MAAM,IAAI,GACR,kBAAkB;gCAClB,cAAc;gCACd,YAAY,CAAC,UAAU;gCACvB,YAAY,CAAC,KAAK;gCAClB,EAAE,CAAC;4BACL,kBAAkB,GAAG,EAAE,GAAG,IAAI,EAAE,GAAG,KAAK,EAAE,CAAC;wBAC7C,CAAC;6BAAM,CAAC;4BACN,kBAAkB,GAAG,EAAE,GAAG,KAAK,EAAE,CAAC;wBACpC,CAAC;wBACD,cAAc,GAAG,KAAK,CAAC;oBACzB,CAAC;oBACD,IAAI,OAAO,IAAI,IAAI,EAAE,CAAC;wBACpB,cAAc,GAAG,OAAO,CAAC;wBACzB,IAAI,YAAY,IAAI,IAAI;4BAAE,mBAAmB,GAAG,YAAY,CAAC;oBAC/D,CAAC;oBACD,IAAI,CAAC,cAAc,EAAE,CAAC;wBACpB,cAAc,GAAG,IAAI,CAAC;wBAEtB,OAAO,CAAC,OAAO,EAAE,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;oBAC3C,CAAC;oBACD,2BAA2B,CAAC,QAAQ,CAAC,CAAC;oBACtC,WAAW,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;wBACzB,EAAE,CAAC,YAAY,CAAC,CAAC;oBACnB,CAAC,CAAC,CAAC;oBACH,OAAO;gBACT,CAAC;gBAED,IAAI,IAAqC,CAAC;gBAC1C,IAAI,gBAAgB,EAAE,CAAC;oBACrB,IAAI,GAAG,EAAE,GAAG,CAAC,YAAY,CAAC,UAAU,IAAI,YAAY,CAAC,KAAK,IAAI,EAAE,CAAC,EAAE,CAAC;gBACtE,CAAC;gBACD,IAAI,KAAK,IAAI,IAAI,EAAE,CAAC;oBAClB,IAAI,IAAI,IAAI,IAAI;wBAAE,MAAM,CAAC,MAAM,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;;wBACxC,IAAI,GAAG,KAAK,CAAC;gBACpB,CAAC;gBACD,IAAI,KAAK,YAAY,CAAC,UAAU,IAAI,YAAY,CAAC,KAAK,IAAI,EAAE,CAAC;gBAC7D,MAAM,UAAU,GAAG,QAAQ,IAAI,YAAY,CAAC,QAAQ,CAAC,QAAQ,CAAC;gBAC9D,MAAM,MAAM,GAAG,aAAa,CAAC,UAAU,CAAC,CAAC;gBACzC,IAAI,SAKS,CAAC;gBACd,IAAI,OAAO,IAAI,IAAI,EAAE,CAAC;oBACpB,MAAM,EAAE,GAAG,YAAY,IAAI,mBAAmB,CAAC,UAAU,CAAC,CAAC;oBAC3D,IAAI,EAAE,IAAI,IAAI;wBAAE,SAAS,GAAG,EAAE,OAAO,EAAE,YAAY,EAAE,EAAE,EAAE,CAAC;gBAC5D,CAAC;gBACD,MAAM,MAAM,GAAG,WAAW,CAAC,IAAI,EAAE,MAAM,EAAE,SAAS,CAAC,CAAC;gBACpD,IAAI,cAAc,GAAG,UAAU,CAAC;gBAChC,IAAI,cAAc,KAAK,EAAE;oBAAE,cAAc,GAAG,GAAG,CAAC;gBAChD,IAAI,SAAS,GAAG,YAAY,CAAC,KAAK,CAAC;gBACnC,IAAI,cAAc,KAAK,YAAY,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;oBACtD,SAAS,GAAG,eAAe,CAAC,UAAU,EAAE;wBACtC,GAAG,MAAM,CAAC,QAAQ;wBAClB,QAAQ,EAAE,cAAc;qBACzB,CAAC,CAAC;gBACL,CAAC;gBACD,MAAM,eAAe,GAAG,MAAM,IAAI,aAAa,CAAC,cAAc,CAAC,CAAC;gBAChE,MAAM,WAAW,GAAG,aAAa,CAAC,MAAM,CAAC,CAAC;gBAC1C,MAAM,KAAK,GAAG,eAAe,CAAC,eAAe,EAAE,WAAW,CAAC,CAAC;gBAC5D,MAAM,QAAQ,GAAG,YAAY,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;gBAChD,YAAY,GAAG;oBACb,aAAa,EAAE,KAAK;oBACpB,QAAQ,EAAE;wBACR,GAAG,YAAY,CAAC,QAAQ;wBACxB,QAAQ,EAAE,cAAc;wBACxB,MAAM;qBACP;oBACD,KAAK,EAAE,SAAS;oBAChB,aAAa,EAAE,QAAQ;oBACvB,SAAS,EAAE,MAAM;oBACjB,KAAK,EAAE,WAAW;oBAClB,UAAU,EAAE,KAAK;iBACC,CAAC;gBACrB,IAAI,aAAa,GAAG,cAAc,CAAC;gBACnC,IAAI,aAAa,KAAK,EAAE;oBAAE,aAAa,GAAG,GAAG,CAAC;gBAC9C,MAAM,WAAW,GAAG,EAAE,QAAQ,EAAE,aAAa,EAAE,MAAM,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC;gBAClE,IAAI,OAAO;oBAAE,OAAO,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;;oBACjC,OAAO,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;gBAC/B,kBAAkB,EAAE,CAAC;oBACnB,IAAI,EAAE,UAAU;oBAChB,IAAI,EAAE,WAAW;oBACjB,KAAK,EAAE,CAAC;oBACR,QAAQ,EAAE,aAAa;oBACvB,MAAM;oBACN,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;iBACtB,CAAC,CAAC;YACL,CAAC;YAAC,MAAM,CAAC;YAET,CAAC;QACH,CAAC;KACF,CAAC;IAGF,IAAI,CAAC,cAAc,IAAI,aAAa,CAAC,IAAI,OAAO,MAAM,KAAK,WAAW,EAAE,CAAC;QAEvE,CAAC,KAAK,IAAI,EAAE;YACV,IAAI,CAAC;gBACH,MAAM,GAAG,GAAQ,MAAM,MAAM,CAAC,eAAe,CAAC,CAAC;gBAE/C,IAAI,GAAG,IAAI,IAAI,IAAI,OAAO,GAAG,CAAC,kBAAkB,KAAK,UAAU,EAAE,CAAC;oBAGhE,kBAAkB,GAAG,GAAG,CAAC,kBAAkB,CAAC;wBAC1C,OAAO;wBACP,OAAO,EAAE;4BACP,MAAM,EAAE,cAAc;4BACtB,KAAK,EAAE,aAAa;4BACpB,QAAQ,EAAE,gBAAgB;yBAC3B;qBACF,CAAC,CAAC;gBACL,CAAC;YACH,CAAC;YAAC,MAAM,CAAC;YAET,CAAC;QACH,CAAC,CAAC,EAAE,CAAC;IACP,CAAC;IAED,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,CAAC;AAC9B,CAAC","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"]}