@plumile/router 0.1.11 → 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.
- package/README.md +755 -1
- package/lib/esm/builder.d.ts.map +1 -1
- package/lib/esm/builder.js +10 -3
- package/lib/esm/eslint-rules/index.d.ts +2 -0
- package/lib/esm/eslint-rules/index.d.ts.map +1 -0
- package/lib/esm/eslint-rules/index.js +2 -0
- package/lib/esm/eslint-rules/no-direct-window-location-search.d.ts +4 -0
- package/lib/esm/eslint-rules/no-direct-window-location-search.d.ts.map +1 -0
- package/lib/esm/eslint-rules/no-direct-window-location-search.js +48 -0
- package/lib/esm/history/BrowserHistory.d.ts.map +1 -1
- package/lib/esm/history/BrowserHistory.js +4 -2
- package/lib/esm/index.d.ts +5 -0
- package/lib/esm/index.d.ts.map +1 -1
- package/lib/esm/index.js +6 -1
- package/lib/esm/routing/Link.d.ts +1 -0
- package/lib/esm/routing/Link.d.ts.map +1 -1
- package/lib/esm/routing/Link.js +35 -4
- package/lib/esm/routing/RouteComponentWrapper.d.ts.map +1 -1
- package/lib/esm/routing/RouteComponentWrapper.js +7 -2
- package/lib/esm/routing/createRouter.d.ts +8 -1
- package/lib/esm/routing/createRouter.d.ts.map +1 -1
- package/lib/esm/routing/createRouter.js +540 -17
- package/lib/esm/routing/devtools.d.ts +20 -0
- package/lib/esm/routing/devtools.d.ts.map +1 -0
- package/lib/esm/routing/devtools.js +678 -0
- package/lib/esm/routing/filters.d.ts +97 -0
- package/lib/esm/routing/filters.d.ts.map +1 -0
- package/lib/esm/routing/filters.js +557 -0
- package/lib/esm/routing/index.d.ts +10 -0
- package/lib/esm/routing/index.d.ts.map +1 -1
- package/lib/esm/routing/index.js +11 -1
- package/lib/esm/routing/useActiveFilters.d.ts +9 -0
- package/lib/esm/routing/useActiveFilters.d.ts.map +1 -0
- package/lib/esm/routing/useActiveFilters.js +38 -0
- package/lib/esm/routing/useFilterState.d.ts +10 -0
- package/lib/esm/routing/useFilterState.d.ts.map +1 -0
- package/lib/esm/routing/useFilterState.js +14 -0
- package/lib/esm/routing/useNavigate.d.ts +13 -0
- package/lib/esm/routing/useNavigate.d.ts.map +1 -0
- package/lib/esm/routing/useNavigate.js +11 -0
- package/lib/esm/routing/useNavigateWithQuery.d.ts +15 -0
- package/lib/esm/routing/useNavigateWithQuery.d.ts.map +1 -0
- package/lib/esm/routing/useNavigateWithQuery.js +95 -0
- package/lib/esm/routing/useQuery.d.ts +2 -0
- package/lib/esm/routing/useQuery.d.ts.map +1 -0
- package/lib/esm/routing/useQuery.js +9 -0
- package/lib/esm/routing/useQueryObject.d.ts +18 -0
- package/lib/esm/routing/useQueryObject.d.ts.map +1 -0
- package/lib/esm/routing/useQueryObject.js +107 -0
- package/lib/esm/routing/useQueryState.d.ts +13 -0
- package/lib/esm/routing/useQueryState.d.ts.map +1 -0
- package/lib/esm/routing/useQueryState.js +80 -0
- package/lib/esm/routing/useStableRefEquality.d.ts +5 -0
- package/lib/esm/routing/useStableRefEquality.d.ts.map +1 -0
- package/lib/esm/routing/useStableRefEquality.js +47 -0
- package/lib/esm/routing/useTypedQuery.d.ts +2 -0
- package/lib/esm/routing/useTypedQuery.d.ts.map +1 -0
- package/lib/esm/routing/useTypedQuery.js +36 -0
- package/lib/esm/tools/buildSearch.d.ts +12 -0
- package/lib/esm/tools/buildSearch.d.ts.map +1 -0
- package/lib/esm/tools/buildSearch.js +264 -0
- package/lib/esm/tools/query-dsl.d.ts +28 -0
- package/lib/esm/tools/query-dsl.d.ts.map +1 -0
- package/lib/esm/tools/query-dsl.js +250 -0
- package/lib/esm/tools/query.d.ts +2 -0
- package/lib/esm/tools/query.d.ts.map +1 -0
- package/lib/esm/tools/query.js +43 -0
- package/lib/esm/tools.d.ts +2 -2
- package/lib/esm/tools.d.ts.map +1 -1
- package/lib/esm/tools.js +3 -2
- package/lib/esm/type-tests/query-infer.test-d.d.ts +2 -0
- package/lib/esm/type-tests/query-infer.test-d.d.ts.map +1 -0
- package/lib/esm/type-tests/query-infer.test-d.js +49 -0
- package/lib/esm/types.d.ts +47 -4
- package/lib/esm/types.d.ts.map +1 -1
- package/lib/esm/types.js +1 -1
- package/lib/tsconfig.esm.tsbuildinfo +1 -1
- package/lib/types/builder.d.ts.map +1 -1
- package/lib/types/eslint-rules/index.d.ts +2 -0
- package/lib/types/eslint-rules/index.d.ts.map +1 -0
- package/lib/types/eslint-rules/no-direct-window-location-search.d.ts +4 -0
- package/lib/types/eslint-rules/no-direct-window-location-search.d.ts.map +1 -0
- package/lib/types/history/BrowserHistory.d.ts.map +1 -1
- package/lib/types/index.d.ts +5 -0
- package/lib/types/index.d.ts.map +1 -1
- package/lib/types/routing/Link.d.ts +1 -0
- package/lib/types/routing/Link.d.ts.map +1 -1
- package/lib/types/routing/RouteComponentWrapper.d.ts.map +1 -1
- package/lib/types/routing/createRouter.d.ts +8 -1
- package/lib/types/routing/createRouter.d.ts.map +1 -1
- package/lib/types/routing/devtools.d.ts +20 -0
- package/lib/types/routing/devtools.d.ts.map +1 -0
- package/lib/types/routing/filters.d.ts +97 -0
- package/lib/types/routing/filters.d.ts.map +1 -0
- package/lib/types/routing/index.d.ts +10 -0
- package/lib/types/routing/index.d.ts.map +1 -1
- package/lib/types/routing/useActiveFilters.d.ts +9 -0
- package/lib/types/routing/useActiveFilters.d.ts.map +1 -0
- package/lib/types/routing/useFilterState.d.ts +10 -0
- package/lib/types/routing/useFilterState.d.ts.map +1 -0
- package/lib/types/routing/useNavigate.d.ts +13 -0
- package/lib/types/routing/useNavigate.d.ts.map +1 -0
- package/lib/types/routing/useNavigateWithQuery.d.ts +15 -0
- package/lib/types/routing/useNavigateWithQuery.d.ts.map +1 -0
- package/lib/types/routing/useQuery.d.ts +2 -0
- package/lib/types/routing/useQuery.d.ts.map +1 -0
- package/lib/types/routing/useQueryObject.d.ts +18 -0
- package/lib/types/routing/useQueryObject.d.ts.map +1 -0
- package/lib/types/routing/useQueryState.d.ts +13 -0
- package/lib/types/routing/useQueryState.d.ts.map +1 -0
- package/lib/types/routing/useStableRefEquality.d.ts +5 -0
- package/lib/types/routing/useStableRefEquality.d.ts.map +1 -0
- package/lib/types/routing/useTypedQuery.d.ts +2 -0
- package/lib/types/routing/useTypedQuery.d.ts.map +1 -0
- package/lib/types/tools/buildSearch.d.ts +12 -0
- package/lib/types/tools/buildSearch.d.ts.map +1 -0
- package/lib/types/tools/query-dsl.d.ts +28 -0
- package/lib/types/tools/query-dsl.d.ts.map +1 -0
- package/lib/types/tools/query.d.ts +2 -0
- package/lib/types/tools/query.d.ts.map +1 -0
- package/lib/types/tools.d.ts +2 -2
- package/lib/types/tools.d.ts.map +1 -1
- package/lib/types/type-tests/query-infer.test-d.d.ts +2 -0
- package/lib/types/type-tests/query-infer.test-d.d.ts.map +1 -0
- package/lib/types/types.d.ts +47 -4
- package/lib/types/types.d.ts.map +1 -1
- package/package.json +7 -7
|
@@ -1,55 +1,433 @@
|
|
|
1
1
|
import { buildRoutes } from '../builder.js';
|
|
2
2
|
import { BrowserHistory } from '../history/index.js';
|
|
3
3
|
import { getMatchedRoute, prepareMatch } from '../tools.js';
|
|
4
|
-
|
|
4
|
+
import { parseTypedQuery } from '../tools/query-dsl.js';
|
|
5
|
+
import { parseRawQuery } from '../tools/query.js';
|
|
6
|
+
import { buildSearch } from '../tools/buildSearch.js';
|
|
7
|
+
export default function createRouter(routes, opts = {}) {
|
|
8
|
+
let devtoolsGlobal;
|
|
9
|
+
let devtoolsPanel = false;
|
|
10
|
+
let devtoolsShortcut = 'Alt+Shift+R';
|
|
11
|
+
if (typeof opts.devtools === 'object') {
|
|
12
|
+
const baseEnabled = (() => {
|
|
13
|
+
try {
|
|
14
|
+
return (typeof process !== 'undefined' &&
|
|
15
|
+
process.env.NODE_ENV !== 'production');
|
|
16
|
+
}
|
|
17
|
+
catch {
|
|
18
|
+
return false;
|
|
19
|
+
}
|
|
20
|
+
})();
|
|
21
|
+
devtoolsGlobal = opts.devtools.global ?? baseEnabled;
|
|
22
|
+
devtoolsPanel = !!opts.devtools.panel;
|
|
23
|
+
if (typeof opts.devtools.shortcut === 'string' &&
|
|
24
|
+
opts.devtools.shortcut.trim() !== '') {
|
|
25
|
+
devtoolsShortcut = opts.devtools.shortcut;
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
else {
|
|
29
|
+
const enabled = (() => {
|
|
30
|
+
if (typeof opts.devtools === 'boolean')
|
|
31
|
+
return opts.devtools;
|
|
32
|
+
try {
|
|
33
|
+
return (typeof process !== 'undefined' &&
|
|
34
|
+
process.env.NODE_ENV !== 'production');
|
|
35
|
+
}
|
|
36
|
+
catch {
|
|
37
|
+
return false;
|
|
38
|
+
}
|
|
39
|
+
})();
|
|
40
|
+
devtoolsGlobal = enabled;
|
|
41
|
+
}
|
|
5
42
|
const history = new BrowserHistory();
|
|
6
43
|
const flatRoutes = buildRoutes(routes);
|
|
7
44
|
const route = getMatchedRoute(flatRoutes, history.location);
|
|
8
|
-
const
|
|
45
|
+
const initialRawQuery = parseRawQuery(history.location.search);
|
|
46
|
+
let initialSchema;
|
|
47
|
+
if (route != null) {
|
|
48
|
+
const collected = [];
|
|
49
|
+
const collectedFilters = [];
|
|
50
|
+
for (const r of route.route.routes) {
|
|
51
|
+
const maybe = r.query;
|
|
52
|
+
if (maybe != null) {
|
|
53
|
+
collected.push(maybe);
|
|
54
|
+
}
|
|
55
|
+
const maybeFilter = r.filterSchema;
|
|
56
|
+
if (maybeFilter != null) {
|
|
57
|
+
collectedFilters.push(maybeFilter);
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
if (collected.length > 0) {
|
|
61
|
+
initialSchema = collected.at(-1);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
const initialTypedQuery = parseTypedQuery(initialSchema, initialRawQuery);
|
|
65
|
+
const preparedMatch = prepareMatch(route, initialTypedQuery);
|
|
9
66
|
let currentEntry = {
|
|
10
67
|
forceRerender: false,
|
|
11
68
|
location: history.location,
|
|
12
69
|
route,
|
|
13
70
|
preparedMatch,
|
|
71
|
+
rawSearch: history.location.search,
|
|
72
|
+
query: initialRawQuery,
|
|
73
|
+
typedQuery: initialTypedQuery,
|
|
14
74
|
};
|
|
75
|
+
const initTyped = currentEntry.typedQuery;
|
|
76
|
+
let initPage;
|
|
77
|
+
if (initTyped != null) {
|
|
78
|
+
initPage = initTyped.page;
|
|
79
|
+
}
|
|
80
|
+
if (typeof initPage === 'number' &&
|
|
81
|
+
initPage < 1 &&
|
|
82
|
+
initialSchema != null &&
|
|
83
|
+
initTyped != null) {
|
|
84
|
+
const norm = { ...initTyped, page: 1 };
|
|
85
|
+
const normalizedSearch = buildSearch(norm, initialSchema);
|
|
86
|
+
if (normalizedSearch !== history.location.search) {
|
|
87
|
+
history.set({
|
|
88
|
+
pathname: history.location.pathname,
|
|
89
|
+
search: normalizedSearch,
|
|
90
|
+
hash: '',
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
}
|
|
15
94
|
let nextId = 0;
|
|
16
95
|
const subscribers = new Map();
|
|
17
96
|
const cleanup = history.subscribe((location, forceRerender) => {
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
97
|
+
const samePathname = location.pathname === currentEntry.location.pathname;
|
|
98
|
+
const sameSearch = location.search === currentEntry.rawSearch;
|
|
99
|
+
if (!forceRerender && samePathname && sameSearch) {
|
|
21
100
|
return;
|
|
22
101
|
}
|
|
23
|
-
|
|
24
|
-
|
|
102
|
+
let nextPreparedMatch = currentEntry.preparedMatch;
|
|
103
|
+
let nextRoute = currentEntry.route;
|
|
104
|
+
if (!samePathname) {
|
|
105
|
+
nextRoute = getMatchedRoute(flatRoutes, history.location);
|
|
106
|
+
}
|
|
107
|
+
const query = parseRawQuery(location.search);
|
|
108
|
+
let schema;
|
|
109
|
+
if (nextRoute != null) {
|
|
110
|
+
const collected = [];
|
|
111
|
+
const collectedFilters = [];
|
|
112
|
+
for (const r of nextRoute.route.routes) {
|
|
113
|
+
const maybe = r.query;
|
|
114
|
+
if (maybe != null) {
|
|
115
|
+
collected.push(maybe);
|
|
116
|
+
}
|
|
117
|
+
const maybeFilter = r.filterSchema;
|
|
118
|
+
if (maybeFilter != null) {
|
|
119
|
+
collectedFilters.push(maybeFilter);
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
if (collected.length > 0) {
|
|
123
|
+
schema = collected.at(-1);
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
let typedQuery = parseTypedQuery(schema, query);
|
|
127
|
+
let normalized = false;
|
|
128
|
+
const pg = typedQuery.page;
|
|
129
|
+
if (typeof pg === 'number' && pg < 1) {
|
|
130
|
+
const clone = { ...typedQuery, page: 1 };
|
|
131
|
+
typedQuery = clone;
|
|
132
|
+
normalized = true;
|
|
133
|
+
}
|
|
134
|
+
if (!samePathname) {
|
|
135
|
+
nextPreparedMatch = prepareMatch(nextRoute, typedQuery);
|
|
136
|
+
}
|
|
137
|
+
else if (!sameSearch) {
|
|
138
|
+
nextPreparedMatch = prepareMatch(nextRoute, typedQuery);
|
|
139
|
+
}
|
|
25
140
|
const nextEntry = {
|
|
26
|
-
forceRerender,
|
|
141
|
+
forceRerender: forceRerender || (samePathname && !sameSearch),
|
|
27
142
|
location,
|
|
28
|
-
route,
|
|
29
|
-
preparedMatch,
|
|
143
|
+
route: nextRoute,
|
|
144
|
+
preparedMatch: nextPreparedMatch,
|
|
145
|
+
rawSearch: location.search,
|
|
146
|
+
query,
|
|
147
|
+
typedQuery,
|
|
30
148
|
};
|
|
149
|
+
if (normalized && schema != null) {
|
|
150
|
+
const normalizedSearch = buildSearch(typedQuery, schema);
|
|
151
|
+
if (normalizedSearch !== location.search) {
|
|
152
|
+
history.set({
|
|
153
|
+
pathname: location.pathname,
|
|
154
|
+
search: normalizedSearch,
|
|
155
|
+
hash: '',
|
|
156
|
+
});
|
|
157
|
+
return;
|
|
158
|
+
}
|
|
159
|
+
}
|
|
31
160
|
currentEntry = nextEntry;
|
|
32
161
|
subscribers.forEach((callback) => {
|
|
33
162
|
callback(nextEntry);
|
|
34
163
|
});
|
|
35
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;
|
|
36
410
|
const context = {
|
|
37
411
|
history,
|
|
38
412
|
get() {
|
|
39
413
|
return currentEntry;
|
|
40
414
|
},
|
|
415
|
+
resolveFilterSchema(pathname) {
|
|
416
|
+
return resolveFilterSchema(pathname);
|
|
417
|
+
},
|
|
418
|
+
currentMergedFilterSchema() {
|
|
419
|
+
return resolveFilterSchema(currentEntry.location.pathname);
|
|
420
|
+
},
|
|
41
421
|
preloadCode(pathname) {
|
|
42
422
|
const matches = getMatchedRoute(flatRoutes, {
|
|
43
423
|
...window.location,
|
|
44
424
|
pathname,
|
|
45
425
|
});
|
|
46
|
-
if (matches == null)
|
|
426
|
+
if (matches == null)
|
|
47
427
|
return;
|
|
48
|
-
}
|
|
49
428
|
matches.route.routes.forEach(({ resourcePage }) => {
|
|
50
|
-
if (resourcePage == null)
|
|
429
|
+
if (resourcePage == null)
|
|
51
430
|
return;
|
|
52
|
-
}
|
|
53
431
|
resourcePage.load();
|
|
54
432
|
});
|
|
55
433
|
},
|
|
@@ -63,13 +441,158 @@ export default function createRouter(routes) {
|
|
|
63
441
|
subscribe(callback) {
|
|
64
442
|
nextId += 1;
|
|
65
443
|
const id = nextId;
|
|
66
|
-
|
|
444
|
+
function dispose() {
|
|
67
445
|
subscribers.delete(id);
|
|
68
|
-
}
|
|
446
|
+
}
|
|
69
447
|
subscribers.set(id, callback);
|
|
70
448
|
return dispose;
|
|
71
449
|
},
|
|
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
|
+
});
|
|
507
|
+
return;
|
|
508
|
+
}
|
|
509
|
+
let base;
|
|
510
|
+
if (effectiveInherit) {
|
|
511
|
+
base = { ...(currentEntry.typedQuery ?? currentEntry.query ?? {}) };
|
|
512
|
+
}
|
|
513
|
+
if (query != null) {
|
|
514
|
+
if (base != null)
|
|
515
|
+
Object.assign(base, query);
|
|
516
|
+
else
|
|
517
|
+
base = query;
|
|
518
|
+
}
|
|
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(),
|
|
571
|
+
});
|
|
572
|
+
}
|
|
573
|
+
catch {
|
|
574
|
+
}
|
|
575
|
+
},
|
|
72
576
|
};
|
|
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
|
+
},
|
|
589
|
+
});
|
|
590
|
+
}
|
|
591
|
+
}
|
|
592
|
+
catch {
|
|
593
|
+
}
|
|
594
|
+
})();
|
|
595
|
+
}
|
|
73
596
|
return { cleanup, context };
|
|
74
597
|
}
|
|
75
|
-
//# 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;AAkD5D,MAAM,CAAC,OAAO,UAAU,YAAY,CAClC,MAAyB;IAGzB,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,aAAa,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC;IAC1C,IAAI,YAAY,GAAoB;QAClC,aAAa,EAAE,KAAK;QACpB,QAAQ,EAAE,OAAO,CAAC,QAAQ;QAC1B,KAAK;QACL,aAAa;KACd,CAAC;IAGF,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,IAAI,aAAa,EAAE,CAAC;QAEpB,CAAC;aAAM,IAAI,QAAQ,CAAC,QAAQ,KAAK,YAAY,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;YAEhE,OAAO;QACT,CAAC;QAGD,MAAM,KAAK,GAAG,eAAe,CAAC,UAAU,EAAE,OAAO,CAAC,QAAQ,CAAC,CAAC;QAC5D,MAAM,aAAa,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC;QAC1C,MAAM,SAAS,GAAoB;YACjC,aAAa;YACb,QAAQ;YACR,KAAK;YACL,aAAa;SACd,CAAC;QAGF,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;IAGH,MAAM,OAAO,GAA4B;QACvC,OAAO;QACP,GAAG;YACD,OAAO,YAAY,CAAC;QACtB,CAAC;QACD,WAAW,CAAC,QAAQ;YAElB,MAAM,OAAO,GAAG,eAAe,CAAC,UAAU,EAAE;gBAC1C,GAAG,MAAM,CAAC,QAAQ;gBAClB,QAAQ;aACT,CAAC,CAAC;YAEH,IAAI,OAAO,IAAI,IAAI,EAAE,CAAC;gBACpB,OAAO;YACT,CAAC;YAGD,OAAO,CAAC,KAAK,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,EAAE,YAAY,EAAE,EAAE,EAAE;gBAChD,IAAI,YAAY,IAAI,IAAI,EAAE,CAAC;oBACzB,OAAO;gBACT,CAAC;gBAED,YAAY,CAAC,IAAI,EAAE,CAAC;YACtB,CAAC,CAAC,CAAC;QACL,CAAC;QACD,OAAO,CAAC,QAAQ;YAEd,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,QAAQ;YAEhB,MAAM,IAAI,CAAC,CAAC;YACZ,MAAM,EAAE,GAAG,MAAM,CAAC;YAElB,MAAM,OAAO,GAAG,GAAG,EAAE;gBACnB,WAAW,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;YACzB,CAAC,CAAC;YACF,WAAW,CAAC,GAAG,CAAC,EAAE,EAAE,QAAQ,CAAC,CAAC;YAC9B,OAAO,OAAO,CAAC;QACjB,CAAC;KACF,CAAC;IAGF,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 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 * @example\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): CreateRouterReturn {\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 preparedMatch = prepareMatch(route);\n  let currentEntry: RouteEntry<any> = {\n    forceRerender: false,\n    location: history.location,\n    route,\n    preparedMatch,\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    if (forceRerender) {\n      // Force re-render even if pathname hasn't changed\n    } else if (location.pathname === currentEntry.location.pathname) {\n      // Skip if pathname hasn't changed (avoid unnecessary work)\n      return;\n    }\n\n    // Find matching route and prepare it\n    const route = getMatchedRoute(flatRoutes, history.location);\n    const preparedMatch = prepareMatch(route);\n    const nextEntry: RouteEntry<any> = {\n      forceRerender,\n      location,\n      route,\n      preparedMatch,\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  };\n\n  // Return both the context object and a cleanup function\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"]}
|