@plumile/router 0.1.43 → 0.1.45
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 +14 -10
- package/lib/esm/builder.d.ts +2 -2
- package/lib/esm/builder.d.ts.map +1 -1
- package/lib/esm/builder.js +1 -1
- package/lib/esm/routing/RouteComponentWrapper.d.ts.map +1 -1
- package/lib/esm/routing/RouteComponentWrapper.js +3 -1
- package/lib/esm/routing/createRouter.d.ts +6 -4
- package/lib/esm/routing/createRouter.d.ts.map +1 -1
- package/lib/esm/routing/createRouter.js +24 -4
- package/lib/esm/tools.d.ts +3 -3
- package/lib/esm/tools.d.ts.map +1 -1
- package/lib/esm/tools.js +3 -2
- package/lib/esm/types.d.ts +24 -19
- 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 +2 -2
- package/lib/types/builder.d.ts.map +1 -1
- package/lib/types/routing/RouteComponentWrapper.d.ts.map +1 -1
- package/lib/types/routing/createRouter.d.ts +6 -4
- package/lib/types/routing/createRouter.d.ts.map +1 -1
- package/lib/types/tools.d.ts +3 -3
- package/lib/types/tools.d.ts.map +1 -1
- package/lib/types/types.d.ts +24 -19
- package/lib/types/types.d.ts.map +1 -1
- package/package.json +7 -7
package/README.md
CHANGED
|
@@ -51,7 +51,7 @@ This package is ESM-only. If your tooling expects CommonJS, enable ESM support (
|
|
|
51
51
|
```typescript
|
|
52
52
|
import { Route, getResourcePage } from '@plumile/router';
|
|
53
53
|
|
|
54
|
-
const routes: Route<any, any>[] = [
|
|
54
|
+
const routes: Route<any, any, any>[] = [
|
|
55
55
|
{
|
|
56
56
|
path: '/',
|
|
57
57
|
resourcePage: getResourcePage('Home', () => import('./pages/Home')),
|
|
@@ -147,13 +147,17 @@ function Navigation() {
|
|
|
147
147
|
|
|
148
148
|
### Core Components
|
|
149
149
|
|
|
150
|
-
#### `createRouter(routes: Route[])`
|
|
150
|
+
#### `createRouter<TContext>(routes: Route<TContext, any, any>[], options?)`
|
|
151
151
|
|
|
152
152
|
Creates a router instance with the given route configuration.
|
|
153
153
|
|
|
154
154
|
**Parameters:**
|
|
155
155
|
|
|
156
156
|
- `routes`: Array of route definitions
|
|
157
|
+
- `options?`: Optional configuration
|
|
158
|
+
- `context?`: Static context value or lazy initializer
|
|
159
|
+
- `getContext?`: Resolve a fresh context value per navigation
|
|
160
|
+
- `instrumentations?`: Instrumentations invoked on router events
|
|
157
161
|
|
|
158
162
|
**Returns:**
|
|
159
163
|
|
|
@@ -194,7 +198,7 @@ React context that provides router functionality to components.
|
|
|
194
198
|
|
|
195
199
|
### Route Configuration
|
|
196
200
|
|
|
197
|
-
#### `Route<TPrepared, TVariables>`
|
|
201
|
+
#### `Route<TContext, TPrepared, TVariables>`
|
|
198
202
|
|
|
199
203
|
Route definition interface.
|
|
200
204
|
|
|
@@ -203,8 +207,8 @@ Route definition interface.
|
|
|
203
207
|
- `path?`: string - URL path pattern
|
|
204
208
|
- `children?`: Route[] | Redirect[] - Nested routes
|
|
205
209
|
- `resourcePage?`: ResourcePage - Lazy-loaded component
|
|
206
|
-
- `prepare?`: Function to preload data
|
|
207
|
-
- `render?`: Custom render function
|
|
210
|
+
- `prepare?`: Function to preload data (receives `context`)
|
|
211
|
+
- `render?`: Custom render function (receives `context`)
|
|
208
212
|
|
|
209
213
|
#### `Redirect`
|
|
210
214
|
|
|
@@ -259,11 +263,11 @@ Browser history implementation.
|
|
|
259
263
|
|
|
260
264
|
Finds the matching route for a given location.
|
|
261
265
|
|
|
262
|
-
#### `prepareMatch(match)`
|
|
266
|
+
#### `prepareMatch(match, query?, instrumentation?, context?)`
|
|
263
267
|
|
|
264
268
|
Prepares route data and components for rendering.
|
|
265
269
|
|
|
266
|
-
#### `r<TPrepared, TVariables>(route)`
|
|
270
|
+
#### `r<TContext, TPrepared, TVariables>(route)`
|
|
267
271
|
|
|
268
272
|
Type helper for strongly-typed route definitions.
|
|
269
273
|
|
|
@@ -559,7 +563,7 @@ Options:
|
|
|
559
563
|
### Data Preloading
|
|
560
564
|
|
|
561
565
|
```typescript
|
|
562
|
-
const route: Route<{ user: User }, { id: string }> = {
|
|
566
|
+
const route: Route<any, { user: User }, { id: string }> = {
|
|
563
567
|
path: '/users/:id',
|
|
564
568
|
prepare: async ({ variables }) => {
|
|
565
569
|
const user = await fetchUser(variables.id);
|
|
@@ -575,7 +579,7 @@ const route: Route<{ user: User }, { id: string }> = {
|
|
|
575
579
|
### Custom Route Rendering
|
|
576
580
|
|
|
577
581
|
```typescript
|
|
578
|
-
const route: Route<any, any> = {
|
|
582
|
+
const route: Route<any, any, any> = {
|
|
579
583
|
path: '/protected',
|
|
580
584
|
render: ({ children, prepared }) => {
|
|
581
585
|
if (!userIsAuthenticated()) {
|
|
@@ -644,7 +648,7 @@ interface UserPageParams {
|
|
|
644
648
|
id: string;
|
|
645
649
|
}
|
|
646
650
|
|
|
647
|
-
const userRoute = r<UserPageData, UserPageParams>({
|
|
651
|
+
const userRoute = r<any, UserPageData, UserPageParams>({
|
|
648
652
|
path: '/users/:id',
|
|
649
653
|
prepare: ({ variables }) => {
|
|
650
654
|
// variables.id is typed as string
|
package/lib/esm/builder.d.ts
CHANGED
|
@@ -8,6 +8,6 @@ export declare class FlatRoute<TParams extends ParamData> {
|
|
|
8
8
|
constructor(input: FlatRouteInput<TParams>);
|
|
9
9
|
}
|
|
10
10
|
export declare function isRedirect(route: AnyRoute | Redirect): route is Redirect;
|
|
11
|
-
export declare function buildRoute(routeConfig: (AnyRoute | Redirect)[], parentRoutes?: AnyRoute[], prefix?: string): FlatRoute<ParamData>[];
|
|
12
|
-
export declare function buildRoutes(routeConfig: AnyRoute[]): FlatRoute<ParamData>[];
|
|
11
|
+
export declare function buildRoute<TContext>(routeConfig: (AnyRoute<TContext> | Redirect)[], parentRoutes?: AnyRoute<TContext>[], prefix?: string): FlatRoute<ParamData>[];
|
|
12
|
+
export declare function buildRoutes<TContext>(routeConfig: AnyRoute<TContext>[]): FlatRoute<ParamData>[];
|
|
13
13
|
//# sourceMappingURL=builder.d.ts.map
|
package/lib/esm/builder.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"builder.d.ts","sourceRoot":"","sources":["../../src/builder.ts"],"names":[],"mappings":"AAAA,OAAO,EAAS,KAAK,aAAa,EAAE,KAAK,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAE3E,OAAO,KAAK,EAAE,cAAc,EAAE,QAAQ,EAAS,QAAQ,EAAE,MAAM,YAAY,CAAC;AAQ5E,qBAAa,SAAS,CAAC,OAAO,SAAS,SAAS;IAEvC,IAAI,EAAE,MAAM,CAAC;IAGb,MAAM,EAAE,QAAQ,EAAE,CAAC;IAGnB,aAAa,EAAE,aAAa,CAAC,OAAO,CAAC,CAAC;IAGtC,UAAU,CAAC,EAAE,MAAM,CAAC;gBAOR,KAAK,EAAE,cAAc,CAAC,OAAO,CAAC;CAOlD;AAGD,wBAAgB,UAAU,CAAC,KAAK,EAAE,QAAQ,GAAG,QAAQ,GAAG,KAAK,IAAI,QAAQ,CAGxE;AASD,wBAAgB,UAAU,
|
|
1
|
+
{"version":3,"file":"builder.d.ts","sourceRoot":"","sources":["../../src/builder.ts"],"names":[],"mappings":"AAAA,OAAO,EAAS,KAAK,aAAa,EAAE,KAAK,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAE3E,OAAO,KAAK,EAAE,cAAc,EAAE,QAAQ,EAAS,QAAQ,EAAE,MAAM,YAAY,CAAC;AAQ5E,qBAAa,SAAS,CAAC,OAAO,SAAS,SAAS;IAEvC,IAAI,EAAE,MAAM,CAAC;IAGb,MAAM,EAAE,QAAQ,EAAE,CAAC;IAGnB,aAAa,EAAE,aAAa,CAAC,OAAO,CAAC,CAAC;IAGtC,UAAU,CAAC,EAAE,MAAM,CAAC;gBAOR,KAAK,EAAE,cAAc,CAAC,OAAO,CAAC;CAOlD;AAGD,wBAAgB,UAAU,CAAC,KAAK,EAAE,QAAQ,GAAG,QAAQ,GAAG,KAAK,IAAI,QAAQ,CAGxE;AASD,wBAAgB,UAAU,CAAC,QAAQ,EACjC,WAAW,EAAE,CAAC,QAAQ,CAAC,QAAQ,CAAC,GAAG,QAAQ,CAAC,EAAE,EAC9C,YAAY,GAAE,QAAQ,CAAC,QAAQ,CAAC,EAAO,EACvC,MAAM,SAAK,GACV,SAAS,CAAC,SAAS,CAAC,EAAE,CA8DxB;AAOD,wBAAgB,WAAW,CAAC,QAAQ,EAClC,WAAW,EAAE,QAAQ,CAAC,QAAQ,CAAC,EAAE,GAChC,SAAS,CAAC,SAAS,CAAC,EAAE,CAExB"}
|
package/lib/esm/builder.js
CHANGED
|
@@ -68,4 +68,4 @@ export function buildRoute(routeConfig, parentRoutes = [], prefix = '') {
|
|
|
68
68
|
export function buildRoutes(routeConfig) {
|
|
69
69
|
return buildRoute(routeConfig);
|
|
70
70
|
}
|
|
71
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
71
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"builder.js","sourceRoot":"","sources":["../../src/builder.ts"],"names":[],"mappings":"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","sourcesContent":["import { match, type MatchFunction, type ParamData } from 'path-to-regexp';\n\nimport type { FlatRouteInput, Redirect, Route, AnyRoute } from './types.js';\n\n/**\n * Represents a flattened route with a compiled match function.\n * This is an internal representation used by the router to efficiently match URLs.\n *\n * @template TParams - Route parameter types extracted from the URL path\n */\nexport class FlatRoute<TParams extends ParamData> {\n  /** The URL path pattern for this route */\n  public path: string;\n\n  /** Nested routes that should be rendered within this route */\n  public routes: AnyRoute[];\n\n  /** Compiled function to match URL paths against this route pattern */\n  public matchFunction: MatchFunction<TParams>;\n\n  /** Optional redirect destination if this route should redirect */\n  public redirectTo?: string;\n\n  /**\n   * Creates a new FlatRoute instance.\n   *\n   * @param input - Configuration for the flat route\n   */\n  public constructor(input: FlatRouteInput<TParams>) {\n    const { matchFunction, path, redirectTo, routes } = input;\n    this.path = path;\n    this.redirectTo = redirectTo;\n    this.routes = routes;\n    this.matchFunction = matchFunction;\n  }\n}\n\n/** Narrow a route configuration to a redirect. */\nexport function isRedirect(route: AnyRoute | Redirect): route is Redirect {\n  // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n  return (route as Redirect).to != null;\n}\n\n/**\n * Recursively flattens nested route definitions into `FlatRoute` entries.\n *\n * @param routeConfig - Route or redirect definitions to process.\n * @param parentRoutes - Accumulated parent routes for nesting context.\n * @param prefix - Current path prefix propagated from parents.\n */\nexport function buildRoute<TContext>(\n  routeConfig: (AnyRoute<TContext> | Redirect)[],\n  parentRoutes: AnyRoute<TContext>[] = [],\n  prefix = '',\n): FlatRoute<ParamData>[] {\n  const flatRoutes: FlatRoute<ParamData>[] = [];\n\n  for (const route of routeConfig) {\n    const parts = [];\n    if (prefix !== '') {\n      parts.push(prefix);\n    }\n    if (route.path != null && route.path !== '' && route.path !== '/') {\n      let normalized = route.path;\n      if (normalized.startsWith('/')) {\n        normalized = normalized.slice(1);\n      }\n      parts.push(normalized);\n    }\n\n    const newPath = parts.join('/');\n\n    const { children } = route as Route<any, any, any, any, any>;\n\n    if (!isRedirect(route) && children != null) {\n      const routes = buildRoute(children, [...parentRoutes, route], newPath);\n      flatRoutes.push(...routes);\n      // eslint-disable-next-line no-continue\n      continue;\n    }\n\n    const matchFunction = match(`/${newPath}`, {\n      trailing: false,\n    });\n\n    let path = newPath;\n    if (!newPath.startsWith('/')) {\n      path = `/${newPath}`;\n    }\n\n    if (isRedirect(route)) {\n      let redirectTo = route.to;\n\n      if (!redirectTo.startsWith('/')) {\n        redirectTo = `${path}/${redirectTo}`;\n      }\n      flatRoutes.push(\n        new FlatRoute({\n          path,\n          redirectTo,\n          matchFunction,\n          routes: [...parentRoutes] as AnyRoute[],\n        }),\n      );\n    } else {\n      flatRoutes.push(\n        new FlatRoute({\n          path,\n          matchFunction,\n          routes: [...parentRoutes, route] as AnyRoute[],\n        }),\n      );\n    }\n  }\n\n  return flatRoutes;\n}\n\n/**\n * Top-level convenience to flatten a route configuration into `FlatRoute`s.\n *\n * @param routeConfig - Route definitions to flatten.\n */\nexport function buildRoutes<TContext>(\n  routeConfig: AnyRoute<TContext>[],\n): FlatRoute<ParamData>[] {\n  return buildRoute(routeConfig);\n}\n"]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"RouteComponentWrapper.d.ts","sourceRoot":"","sources":["../../../src/routing/RouteComponentWrapper.tsx"],"names":[],"mappings":"AAAA,OAAO,EAAuB,KAAK,GAAG,EAAE,KAAK,SAAS,EAAE,MAAM,OAAO,CAAC;AAEtE,OAAO,KAAK,EAAE,kBAAkB,EAAE,kBAAkB,EAAE,MAAM,aAAa,CAAC;AAC1E,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAiChD,KAAK,KAAK,CAAC,KAAK,SAAS,SAAS,GAAG,SAAS,IAAI;IAChD,QAAQ,CAAC,EAAE,SAAS,CAAC;IACrB,KAAK,EAAE,kBAAkB,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC;IACxC,aAAa,EAAE,kBAAkB,CAAC;CACnC,CAAC;AAaF,QAAA,MAAM,qBAAqB,GAAI,KAAK,SAAS,SAAS,GAAG,SAAS,EAChE,OAAO,KAAK,CAAC,KAAK,CAAC,KAClB,GAAG,CAAC,OAAO,GAAG,
|
|
1
|
+
{"version":3,"file":"RouteComponentWrapper.d.ts","sourceRoot":"","sources":["../../../src/routing/RouteComponentWrapper.tsx"],"names":[],"mappings":"AAAA,OAAO,EAAuB,KAAK,GAAG,EAAE,KAAK,SAAS,EAAE,MAAM,OAAO,CAAC;AAEtE,OAAO,KAAK,EAAE,kBAAkB,EAAE,kBAAkB,EAAE,MAAM,aAAa,CAAC;AAC1E,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAiChD,KAAK,KAAK,CAAC,KAAK,SAAS,SAAS,GAAG,SAAS,IAAI;IAChD,QAAQ,CAAC,EAAE,SAAS,CAAC;IACrB,KAAK,EAAE,kBAAkB,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC;IACxC,aAAa,EAAE,kBAAkB,CAAC;CACnC,CAAC;AAaF,QAAA,MAAM,qBAAqB,GAAI,KAAK,SAAS,SAAS,GAAG,SAAS,EAChE,OAAO,KAAK,CAAC,KAAK,CAAC,KAClB,GAAG,CAAC,OAAO,GAAG,IA+GhB,CAAC;AAEF,eAAe,qBAAqB,CAAC"}
|
|
@@ -13,6 +13,7 @@ const RouteComponentWrapper = (props) => {
|
|
|
13
13
|
const router = useContext(RoutingContext);
|
|
14
14
|
const currentEntry = router?.get();
|
|
15
15
|
const rawQuery = currentEntry?.query ?? {};
|
|
16
|
+
const routeContext = currentEntry?.context;
|
|
16
17
|
const typedQuery = rawQuery;
|
|
17
18
|
const route = useMemo(() => {
|
|
18
19
|
if (match == null) {
|
|
@@ -36,6 +37,7 @@ const RouteComponentWrapper = (props) => {
|
|
|
36
37
|
preparedRoute,
|
|
37
38
|
route,
|
|
38
39
|
prepared,
|
|
40
|
+
context: routeContext,
|
|
39
41
|
rawQuery,
|
|
40
42
|
query: typedQuery,
|
|
41
43
|
});
|
|
@@ -85,4 +87,4 @@ const RouteComponentWrapper = (props) => {
|
|
|
85
87
|
return _jsx(RouteComponent, { redirectToPathname: pathname, content: content2 });
|
|
86
88
|
};
|
|
87
89
|
export default RouteComponentWrapper;
|
|
88
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"RouteComponentWrapper.js","sourceRoot":"","sources":["../../../src/routing/RouteComponentWrapper.tsx"],"names":[],"mappings":"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","sourcesContent":["import { useContext, useMemo, type JSX, type ReactNode } from 'react';\n\nimport type { PreparedMatchRoute, RouterMatchedRoute } from '../types.js';\nimport type { ParamData } from 'path-to-regexp';\nimport { HttpRedirect } from '../errors/index.js';\n\nimport RoutingContext from './RoutingContext.js';\nimport RouteComponent from './RouteComponent.js';\n\ntype HighlightBoundaryProps = {\n  highlightId: string;\n  children: ReactNode;\n};\n\n/**\n * Wraps the rendered route output with invisible markers so DevTools can map it back.\n */\nfunction RouteHighlightBoundary(props: HighlightBoundaryProps): JSX.Element {\n  const { highlightId, children } = props;\n  return (\n    <>\n      <span\n        data-plumile-route-start={highlightId}\n        aria-hidden=\"true\"\n        style={{ display: 'none' }}\n      />\n      {children}\n      <span\n        data-plumile-route-end={highlightId}\n        aria-hidden=\"true\"\n        style={{ display: 'none' }}\n      />\n    </>\n  );\n}\n\ntype Props<TVars extends ParamData = ParamData> = {\n  children?: ReactNode;\n  match: RouterMatchedRoute<TVars> | null;\n  preparedRoute: PreparedMatchRoute;\n};\n\n/**\n * The `resourcePage` property from the route entry is a Resource, which may or may not be ready.\n * We use a helper child component to unwrap the resource with component.read(), and then\n * render it if its ready.\n *\n * NOTE: calling routeEntry.route.component.read() directly in RouteRenderer woldn't work the\n * way we'd expect. Because that method could throw - either suspending or on error - the error\n * would bubble up to the *caller* of RouteRenderer. We want the suspend/error to bubble up to\n * our ErrorBoundary/Suspense components, so we have to ensure that the suspend/error happens\n * in a child component.\n */\nconst RouteComponentWrapper = <TVars extends ParamData = ParamData>(\n  props: Props<TVars>,\n): JSX.Element | null => {\n  const { children, match, preparedRoute } = props;\n  const { resourcePage, prepared, render, redirectTo, highlightId } =\n    preparedRoute; // routeData\n  const router = useContext(RoutingContext);\n  const currentEntry = router?.get();\n  const rawQuery = currentEntry?.query ?? {};\n  const typedQuery = rawQuery; // legacy typedQuery removed\n\n  const route = useMemo(() => {\n    if (match == null) {\n      return null;\n    }\n\n    return match.route;\n  }, [match]);\n\n  // eslint-disable-next-line @typescript-eslint/promise-function-async\n  const Component = useMemo(() => {\n    if (resourcePage == null) {\n      return;\n    }\n\n    // eslint-disable-next-line consistent-return\n    return resourcePage.read();\n  }, [resourcePage]);\n\n  let content: JSX.Element | null | undefined | Promise<JSX.Element | null> =\n    null;\n  let redirectToPathname: string | null = null;\n\n  if (render != null) {\n    try {\n      content = render({\n        children,\n        // @ts-expect-error: OK can be a suspend\n        Component,\n        preparedRoute,\n        route,\n        prepared,\n        rawQuery,\n        query: typedQuery,\n      });\n    } catch (error) {\n      if (error instanceof HttpRedirect && router != null) {\n        content = null;\n        redirectToPathname = error.redirectTo;\n        // If it's suspended\n      } else if (error instanceof Promise) {\n        content = error;\n      } else if (typeof error === 'undefined') {\n        content = undefined;\n        redirectToPathname = null;\n      } else {\n        // eslint-disable-next-line no-console\n        console.error('ERROR: ', error);\n        content = null;\n        redirectToPathname = null;\n      }\n    }\n  }\n\n  const pathname: string | null = redirectTo ?? redirectToPathname ?? null;\n\n  const wrapWithHighlight = (node: ReactNode): JSX.Element => {\n    if (typeof highlightId !== 'string') {\n      return <>{node}</>;\n    }\n    return (\n      <RouteHighlightBoundary highlightId={highlightId}>\n        {node}\n      </RouteHighlightBoundary>\n    );\n  };\n\n  if (render != null) {\n    if (content instanceof Promise) {\n      // eslint-disable-next-line @typescript-eslint/only-throw-error\n      throw content;\n    }\n\n    if (pathname != null) {\n      return <RouteComponent redirectToPathname={pathname} />;\n    }\n\n    if (typeof content !== 'undefined') {\n      return wrapWithHighlight(content ?? null);\n    }\n  }\n\n  if (Component == null) {\n    return null;\n  }\n\n  const componentNode = (\n    // @ts-expect-error: OK - component type inference from resource loader is dynamic\n    <Component\n      match={match}\n      preparedRoute={preparedRoute}\n      route={route}\n      prepared={prepared}\n      query={typedQuery}\n    >\n      {children}\n    </Component>\n  );\n  const content2 = wrapWithHighlight(componentNode);\n\n  return <RouteComponent redirectToPathname={pathname} content={content2} />;\n};\n\nexport default RouteComponentWrapper;\n"]}
|
|
90
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"RouteComponentWrapper.js","sourceRoot":"","sources":["../../../src/routing/RouteComponentWrapper.tsx"],"names":[],"mappings":"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","sourcesContent":["import { useContext, useMemo, type JSX, type ReactNode } from 'react';\n\nimport type { PreparedMatchRoute, RouterMatchedRoute } from '../types.js';\nimport type { ParamData } from 'path-to-regexp';\nimport { HttpRedirect } from '../errors/index.js';\n\nimport RoutingContext from './RoutingContext.js';\nimport RouteComponent from './RouteComponent.js';\n\ntype HighlightBoundaryProps = {\n  highlightId: string;\n  children: ReactNode;\n};\n\n/**\n * Wraps the rendered route output with invisible markers so DevTools can map it back.\n */\nfunction RouteHighlightBoundary(props: HighlightBoundaryProps): JSX.Element {\n  const { highlightId, children } = props;\n  return (\n    <>\n      <span\n        data-plumile-route-start={highlightId}\n        aria-hidden=\"true\"\n        style={{ display: 'none' }}\n      />\n      {children}\n      <span\n        data-plumile-route-end={highlightId}\n        aria-hidden=\"true\"\n        style={{ display: 'none' }}\n      />\n    </>\n  );\n}\n\ntype Props<TVars extends ParamData = ParamData> = {\n  children?: ReactNode;\n  match: RouterMatchedRoute<TVars> | null;\n  preparedRoute: PreparedMatchRoute;\n};\n\n/**\n * The `resourcePage` property from the route entry is a Resource, which may or may not be ready.\n * We use a helper child component to unwrap the resource with component.read(), and then\n * render it if its ready.\n *\n * NOTE: calling routeEntry.route.component.read() directly in RouteRenderer woldn't work the\n * way we'd expect. Because that method could throw - either suspending or on error - the error\n * would bubble up to the *caller* of RouteRenderer. We want the suspend/error to bubble up to\n * our ErrorBoundary/Suspense components, so we have to ensure that the suspend/error happens\n * in a child component.\n */\nconst RouteComponentWrapper = <TVars extends ParamData = ParamData>(\n  props: Props<TVars>,\n): JSX.Element | null => {\n  const { children, match, preparedRoute } = props;\n  const { resourcePage, prepared, render, redirectTo, highlightId } =\n    preparedRoute; // routeData\n  const router = useContext(RoutingContext);\n  const currentEntry = router?.get();\n  const rawQuery = currentEntry?.query ?? {};\n  const routeContext = currentEntry?.context;\n  const typedQuery = rawQuery; // legacy typedQuery removed\n\n  const route = useMemo(() => {\n    if (match == null) {\n      return null;\n    }\n\n    return match.route;\n  }, [match]);\n\n  // eslint-disable-next-line @typescript-eslint/promise-function-async\n  const Component = useMemo(() => {\n    if (resourcePage == null) {\n      return;\n    }\n\n    // eslint-disable-next-line consistent-return\n    return resourcePage.read();\n  }, [resourcePage]);\n\n  let content: JSX.Element | null | undefined | Promise<JSX.Element | null> =\n    null;\n  let redirectToPathname: string | null = null;\n\n  if (render != null) {\n    try {\n      content = render({\n        children,\n        // @ts-expect-error: OK can be a suspend\n        Component,\n        preparedRoute,\n        route,\n        prepared,\n        context: routeContext,\n        rawQuery,\n        query: typedQuery,\n      });\n    } catch (error) {\n      if (error instanceof HttpRedirect && router != null) {\n        content = null;\n        redirectToPathname = error.redirectTo;\n        // If it's suspended\n      } else if (error instanceof Promise) {\n        content = error;\n      } else if (typeof error === 'undefined') {\n        content = undefined;\n        redirectToPathname = null;\n      } else {\n        // eslint-disable-next-line no-console\n        console.error('ERROR: ', error);\n        content = null;\n        redirectToPathname = null;\n      }\n    }\n  }\n\n  const pathname: string | null = redirectTo ?? redirectToPathname ?? null;\n\n  const wrapWithHighlight = (node: ReactNode): JSX.Element => {\n    if (typeof highlightId !== 'string') {\n      return <>{node}</>;\n    }\n    return (\n      <RouteHighlightBoundary highlightId={highlightId}>\n        {node}\n      </RouteHighlightBoundary>\n    );\n  };\n\n  if (render != null) {\n    if (content instanceof Promise) {\n      // eslint-disable-next-line @typescript-eslint/only-throw-error\n      throw content;\n    }\n\n    if (pathname != null) {\n      return <RouteComponent redirectToPathname={pathname} />;\n    }\n\n    if (typeof content !== 'undefined') {\n      return wrapWithHighlight(content ?? null);\n    }\n  }\n\n  if (Component == null) {\n    return null;\n  }\n\n  const componentNode = (\n    // @ts-expect-error: OK - component type inference from resource loader is dynamic\n    <Component\n      match={match}\n      preparedRoute={preparedRoute}\n      route={route}\n      prepared={prepared}\n      query={typedQuery}\n    >\n      {children}\n    </Component>\n  );\n  const content2 = wrapWithHighlight(componentNode);\n\n  return <RouteComponent redirectToPathname={pathname} content={content2} />;\n};\n\nexport default RouteComponentWrapper;\n"]}
|
|
@@ -1,11 +1,13 @@
|
|
|
1
1
|
import { type RoutingContextType, type AnyRoute, type PreparedAccess } from '../types.js';
|
|
2
2
|
import { type InstrumentationAPI } from '../instrumentation/Instrumentation.js';
|
|
3
|
-
export type CreateRouterReturn<R extends AnyRoute[]> = {
|
|
3
|
+
export type CreateRouterReturn<TContext, R extends AnyRoute<TContext>[]> = {
|
|
4
4
|
cleanup: () => void;
|
|
5
|
-
context: RoutingContextType<any> & PreparedAccess<R>;
|
|
5
|
+
context: RoutingContextType<any, any, any, TContext> & PreparedAccess<R>;
|
|
6
6
|
};
|
|
7
|
-
export type CreateRouterOptions = {
|
|
7
|
+
export type CreateRouterOptions<TContext> = {
|
|
8
8
|
instrumentations?: InstrumentationAPI[];
|
|
9
|
+
context?: TContext | (() => TContext);
|
|
10
|
+
getContext?: () => TContext;
|
|
9
11
|
};
|
|
10
|
-
export default function createRouter<R extends AnyRoute[]>(routes: [...R] | AnyRoute[], options?: CreateRouterOptions): CreateRouterReturn<R extends AnyRoute[] ? R : AnyRoute[]>;
|
|
12
|
+
export default function createRouter<TContext = unknown, R extends AnyRoute<TContext>[] = AnyRoute<TContext>[]>(routes: [...R] | AnyRoute<TContext>[], options?: CreateRouterOptions<TContext>): CreateRouterReturn<TContext, R extends AnyRoute<TContext>[] ? R : AnyRoute<TContext>[]>;
|
|
11
13
|
//# sourceMappingURL=createRouter.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"createRouter.d.ts","sourceRoot":"","sources":["../../../src/routing/createRouter.ts"],"names":[],"mappings":"AAMA,OAAO,EAEL,KAAK,kBAAkB,EAEvB,KAAK,QAAQ,EACb,KAAK,cAAc,EAGpB,MAAM,aAAa,CAAC;AACrB,OAAO,EAEL,KAAK,kBAAkB,EAKxB,MAAM,uCAAuC,CAAC;AAM/C,MAAM,MAAM,kBAAkB,CAAC,CAAC,SAAS,QAAQ,EAAE,IAAI;
|
|
1
|
+
{"version":3,"file":"createRouter.d.ts","sourceRoot":"","sources":["../../../src/routing/createRouter.ts"],"names":[],"mappings":"AAMA,OAAO,EAEL,KAAK,kBAAkB,EAEvB,KAAK,QAAQ,EACb,KAAK,cAAc,EAGpB,MAAM,aAAa,CAAC;AACrB,OAAO,EAEL,KAAK,kBAAkB,EAKxB,MAAM,uCAAuC,CAAC;AAM/C,MAAM,MAAM,kBAAkB,CAAC,QAAQ,EAAE,CAAC,SAAS,QAAQ,CAAC,QAAQ,CAAC,EAAE,IAAI;IAEzE,OAAO,EAAE,MAAM,IAAI,CAAC;IAEpB,OAAO,EAAE,kBAAkB,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,QAAQ,CAAC,GAAG,cAAc,CAAC,CAAC,CAAC,CAAC;CAC1E,CAAC;AAwCF,MAAM,MAAM,mBAAmB,CAAC,QAAQ,IAAI;IAE1C,gBAAgB,CAAC,EAAE,kBAAkB,EAAE,CAAC;IAExC,OAAO,CAAC,EAAE,QAAQ,GAAG,CAAC,MAAM,QAAQ,CAAC,CAAC;IAEtC,UAAU,CAAC,EAAE,MAAM,QAAQ,CAAC;CAC7B,CAAC;AASF,MAAM,CAAC,OAAO,UAAU,YAAY,CAClC,QAAQ,GAAG,OAAO,EAClB,CAAC,SAAS,QAAQ,CAAC,QAAQ,CAAC,EAAE,GAAG,QAAQ,CAAC,QAAQ,CAAC,EAAE,EAErD,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,GAAG,QAAQ,CAAC,QAAQ,CAAC,EAAE,EACrC,OAAO,GAAE,mBAAmB,CAAC,QAAQ,CAAM,GAC1C,kBAAkB,CACnB,QAAQ,EACR,CAAC,SAAS,QAAQ,CAAC,QAAQ,CAAC,EAAE,GAAG,CAAC,GAAG,QAAQ,CAAC,QAAQ,CAAC,EAAE,CAC1D,CAkwBA"}
|
|
@@ -8,6 +8,21 @@ import { createInstrumentationRegistry, } from '../instrumentation/Instrumentati
|
|
|
8
8
|
export default function createRouter(routes, options = {}) {
|
|
9
9
|
const history = new BrowserHistory();
|
|
10
10
|
const registry = createInstrumentationRegistry(options.instrumentations ?? []);
|
|
11
|
+
let staticContext;
|
|
12
|
+
if (options.getContext == null) {
|
|
13
|
+
if (typeof options.context === 'function') {
|
|
14
|
+
staticContext = options.context();
|
|
15
|
+
}
|
|
16
|
+
else {
|
|
17
|
+
staticContext = options.context;
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
function resolveContext() {
|
|
21
|
+
if (options.getContext != null) {
|
|
22
|
+
return options.getContext();
|
|
23
|
+
}
|
|
24
|
+
return staticContext;
|
|
25
|
+
}
|
|
11
26
|
let pendingNavigationOrigin;
|
|
12
27
|
let lastNavigationOrigin = 'external';
|
|
13
28
|
let lastPreloadSignature = null;
|
|
@@ -194,12 +209,14 @@ export default function createRouter(routes, options = {}) {
|
|
|
194
209
|
location: initialLocationSnapshot,
|
|
195
210
|
};
|
|
196
211
|
}
|
|
197
|
-
const
|
|
212
|
+
const initialContext = resolveContext();
|
|
213
|
+
const preparedMatch = prepareMatch(route, initialFilters, initialInstrumentation, initialContext);
|
|
198
214
|
let currentEntry = {
|
|
199
215
|
forceRerender: false,
|
|
200
216
|
location: history.location,
|
|
201
217
|
route,
|
|
202
218
|
preparedMatch,
|
|
219
|
+
context: initialContext,
|
|
203
220
|
rawSearch: history.location.search,
|
|
204
221
|
query: initialRawQuery,
|
|
205
222
|
filters: initialFilters,
|
|
@@ -300,6 +317,7 @@ export default function createRouter(routes, options = {}) {
|
|
|
300
317
|
if (!forceRerender && samePathname && sameSearch) {
|
|
301
318
|
return;
|
|
302
319
|
}
|
|
320
|
+
const nextContext = resolveContext();
|
|
303
321
|
let nextPreparedMatch = currentEntry.preparedMatch;
|
|
304
322
|
let nextRoute = currentEntry.route;
|
|
305
323
|
if (!samePathname) {
|
|
@@ -351,13 +369,14 @@ export default function createRouter(routes, options = {}) {
|
|
|
351
369
|
location: locationPayload,
|
|
352
370
|
};
|
|
353
371
|
}
|
|
354
|
-
nextPreparedMatch = prepareMatch(nextRoute, undefined, updateInstrumentation);
|
|
372
|
+
nextPreparedMatch = prepareMatch(nextRoute, undefined, updateInstrumentation, nextContext);
|
|
355
373
|
}
|
|
356
374
|
const nextEntry = {
|
|
357
375
|
forceRerender: forceRerender || (samePathname && !sameSearch),
|
|
358
376
|
location,
|
|
359
377
|
route: nextRoute,
|
|
360
378
|
preparedMatch: nextPreparedMatch,
|
|
379
|
+
context: nextContext,
|
|
361
380
|
rawSearch: location.search,
|
|
362
381
|
query,
|
|
363
382
|
filters,
|
|
@@ -470,7 +489,8 @@ export default function createRouter(routes, options = {}) {
|
|
|
470
489
|
location: locationSnapshot,
|
|
471
490
|
};
|
|
472
491
|
}
|
|
473
|
-
|
|
492
|
+
const preloadContext = resolveContext();
|
|
493
|
+
prepareMatch(matches, undefined, instrumentation, preloadContext);
|
|
474
494
|
},
|
|
475
495
|
subscribe(callback) {
|
|
476
496
|
nextId += 1;
|
|
@@ -571,4 +591,4 @@ export default function createRouter(routes, options = {}) {
|
|
|
571
591
|
}
|
|
572
592
|
return { context, cleanup };
|
|
573
593
|
}
|
|
574
|
-
//# 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 { parse as parseFilters } from '@plumile/filter-query';\nimport {\n  type RouteEntry,\n  type RoutingContextType,\n  type SubscribeCallback,\n  type AnyRoute,\n  type PreparedAccess,\n  type NavigateOverloads,\n  type PreloadTarget,\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';\n\n/**\n * Return type for the createRouter function.\n */\nexport type CreateRouterReturn<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<any> & PreparedAccess<R>;\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 = {\n  /** Instrumentations to attach to the router (DevTools bridge, logger, ...). */\n  instrumentations?: InstrumentationAPI[];\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<R extends AnyRoute[]>(\n  routes: [...R] | AnyRoute[],\n  options: CreateRouterOptions = {},\n): CreateRouterReturn<R extends AnyRoute[] ? R : AnyRoute[]> {\n  // Initialize browser history manager\n  const history = new BrowserHistory();\n\n  const registry = createInstrumentationRegistry(\n    options.instrumentations ?? [],\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 (raw === undefined) {\n      return undefined;\n    }\n    switch (raw) {\n      case 'link-click':\n      case 'link-hover':\n      case 'programmatic':\n      case 'popstate-back':\n      case 'popstate-forward':\n      case 'popstate-unknown':\n      case 'external':\n      case 'normalize':\n      case 'preload-hover':\n        return raw;\n      default:\n        return undefined;\n    }\n  }\n\n  const FILTER_OPERATORS = new Set([\n    'gt',\n    'gte',\n    'lt',\n    'lte',\n    'eq',\n    'neq',\n    'contains',\n    'sw',\n    'ew',\n    'between',\n    'in',\n    'nin',\n  ]);\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 (FILTER_OPERATORS.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: RouteEntry<any> | 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  /** Builds a snapshot representation of the current router entry. */\n  function buildEntrySnapshot(entry: RouteEntry<any>): RouterEntrySnapshot {\n    return {\n      location: normalizeLocation(entry.location),\n      routePath: readMatchedRoutePath(entry),\n      preparedMatch: entry.preparedMatch,\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: RouteEntry<any>): void {\n    registry.notifyEntryChange(buildEntrySnapshot(entry));\n  }\n\n  /** Creates a deduplication key for preload events. */\n  function buildPreloadKey(mode: 'code' | 'full', pathname: string): 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: 'push' | 'replace' | 'normalize',\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 initialRawQuery = parseRawQuery(history.location.search);\n  // Direct schema: only the matched route's own querySchema (no hierarchical discovery)\n  let initialUnifiedSchema: any;\n  if (route != null) {\n    const last = route.route.routes.at(-1) as unknown as {\n      querySchema?: unknown;\n    } | null;\n    if (last != null) {\n      initialUnifiedSchema = last.querySchema;\n    }\n  }\n  let initialFilters: Record<string, unknown> | undefined;\n  let initialFilterDiagnostics: any[] | undefined;\n  if (initialUnifiedSchema != null) {\n    const parsed = parseFilters(history.location.search, initialUnifiedSchema);\n    initialFilters = parsed.filters as unknown as Record<string, unknown>;\n    initialFilterDiagnostics = parsed.diagnostics as unknown[];\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 preparedMatch = prepareMatch(\n    route,\n    initialFilters,\n    initialInstrumentation,\n  );\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 removed (unified into filters/query)\n    filters: initialFilters,\n    filterDiagnostics: initialFilterDiagnostics,\n    activeQuerySchema: initialUnifiedSchema,\n  };\n\n  // Initial normalization pass (e.g., clamp page)\n  const initTyped = currentEntry.filters as unknown as\n    | Record<string, unknown>\n    | undefined;\n  let initPageNumeric: number | undefined;\n  if (initTyped != null) {\n    const rawPage = initTyped.page;\n    if (typeof rawPage === 'number') initPageNumeric = rawPage;\n    else if (\n      rawPage != null &&\n      typeof rawPage === 'object' &&\n      !Array.isArray(rawPage) &&\n      Object.prototype.hasOwnProperty.call(rawPage, 'eq') &&\n      typeof (rawPage as Record<string, unknown>).eq === 'number'\n    ) {\n      initPageNumeric = (rawPage as Record<string, unknown>).eq as number;\n    }\n  }\n  if (\n    typeof initPageNumeric === 'number' &&\n    initPageNumeric < 1 &&\n    initialUnifiedSchema != null &&\n    initTyped != null\n  ) {\n    const norm = { ...initTyped, page: { eq: 1 } } as Record<string, unknown>;\n    currentEntry.filters = norm; // immediate visibility\n    const normalizedSearch = buildCombinedSearch({\n      filters: norm,\n      query: initialRawQuery as Record<string, unknown>,\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<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 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      // 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 query = parseRawQuery(location.search);\n      // Determine schema from deepest matched route\n      let querySchema: any;\n      if (nextRoute != null) {\n        const last = nextRoute.route.routes.at(-1) as unknown as {\n          querySchema?: unknown;\n        } | null;\n        if (last != null) {\n          querySchema = last.querySchema;\n        }\n      }\n      // Parse unified filters\n      let filters: Record<string, unknown> | undefined;\n      let filterDiagnostics: any[] | undefined;\n      if (querySchema != null) {\n        const parsedF = parseFilters(location.search, querySchema);\n        filters = parsedF.filters as unknown as Record<string, unknown>;\n        filterDiagnostics = parsedF.diagnostics as unknown[];\n      }\n      // Normalization: clamp page >= 1 if numeric page present\n      let normalized = false;\n      let pg: unknown;\n      if (filters != null) {\n        const f: Record<string, unknown> = filters;\n        const maybe = f.page;\n        if (\n          maybe != null &&\n          typeof maybe === 'object' &&\n          !Array.isArray(maybe) &&\n          Object.prototype.hasOwnProperty.call(maybe, 'eq')\n        ) {\n          pg = (maybe as Record<string, unknown>).eq;\n        } else {\n          pg = maybe;\n        }\n      }\n      if (typeof pg === 'number' && pg < 1) {\n        const clone = { ...(filters as Record<string, any>), page: { eq: 1 } };\n        filters = clone;\n        normalized = true;\n      }\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          undefined,\n          updateInstrumentation,\n        );\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        filters,\n        filterDiagnostics,\n        activeQuerySchema: querySchema,\n      };\n\n      // If normalization changed the typed query we trigger a replace with normalized search\n      if (normalized && querySchema != null) {\n        const normalizedSearch = buildCombinedSearch({\n          filters,\n          query: query as Record<string, unknown>,\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  // The router context object that will be passed to React Context\n  const context: RoutingContextType<any> &\n    PreparedAccess<AnyRoute[]> & {\n      navigate: NavigateOverloads<AnyRoute[]>;\n    } = {\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      prepareMatch(matches, undefined, instrumentation);\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({ pathname, query, filters: navFilters, replace }: any) {\n      const current = currentEntry;\n      let targetPathname = pathname;\n      if (targetPathname === undefined) {\n        targetPathname = current.location.pathname;\n      }\n\n      let destSchema: any;\n      const destinationRoute = getMatchedRoute(flatRoutes, {\n        ...history.location,\n        pathname: targetPathname,\n      });\n      const lastRoute = destinationRoute?.route.routes.at(-1) as\n        | {\n            querySchema?: unknown;\n          }\n        | null\n        | undefined;\n      if (lastRoute != null) {\n        destSchema = lastRoute.querySchema;\n      }\n      destSchema ??= current.activeQuerySchema;\n\n      const typedNavFilters = navFilters as Record<string, unknown> | undefined;\n      const hasSchema = destSchema != null;\n      const useQueryAsFilters =\n        typedNavFilters == null &&\n        hasSchema &&\n        query != null &&\n        looksLikeFilters(query);\n      let filtersInput = typedNavFilters;\n      if (filtersInput == null) {\n        if (useQueryAsFilters) {\n          filtersInput = query as Record<string, unknown>;\n        } else {\n          filtersInput = current.filters;\n        }\n      }\n      let rawQueryInput: Record<string, unknown> | undefined;\n      if (useQueryAsFilters) {\n        rawQueryInput = current.query as Record<string, unknown>;\n      } else {\n        rawQueryInput = (query ?? current.query) as\n          | Record<string, unknown>\n          | undefined;\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(path) {\n      try {\n        const entry = currentEntry;\n        for (const preparedRoute of entry.preparedMatch.routes) {\n          if (preparedRoute.path === path) {\n            return preparedRoute.prepared as any;\n          }\n        }\n        return undefined;\n      } catch {\n        return undefined;\n      }\n    },\n  };\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"]}
|
|
594
|
+
//# 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,aAAa,EAAE,MAAM,mBAAmB,CAAC;AAClD,OAAO,mBAAmB,MAAM,iCAAiC,CAAC;AAClE,OAAO,EAAE,KAAK,IAAI,YAAY,EAAE,MAAM,uBAAuB,CAAC;AAU9D,OAAO,EACL,6BAA6B,GAM9B,MAAM,uCAAuC,CAAC;AAmE/C,MAAM,CAAC,OAAO,UAAU,YAAY,CAIlC,MAAqC,EACrC,UAAyC,EAAE;IAM3C,MAAM,OAAO,GAAG,IAAI,cAAc,EAAE,CAAC;IAErC,MAAM,QAAQ,GAAG,6BAA6B,CAC5C,OAAO,CAAC,gBAAgB,IAAI,EAAE,CAC/B,CAAC;IAEF,IAAI,aAAmC,CAAC;IACxC,IAAI,OAAO,CAAC,UAAU,IAAI,IAAI,EAAE,CAAC;QAC/B,IAAI,OAAO,OAAO,CAAC,OAAO,KAAK,UAAU,EAAE,CAAC;YAC1C,aAAa,GAAI,OAAO,CAAC,OAA0B,EAAE,CAAC;QACxD,CAAC;aAAM,CAAC;YACN,aAAa,GAAG,OAAO,CAAC,OAAO,CAAC;QAClC,CAAC;IACH,CAAC;IAGD,SAAS,cAAc;QACrB,IAAI,OAAO,CAAC,UAAU,IAAI,IAAI,EAAE,CAAC;YAC/B,OAAO,OAAO,CAAC,UAAU,EAAE,CAAC;QAC9B,CAAC;QACD,OAAO,aAAyB,CAAC;IACnC,CAAC;IAED,IAAI,uBAA2D,CAAC;IAChE,IAAI,oBAAoB,GAA2B,UAAU,CAAC;IAC9D,IAAI,oBAAoB,GAGb,IAAI,CAAC;IAKhB,SAAS,iBAAiB,CAAC,QAI1B;QACC,IAAI,aAAa,GAAG,EAAE,CAAC;QACvB,IAAI,OAAO,QAAQ,CAAC,QAAQ,KAAK,QAAQ,EAAE,CAAC;YAC1C,aAAa,GAAG,QAAQ,CAAC,QAAQ,CAAC;QACpC,CAAC;QACD,IAAI,WAAW,GAAG,EAAE,CAAC;QACrB,IAAI,OAAO,QAAQ,CAAC,MAAM,KAAK,QAAQ,EAAE,CAAC;YACxC,WAAW,GAAG,QAAQ,CAAC,MAAM,CAAC;QAChC,CAAC;QACD,IAAI,SAAS,GAAG,EAAE,CAAC;QACnB,IAAI,OAAO,QAAQ,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;YACtC,SAAS,GAAG,QAAQ,CAAC,IAAI,CAAC;QAC5B,CAAC;QACD,OAAO;YACL,QAAQ,EAAE,aAAa;YACvB,MAAM,EAAE,WAAW;YACnB,IAAI,EAAE,SAAS;SAChB,CAAC;IACJ,CAAC;IAGD,SAAS,SAAS,CAAC,KAAkB;QACnC,QAAQ,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;IAC5B,CAAC;IAKD,SAAS,SAAS,CAAC,GAAY;QAC7B,IAAI,GAAG,KAAK,SAAS,EAAE,CAAC;YACtB,OAAO,SAAS,CAAC;QACnB,CAAC;QACD,QAAQ,GAAG,EAAE,CAAC;YACZ,KAAK,YAAY,CAAC;YAClB,KAAK,YAAY,CAAC;YAClB,KAAK,cAAc,CAAC;YACpB,KAAK,eAAe,CAAC;YACrB,KAAK,kBAAkB,CAAC;YACxB,KAAK,kBAAkB,CAAC;YACxB,KAAK,UAAU,CAAC;YAChB,KAAK,WAAW,CAAC;YACjB,KAAK,eAAe;gBAClB,OAAO,GAAG,CAAC;YACb;gBACE,OAAO,SAAS,CAAC;QACrB,CAAC;IACH,CAAC;IAED,MAAM,gBAAgB,GAAG,IAAI,GAAG,CAAC;QAC/B,IAAI;QACJ,KAAK;QACL,IAAI;QACJ,KAAK;QACL,IAAI;QACJ,KAAK;QACL,UAAU;QACV,IAAI;QACJ,IAAI;QACJ,SAAS;QACT,IAAI;QACJ,KAAK;KACN,CAAC,CAAC;IAGH,SAAS,gBAAgB,CAAC,KAAc;QACtC,IAAI,KAAK,IAAI,IAAI,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC;YACvE,OAAO,KAAK,CAAC;QACf,CAAC;QACD,MAAM,MAAM,GAAG,KAAgC,CAAC;QAChD,KAAK,MAAM,UAAU,IAAI,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC;YAC/C,IACE,UAAU,IAAI,IAAI;gBAClB,OAAO,UAAU,KAAK,QAAQ;gBAC9B,CAAC,KAAK,CAAC,OAAO,CAAC,UAAU,CAAC,EAC1B,CAAC;gBACD,KAAK,MAAM,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,UAAqC,CAAC,EAAE,CAAC;oBACrE,IAAI,gBAAgB,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC;wBAC9B,OAAO,IAAI,CAAC;oBACd,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAMD,SAAS,uBAAuB,CAC9B,OAAoC;QAEpC,MAAM,cAAc,GAAG,uBAAuB,CAAC;QAC/C,IAAI,MAA0C,CAAC;QAE/C,MAAM,aAAa,GAAG,OAAO,EAAE,MAAM,CAAC;QACtC,IAAI,aAAa,KAAK,SAAS,EAAE,CAAC;YAChC,MAAM,GAAG,SAAS,CAAC,aAAa,CAAC,CAAC;QACpC,CAAC;QAED,MAAM,KAAK,cAAc,CAAC;QAC1B,MAAM,KAAK,UAAU,CAAC;QACtB,uBAAuB,GAAG,SAAS,CAAC;QACpC,OAAO,MAAM,CAAC;IAChB,CAAC;IAMD,SAAS,oBAAoB,CAC3B,KAAkC;QAElC,IAAI,KAAK,KAAK,SAAS,EAAE,CAAC;YACxB,OAAO,SAAS,CAAC;QACnB,CAAC;QACD,MAAM,YAAY,GAAG,KAAK,CAAC,KAAK,CAAC;QACjC,IAAI,YAAY,KAAK,IAAI,EAAE,CAAC;YAC1B,OAAO,SAAS,CAAC;QACnB,CAAC;QACD,MAAM,WAAW,GAAG,YAAY,CAAC,KAAK,CAAC,MAAM,CAAC;QAC9C,IAAI,KAAK,CAAC,OAAO,CAAC,WAAW,CAAC,EAAE,CAAC;YAC/B,MAAM,SAAS,GAAG,WAAW,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;YACrC,IAAI,SAAS,KAAK,SAAS,EAAE,CAAC;gBAC5B,MAAM,eAAe,GAAG,SAAoC,CAAC;gBAC7D,MAAM,SAAS,GAAG,eAAe,CAAC,IAAI,CAAC;gBACvC,IAAI,OAAO,SAAS,KAAK,QAAQ,EAAE,CAAC;oBAClC,OAAO,SAAS,CAAC;gBACnB,CAAC;YACH,CAAC;QACH,CAAC;QACD,MAAM,QAAQ,GAAG,YAAY,CAAC,KAAK,CAAC,IAAI,CAAC;QACzC,IAAI,OAAO,QAAQ,KAAK,QAAQ,EAAE,CAAC;YACjC,OAAO,QAAQ,CAAC;QAClB,CAAC;QACD,OAAO,SAAS,CAAC;IACnB,CAAC;IAGD,SAAS,kBAAkB,CAAC,KAAsB;QAChD,OAAO;YACL,QAAQ,EAAE,iBAAiB,CAAC,KAAK,CAAC,QAAQ,CAAC;YAC3C,SAAS,EAAE,oBAAoB,CAAC,KAAK,CAAC;YACtC,aAAa,EAAE,KAAK,CAAC,aAAa;YAClC,OAAO,EAAE,KAAK,CAAC,OAAO;YACtB,iBAAiB,EAAE,KAAK,CAAC,iBAAiB;YAC1C,iBAAiB,EAAE,KAAK,CAAC,iBAAiB;SAC3C,CAAC;IACJ,CAAC;IAGD,SAAS,WAAW,CAAC,KAAsB;QACzC,QAAQ,CAAC,iBAAiB,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC,CAAC;IACxD,CAAC;IAGD,SAAS,eAAe,CAAC,IAAqB,EAAE,QAAgB;QAC9D,OAAO,GAAG,IAAI,IAAI,QAAQ,EAAE,CAAC;IAC/B,CAAC;IAGD,SAAS,gBAAgB,CACvB,IAAqB,EACrB,QAAgB,EAChB,MAA8B;QAE9B,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACvB,MAAM,GAAG,GAAG,eAAe,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;QAC5C,MAAM,gBAAgB,GACpB,oBAAoB,KAAK,IAAI;YAC7B,oBAAoB,CAAC,GAAG,KAAK,GAAG;YAChC,GAAG,GAAG,oBAAoB,CAAC,SAAS,GAAG,EAAE,CAAC;QAE5C,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACtB,SAAS,CAAC;gBACR,IAAI,EAAE,SAAS;gBACf,MAAM;gBACN,SAAS,EAAE,GAAG;gBACd,QAAQ,EAAE,iBAAiB,CAAC,OAAO,CAAC,QAAQ,CAAC;gBAC7C,cAAc,EAAE,QAAQ;gBACxB,IAAI;aACL,CAAC,CAAC;QACL,CAAC;QAED,oBAAoB,GAAG,EAAE,GAAG,EAAE,SAAS,EAAE,GAAG,EAAE,CAAC;IACjD,CAAC;IAGD,SAAS,gBAAgB,CACvB,MAAwC,EACxC,MAA8B,EAC9B,QAA8D,EAC9D,OAAiC;QAEjC,uBAAuB,GAAG,MAAM,CAAC;QACjC,SAAS,CAAC;YACR,IAAI,EAAE,SAAS;YACf,MAAM;YACN,MAAM;YACN,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;YACrB,QAAQ,EAAE,iBAAiB,CAAC,QAAQ,CAAC;YACrC,OAAO;SACR,CAAC,CAAC;IACL,CAAC;IAGD,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,oBAAyB,CAAC;IAC9B,IAAI,KAAK,IAAI,IAAI,EAAE,CAAC;QAClB,MAAM,IAAI,GAAG,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,CAE7B,CAAC;QACT,IAAI,IAAI,IAAI,IAAI,EAAE,CAAC;YACjB,oBAAoB,GAAG,IAAI,CAAC,WAAW,CAAC;QAC1C,CAAC;IACH,CAAC;IACD,IAAI,cAAmD,CAAC;IACxD,IAAI,wBAA2C,CAAC;IAChD,IAAI,oBAAoB,IAAI,IAAI,EAAE,CAAC;QACjC,MAAM,MAAM,GAAG,YAAY,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,EAAE,oBAAoB,CAAC,CAAC;QAC3E,cAAc,GAAG,MAAM,CAAC,OAA6C,CAAC;QACtE,wBAAwB,GAAG,MAAM,CAAC,WAAwB,CAAC;IAC7D,CAAC;IACD,MAAM,uBAAuB,GAAG,iBAAiB,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;IACpE,IAAI,sBAMS,CAAC;IACd,IAAI,KAAK,IAAI,IAAI,EAAE,CAAC;QAClB,sBAAsB,GAAG;YACvB,IAAI,EAAE,CAAC,KAAkB,EAAE,EAAE;gBAC3B,SAAS,CAAC,KAAK,CAAC,CAAC;YACnB,CAAC;YACD,MAAM,EAAE,oBAAoB;YAC5B,QAAQ,EAAE,uBAAuB;SAClC,CAAC;IACJ,CAAC;IACD,MAAM,cAAc,GAAG,cAAc,EAAE,CAAC;IACxC,MAAM,aAAa,GAAG,YAAY,CAChC,KAAK,EACL,cAAc,EACd,sBAAsB,EACtB,cAAc,CACf,CAAC;IAEF,IAAI,YAAY,GAAmC;QACjD,aAAa,EAAE,KAAK;QACpB,QAAQ,EAAE,OAAO,CAAC,QAAQ;QAC1B,KAAK;QACL,aAAa;QACb,OAAO,EAAE,cAAc;QACvB,SAAS,EAAE,OAAO,CAAC,QAAQ,CAAC,MAAM;QAClC,KAAK,EAAE,eAAe;QAEtB,OAAO,EAAE,cAAc;QACvB,iBAAiB,EAAE,wBAAwB;QAC3C,iBAAiB,EAAE,oBAAoB;KACxC,CAAC;IAGF,MAAM,SAAS,GAAG,YAAY,CAAC,OAElB,CAAC;IACd,IAAI,eAAmC,CAAC;IACxC,IAAI,SAAS,IAAI,IAAI,EAAE,CAAC;QACtB,MAAM,OAAO,GAAG,SAAS,CAAC,IAAI,CAAC;QAC/B,IAAI,OAAO,OAAO,KAAK,QAAQ;YAAE,eAAe,GAAG,OAAO,CAAC;aACtD,IACH,OAAO,IAAI,IAAI;YACf,OAAO,OAAO,KAAK,QAAQ;YAC3B,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC;YACvB,MAAM,CAAC,SAAS,CAAC,cAAc,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC;YACnD,OAAQ,OAAmC,CAAC,EAAE,KAAK,QAAQ,EAC3D,CAAC;YACD,eAAe,GAAI,OAAmC,CAAC,EAAY,CAAC;QACtE,CAAC;IACH,CAAC;IACD,IACE,OAAO,eAAe,KAAK,QAAQ;QACnC,eAAe,GAAG,CAAC;QACnB,oBAAoB,IAAI,IAAI;QAC5B,SAAS,IAAI,IAAI,EACjB,CAAC;QACD,MAAM,IAAI,GAAG,EAAE,GAAG,SAAS,EAAE,IAAI,EAAE,EAAE,EAAE,EAAE,CAAC,EAAE,EAA6B,CAAC;QAC1E,YAAY,CAAC,OAAO,GAAG,IAAI,CAAC;QAC5B,MAAM,gBAAgB,GAAG,mBAAmB,CAAC;YAC3C,OAAO,EAAE,IAAI;YACb,KAAK,EAAE,eAA0C;YACjD,WAAW,EAAE,oBAAoB;SAClC,CAAC,CAAC;QACH,IAAI,gBAAgB,KAAK,OAAO,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC;YACjD,gBAAgB,CACd,WAAW,EACX,WAAW,EACX;gBACE,QAAQ,EAAE,OAAO,CAAC,QAAQ,CAAC,QAAQ;gBACnC,MAAM,EAAE,gBAAgB;gBACxB,IAAI,EAAE,EAAE;aACT,EACD,EAAE,MAAM,EAAE,oBAAoB,EAAE,CACjC,CAAC;YACF,MAAM,qBAAqB,GAAwB;gBACjD,MAAM,EAAE,WAAW;gBACnB,OAAO,EAAE,WAAW;aACrB,CAAC;YACF,OAAO,CAAC,GAAG,CAAC;gBACV,QAAQ,EAAE,OAAO,CAAC,QAAQ,CAAC,QAAQ;gBACnC,MAAM,EAAE,gBAAgB;gBACxB,IAAI,EAAE,EAAE;gBACR,YAAY,EAAE,qBAAqB;aACpC,CAAC,CAAC;YAEH,YAAY,GAAG;gBACb,GAAG,YAAY;gBACf,QAAQ,EAAE,EAAE,GAAG,YAAY,CAAC,QAAQ,EAAE,MAAM,EAAE,gBAAgB,EAAE;gBAChE,SAAS,EAAE,gBAAgB;aAC5B,CAAC;QACJ,CAAC;IACH,CAAC;IAED,WAAW,CAAC,YAAY,CAAC,CAAC;IAC1B,SAAS,CAAC;QACR,IAAI,EAAE,UAAU;QAChB,MAAM,EAAE,oBAAoB;QAC5B,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;QACrB,QAAQ,EAAE,iBAAiB,CAAC,YAAY,CAAC,QAAQ,CAAC;QAClD,SAAS,EAAE,oBAAoB,CAAC,YAAY,CAAC;KAC9C,CAAC,CAAC;IAGH,IAAI,MAAM,GAAG,CAAC,CAAC;IACf,MAAM,WAAW,GAAG,IAAI,GAAG,EAAiD,CAAC;IAK7E,MAAM,cAAc,GAAG,OAAO,CAAC,SAAS,CACtC,CAAC,QAAQ,EAAE,aAAa,EAAE,YAAY,EAAE,EAAE;QACxC,MAAM,MAAM,GAAG,uBAAuB,CAAC,YAAY,CAAC,CAAC;QACrD,oBAAoB,GAAG,MAAM,CAAC;QAC9B,MAAM,eAAe,GAAG,iBAAiB,CAAC,QAAQ,CAAC,CAAC;QAEpD,MAAM,kBAAkB,GAAG,YAAY,EAAE,MAAM,CAAC;QAChD,IAAI,kBAAkB,KAAK,SAAS,EAAE,CAAC;YACrC,IAAI,cAAoD,CAAC;YACzD,IAAI,OAAO,YAAY,EAAE,YAAY,KAAK,QAAQ,EAAE,CAAC;gBACnD,cAAc,GAAG,EAAE,YAAY,EAAE,YAAY,CAAC,YAAY,EAAE,CAAC;YAC/D,CAAC;YAED,IACE,kBAAkB,KAAK,eAAe;gBACtC,kBAAkB,KAAK,kBAAkB;gBACzC,kBAAkB,KAAK,kBAAkB;gBACzC,kBAAkB,KAAK,UAAU,EACjC,CAAC;gBACD,IAAI,SAAS,GAAmC,SAAS,CAAC;gBAC1D,IAAI,kBAAkB,KAAK,eAAe,EAAE,CAAC;oBAC3C,SAAS,GAAG,MAAM,CAAC;gBACrB,CAAC;qBAAM,IAAI,kBAAkB,KAAK,kBAAkB,EAAE,CAAC;oBACrD,SAAS,GAAG,SAAS,CAAC;gBACxB,CAAC;gBACD,SAAS,CAAC;oBACR,IAAI,EAAE,UAAU;oBAChB,MAAM,EAAE,MAAM;oBACd,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;oBACrB,QAAQ,EAAE,eAAe;oBACzB,SAAS;oBACT,OAAO,EAAE,cAAc;iBACxB,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,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;QAED,MAAM,WAAW,GAAG,cAAc,EAAE,CAAC;QAIrC,IAAI,iBAAiB,GAAG,YAAY,CAAC,aAAa,CAAC;QACnD,IAAI,SAAS,GAAG,YAAY,CAAC,KAAK,CAAC;QAEnC,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,SAAS,GAAG,eAAe,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;QACpD,CAAC;QAGD,MAAM,KAAK,GAAG,aAAa,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;QAE7C,IAAI,WAAgB,CAAC;QACrB,IAAI,SAAS,IAAI,IAAI,EAAE,CAAC;YACtB,MAAM,IAAI,GAAG,SAAS,CAAC,KAAK,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,CAEjC,CAAC;YACT,IAAI,IAAI,IAAI,IAAI,EAAE,CAAC;gBACjB,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;YACjC,CAAC;QACH,CAAC;QAED,IAAI,OAA4C,CAAC;QACjD,IAAI,iBAAoC,CAAC;QACzC,IAAI,WAAW,IAAI,IAAI,EAAE,CAAC;YACxB,MAAM,OAAO,GAAG,YAAY,CAAC,QAAQ,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC;YAC3D,OAAO,GAAG,OAAO,CAAC,OAA6C,CAAC;YAChE,iBAAiB,GAAG,OAAO,CAAC,WAAwB,CAAC;QACvD,CAAC;QAED,IAAI,UAAU,GAAG,KAAK,CAAC;QACvB,IAAI,EAAW,CAAC;QAChB,IAAI,OAAO,IAAI,IAAI,EAAE,CAAC;YACpB,MAAM,CAAC,GAA4B,OAAO,CAAC;YAC3C,MAAM,KAAK,GAAG,CAAC,CAAC,IAAI,CAAC;YACrB,IACE,KAAK,IAAI,IAAI;gBACb,OAAO,KAAK,KAAK,QAAQ;gBACzB,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;gBACrB,MAAM,CAAC,SAAS,CAAC,cAAc,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,EACjD,CAAC;gBACD,EAAE,GAAI,KAAiC,CAAC,EAAE,CAAC;YAC7C,CAAC;iBAAM,CAAC;gBACN,EAAE,GAAG,KAAK,CAAC;YACb,CAAC;QACH,CAAC;QACD,IAAI,OAAO,EAAE,KAAK,QAAQ,IAAI,EAAE,GAAG,CAAC,EAAE,CAAC;YACrC,MAAM,KAAK,GAAG,EAAE,GAAI,OAA+B,EAAE,IAAI,EAAE,EAAE,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC;YACvE,OAAO,GAAG,KAAK,CAAC;YAChB,UAAU,GAAG,IAAI,CAAC;QACpB,CAAC;QAGD,IAAI,CAAC,YAAY,IAAI,CAAC,UAAU,EAAE,CAAC;YACjC,IAAI,qBAMS,CAAC;YACd,IAAI,SAAS,IAAI,IAAI,EAAE,CAAC;gBACtB,qBAAqB,GAAG;oBACtB,IAAI,EAAE,CAAC,KAAkB,EAAE,EAAE;wBAC3B,SAAS,CAAC,KAAK,CAAC,CAAC;oBACnB,CAAC;oBACD,MAAM,EAAE,MAAM;oBACd,QAAQ,EAAE,eAAe;iBAC1B,CAAC;YACJ,CAAC;YACD,iBAAiB,GAAG,YAAY,CAC9B,SAAS,EACT,SAAS,EACT,qBAAqB,EACrB,WAAW,CACZ,CAAC;QACJ,CAAC;QAED,MAAM,SAAS,GAAmC;YAChD,aAAa,EAAE,aAAa,IAAI,CAAC,YAAY,IAAI,CAAC,UAAU,CAAC;YAC7D,QAAQ;YACR,KAAK,EAAE,SAAS;YAChB,aAAa,EAAE,iBAAiB;YAChC,OAAO,EAAE,WAAW;YACpB,SAAS,EAAE,QAAQ,CAAC,MAAM;YAC1B,KAAK;YACL,OAAO;YACP,iBAAiB;YACjB,iBAAiB,EAAE,WAAW;SAC/B,CAAC;QAGF,IAAI,UAAU,IAAI,WAAW,IAAI,IAAI,EAAE,CAAC;YACtC,MAAM,gBAAgB,GAAG,mBAAmB,CAAC;gBAC3C,OAAO;gBACP,KAAK,EAAE,KAAgC;gBACvC,WAAW;aACZ,CAAC,CAAC;YACH,IAAI,gBAAgB,KAAK,QAAQ,CAAC,MAAM,EAAE,CAAC;gBACzC,IAAI,aAAa,GAAG,gBAAgB,CAAC;gBACrC,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBAC/D,aAAa,GAAG,IAAI,aAAa,EAAE,CAAC;gBACtC,CAAC;gBACD,gBAAgB,CACd,WAAW,EACX,WAAW,EACX;oBACE,QAAQ,EAAE,QAAQ,CAAC,QAAQ;oBAC3B,MAAM,EAAE,aAAa;oBACrB,IAAI,EAAE,EAAE;iBACT,EACD,EAAE,MAAM,EAAE,oBAAoB,EAAE,CACjC,CAAC;gBACF,MAAM,uBAAuB,GAAwB;oBACnD,MAAM,EAAE,WAAW;oBACnB,OAAO,EAAE,WAAW;iBACrB,CAAC;gBACF,OAAO,CAAC,GAAG,CAAC;oBACV,QAAQ,EAAE,QAAQ,CAAC,QAAQ;oBAC3B,MAAM,EAAE,aAAa;oBACrB,IAAI,EAAE,EAAE;oBACR,YAAY,EAAE,uBAAuB;iBACtC,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;QACH,WAAW,CAAC,SAAS,CAAC,CAAC;QACvB,SAAS,CAAC;YACR,IAAI,EAAE,UAAU;YAChB,MAAM,EAAE,MAAM;YACd,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;YACrB,QAAQ,EAAE,eAAe;YACzB,SAAS,EAAE,oBAAoB,CAAC,SAAS,CAAC;SAC3C,CAAC,CAAC;IACL,CAAC,CACF,CAAC;IAGF,SAAS,sBAAsB,CAAC,MAAqB;QAKnD,IAAI,kBAAkB,GAAG,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAC;QACnD,IAAI,OAAO,MAAM,CAAC,QAAQ,KAAK,QAAQ,IAAI,MAAM,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtE,kBAAkB,GAAG,MAAM,CAAC,QAAQ,CAAC;QACvC,CAAC;QAED,IAAI,gBAAgB,GAAG,EAAE,CAAC;QAC1B,IAAI,OAAO,MAAM,CAAC,MAAM,KAAK,QAAQ,EAAE,CAAC;YACtC,gBAAgB,GAAG,MAAM,CAAC,MAAM,CAAC;QACnC,CAAC;QAED,IAAI,gBAAgB,GAA2B,cAAc,CAAC;QAC9D,MAAM,YAAY,GAAG,SAAS,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QAC9C,IAAI,YAAY,KAAK,SAAS,EAAE,CAAC;YAC/B,gBAAgB,GAAG,YAAY,CAAC;QAClC,CAAC;QACD,OAAO;YACL,QAAQ,EAAE,kBAAkB;YAC5B,MAAM,EAAE,gBAAgB;YACxB,MAAM,EAAE,gBAAgB;SACzB,CAAC;IACJ,CAAC;IAGD,MAAM,OAAO,GAGP;QACJ,OAAO;QACP,GAAG;YACD,OAAO,YAAY,CAAC;QACtB,CAAC;QACD,WAAW,CAAC,MAAM;YAChB,MAAM,UAAU,GAAG,sBAAsB,CAAC,MAAM,CAAC,CAAC;YAClD,gBAAgB,CAAC,MAAM,EAAE,UAAU,CAAC,QAAQ,EAAE,UAAU,CAAC,MAAM,CAAC,CAAC;YACjE,MAAM,OAAO,GAAG,eAAe,CAAC,UAAU,EAAE;gBAC1C,GAAG,OAAO,CAAC,QAAQ;gBACnB,QAAQ,EAAE,UAAU,CAAC,QAAQ;aAC9B,CAAC,CAAC;YAEH,IAAI,OAAO,IAAI,IAAI,EAAE,CAAC;gBACpB,OAAO;YACT,CAAC;YAED,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,MAAM;YACZ,MAAM,UAAU,GAAG,sBAAsB,CAAC,MAAM,CAAC,CAAC;YAClD,gBAAgB,CAAC,MAAM,EAAE,UAAU,CAAC,QAAQ,EAAE,UAAU,CAAC,MAAM,CAAC,CAAC;YACjE,MAAM,OAAO,GAAG,eAAe,CAAC,UAAU,EAAE;gBAC1C,GAAG,OAAO,CAAC,QAAQ;gBACnB,QAAQ,EAAE,UAAU,CAAC,QAAQ;aAC9B,CAAC,CAAC;YACH,MAAM,gBAAgB,GAA2B;gBAC/C,QAAQ,EAAE,UAAU,CAAC,QAAQ;gBAC7B,MAAM,EAAE,UAAU,CAAC,MAAM;gBACzB,IAAI,EAAE,EAAE;aACT,CAAC;YACF,IAAI,eAMS,CAAC;YACd,IAAI,OAAO,IAAI,IAAI,EAAE,CAAC;gBACpB,eAAe,GAAG;oBAChB,IAAI,EAAE,CAAC,KAAkB,EAAE,EAAE;wBAC3B,SAAS,CAAC,KAAK,CAAC,CAAC;oBACnB,CAAC;oBACD,MAAM,EAAE,UAAU,CAAC,MAAM;oBACzB,QAAQ,EAAE,gBAAgB;iBAC3B,CAAC;YACJ,CAAC;YACD,MAAM,cAAc,GAAG,cAAc,EAAE,CAAC;YACxC,YAAY,CAAC,OAAO,EAAE,SAAS,EAAE,eAAe,EAAE,cAAc,CAAC,CAAC;QACpE,CAAC;QACD,SAAS,CAAC,QAAQ;YAChB,MAAM,IAAI,CAAC,CAAC;YACZ,MAAM,EAAE,GAAG,MAAM,CAAC;YAElB,SAAS,eAAe;gBACtB,WAAW,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;YACzB,CAAC;YACD,WAAW,CAAC,GAAG,CAAC,EAAE,EAAE,QAAQ,CAAC,CAAC;YAC9B,OAAO,eAAe,CAAC;QACzB,CAAC;QACD,QAAQ,CAAC,EAAE,QAAQ,EAAE,KAAK,EAAE,OAAO,EAAE,UAAU,EAAE,OAAO,EAAO;YAC7D,MAAM,OAAO,GAAG,YAAY,CAAC;YAC7B,IAAI,cAAc,GAAG,QAAQ,CAAC;YAC9B,IAAI,cAAc,KAAK,SAAS,EAAE,CAAC;gBACjC,cAAc,GAAG,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAC;YAC7C,CAAC;YAED,IAAI,UAAe,CAAC;YACpB,MAAM,gBAAgB,GAAG,eAAe,CAAC,UAAU,EAAE;gBACnD,GAAG,OAAO,CAAC,QAAQ;gBACnB,QAAQ,EAAE,cAAc;aACzB,CAAC,CAAC;YACH,MAAM,SAAS,GAAG,gBAAgB,EAAE,KAAK,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC,CAKzC,CAAC;YACd,IAAI,SAAS,IAAI,IAAI,EAAE,CAAC;gBACtB,UAAU,GAAG,SAAS,CAAC,WAAW,CAAC;YACrC,CAAC;YACD,UAAU,KAAK,OAAO,CAAC,iBAAiB,CAAC;YAEzC,MAAM,eAAe,GAAG,UAAiD,CAAC;YAC1E,MAAM,SAAS,GAAG,UAAU,IAAI,IAAI,CAAC;YACrC,MAAM,iBAAiB,GACrB,eAAe,IAAI,IAAI;gBACvB,SAAS;gBACT,KAAK,IAAI,IAAI;gBACb,gBAAgB,CAAC,KAAK,CAAC,CAAC;YAC1B,IAAI,YAAY,GAAG,eAAe,CAAC;YACnC,IAAI,YAAY,IAAI,IAAI,EAAE,CAAC;gBACzB,IAAI,iBAAiB,EAAE,CAAC;oBACtB,YAAY,GAAG,KAAgC,CAAC;gBAClD,CAAC;qBAAM,CAAC;oBACN,YAAY,GAAG,OAAO,CAAC,OAAO,CAAC;gBACjC,CAAC;YACH,CAAC;YACD,IAAI,aAAkD,CAAC;YACvD,IAAI,iBAAiB,EAAE,CAAC;gBACtB,aAAa,GAAG,OAAO,CAAC,KAAgC,CAAC;YAC3D,CAAC;iBAAM,CAAC;gBACN,aAAa,GAAG,CAAC,KAAK,IAAI,OAAO,CAAC,KAAK,CAE1B,CAAC;YAChB,CAAC;YACD,MAAM,MAAM,GAAG,mBAAmB,CAAC;gBACjC,OAAO,EAAE,YAAY;gBACrB,KAAK,EAAE,aAAa;gBACpB,WAAW,EAAE,UAAU;aACxB,CAAC,CAAC;YACH,MAAM,WAAW,GAAG,EAAE,QAAQ,EAAE,cAAc,EAAE,MAAM,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC;YACnE,MAAM,MAAM,GAA2B,cAAc,CAAC;YACtD,MAAM,YAAY,GAAwB;gBACxC,MAAM;gBACN,OAAO,EAAE,cAAc;aACxB,CAAC;YAEF,IAAI,OAAO,KAAK,IAAI,EAAE,CAAC;gBACrB,gBAAgB,CAAC,SAAS,EAAE,MAAM,EAAE,WAAW,EAAE;oBAC/C,OAAO,EAAE,cAAc;iBACxB,CAAC,CAAC;gBACH,OAAO,CAAC,GAAG,CAAC;oBACV,GAAG,WAAW;oBACd,YAAY;iBACb,CAAC,CAAC;YACL,CAAC;iBAAM,CAAC;gBACN,gBAAgB,CAAC,MAAM,EAAE,MAAM,EAAE,WAAW,EAAE;oBAC5C,OAAO,EAAE,cAAc;iBACxB,CAAC,CAAC;gBACH,OAAO,CAAC,IAAI,CAAC;oBACX,GAAG,WAAW;oBACd,YAAY;iBACb,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QACD,WAAW,CAAC,IAAI;YACd,IAAI,CAAC;gBACH,MAAM,KAAK,GAAG,YAAY,CAAC;gBAC3B,KAAK,MAAM,aAAa,IAAI,KAAK,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC;oBACvD,IAAI,aAAa,CAAC,IAAI,KAAK,IAAI,EAAE,CAAC;wBAChC,OAAO,aAAa,CAAC,QAAe,CAAC;oBACvC,CAAC;gBACH,CAAC;gBACD,OAAO,SAAS,CAAC;YACnB,CAAC;YAAC,MAAM,CAAC;gBACP,OAAO,SAAS,CAAC;YACnB,CAAC;QACH,CAAC;KACF,CAAC;IAGF,SAAS,OAAO;QACd,cAAc,EAAE,CAAC;QACjB,WAAW,CAAC,KAAK,EAAE,CAAC;QACpB,QAAQ,CAAC,OAAO,EAAE,CAAC;IACrB,CAAC;IAGD,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 { parseRawQuery } from '../tools/query.js';\nimport buildCombinedSearch from '../tools/buildCombinedSearch.js';\nimport { parse as parseFilters } from '@plumile/filter-query';\nimport {\n  type RouteEntry,\n  type RoutingContextType,\n  type SubscribeCallback,\n  type AnyRoute,\n  type PreparedAccess,\n  type NavigateOverloads,\n  type PreloadTarget,\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';\n\n/**\n * Return type for the createRouter function.\n */\nexport type CreateRouterReturn<TContext, R extends AnyRoute<TContext>[]> = {\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, any, any, TContext> & PreparedAccess<R>;\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<TContext>[] = AnyRoute<TContext>[],\n>(\n  routes: [...R] | AnyRoute<TContext>[],\n  options: CreateRouterOptions<TContext> = {},\n): CreateRouterReturn<\n  TContext,\n  R extends AnyRoute<TContext>[] ? R : AnyRoute<TContext>[]\n> {\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 (raw === undefined) {\n      return undefined;\n    }\n    switch (raw) {\n      case 'link-click':\n      case 'link-hover':\n      case 'programmatic':\n      case 'popstate-back':\n      case 'popstate-forward':\n      case 'popstate-unknown':\n      case 'external':\n      case 'normalize':\n      case 'preload-hover':\n        return raw;\n      default:\n        return undefined;\n    }\n  }\n\n  const FILTER_OPERATORS = new Set([\n    'gt',\n    'gte',\n    'lt',\n    'lte',\n    'eq',\n    'neq',\n    'contains',\n    'sw',\n    'ew',\n    'between',\n    'in',\n    'nin',\n  ]);\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 (FILTER_OPERATORS.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: RouteEntry<any> | 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  /** Builds a snapshot representation of the current router entry. */\n  function buildEntrySnapshot(entry: RouteEntry<any>): RouterEntrySnapshot {\n    return {\n      location: normalizeLocation(entry.location),\n      routePath: readMatchedRoutePath(entry),\n      preparedMatch: entry.preparedMatch,\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: RouteEntry<any>): void {\n    registry.notifyEntryChange(buildEntrySnapshot(entry));\n  }\n\n  /** Creates a deduplication key for preload events. */\n  function buildPreloadKey(mode: 'code' | 'full', pathname: string): 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: 'push' | 'replace' | 'normalize',\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 initialRawQuery = parseRawQuery(history.location.search);\n  // Direct schema: only the matched route's own querySchema (no hierarchical discovery)\n  let initialUnifiedSchema: any;\n  if (route != null) {\n    const last = route.route.routes.at(-1) as unknown as {\n      querySchema?: unknown;\n    } | null;\n    if (last != null) {\n      initialUnifiedSchema = last.querySchema;\n    }\n  }\n  let initialFilters: Record<string, unknown> | undefined;\n  let initialFilterDiagnostics: any[] | undefined;\n  if (initialUnifiedSchema != null) {\n    const parsed = parseFilters(history.location.search, initialUnifiedSchema);\n    initialFilters = parsed.filters as unknown as Record<string, unknown>;\n    initialFilterDiagnostics = parsed.diagnostics as unknown[];\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    initialFilters,\n    initialInstrumentation,\n    initialContext,\n  );\n  // Helper to build the raw query object from a search string\n  let currentEntry: RouteEntry<any, any, TContext> = {\n    forceRerender: false,\n    location: history.location,\n    route,\n    preparedMatch,\n    context: initialContext,\n    rawSearch: history.location.search,\n    query: initialRawQuery,\n    // typedQuery removed (unified into filters/query)\n    filters: initialFilters,\n    filterDiagnostics: initialFilterDiagnostics,\n    activeQuerySchema: initialUnifiedSchema,\n  };\n\n  // Initial normalization pass (e.g., clamp page)\n  const initTyped = currentEntry.filters as unknown as\n    | Record<string, unknown>\n    | undefined;\n  let initPageNumeric: number | undefined;\n  if (initTyped != null) {\n    const rawPage = initTyped.page;\n    if (typeof rawPage === 'number') initPageNumeric = rawPage;\n    else if (\n      rawPage != null &&\n      typeof rawPage === 'object' &&\n      !Array.isArray(rawPage) &&\n      Object.prototype.hasOwnProperty.call(rawPage, 'eq') &&\n      typeof (rawPage as Record<string, unknown>).eq === 'number'\n    ) {\n      initPageNumeric = (rawPage as Record<string, unknown>).eq as number;\n    }\n  }\n  if (\n    typeof initPageNumeric === 'number' &&\n    initPageNumeric < 1 &&\n    initialUnifiedSchema != null &&\n    initTyped != null\n  ) {\n    const norm = { ...initTyped, page: { eq: 1 } } as Record<string, unknown>;\n    currentEntry.filters = norm; // immediate visibility\n    const normalizedSearch = buildCombinedSearch({\n      filters: norm,\n      query: initialRawQuery as Record<string, unknown>,\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<number, SubscribeCallback<any, any, TContext>>();\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 query = parseRawQuery(location.search);\n      // Determine schema from deepest matched route\n      let querySchema: any;\n      if (nextRoute != null) {\n        const last = nextRoute.route.routes.at(-1) as unknown as {\n          querySchema?: unknown;\n        } | null;\n        if (last != null) {\n          querySchema = last.querySchema;\n        }\n      }\n      // Parse unified filters\n      let filters: Record<string, unknown> | undefined;\n      let filterDiagnostics: any[] | undefined;\n      if (querySchema != null) {\n        const parsedF = parseFilters(location.search, querySchema);\n        filters = parsedF.filters as unknown as Record<string, unknown>;\n        filterDiagnostics = parsedF.diagnostics as unknown[];\n      }\n      // Normalization: clamp page >= 1 if numeric page present\n      let normalized = false;\n      let pg: unknown;\n      if (filters != null) {\n        const f: Record<string, unknown> = filters;\n        const maybe = f.page;\n        if (\n          maybe != null &&\n          typeof maybe === 'object' &&\n          !Array.isArray(maybe) &&\n          Object.prototype.hasOwnProperty.call(maybe, 'eq')\n        ) {\n          pg = (maybe as Record<string, unknown>).eq;\n        } else {\n          pg = maybe;\n        }\n      }\n      if (typeof pg === 'number' && pg < 1) {\n        const clone = { ...(filters as Record<string, any>), page: { eq: 1 } };\n        filters = clone;\n        normalized = true;\n      }\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          undefined,\n          updateInstrumentation,\n          nextContext,\n        );\n      }\n\n      const nextEntry: RouteEntry<any, any, TContext> = {\n        forceRerender: forceRerender || (samePathname && !sameSearch),\n        location,\n        route: nextRoute,\n        preparedMatch: nextPreparedMatch,\n        context: nextContext,\n        rawSearch: location.search,\n        query,\n        filters,\n        filterDiagnostics,\n        activeQuerySchema: querySchema,\n      };\n\n      // If normalization changed the typed query we trigger a replace with normalized search\n      if (normalized && querySchema != null) {\n        const normalizedSearch = buildCombinedSearch({\n          filters,\n          query: query as Record<string, unknown>,\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  // The router context object that will be passed to React Context\n  const context: RoutingContextType<any, any, any, TContext> &\n    PreparedAccess<AnyRoute<TContext>[]> & {\n      navigate: NavigateOverloads<AnyRoute<TContext>[]>;\n    } = {\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      prepareMatch(matches, undefined, instrumentation, preloadContext);\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({ pathname, query, filters: navFilters, replace }: any) {\n      const current = currentEntry;\n      let targetPathname = pathname;\n      if (targetPathname === undefined) {\n        targetPathname = current.location.pathname;\n      }\n\n      let destSchema: any;\n      const destinationRoute = getMatchedRoute(flatRoutes, {\n        ...history.location,\n        pathname: targetPathname,\n      });\n      const lastRoute = destinationRoute?.route.routes.at(-1) as\n        | {\n            querySchema?: unknown;\n          }\n        | null\n        | undefined;\n      if (lastRoute != null) {\n        destSchema = lastRoute.querySchema;\n      }\n      destSchema ??= current.activeQuerySchema;\n\n      const typedNavFilters = navFilters as Record<string, unknown> | undefined;\n      const hasSchema = destSchema != null;\n      const useQueryAsFilters =\n        typedNavFilters == null &&\n        hasSchema &&\n        query != null &&\n        looksLikeFilters(query);\n      let filtersInput = typedNavFilters;\n      if (filtersInput == null) {\n        if (useQueryAsFilters) {\n          filtersInput = query as Record<string, unknown>;\n        } else {\n          filtersInput = current.filters;\n        }\n      }\n      let rawQueryInput: Record<string, unknown> | undefined;\n      if (useQueryAsFilters) {\n        rawQueryInput = current.query as Record<string, unknown>;\n      } else {\n        rawQueryInput = (query ?? current.query) as\n          | Record<string, unknown>\n          | undefined;\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(path) {\n      try {\n        const entry = currentEntry;\n        for (const preparedRoute of entry.preparedMatch.routes) {\n          if (preparedRoute.path === path) {\n            return preparedRoute.prepared as any;\n          }\n        }\n        return undefined;\n      } catch {\n        return undefined;\n      }\n    },\n  };\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"]}
|
package/lib/esm/tools.d.ts
CHANGED
|
@@ -3,10 +3,10 @@ import { type PrepareResult, type PreparedMatch, type Route, type RouterMatchedR
|
|
|
3
3
|
import type { RouterPrepareEvent, RouterNavigationSource, RouterLocationSnapshot } from './instrumentation/Instrumentation.js';
|
|
4
4
|
import type { FlatRoute } from './builder.js';
|
|
5
5
|
export declare function getMatchedRoute<TParams extends ParamData>(routes: FlatRoute<TParams>[], location: Location): RouterMatchedRoute<TParams> | null;
|
|
6
|
-
export declare function prepareMatch<TParams extends ParamData>(match: RouterMatchedRoute<TParams> | null, query?: Record<string, unknown>, instrumentation?: {
|
|
6
|
+
export declare function prepareMatch<TParams extends ParamData, TContext = unknown>(match: RouterMatchedRoute<TParams> | null, query?: Record<string, unknown>, instrumentation?: {
|
|
7
7
|
emit: (event: RouterPrepareEvent) => void;
|
|
8
8
|
source: RouterNavigationSource;
|
|
9
9
|
location: RouterLocationSnapshot;
|
|
10
|
-
}): PreparedMatch<TParams>;
|
|
11
|
-
export declare function r<TPrepared extends PrepareResult = PrepareResult, TVariables extends ParamData = ParamData, TQuery extends Record<string, unknown> = Record<string, unknown>>(route: Route<TPrepared, TVariables, TQuery>): Route<TPrepared, TVariables, TQuery>;
|
|
10
|
+
}, context?: TContext): PreparedMatch<TParams>;
|
|
11
|
+
export declare function r<TContext = unknown, TPrepared extends PrepareResult = PrepareResult, TVariables extends ParamData = ParamData, TQuery extends Record<string, unknown> = Record<string, unknown>>(route: Route<TContext, TPrepared, TVariables, TQuery>): Route<TContext, TPrepared, TVariables, TQuery>;
|
|
12
12
|
//# sourceMappingURL=tools.d.ts.map
|