@plumile/router 0.1.30 → 0.1.31

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (35) hide show
  1. package/lib/esm/ResourcePage.d.ts +1 -0
  2. package/lib/esm/ResourcePage.d.ts.map +1 -1
  3. package/lib/esm/ResourcePage.js +4 -1
  4. package/lib/esm/history/BrowserHistory.d.ts +7 -2
  5. package/lib/esm/history/BrowserHistory.d.ts.map +1 -1
  6. package/lib/esm/history/BrowserHistory.js +96 -18
  7. package/lib/esm/history/types.d.ts +8 -1
  8. package/lib/esm/history/types.d.ts.map +1 -1
  9. package/lib/esm/history/types.js +1 -1
  10. package/lib/esm/routing/Link.d.ts.map +1 -1
  11. package/lib/esm/routing/Link.js +16 -1
  12. package/lib/esm/routing/createRouter.d.ts +1 -1
  13. package/lib/esm/routing/createRouter.d.ts.map +1 -1
  14. package/lib/esm/routing/createRouter.js +363 -24
  15. package/lib/esm/tools.d.ts +6 -2
  16. package/lib/esm/tools.d.ts.map +1 -1
  17. package/lib/esm/tools.js +47 -2
  18. package/lib/esm/types.d.ts +47 -0
  19. package/lib/esm/types.d.ts.map +1 -1
  20. package/lib/esm/types.js +28 -2
  21. package/lib/tsconfig.esm.tsbuildinfo +1 -1
  22. package/lib/types/ResourcePage.d.ts +1 -0
  23. package/lib/types/ResourcePage.d.ts.map +1 -1
  24. package/lib/types/history/BrowserHistory.d.ts +7 -2
  25. package/lib/types/history/BrowserHistory.d.ts.map +1 -1
  26. package/lib/types/history/types.d.ts +8 -1
  27. package/lib/types/history/types.d.ts.map +1 -1
  28. package/lib/types/routing/Link.d.ts.map +1 -1
  29. package/lib/types/routing/createRouter.d.ts +1 -1
  30. package/lib/types/routing/createRouter.d.ts.map +1 -1
  31. package/lib/types/tools.d.ts +6 -2
  32. package/lib/types/tools.d.ts.map +1 -1
  33. package/lib/types/types.d.ts +47 -0
  34. package/lib/types/types.d.ts.map +1 -1
  35. package/package.json +3 -3
@@ -8,6 +8,7 @@ export declare class ResourcePage {
8
8
  constructor(loader: ResourcePageLoader, moduleId: string);
9
9
  load(): Promise<ResourcePageReturn>;
10
10
  get(): ResourcePageReturn | undefined;
11
+ getModuleId(): string;
11
12
  read(): ResourcePageReturn | Promise<ResourcePageReturn> | Error;
12
13
  private __load;
13
14
  }
