@plumile/router 0.1.48 → 0.1.50
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 +78 -2
- package/lib/esm/ResourcePage.d.ts +1 -5
- package/lib/esm/ResourcePage.d.ts.map +1 -1
- package/lib/esm/ResourcePage.js +13 -42
- package/lib/esm/asyncResource.d.ts +8 -0
- package/lib/esm/asyncResource.d.ts.map +1 -0
- package/lib/esm/asyncResource.js +101 -0
- package/lib/esm/builder.d.ts +7 -7
- package/lib/esm/builder.d.ts.map +1 -1
- package/lib/esm/builder.js +5 -4
- package/lib/esm/index.d.ts +1 -0
- package/lib/esm/index.d.ts.map +1 -1
- package/lib/esm/index.js +2 -1
- package/lib/esm/instrumentation/Instrumentation.d.ts +37 -13
- package/lib/esm/instrumentation/Instrumentation.d.ts.map +1 -1
- package/lib/esm/instrumentation/Instrumentation.js +1 -1
- package/lib/esm/prepareResource.d.ts +4 -0
- package/lib/esm/prepareResource.d.ts.map +1 -0
- package/lib/esm/prepareResource.js +11 -0
- package/lib/esm/routing/RouteComponentWrapper.d.ts.map +1 -1
- package/lib/esm/routing/RouteComponentWrapper.js +14 -3
- package/lib/esm/routing/createRouter.d.ts +9 -3
- package/lib/esm/routing/createRouter.d.ts.map +1 -1
- package/lib/esm/routing/createRouter.js +155 -145
- package/lib/esm/tools.d.ts +7 -4
- package/lib/esm/tools.d.ts.map +1 -1
- package/lib/esm/tools.js +93 -21
- package/lib/esm/types.d.ts +131 -54
- package/lib/esm/types.d.ts.map +1 -1
- package/lib/esm/types.js +1 -1
- package/lib/esm/values.d.ts +19 -0
- package/lib/esm/values.d.ts.map +1 -0
- package/lib/esm/values.js +53 -0
- package/lib/tsconfig.esm.tsbuildinfo +1 -1
- package/lib/types/ResourcePage.d.ts +1 -5
- package/lib/types/ResourcePage.d.ts.map +1 -1
- package/lib/types/asyncResource.d.ts +8 -0
- package/lib/types/asyncResource.d.ts.map +1 -0
- package/lib/types/builder.d.ts +7 -7
- package/lib/types/builder.d.ts.map +1 -1
- package/lib/types/index.d.ts +1 -0
- package/lib/types/index.d.ts.map +1 -1
- package/lib/types/instrumentation/Instrumentation.d.ts +37 -13
- package/lib/types/instrumentation/Instrumentation.d.ts.map +1 -1
- package/lib/types/prepareResource.d.ts +4 -0
- package/lib/types/prepareResource.d.ts.map +1 -0
- package/lib/types/routing/RouteComponentWrapper.d.ts.map +1 -1
- package/lib/types/routing/createRouter.d.ts +9 -3
- package/lib/types/routing/createRouter.d.ts.map +1 -1
- package/lib/types/tools.d.ts +7 -4
- package/lib/types/tools.d.ts.map +1 -1
- package/lib/types/types.d.ts +131 -54
- package/lib/types/types.d.ts.map +1 -1
- package/lib/types/values.d.ts +19 -0
- package/lib/types/values.d.ts.map +1 -0
- package/package.json +2 -2
|
@@ -56,4 +56,4 @@ export function createInstrumentationRegistry(initial = []) {
|
|
|
56
56
|
},
|
|
57
57
|
};
|
|
58
58
|
}
|
|
59
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Instrumentation.js","sourceRoot":"","sources":["../../../src/instrumentation/Instrumentation.ts"],"names":[],"mappings":"AAoHA,MAAM,UAAU,6BAA6B,CAC3C,UAAgC,EAAE;IAElC,MAAM,SAAS,GAAG,IAAI,GAAG,EAAsB,CAAC;IAChD,OAAO,CAAC,OAAO,CAAC,CAAC,eAAe,EAAE,EAAE;QAClC,SAAS,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;IACjC,CAAC,CAAC,CAAC;IAEH,IAAI,QAAQ,GAAG,KAAK,CAAC;IAKrB,SAAS,KAAK;QACZ,IAAI,QAAQ,EAAE,CAAC;YACb,MAAM,IAAI,KAAK,CAAC,uCAAuC,CAAC,CAAC;QAC3D,CAAC;IACH,CAAC;IAMD,SAAS,UAAU,CACjB,eAAmC,EACnC,MAAgC,EAChC,OAAkB;QAElB,MAAM,OAAO,GAAG,eAAe,CAAC,MAAM,CAAC,CAAC;QACxC,IAAI,OAAO,OAAO,KAAK,UAAU,EAAE,CAAC;YAClC,OAAO;QACT,CAAC;QACD,IAAI,CAAC;YACF,OAAwC,CAAC,GAAG,OAAO,CAAC,CAAC;QACxD,CAAC;QAAC,MAAM,CAAC;QAET,CAAC;IACH,CAAC;IAED,OAAO;QACL,QAAQ,CAAC,eAAe;YACtB,KAAK,EAAE,CAAC;YACR,SAAS,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;QACjC,CAAC;QACD,UAAU,CAAC,eAAe;YACxB,KAAK,EAAE,CAAC;YACR,SAAS,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;QACpC,CAAC;QACD,SAAS,CAAC,KAAK;YACb,KAAK,EAAE,CAAC;YACR,SAAS,CAAC,OAAO,CAAC,CAAC,eAAe,EAAE,EAAE;gBACpC,UAAU,CAAC,eAAe,EAAE,SAAS,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;YAClD,CAAC,CAAC,CAAC;QACL,CAAC;QACD,iBAAiB,CAAC,KAAK;YACrB,KAAK,EAAE,CAAC;YACR,SAAS,CAAC,OAAO,CAAC,CAAC,eAAe,EAAE,EAAE;gBACpC,UAAU,CAAC,eAAe,EAAE,eAAe,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;YACxD,CAAC,CAAC,CAAC;QACL,CAAC;QACD,OAAO;YACL,IAAI,QAAQ,EAAE,CAAC;gBACb,OAAO;YACT,CAAC;YACD,SAAS,CAAC,OAAO,CAAC,CAAC,eAAe,EAAE,EAAE;gBACpC,UAAU,CAAC,eAAe,EAAE,SAAS,EAAE,EAAE,CAAC,CAAC;YAC7C,CAAC,CAAC,CAAC;YACH,SAAS,CAAC,KAAK,EAAE,CAAC;YAClB,QAAQ,GAAG,IAAI,CAAC;QAClB,CAAC;QACD,IAAI,IAAI;YACN,OAAO,SAAS,CAAC,IAAI,CAAC;QACxB,CAAC;KACF,CAAC;AACJ,CAAC","sourcesContent":["import type { RouteEntry } from '../types.js';\n\n/**\n * Enumerates the origin of navigation-related events exposed to instrumentations.\n *\n * Kept as string literals to avoid leaking debug enums into the production bundle.\n */\nexport type RouterNavigationSource =\n  | 'link-click'\n  | 'link-hover'\n  | 'preload-hover'\n  | 'programmatic'\n  | 'popstate-back'\n  | 'popstate-forward'\n  | 'popstate-unknown'\n  | 'external'\n  | 'normalize';\n\n/**\n * Compact location snapshot shared by all instrumentation events.\n */\nexport type RouterLocationSnapshot = {\n  pathname: string;\n  search: string;\n  hash: string;\n};\n\n/**\n * Summary of the current router entry that can be broadcast to instrumentations.\n * The shape intentionally mirrors public data already available to applications.\n */\nexport type RouterEntrySnapshot = {\n  location: RouterLocationSnapshot;\n  routePath?: string;\n  preparedMatch?: RouteEntry<any>['preparedMatch'];\n  filters?: RouteEntry<any>['filters'];\n  filterDiagnostics?: RouteEntry<any>['filterDiagnostics'];\n  activeQuerySchema?: RouteEntry<any>['activeQuerySchema'];\n};\n\n/**\n * Base payload shared by router events. Each event extends the base with\n * specific details relevant to the instrumentation.\n */\nexport type RouterEventBase = {\n  timestamp: number;\n  location: RouterLocationSnapshot;\n};\n\nexport type RouterSnapshotEvent = RouterEventBase & {\n  kind: 'snapshot';\n  source: RouterNavigationSource;\n  routePath?: string;\n};\n\nexport type RouterPreloadEvent = RouterEventBase & {\n  kind: 'preload';\n  source: RouterNavigationSource;\n  targetPathname: string;\n  mode: 'code' | 'full';\n};\n\nexport type RouterHistoryEvent = RouterEventBase & {\n  kind: 'history';\n  source: RouterNavigationSource;\n  action: 'push' | 'replace' | 'normalize';\n  details?: Record<string, unknown>;\n};\n\nexport type RouterPopstateEvent = RouterEventBase & {\n  kind: 'popstate';\n  source: RouterNavigationSource;\n  direction: 'back' | 'forward' | 'unknown';\n  details?: Record<string, unknown>;\n};\n\nexport type RouterPrepareEvent = RouterEventBase & {\n  kind: 'prepare-start' | 'prepare-end';\n  source: RouterNavigationSource;\n  routePath?: string;\n  moduleId?: string;\n  durationMs?: number;\n  hasPrepared?: boolean;\n};\n\nexport type RouterEvent =\n  | RouterSnapshotEvent\n  | RouterPreloadEvent\n  | RouterHistoryEvent\n  | RouterPopstateEvent\n  | RouterPrepareEvent;\n\n/**\n * Instrumentations can observe router events, entry changes, and optionally\n * dispose resources when the registry is torn down.\n */\nexport type InstrumentationAPI = {\n  onEvent?: (event: RouterEvent) => void;\n  onEntryChange?: (entry: RouterEntrySnapshot) => void;\n  dispose?: () => void;\n};\n\nexport type InstrumentationRegistry = {\n  register: (instrumentation: InstrumentationAPI) => void;\n  unregister: (instrumentation: InstrumentationAPI) => void;\n  emitEvent: (event: RouterEvent) => void;\n  notifyEntryChange: (entry: RouterEntrySnapshot) => void;\n  dispose: () => void;\n  readonly size: number;\n};\n\n/**\n * Creates a registry coordinating a set of instrumentations. The registry keeps\n * consumer code simple: router internals broadcast events without caring about\n * the number or nature of attached instrumentations.\n */\nexport function createInstrumentationRegistry(\n  initial: InstrumentationAPI[] = [],\n): InstrumentationRegistry {\n  const listeners = new Set<InstrumentationAPI>();\n  initial.forEach((instrumentation) => {\n    listeners.add(instrumentation);\n  });\n\n  let disposed = false;\n\n  /**\n   * Throws when the registry has already been disposed. Guards event emission.\n   */\n  function guard() {\n    if (disposed) {\n      throw new Error('Instrumentation registry was disposed');\n    }\n  }\n\n  /**\n   * Safely invokes an optional handler provided by an instrumentation.\n   * Failures are swallowed so router execution continues uninterrupted.\n   */\n  function safeInvoke(\n    instrumentation: InstrumentationAPI,\n    method: keyof InstrumentationAPI,\n    payload: unknown[],\n  ): void {\n    const handler = instrumentation[method];\n    if (typeof handler !== 'function') {\n      return;\n    }\n    try {\n      (handler as (...args: unknown[]) => void)(...payload);\n    } catch {\n      // Instrumentations should never break router execution. Errors are swallowed.\n    }\n  }\n\n  return {\n    register(instrumentation) {\n      guard();\n      listeners.add(instrumentation);\n    },\n    unregister(instrumentation) {\n      guard();\n      listeners.delete(instrumentation);\n    },\n    emitEvent(event) {\n      guard();\n      listeners.forEach((instrumentation) => {\n        safeInvoke(instrumentation, 'onEvent', [event]);\n      });\n    },\n    notifyEntryChange(entry) {\n      guard();\n      listeners.forEach((instrumentation) => {\n        safeInvoke(instrumentation, 'onEntryChange', [entry]);\n      });\n    },\n    dispose() {\n      if (disposed) {\n        return;\n      }\n      listeners.forEach((instrumentation) => {\n        safeInvoke(instrumentation, 'dispose', []);\n      });\n      listeners.clear();\n      disposed = true;\n    },\n    get size() {\n      return listeners.size;\n    },\n  };\n}\n"]}
|
|
59
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Instrumentation.js","sourceRoot":"","sources":["../../../src/instrumentation/Instrumentation.ts"],"names":[],"mappings":"AAmJA,MAAM,UAAU,6BAA6B,CAC3C,UAAgC,EAAE;IAElC,MAAM,SAAS,GAAG,IAAI,GAAG,EAAsB,CAAC;IAChD,OAAO,CAAC,OAAO,CAAC,CAAC,eAAe,EAAE,EAAE;QAClC,SAAS,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;IACjC,CAAC,CAAC,CAAC;IAEH,IAAI,QAAQ,GAAG,KAAK,CAAC;IAKrB,SAAS,KAAK;QACZ,IAAI,QAAQ,EAAE,CAAC;YACb,MAAM,IAAI,KAAK,CAAC,uCAAuC,CAAC,CAAC;QAC3D,CAAC;IACH,CAAC;IAMD,SAAS,UAAU,CACjB,eAAmC,EACnC,MAAgC,EAChC,OAAkB;QAElB,MAAM,OAAO,GAAG,eAAe,CAAC,MAAM,CAAC,CAAC;QACxC,IAAI,OAAO,OAAO,KAAK,UAAU,EAAE,CAAC;YAClC,OAAO;QACT,CAAC;QACD,IAAI,CAAC;YACF,OAAwC,CAAC,GAAG,OAAO,CAAC,CAAC;QACxD,CAAC;QAAC,MAAM,CAAC;QAET,CAAC;IACH,CAAC;IAED,OAAO;QACL,QAAQ,CAAC,eAAe;YACtB,KAAK,EAAE,CAAC;YACR,SAAS,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;QACjC,CAAC;QACD,UAAU,CAAC,eAAe;YACxB,KAAK,EAAE,CAAC;YACR,SAAS,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;QACpC,CAAC;QACD,SAAS,CAAC,KAAK;YACb,KAAK,EAAE,CAAC;YACR,SAAS,CAAC,OAAO,CAAC,CAAC,eAAe,EAAE,EAAE;gBACpC,UAAU,CAAC,eAAe,EAAE,SAAS,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;YAClD,CAAC,CAAC,CAAC;QACL,CAAC;QACD,iBAAiB,CAAC,KAAK;YACrB,KAAK,EAAE,CAAC;YACR,SAAS,CAAC,OAAO,CAAC,CAAC,eAAe,EAAE,EAAE;gBACpC,UAAU,CAAC,eAAe,EAAE,eAAe,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;YACxD,CAAC,CAAC,CAAC;QACL,CAAC;QACD,OAAO;YACL,IAAI,QAAQ,EAAE,CAAC;gBACb,OAAO;YACT,CAAC;YACD,SAAS,CAAC,OAAO,CAAC,CAAC,eAAe,EAAE,EAAE;gBACpC,UAAU,CAAC,eAAe,EAAE,SAAS,EAAE,EAAE,CAAC,CAAC;YAC7C,CAAC,CAAC,CAAC;YACH,SAAS,CAAC,KAAK,EAAE,CAAC;YAClB,QAAQ,GAAG,IAAI,CAAC;QAClB,CAAC;QACD,IAAI,IAAI;YACN,OAAO,SAAS,CAAC,IAAI,CAAC;QACxB,CAAC;KACF,CAAC;AACJ,CAAC","sourcesContent":["import type {\n  AsyncResourceStatus,\n  PreparedValueResource,\n  UnknownFilters,\n} from '../types.js';\nimport type { Schema as FilterSchema } from '@plumile/filter-query';\nimport type {\n  RouterHistoryActionValue,\n  RouterNavigationSourceValue,\n  RouterPopstateDirectionValue,\n  RouterPreloadModeValue,\n  RouterPrepareStatusValue,\n} from '../values.js';\n\n/**\n * Enumerates the origin of navigation-related events exposed to instrumentations.\n *\n * Kept as string literals to avoid leaking debug enums into the production bundle.\n */\nexport type RouterNavigationSource = RouterNavigationSourceValue;\n\n/**\n * Compact location snapshot shared by all instrumentation events.\n */\nexport type RouterLocationSnapshot = {\n  pathname: string;\n  search: string;\n  hash: string;\n};\n\nexport type RouterPreparedSegmentSnapshot = {\n  path?: string;\n  fullPath?: string;\n  segmentIndex: number;\n  preparedResource?: PreparedValueResource;\n  redirectTo: string | null;\n  highlightId?: string;\n  querySchema?: FilterSchema;\n  preparedStatus?: AsyncResourceStatus;\n};\n\nexport type RouterPreparedMatchSnapshot = {\n  segments: RouterPreparedSegmentSnapshot[];\n  routes: RouterPreparedSegmentSnapshot[];\n};\n\n/**\n * Summary of the current router entry that can be broadcast to instrumentations.\n * The shape intentionally mirrors public data already available to applications.\n */\nexport type RouterEntrySnapshot = {\n  location: RouterLocationSnapshot;\n  routePath?: string;\n  preparedMatch?: RouterPreparedMatchSnapshot;\n  filters?: UnknownFilters;\n  filterDiagnostics?: unknown[];\n  activeQuerySchema?: FilterSchema;\n};\n\n/**\n * Base payload shared by router events. Each event extends the base with\n * specific details relevant to the instrumentation.\n */\nexport type RouterEventBase = {\n  timestamp: number;\n  location: RouterLocationSnapshot;\n};\n\nexport type RouterSnapshotEvent = RouterEventBase & {\n  kind: 'snapshot';\n  source: RouterNavigationSource;\n  routePath?: string;\n};\n\nexport type RouterPreloadEvent = RouterEventBase & {\n  kind: 'preload';\n  source: RouterNavigationSource;\n  targetPathname: string;\n  mode: RouterPreloadModeValue;\n};\n\nexport type RouterHistoryEvent = RouterEventBase & {\n  kind: 'history';\n  source: RouterNavigationSource;\n  action: RouterHistoryActionValue;\n  details?: Record<string, unknown>;\n};\n\nexport type RouterPopstateEvent = RouterEventBase & {\n  kind: 'popstate';\n  source: RouterNavigationSource;\n  direction: RouterPopstateDirectionValue;\n  details?: Record<string, unknown>;\n};\n\nexport type RouterPrepareStartEvent = RouterEventBase & {\n  kind: 'prepare-start';\n  source: RouterNavigationSource;\n  routePath?: string;\n  moduleId?: string;\n};\n\nexport type RouterPrepareCompleteEvent = RouterEventBase & {\n  kind: 'prepare-complete';\n  source: RouterNavigationSource;\n  routePath?: string;\n  moduleId?: string;\n  durationMs: number;\n  hasPrepared: boolean;\n  status: Exclude<RouterPrepareStatusValue, 'pending'>;\n};\n\nexport type RouterPrepareEvent =\n  | RouterPrepareStartEvent\n  | RouterPrepareCompleteEvent;\n\nexport type RouterEvent =\n  | RouterSnapshotEvent\n  | RouterPreloadEvent\n  | RouterHistoryEvent\n  | RouterPopstateEvent\n  | RouterPrepareEvent;\n\n/**\n * Instrumentations can observe router events, entry changes, and optionally\n * dispose resources when the registry is torn down.\n */\nexport type InstrumentationAPI = {\n  onEvent?: (event: RouterEvent) => void;\n  onEntryChange?: (entry: RouterEntrySnapshot) => void;\n  dispose?: () => void;\n};\n\nexport type InstrumentationRegistry = {\n  register: (instrumentation: InstrumentationAPI) => void;\n  unregister: (instrumentation: InstrumentationAPI) => void;\n  emitEvent: (event: RouterEvent) => void;\n  notifyEntryChange: (entry: RouterEntrySnapshot) => void;\n  dispose: () => void;\n  readonly size: number;\n};\n\n/**\n * Creates a registry coordinating a set of instrumentations. The registry keeps\n * consumer code simple: router internals broadcast events without caring about\n * the number or nature of attached instrumentations.\n */\nexport function createInstrumentationRegistry(\n  initial: InstrumentationAPI[] = [],\n): InstrumentationRegistry {\n  const listeners = new Set<InstrumentationAPI>();\n  initial.forEach((instrumentation) => {\n    listeners.add(instrumentation);\n  });\n\n  let disposed = false;\n\n  /**\n   * Throws when the registry has already been disposed. Guards event emission.\n   */\n  function guard() {\n    if (disposed) {\n      throw new Error('Instrumentation registry was disposed');\n    }\n  }\n\n  /**\n   * Safely invokes an optional handler provided by an instrumentation.\n   * Failures are swallowed so router execution continues uninterrupted.\n   */\n  function safeInvoke(\n    instrumentation: InstrumentationAPI,\n    method: keyof InstrumentationAPI,\n    payload: unknown[],\n  ): void {\n    const handler = instrumentation[method];\n    if (typeof handler !== 'function') {\n      return;\n    }\n    try {\n      (handler as (...args: unknown[]) => void)(...payload);\n    } catch {\n      // Instrumentations should never break router execution. Errors are swallowed.\n    }\n  }\n\n  return {\n    register(instrumentation) {\n      guard();\n      listeners.add(instrumentation);\n    },\n    unregister(instrumentation) {\n      guard();\n      listeners.delete(instrumentation);\n    },\n    emitEvent(event) {\n      guard();\n      listeners.forEach((instrumentation) => {\n        safeInvoke(instrumentation, 'onEvent', [event]);\n      });\n    },\n    notifyEntryChange(entry) {\n      guard();\n      listeners.forEach((instrumentation) => {\n        safeInvoke(instrumentation, 'onEntryChange', [entry]);\n      });\n    },\n    dispose() {\n      if (disposed) {\n        return;\n      }\n      listeners.forEach((instrumentation) => {\n        safeInvoke(instrumentation, 'dispose', []);\n      });\n      listeners.clear();\n      disposed = true;\n    },\n    get size() {\n      return listeners.size;\n    },\n  };\n}\n"]}
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
import type { MaybePromise, PrepareResult, PreparedValueResource } from './types.js';
|
|
2
|
+
export declare function createPreparedValueResource<TPrepared extends PrepareResult = PrepareResult>(loader: () => MaybePromise<TPrepared>): PreparedValueResource<TPrepared>;
|
|
3
|
+
export declare function readPreparedRouteValue(preparedResource?: PreparedValueResource): unknown;
|
|
4
|
+
//# sourceMappingURL=prepareResource.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"prepareResource.d.ts","sourceRoot":"","sources":["../../src/prepareResource.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EACV,YAAY,EACZ,aAAa,EACb,qBAAqB,EACtB,MAAM,YAAY,CAAC;AAQpB,wBAAgB,2BAA2B,CACzC,SAAS,SAAS,aAAa,GAAG,aAAa,EAC/C,MAAM,EAAE,MAAM,YAAY,CAAC,SAAS,CAAC,GAAG,qBAAqB,CAAC,SAAS,CAAC,CAEzE;AAQD,wBAAgB,sBAAsB,CACpC,gBAAgB,CAAC,EAAE,qBAAqB,GACvC,OAAO,CAKT"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { createAsyncResource } from './asyncResource.js';
|
|
2
|
+
export function createPreparedValueResource(loader) {
|
|
3
|
+
return createAsyncResource(loader);
|
|
4
|
+
}
|
|
5
|
+
export function readPreparedRouteValue(preparedResource) {
|
|
6
|
+
if (preparedResource == null) {
|
|
7
|
+
return undefined;
|
|
8
|
+
}
|
|
9
|
+
return preparedResource.read();
|
|
10
|
+
}
|
|
11
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJlcGFyZVJlc291cmNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3ByZXBhcmVSZXNvdXJjZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQWF6RCxNQUFNLFVBQVUsMkJBQTJCLENBRXpDLE1BQXFDO0lBQ3JDLE9BQU8sbUJBQW1CLENBQUMsTUFBTSxDQUFDLENBQUM7QUFDckMsQ0FBQztBQVFELE1BQU0sVUFBVSxzQkFBc0IsQ0FDcEMsZ0JBQXdDO0lBRXhDLElBQUksZ0JBQWdCLElBQUksSUFBSSxFQUFFLENBQUM7UUFDN0IsT0FBTyxTQUFTLENBQUM7SUFDbkIsQ0FBQztJQUNELE9BQU8sZ0JBQWdCLENBQUMsSUFBSSxFQUFFLENBQUM7QUFDakMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGNyZWF0ZUFzeW5jUmVzb3VyY2UgfSBmcm9tICcuL2FzeW5jUmVzb3VyY2UuanMnO1xuaW1wb3J0IHR5cGUge1xuICBNYXliZVByb21pc2UsXG4gIFByZXBhcmVSZXN1bHQsXG4gIFByZXBhcmVkVmFsdWVSZXNvdXJjZSxcbn0gZnJvbSAnLi90eXBlcy5qcyc7XG5cbi8qKlxuICogQ3JlYXRlcyB0aGUgaW50ZXJuYWwgcmVzb3VyY2UgdXNlZCB0byBzdG9yZSByb3V0ZSBwcmVwYXJlIHZhbHVlcy5cbiAqXG4gKiBAcGFyYW0gbG9hZGVyIC0gRnVuY3Rpb24gcHJvZHVjaW5nIHRoZSBwcmVwYXJlZCByb3V0ZSB2YWx1ZS5cbiAqIEByZXR1cm5zIFN1c3BlbnNlLWZyaWVuZGx5IHJlYWRhYmxlIHJlc291cmNlIGZvciB0aGUgcHJlcGFyZWQgdmFsdWUuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVQcmVwYXJlZFZhbHVlUmVzb3VyY2U8XG4gIFRQcmVwYXJlZCBleHRlbmRzIFByZXBhcmVSZXN1bHQgPSBQcmVwYXJlUmVzdWx0LFxuPihsb2FkZXI6ICgpID0+IE1heWJlUHJvbWlzZTxUUHJlcGFyZWQ+KTogUHJlcGFyZWRWYWx1ZVJlc291cmNlPFRQcmVwYXJlZD4ge1xuICByZXR1cm4gY3JlYXRlQXN5bmNSZXNvdXJjZShsb2FkZXIpO1xufVxuXG4vKipcbiAqIFJlYWRzIHRoZSByZXNvbHZlZCBwcmVwYXJlZCB2YWx1ZSBvciBzdXNwZW5kcy9yZXRocm93cyB0aHJvdWdoIHRoZSByZXNvdXJjZS5cbiAqXG4gKiBAcGFyYW0gcHJlcGFyZWRSZXNvdXJjZSAtIFByZXBhcmVkIHZhbHVlIHJlc291cmNlIGF0dGFjaGVkIHRvIGEgcm91dGUuXG4gKiBAcmV0dXJucyBSZXNvbHZlZCBwcmVwYXJlZCB2YWx1ZSB3aGVuIGF2YWlsYWJsZS5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHJlYWRQcmVwYXJlZFJvdXRlVmFsdWUoXG4gIHByZXBhcmVkUmVzb3VyY2U/OiBQcmVwYXJlZFZhbHVlUmVzb3VyY2UsXG4pOiB1bmtub3duIHtcbiAgaWYgKHByZXBhcmVkUmVzb3VyY2UgPT0gbnVsbCkge1xuICAgIHJldHVybiB1bmRlZmluZWQ7XG4gIH1cbiAgcmV0dXJuIHByZXBhcmVkUmVzb3VyY2UucmVhZCgpO1xufVxuIl19
|
|
@@ -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;
|
|
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;AAkChD,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,IAgIhB,CAAC;AAEF,eAAe,qBAAqB,CAAC"}
|
|
@@ -3,24 +3,31 @@ import { useContext, useMemo } from 'react';
|
|
|
3
3
|
import { HttpRedirect } from '../errors/index.js';
|
|
4
4
|
import RoutingContext from './RoutingContext.js';
|
|
5
5
|
import RouteComponent from './RouteComponent.js';
|
|
6
|
+
import { readPreparedRouteValue } from '../prepareResource.js';
|
|
6
7
|
function RouteHighlightBoundary(props) {
|
|
7
8
|
const { highlightId, children } = props;
|
|
8
9
|
return (_jsxs(_Fragment, { children: [_jsx("span", { "data-plumile-route-start": highlightId, "aria-hidden": "true", style: { display: 'none' } }), children, _jsx("span", { "data-plumile-route-end": highlightId, "aria-hidden": "true", style: { display: 'none' } })] }));
|
|
9
10
|
}
|
|
10
11
|
const RouteComponentWrapper = (props) => {
|
|
11
12
|
const { children, match, preparedRoute } = props;
|
|
12
|
-
const { resourcePage,
|
|
13
|
+
const { resourcePage, preparedResource, render, redirectTo, highlightId } = preparedRoute;
|
|
13
14
|
const router = useContext(RoutingContext);
|
|
14
15
|
const currentEntry = router?.get();
|
|
15
16
|
const rawQuery = currentEntry?.query ?? {};
|
|
16
17
|
const routeContext = currentEntry?.context;
|
|
17
18
|
const typedQuery = rawQuery;
|
|
19
|
+
const preparedSegments = currentEntry?.preparedMatch.segments ?? [];
|
|
20
|
+
const { segmentIndex } = preparedRoute;
|
|
21
|
+
const leafSegment = preparedSegments.at(-1) ?? preparedRoute;
|
|
18
22
|
const route = useMemo(() => {
|
|
19
23
|
if (match == null) {
|
|
20
24
|
return null;
|
|
21
25
|
}
|
|
22
26
|
return match.route;
|
|
23
27
|
}, [match]);
|
|
28
|
+
const prepared = useMemo(() => {
|
|
29
|
+
return readPreparedRouteValue(preparedResource);
|
|
30
|
+
}, [preparedResource]);
|
|
24
31
|
const Component = useMemo(() => {
|
|
25
32
|
if (resourcePage == null) {
|
|
26
33
|
return;
|
|
@@ -34,7 +41,11 @@ const RouteComponentWrapper = (props) => {
|
|
|
34
41
|
content = render({
|
|
35
42
|
children,
|
|
36
43
|
Component,
|
|
44
|
+
preparedSegment: preparedRoute,
|
|
37
45
|
preparedRoute,
|
|
46
|
+
preparedSegments,
|
|
47
|
+
segmentIndex,
|
|
48
|
+
leafSegment,
|
|
38
49
|
route,
|
|
39
50
|
prepared,
|
|
40
51
|
context: routeContext,
|
|
@@ -82,9 +93,9 @@ const RouteComponentWrapper = (props) => {
|
|
|
82
93
|
if (Component == null) {
|
|
83
94
|
return null;
|
|
84
95
|
}
|
|
85
|
-
const componentNode = (_jsx(Component, { match: match, preparedRoute: preparedRoute, route: route, prepared: prepared, query: typedQuery, children: children }));
|
|
96
|
+
const componentNode = (_jsx(Component, { match: match, preparedRoute: preparedRoute, route: route, prepared: prepared, query: typedQuery, preparedSegments: preparedSegments, segmentIndex: segmentIndex, leafSegment: leafSegment, children: children }));
|
|
86
97
|
const content2 = wrapWithHighlight(componentNode);
|
|
87
98
|
return _jsx(RouteComponent, { redirectToPathname: pathname, content: content2 });
|
|
88
99
|
};
|
|
89
100
|
export default RouteComponentWrapper;
|
|
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"]}
|
|
101
|
+
//# 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';\nimport { readPreparedRouteValue } from '../prepareResource.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, preparedResource, 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  const preparedSegments = currentEntry?.preparedMatch.segments ?? [];\n  const { segmentIndex } = preparedRoute;\n  const leafSegment = preparedSegments.at(-1) ?? preparedRoute;\n\n  const route = useMemo(() => {\n    if (match == null) {\n      return null;\n    }\n\n    return match.route;\n  }, [match]);\n\n  const prepared = useMemo(() => {\n    return readPreparedRouteValue(preparedResource);\n  }, [preparedResource]);\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:\n    | JSX.Element\n    | null\n    | undefined\n    | Promise<JSX.Element | null | undefined> = 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        preparedSegment: preparedRoute,\n        preparedRoute,\n        preparedSegments,\n        segmentIndex,\n        leafSegment,\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      preparedSegments={preparedSegments}\n      segmentIndex={segmentIndex}\n      leafSegment={leafSegment}\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,13 +1,19 @@
|
|
|
1
|
-
import { type
|
|
1
|
+
import { type Schema as FilterSchema } from '@plumile/filter-query';
|
|
2
|
+
import type { ParamData } from 'path-to-regexp';
|
|
3
|
+
import { type RoutingContextType, type AnyRoute, type PreparedAccess, type NavigateOverloads, type UnknownQueryShape } from '../types.js';
|
|
2
4
|
import { type InstrumentationAPI } from '../instrumentation/Instrumentation.js';
|
|
5
|
+
type RouterQuerySchema = FilterSchema | undefined;
|
|
3
6
|
export type CreateRouterReturn<TContext, R extends AnyRoute<TContext>[]> = {
|
|
4
7
|
cleanup: () => void;
|
|
5
|
-
context: RoutingContextType<
|
|
8
|
+
context: RoutingContextType<ParamData, RouterQuerySchema, UnknownQueryShape, TContext> & PreparedAccess<R> & {
|
|
9
|
+
navigate: NavigateOverloads<R>;
|
|
10
|
+
};
|
|
6
11
|
};
|
|
7
12
|
export type CreateRouterOptions<TContext> = {
|
|
8
13
|
instrumentations?: InstrumentationAPI[];
|
|
9
14
|
context?: TContext | (() => TContext);
|
|
10
15
|
getContext?: () => TContext;
|
|
11
16
|
};
|
|
12
|
-
export default function createRouter<TContext = unknown, R extends AnyRoute<TContext>[] = AnyRoute<TContext>[]>(routes: [...R] | AnyRoute<TContext>[], options?: CreateRouterOptions<TContext>): CreateRouterReturn<TContext, R
|
|
17
|
+
export default function createRouter<TContext = unknown, R extends AnyRoute<TContext>[] = AnyRoute<TContext>[]>(routes: [...R] | AnyRoute<TContext>[], options?: CreateRouterOptions<TContext>): CreateRouterReturn<TContext, R>;
|
|
18
|
+
export {};
|
|
13
19
|
//# sourceMappingURL=createRouter.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"createRouter.d.ts","sourceRoot":"","sources":["../../../src/routing/createRouter.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"createRouter.d.ts","sourceRoot":"","sources":["../../../src/routing/createRouter.ts"],"names":[],"mappings":"AAKA,OAAO,EAIL,KAAK,MAAM,IAAI,YAAY,EAC5B,MAAM,uBAAuB,CAAC;AAC/B,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAChD,OAAO,EAEL,KAAK,kBAAkB,EAEvB,KAAK,QAAQ,EACb,KAAK,cAAc,EACnB,KAAK,iBAAiB,EAOtB,KAAK,iBAAiB,EACvB,MAAM,aAAa,CAAC;AACrB,OAAO,EAEL,KAAK,kBAAkB,EAKxB,MAAM,uCAAuC,CAAC;AAQ/C,KAAK,iBAAiB,GAAG,YAAY,GAAG,SAAS,CAAC;AAgBlD,MAAM,MAAM,kBAAkB,CAAC,QAAQ,EAAE,CAAC,SAAS,QAAQ,CAAC,QAAQ,CAAC,EAAE,IAAI;IAEzE,OAAO,EAAE,MAAM,IAAI,CAAC;IAEpB,OAAO,EAAE,kBAAkB,CACzB,SAAS,EACT,iBAAiB,EACjB,iBAAiB,EACjB,QAAQ,CACT,GACC,cAAc,CAAC,CAAC,CAAC,GAAG;QAClB,QAAQ,EAAE,iBAAiB,CAAC,CAAC,CAAC,CAAC;KAChC,CAAC;CACL,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,CAAC,QAAQ,EAAE,CAAC,CAAC,CAk0BjC"}
|