@plumile/router 0.1.42 → 0.1.44
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 +27 -14
- 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/Link.d.ts +4 -3
- package/lib/esm/routing/Link.d.ts.map +1 -1
- package/lib/esm/routing/Link.js +33 -7
- 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 +80 -6
- package/lib/esm/routing/index.d.ts +3 -0
- package/lib/esm/routing/index.d.ts.map +1 -1
- package/lib/esm/routing/index.js +4 -1
- package/lib/esm/routing/useLocation.d.ts +2 -0
- package/lib/esm/routing/useLocation.d.ts.map +1 -0
- package/lib/esm/routing/useLocation.js +24 -0
- package/lib/esm/routing/useNavigate.d.ts +2 -1
- package/lib/esm/routing/useNavigate.d.ts.map +1 -1
- package/lib/esm/routing/useNavigate.js +1 -1
- package/lib/esm/routing/usePathname.d.ts +2 -0
- package/lib/esm/routing/usePathname.d.ts.map +1 -0
- package/lib/esm/routing/usePathname.js +9 -0
- package/lib/esm/routing/useSearchParams.d.ts +11 -0
- package/lib/esm/routing/useSearchParams.d.ts.map +1 -0
- package/lib/esm/routing/useSearchParams.js +67 -0
- package/lib/esm/tools/buildCombinedSearch.d.ts +1 -0
- package/lib/esm/tools/buildCombinedSearch.d.ts.map +1 -1
- package/lib/esm/tools/buildCombinedSearch.js +61 -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/Link.d.ts +4 -3
- package/lib/types/routing/Link.d.ts.map +1 -1
- package/lib/types/routing/RouteComponentWrapper.d.ts.map +1 -1
- package/lib/types/routing/createRouter.d.ts +6 -4
- package/lib/types/routing/createRouter.d.ts.map +1 -1
- package/lib/types/routing/index.d.ts +3 -0
- package/lib/types/routing/index.d.ts.map +1 -1
- package/lib/types/routing/useLocation.d.ts +2 -0
- package/lib/types/routing/useLocation.d.ts.map +1 -0
- package/lib/types/routing/useNavigate.d.ts +2 -1
- package/lib/types/routing/useNavigate.d.ts.map +1 -1
- package/lib/types/routing/usePathname.d.ts +2 -0
- package/lib/types/routing/usePathname.d.ts.map +1 -0
- package/lib/types/routing/useSearchParams.d.ts +11 -0
- package/lib/types/routing/useSearchParams.d.ts.map +1 -0
- package/lib/types/tools/buildCombinedSearch.d.ts +1 -0
- package/lib/types/tools/buildCombinedSearch.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 +3 -3
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')),
|
|
@@ -136,6 +136,9 @@ function Navigation() {
|
|
|
136
136
|
| `useNavigate()` | `() => Navigate` | Imperative navigation with type-safe params and filters. | `const navigate = useNavigate(); navigate({ pathname: '/products', filters: { page: { eq: 2 } } });` |
|
|
137
137
|
| `useFilters(schema)` | `(schema) => [filters, actions]` | Read and mutate filters inferred from a schema. | `const [filters, actions] = useFilters(productFilters); actions.set('price', 'gt', 10);` |
|
|
138
138
|
| `useQuery()` | `() => Record<string, string \| string[]>` | Access the raw query aggregation (legacy/simple use). | `const query = useQuery();` |
|
|
139
|
+
| `useLocation()` | `() => Location` | Read the current location object. | `const location = useLocation();` |
|
|
140
|
+
| `usePathname()` | `() => string` | Read the current pathname. | `const pathname = usePathname();` |
|
|
141
|
+
| `useSearchParams()` | `() => SearchParamsActions` | Read and update URLSearchParams. | `const { params, setParam } = useSearchParams();` |
|
|
139
142
|
| `useQueryState(key, options?)` | `(key, options?) => [value, setValue]` | Two-way binding for a single query parameter with default/replace options. | `const [page, setPage] = useQueryState('page', { defaultValue: 1 });` |
|
|
140
143
|
| `useFilterDiagnostics()` | `() => Diagnostic[]` | Surface parsing issues (unknown fields/operators) for UI or logging. | `const diagnostics = useFilterDiagnostics();` |
|
|
141
144
|
| `useAllQuery(options?)` | `(options?) => QueryLike` | Merge filters and raw query, helpful during migrations. | `const all = useAllQuery();` |
|
|
@@ -144,13 +147,17 @@ function Navigation() {
|
|
|
144
147
|
|
|
145
148
|
### Core Components
|
|
146
149
|
|
|
147
|
-
#### `createRouter(routes: Route[])`
|
|
150
|
+
#### `createRouter<TContext>(routes: Route<TContext, any, any>[], options?)`
|
|
148
151
|
|
|
149
152
|
Creates a router instance with the given route configuration.
|
|
150
153
|
|
|
151
154
|
**Parameters:**
|
|
152
155
|
|
|
153
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
|
|
154
161
|
|
|
155
162
|
**Returns:**
|
|
156
163
|
|
|
@@ -173,12 +180,17 @@ Navigation component that handles client-side routing.
|
|
|
173
180
|
|
|
174
181
|
**Props:**
|
|
175
182
|
|
|
176
|
-
- `to`: string - Destination path
|
|
183
|
+
- `to`: string | HistoryLocation - Destination path (supports `search`/`hash`)
|
|
184
|
+
- `filters?`: object - Filters object serialized with the active query schema
|
|
185
|
+
- `query?`: object - Raw query params merged as non-schema keys
|
|
177
186
|
- `exact?`: boolean - Exact path matching for active state
|
|
178
187
|
- `activeClassName?`: string - CSS class when link is active
|
|
179
188
|
- `className?`: string - Base CSS class
|
|
180
|
-
- `
|
|
181
|
-
- `
|
|
189
|
+
- `preloadOnMouseEnter?`: boolean - Preload route on hover
|
|
190
|
+
- `preloadOnMouseDown?`: boolean - Preload route on mouse down
|
|
191
|
+
- `href?`: string - Explicit href override
|
|
192
|
+
- `target?`: string - Target attribute for the link
|
|
193
|
+
- `onClick?`: (event) => void - Click handler
|
|
182
194
|
|
|
183
195
|
#### `RoutingContext`
|
|
184
196
|
|
|
@@ -186,7 +198,7 @@ React context that provides router functionality to components.
|
|
|
186
198
|
|
|
187
199
|
### Route Configuration
|
|
188
200
|
|
|
189
|
-
#### `Route<TPrepared, TVariables>`
|
|
201
|
+
#### `Route<TContext, TPrepared, TVariables>`
|
|
190
202
|
|
|
191
203
|
Route definition interface.
|
|
192
204
|
|
|
@@ -195,8 +207,8 @@ Route definition interface.
|
|
|
195
207
|
- `path?`: string - URL path pattern
|
|
196
208
|
- `children?`: Route[] | Redirect[] - Nested routes
|
|
197
209
|
- `resourcePage?`: ResourcePage - Lazy-loaded component
|
|
198
|
-
- `prepare?`: Function to preload data
|
|
199
|
-
- `render?`: Custom render function
|
|
210
|
+
- `prepare?`: Function to preload data (receives `context`)
|
|
211
|
+
- `render?`: Custom render function (receives `context`)
|
|
200
212
|
|
|
201
213
|
#### `Redirect`
|
|
202
214
|
|
|
@@ -251,11 +263,11 @@ Browser history implementation.
|
|
|
251
263
|
|
|
252
264
|
Finds the matching route for a given location.
|
|
253
265
|
|
|
254
|
-
#### `prepareMatch(match)`
|
|
266
|
+
#### `prepareMatch(match, query?, instrumentation?, context?)`
|
|
255
267
|
|
|
256
268
|
Prepares route data and components for rendering.
|
|
257
269
|
|
|
258
|
-
#### `r<TPrepared, TVariables>(route)`
|
|
270
|
+
#### `r<TContext, TPrepared, TVariables>(route)`
|
|
259
271
|
|
|
260
272
|
Type helper for strongly-typed route definitions.
|
|
261
273
|
|
|
@@ -361,8 +373,9 @@ import { buildCombinedSearch } from '@plumile/router';
|
|
|
361
373
|
const search = buildCombinedSearch({
|
|
362
374
|
filters: { page: { eq: 1 }, price: { gt: 10 } },
|
|
363
375
|
querySchema: productFilters,
|
|
376
|
+
query: { ref: 'promo' },
|
|
364
377
|
});
|
|
365
|
-
// => '?page=1&price.gt=10'
|
|
378
|
+
// => '?page=1&price.gt=10&ref=promo'
|
|
366
379
|
```
|
|
367
380
|
|
|
368
381
|
Guideline: If you need to derive lightweight projections (e.g. `const { page } = typed`), you can still destructure; but avoid spreading into a new object if you rely on reference equality downstream.
|
|
@@ -550,7 +563,7 @@ Options:
|
|
|
550
563
|
### Data Preloading
|
|
551
564
|
|
|
552
565
|
```typescript
|
|
553
|
-
const route: Route<{ user: User }, { id: string }> = {
|
|
566
|
+
const route: Route<any, { user: User }, { id: string }> = {
|
|
554
567
|
path: '/users/:id',
|
|
555
568
|
prepare: async ({ variables }) => {
|
|
556
569
|
const user = await fetchUser(variables.id);
|
|
@@ -566,7 +579,7 @@ const route: Route<{ user: User }, { id: string }> = {
|
|
|
566
579
|
### Custom Route Rendering
|
|
567
580
|
|
|
568
581
|
```typescript
|
|
569
|
-
const route: Route<any, any> = {
|
|
582
|
+
const route: Route<any, any, any> = {
|
|
570
583
|
path: '/protected',
|
|
571
584
|
render: ({ children, prepared }) => {
|
|
572
585
|
if (!userIsAuthenticated()) {
|
|
@@ -635,7 +648,7 @@ interface UserPageParams {
|
|
|
635
648
|
id: string;
|
|
636
649
|
}
|
|
637
650
|
|
|
638
|
-
const userRoute = r<UserPageData, UserPageParams>({
|
|
651
|
+
const userRoute = r<any, UserPageData, UserPageParams>({
|
|
639
652
|
path: '/users/:id',
|
|
640
653
|
prepare: ({ variables }) => {
|
|
641
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,7 +1,7 @@
|
|
|
1
1
|
import React, { type HTMLAttributeAnchorTarget, type ReactNode } from 'react';
|
|
2
2
|
import type { BrowserHistory, HistoryLocation } from '../history/index.js';
|
|
3
3
|
export declare function isCurrentPathname(exact: boolean, history: BrowserHistory | undefined, pathname: string): boolean;
|
|
4
|
-
type Props<
|
|
4
|
+
type Props<TFilters extends Record<string, unknown> = Record<string, unknown>, TRawQuery extends Record<string, unknown> = Record<string, unknown>> = {
|
|
5
5
|
activeClassName?: string;
|
|
6
6
|
children: ReactNode;
|
|
7
7
|
className?: string;
|
|
@@ -15,8 +15,9 @@ type Props<TQuery extends Record<string, unknown> = Record<string, unknown>> = {
|
|
|
15
15
|
onMouseOver?: React.MouseEventHandler<HTMLAnchorElement>;
|
|
16
16
|
target?: HTMLAttributeAnchorTarget;
|
|
17
17
|
to?: HistoryLocation | string;
|
|
18
|
-
|
|
18
|
+
filters?: TFilters;
|
|
19
|
+
query?: TRawQuery;
|
|
19
20
|
};
|
|
20
|
-
declare const Link: React.ForwardRefExoticComponent<Props<Record<string, unknown>> & React.RefAttributes<HTMLAnchorElement>>;
|
|
21
|
+
declare const Link: React.ForwardRefExoticComponent<Props<Record<string, unknown>, Record<string, unknown>> & React.RefAttributes<HTMLAnchorElement>>;
|
|
21
22
|
export default Link;
|
|
22
23
|
//# sourceMappingURL=Link.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"Link.d.ts","sourceRoot":"","sources":["../../../src/routing/Link.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,EACZ,KAAK,yBAAyB,EAE9B,KAAK,SAAS,EAIf,MAAM,OAAO,CAAC;AAEf,OAAO,KAAK,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AAa3E,wBAAgB,iBAAiB,CAC/B,KAAK,EAAE,OAAO,EACd,OAAO,EAAE,cAAc,GAAG,SAAS,EACnC,QAAQ,EAAE,MAAM,GACf,OAAO,CAmBT;AAKD,KAAK,KAAK,CAAC,MAAM,SAAS,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,
|
|
1
|
+
{"version":3,"file":"Link.d.ts","sourceRoot":"","sources":["../../../src/routing/Link.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,EACZ,KAAK,yBAAyB,EAE9B,KAAK,SAAS,EAIf,MAAM,OAAO,CAAC;AAEf,OAAO,KAAK,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AAa3E,wBAAgB,iBAAiB,CAC/B,KAAK,EAAE,OAAO,EACd,OAAO,EAAE,cAAc,GAAG,SAAS,EACnC,QAAQ,EAAE,MAAM,GACf,OAAO,CAmBT;AAKD,KAAK,KAAK,CACR,QAAQ,SAAS,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAClE,SAAS,SAAS,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,IACjE;IAEF,eAAe,CAAC,EAAE,MAAM,CAAC;IAEzB,QAAQ,EAAE,SAAS,CAAC;IAEpB,SAAS,CAAC,EAAE,MAAM,CAAC;IAEnB,KAAK,CAAC,EAAE,OAAO,CAAC;IAEhB,mBAAmB,CAAC,EAAE,OAAO,CAAC;IAE9B,kBAAkB,CAAC,EAAE,OAAO,CAAC;IAE7B,IAAI,CAAC,EAAE,MAAM,CAAC;IAEd,UAAU,CAAC,EAAE,OAAO,CAAC;IAErB,OAAO,CAAC,EAAE,KAAK,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,CAAC;IAErD,OAAO,CAAC,EAAE,KAAK,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,CAAC;IAErD,WAAW,CAAC,EAAE,KAAK,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,CAAC;IAEzD,MAAM,CAAC,EAAE,yBAAyB,CAAC;IAEnC,EAAE,CAAC,EAAE,eAAe,GAAG,MAAM,CAAC;IAE9B,OAAO,CAAC,EAAE,QAAQ,CAAC;IAEnB,KAAK,CAAC,EAAE,SAAS,CAAC;CACnB,CAAC;AAyBF,QAAA,MAAM,IAAI,mIAqMR,CAAC;AAEH,eAAe,IAAI,CAAC"}
|
package/lib/esm/routing/Link.js
CHANGED
|
@@ -20,7 +20,7 @@ export function isCurrentPathname(exact, history, pathname) {
|
|
|
20
20
|
}
|
|
21
21
|
const Link = forwardRef((props, ref) => {
|
|
22
22
|
const router = useContext(RoutingContext);
|
|
23
|
-
const { activeClassName, children, className, exact = false, isDisabled = false, onClick, onFocus, onMouseOver, preloadOnMouseEnter = false, preloadOnMouseDown = true, target, to, href, query, } = props;
|
|
23
|
+
const { activeClassName, children, className, exact = false, isDisabled = false, onClick, onFocus, onMouseOver, preloadOnMouseEnter = false, preloadOnMouseDown = true, target, to, href, filters, query, } = props;
|
|
24
24
|
const pathname = useMemo(() => {
|
|
25
25
|
if (isDisabled) {
|
|
26
26
|
return '#';
|
|
@@ -50,14 +50,37 @@ const Link = forwardRef((props, ref) => {
|
|
|
50
50
|
}
|
|
51
51
|
return `?${to.search}`;
|
|
52
52
|
}
|
|
53
|
-
|
|
53
|
+
const filterInput = filters ?? query;
|
|
54
|
+
let rawQueryInput;
|
|
55
|
+
if (filters != null) {
|
|
56
|
+
rawQueryInput = query;
|
|
57
|
+
}
|
|
58
|
+
if (filterInput == null && rawQueryInput == null)
|
|
59
|
+
return '';
|
|
60
|
+
if (router == null)
|
|
54
61
|
return '';
|
|
55
62
|
const entry = router.get();
|
|
56
63
|
return buildCombinedSearch({
|
|
57
|
-
filters:
|
|
64
|
+
filters: filterInput,
|
|
65
|
+
query: rawQueryInput,
|
|
58
66
|
querySchema: entry.activeQuerySchema,
|
|
59
67
|
});
|
|
60
|
-
}, [isDisabled, query, router, to]);
|
|
68
|
+
}, [filters, isDisabled, query, router, to]);
|
|
69
|
+
const hash = useMemo(() => {
|
|
70
|
+
if (isDisabled) {
|
|
71
|
+
return '';
|
|
72
|
+
}
|
|
73
|
+
if (to != null && typeof to === 'object' && to.hash != null) {
|
|
74
|
+
if (to.hash === '') {
|
|
75
|
+
return '';
|
|
76
|
+
}
|
|
77
|
+
if (to.hash.startsWith('#')) {
|
|
78
|
+
return to.hash;
|
|
79
|
+
}
|
|
80
|
+
return `#${to.hash}`;
|
|
81
|
+
}
|
|
82
|
+
return '';
|
|
83
|
+
}, [isDisabled, to]);
|
|
61
84
|
const initialIsActive = isCurrentPathname(exact, router?.history, pathname);
|
|
62
85
|
const [isActive, setIsActive] = React.useState(initialIsActive);
|
|
63
86
|
useEffect(() => {
|
|
@@ -97,13 +120,13 @@ const Link = forwardRef((props, ref) => {
|
|
|
97
120
|
router.history.push({
|
|
98
121
|
pathname,
|
|
99
122
|
search,
|
|
100
|
-
hash
|
|
123
|
+
hash,
|
|
101
124
|
debugContext: {
|
|
102
125
|
origin: 'link-click',
|
|
103
126
|
trigger: 'link',
|
|
104
127
|
},
|
|
105
128
|
});
|
|
106
|
-
}, [isDisabled, onClick, pathname, router, target, search]);
|
|
129
|
+
}, [hash, isDisabled, onClick, pathname, router, target, search]);
|
|
107
130
|
const handleMouseEnter = useCallback(() => {
|
|
108
131
|
if (router == null || isDisabled) {
|
|
109
132
|
return;
|
|
@@ -126,7 +149,10 @@ const Link = forwardRef((props, ref) => {
|
|
|
126
149
|
if (typeof search === 'string' && search.length > 0) {
|
|
127
150
|
hrefValue = `${pathname}${search}`;
|
|
128
151
|
}
|
|
152
|
+
if (typeof hash === 'string' && hash.length > 0) {
|
|
153
|
+
hrefValue = `${hrefValue}${hash}`;
|
|
154
|
+
}
|
|
129
155
|
return (_jsx("a", { className: cx(classNames), href: hrefValue, onClick: handleClick, onFocus: onFocus, onMouseDown: handleMouseDown, onMouseEnter: handleMouseEnter, onMouseOver: onMouseOver, ref: ref, target: target, children: children }));
|
|
130
156
|
});
|
|
131
157
|
export default Link;
|
|
132
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Link.js","sourceRoot":"","sources":["../../../src/routing/Link.tsx"],"names":[],"mappings":"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","sourcesContent":["import React, {\n  type HTMLAttributeAnchorTarget,\n  type MouseEvent,\n  type ReactNode,\n  forwardRef,\n  useEffect,\n  useMemo,\n} from 'react';\n\nimport type { BrowserHistory, HistoryLocation } from '../history/index.js';\n\nimport { cx } from './../tools/index.js';\nimport buildCombinedSearch from '../tools/buildCombinedSearch.js';\n\nimport RoutingContext from './RoutingContext.js';\n// (filter-query stringify not used here after simplifying manual serialization)\n\nconst { useCallback, useContext } = React;\n\n/**\n * Check if the current pathname matches the given pathname.\n */\nexport function isCurrentPathname(\n  exact: boolean,\n  history: BrowserHistory | undefined,\n  pathname: string,\n): boolean {\n  if (history == null) {\n    return false;\n  }\n\n  // @ts-expect-error: OK\n  const resolvedUrl = new URL(pathname, document.location);\n\n  const resolvedPathname = resolvedUrl.pathname;\n\n  if (exact && history.location.pathname === resolvedPathname) {\n    return true;\n  }\n\n  if (!exact && history.location.pathname.startsWith(resolvedPathname)) {\n    return true;\n  }\n\n  return false;\n}\n\n/**\n * Props for the Link component.\n */\ntype Props<TQuery extends Record<string, unknown> = Record<string, unknown>> = {\n  /** CSS class to apply when the link is active (matches current route) */\n  activeClassName?: string;\n  /** Child components to render inside the link */\n  children: ReactNode;\n  /** Base CSS class for the link */\n  className?: string;\n  /** Whether to use exact path matching for active state */\n  exact?: boolean;\n  /** Whether to preload the route when mouse enters the link */\n  preloadOnMouseEnter?: boolean;\n  /** Whether to preload the route when mouse is pressed down */\n  preloadOnMouseDown?: boolean;\n  /** Direct href attribute (overrides 'to' prop) */\n  href?: string;\n  /** Whether the link should be disabled */\n  isDisabled?: boolean;\n  /** Click handler */\n  onClick?: React.MouseEventHandler<HTMLAnchorElement>;\n  /** Focus handler */\n  onFocus?: React.FocusEventHandler<HTMLAnchorElement>;\n  /** Mouse over handler */\n  onMouseOver?: React.MouseEventHandler<HTMLAnchorElement>;\n  /** Target attribute for the link */\n  target?: HTMLAttributeAnchorTarget;\n  /** Destination for the link (can be string path or location object) */\n  to?: HistoryLocation | string;\n  /** Optional query object to serialize using destination route schema (ignored if `to.search` is set) */\n  query?: TQuery;\n};\n\n/**\n * Navigation component that provides client-side routing with preloading capabilities.\n *\n * This component integrates with the router's RoutingContext to provide smooth navigation\n * with optional preloading of route components and data. It automatically applies active\n * styling when the current route matches the link's destination.\n *\n * @example\n * ```tsx\n * // Basic usage\n * <Link to=\"/users/123\">View Profile</Link>\n *\n * // With active styling\n * <Link to=\"/dashboard\" activeClassName=\"active\" exact>\n *   Dashboard\n * </Link>\n *\n * // With preloading on hover\n * <Link to=\"/heavy-page\" preloadOnMouseEnter>\n *   Heavy Page\n * </Link>\n * ```\n */\nconst Link = forwardRef<HTMLAnchorElement, Props>((props: Props, ref) => {\n  const router = useContext(RoutingContext);\n  const {\n    activeClassName,\n    children,\n    className,\n    exact = false,\n    isDisabled = false,\n    onClick,\n    onFocus,\n    onMouseOver,\n    preloadOnMouseEnter = false,\n    preloadOnMouseDown = true,\n    target,\n    to,\n    href,\n    query,\n  } = props;\n\n  const pathname = useMemo(() => {\n    if (isDisabled) {\n      return '#';\n    }\n\n    let newHref = href;\n\n    if (newHref == null) {\n      if (typeof to === 'string') {\n        newHref = to;\n      } else if (to?.pathname != null) {\n        newHref = to.pathname;\n      }\n    }\n\n    newHref ??= '#';\n\n    return newHref;\n  }, [href, isDisabled, to]);\n\n  // Resolve search string from explicit `to.search` or from provided query\n  // treated as filters with the active schema.\n  const search = useMemo(() => {\n    if (isDisabled) {\n      return '';\n    }\n    if (to != null && typeof to === 'object' && to.search != null) {\n      if (to.search === '') {\n        return '';\n      }\n      if (to.search.startsWith('?')) {\n        return to.search;\n      }\n      return `?${to.search}`;\n    }\n    if (query == null || router == null) return '';\n    const entry = router.get();\n    return buildCombinedSearch({\n      filters: query as any,\n      querySchema: entry.activeQuerySchema as any,\n    });\n  }, [isDisabled, query, router, to]);\n\n  const initialIsActive = isCurrentPathname(exact, router?.history, pathname);\n  const [isActive, setIsActive] = React.useState(initialIsActive);\n\n  useEffect(() => {\n    const onChange = () => {\n      const newIsActive = isCurrentPathname(exact, router?.history, pathname);\n      setIsActive(newIsActive);\n    };\n\n    if (router == null) {\n      return () => {};\n    }\n\n    router.history.subscribe(onChange);\n\n    return () => {\n      router.history.unsubscribe(onChange);\n    };\n  }, [exact, pathname, router]);\n\n  const classNames = [className];\n\n  if (isActive) {\n    classNames.push(activeClassName);\n  }\n\n  // When the user clicks, change route\n  const handleClick = useCallback(\n    (event: MouseEvent<HTMLAnchorElement>) => {\n      if (!isDisabled && typeof onClick === 'function') {\n        onClick(event);\n      }\n\n      if (\n        event.defaultPrevented ||\n        event.metaKey ||\n        event.altKey ||\n        event.ctrlKey ||\n        event.shiftKey ||\n        event.button !== 0 ||\n        (target != null && target !== '_self')\n      ) {\n        return;\n      }\n\n      event.preventDefault();\n      if (router == null || isDisabled) {\n        return;\n      }\n\n      router.history.push({\n        pathname,\n        search,\n        hash: '',\n        debugContext: {\n          origin: 'link-click',\n          trigger: 'link',\n        },\n      });\n    },\n    [isDisabled, onClick, pathname, router, target, search],\n  );\n\n  // Callback to preload just the code for the route:\n  // we pass this to onMouseEnter, which is a weaker signal\n  // that the user *may* navigate to the route.\n  const handleMouseEnter = useCallback(() => {\n    if (router == null || isDisabled) {\n      return;\n    }\n\n    const target = { pathname, search, source: 'preload-hover' } as const;\n    if (preloadOnMouseEnter) {\n      router.preload(target);\n    } else {\n      router.preloadCode(target);\n    }\n  }, [router, isDisabled, preloadOnMouseEnter, pathname, search]);\n\n  // Callback to preload the code and data for the route:\n  // we pass this to onMouseDown, since this is a stronger\n  // signal that the user will likely complete the navigation\n  const handleMouseDown = useCallback(() => {\n    if (router == null || isDisabled || !preloadOnMouseDown) {\n      return;\n    }\n    router.preload({ pathname, search, source: 'link-click' });\n  }, [router, isDisabled, preloadOnMouseDown, pathname, search]);\n\n  let hrefValue = pathname;\n  if (typeof search === 'string' && search.length > 0) {\n    hrefValue = `${pathname}${search}`;\n  }\n  return (\n    <a\n      className={cx(classNames)}\n      href={hrefValue}\n      onClick={handleClick}\n      onFocus={onFocus}\n      onMouseDown={handleMouseDown}\n      onMouseEnter={handleMouseEnter}\n      onMouseOver={onMouseOver}\n      ref={ref}\n      target={target}\n    >\n      {children}\n    </a>\n  );\n});\n\nexport default Link;\n"]}
|
|
158
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Link.js","sourceRoot":"","sources":["../../../src/routing/Link.tsx"],"names":[],"mappings":"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","sourcesContent":["import React, {\n  type HTMLAttributeAnchorTarget,\n  type MouseEvent,\n  type ReactNode,\n  forwardRef,\n  useEffect,\n  useMemo,\n} from 'react';\n\nimport type { BrowserHistory, HistoryLocation } from '../history/index.js';\n\nimport { cx } from './../tools/index.js';\nimport buildCombinedSearch from '../tools/buildCombinedSearch.js';\n\nimport RoutingContext from './RoutingContext.js';\n// (filter-query stringify not used here after simplifying manual serialization)\n\nconst { useCallback, useContext } = React;\n\n/**\n * Check if the current pathname matches the given pathname.\n */\nexport function isCurrentPathname(\n  exact: boolean,\n  history: BrowserHistory | undefined,\n  pathname: string,\n): boolean {\n  if (history == null) {\n    return false;\n  }\n\n  // @ts-expect-error: OK\n  const resolvedUrl = new URL(pathname, document.location);\n\n  const resolvedPathname = resolvedUrl.pathname;\n\n  if (exact && history.location.pathname === resolvedPathname) {\n    return true;\n  }\n\n  if (!exact && history.location.pathname.startsWith(resolvedPathname)) {\n    return true;\n  }\n\n  return false;\n}\n\n/**\n * Props for the Link component.\n */\ntype Props<\n  TFilters extends Record<string, unknown> = Record<string, unknown>,\n  TRawQuery extends Record<string, unknown> = Record<string, unknown>,\n> = {\n  /** CSS class to apply when the link is active (matches current route) */\n  activeClassName?: string;\n  /** Child components to render inside the link */\n  children: ReactNode;\n  /** Base CSS class for the link */\n  className?: string;\n  /** Whether to use exact path matching for active state */\n  exact?: boolean;\n  /** Whether to preload the route when mouse enters the link */\n  preloadOnMouseEnter?: boolean;\n  /** Whether to preload the route when mouse is pressed down */\n  preloadOnMouseDown?: boolean;\n  /** Direct href attribute (overrides 'to' prop) */\n  href?: string;\n  /** Whether the link should be disabled */\n  isDisabled?: boolean;\n  /** Click handler */\n  onClick?: React.MouseEventHandler<HTMLAnchorElement>;\n  /** Focus handler */\n  onFocus?: React.FocusEventHandler<HTMLAnchorElement>;\n  /** Mouse over handler */\n  onMouseOver?: React.MouseEventHandler<HTMLAnchorElement>;\n  /** Target attribute for the link */\n  target?: HTMLAttributeAnchorTarget;\n  /** Destination for the link (can be string path or location object) */\n  to?: HistoryLocation | string;\n  /** Optional filters object to serialize using destination route schema (ignored if `to.search` is set) */\n  filters?: TFilters;\n  /** Optional raw query params (non-schema keys) to merge (ignored if `to.search` is set) */\n  query?: TRawQuery;\n};\n\n/**\n * Navigation component that provides client-side routing with preloading capabilities.\n *\n * This component integrates with the router's RoutingContext to provide smooth navigation\n * with optional preloading of route components and data. It automatically applies active\n * styling when the current route matches the link's destination.\n *\n * @example\n * ```tsx\n * // Basic usage\n * <Link to=\"/users/123\">View Profile</Link>\n *\n * // With active styling\n * <Link to=\"/dashboard\" activeClassName=\"active\" exact>\n *   Dashboard\n * </Link>\n *\n * // With preloading on hover\n * <Link to=\"/heavy-page\" preloadOnMouseEnter>\n *   Heavy Page\n * </Link>\n * ```\n */\nconst Link = forwardRef<HTMLAnchorElement, Props>((props: Props, ref) => {\n  const router = useContext(RoutingContext);\n  const {\n    activeClassName,\n    children,\n    className,\n    exact = false,\n    isDisabled = false,\n    onClick,\n    onFocus,\n    onMouseOver,\n    preloadOnMouseEnter = false,\n    preloadOnMouseDown = true,\n    target,\n    to,\n    href,\n    filters,\n    query,\n  } = props;\n\n  const pathname = useMemo(() => {\n    if (isDisabled) {\n      return '#';\n    }\n\n    let newHref = href;\n\n    if (newHref == null) {\n      if (typeof to === 'string') {\n        newHref = to;\n      } else if (to?.pathname != null) {\n        newHref = to.pathname;\n      }\n    }\n\n    newHref ??= '#';\n\n    return newHref;\n  }, [href, isDisabled, to]);\n\n  // Resolve search string from explicit `to.search` or from provided query\n  // treated as filters with the active schema.\n  const search = useMemo(() => {\n    if (isDisabled) {\n      return '';\n    }\n    if (to != null && typeof to === 'object' && to.search != null) {\n      if (to.search === '') {\n        return '';\n      }\n      if (to.search.startsWith('?')) {\n        return to.search;\n      }\n      return `?${to.search}`;\n    }\n    const filterInput = filters ?? query;\n    let rawQueryInput: typeof query | undefined;\n    if (filters != null) {\n      rawQueryInput = query;\n    }\n    if (filterInput == null && rawQueryInput == null) return '';\n    if (router == null) return '';\n    const entry = router.get();\n    return buildCombinedSearch({\n      filters: filterInput as any,\n      query: rawQueryInput as any,\n      querySchema: entry.activeQuerySchema as any,\n    });\n  }, [filters, isDisabled, query, router, to]);\n\n  const hash = useMemo(() => {\n    if (isDisabled) {\n      return '';\n    }\n    if (to != null && typeof to === 'object' && to.hash != null) {\n      if (to.hash === '') {\n        return '';\n      }\n      if (to.hash.startsWith('#')) {\n        return to.hash;\n      }\n      return `#${to.hash}`;\n    }\n    return '';\n  }, [isDisabled, to]);\n\n  const initialIsActive = isCurrentPathname(exact, router?.history, pathname);\n  const [isActive, setIsActive] = React.useState(initialIsActive);\n\n  useEffect(() => {\n    const onChange = () => {\n      const newIsActive = isCurrentPathname(exact, router?.history, pathname);\n      setIsActive(newIsActive);\n    };\n\n    if (router == null) {\n      return () => {};\n    }\n\n    router.history.subscribe(onChange);\n\n    return () => {\n      router.history.unsubscribe(onChange);\n    };\n  }, [exact, pathname, router]);\n\n  const classNames = [className];\n\n  if (isActive) {\n    classNames.push(activeClassName);\n  }\n\n  // When the user clicks, change route\n  const handleClick = useCallback(\n    (event: MouseEvent<HTMLAnchorElement>) => {\n      if (!isDisabled && typeof onClick === 'function') {\n        onClick(event);\n      }\n\n      if (\n        event.defaultPrevented ||\n        event.metaKey ||\n        event.altKey ||\n        event.ctrlKey ||\n        event.shiftKey ||\n        event.button !== 0 ||\n        (target != null && target !== '_self')\n      ) {\n        return;\n      }\n\n      event.preventDefault();\n      if (router == null || isDisabled) {\n        return;\n      }\n\n      router.history.push({\n        pathname,\n        search,\n        hash,\n        debugContext: {\n          origin: 'link-click',\n          trigger: 'link',\n        },\n      });\n    },\n    [hash, isDisabled, onClick, pathname, router, target, search],\n  );\n\n  // Callback to preload just the code for the route:\n  // we pass this to onMouseEnter, which is a weaker signal\n  // that the user *may* navigate to the route.\n  const handleMouseEnter = useCallback(() => {\n    if (router == null || isDisabled) {\n      return;\n    }\n\n    const target = { pathname, search, source: 'preload-hover' } as const;\n    if (preloadOnMouseEnter) {\n      router.preload(target);\n    } else {\n      router.preloadCode(target);\n    }\n  }, [router, isDisabled, preloadOnMouseEnter, pathname, search]);\n\n  // Callback to preload the code and data for the route:\n  // we pass this to onMouseDown, since this is a stronger\n  // signal that the user will likely complete the navigation\n  const handleMouseDown = useCallback(() => {\n    if (router == null || isDisabled || !preloadOnMouseDown) {\n      return;\n    }\n    router.preload({ pathname, search, source: 'link-click' });\n  }, [router, isDisabled, preloadOnMouseDown, pathname, search]);\n\n  let hrefValue = pathname;\n  if (typeof search === 'string' && search.length > 0) {\n    hrefValue = `${pathname}${search}`;\n  }\n  if (typeof hash === 'string' && hash.length > 0) {\n    hrefValue = `${hrefValue}${hash}`;\n  }\n  return (\n    <a\n      className={cx(classNames)}\n      href={hrefValue}\n      onClick={handleClick}\n      onFocus={onFocus}\n      onMouseDown={handleMouseDown}\n      onMouseEnter={handleMouseEnter}\n      onMouseOver={onMouseOver}\n      ref={ref}\n      target={target}\n    >\n      {children}\n    </a>\n  );\n});\n\nexport default Link;\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"}
|