@@ -1 +1 @@
1
- {"version":3,"file":"ResourcePage.d.ts","sourceRoot":"","sources":["../../src/ResourcePage.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,kBAAkB,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAC;AA2BzE,qBAAa,YAAY;IAEvB,OAAO,CAAC,OAAO,CAAe;IAG9B,OAAO,CAAC,QAAQ,CAAqB;IAGrC,OAAO,CAAC,SAAS,CAAqC;IAGtD,OAAO,CAAC,QAAQ,CAA4B;IAI5C,OAAO,CAAC,UAAU,CAAS;gBAQR,MAAM,EAAE,kBAAkB,EAAE,QAAQ,EAAE,MAAM;IAclD,IAAI,IAAI,OAAO,CAAC,kBAAkB,CAAC;IA8BzC,GAAG,IAAI,kBAAkB,GAAG,SAAS;IAmBrC,IAAI,IAAI,kBAAkB,GAAG,OAAO,CAAC,kBAAkB,CAAC,GAAG,KAAK;YAkBzD,MAAM;CAKrB;AAsBD,wBAAgB,eAAe,CAC7B,QAAQ,EAAE,MAAM,EAChB,MAAM,EAAE,kBAAkB,GACzB,YAAY,GAAG,IAAI,CAOrB"}
1
+ {"version":3,"file":"ResourcePage.d.ts","sourceRoot":"","sources":["../../src/ResourcePage.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,kBAAkB,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAC;AA2BzE,qBAAa,YAAY;IAEvB,OAAO,CAAC,OAAO,CAAe;IAG9B,OAAO,CAAC,QAAQ,CAAqB;IAGrC,OAAO,CAAC,SAAS,CAAqC;IAGtD,OAAO,CAAC,QAAQ,CAA4B;IAG5C,OAAO,CAAC,UAAU,CAAS;gBAQR,MAAM,EAAE,kBAAkB,EAAE,QAAQ,EAAE,MAAM;IAclD,IAAI,IAAI,OAAO,CAAC,kBAAkB,CAAC;IA8BzC,GAAG,IAAI,kBAAkB,GAAG,SAAS;IAWrC,WAAW,IAAI,MAAM;IAerB,IAAI,IAAI,kBAAkB,GAAG,OAAO,CAAC,kBAAkB,CAAC,GAAG,KAAK;YAkBzD,MAAM;CAKrB;AAsBD,wBAAgB,eAAe,CAC7B,QAAQ,EAAE,MAAM,EAChB,MAAM,EAAE,kBAAkB,GACzB,YAAY,GAAG,IAAI,CAOrB"}
@@ -37,6 +37,9 @@ export class ResourcePage {
37
37
  }
38
38
  return undefined;
39
39
  }
40
+ getModuleId() {
41
+ return this.__moduleId;
42
+ }
40
43
  read() {
41
44
  if (this.__result != null) {
42
45
  return this.__result;
@@ -61,4 +64,4 @@ export function getResourcePage(moduleId, loader) {
61
64
  }
62
65
  return resource;
63
66
  }
64
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ResourcePage.js","sourceRoot":"","sources":["../../src/ResourcePage.tsx"],"names":[],"mappings":"AAQA,MAAM,eAAe,GAAG,IAAI,GAAG,EAAwB,CAAC;AAmBxD,MAAM,OAAO,YAAY;IAEf,OAAO,CAAe;IAGtB,QAAQ,CAAqB;IAG7B,SAAS,CAAqC;IAG9C,QAAQ,CAA4B;IAIpC,UAAU,CAAS;IAQ3B,YAAmB,MAA0B,EAAE,QAAgB;QAC7D,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;QACpB,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC;QACvB,IAAI,CAAC,UAAU,GAAG,QAAQ,CAAC;QAC3B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;IACvB,CAAC;IAQM,KAAK,CAAC,IAAI;QACf,IAAI,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC;QAC7B,IAAI,OAAO,IAAI,IAAI,EAAE,CAAC;YACpB,OAAO,GAAG,IAAI,CAAC,MAAM,EAAE;iBACpB,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE;gBAEf,IAAI,MAAM,CAAC,OAAO,IAAI,IAAI,EAAE,CAAC;oBAG3B,MAAM,GAAG,MAAM,CAAC,OAAO,CAAC;gBAC1B,CAAC;gBACD,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC;gBACvB,OAAO,MAAM,CAAC;YAChB,CAAC,CAAC;iBACD,KAAK,CAAC,CAAC,KAAK,EAAE,EAAE;gBACf,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;gBACrB,MAAM,KAAK,CAAC;YACd,CAAC,CAAC,CAAC;YACL,IAAI,CAAC,SAAS,GAAG,OAAO,CAAC;QAC3B,CAAC;QACD,OAAO,OAAO,CAAC;IACjB,CAAC;IASM,GAAG;QACR,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,EAAE,CAAC;YAC1B,OAAO,IAAI,CAAC,QAAQ,CAAC;QACvB,CAAC;QAED,OAAO,SAAS,CAAC;IACnB,CAAC;IAaM,IAAI;QACT,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,EAAE,CAAC;YAC1B,OAAO,IAAI,CAAC,QAAQ,CAAC;QACvB,CAAC;QAED,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,EAAE,CAAC;YACzB,MAAM,IAAI,CAAC,OAAO,CAAC;QACrB,CAAC;aAAM,CAAC;YAEN,MAAM,IAAI,CAAC,SAAS,CAAC;QACvB,CAAC;IACH,CAAC;IAOO,KAAK,CAAC,MAAM;QAClB,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC;QAElC,OAAO,OAAO,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC;IACnC,CAAC;CACF;AAsBD,MAAM,UAAU,eAAe,CAC7B,QAAgB,EAChB,MAA0B;IAE1B,IAAI,QAAQ,GAAG,eAAe,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IAC7C,IAAI,QAAQ,IAAI,IAAI,EAAE,CAAC;QACrB,QAAQ,GAAG,IAAI,YAAY,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;QAC9C,eAAe,CAAC,GAAG,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;IAC1C,CAAC;IACD,OAAO,QAAQ,CAAC;AAClB,CAAC","sourcesContent":["import type { ResourcePageLoader, ResourcePageReturn } from './types.js';\n\n/**\n * A cache of resources to avoid loading the same module twice. This is important\n * because Webpack dynamic imports only expose an asynchronous API for loading\n * modules, so to be able to access already-loaded modules synchronously we\n * must have stored the previous result somewhere.\n */\nconst resourcePageMap = new Map<string, ResourcePage>();\n\n/**\n * A resource manager for lazy-loaded React components with Suspense integration.\n * This class handles the loading state, caching, and Suspense integration for\n * dynamically imported components.\n *\n * @example\n * ```typescript\n * const resource = new ResourcePage(\n *   () => import('./MyComponent'),\n *   'MyComponent'\n * );\n *\n * // In a React component with Suspense boundary:\n * const Component = resource.read();\n * return <Component />;\n * ```\n */\nexport class ResourcePage {\n  /** Error state if the resource failed to load */\n  private __error: Error | null;\n\n  /** Function to load the resource */\n  private __loader: ResourcePageLoader;\n\n  /** Promise representing the loading state */\n  private __promise: Promise<ResourcePageReturn> | null;\n\n  /** The loaded resource result */\n  private __result: ResourcePageReturn | null;\n\n  /** Unique identifier for this resource */\n  // @ts-expect-error: OK\n  private __moduleId: string;\n\n  /**\n   * Creates a new ResourcePage instance.\n   *\n   * @param loader - Function that returns a Promise resolving to the component\n   * @param moduleId - Unique identifier for caching purposes\n   */\n  public constructor(loader: ResourcePageLoader, moduleId: string) {\n    this.__error = null;\n    this.__loader = loader;\n    this.__moduleId = moduleId;\n    this.__promise = null;\n    this.__result = null;\n  }\n\n  /**\n   * Loads the resource if not already loaded or loading.\n   * This method can be called multiple times safely - it will return the same promise.\n   *\n   * @returns Promise that resolves to the loaded component\n   */\n  public async load(): Promise<ResourcePageReturn> {\n    let promise = this.__promise;\n    if (promise == null) {\n      promise = this.__load()\n        .then((result) => {\n          // @ts-expect-error: OK\n          if (result.default != null) {\n            // @ts-expect-error: OK\n            // eslint-disable-next-line no-param-reassign\n            result = result.default;\n          }\n          this.__result = result;\n          return result;\n        })\n        .catch((error) => {\n          this.__error = error;\n          throw error;\n        });\n      this.__promise = promise;\n    }\n    return promise;\n  }\n\n  /**\n   * Returns the loaded component if available, undefined otherwise.\n   * This method can be used to check if the component is already loaded\n   * without triggering a load or throwing an error.\n   *\n   * @returns The loaded component or undefined if not yet loaded\n   */\n  public get(): ResourcePageReturn | undefined {\n    if (this.__result != null) {\n      return this.__result;\n    }\n\n    return undefined;\n  }\n\n  /**\n   * Reads the resource with React Suspense integration.\n   * This is the key method for integrating with React Suspense:\n   * - Returns the component if loaded\n   * - Throws an error if loading failed\n   * - Throws a Promise if still loading (Suspense will catch this)\n   *\n   * @returns The loaded component\n   * @throws Promise when loading (caught by Suspense)\n   * @throws Error when loading failed\n   */\n  public read(): ResourcePageReturn | Promise<ResourcePageReturn> | Error {\n    if (this.__result != null) {\n      return this.__result;\n    }\n\n    if (this.__error != null) {\n      throw this.__error;\n    } else {\n      // eslint-disable-next-line @typescript-eslint/only-throw-error\n      throw this.__promise;\n    }\n  }\n\n  /**\n   * Internal method to execute the loader function.\n   *\n   * @returns Promise resolving to the component\n   */\n  private async __load(): Promise<ResourcePageReturn> {\n    const { __loader: loader } = this;\n\n    return Promise.resolve(loader());\n  }\n}\n\n/**\n * Factory function to create or retrieve a cached ResourcePage instance.\n * This function implements a singleton pattern per moduleId to ensure that\n * the same component is not loaded multiple times.\n *\n * @param moduleId - Globally unique identifier for the resource used for caching\n * @param loader - Function that returns a Promise resolving to the component\n * @returns ResourcePage instance (cached if previously created)\n *\n * @example\n * ```typescript\n * // Create a resource for lazy loading\n * const resource = getResourcePage('UserProfile', () => import('./UserProfile'));\n * resource.load();\n *\n * // In a React component with Suspense boundary:\n * const UserProfile = resource.read();\n * return <UserProfile userId={userId} />;\n * ```\n */\nexport function getResourcePage(\n  moduleId: string,\n  loader: ResourcePageLoader,\n): ResourcePage | null {\n  let resource = resourcePageMap.get(moduleId);\n  if (resource == null) {\n    resource = new ResourcePage(loader, moduleId);\n    resourcePageMap.set(moduleId, resource);\n  }\n  return resource;\n}\n"]}
67
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ResourcePage.js","sourceRoot":"","sources":["../../src/ResourcePage.tsx"],"names":[],"mappings":"AAQA,MAAM,eAAe,GAAG,IAAI,GAAG,EAAwB,CAAC;AAmBxD,MAAM,OAAO,YAAY;IAEf,OAAO,CAAe;IAGtB,QAAQ,CAAqB;IAG7B,SAAS,CAAqC;IAG9C,QAAQ,CAA4B;IAGpC,UAAU,CAAS;IAQ3B,YAAmB,MAA0B,EAAE,QAAgB;QAC7D,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;QACpB,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC;QACvB,IAAI,CAAC,UAAU,GAAG,QAAQ,CAAC;QAC3B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;IACvB,CAAC;IAQM,KAAK,CAAC,IAAI;QACf,IAAI,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC;QAC7B,IAAI,OAAO,IAAI,IAAI,EAAE,CAAC;YACpB,OAAO,GAAG,IAAI,CAAC,MAAM,EAAE;iBACpB,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE;gBAEf,IAAI,MAAM,CAAC,OAAO,IAAI,IAAI,EAAE,CAAC;oBAG3B,MAAM,GAAG,MAAM,CAAC,OAAO,CAAC;gBAC1B,CAAC;gBACD,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC;gBACvB,OAAO,MAAM,CAAC;YAChB,CAAC,CAAC;iBACD,KAAK,CAAC,CAAC,KAAK,EAAE,EAAE;gBACf,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;gBACrB,MAAM,KAAK,CAAC;YACd,CAAC,CAAC,CAAC;YACL,IAAI,CAAC,SAAS,GAAG,OAAO,CAAC;QAC3B,CAAC;QACD,OAAO,OAAO,CAAC;IACjB,CAAC;IASM,GAAG;QACR,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,EAAE,CAAC;YAC1B,OAAO,IAAI,CAAC,QAAQ,CAAC;QACvB,CAAC;QAED,OAAO,SAAS,CAAC;IACnB,CAAC;IAKM,WAAW;QAChB,OAAO,IAAI,CAAC,UAAU,CAAC;IACzB,CAAC;IAaM,IAAI;QACT,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,EAAE,CAAC;YAC1B,OAAO,IAAI,CAAC,QAAQ,CAAC;QACvB,CAAC;QAED,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,EAAE,CAAC;YACzB,MAAM,IAAI,CAAC,OAAO,CAAC;QACrB,CAAC;aAAM,CAAC;YAEN,MAAM,IAAI,CAAC,SAAS,CAAC;QACvB,CAAC;IACH,CAAC;IAOO,KAAK,CAAC,MAAM;QAClB,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC;QAElC,OAAO,OAAO,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC;IACnC,CAAC;CACF;AAsBD,MAAM,UAAU,eAAe,CAC7B,QAAgB,EAChB,MAA0B;IAE1B,IAAI,QAAQ,GAAG,eAAe,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IAC7C,IAAI,QAAQ,IAAI,IAAI,EAAE,CAAC;QACrB,QAAQ,GAAG,IAAI,YAAY,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;QAC9C,eAAe,CAAC,GAAG,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;IAC1C,CAAC;IACD,OAAO,QAAQ,CAAC;AAClB,CAAC","sourcesContent":["import type { ResourcePageLoader, ResourcePageReturn } from './types.js';\n\n/**\n * A cache of resources to avoid loading the same module twice. This is important\n * because Webpack dynamic imports only expose an asynchronous API for loading\n * modules, so to be able to access already-loaded modules synchronously we\n * must have stored the previous result somewhere.\n */\nconst resourcePageMap = new Map<string, ResourcePage>();\n\n/**\n * A resource manager for lazy-loaded React components with Suspense integration.\n * This class handles the loading state, caching, and Suspense integration for\n * dynamically imported components.\n *\n * @example\n * ```typescript\n * const resource = new ResourcePage(\n *   () => import('./MyComponent'),\n *   'MyComponent'\n * );\n *\n * // In a React component with Suspense boundary:\n * const Component = resource.read();\n * return <Component />;\n * ```\n */\nexport class ResourcePage {\n  /** Error state if the resource failed to load */\n  private __error: Error | null;\n\n  /** Function to load the resource */\n  private __loader: ResourcePageLoader;\n\n  /** Promise representing the loading state */\n  private __promise: Promise<ResourcePageReturn> | null;\n\n  /** The loaded resource result */\n  private __result: ResourcePageReturn | null;\n\n  /** Unique identifier for this resource */\n  private __moduleId: string;\n\n  /**\n   * Creates a new ResourcePage instance.\n   *\n   * @param loader - Function that returns a Promise resolving to the component\n   * @param moduleId - Unique identifier for caching purposes\n   */\n  public constructor(loader: ResourcePageLoader, moduleId: string) {\n    this.__error = null;\n    this.__loader = loader;\n    this.__moduleId = moduleId;\n    this.__promise = null;\n    this.__result = null;\n  }\n\n  /**\n   * Loads the resource if not already loaded or loading.\n   * This method can be called multiple times safely - it will return the same promise.\n   *\n   * @returns Promise that resolves to the loaded component\n   */\n  public async load(): Promise<ResourcePageReturn> {\n    let promise = this.__promise;\n    if (promise == null) {\n      promise = this.__load()\n        .then((result) => {\n          // @ts-expect-error: OK\n          if (result.default != null) {\n            // @ts-expect-error: OK\n            // eslint-disable-next-line no-param-reassign\n            result = result.default;\n          }\n          this.__result = result;\n          return result;\n        })\n        .catch((error) => {\n          this.__error = error;\n          throw error;\n        });\n      this.__promise = promise;\n    }\n    return promise;\n  }\n\n  /**\n   * Returns the loaded component if available, undefined otherwise.\n   * This method can be used to check if the component is already loaded\n   * without triggering a load or throwing an error.\n   *\n   * @returns The loaded component or undefined if not yet loaded\n   */\n  public get(): ResourcePageReturn | undefined {\n    if (this.__result != null) {\n      return this.__result;\n    }\n\n    return undefined;\n  }\n\n  /**\n   * Returns the module identifier associated with this resource (debug helper).\n   */\n  public getModuleId(): string {\n    return this.__moduleId;\n  }\n\n  /**\n   * Reads the resource with React Suspense integration.\n   * This is the key method for integrating with React Suspense:\n   * - Returns the component if loaded\n   * - Throws an error if loading failed\n   * - Throws a Promise if still loading (Suspense will catch this)\n   *\n   * @returns The loaded component\n   * @throws Promise when loading (caught by Suspense)\n   * @throws Error when loading failed\n   */\n  public read(): ResourcePageReturn | Promise<ResourcePageReturn> | Error {\n    if (this.__result != null) {\n      return this.__result;\n    }\n\n    if (this.__error != null) {\n      throw this.__error;\n    } else {\n      // eslint-disable-next-line @typescript-eslint/only-throw-error\n      throw this.__promise;\n    }\n  }\n\n  /**\n   * Internal method to execute the loader function.\n   *\n   * @returns Promise resolving to the component\n   */\n  private async __load(): Promise<ResourcePageReturn> {\n    const { __loader: loader } = this;\n\n    return Promise.resolve(loader());\n  }\n}\n\n/**\n * Factory function to create or retrieve a cached ResourcePage instance.\n * This function implements a singleton pattern per moduleId to ensure that\n * the same component is not loaded multiple times.\n *\n * @param moduleId - Globally unique identifier for the resource used for caching\n * @param loader - Function that returns a Promise resolving to the component\n * @returns ResourcePage instance (cached if previously created)\n *\n * @example\n * ```typescript\n * // Create a resource for lazy loading\n * const resource = getResourcePage('UserProfile', () => import('./UserProfile'));\n * resource.load();\n *\n * // In a React component with Suspense boundary:\n * const UserProfile = resource.read();\n * return <UserProfile userId={userId} />;\n * ```\n */\nexport function getResourcePage(\n  moduleId: string,\n  loader: ResourcePageLoader,\n): ResourcePage | null {\n  let resource = resourcePageMap.get(moduleId);\n  if (resource == null) {\n    resource = new ResourcePage(loader, moduleId);\n    resourcePageMap.set(moduleId, resource);\n  }\n  return resource;\n}\n"]}
@@ -2,6 +2,9 @@ import type { History, HistoryListener, HistoryLocation } from './types.js';
2
2
  export declare function createPath(location: HistoryLocation): string;
3
3
  export default class BrowserHistory implements History {
4
4
  private __listeners;
5
+ private __pendingDebugContext;
6
+ private __historyIndex;
7
+ private __currentIndex;
5
8
  constructor();
6
9
  get location(): Location;
7
10
  init(): void;
@@ -9,8 +12,10 @@ export default class BrowserHistory implements History {
9
12
  push(location: HistoryLocation): void;
10
13
  unsubscribe(listener: HistoryListener): void;
11
14
  subscribe(listener: HistoryListener): () => void;
12
- private __fire;
13
- private __doFire;
15
+ private __notify;
16
+ private __consumePendingDebugContext;
14
17
  private __subscribe;
18
+ private __handlePopstate;
19
+ private __ensureStateIndex;
15
20
  }
16
21
  //# sourceMappingURL=BrowserHistory.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"BrowserHistory.d.ts","sourceRoot":"","sources":["../../../src/history/BrowserHistory.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,OAAO,EAAE,eAAe,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAQ5E,wBAAgB,UAAU,CAAC,QAAQ,EAAE,eAAe,GAAG,MAAM,CAI5D;AAOD,MAAM,CAAC,OAAO,OAAO,cAAe,YAAW,OAAO;IAEpD,OAAO,CAAC,WAAW,CAAyB;;IAe5C,IAAW,QAAQ,IAAI,QAAQ,CAU9B;IAKM,IAAI,IAAI,IAAI;IAUZ,GAAG,CAAC,QAAQ,EAAE,eAAe,GAAG,IAAI;IAapC,IAAI,CAAC,QAAQ,EAAE,eAAe,GAAG,IAAI;IAYrC,WAAW,CAAC,QAAQ,EAAE,eAAe,GAAG,IAAI;IAY5C,SAAS,CAAC,QAAQ,EAAE,eAAe,GAAG,MAAM,IAAI;IAWvD,OAAO,CAAC,MAAM,CAOZ;IAOF,OAAO,CAAC,QAAQ,CAOd;IAKF,OAAO,CAAC,WAAW;CAGpB"}
1
+ {"version":3,"file":"BrowserHistory.d.ts","sourceRoot":"","sources":["../../../src/history/BrowserHistory.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACV,OAAO,EAEP,eAAe,EACf,eAAe,EAChB,MAAM,YAAY,CAAC;AAQpB,wBAAgB,UAAU,CAAC,QAAQ,EAAE,eAAe,GAAG,MAAM,CAI5D;AAOD,MAAM,CAAC,OAAO,OAAO,cAAe,YAAW,OAAO;IAEpD,OAAO,CAAC,WAAW,CAAyB;IAE5C,OAAO,CAAC,qBAAqB,CAAoC;IAEjE,OAAO,CAAC,cAAc,CAAK;IAE3B,OAAO,CAAC,cAAc,CAAK;;IAgB3B,IAAW,QAAQ,IAAI,QAAQ,CAU9B;IAKM,IAAI,IAAI,IAAI;IAUZ,GAAG,CAAC,QAAQ,EAAE,eAAe,GAAG,IAAI;IA+BpC,IAAI,CAAC,QAAQ,EAAE,eAAe,GAAG,IAAI;IAiCrC,WAAW,CAAC,QAAQ,EAAE,eAAe,GAAG,IAAI;IAY5C,SAAS,CAAC,QAAQ,EAAE,eAAe,GAAG,MAAM,IAAI;IAYvD,OAAO,CAAC,QAAQ,CAQd;IAEF,OAAO,CAAC,4BAA4B;IAYpC,OAAO,CAAC,WAAW;IAInB,OAAO,CAAC,gBAAgB,CA2BtB;IAEF,OAAO,CAAC,kBAAkB;CA2B3B"}
@@ -4,8 +4,12 @@ export function createPath(location) {
4
4
  }
5
5
  export default class BrowserHistory {
6
6
  __listeners = [];
7
+ __pendingDebugContext = null;
8
+ __historyIndex = 0;
9
+ __currentIndex = 0;
7
10
  constructor() {
8
11
  this.init();
12
+ this.__ensureStateIndex();
9
13
  }
10
14
  get location() {
11
15
  return {
@@ -18,14 +22,40 @@ export default class BrowserHistory {
18
22
  this.__subscribe();
19
23
  }
20
24
  set(location) {
21
- const path = createPath(location);
22
- window.history.replaceState({}, '', path);
23
- this.__doFire(true);
25
+ const { debugContext, ...rest } = location;
26
+ const path = createPath(rest);
27
+ if (debugContext !== undefined) {
28
+ this.__pendingDebugContext = {
29
+ ...debugContext,
30
+ historyIndex: this.__currentIndex,
31
+ };
32
+ }
33
+ else {
34
+ this.__pendingDebugContext = { historyIndex: this.__currentIndex };
35
+ }
36
+ window.history.replaceState({
37
+ __plumileRouterIndex: this.__currentIndex,
38
+ }, '', path);
39
+ this.__notify(true);
24
40
  }
25
41
  push(location) {
26
- const path = createPath(location);
27
- window.history.pushState({}, '', path);
28
- this.__fire();
42
+ const { debugContext, ...rest } = location;
43
+ const path = createPath(rest);
44
+ this.__historyIndex += 1;
45
+ this.__currentIndex = this.__historyIndex;
46
+ if (debugContext !== undefined) {
47
+ this.__pendingDebugContext = {
48
+ ...debugContext,
49
+ historyIndex: this.__currentIndex,
50
+ };
51
+ }
52
+ else {
53
+ this.__pendingDebugContext = { historyIndex: this.__currentIndex };
54
+ }
55
+ window.history.pushState({
56
+ __plumileRouterIndex: this.__currentIndex,
57
+ }, '', path);
58
+ this.__notify(false);
29
59
  }
30
60
  unsubscribe(listener) {
31
61
  this.__listeners = this.__listeners.filter((item) => {
@@ -38,24 +68,72 @@ export default class BrowserHistory {
38
68
  this.unsubscribe(listener);
39
69
  };
40
70
  }
41
- __fire = () => {
71
+ __notify = (forceRerender) => {
42
72
  setTimeout(() => {
43
73
  const { location } = this;
74
+ const context = this.__consumePendingDebugContext();
44
75
  this.__listeners.forEach((listener) => {
45
- listener(location, false);
46
- });
47
- }, 0);
48
- };
49
- __doFire = (forceRerender = false) => {
50
- setTimeout(() => {
51
- const { location } = this;
52
- this.__listeners.forEach((listener) => {
53
- listener(location, forceRerender);
76
+ listener(location, forceRerender, context);
54
77
  });
55
78
  }, 0);
56
79
  };
80
+ __consumePendingDebugContext() {
81
+ if (this.__pendingDebugContext == null) {
82
+ return undefined;
83
+ }
84
+ const context = this.__pendingDebugContext;
85
+ this.__pendingDebugContext = null;
86
+ return context;
87
+ }
57
88
  __subscribe() {
58
- window.addEventListener('popstate', this.__fire);
89
+ window.addEventListener('popstate', this.__handlePopstate);
90
+ }
91
+ __handlePopstate = (event) => {
92
+ let origin;
93
+ let historyIndex;
94
+ const state = event.state;
95
+ if (state != null && typeof state.__plumileRouterIndex === 'number') {
96
+ historyIndex = state.__plumileRouterIndex;
97
+ if (historyIndex < this.__currentIndex) {
98
+ origin = 'popstate-back';
99
+ }
100
+ else if (historyIndex > this.__currentIndex) {
101
+ origin = 'popstate-forward';
102
+ }
103
+ else {
104
+ origin = 'popstate-unknown';
105
+ }
106
+ this.__currentIndex = historyIndex;
107
+ if (historyIndex > this.__historyIndex) {
108
+ this.__historyIndex = historyIndex;
109
+ }
110
+ }
111
+ else {
112
+ origin = 'external';
113
+ }
114
+ this.__pendingDebugContext = {
115
+ origin,
116
+ trigger: 'popstate',
117
+ historyIndex,
118
+ };
119
+ this.__notify(false);
120
+ };
121
+ __ensureStateIndex() {
122
+ const state = window.history.state;
123
+ if (state != null && typeof state.__plumileRouterIndex === 'number') {
124
+ this.__currentIndex = state.__plumileRouterIndex;
125
+ this.__historyIndex = state.__plumileRouterIndex;
126
+ return;
127
+ }
128
+ const { pathname, search, hash } = window.location;
129
+ let targetUrl = `${pathname}${search}${hash}`;
130
+ if (typeof window.location.href === 'string' &&
131
+ window.location.href !== '') {
132
+ targetUrl = window.location.href;
133
+ }
134
+ window.history.replaceState({
135
+ __plumileRouterIndex: this.__currentIndex,
136
+ }, '', targetUrl);
59
137
  }
60
138
  }
61
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"BrowserHistory.js","sourceRoot":"","sources":["../../../src/history/BrowserHistory.ts"],"names":[],"mappings":"AAQA,MAAM,UAAU,UAAU,CAAC,QAAyB;IAClD,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,EAAE,EAAE,IAAI,GAAG,EAAE,EAAE,GAAG,QAAQ,CAAC;IAEtD,OAAO,GAAG,QAAQ,GAAG,MAAM,GAAG,IAAI,EAAE,CAAC;AACvC,CAAC;AAOD,MAAM,CAAC,OAAO,OAAO,cAAc;IAEzB,WAAW,GAAsB,EAAE,CAAC;IAK5C;QACE,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAQD,IAAW,QAAQ;QAIjB,OAAO;YAEL,QAAQ,EAAE,MAAM,CAAC,QAAQ,CAAC,QAAQ;YAClC,MAAM,EAAE,MAAM,CAAC,QAAQ,CAAC,MAAM;YAC9B,IAAI,EAAE,MAAM,CAAC,QAAQ,CAAC,IAAI;SACJ,CAAC;IAC3B,CAAC;IAKM,IAAI;QACT,IAAI,CAAC,WAAW,EAAE,CAAC;IACrB,CAAC;IAQM,GAAG,CAAC,QAAyB;QAClC,MAAM,IAAI,GAAG,UAAU,CAAC,QAAQ,CAAC,CAAC;QAElC,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,EAAE,EAAE,EAAE,IAAI,CAAC,CAAC;QAC1C,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;IACtB,CAAC;IAQM,IAAI,CAAC,QAAyB;QACnC,MAAM,IAAI,GAAG,UAAU,CAAC,QAAQ,CAAC,CAAC;QAElC,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,EAAE,EAAE,EAAE,IAAI,CAAC,CAAC;QACvC,IAAI,CAAC,MAAM,EAAE,CAAC;IAChB,CAAC;IAOM,WAAW,CAAC,QAAyB;QAC1C,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE;YAClD,OAAO,IAAI,KAAK,QAAQ,CAAC;QAC3B,CAAC,CAAC,CAAC;IACL,CAAC;IAQM,SAAS,CAAC,QAAyB;QACxC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAChC,OAAO,GAAG,EAAE;YACV,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;QAC7B,CAAC,CAAC;IACJ,CAAC;IAMO,MAAM,GAAG,GAAS,EAAE;QAC1B,UAAU,CAAC,GAAG,EAAE;YACd,MAAM,EAAE,QAAQ,EAAE,GAAG,IAAI,CAAC;YAC1B,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE;gBACpC,QAAQ,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;YAC5B,CAAC,CAAC,CAAC;QACL,CAAC,EAAE,CAAC,CAAC,CAAC;IACR,CAAC,CAAC;IAOM,QAAQ,GAAG,CAAC,aAAa,GAAG,KAAK,EAAE,EAAE;QAC3C,UAAU,CAAC,GAAG,EAAE;YACd,MAAM,EAAE,QAAQ,EAAE,GAAG,IAAI,CAAC;YAC1B,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE;gBACpC,QAAQ,CAAC,QAAQ,EAAE,aAAa,CAAC,CAAC;YACpC,CAAC,CAAC,CAAC;QACL,CAAC,EAAE,CAAC,CAAC,CAAC;IACR,CAAC,CAAC;IAKM,WAAW;QACjB,MAAM,CAAC,gBAAgB,CAAC,UAAU,EAAE,IAAI,CAAC,MAAM,CAAC,CAAC;IACnD,CAAC;CACF","sourcesContent":["import type { History, HistoryListener, HistoryLocation } from './types.js';\n\n/**\n * Creates a complete URL path from a HistoryLocation object.\n *\n * @param location - Location object containing pathname, search, and hash\n * @returns Complete path string combining all components\n */\nexport function createPath(location: HistoryLocation): string {\n  const { pathname, search = '', hash = '' } = location;\n\n  return `${pathname}${search}${hash}`;\n}\n\n/**\n * Browser history implementation using the HTML5 History API.\n * Manages browser navigation state and provides a consistent interface\n * for programmatic navigation and location change notifications.\n */\nexport default class BrowserHistory implements History {\n  /** Array of listeners for location changes */\n  private __listeners: HistoryListener[] = [];\n\n  /**\n   * Creates a new BrowserHistory instance and initializes event listeners.\n   */\n  public constructor() {\n    this.init();\n  }\n\n  /**\n   * Gets the current browser location.\n   *\n   * @returns Current window.location object\n   */\n  // eslint-disable-next-line class-methods-use-this\n  public get location(): Location {\n    // Some test DOM environments (e.g. happy-dom) expose non-enumerable\n    // Location properties so a simple spread returns an empty object.\n    // Return an explicit plain object with the fields we rely on.\n    return {\n      // pathname/search/hash are sufficient for our router logic\n      pathname: window.location.pathname,\n      search: window.location.search,\n      hash: window.location.hash,\n    } as unknown as Location;\n  }\n\n  /**\n   * Initializes the history instance by setting up event listeners.\n   */\n  public init(): void {\n    this.__subscribe();\n  }\n\n  /**\n   * Replaces the current history entry with a new location.\n   * This updates the URL without creating a new history entry.\n   *\n   * @param location - New location to set\n   */\n  public set(location: HistoryLocation): void {\n    const path = createPath(location);\n\n    window.history.replaceState({}, '', path);\n    this.__doFire(true);\n  }\n\n  /**\n   * Navigates to a new location by pushing a new history entry.\n   * This creates a new entry in the browser's history stack.\n   *\n   * @param location - Location to navigate to\n   */\n  public push(location: HistoryLocation): void {\n    const path = createPath(location);\n\n    window.history.pushState({}, '', path);\n    this.__fire();\n  }\n\n  /**\n   * Removes a listener from the history change notifications.\n   *\n   * @param listener - Listener function to remove\n   */\n  public unsubscribe(listener: HistoryListener): void {\n    this.__listeners = this.__listeners.filter((item) => {\n      return item !== listener;\n    });\n  }\n\n  /**\n   * Subscribes to history changes.\n   *\n   * @param listener - Function to call when location changes\n   * @returns Unsubscribe function\n   */\n  public subscribe(listener: HistoryListener): () => void {\n    this.__listeners.push(listener);\n    return () => {\n      this.unsubscribe(listener);\n    };\n  }\n\n  /**\n   * Notifies all listeners of a location change.\n   * Uses setTimeout to ensure the notification happens after the current execution.\n   */\n  private __fire = (): void => {\n    setTimeout(() => {\n      const { location } = this;\n      this.__listeners.forEach((listener) => {\n        listener(location, false);\n      });\n    }, 0);\n  };\n\n  /**\n   * Notifies all listeners of a location change with optional force rerender.\n   *\n   * @param forceRerender - Whether to force a re-render\n   */\n  private __doFire = (forceRerender = false) => {\n    setTimeout(() => {\n      const { location } = this;\n      this.__listeners.forEach((listener) => {\n        listener(location, forceRerender);\n      });\n    }, 0);\n  };\n\n  /**\n   * Sets up the popstate event listener for browser back/forward navigation.\n   */\n  private __subscribe() {\n    window.addEventListener('popstate', this.__fire);\n  }\n}\n"]}
139
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"BrowserHistory.js","sourceRoot":"","sources":["../../../src/history/BrowserHistory.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  History,\n  HistoryDebugContext,\n  HistoryListener,\n  HistoryLocation,\n} from './types.js';\n\n/**\n * Creates a complete URL path from a HistoryLocation object.\n *\n * @param location - Location object containing pathname, search, and hash\n * @returns Complete path string combining all components\n */\nexport function createPath(location: HistoryLocation): string {\n  const { pathname, search = '', hash = '' } = location;\n\n  return `${pathname}${search}${hash}`;\n}\n\n/**\n * Browser history implementation using the HTML5 History API.\n * Manages browser navigation state and provides a consistent interface\n * for programmatic navigation and location change notifications.\n */\nexport default class BrowserHistory implements History {\n  /** Array of listeners for location changes */\n  private __listeners: HistoryListener[] = [];\n  /** Debug metadata captured for the next notification */\n  private __pendingDebugContext: HistoryDebugContext | null = null;\n  /** Monotonic counter attached to pushState/replaceState */\n  private __historyIndex = 0;\n  /** Current index representing the active entry */\n  private __currentIndex = 0;\n\n  /**\n   * Creates a new BrowserHistory instance and initializes event listeners.\n   */\n  public constructor() {\n    this.init();\n    this.__ensureStateIndex();\n  }\n\n  /**\n   * Gets the current browser location.\n   *\n   * @returns Current window.location object\n   */\n  // eslint-disable-next-line class-methods-use-this\n  public get location(): Location {\n    // Some test DOM environments (e.g. happy-dom) expose non-enumerable\n    // Location properties so a simple spread returns an empty object.\n    // Return an explicit plain object with the fields we rely on.\n    return {\n      // pathname/search/hash are sufficient for our router logic\n      pathname: window.location.pathname,\n      search: window.location.search,\n      hash: window.location.hash,\n    } as unknown as Location;\n  }\n\n  /**\n   * Initializes the history instance by setting up event listeners.\n   */\n  public init(): void {\n    this.__subscribe();\n  }\n\n  /**\n   * Replaces the current history entry with a new location.\n   * This updates the URL without creating a new history entry.\n   *\n   * @param location - New location to set\n   */\n  public set(location: HistoryLocation): void {\n    const { debugContext, ...rest } = location as HistoryLocation & {\n      debugContext?: HistoryDebugContext;\n    };\n    const path = createPath(rest);\n\n    if (debugContext !== undefined) {\n      this.__pendingDebugContext = {\n        ...debugContext,\n        historyIndex: this.__currentIndex,\n      };\n    } else {\n      this.__pendingDebugContext = { historyIndex: this.__currentIndex };\n    }\n\n    window.history.replaceState(\n      {\n        __plumileRouterIndex: this.__currentIndex,\n      },\n      '',\n      path,\n    );\n    this.__notify(true);\n  }\n\n  /**\n   * Navigates to a new location by pushing a new history entry.\n   * This creates a new entry in the browser's history stack.\n   *\n   * @param location - Location to navigate to\n   */\n  public push(location: HistoryLocation): void {\n    const { debugContext, ...rest } = location as HistoryLocation & {\n      debugContext?: HistoryDebugContext;\n    };\n    const path = createPath(rest);\n\n    this.__historyIndex += 1;\n    this.__currentIndex = this.__historyIndex;\n\n    if (debugContext !== undefined) {\n      this.__pendingDebugContext = {\n        ...debugContext,\n        historyIndex: this.__currentIndex,\n      };\n    } else {\n      this.__pendingDebugContext = { historyIndex: this.__currentIndex };\n    }\n\n    window.history.pushState(\n      {\n        __plumileRouterIndex: this.__currentIndex,\n      },\n      '',\n      path,\n    );\n    this.__notify(false);\n  }\n\n  /**\n   * Removes a listener from the history change notifications.\n   *\n   * @param listener - Listener function to remove\n   */\n  public unsubscribe(listener: HistoryListener): void {\n    this.__listeners = this.__listeners.filter((item) => {\n      return item !== listener;\n    });\n  }\n\n  /**\n   * Subscribes to history changes.\n   *\n   * @param listener - Function to call when location changes\n   * @returns Unsubscribe function\n   */\n  public subscribe(listener: HistoryListener): () => void {\n    this.__listeners.push(listener);\n    return () => {\n      this.unsubscribe(listener);\n    };\n  }\n\n  /**\n   * Notifies all listeners of a location change.\n   *\n   * Uses setTimeout to ensure the notification happens after the current execution.\n   */\n  private __notify = (forceRerender: boolean): void => {\n    setTimeout(() => {\n      const { location } = this;\n      const context = this.__consumePendingDebugContext();\n      this.__listeners.forEach((listener) => {\n        listener(location, forceRerender, context);\n      });\n    }, 0);\n  };\n\n  private __consumePendingDebugContext(): HistoryDebugContext | undefined {\n    if (this.__pendingDebugContext == null) {\n      return undefined;\n    }\n    const context = this.__pendingDebugContext;\n    this.__pendingDebugContext = null;\n    return context;\n  }\n\n  /**\n   * Sets up the popstate event listener for browser back/forward navigation.\n   */\n  private __subscribe() {\n    window.addEventListener('popstate', this.__handlePopstate);\n  }\n\n  private __handlePopstate = (event: PopStateEvent): void => {\n    let origin: string | undefined;\n    let historyIndex: number | undefined;\n    const state = event.state as { __plumileRouterIndex?: number } | null;\n    if (state != null && typeof state.__plumileRouterIndex === 'number') {\n      historyIndex = state.__plumileRouterIndex;\n      if (historyIndex < this.__currentIndex) {\n        origin = 'popstate-back';\n      } else if (historyIndex > this.__currentIndex) {\n        origin = 'popstate-forward';\n      } else {\n        origin = 'popstate-unknown';\n      }\n      this.__currentIndex = historyIndex;\n      if (historyIndex > this.__historyIndex) {\n        this.__historyIndex = historyIndex;\n      }\n    } else {\n      origin = 'external';\n    }\n\n    this.__pendingDebugContext = {\n      origin,\n      trigger: 'popstate',\n      historyIndex,\n    };\n    this.__notify(false);\n  };\n\n  private __ensureStateIndex(): void {\n    const state = window.history.state as {\n      __plumileRouterIndex?: number;\n    } | null;\n    if (state != null && typeof state.__plumileRouterIndex === 'number') {\n      this.__currentIndex = state.__plumileRouterIndex;\n      this.__historyIndex = state.__plumileRouterIndex;\n      return;\n    }\n\n    const { pathname, search, hash } = window.location;\n    let targetUrl = `${pathname}${search}${hash}`;\n    if (\n      typeof window.location.href === 'string' &&\n      window.location.href !== ''\n    ) {\n      targetUrl = window.location.href;\n    }\n\n    window.history.replaceState(\n      {\n        __plumileRouterIndex: this.__currentIndex,\n      },\n      '',\n      targetUrl,\n    );\n  }\n}\n"]}
@@ -1,12 +1,19 @@
1
- export type HistoryListener = (location: Location, forceRerender: boolean) => void;
1
+ export type HistoryDebugContext = {
2
+ origin?: string;
3
+ trigger?: string;
4
+ historyIndex?: number;
5
+ };
6
+ export type HistoryListener = (location: Location, forceRerender: boolean, debugContext?: HistoryDebugContext) => void;
2
7
  export type HistoryLocation = {
3
8
  hash?: string;
4
9
  pathname: string;
5
10
  search?: string;
11
+ debugContext?: HistoryDebugContext;
6
12
  };
7
13
  export interface History {
8
14
  init: () => void;
9
15
  subscribe: (listener: HistoryListener) => () => void;
10
16
  push: (location: HistoryLocation) => void;
17
+ set: (location: HistoryLocation) => void;
11
18
  }
12
19
  //# sourceMappingURL=types.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../src/history/types.ts"],"names":[],"mappings":"AAOA,MAAM,MAAM,eAAe,GAAG,CAC5B,QAAQ,EAAE,QAAQ,EAClB,aAAa,EAAE,OAAO,KACnB,IAAI,CAAC;AAMV,MAAM,MAAM,eAAe,GAAG;IAE5B,IAAI,CAAC,EAAE,MAAM,CAAC;IAEd,QAAQ,EAAE,MAAM,CAAC;IAEjB,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB,CAAC;AAMF,MAAM,WAAW,OAAO;IAEtB,IAAI,EAAE,MAAM,IAAI,CAAC;IAEjB,SAAS,EAAE,CAAC,QAAQ,EAAE,eAAe,KAAK,MAAM,IAAI,CAAC;IAErD,IAAI,EAAE,CAAC,QAAQ,EAAE,eAAe,KAAK,IAAI,CAAC;CAC3C"}
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../../src/history/types.ts"],"names":[],"mappings":"AAOA,MAAM,MAAM,mBAAmB,GAAG;IAChC,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,YAAY,CAAC,EAAE,MAAM,CAAC;CACvB,CAAC;AAEF,MAAM,MAAM,eAAe,GAAG,CAC5B,QAAQ,EAAE,QAAQ,EAClB,aAAa,EAAE,OAAO,EACtB,YAAY,CAAC,EAAE,mBAAmB,KAC/B,IAAI,CAAC;AAMV,MAAM,MAAM,eAAe,GAAG;IAE5B,IAAI,CAAC,EAAE,MAAM,CAAC;IAEd,QAAQ,EAAE,MAAM,CAAC;IAEjB,MAAM,CAAC,EAAE,MAAM,CAAC;IAEhB,YAAY,CAAC,EAAE,mBAAmB,CAAC;CACpC,CAAC;AAMF,MAAM,WAAW,OAAO;IAEtB,IAAI,EAAE,MAAM,IAAI,CAAC;IAEjB,SAAS,EAAE,CAAC,QAAQ,EAAE,eAAe,KAAK,MAAM,IAAI,CAAC;IAErD,IAAI,EAAE,CAAC,QAAQ,EAAE,eAAe,KAAK,IAAI,CAAC;IAE1C,GAAG,EAAE,CAAC,QAAQ,EAAE,eAAe,KAAK,IAAI,CAAC;CAC1C"}
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvaGlzdG9yeS90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDYWxsYmFjayBmdW5jdGlvbiBmb3IgaGlzdG9yeSBjaGFuZ2VzLlxuICogQ2FsbGVkIHdoZW5ldmVyIHRoZSBicm93c2VyIGxvY2F0aW9uIGNoYW5nZXMuXG4gKlxuICogQHBhcmFtIGxvY2F0aW9uIC0gVGhlIG5ldyBicm93c2VyIGxvY2F0aW9uXG4gKiBAcGFyYW0gZm9yY2VSZXJlbmRlciAtIFdoZXRoZXIgdG8gZm9yY2UgYSByZS1yZW5kZXIgZXZlbiBpZiB0aGUgcGF0aCBpcyB0aGUgc2FtZVxuICovXG5leHBvcnQgdHlwZSBIaXN0b3J5TGlzdGVuZXIgPSAoXG4gIGxvY2F0aW9uOiBMb2NhdGlvbixcbiAgZm9yY2VSZXJlbmRlcjogYm9vbGVhbixcbikgPT4gdm9pZDtcblxuLyoqXG4gKiBSZXByZXNlbnRzIGEgbG9jYXRpb24gZm9yIGhpc3RvcnkgbmF2aWdhdGlvbi5cbiAqIFRoaXMgaXMgYSBzdWJzZXQgb2YgdGhlIGJyb3dzZXIncyBMb2NhdGlvbiBpbnRlcmZhY2UuXG4gKi9cbmV4cG9ydCB0eXBlIEhpc3RvcnlMb2NhdGlvbiA9IHtcbiAgLyoqIFVSTCBoYXNoIChmcmFnbWVudCBpZGVudGlmaWVyKSAqL1xuICBoYXNoPzogc3RyaW5nO1xuICAvKiogVVJMIHBhdGhuYW1lICovXG4gIHBhdGhuYW1lOiBzdHJpbmc7XG4gIC8qKiBVUkwgc2VhcmNoIHBhcmFtZXRlcnMgKi9cbiAgc2VhcmNoPzogc3RyaW5nO1xufTtcblxuLyoqXG4gKiBJbnRlcmZhY2UgZm9yIGhpc3RvcnkgbWFuYWdlbWVudCBpbXBsZW1lbnRhdGlvbnMuXG4gKiBQcm92aWRlcyBtZXRob2RzIGZvciBuYXZpZ2F0aW9uIGFuZCBsaXN0ZW5pbmcgdG8gbG9jYXRpb24gY2hhbmdlcy5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBIaXN0b3J5IHtcbiAgLyoqIEluaXRpYWxpemUgdGhlIGhpc3RvcnkgaW5zdGFuY2UgKi9cbiAgaW5pdDogKCkgPT4gdm9pZDtcbiAgLyoqIFN1YnNjcmliZSB0byBsb2NhdGlvbiBjaGFuZ2VzICovXG4gIHN1YnNjcmliZTogKGxpc3RlbmVyOiBIaXN0b3J5TGlzdGVuZXIpID0+ICgpID0+IHZvaWQ7XG4gIC8qKiBOYXZpZ2F0ZSB0byBhIG5ldyBsb2NhdGlvbiAqL1xuICBwdXNoOiAobG9jYXRpb246IEhpc3RvcnlMb2NhdGlvbikgPT4gdm9pZDtcbn1cbiJdfQ==
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvaGlzdG9yeS90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDYWxsYmFjayBmdW5jdGlvbiBmb3IgaGlzdG9yeSBjaGFuZ2VzLlxuICogQ2FsbGVkIHdoZW5ldmVyIHRoZSBicm93c2VyIGxvY2F0aW9uIGNoYW5nZXMuXG4gKlxuICogQHBhcmFtIGxvY2F0aW9uIC0gVGhlIG5ldyBicm93c2VyIGxvY2F0aW9uXG4gKiBAcGFyYW0gZm9yY2VSZXJlbmRlciAtIFdoZXRoZXIgdG8gZm9yY2UgYSByZS1yZW5kZXIgZXZlbiBpZiB0aGUgcGF0aCBpcyB0aGUgc2FtZVxuICovXG5leHBvcnQgdHlwZSBIaXN0b3J5RGVidWdDb250ZXh0ID0ge1xuICBvcmlnaW4/OiBzdHJpbmc7XG4gIHRyaWdnZXI/OiBzdHJpbmc7XG4gIGhpc3RvcnlJbmRleD86IG51bWJlcjtcbn07XG5cbmV4cG9ydCB0eXBlIEhpc3RvcnlMaXN0ZW5lciA9IChcbiAgbG9jYXRpb246IExvY2F0aW9uLFxuICBmb3JjZVJlcmVuZGVyOiBib29sZWFuLFxuICBkZWJ1Z0NvbnRleHQ/OiBIaXN0b3J5RGVidWdDb250ZXh0LFxuKSA9PiB2b2lkO1xuXG4vKipcbiAqIFJlcHJlc2VudHMgYSBsb2NhdGlvbiBmb3IgaGlzdG9yeSBuYXZpZ2F0aW9uLlxuICogVGhpcyBpcyBhIHN1YnNldCBvZiB0aGUgYnJvd3NlcidzIExvY2F0aW9uIGludGVyZmFjZS5cbiAqL1xuZXhwb3J0IHR5cGUgSGlzdG9yeUxvY2F0aW9uID0ge1xuICAvKiogVVJMIGhhc2ggKGZyYWdtZW50IGlkZW50aWZpZXIpICovXG4gIGhhc2g/OiBzdHJpbmc7XG4gIC8qKiBVUkwgcGF0aG5hbWUgKi9cbiAgcGF0aG5hbWU6IHN0cmluZztcbiAgLyoqIFVSTCBzZWFyY2ggcGFyYW1ldGVycyAqL1xuICBzZWFyY2g/OiBzdHJpbmc7XG4gIC8qKiBPcHRpb25hbCBkZWJ1ZyBtZXRhZGF0YSBwcm9wYWdhdGVkIHRocm91Z2ggbmF2aWdhdGlvbiBldmVudHMgKi9cbiAgZGVidWdDb250ZXh0PzogSGlzdG9yeURlYnVnQ29udGV4dDtcbn07XG5cbi8qKlxuICogSW50ZXJmYWNlIGZvciBoaXN0b3J5IG1hbmFnZW1lbnQgaW1wbGVtZW50YXRpb25zLlxuICogUHJvdmlkZXMgbWV0aG9kcyBmb3IgbmF2aWdhdGlvbiBhbmQgbGlzdGVuaW5nIHRvIGxvY2F0aW9uIGNoYW5nZXMuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgSGlzdG9yeSB7XG4gIC8qKiBJbml0aWFsaXplIHRoZSBoaXN0b3J5IGluc3RhbmNlICovXG4gIGluaXQ6ICgpID0+IHZvaWQ7XG4gIC8qKiBTdWJzY3JpYmUgdG8gbG9jYXRpb24gY2hhbmdlcyAqL1xuICBzdWJzY3JpYmU6IChsaXN0ZW5lcjogSGlzdG9yeUxpc3RlbmVyKSA9PiAoKSA9PiB2b2lkO1xuICAvKiogTmF2aWdhdGUgdG8gYSBuZXcgbG9jYXRpb24gKi9cbiAgcHVzaDogKGxvY2F0aW9uOiBIaXN0b3J5TG9jYXRpb24pID0+IHZvaWQ7XG4gIC8qKiBSZXBsYWNlIHRoZSBjdXJyZW50IGhpc3RvcnkgZW50cnkgKi9cbiAgc2V0OiAobG9jYXRpb246IEhpc3RvcnlMb2NhdGlvbikgPT4gdm9pZDtcbn1cbiJdfQ==
@@ -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,IAAI;IAE7E,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,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB,CAAC;AAyBF,QAAA,MAAM,IAAI,0GAuJR,CAAC;AAEH,eAAe,IAAI,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;AAc3E,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,IAAI;IAE7E,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,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB,CAAC;AAyBF,QAAA,MAAM,IAAI,0GAsLR,CAAC;AAEH,eAAe,IAAI,CAAC"}
@@ -1,5 +1,6 @@
1
1
  import { jsx as _jsx } from "react/jsx-runtime";
2
2
  import React, { forwardRef, useEffect, useMemo, } from 'react';
3
+ import { NavigationOrigin, RouterDebugEventKind } from '../types.js';
3
4
  import { cx } from './../tools/index.js';
4
5
  import buildCombinedSearch from '../tools/buildCombinedSearch.js';
5
6
  import RoutingContext from './RoutingContext.js';
@@ -82,9 +83,20 @@ const Link = forwardRef((props, ref) => {
82
83
  if (router == null || isDisabled) {
83
84
  return;
84
85
  }
86
+ router.__debug?.recordNavigationIntent(NavigationOrigin.LinkClick);
87
+ router.__debug?.recordHistoryAction(RouterDebugEventKind.HistoryPush, NavigationOrigin.LinkClick, {
88
+ pathname,
89
+ search,
90
+ hash: '',
91
+ }, { trigger: 'link' });
85
92
  router.history.push({
86
93
  pathname,
87
94
  search,
95
+ hash: '',
96
+ debugContext: {
97
+ origin: NavigationOrigin.LinkClick,
98
+ trigger: 'link',
99
+ },
88
100
  });
89
101
  }, [isDisabled, onClick, pathname, router, target, search]);
90
102
  const handleMouseEnter = useCallback(() => {
@@ -92,9 +104,11 @@ const Link = forwardRef((props, ref) => {
92
104
  return;
93
105
  }
94
106
  if (preloadOnMouseEnter) {
107
+ router.__debug?.recordPreload(RouterDebugEventKind.Preload, NavigationOrigin.PreloadHover, pathname);
95
108
  router.preload(pathname);
96
109
  }
97
110
  else {
111
+ router.__debug?.recordPreload(RouterDebugEventKind.PreloadCode, NavigationOrigin.PreloadHover, pathname);
98
112
  router.preloadCode(pathname);
99
113
  }
100
114
  }, [router, isDisabled, preloadOnMouseEnter, pathname]);
@@ -102,6 +116,7 @@ const Link = forwardRef((props, ref) => {
102
116
  if (router == null || isDisabled || !preloadOnMouseDown) {
103
117
  return;
104
118
  }
119
+ router.__debug?.recordPreload(RouterDebugEventKind.Preload, NavigationOrigin.LinkClick, pathname);
105
120
  router.preload(pathname);
106
121
  }, [router, isDisabled, preloadOnMouseDown, pathname]);
107
122
  let hrefValue = pathname;
@@ -111,4 +126,4 @@ const Link = forwardRef((props, ref) => {
111
126
  return (_jsx("a", { className: cx(classNames), href: hrefValue, onClick: handleClick, onFocus: onFocus, onMouseDown: handleMouseDown, onMouseEnter: handleMouseEnter, onMouseOver: onMouseOver, ref: ref, target: target, children: children }));
112
127
  });
113
128
  export default Link;
114
- //# 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 */\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 provided query treated as filters w/ active schema.\n  const search = useMemo(() => {\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  }, [query, router]);\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      });\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    if (preloadOnMouseEnter) {\n      router.preload(pathname);\n    } else {\n      router.preloadCode(pathname);\n    }\n  }, [router, isDisabled, preloadOnMouseEnter, pathname]);\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);\n  }, [router, isDisabled, preloadOnMouseDown, pathname]);\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"]}
129
+ //# 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';\nimport { NavigationOrigin, RouterDebugEventKind } from '../types.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 */\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 provided query treated as filters w/ active schema.\n  const search = useMemo(() => {\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  }, [query, router]);\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.__debug?.recordNavigationIntent(NavigationOrigin.LinkClick);\n      router.__debug?.recordHistoryAction(\n        RouterDebugEventKind.HistoryPush,\n        NavigationOrigin.LinkClick,\n        {\n          pathname,\n          search,\n          hash: '',\n        },\n        { trigger: 'link' },\n      );\n      router.history.push({\n        pathname,\n        search,\n        hash: '',\n        debugContext: {\n          origin: NavigationOrigin.LinkClick,\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    if (preloadOnMouseEnter) {\n      router.__debug?.recordPreload(\n        RouterDebugEventKind.Preload,\n        NavigationOrigin.PreloadHover,\n        pathname,\n      );\n      router.preload(pathname);\n    } else {\n      router.__debug?.recordPreload(\n        RouterDebugEventKind.PreloadCode,\n        NavigationOrigin.PreloadHover,\n        pathname,\n      );\n      router.preloadCode(pathname);\n    }\n  }, [router, isDisabled, preloadOnMouseEnter, pathname]);\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.__debug?.recordPreload(\n      RouterDebugEventKind.Preload,\n      NavigationOrigin.LinkClick,\n      pathname,\n    );\n    router.preload(pathname);\n  }, [router, isDisabled, preloadOnMouseDown, pathname]);\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"]}
@@ -1,4 +1,4 @@
1
- import type { RoutingContextType, AnyRoute, PreparedAccess } from '../types.js';
1
+ import { type RoutingContextType, type AnyRoute, type PreparedAccess } from '../types.js';
2
2
  export type CreateRouterReturn<R extends AnyRoute[]> = {
3
3
  cleanup: () => void;
4
4
  context: RoutingContextType<any> & PreparedAccess<R>;
@@ -1 +1 @@
1
- {"version":3,"file":"createRouter.d.ts","sourceRoot":"","sources":["../../../src/routing/createRouter.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAEV,kBAAkB,EAElB,QAAQ,EACR,cAAc,EAEf,MAAM,aAAa,CAAC;AAKrB,MAAM,MAAM,kBAAkB,CAAC,CAAC,SAAS,QAAQ,EAAE,IAAI;IAErD,OAAO,EAAE,MAAM,IAAI,CAAC;IAEpB,OAAO,EAAE,kBAAkB,CAAC,GAAG,CAAC,GAAG,cAAc,CAAC,CAAC,CAAC,CAAC;CACtD,CAAC;AAwCF,MAAM,MAAM,mBAAmB,GAAG;IAKhC,KAAK,CAAC,EAAE,OAAO,CAAC;CACjB,CAAC;AASF,MAAM,CAAC,OAAO,UAAU,YAAY,CAAC,CAAC,SAAS,QAAQ,EAAE,EACvD,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,GAAG,QAAQ,EAAE,EAC3B,OAAO,GAAE,mBAAwB,GAChC,kBAAkB,CAAC,CAAC,SAAS,QAAQ,EAAE,GAAG,CAAC,GAAG,QAAQ,EAAE,CAAC,CA8T3D"}
1
+ {"version":3,"file":"createRouter.d.ts","sourceRoot":"","sources":["../../../src/routing/createRouter.ts"],"names":[],"mappings":"AAMA,OAAO,EAML,KAAK,kBAAkB,EAEvB,KAAK,QAAQ,EACb,KAAK,cAAc,EAEpB,MAAM,aAAa,CAAC;AAMrB,MAAM,MAAM,kBAAkB,CAAC,CAAC,SAAS,QAAQ,EAAE,IAAI;IAErD,OAAO,EAAE,MAAM,IAAI,CAAC;IAEpB,OAAO,EAAE,kBAAkB,CAAC,GAAG,CAAC,GAAG,cAAc,CAAC,CAAC,CAAC,CAAC;CACtD,CAAC;AAwCF,MAAM,MAAM,mBAAmB,GAAG;IAKhC,KAAK,CAAC,EAAE,OAAO,CAAC;CACjB,CAAC;AASF,MAAM,CAAC,OAAO,UAAU,YAAY,CAAC,CAAC,SAAS,QAAQ,EAAE,EACvD,MAAM,EAAE,CAAC,GAAG,CAAC,CAAC,GAAG,QAAQ,EAAE,EAC3B,OAAO,GAAE,mBAAwB,GAChC,kBAAkB,CAAC,CAAC,SAAS,QAAQ,EAAE,GAAG,CAAC,GAAG,QAAQ,EAAE,CAAC,CAsyB3D"}