@plumile/router 0.1.52 → 0.1.54
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/ResourcePage.d.ts +12 -0
- package/lib/ResourcePage.d.ts.map +1 -0
- package/lib/ResourcePage.js +38 -0
- package/lib/asyncResource.d.ts +8 -0
- package/lib/asyncResource.d.ts.map +1 -0
- package/lib/asyncResource.js +101 -0
- package/lib/builder.d.ts +13 -0
- package/lib/builder.d.ts.map +1 -0
- package/lib/builder.js +72 -0
- package/lib/errors/HttpRedirect.d.ts +6 -0
- package/lib/errors/HttpRedirect.d.ts.map +1 -0
- package/lib/errors/HttpRedirect.js +11 -0
- package/lib/errors/index.d.ts +2 -0
- package/lib/errors/index.d.ts.map +1 -0
- package/lib/errors/index.js +2 -0
- package/lib/eslint-rules/index.d.ts +2 -0
- package/lib/eslint-rules/index.d.ts.map +1 -0
- package/lib/eslint-rules/index.js +2 -0
- package/lib/eslint-rules/no-direct-window-location-search.d.ts +4 -0
- package/lib/eslint-rules/no-direct-window-location-search.d.ts.map +1 -0
- package/lib/eslint-rules/no-direct-window-location-search.js +48 -0
- package/lib/history/BrowserHistory.d.ts +21 -0
- package/lib/history/BrowserHistory.d.ts.map +1 -0
- package/lib/history/BrowserHistory.js +139 -0
- package/lib/history/index.d.ts +3 -0
- package/lib/history/index.d.ts.map +1 -0
- package/lib/history/index.js +2 -0
- package/lib/history/types.d.ts +19 -0
- package/lib/history/types.d.ts.map +1 -0
- package/lib/history/types.js +2 -0
- package/lib/index.d.ts +10 -0
- package/lib/index.d.ts.map +1 -0
- package/lib/index.js +9 -0
- package/lib/instrumentation/Instrumentation.d.ts +90 -0
- package/lib/instrumentation/Instrumentation.d.ts.map +1 -0
- package/lib/instrumentation/Instrumentation.js +59 -0
- package/lib/instrumentation/adapters/devtoolsBridge.d.ts +14 -0
- package/lib/instrumentation/adapters/devtoolsBridge.d.ts.map +1 -0
- package/lib/instrumentation/adapters/devtoolsBridge.js +133 -0
- package/lib/instrumentation/adapters/logger.d.ts +10 -0
- package/lib/instrumentation/adapters/logger.d.ts.map +1 -0
- package/lib/instrumentation/adapters/logger.js +19 -0
- package/lib/instrumentation/index.d.ts +4 -0
- package/lib/instrumentation/index.d.ts.map +1 -0
- package/lib/instrumentation/index.js +4 -0
- package/lib/prepareResource.d.ts +4 -0
- package/lib/prepareResource.d.ts.map +1 -0
- package/lib/prepareResource.js +11 -0
- package/lib/routing/Link.d.ts +23 -0
- package/lib/routing/Link.d.ts.map +1 -0
- package/lib/routing/Link.js +158 -0
- package/lib/routing/RouteComponent.d.ts +8 -0
- package/lib/routing/RouteComponent.d.ts.map +1 -0
- package/lib/routing/RouteComponent.js +20 -0
- package/lib/routing/RouteComponentWrapper.d.ts +11 -0
- package/lib/routing/RouteComponentWrapper.d.ts.map +1 -0
- package/lib/routing/RouteComponentWrapper.js +101 -0
- package/lib/routing/RouterRenderer.d.ts +10 -0
- package/lib/routing/RouterRenderer.d.ts.map +1 -0
- package/lib/routing/RouterRenderer.js +67 -0
- package/lib/routing/RoutingContext.d.ts +5 -0
- package/lib/routing/RoutingContext.d.ts.map +1 -0
- package/lib/routing/RoutingContext.js +4 -0
- package/lib/routing/createRouter.d.ts +19 -0
- package/lib/routing/createRouter.d.ts.map +1 -0
- package/lib/routing/createRouter.js +604 -0
- package/lib/routing/index.d.ts +16 -0
- package/lib/routing/index.d.ts.map +1 -0
- package/lib/routing/index.js +16 -0
- package/lib/routing/useAllQuery.d.ts +7 -0
- package/lib/routing/useAllQuery.d.ts.map +1 -0
- package/lib/routing/useAllQuery.js +31 -0
- package/lib/routing/useFilterDiagnostics.d.ts +2 -0
- package/lib/routing/useFilterDiagnostics.d.ts.map +1 -0
- package/lib/routing/useFilterDiagnostics.js +11 -0
- package/lib/routing/useFilters.d.ts +8 -0
- package/lib/routing/useFilters.d.ts.map +1 -0
- package/lib/routing/useFilters.js +65 -0
- package/lib/routing/useLocation.d.ts +2 -0
- package/lib/routing/useLocation.d.ts.map +1 -0
- package/lib/routing/useLocation.js +24 -0
- package/lib/routing/useNavigate.d.ts +7 -0
- package/lib/routing/useNavigate.d.ts.map +1 -0
- package/lib/routing/useNavigate.js +11 -0
- package/lib/routing/usePathname.d.ts +2 -0
- package/lib/routing/usePathname.d.ts.map +1 -0
- package/lib/routing/usePathname.js +9 -0
- package/lib/routing/useQuery.d.ts +2 -0
- package/lib/routing/useQuery.d.ts.map +1 -0
- package/lib/routing/useQuery.js +9 -0
- package/lib/routing/useQueryState.d.ts +13 -0
- package/lib/routing/useQueryState.d.ts.map +1 -0
- package/lib/routing/useQueryState.js +45 -0
- package/lib/routing/useSearchParams.d.ts +11 -0
- package/lib/routing/useSearchParams.d.ts.map +1 -0
- package/lib/routing/useSearchParams.js +67 -0
- package/lib/tools/buildCombinedSearch.d.ts +8 -0
- package/lib/tools/buildCombinedSearch.d.ts.map +1 -0
- package/lib/tools/buildCombinedSearch.js +76 -0
- package/lib/tools/index.d.ts +3 -0
- package/lib/tools/index.d.ts.map +1 -0
- package/lib/tools/index.js +13 -0
- package/lib/tools/query.d.ts +2 -0
- package/lib/tools/query.d.ts.map +1 -0
- package/lib/tools/query.js +43 -0
- package/lib/tools.d.ts +15 -0
- package/lib/tools.d.ts.map +1 -0
- package/lib/tools.js +179 -0
- package/lib/tsconfig.esm.tsbuildinfo +1 -1
- package/lib/types.d.ts +245 -0
- package/lib/types.d.ts.map +1 -0
- package/lib/types.js +2 -0
- package/lib/values.d.ts +19 -0
- package/lib/values.d.ts.map +1 -0
- package/lib/values.js +53 -0
- package/package.json +2 -2
|
@@ -0,0 +1,604 @@
|
|
|
1
|
+
import { buildRoutes } from '../builder.js';
|
|
2
|
+
import { BrowserHistory } from '../history/index.js';
|
|
3
|
+
import { getMatchedRoute, prepareMatch } from '../tools.js';
|
|
4
|
+
import { parseRawQuery } from '../tools/query.js';
|
|
5
|
+
import buildCombinedSearch from '../tools/buildCombinedSearch.js';
|
|
6
|
+
import { FILTER_OPERATORS, parse as parseFilters, } from '@plumile/filter-query';
|
|
7
|
+
import { createInstrumentationRegistry, } from '../instrumentation/Instrumentation.js';
|
|
8
|
+
import { isRouterNavigationSource, } from '../values.js';
|
|
9
|
+
export default function createRouter(routes, options = {}) {
|
|
10
|
+
const history = new BrowserHistory();
|
|
11
|
+
const registry = createInstrumentationRegistry(options.instrumentations ?? []);
|
|
12
|
+
let staticContext;
|
|
13
|
+
if (options.getContext == null) {
|
|
14
|
+
if (typeof options.context === 'function') {
|
|
15
|
+
staticContext = options.context();
|
|
16
|
+
}
|
|
17
|
+
else {
|
|
18
|
+
staticContext = options.context;
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
function resolveContext() {
|
|
22
|
+
if (options.getContext != null) {
|
|
23
|
+
return options.getContext();
|
|
24
|
+
}
|
|
25
|
+
return staticContext;
|
|
26
|
+
}
|
|
27
|
+
let pendingNavigationOrigin;
|
|
28
|
+
let lastNavigationOrigin = 'external';
|
|
29
|
+
let lastPreloadSignature = null;
|
|
30
|
+
function normalizeLocation(location) {
|
|
31
|
+
let pathnameValue = '';
|
|
32
|
+
if (typeof location.pathname === 'string') {
|
|
33
|
+
pathnameValue = location.pathname;
|
|
34
|
+
}
|
|
35
|
+
let searchValue = '';
|
|
36
|
+
if (typeof location.search === 'string') {
|
|
37
|
+
searchValue = location.search;
|
|
38
|
+
}
|
|
39
|
+
let hashValue = '';
|
|
40
|
+
if (typeof location.hash === 'string') {
|
|
41
|
+
hashValue = location.hash;
|
|
42
|
+
}
|
|
43
|
+
return {
|
|
44
|
+
pathname: pathnameValue,
|
|
45
|
+
search: searchValue,
|
|
46
|
+
hash: hashValue,
|
|
47
|
+
};
|
|
48
|
+
}
|
|
49
|
+
function emitEvent(event) {
|
|
50
|
+
registry.emitEvent(event);
|
|
51
|
+
}
|
|
52
|
+
function mapOrigin(raw) {
|
|
53
|
+
if (isRouterNavigationSource(raw)) {
|
|
54
|
+
return raw;
|
|
55
|
+
}
|
|
56
|
+
return undefined;
|
|
57
|
+
}
|
|
58
|
+
const filterOperators = new Set(FILTER_OPERATORS);
|
|
59
|
+
function looksLikeFilters(value) {
|
|
60
|
+
if (value == null || typeof value !== 'object' || Array.isArray(value)) {
|
|
61
|
+
return false;
|
|
62
|
+
}
|
|
63
|
+
const record = value;
|
|
64
|
+
for (const fieldValue of Object.values(record)) {
|
|
65
|
+
if (fieldValue != null &&
|
|
66
|
+
typeof fieldValue === 'object' &&
|
|
67
|
+
!Array.isArray(fieldValue)) {
|
|
68
|
+
for (const key of Object.keys(fieldValue)) {
|
|
69
|
+
if (filterOperators.has(key)) {
|
|
70
|
+
return true;
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
return false;
|
|
76
|
+
}
|
|
77
|
+
function resolveNavigationOrigin(context) {
|
|
78
|
+
const fallbackOrigin = pendingNavigationOrigin;
|
|
79
|
+
let origin;
|
|
80
|
+
const contextOrigin = context?.origin;
|
|
81
|
+
if (contextOrigin !== undefined) {
|
|
82
|
+
origin = mapOrigin(contextOrigin);
|
|
83
|
+
}
|
|
84
|
+
origin ??= fallbackOrigin;
|
|
85
|
+
origin ??= 'external';
|
|
86
|
+
pendingNavigationOrigin = undefined;
|
|
87
|
+
return origin;
|
|
88
|
+
}
|
|
89
|
+
function readMatchedRoutePath(entry) {
|
|
90
|
+
if (entry === undefined) {
|
|
91
|
+
return undefined;
|
|
92
|
+
}
|
|
93
|
+
const matchedRoute = entry.route;
|
|
94
|
+
if (matchedRoute === null) {
|
|
95
|
+
return undefined;
|
|
96
|
+
}
|
|
97
|
+
const childRoutes = matchedRoute.route.routes;
|
|
98
|
+
if (Array.isArray(childRoutes)) {
|
|
99
|
+
const lastChild = childRoutes.at(-1);
|
|
100
|
+
if (lastChild !== undefined) {
|
|
101
|
+
const lastChildRecord = lastChild;
|
|
102
|
+
const childPath = lastChildRecord.path;
|
|
103
|
+
if (typeof childPath === 'string') {
|
|
104
|
+
return childPath;
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
const rootPath = matchedRoute.route.path;
|
|
109
|
+
if (typeof rootPath === 'string') {
|
|
110
|
+
return rootPath;
|
|
111
|
+
}
|
|
112
|
+
return undefined;
|
|
113
|
+
}
|
|
114
|
+
function readActiveQuerySchema(route) {
|
|
115
|
+
return route?.route.routes.at(-1)?.querySchema;
|
|
116
|
+
}
|
|
117
|
+
function parseSearchState(search, querySchema) {
|
|
118
|
+
const query = parseRawQuery(search);
|
|
119
|
+
if (querySchema == null) {
|
|
120
|
+
return { query };
|
|
121
|
+
}
|
|
122
|
+
const parsed = parseFilters(search, querySchema);
|
|
123
|
+
return {
|
|
124
|
+
query,
|
|
125
|
+
filters: parsed.filters,
|
|
126
|
+
filterDiagnostics: parsed.diagnostics,
|
|
127
|
+
};
|
|
128
|
+
}
|
|
129
|
+
function readPageValue(filters) {
|
|
130
|
+
if (filters == null) {
|
|
131
|
+
return undefined;
|
|
132
|
+
}
|
|
133
|
+
const rawPage = filters.page;
|
|
134
|
+
if (typeof rawPage === 'number') {
|
|
135
|
+
return rawPage;
|
|
136
|
+
}
|
|
137
|
+
if (rawPage != null &&
|
|
138
|
+
typeof rawPage === 'object' &&
|
|
139
|
+
!Array.isArray(rawPage) &&
|
|
140
|
+
typeof rawPage.eq === 'number') {
|
|
141
|
+
return rawPage.eq;
|
|
142
|
+
}
|
|
143
|
+
return undefined;
|
|
144
|
+
}
|
|
145
|
+
function normalizePageFilters(filters) {
|
|
146
|
+
const pageValue = readPageValue(filters);
|
|
147
|
+
if (pageValue == null || pageValue >= 1 || filters == null) {
|
|
148
|
+
return {
|
|
149
|
+
filters,
|
|
150
|
+
normalized: false,
|
|
151
|
+
};
|
|
152
|
+
}
|
|
153
|
+
return {
|
|
154
|
+
filters: {
|
|
155
|
+
...filters,
|
|
156
|
+
page: { eq: 1 },
|
|
157
|
+
},
|
|
158
|
+
normalized: true,
|
|
159
|
+
};
|
|
160
|
+
}
|
|
161
|
+
function buildPrepareQueryInput(state, querySchema) {
|
|
162
|
+
if (querySchema != null) {
|
|
163
|
+
return state.filters ?? {};
|
|
164
|
+
}
|
|
165
|
+
return state.query;
|
|
166
|
+
}
|
|
167
|
+
function buildEntrySnapshot(entry) {
|
|
168
|
+
const preparedSegments = entry.preparedMatch.segments.map((segment) => {
|
|
169
|
+
return {
|
|
170
|
+
path: segment.path,
|
|
171
|
+
fullPath: segment.fullPath,
|
|
172
|
+
segmentIndex: segment.segmentIndex,
|
|
173
|
+
preparedResource: segment.preparedResource,
|
|
174
|
+
redirectTo: segment.redirectTo,
|
|
175
|
+
highlightId: segment.highlightId,
|
|
176
|
+
querySchema: segment.querySchema,
|
|
177
|
+
preparedStatus: segment.preparedResource?.getStatus(),
|
|
178
|
+
};
|
|
179
|
+
});
|
|
180
|
+
return {
|
|
181
|
+
location: normalizeLocation(entry.location),
|
|
182
|
+
routePath: readMatchedRoutePath(entry),
|
|
183
|
+
preparedMatch: {
|
|
184
|
+
segments: preparedSegments,
|
|
185
|
+
routes: preparedSegments,
|
|
186
|
+
},
|
|
187
|
+
filters: entry.filters,
|
|
188
|
+
filterDiagnostics: entry.filterDiagnostics,
|
|
189
|
+
activeQuerySchema: entry.activeQuerySchema,
|
|
190
|
+
};
|
|
191
|
+
}
|
|
192
|
+
function notifyEntry(entry) {
|
|
193
|
+
registry.notifyEntryChange(buildEntrySnapshot(entry));
|
|
194
|
+
}
|
|
195
|
+
function buildPreloadKey(mode, pathname) {
|
|
196
|
+
return `${mode}:${pathname}`;
|
|
197
|
+
}
|
|
198
|
+
function emitPreloadEvent(mode, pathname, source) {
|
|
199
|
+
const now = Date.now();
|
|
200
|
+
const key = buildPreloadKey(mode, pathname);
|
|
201
|
+
const recentlyRecorded = lastPreloadSignature !== null &&
|
|
202
|
+
lastPreloadSignature.key === key &&
|
|
203
|
+
now - lastPreloadSignature.timestamp < 50;
|
|
204
|
+
if (!recentlyRecorded) {
|
|
205
|
+
emitEvent({
|
|
206
|
+
kind: 'preload',
|
|
207
|
+
source,
|
|
208
|
+
timestamp: now,
|
|
209
|
+
location: normalizeLocation(history.location),
|
|
210
|
+
targetPathname: pathname,
|
|
211
|
+
mode,
|
|
212
|
+
});
|
|
213
|
+
}
|
|
214
|
+
lastPreloadSignature = { key, timestamp: now };
|
|
215
|
+
}
|
|
216
|
+
function emitHistoryEvent(action, source, location, details) {
|
|
217
|
+
pendingNavigationOrigin = source;
|
|
218
|
+
emitEvent({
|
|
219
|
+
kind: 'history',
|
|
220
|
+
source,
|
|
221
|
+
action,
|
|
222
|
+
timestamp: Date.now(),
|
|
223
|
+
location: normalizeLocation(location),
|
|
224
|
+
details,
|
|
225
|
+
});
|
|
226
|
+
}
|
|
227
|
+
const flatRoutes = buildRoutes(routes);
|
|
228
|
+
const route = getMatchedRoute(flatRoutes, history.location);
|
|
229
|
+
const initialUnifiedSchema = readActiveQuerySchema(route);
|
|
230
|
+
const initialSearchState = parseSearchState(history.location.search, initialUnifiedSchema);
|
|
231
|
+
const normalizedInitialFilters = normalizePageFilters(initialSearchState.filters);
|
|
232
|
+
const initialLocationSnapshot = normalizeLocation(history.location);
|
|
233
|
+
let initialInstrumentation;
|
|
234
|
+
if (route != null) {
|
|
235
|
+
initialInstrumentation = {
|
|
236
|
+
emit: (event) => {
|
|
237
|
+
emitEvent(event);
|
|
238
|
+
},
|
|
239
|
+
source: lastNavigationOrigin,
|
|
240
|
+
location: initialLocationSnapshot,
|
|
241
|
+
};
|
|
242
|
+
}
|
|
243
|
+
const initialContext = resolveContext();
|
|
244
|
+
const preparedMatch = prepareMatch(route, buildPrepareQueryInput({
|
|
245
|
+
...initialSearchState,
|
|
246
|
+
filters: normalizedInitialFilters.filters,
|
|
247
|
+
}, initialUnifiedSchema), initialInstrumentation, initialContext);
|
|
248
|
+
let currentEntry = {
|
|
249
|
+
forceRerender: false,
|
|
250
|
+
location: history.location,
|
|
251
|
+
route,
|
|
252
|
+
preparedMatch,
|
|
253
|
+
context: initialContext,
|
|
254
|
+
rawSearch: history.location.search,
|
|
255
|
+
query: initialSearchState.query,
|
|
256
|
+
filters: normalizedInitialFilters.filters,
|
|
257
|
+
filterDiagnostics: initialSearchState.filterDiagnostics,
|
|
258
|
+
activeQuerySchema: initialUnifiedSchema,
|
|
259
|
+
};
|
|
260
|
+
if (normalizedInitialFilters.normalized &&
|
|
261
|
+
initialUnifiedSchema != null &&
|
|
262
|
+
normalizedInitialFilters.filters != null) {
|
|
263
|
+
currentEntry.filters = normalizedInitialFilters.filters;
|
|
264
|
+
const normalizedSearch = buildCombinedSearch({
|
|
265
|
+
filters: normalizedInitialFilters.filters,
|
|
266
|
+
query: initialSearchState.query,
|
|
267
|
+
querySchema: initialUnifiedSchema,
|
|
268
|
+
});
|
|
269
|
+
if (normalizedSearch !== history.location.search) {
|
|
270
|
+
emitHistoryEvent('normalize', 'normalize', {
|
|
271
|
+
pathname: history.location.pathname,
|
|
272
|
+
search: normalizedSearch,
|
|
273
|
+
hash: '',
|
|
274
|
+
}, { reason: 'initial-page-clamp' });
|
|
275
|
+
const normalizeDebugContext = {
|
|
276
|
+
origin: 'normalize',
|
|
277
|
+
trigger: 'normalize',
|
|
278
|
+
};
|
|
279
|
+
history.set({
|
|
280
|
+
pathname: history.location.pathname,
|
|
281
|
+
search: normalizedSearch,
|
|
282
|
+
hash: '',
|
|
283
|
+
debugContext: normalizeDebugContext,
|
|
284
|
+
});
|
|
285
|
+
currentEntry = {
|
|
286
|
+
...currentEntry,
|
|
287
|
+
location: { ...currentEntry.location, search: normalizedSearch },
|
|
288
|
+
rawSearch: normalizedSearch,
|
|
289
|
+
};
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
notifyEntry(currentEntry);
|
|
293
|
+
emitEvent({
|
|
294
|
+
kind: 'snapshot',
|
|
295
|
+
source: lastNavigationOrigin,
|
|
296
|
+
timestamp: Date.now(),
|
|
297
|
+
location: normalizeLocation(currentEntry.location),
|
|
298
|
+
routePath: readMatchedRoutePath(currentEntry),
|
|
299
|
+
});
|
|
300
|
+
let nextId = 0;
|
|
301
|
+
const subscribers = new Map();
|
|
302
|
+
const disposeHistory = history.subscribe((location, forceRerender, debugContext) => {
|
|
303
|
+
const origin = resolveNavigationOrigin(debugContext);
|
|
304
|
+
lastNavigationOrigin = origin;
|
|
305
|
+
const locationPayload = normalizeLocation(location);
|
|
306
|
+
const debugContextOrigin = debugContext?.origin;
|
|
307
|
+
if (debugContextOrigin !== undefined) {
|
|
308
|
+
let historyDetails;
|
|
309
|
+
if (typeof debugContext?.historyIndex === 'number') {
|
|
310
|
+
historyDetails = { historyIndex: debugContext.historyIndex };
|
|
311
|
+
}
|
|
312
|
+
if (debugContextOrigin === 'popstate-back' ||
|
|
313
|
+
debugContextOrigin === 'popstate-forward' ||
|
|
314
|
+
debugContextOrigin === 'popstate-unknown' ||
|
|
315
|
+
debugContextOrigin === 'external') {
|
|
316
|
+
let direction = 'unknown';
|
|
317
|
+
if (debugContextOrigin === 'popstate-back') {
|
|
318
|
+
direction = 'back';
|
|
319
|
+
}
|
|
320
|
+
else if (debugContextOrigin === 'popstate-forward') {
|
|
321
|
+
direction = 'forward';
|
|
322
|
+
}
|
|
323
|
+
emitEvent({
|
|
324
|
+
kind: 'popstate',
|
|
325
|
+
source: origin,
|
|
326
|
+
timestamp: Date.now(),
|
|
327
|
+
location: locationPayload,
|
|
328
|
+
direction,
|
|
329
|
+
details: historyDetails,
|
|
330
|
+
});
|
|
331
|
+
}
|
|
332
|
+
}
|
|
333
|
+
const samePathname = location.pathname === currentEntry.location.pathname;
|
|
334
|
+
const sameSearch = location.search === currentEntry.rawSearch;
|
|
335
|
+
if (!forceRerender && samePathname && sameSearch) {
|
|
336
|
+
return;
|
|
337
|
+
}
|
|
338
|
+
const nextContext = resolveContext();
|
|
339
|
+
let nextPreparedMatch = currentEntry.preparedMatch;
|
|
340
|
+
let nextRoute = currentEntry.route;
|
|
341
|
+
if (!samePathname) {
|
|
342
|
+
nextRoute = getMatchedRoute(flatRoutes, location);
|
|
343
|
+
}
|
|
344
|
+
const querySchema = readActiveQuerySchema(nextRoute);
|
|
345
|
+
const parsedSearchState = parseSearchState(location.search, querySchema);
|
|
346
|
+
const normalizedFilters = normalizePageFilters(parsedSearchState.filters);
|
|
347
|
+
if (!samePathname || !sameSearch) {
|
|
348
|
+
let updateInstrumentation;
|
|
349
|
+
if (nextRoute != null) {
|
|
350
|
+
updateInstrumentation = {
|
|
351
|
+
emit: (event) => {
|
|
352
|
+
emitEvent(event);
|
|
353
|
+
},
|
|
354
|
+
source: origin,
|
|
355
|
+
location: locationPayload,
|
|
356
|
+
};
|
|
357
|
+
}
|
|
358
|
+
nextPreparedMatch = prepareMatch(nextRoute, buildPrepareQueryInput({
|
|
359
|
+
...parsedSearchState,
|
|
360
|
+
filters: normalizedFilters.filters,
|
|
361
|
+
}, querySchema), updateInstrumentation, nextContext);
|
|
362
|
+
}
|
|
363
|
+
const nextEntry = {
|
|
364
|
+
forceRerender: forceRerender || (samePathname && !sameSearch),
|
|
365
|
+
location,
|
|
366
|
+
route: nextRoute,
|
|
367
|
+
preparedMatch: nextPreparedMatch,
|
|
368
|
+
context: nextContext,
|
|
369
|
+
rawSearch: location.search,
|
|
370
|
+
query: parsedSearchState.query,
|
|
371
|
+
filters: normalizedFilters.filters,
|
|
372
|
+
filterDiagnostics: parsedSearchState.filterDiagnostics,
|
|
373
|
+
activeQuerySchema: querySchema,
|
|
374
|
+
};
|
|
375
|
+
if (normalizedFilters.normalized && querySchema != null) {
|
|
376
|
+
const normalizedSearch = buildCombinedSearch({
|
|
377
|
+
filters: normalizedFilters.filters,
|
|
378
|
+
query: parsedSearchState.query,
|
|
379
|
+
querySchema,
|
|
380
|
+
});
|
|
381
|
+
if (normalizedSearch !== location.search) {
|
|
382
|
+
let nextSearchStr = normalizedSearch;
|
|
383
|
+
if (!nextSearchStr.startsWith('?') && nextSearchStr.length > 0) {
|
|
384
|
+
nextSearchStr = `?${nextSearchStr}`;
|
|
385
|
+
}
|
|
386
|
+
emitHistoryEvent('normalize', 'normalize', {
|
|
387
|
+
pathname: location.pathname,
|
|
388
|
+
search: nextSearchStr,
|
|
389
|
+
hash: '',
|
|
390
|
+
}, { reason: 'runtime-page-clamp' });
|
|
391
|
+
const runtimeNormalizeContext = {
|
|
392
|
+
origin: 'normalize',
|
|
393
|
+
trigger: 'normalize',
|
|
394
|
+
};
|
|
395
|
+
history.set({
|
|
396
|
+
pathname: location.pathname,
|
|
397
|
+
search: nextSearchStr,
|
|
398
|
+
hash: '',
|
|
399
|
+
debugContext: runtimeNormalizeContext,
|
|
400
|
+
});
|
|
401
|
+
return;
|
|
402
|
+
}
|
|
403
|
+
}
|
|
404
|
+
currentEntry = nextEntry;
|
|
405
|
+
subscribers.forEach((callback) => {
|
|
406
|
+
callback(nextEntry);
|
|
407
|
+
});
|
|
408
|
+
notifyEntry(nextEntry);
|
|
409
|
+
emitEvent({
|
|
410
|
+
kind: 'snapshot',
|
|
411
|
+
source: origin,
|
|
412
|
+
timestamp: Date.now(),
|
|
413
|
+
location: locationPayload,
|
|
414
|
+
routePath: readMatchedRoutePath(nextEntry),
|
|
415
|
+
});
|
|
416
|
+
});
|
|
417
|
+
function normalizePreloadTarget(target) {
|
|
418
|
+
let normalizedPathname = history.location.pathname;
|
|
419
|
+
if (typeof target.pathname === 'string' && target.pathname.length > 0) {
|
|
420
|
+
normalizedPathname = target.pathname;
|
|
421
|
+
}
|
|
422
|
+
let normalizedSearch = '';
|
|
423
|
+
if (typeof target.search === 'string') {
|
|
424
|
+
normalizedSearch = target.search;
|
|
425
|
+
}
|
|
426
|
+
let normalizedSource = 'programmatic';
|
|
427
|
+
const mappedSource = mapOrigin(target.source);
|
|
428
|
+
if (mappedSource !== undefined) {
|
|
429
|
+
normalizedSource = mappedSource;
|
|
430
|
+
}
|
|
431
|
+
return {
|
|
432
|
+
pathname: normalizedPathname,
|
|
433
|
+
search: normalizedSearch,
|
|
434
|
+
source: normalizedSource,
|
|
435
|
+
};
|
|
436
|
+
}
|
|
437
|
+
function getPrepared(path) {
|
|
438
|
+
try {
|
|
439
|
+
const entry = currentEntry;
|
|
440
|
+
for (const preparedRoute of entry.preparedMatch.segments) {
|
|
441
|
+
if (preparedRoute.path === path || preparedRoute.fullPath === path) {
|
|
442
|
+
const preparedValue = preparedRoute.preparedResource?.get();
|
|
443
|
+
return preparedValue;
|
|
444
|
+
}
|
|
445
|
+
}
|
|
446
|
+
return undefined;
|
|
447
|
+
}
|
|
448
|
+
catch {
|
|
449
|
+
return undefined;
|
|
450
|
+
}
|
|
451
|
+
}
|
|
452
|
+
function getPreparedSegments(path) {
|
|
453
|
+
return currentEntry.preparedMatch.segments
|
|
454
|
+
.filter((preparedRoute) => {
|
|
455
|
+
return preparedRoute.path === path || preparedRoute.fullPath === path;
|
|
456
|
+
})
|
|
457
|
+
.map((preparedRoute) => {
|
|
458
|
+
return {
|
|
459
|
+
path: preparedRoute.path,
|
|
460
|
+
fullPath: preparedRoute.fullPath,
|
|
461
|
+
segmentIndex: preparedRoute.segmentIndex,
|
|
462
|
+
preparedResource: preparedRoute.preparedResource,
|
|
463
|
+
redirectTo: preparedRoute.redirectTo,
|
|
464
|
+
resourcePage: preparedRoute.resourcePage,
|
|
465
|
+
highlightId: preparedRoute.highlightId,
|
|
466
|
+
querySchema: preparedRoute.querySchema,
|
|
467
|
+
};
|
|
468
|
+
});
|
|
469
|
+
}
|
|
470
|
+
const context = {
|
|
471
|
+
history,
|
|
472
|
+
get() {
|
|
473
|
+
return currentEntry;
|
|
474
|
+
},
|
|
475
|
+
preloadCode(target) {
|
|
476
|
+
const normalized = normalizePreloadTarget(target);
|
|
477
|
+
emitPreloadEvent('code', normalized.pathname, normalized.source);
|
|
478
|
+
const matches = getMatchedRoute(flatRoutes, {
|
|
479
|
+
...history.location,
|
|
480
|
+
pathname: normalized.pathname,
|
|
481
|
+
});
|
|
482
|
+
if (matches == null) {
|
|
483
|
+
return;
|
|
484
|
+
}
|
|
485
|
+
matches.route.routes.forEach(({ resourcePage }) => {
|
|
486
|
+
if (resourcePage == null) {
|
|
487
|
+
return;
|
|
488
|
+
}
|
|
489
|
+
resourcePage.load();
|
|
490
|
+
});
|
|
491
|
+
},
|
|
492
|
+
preload(target) {
|
|
493
|
+
const normalized = normalizePreloadTarget(target);
|
|
494
|
+
emitPreloadEvent('full', normalized.pathname, normalized.source);
|
|
495
|
+
const matches = getMatchedRoute(flatRoutes, {
|
|
496
|
+
...history.location,
|
|
497
|
+
pathname: normalized.pathname,
|
|
498
|
+
});
|
|
499
|
+
const locationSnapshot = {
|
|
500
|
+
pathname: normalized.pathname,
|
|
501
|
+
search: normalized.search,
|
|
502
|
+
hash: '',
|
|
503
|
+
};
|
|
504
|
+
let instrumentation;
|
|
505
|
+
if (matches != null) {
|
|
506
|
+
instrumentation = {
|
|
507
|
+
emit: (event) => {
|
|
508
|
+
emitEvent(event);
|
|
509
|
+
},
|
|
510
|
+
source: normalized.source,
|
|
511
|
+
location: locationSnapshot,
|
|
512
|
+
};
|
|
513
|
+
}
|
|
514
|
+
const preloadContext = resolveContext();
|
|
515
|
+
const preloadQuerySchema = readActiveQuerySchema(matches);
|
|
516
|
+
const preloadSearchState = parseSearchState(normalized.search, preloadQuerySchema);
|
|
517
|
+
const preparedMatch = prepareMatch(matches, buildPrepareQueryInput(preloadSearchState, preloadQuerySchema), instrumentation, preloadContext);
|
|
518
|
+
preparedMatch.segments.forEach((route) => {
|
|
519
|
+
if (route.preparedResource != null) {
|
|
520
|
+
route.preparedResource.load();
|
|
521
|
+
}
|
|
522
|
+
});
|
|
523
|
+
},
|
|
524
|
+
subscribe(callback) {
|
|
525
|
+
nextId += 1;
|
|
526
|
+
const id = nextId;
|
|
527
|
+
function disposeCallback() {
|
|
528
|
+
subscribers.delete(id);
|
|
529
|
+
}
|
|
530
|
+
subscribers.set(id, callback);
|
|
531
|
+
return disposeCallback;
|
|
532
|
+
},
|
|
533
|
+
navigate({ pathname, query, filters: navFilters, replace, }) {
|
|
534
|
+
const current = currentEntry;
|
|
535
|
+
let targetPathname = pathname;
|
|
536
|
+
targetPathname ??= current.location.pathname;
|
|
537
|
+
const destinationRoute = getMatchedRoute(flatRoutes, {
|
|
538
|
+
...history.location,
|
|
539
|
+
pathname: targetPathname,
|
|
540
|
+
});
|
|
541
|
+
let destSchema = readActiveQuerySchema(destinationRoute);
|
|
542
|
+
destSchema = destSchema ?? current.activeQuerySchema;
|
|
543
|
+
const hasSchema = destSchema != null;
|
|
544
|
+
const useQueryAsFilters = navFilters == null &&
|
|
545
|
+
hasSchema &&
|
|
546
|
+
query != null &&
|
|
547
|
+
looksLikeFilters(query);
|
|
548
|
+
let filtersInput = navFilters;
|
|
549
|
+
if (filtersInput == null) {
|
|
550
|
+
if (useQueryAsFilters) {
|
|
551
|
+
filtersInput = query;
|
|
552
|
+
}
|
|
553
|
+
else {
|
|
554
|
+
filtersInput = current.filters;
|
|
555
|
+
}
|
|
556
|
+
}
|
|
557
|
+
let rawQueryInput;
|
|
558
|
+
if (useQueryAsFilters) {
|
|
559
|
+
rawQueryInput = current.query;
|
|
560
|
+
}
|
|
561
|
+
else {
|
|
562
|
+
rawQueryInput = query ?? current.query;
|
|
563
|
+
}
|
|
564
|
+
const search = buildCombinedSearch({
|
|
565
|
+
filters: filtersInput,
|
|
566
|
+
query: rawQueryInput,
|
|
567
|
+
querySchema: destSchema,
|
|
568
|
+
});
|
|
569
|
+
const locationObj = { pathname: targetPathname, search, hash: '' };
|
|
570
|
+
const origin = 'programmatic';
|
|
571
|
+
const debugContext = {
|
|
572
|
+
origin,
|
|
573
|
+
trigger: 'programmatic',
|
|
574
|
+
};
|
|
575
|
+
if (replace === true) {
|
|
576
|
+
emitHistoryEvent('replace', origin, locationObj, {
|
|
577
|
+
trigger: 'programmatic',
|
|
578
|
+
});
|
|
579
|
+
history.set({
|
|
580
|
+
...locationObj,
|
|
581
|
+
debugContext,
|
|
582
|
+
});
|
|
583
|
+
}
|
|
584
|
+
else {
|
|
585
|
+
emitHistoryEvent('push', origin, locationObj, {
|
|
586
|
+
trigger: 'programmatic',
|
|
587
|
+
});
|
|
588
|
+
history.push({
|
|
589
|
+
...locationObj,
|
|
590
|
+
debugContext,
|
|
591
|
+
});
|
|
592
|
+
}
|
|
593
|
+
},
|
|
594
|
+
getPrepared,
|
|
595
|
+
getPreparedSegments,
|
|
596
|
+
};
|
|
597
|
+
function cleanup() {
|
|
598
|
+
disposeHistory();
|
|
599
|
+
subscribers.clear();
|
|
600
|
+
registry.dispose();
|
|
601
|
+
}
|
|
602
|
+
return { context, cleanup };
|
|
603
|
+
}
|
|
604
|
+
//# 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 { parseRawQuery } from '../tools/query.js';\nimport buildCombinedSearch from '../tools/buildCombinedSearch.js';\nimport {\n  FILTER_OPERATORS,\n  parse as parseFilters,\n  type Diagnostic as FilterDiagnostic,\n  type Schema as FilterSchema,\n} from '@plumile/filter-query';\nimport type { ParamData } from 'path-to-regexp';\nimport {\n  type RouteEntry,\n  type RoutingContextType,\n  type SubscribeCallback,\n  type AnyRoute,\n  type PreparedAccess,\n  type NavigateOverloads,\n  type PreloadTarget,\n  type PreparedOf,\n  type PreparedRouteSegmentSnapshot,\n  type RouteUnion,\n  type RawQuery,\n  type UnknownFilters,\n  type UnknownQueryShape,\n} from '../types.js';\nimport {\n  createInstrumentationRegistry,\n  type InstrumentationAPI,\n  type RouterEntrySnapshot,\n  type RouterLocationSnapshot,\n  type RouterNavigationSource,\n  type RouterEvent,\n} from '../instrumentation/Instrumentation.js';\nimport type { HistoryDebugContext } from '../history/types.js';\nimport {\n  isRouterNavigationSource,\n  type RouterHistoryActionValue,\n  type RouterPreloadModeValue,\n} from '../values.js';\n\ntype RouterQuerySchema = FilterSchema | undefined;\ntype RouterEntryState<TContext> = RouteEntry<\n  ParamData,\n  RouterQuerySchema,\n  TContext\n>;\ntype ParsedSearchState = {\n  query: RawQuery;\n  filters?: UnknownFilters;\n  filterDiagnostics?: FilterDiagnostic[];\n};\ntype PrepareQueryInput = Record<string, unknown>;\n\n/**\n * Return type for the createRouter function.\n */\nexport type CreateRouterReturn<TContext, R extends AnyRoute[]> = {\n  /** Function to clean up router listeners and resources */\n  cleanup: () => void;\n  /** Router context object for the React Context Provider */\n  context: RoutingContextType<\n    ParamData,\n    RouterQuerySchema,\n    UnknownQueryShape,\n    TContext\n  > &\n    PreparedAccess<R> & {\n      navigate: NavigateOverloads<R>;\n    };\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 */\n/**\n * Create a router (typed overload). When called with a const tuple of routes, generics are preserved.\n */\n/**\n * Optional configuration for `createRouter`.\n */\nexport type CreateRouterOptions<TContext> = {\n  /** Instrumentations to attach to the router (DevTools bridge, logger, ...). */\n  instrumentations?: InstrumentationAPI[];\n  /** Static context value or lazy initializer for the router. */\n  context?: TContext | (() => TContext);\n  /** Resolve a fresh context value per navigation. */\n  getContext?: () => TContext;\n};\n\n/**\n * Creates a router instance from a route configuration.\n *\n * @param routes - Route definitions to register.\n * @param options - Optional instrumentation settings used for exposing development tooling.\n * @param options.instrumentations - Instrumentations invoked on router events (development only).\n */\nexport default function createRouter<\n  TContext = unknown,\n  R extends AnyRoute[] = AnyRoute[],\n>(\n  routes: [...R] | AnyRoute[],\n  options: CreateRouterOptions<TContext> = {},\n): CreateRouterReturn<TContext, R> {\n  // Initialize browser history manager\n  const history = new BrowserHistory();\n\n  const registry = createInstrumentationRegistry(\n    options.instrumentations ?? [],\n  );\n\n  let staticContext: TContext | undefined;\n  if (options.getContext == null) {\n    if (typeof options.context === 'function') {\n      staticContext = (options.context as () => TContext)();\n    } else {\n      staticContext = options.context;\n    }\n  }\n\n  /** Resolve the context for the current navigation. */\n  function resolveContext(): TContext {\n    if (options.getContext != null) {\n      return options.getContext();\n    }\n    return staticContext as TContext;\n  }\n\n  let pendingNavigationOrigin: RouterNavigationSource | undefined;\n  let lastNavigationOrigin: RouterNavigationSource = 'external';\n  let lastPreloadSignature: {\n    key: string;\n    timestamp: number;\n  } | null = null;\n\n  /**\n   * Converts a partial location object into a full location structure with string fields.\n   */\n  function normalizeLocation(location: {\n    pathname: string;\n    search?: string;\n    hash?: string;\n  }): { pathname: string; search: string; hash: string } {\n    let pathnameValue = '';\n    if (typeof location.pathname === 'string') {\n      pathnameValue = location.pathname;\n    }\n    let searchValue = '';\n    if (typeof location.search === 'string') {\n      searchValue = location.search;\n    }\n    let hashValue = '';\n    if (typeof location.hash === 'string') {\n      hashValue = location.hash;\n    }\n    return {\n      pathname: pathnameValue,\n      search: searchValue,\n      hash: hashValue,\n    };\n  }\n\n  /** Emits a router event to every registered instrumentation. */\n  function emitEvent(event: RouterEvent): void {\n    registry.emitEvent(event);\n  }\n\n  /**\n   * Attempts to coerce an arbitrary string into a known navigation source.\n   */\n  function mapOrigin(raw?: string): RouterNavigationSource | undefined {\n    if (isRouterNavigationSource(raw)) {\n      return raw;\n    }\n    return undefined;\n  }\n\n  const filterOperators = new Set<string>(FILTER_OPERATORS);\n\n  /** Heuristic: detect filter-like objects to preserve legacy navigate(query) behavior. */\n  function looksLikeFilters(value: unknown): boolean {\n    if (value == null || typeof value !== 'object' || Array.isArray(value)) {\n      return false;\n    }\n    const record = value as Record<string, unknown>;\n    for (const fieldValue of Object.values(record)) {\n      if (\n        fieldValue != null &&\n        typeof fieldValue === 'object' &&\n        !Array.isArray(fieldValue)\n      ) {\n        for (const key of Object.keys(fieldValue as Record<string, unknown>)) {\n          if (filterOperators.has(key)) {\n            return true;\n          }\n        }\n      }\n    }\n    return false;\n  }\n\n  /**\n   * Determines the most appropriate origin for the current navigation event.\n   */\n  /** Resolves the navigation source combining history context and pending intent. */\n  function resolveNavigationOrigin(\n    context?: HistoryDebugContext | null,\n  ): RouterNavigationSource {\n    const fallbackOrigin = pendingNavigationOrigin;\n    let origin: RouterNavigationSource | undefined;\n\n    const contextOrigin = context?.origin;\n    if (contextOrigin !== undefined) {\n      origin = mapOrigin(contextOrigin);\n    }\n\n    origin ??= fallbackOrigin;\n    origin ??= 'external';\n    pendingNavigationOrigin = undefined;\n    return origin;\n  }\n\n  /**\n   * Extracts the inner-most matched route path from a route entry when available.\n   */\n  /** Returns the deepest matched route path from a route entry if available. */\n  function readMatchedRoutePath(\n    entry: RouterEntryState<TContext> | undefined,\n  ): string | undefined {\n    if (entry === undefined) {\n      return undefined;\n    }\n    const matchedRoute = entry.route;\n    if (matchedRoute === null) {\n      return undefined;\n    }\n    const childRoutes = matchedRoute.route.routes;\n    if (Array.isArray(childRoutes)) {\n      const lastChild = childRoutes.at(-1);\n      if (lastChild !== undefined) {\n        const lastChildRecord = lastChild as Record<string, unknown>;\n        const childPath = lastChildRecord.path;\n        if (typeof childPath === 'string') {\n          return childPath;\n        }\n      }\n    }\n    const rootPath = matchedRoute.route.path;\n    if (typeof rootPath === 'string') {\n      return rootPath;\n    }\n    return undefined;\n  }\n\n  /** Returns the deepest active query schema for a matched route. */\n  function readActiveQuerySchema(\n    route: RouterEntryState<TContext>['route'],\n  ): RouterQuerySchema {\n    return route?.route.routes.at(-1)?.querySchema;\n  }\n\n  /** Parses raw query and typed filters for the provided search string. */\n  function parseSearchState(\n    search: string,\n    querySchema: RouterQuerySchema,\n  ): ParsedSearchState {\n    const query = parseRawQuery(search);\n    if (querySchema == null) {\n      return { query };\n    }\n\n    const parsed = parseFilters(search, querySchema);\n    return {\n      query,\n      filters: parsed.filters as UnknownFilters,\n      filterDiagnostics: parsed.diagnostics,\n    };\n  }\n\n  /** Reads a page number from either `page` or `page.eq` filter shapes. */\n  function readPageValue(filters?: UnknownFilters): number | undefined {\n    if (filters == null) {\n      return undefined;\n    }\n\n    const rawPage = filters.page;\n    if (typeof rawPage === 'number') {\n      return rawPage;\n    }\n    if (\n      rawPage != null &&\n      typeof rawPage === 'object' &&\n      !Array.isArray(rawPage) &&\n      typeof (rawPage as { eq?: unknown }).eq === 'number'\n    ) {\n      return (rawPage as { eq: number }).eq;\n    }\n    return undefined;\n  }\n\n  /** Normalizes the page filter to a minimum value of 1 when present. */\n  function normalizePageFilters(filters?: UnknownFilters): {\n    filters?: UnknownFilters;\n    normalized: boolean;\n  } {\n    const pageValue = readPageValue(filters);\n    if (pageValue == null || pageValue >= 1 || filters == null) {\n      return {\n        filters,\n        normalized: false,\n      };\n    }\n\n    return {\n      filters: {\n        ...filters,\n        page: { eq: 1 },\n      },\n      normalized: true,\n    };\n  }\n\n  /** Converts router search state into the object forwarded to `prepare`. */\n  function buildPrepareQueryInput(\n    state: ParsedSearchState,\n    querySchema: RouterQuerySchema,\n  ): PrepareQueryInput {\n    if (querySchema != null) {\n      return state.filters ?? {};\n    }\n    return state.query;\n  }\n\n  /** Builds a snapshot representation of the current router entry. */\n  function buildEntrySnapshot(\n    entry: RouterEntryState<TContext>,\n  ): RouterEntrySnapshot {\n    const preparedSegments = entry.preparedMatch.segments.map((segment) => {\n      return {\n        path: segment.path,\n        fullPath: segment.fullPath,\n        segmentIndex: segment.segmentIndex,\n        preparedResource: segment.preparedResource,\n        redirectTo: segment.redirectTo,\n        highlightId: segment.highlightId,\n        querySchema: segment.querySchema,\n        preparedStatus: segment.preparedResource?.getStatus(),\n      };\n    });\n\n    return {\n      location: normalizeLocation(entry.location),\n      routePath: readMatchedRoutePath(entry),\n      preparedMatch: {\n        segments: preparedSegments,\n        routes: preparedSegments,\n      },\n      filters: entry.filters,\n      filterDiagnostics: entry.filterDiagnostics,\n      activeQuerySchema: entry.activeQuerySchema,\n    };\n  }\n\n  /** Notifies instrumentations of a new active entry. */\n  function notifyEntry(entry: RouterEntryState<TContext>): void {\n    registry.notifyEntryChange(buildEntrySnapshot(entry));\n  }\n\n  /** Creates a deduplication key for preload events. */\n  function buildPreloadKey(\n    mode: RouterPreloadModeValue,\n    pathname: string,\n  ): string {\n    return `${mode}:${pathname}`;\n  }\n\n  /** Emits a preload or preload-code instrumentation event. */\n  function emitPreloadEvent(\n    mode: 'code' | 'full',\n    pathname: string,\n    source: RouterNavigationSource,\n  ): void {\n    const now = Date.now();\n    const key = buildPreloadKey(mode, pathname);\n    const recentlyRecorded =\n      lastPreloadSignature !== null &&\n      lastPreloadSignature.key === key &&\n      now - lastPreloadSignature.timestamp < 50;\n\n    if (!recentlyRecorded) {\n      emitEvent({\n        kind: 'preload',\n        source,\n        timestamp: now,\n        location: normalizeLocation(history.location),\n        targetPathname: pathname,\n        mode,\n      });\n    }\n\n    lastPreloadSignature = { key, timestamp: now };\n  }\n\n  /** Emits a history mutation event (push/replace/normalize). */\n  function emitHistoryEvent(\n    action: RouterHistoryActionValue,\n    source: RouterNavigationSource,\n    location: { pathname: string; search?: string; hash?: string },\n    details?: Record<string, unknown>,\n  ): void {\n    pendingNavigationOrigin = source;\n    emitEvent({\n      kind: 'history',\n      source,\n      action,\n      timestamp: Date.now(),\n      location: normalizeLocation(location),\n      details,\n    });\n  }\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 initialUnifiedSchema = readActiveQuerySchema(route);\n  const initialSearchState = parseSearchState(\n    history.location.search,\n    initialUnifiedSchema,\n  );\n  const normalizedInitialFilters = normalizePageFilters(\n    initialSearchState.filters,\n  );\n  const initialLocationSnapshot = normalizeLocation(history.location);\n  let initialInstrumentation:\n    | {\n        emit: (event: RouterEvent) => void;\n        source: RouterNavigationSource;\n        location: RouterLocationSnapshot;\n      }\n    | undefined;\n  if (route != null) {\n    initialInstrumentation = {\n      emit: (event: RouterEvent) => {\n        emitEvent(event);\n      },\n      source: lastNavigationOrigin,\n      location: initialLocationSnapshot,\n    };\n  }\n  const initialContext = resolveContext();\n  const preparedMatch = prepareMatch(\n    route,\n    buildPrepareQueryInput(\n      {\n        ...initialSearchState,\n        filters: normalizedInitialFilters.filters,\n      },\n      initialUnifiedSchema,\n    ),\n    initialInstrumentation,\n    initialContext,\n  );\n  // Helper to build the raw query object from a search string\n  let currentEntry: RouterEntryState<TContext> = {\n    forceRerender: false,\n    location: history.location,\n    route,\n    preparedMatch,\n    context: initialContext,\n    rawSearch: history.location.search,\n    query: initialSearchState.query,\n    // typedQuery removed (unified into filters/query)\n    filters: normalizedInitialFilters.filters,\n    filterDiagnostics: initialSearchState.filterDiagnostics,\n    activeQuerySchema: initialUnifiedSchema,\n  };\n\n  // Initial normalization pass (e.g., clamp page)\n  if (\n    normalizedInitialFilters.normalized &&\n    initialUnifiedSchema != null &&\n    normalizedInitialFilters.filters != null\n  ) {\n    currentEntry.filters = normalizedInitialFilters.filters;\n    const normalizedSearch = buildCombinedSearch({\n      filters: normalizedInitialFilters.filters,\n      query: initialSearchState.query,\n      querySchema: initialUnifiedSchema,\n    }); // returns '' or string starting with '?'\n    if (normalizedSearch !== history.location.search) {\n      emitHistoryEvent(\n        'normalize',\n        'normalize',\n        {\n          pathname: history.location.pathname,\n          search: normalizedSearch,\n          hash: '',\n        },\n        { reason: 'initial-page-clamp' },\n      );\n      const normalizeDebugContext: HistoryDebugContext = {\n        origin: 'normalize',\n        trigger: 'normalize',\n      };\n      history.set({\n        pathname: history.location.pathname,\n        search: normalizedSearch,\n        hash: '',\n        debugContext: normalizeDebugContext,\n      });\n      // Update currentEntry.location to reflect new search directly (history.set triggers async subscriber)\n      currentEntry = {\n        ...currentEntry,\n        location: { ...currentEntry.location, search: normalizedSearch },\n        rawSearch: normalizedSearch,\n      };\n    }\n  }\n\n  notifyEntry(currentEntry);\n  emitEvent({\n    kind: 'snapshot',\n    source: lastNavigationOrigin,\n    timestamp: Date.now(),\n    location: normalizeLocation(currentEntry.location),\n    routePath: readMatchedRoutePath(currentEntry),\n  });\n\n  // Maintain a set of subscribers to the active route entry\n  let nextId = 0;\n  const subscribers = new Map<\n    number,\n    SubscribeCallback<ParamData, RouterQuerySchema, TContext>\n  >();\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 disposeHistory = history.subscribe(\n    (location, forceRerender, debugContext) => {\n      const origin = resolveNavigationOrigin(debugContext);\n      lastNavigationOrigin = origin;\n      const locationPayload = normalizeLocation(location);\n\n      const debugContextOrigin = debugContext?.origin;\n      if (debugContextOrigin !== undefined) {\n        let historyDetails: { historyIndex: number } | undefined;\n        if (typeof debugContext?.historyIndex === 'number') {\n          historyDetails = { historyIndex: debugContext.historyIndex };\n        }\n\n        if (\n          debugContextOrigin === 'popstate-back' ||\n          debugContextOrigin === 'popstate-forward' ||\n          debugContextOrigin === 'popstate-unknown' ||\n          debugContextOrigin === 'external'\n        ) {\n          let direction: 'back' | 'forward' | 'unknown' = 'unknown';\n          if (debugContextOrigin === 'popstate-back') {\n            direction = 'back';\n          } else if (debugContextOrigin === 'popstate-forward') {\n            direction = 'forward';\n          }\n          emitEvent({\n            kind: 'popstate',\n            source: origin,\n            timestamp: Date.now(),\n            location: locationPayload,\n            direction,\n            details: historyDetails,\n          });\n        }\n      }\n\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      const nextContext = resolveContext();\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        nextRoute = getMatchedRoute(flatRoutes, location);\n      }\n\n      // Build raw query object (basic aggregation) from location.search\n      const querySchema = readActiveQuerySchema(nextRoute);\n      const parsedSearchState = parseSearchState(location.search, querySchema);\n      const normalizedFilters = normalizePageFilters(parsedSearchState.filters);\n\n      // If only the search changed (same pathname) we still need to re-run prepare\n      if (!samePathname || !sameSearch) {\n        let updateInstrumentation:\n          | {\n              emit: (event: RouterEvent) => void;\n              source: RouterNavigationSource;\n              location: RouterLocationSnapshot;\n            }\n          | undefined;\n        if (nextRoute != null) {\n          updateInstrumentation = {\n            emit: (event: RouterEvent) => {\n              emitEvent(event);\n            },\n            source: origin,\n            location: locationPayload,\n          };\n        }\n        nextPreparedMatch = prepareMatch(\n          nextRoute,\n          buildPrepareQueryInput(\n            {\n              ...parsedSearchState,\n              filters: normalizedFilters.filters,\n            },\n            querySchema,\n          ),\n          updateInstrumentation,\n          nextContext,\n        );\n      }\n\n      const nextEntry: RouterEntryState<TContext> = {\n        forceRerender: forceRerender || (samePathname && !sameSearch),\n        location,\n        route: nextRoute,\n        preparedMatch: nextPreparedMatch,\n        context: nextContext,\n        rawSearch: location.search,\n        query: parsedSearchState.query,\n        filters: normalizedFilters.filters,\n        filterDiagnostics: parsedSearchState.filterDiagnostics,\n        activeQuerySchema: querySchema,\n      };\n\n      // If normalization changed the typed query we trigger a replace with normalized search\n      if (normalizedFilters.normalized && querySchema != null) {\n        const normalizedSearch = buildCombinedSearch({\n          filters: normalizedFilters.filters,\n          query: parsedSearchState.query,\n          querySchema,\n        });\n        if (normalizedSearch !== location.search) {\n          let nextSearchStr = normalizedSearch;\n          if (!nextSearchStr.startsWith('?') && nextSearchStr.length > 0) {\n            nextSearchStr = `?${nextSearchStr}`;\n          }\n          emitHistoryEvent(\n            'normalize',\n            'normalize',\n            {\n              pathname: location.pathname,\n              search: nextSearchStr,\n              hash: '',\n            },\n            { reason: 'runtime-page-clamp' },\n          );\n          const runtimeNormalizeContext: HistoryDebugContext = {\n            origin: 'normalize',\n            trigger: 'normalize',\n          };\n          history.set({\n            pathname: location.pathname,\n            search: nextSearchStr,\n            hash: '',\n            debugContext: runtimeNormalizeContext,\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      notifyEntry(nextEntry);\n      emitEvent({\n        kind: 'snapshot',\n        source: origin,\n        timestamp: Date.now(),\n        location: locationPayload,\n        routePath: readMatchedRoutePath(nextEntry),\n      });\n    },\n  );\n\n  /** Normalizes user-provided preload targets into consistent structures. */\n  function normalizePreloadTarget(target: PreloadTarget): {\n    pathname: string;\n    search: string;\n    source: RouterNavigationSource;\n  } {\n    let normalizedPathname = history.location.pathname;\n    if (typeof target.pathname === 'string' && target.pathname.length > 0) {\n      normalizedPathname = target.pathname;\n    }\n\n    let normalizedSearch = '';\n    if (typeof target.search === 'string') {\n      normalizedSearch = target.search;\n    }\n\n    let normalizedSource: RouterNavigationSource = 'programmatic';\n    const mappedSource = mapOrigin(target.source);\n    if (mappedSource !== undefined) {\n      normalizedSource = mappedSource;\n    }\n    return {\n      pathname: normalizedPathname,\n      search: normalizedSearch,\n      source: normalizedSource,\n    };\n  }\n\n  /**\n   * Returns the prepared value for the first active segment matching the provided path.\n   *\n   * @param path - Route segment path or full path to inspect.\n   */\n  function getPrepared(path: string): PreparedOf<RouteUnion<R>> | undefined {\n    try {\n      const entry = currentEntry;\n      for (const preparedRoute of entry.preparedMatch.segments) {\n        if (preparedRoute.path === path || preparedRoute.fullPath === path) {\n          const preparedValue = preparedRoute.preparedResource?.get();\n          return preparedValue as PreparedOf<RouteUnion<R>> | undefined;\n        }\n      }\n      return undefined;\n    } catch {\n      return undefined;\n    }\n  }\n\n  /**\n   * Returns every active prepared segment matching the provided path.\n   *\n   * @param path - Route segment path or full path to inspect.\n   */\n  function getPreparedSegments(path: string): PreparedRouteSegmentSnapshot[] {\n    return currentEntry.preparedMatch.segments\n      .filter((preparedRoute) => {\n        return preparedRoute.path === path || preparedRoute.fullPath === path;\n      })\n      .map((preparedRoute) => {\n        return {\n          path: preparedRoute.path,\n          fullPath: preparedRoute.fullPath,\n          segmentIndex: preparedRoute.segmentIndex,\n          preparedResource: preparedRoute.preparedResource,\n          redirectTo: preparedRoute.redirectTo,\n          resourcePage: preparedRoute.resourcePage,\n          highlightId: preparedRoute.highlightId,\n          querySchema: preparedRoute.querySchema,\n        };\n      });\n  }\n\n  // The router context object that will be passed to React Context\n  const context = {\n    history,\n    get() {\n      return currentEntry;\n    },\n    preloadCode(target) {\n      const normalized = normalizePreloadTarget(target);\n      emitPreloadEvent('code', normalized.pathname, normalized.source);\n      const matches = getMatchedRoute(flatRoutes, {\n        ...history.location,\n        pathname: normalized.pathname,\n      });\n\n      if (matches == null) {\n        return;\n      }\n\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(target) {\n      const normalized = normalizePreloadTarget(target);\n      emitPreloadEvent('full', normalized.pathname, normalized.source);\n      const matches = getMatchedRoute(flatRoutes, {\n        ...history.location,\n        pathname: normalized.pathname,\n      });\n      const locationSnapshot: RouterLocationSnapshot = {\n        pathname: normalized.pathname,\n        search: normalized.search,\n        hash: '',\n      };\n      let instrumentation:\n        | {\n            emit: (event: RouterEvent) => void;\n            source: RouterNavigationSource;\n            location: RouterLocationSnapshot;\n          }\n        | undefined;\n      if (matches != null) {\n        instrumentation = {\n          emit: (event: RouterEvent) => {\n            emitEvent(event);\n          },\n          source: normalized.source,\n          location: locationSnapshot,\n        };\n      }\n      const preloadContext = resolveContext();\n      const preloadQuerySchema = readActiveQuerySchema(matches);\n      const preloadSearchState = parseSearchState(\n        normalized.search,\n        preloadQuerySchema,\n      );\n      const preparedMatch = prepareMatch(\n        matches,\n        buildPrepareQueryInput(preloadSearchState, preloadQuerySchema),\n        instrumentation,\n        preloadContext,\n      );\n      preparedMatch.segments.forEach((route) => {\n        if (route.preparedResource != null) {\n          // eslint-disable-next-line @typescript-eslint/no-floating-promises\n          route.preparedResource.load();\n        }\n      });\n    },\n    subscribe(callback) {\n      nextId += 1;\n      const id = nextId;\n      /** Removes the subscriber previously registered via subscribe. */\n      function disposeCallback() {\n        subscribers.delete(id);\n      }\n      subscribers.set(id, callback);\n      return disposeCallback;\n    },\n    navigate({\n      pathname,\n      query,\n      filters: navFilters,\n      replace,\n    }: {\n      pathname?: string;\n      query?: UnknownQueryShape;\n      filters?: UnknownFilters;\n      replace?: boolean;\n    }) {\n      const current = currentEntry;\n      let targetPathname = pathname;\n      targetPathname ??= current.location.pathname;\n\n      const destinationRoute = getMatchedRoute(flatRoutes, {\n        ...history.location,\n        pathname: targetPathname,\n      });\n      let destSchema = readActiveQuerySchema(destinationRoute);\n      destSchema = destSchema ?? current.activeQuerySchema;\n\n      const hasSchema = destSchema != null;\n      const useQueryAsFilters =\n        navFilters == null &&\n        hasSchema &&\n        query != null &&\n        looksLikeFilters(query);\n      let filtersInput = navFilters;\n      if (filtersInput == null) {\n        if (useQueryAsFilters) {\n          filtersInput = query;\n        } else {\n          filtersInput = current.filters as UnknownFilters | undefined;\n        }\n      }\n      let rawQueryInput: UnknownQueryShape | RawQuery | undefined;\n      if (useQueryAsFilters) {\n        rawQueryInput = current.query;\n      } else {\n        rawQueryInput = query ?? current.query;\n      }\n      const search = buildCombinedSearch({\n        filters: filtersInput,\n        query: rawQueryInput,\n        querySchema: destSchema,\n      });\n      const locationObj = { pathname: targetPathname, search, hash: '' };\n      const origin: RouterNavigationSource = 'programmatic';\n      const debugContext: HistoryDebugContext = {\n        origin,\n        trigger: 'programmatic',\n      };\n\n      if (replace === true) {\n        emitHistoryEvent('replace', origin, locationObj, {\n          trigger: 'programmatic',\n        });\n        history.set({\n          ...locationObj,\n          debugContext,\n        });\n      } else {\n        emitHistoryEvent('push', origin, locationObj, {\n          trigger: 'programmatic',\n        });\n        history.push({\n          ...locationObj,\n          debugContext,\n        });\n      }\n    },\n    getPrepared,\n    getPreparedSegments,\n  } as CreateRouterReturn<TContext, R>['context'];\n\n  /** Releases router resources and detaches all instrumentations. */\n  function cleanup(): void {\n    disposeHistory();\n    subscribers.clear();\n    registry.dispose();\n  }\n\n  // Return both the context object and a cleanup function\n  return { context, cleanup };\n}\n"]}
|