@manyducks.co/dolla 2.0.0-alpha.60 → 2.0.0-alpha.61

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 (50) hide show
  1. package/dist/core/context.d.ts +0 -2
  2. package/dist/core/context.test.d.ts +1 -0
  3. package/dist/core/index.d.ts +4 -5
  4. package/dist/core/logger.test.d.ts +0 -0
  5. package/dist/core/markup.d.ts +16 -37
  6. package/dist/core/markup.test.d.ts +0 -0
  7. package/dist/core/mount.test.d.ts +0 -0
  8. package/dist/core/nodes/_markup.d.ts +36 -0
  9. package/dist/core/nodes/dom.d.ts +5 -6
  10. package/dist/core/nodes/dynamic.d.ts +6 -13
  11. package/dist/core/nodes/{html.d.ts → element.d.ts} +9 -8
  12. package/dist/core/nodes/portal.d.ts +6 -7
  13. package/dist/core/nodes/repeat.d.ts +7 -5
  14. package/dist/core/nodes/view.d.ts +6 -5
  15. package/dist/core/ref.test.d.ts +1 -0
  16. package/dist/core/signals.d.ts +17 -2
  17. package/dist/core/views/default-crash-view.d.ts +5 -2
  18. package/dist/hooks/index.d.ts +64 -0
  19. package/dist/hooks/index.test.d.ts +1 -0
  20. package/dist/hooks.js +69 -0
  21. package/dist/hooks.js.map +1 -0
  22. package/dist/i18n.js +49 -49
  23. package/dist/i18n.js.map +1 -1
  24. package/dist/index.js +50 -60
  25. package/dist/index.js.map +1 -1
  26. package/dist/jsx-dev-runtime.js +1 -1
  27. package/dist/jsx-runtime.js +1 -1
  28. package/dist/logger-Bl496yfY.js +91 -0
  29. package/dist/logger-Bl496yfY.js.map +1 -0
  30. package/dist/markup-CX27GJ1M.js +1030 -0
  31. package/dist/markup-CX27GJ1M.js.map +1 -0
  32. package/dist/ref-BD79iqlg.js +15 -0
  33. package/dist/ref-BD79iqlg.js.map +1 -0
  34. package/dist/router/router.d.ts +29 -18
  35. package/dist/{router-BpuJQ6OA.js → router-CjCkk4dA.js} +217 -190
  36. package/dist/router-CjCkk4dA.js.map +1 -0
  37. package/dist/router.js +1 -1
  38. package/dist/signals-gCwiIe5X.js +450 -0
  39. package/dist/signals-gCwiIe5X.js.map +1 -0
  40. package/dist/types.d.ts +1 -1
  41. package/docs/hooks.md +211 -0
  42. package/docs/i18n.md +1 -1
  43. package/package.json +5 -1
  44. package/vite.config.js +1 -0
  45. package/dist/core/symbols.d.ts +0 -2
  46. package/dist/logger-MPwl-Xqu.js +0 -524
  47. package/dist/logger-MPwl-Xqu.js.map +0 -1
  48. package/dist/markup-BGlfQYQk.js +0 -996
  49. package/dist/markup-BGlfQYQk.js.map +0 -1
  50. package/dist/router-BpuJQ6OA.js.map +0 -1
@@ -1 +0,0 @@
1
- {"version":3,"file":"markup-BGlfQYQk.js","sources":["../src/core/context.ts","../src/core/symbols.ts","../src/core/nodes/dom.ts","../src/core/nodes/dynamic.ts","../src/core/nodes/view.ts","../src/core/nodes/html.ts","../src/core/nodes/portal.ts","../src/core/nodes/repeat.ts","../src/core/markup.ts"],"sourcesContent":["import { isFunction, typeOf } from \"../typeChecking\";\nimport type { Store } from \"../types\";\nimport { getUniqueId } from \"../utils\";\nimport { createLogger, type Logger, type LoggerOptions } from \"./logger\";\nimport { effect, type EffectFn, get, type MaybeSignal, untracked, type UnsubscribeFn } from \"./signals\";\n\nexport enum LifecycleEvent {\n WILL_MOUNT,\n DID_MOUNT,\n WILL_UNMOUNT,\n DID_UNMOUNT,\n DISPOSE,\n}\n\ntype LifecycleListener = () => void;\n\nenum LifecycleState {\n Unmounted = 0,\n WillMount = 1,\n DidMount = 2,\n WillUnmount = 3,\n DidUnmount = 4,\n Disposed = 5,\n}\n\nconst NAME = Symbol(\"name\");\nconst LIFECYCLE = Symbol(\"lifecycle\");\nconst PARENT = Symbol(\"parent\");\nconst STORES = Symbol(\"stores\");\nconst STATE = Symbol(\"state\");\n\n/**\n * Manages lifecycle events for a Context.\n */\nclass ContextLifecycle {\n private context;\n\n state = LifecycleState.Unmounted;\n listeners = new Map<LifecycleEvent, Set<LifecycleListener>>();\n bound?: Set<Context>;\n\n constructor(context: Context) {\n this.context = context;\n }\n\n /**\n * Listen for a certain event to be emitted. Listeners are called when the event results in a state change.\n */\n on<E extends LifecycleEvent>(event: E, listener: LifecycleListener) {\n const listeners = this.listeners.get(event);\n if (!listeners) {\n this.listeners.set(event, new Set([listener]));\n } else {\n listeners.add(listener);\n }\n }\n\n /**\n * Stop a particular listener from being called when an event is emitted.\n */\n off<E extends LifecycleEvent>(event: E, listener: LifecycleListener) {\n const listeners = this.listeners.get(event);\n if (listeners) {\n listeners.delete(listener);\n if (listeners.size === 0) {\n this.listeners.delete(event);\n }\n }\n }\n\n /**\n * Advance the lifecycle state machine.\n */\n emit<E extends LifecycleEvent>(event: E) {\n switch (event) {\n case LifecycleEvent.WILL_MOUNT: {\n if (this.state < LifecycleState.WillMount) {\n this.state = LifecycleState.WillMount;\n this.notify(event);\n } else {\n this.context.crash(new Error(`Tried to WILL_MOUNT context at state ${this.state}`));\n }\n break;\n }\n case LifecycleEvent.DID_MOUNT: {\n if (this.state >= LifecycleState.WillMount && this.state < LifecycleState.DidMount) {\n this.state = LifecycleState.DidMount;\n this.notify(event);\n } else {\n this.context.crash(new Error(`Tried to WILL_UNMOUNT context at state ${this.state}`));\n }\n break;\n }\n case LifecycleEvent.WILL_UNMOUNT: {\n if (this.state >= LifecycleState.DidMount && this.state < LifecycleState.WillUnmount) {\n this.notify(event);\n this.state = LifecycleState.WillUnmount;\n } else {\n this.context.crash(new Error(`Tried to WILL_UNMOUNT context at state ${this.state}`));\n }\n break;\n }\n case LifecycleEvent.DID_UNMOUNT: {\n if (this.state >= LifecycleState.WillUnmount && this.state < LifecycleState.DidUnmount) {\n // Loop back to .Unmounted\n this.state = LifecycleState.DidUnmount % LifecycleState.DidUnmount;\n this.notify(event);\n } else {\n this.context.crash(new Error(`Tried to DID_UNMOUNT context at state ${this.state}`));\n }\n break;\n }\n case LifecycleEvent.DISPOSE: {\n if (this.state === LifecycleState.Unmounted) {\n this.notify(event);\n this.listeners.clear();\n this.bound = undefined;\n this.context[STATE] = undefined;\n this.context[STORES] = undefined;\n this.state = LifecycleState.Disposed;\n } else {\n this.context.crash(new Error(`Tried to DISPOSE context at state ${this.state}`));\n }\n break;\n }\n }\n }\n\n /**\n * Bind `context` to this lifecycle; when any event is emitted here it will be emitted for `context` as well.\n */\n bind(context: Context) {\n if (!this.bound) {\n this.bound = new Set([context]);\n } else {\n this.bound.add(context);\n }\n }\n\n /**\n * Call all the event's listeners and re-emit to bound contexts.\n */\n private notify<E extends LifecycleEvent>(event: E) {\n // Call listener functions.\n const listeners = this.listeners.get(event);\n if (listeners) {\n for (const listener of listeners) {\n listener();\n }\n }\n // Emit to bound contexts.\n if (this.bound) {\n for (const context of this.bound) {\n context[LIFECYCLE].emit(event);\n }\n }\n }\n}\n\nexport interface ContextOptions {\n logger?: LoggerOptions;\n}\n\nexport interface LinkedContextOptions extends ContextOptions {\n bindLifecycleToParent?: boolean;\n}\n\nexport interface Context extends Logger {}\n\nexport class Context implements Logger {\n #name: MaybeSignal<string>;\n\n [NAME]: string;\n [LIFECYCLE] = new ContextLifecycle(this);\n [PARENT]?: Context;\n [STORES]?: Map<Store<any, any>, any>;\n [STATE]?: Map<any, any>;\n\n get isMounted() {\n const { state } = this[LIFECYCLE];\n return state >= LifecycleState.DidMount && state < LifecycleState.DidUnmount;\n }\n\n /**\n * Returns a new Context with this one as its parent.\n */\n static linked(parent: Context, name: MaybeSignal<string>, options?: LinkedContextOptions): Context {\n const context = new Context(name, options);\n context[PARENT] = parent;\n if (options?.bindLifecycleToParent) parent[LIFECYCLE].bind(context);\n return context;\n }\n\n /**\n * Emit a lifecycle event to `context`.\n */\n static emit(context: Context, event: LifecycleEvent) {\n context[LIFECYCLE].emit(event);\n }\n\n /**\n * Traverses _parent contexts until arriving at one that doesn't have a parent itself.\n * Returns null if this context is the parent.\n */\n static getRoot(context: Context) {\n // This is like the programming version of \"Buffalo buffalo buffalo...\"\n let parent = context[PARENT];\n while (parent?.[PARENT]) {\n parent = parent[PARENT];\n }\n return parent ?? null;\n }\n\n constructor(name: MaybeSignal<string>, options?: ContextOptions) {\n this.#name = name;\n this[NAME] = untracked(name);\n\n // Add logger methods.\n const logger = createLogger(() => get(this.#name), options?.logger);\n const descriptors = Object.getOwnPropertyDescriptors(logger);\n for (const key in descriptors) {\n Object.defineProperty(this, key, descriptors[key]);\n }\n }\n\n /**\n * Returns the current name of this context.\n */\n getName(): string {\n return untracked(this.#name);\n }\n\n /**\n * Sets a new name for this context.\n */\n setName(name: MaybeSignal<string>) {\n this.#name = name;\n this[NAME] = untracked(name); // Try to store name as a readable string for debugging purposes.\n }\n\n /**\n * Creates an instance of a store and attaches it to this context.\n */\n addStore<T>(store: Store<any, T>, options?: any): this {\n if (this[STORES]?.get(store)) {\n let name = store.name ? `'${store.name}'` : \"this store\";\n throw this.crash(new StoreError(`An instance of ${name} was already added on this context.`));\n }\n\n const context = Context.linked(this, store.name, {\n bindLifecycleToParent: true,\n logger: { tag: getUniqueId(), tagName: \"uid\" },\n });\n try {\n if (!this[STORES]) this[STORES] = new Map();\n const result = store.call(context, options, context);\n this[STORES].set(store, result);\n } catch (error) {\n throw this.crash(error as Error);\n }\n\n return this;\n }\n\n /**\n * Retrieves the nearest instance of `store`. If this context doesn't have it, the parent context is checked. This process continues until either:\n * 1. An instance of the store is found and returned.\n * 2. No instance is found and an error is thrown.\n */\n getStore<T>(store: Store<any, T>): T {\n if (!isFunction(store)) {\n throw new StoreError(`Invalid store.`);\n }\n let context: Context = this;\n let result: unknown;\n while (true) {\n result = context[STORES]?.get(store);\n if (result == null && context[PARENT] != null) {\n context = context[PARENT];\n } else {\n break;\n }\n }\n if (result == null) {\n throw this.crash(new StoreError(`Store '${store.name}' is not provided by this context.`));\n }\n return result as T;\n }\n\n /**\n * Schedule a callback function to run just before this context is mounted.\n */\n beforeMount(listener: LifecycleListener) {\n this[LIFECYCLE].on(LifecycleEvent.WILL_MOUNT, listener);\n return () => this[LIFECYCLE].off(LifecycleEvent.WILL_MOUNT, listener);\n }\n\n /**\n * Schedule a callback function to run after this context is mounted.\n */\n onMount(listener: LifecycleListener) {\n this[LIFECYCLE].on(LifecycleEvent.DID_MOUNT, listener);\n return () => this[LIFECYCLE].off(LifecycleEvent.DID_MOUNT, listener);\n }\n\n /**\n * Schedule a callback function to run just before this context is unmounted.\n */\n beforeUnmount(listener: LifecycleListener) {\n this[LIFECYCLE].on(LifecycleEvent.WILL_UNMOUNT, listener);\n return () => this[LIFECYCLE].off(LifecycleEvent.WILL_UNMOUNT, listener);\n }\n\n /**\n * Schedule a callback function to run after this context is unmounted.\n */\n onUnmount(listener: LifecycleListener) {\n this[LIFECYCLE].on(LifecycleEvent.DID_UNMOUNT, listener);\n return () => this[LIFECYCLE].off(LifecycleEvent.DID_UNMOUNT, listener);\n }\n\n effect(callback: EffectFn) {\n const fn = () => {\n try {\n return callback(); // Return callback so cleanup function passes through to effect handler\n } catch (error) {\n this.error(error);\n if (error instanceof Error) {\n this.crash(error);\n } else if (typeof error === \"string\") {\n this.crash(new Error(error));\n } else {\n this.crash(new Error(`Unknown error thrown in effect callback`));\n }\n }\n };\n\n if (this[LIFECYCLE].state >= LifecycleState.WillMount) {\n // This code is probably in a lifecycle hook; run the effect immediately and trigger unsubscribe when context unmounts.\n const unsubscribe = effect(fn);\n this[LIFECYCLE].on(LifecycleEvent.DID_UNMOUNT, unsubscribe);\n return unsubscribe;\n } else {\n // Prime the effect to run when the context is mounted and unsubscribe when unmounted, unless unsubscribed before `willMount`.\n let unsubscribe: UnsubscribeFn | undefined;\n let disposed = false;\n this[LIFECYCLE].on(LifecycleEvent.WILL_MOUNT, () => {\n if (!disposed) {\n unsubscribe = effect(fn);\n this[LIFECYCLE].on(LifecycleEvent.DID_UNMOUNT, unsubscribe);\n }\n });\n return () => {\n if (unsubscribe != null) {\n disposed = true;\n unsubscribe();\n }\n };\n }\n }\n\n /**\n * Gets the value stored at `key`, or returns the `defaultValue` if none is set.\n */\n getState<T>(key: any, defaultValue: T): T;\n\n /**\n * Gets the value stored at `key`, or throws an error if none is set.\n */\n getState<T>(key: any): T;\n\n /**\n * Returns a Map containing all state values available to this context.\n */\n getState(): Map<any, any>;\n\n getState<T>(key?: any, defaultValue?: T): T | Map<any, any> {\n if (arguments.length > 0) {\n // Get value by key\n let context: Context = this;\n let value: any;\n while (true) {\n value = context[STATE]?.get(key);\n if (value === undefined && context[PARENT] != null) {\n context = context[PARENT];\n } else {\n break;\n }\n }\n if (value === undefined) {\n if (arguments.length > 1) {\n return defaultValue!;\n } else {\n throw new Error(`Expected a value for '${String(key)}' but got undefined.`);\n }\n }\n return value;\n } else {\n // Get merged values\n let context: Context = this;\n const entries: [any, any][] = [];\n while (true) {\n if (context[STATE]) {\n entries.push(...context[STATE].entries());\n }\n if (context[PARENT] != null) {\n context = context[PARENT];\n } else {\n break;\n }\n }\n return new Map(entries.reverse());\n }\n }\n\n /**\n * Stores `value` at `key` in this context's state.\n */\n setState<T>(key: any, value: T): void;\n\n /**\n * For each tuple in `entries`, stores `value` at `key` in this context's state.\n */\n setState(entries: [key: any, value: any][]): void;\n\n setState() {\n if (!this[STATE]) {\n this[STATE] = new Map();\n }\n if (arguments.length === 2) {\n this[STATE].set(arguments[0], arguments[1]);\n } else if (typeOf(arguments[0]) === \"array\") {\n for (const [key, value] of arguments[0]) {\n if (value === undefined) {\n this[STATE].delete(key);\n } else {\n this[STATE].set(key, value);\n }\n }\n } else {\n throw new Error(`Invalid arguments.`);\n }\n\n return this;\n }\n}\n\nexport function createContext(name: MaybeSignal<string>, options?: ContextOptions) {\n return new Context(name, options);\n}\n\nexport class StoreError extends Error {}\n","export const TYPE = Symbol(\"type\");\nexport const MARKUP_NODE = Symbol(\"MarkupNode\");\n","import type { MarkupNode } from \"../markup\";\nimport { MARKUP_NODE, TYPE } from \"../symbols\";\n\n/**\n * Lightweight MarkupNode wrapper for a plain DOM node.\n */\nexport class DOMNode implements MarkupNode {\n [TYPE] = MARKUP_NODE;\n\n root: Node;\n\n constructor(node: Node) {\n this.root = node;\n }\n\n isMounted() {\n return this.root.parentNode != null;\n }\n\n mount(parent: Element, after?: Node) {\n parent.insertBefore(this.root, after?.nextSibling ?? null);\n }\n\n unmount(skipDOM = false) {\n if (!skipDOM && this.root.parentNode) {\n this.root.parentNode.removeChild(this.root);\n }\n }\n\n move(parent: Element, after?: Node) {\n if (\"moveBefore\" in parent && this.root instanceof Element) {\n try {\n (parent as any).moveBefore(this.root, after?.nextSibling ?? null);\n } catch {\n this.mount(parent, after);\n }\n } else {\n this.mount(parent, after);\n }\n }\n}\n","import { toArray } from \"../../utils.js\";\nimport type { Context } from \"../context.js\";\nimport { toMarkupNodes, type MarkupNode } from \"../markup.js\";\nimport { effect, untracked, type Signal, type UnsubscribeFn } from \"../signals.js\";\nimport { MARKUP_NODE, TYPE } from \"../symbols.js\";\n\n/**\n * Displays dynamic children without a parent element.\n * Renders a Reactive value via a render function.\n *\n * This is probably the most used element type aside from HTML.\n */\nexport class Dynamic implements MarkupNode {\n [TYPE] = MARKUP_NODE;\n\n root = document.createTextNode(\"\");\n\n private children: MarkupNode[] = [];\n private context: Context;\n\n private $slot: Signal<any>;\n private unsubscribe?: UnsubscribeFn;\n\n constructor(context: Context, $slot: Signal<any>) {\n this.context = context;\n this.$slot = $slot;\n }\n\n isMounted() {\n return this.root.parentElement != null;\n }\n\n mount(parent: Node, after?: Node) {\n if (!this.isMounted()) {\n parent.insertBefore(this.root, after?.nextSibling ?? null);\n\n this.unsubscribe = effect(() => {\n try {\n const content = this.$slot();\n untracked(() => {\n this.update(toArray(content));\n });\n } catch (error) {\n this.context.crash(error as Error);\n }\n });\n }\n }\n\n unmount(skipDOM = false) {\n this.unsubscribe?.();\n\n if (this.isMounted()) {\n this.cleanup(skipDOM);\n this.root.parentNode?.removeChild(this.root);\n }\n }\n\n move(parent: Element, after?: Node) {\n if (\"moveBefore\" in parent) {\n try {\n (parent as any).moveBefore(this.root, after?.nextSibling ?? null);\n for (let i = 0; i < this.children.length; i++) {\n this.children[i].move(parent, this.children[i - 1]?.root ?? this.root);\n }\n (parent as any).moveBefore(this.root, this.children.at(-1)?.root?.nextSibling ?? null);\n } catch {\n this.mount(parent, after);\n }\n } else {\n this.mount(parent, after);\n }\n }\n\n private cleanup(skipDOM: boolean) {\n for (const element of this.children) {\n if (element.isMounted()) element.unmount(skipDOM);\n }\n this.children.length = 0;\n }\n\n private update(content: any[]) {\n this.cleanup(false);\n\n if (content.length === 0 || !this.isMounted()) return;\n\n const nodes = toMarkupNodes(this.context, content);\n\n for (const node of nodes) {\n const previous = this.children.at(-1)?.root || this.root;\n node.mount(this.root.parentElement!, previous);\n this.children.push(node);\n }\n\n this.moveMarker();\n }\n\n /**\n * Move marker node to end of children.\n */\n private moveMarker() {\n const parent = this.root.parentElement!;\n const lastChildNextSibling = this.children.at(-1)?.root?.nextSibling ?? null;\n if (\"moveBefore\" in parent) {\n (parent as any).moveBefore(this.root, lastChildNextSibling);\n } else {\n parent.insertBefore(this.root, lastChildNextSibling);\n }\n }\n}\n","import type { View } from \"../../types.js\";\nimport { getUniqueId } from \"../../utils.js\";\nimport { Context, LifecycleEvent } from \"../context.js\";\nimport { render, type MarkupNode } from \"../markup.js\";\nimport { TYPE, MARKUP_NODE } from \"../symbols.js\";\n\nexport const VIEW = Symbol(\"View\");\n\nexport class ViewInstance<P> implements MarkupNode {\n [TYPE] = MARKUP_NODE;\n\n uniqueId = getUniqueId();\n context: Context;\n props;\n view;\n\n node?: MarkupNode;\n\n get root() {\n return this.node?.root!;\n }\n\n /**\n * @param context - Parent contenxt to link to.\n * @param view - View function to mount.\n * @param props - Props to pass to view function.\n */\n constructor(context: Context, view: View<P>, props: P) {\n this.context = Context.linked(context, view.name ?? \"anonymous view\", {\n logger: {\n tag: this.uniqueId,\n tagName: \"uid\",\n },\n });\n this.context.setState(VIEW, this);\n this.props = props;\n this.view = view;\n }\n\n isMounted() {\n return this.node?.isMounted() ?? false;\n }\n\n mount(parent: Element, after?: Node) {\n // Don't run lifecycle hooks or initialize if already connected.\n // Calling connect again can be used to re-order elements that are already connected to the DOM.\n const wasMounted = this.isMounted();\n\n if (!wasMounted) {\n const { context, props, view } = this;\n try {\n const result = view.call(context, props, context);\n if (result != null && result !== false) {\n this.node = render(result, context);\n }\n } catch (error) {\n if (error instanceof Error) {\n context.crash(error);\n }\n throw error;\n }\n\n Context.emit(this.context, LifecycleEvent.WILL_MOUNT);\n }\n\n if (this.node) {\n this.node.mount(parent, after);\n }\n\n if (!wasMounted) Context.emit(this.context, LifecycleEvent.DID_MOUNT);\n }\n\n unmount(skipDOM = false) {\n Context.emit(this.context, LifecycleEvent.WILL_UNMOUNT);\n\n if (this.node) {\n this.node.unmount(skipDOM);\n }\n\n Context.emit(this.context, LifecycleEvent.DID_UNMOUNT);\n Context.emit(this.context, LifecycleEvent.DISPOSE);\n }\n\n move(parent: Element, after?: Node) {\n this.node?.move(parent, after);\n }\n}\n","import { isFunction, isObject, isString } from \"../../typeChecking.js\";\nimport { omit, toArray, toCamelCase } from \"../../utils.js\";\nimport { Context, LifecycleEvent } from \"../context.js\";\nimport { getEnv } from \"../env.js\";\nimport { toMarkupNodes, type MarkupNode } from \"../markup.js\";\nimport { effect, get, type MaybeSignal, type Signal, type Source, type UnsubscribeFn } from \"../signals.js\";\nimport { MARKUP_NODE, TYPE } from \"../symbols.js\";\nimport { VIEW, ViewInstance } from \"./view.js\";\n\nconst isCamelCaseEventName = (key: string) => /^on[A-Z]/.test(key);\n\nconst IS_SVG = Symbol(\"HTML.isSVG\");\n\n// Attributes in this list will not be forwarded to the DOM node.\nconst privateProps = [\"class\", \"className\", \"ref\", \"mixin\", \"children\"];\n\nexport class HTML implements MarkupNode {\n [TYPE] = MARKUP_NODE;\n\n root: HTMLElement | SVGElement;\n\n private parentContext: Context;\n private context: Context;\n\n tag;\n props: Record<string, any>;\n\n private childNodes: MarkupNode[] = [];\n private unsubscribers: UnsubscribeFn[] = [];\n\n // Track the ref so we can nullify it on unmount.\n private ref?: Source<any>;\n\n // Prevents 'onClickOutside' handlers from firing in the same cycle in which the element is connected.\n private canClickAway = false;\n\n constructor(context: Context, tag: string, props: Record<string, any>) {\n this.parentContext = context;\n this.tag = tag;\n this.props = props;\n\n this.context = Context.linked(this.parentContext, getLoggerName.bind(this));\n\n // This and all nested views will be created as SVG elements.\n if (tag.toLowerCase() === \"svg\") {\n this.context.setState(IS_SVG, true);\n }\n\n // Create node with the appropriate constructor.\n if (this.context.getState(IS_SVG, false)) {\n this.root = document.createElementNS(\"http://www.w3.org/2000/svg\", tag);\n } else {\n this.root = document.createElement(tag);\n }\n\n // Add view name as a data attribute in development mode for better debugging.\n if (getEnv() === \"development\") {\n const view = this.context.getState<ViewInstance<any> | null>(VIEW, null);\n if (view) {\n this.root.dataset.view = view.context.getName();\n }\n }\n\n if (props.mixin) {\n for (const fn of toArray(props.mixin)) {\n fn(\n this.root,\n Context.linked(this.context, getLoggerName.bind(this), {\n bindLifecycleToParent: true,\n logger: { tagName: fn.name === \"mixin\" ? undefined : \"mixin\", tag: fn.name },\n }),\n );\n }\n }\n\n const classes = props.className ?? props.class;\n\n this.applyProps(this.root, omit(privateProps, props));\n if (props.style) this.applyStyles(this.root, props.style, this.unsubscribers);\n if (classes) this.applyClasses(this.root, classes, this.unsubscribers);\n\n if (props.ref) {\n if (isFunction(props.ref)) {\n this.ref = props.ref;\n this.ref(this.root);\n } else {\n throw new Error(\"Expected ref to be a function. Got: \" + props.ref);\n }\n }\n\n if (props.children) {\n this.childNodes = toMarkupNodes(this.context, props.children);\n }\n }\n\n isMounted() {\n return this.context.isMounted;\n // return this.root?.parentNode != null;\n }\n\n mount(parent: Node, after?: Node) {\n const wasMounted = this.isMounted();\n\n if (!wasMounted) {\n Context.emit(this.context, LifecycleEvent.WILL_MOUNT);\n\n for (let i = 0; i < this.childNodes.length; i++) {\n const child = this.childNodes[i];\n const previous = i > 0 ? this.childNodes[i - 1].root : undefined;\n child.mount(this.root!, previous);\n }\n }\n\n parent.insertBefore(this.root!, after?.nextSibling ?? null);\n\n this.canClickAway = true;\n\n if (!wasMounted) Context.emit(this.context, LifecycleEvent.DID_MOUNT);\n }\n\n unmount(skipDOM = false) {\n Context.emit(this.context, LifecycleEvent.WILL_UNMOUNT);\n\n if (!skipDOM) {\n this.root!.parentNode?.removeChild(this.root!);\n }\n\n for (const child of this.childNodes) {\n child.unmount(true);\n }\n\n this.canClickAway = false;\n\n for (const unsubscribe of this.unsubscribers) {\n unsubscribe();\n }\n this.unsubscribers.length = 0;\n\n if (this.ref) {\n this.ref(undefined);\n }\n\n Context.emit(this.context, LifecycleEvent.DID_UNMOUNT);\n Context.emit(this.context, LifecycleEvent.DISPOSE);\n }\n\n move(parent: Element, after?: Node) {\n if (\"moveBefore\" in parent) {\n try {\n (parent as any).moveBefore(this.root!, after?.nextSibling ?? null);\n } catch {\n this.mount(parent, after);\n }\n } else {\n this.mount(parent, after);\n }\n }\n\n private attachProp<T>(value: MaybeSignal<T>, callback: (value: T) => void) {\n if (isFunction(value)) {\n this.unsubscribers.push(\n effect(() => {\n try {\n callback((value as Signal<T>)());\n } catch (error) {\n this.context.error(error);\n this.context.crash(error as Error);\n }\n }),\n );\n } else {\n callback(value);\n }\n }\n\n private applyProps(element: HTMLElement | SVGElement, props: Record<string, unknown>) {\n for (const key in props) {\n const value = props[key];\n\n if (key === \"on:clickoutside\" || key === \"onClickOutside\" || key === \"onclickoutside\") {\n // Synthetic onclickoutside event.\n\n const listener = (e: Event) => {\n if (this.canClickAway && !element.contains(e.target as any)) {\n (value as (e: Event) => void)(e);\n }\n };\n\n // Capture is set to avoid intermediate elements cancelling this event before it gets to the window.\n const options = { capture: true };\n\n window.addEventListener(\"click\", listener, options);\n this.unsubscribers.push(() => {\n window.removeEventListener(\"click\", listener, options);\n });\n } else if (key.startsWith(\"prop:\")) {\n // Keys starting with `prop:` are set as props.\n\n const _key = key.substring(5);\n this.attachProp(value, (current) => {\n (element as any)[_key] = current;\n });\n } else if (key.startsWith(\"on:\")) {\n // Keys starting with `on:` are treated as event listeners.\n\n const _key = key.substring(3);\n let _prev: EventListener | undefined;\n if (isFunction(value)) {\n element.addEventListener(_key, value as EventListener);\n this.unsubscribers.push(() => {\n element.removeEventListener(_key, value as EventListener);\n });\n } else {\n this.attachProp(value as MaybeSignal<EventListener>, (current) => {\n if (!current && _prev) {\n element.removeEventListener(_key, _prev);\n } else if (current != null) {\n if (_prev && _prev !== current) {\n element.removeEventListener(_key, _prev);\n }\n element.addEventListener(_key, current);\n }\n _prev = current;\n });\n }\n } else if (key.startsWith(\"attr:\")) {\n // Keys starting with `attr:` are set as attributes.\n\n const _key = key.substring(5).toLowerCase();\n this.attachProp(value, (current) => {\n if (current != null) {\n element.setAttribute(_key, String(current));\n } else {\n element.removeAttribute(_key);\n }\n });\n } else if (isFunction(value) && isCamelCaseEventName(key)) {\n // camelCase event names are applied with addEventListener.\n\n const eventName = key.slice(2).toLowerCase();\n\n const listener: (e: Event) => void = value as (e: Event) => void;\n\n element.addEventListener(eventName, listener);\n this.unsubscribers.push(() => {\n element.removeEventListener(eventName, listener);\n });\n } else if (key.startsWith(\"on\") && isFunction(value) && knownEventNames.includes(key.substring(2))) {\n // Event handler properties (element.onsomething = function () {})\n\n (element as any)[key] = value;\n this.unsubscribers.push(() => {\n (element as any)[key] = undefined;\n });\n } else if (key.includes(\"-\")) {\n // Names with dashes in them are not valid prop names, so they are treated as attributes.\n\n this.attachProp(value, (current) => {\n if (current == null) {\n element.removeAttribute(key);\n } else {\n element.setAttribute(key, String(current));\n }\n });\n } else if (this.context.getState(IS_SVG, false)) {\n // SVG gets everything set as an attribute.\n\n // TODO: This isn't exactly right. SVGElement supports props as well.\n this.attachProp(value, (current) => {\n if (current != null) {\n element.setAttribute(key, String(props[key]));\n } else {\n element.removeAttribute(key);\n }\n });\n } else {\n // Special handling for other props on a case by case basis.\n\n switch (key) {\n case \"contentEditable\":\n case \"value\":\n this.attachProp(value, (current) => {\n (element as any)[key] = String(current);\n });\n break;\n\n case \"for\":\n this.attachProp(value, (current) => {\n (element as any).htmlFor = current;\n });\n break;\n\n case \"innerHTML\":\n this.attachProp(value, (current) => {\n (element as any).innerHTML = current;\n });\n break;\n\n case \"title\":\n this.attachProp(value, (current) => {\n if (current == null) {\n (element as any).removeAttribute(key);\n } else {\n (element as any).setAttribute(key, String(current));\n }\n });\n\n case \"checked\":\n this.attachProp(value, (current) => {\n (element as any).checked = current;\n\n // Set attribute also or styles don't take effect.\n if (current) {\n element.setAttribute(\"checked\", \"\");\n } else {\n element.removeAttribute(\"checked\");\n }\n });\n break;\n\n case \"dataset\":\n let applied: Record<string, string> = {};\n this.attachProp(value, (_next) => {\n if (isObject(_next)) {\n // Convert keys to camelCase and make sure values are strings.\n const next: Record<string, string> = {};\n for (const _key in _next) {\n next[toCamelCase(_key)] = String(_next[_key]);\n }\n\n for (const key in applied) {\n // Key removed.\n if (Object.hasOwn(applied, key) && !Object.hasOwn(next, key)) {\n delete element.dataset[key];\n delete applied[key];\n }\n }\n\n for (const key in next) {\n // Value changed (or not set).\n if (applied[key] !== next[key]) {\n element.dataset[key] = next[key];\n applied[key] = next[key];\n }\n }\n } else {\n for (const key in applied) {\n delete element.dataset[key];\n }\n }\n });\n break;\n\n case \"autocomplete\":\n case \"autocapitalize\":\n this.attachProp(value, (current) => {\n if (typeof current === \"string\") {\n (element as any)[key] = current;\n } else if (current) {\n (element as any)[key] = \"on\";\n } else {\n (element as any)[key] = \"off\";\n }\n });\n break;\n\n default: {\n // Fall back to setting as a property.\n this.attachProp(value, (current) => {\n (element as any)[key] = current;\n });\n break;\n }\n }\n }\n }\n }\n\n private applyStyles(element: HTMLElement | SVGElement, styles: unknown, unsubscribers: UnsubscribeFn[]) {\n const propUnsubscribers: UnsubscribeFn[] = [];\n\n if (isFunction(styles)) {\n let unapply: () => void;\n\n const unsubscribe = effect(() => {\n if (isFunction(unapply)) {\n unapply();\n }\n element.style.cssText = \"\";\n unapply = this.applyStyles(element, get(styles), unsubscribers);\n });\n\n unsubscribers.push(unsubscribe);\n propUnsubscribers.push(unsubscribe);\n } else {\n const mapped = getStyleMap(styles);\n\n for (const name in mapped) {\n const { value, priority } = mapped[name];\n\n if (isFunction(value)) {\n const unsubscribe = effect(() => {\n if (get(value)) {\n element.style.setProperty(name, String(get(value)), priority);\n } else {\n element.style.removeProperty(name);\n }\n });\n\n unsubscribers.push(unsubscribe);\n propUnsubscribers.push(unsubscribe);\n } else if (value != undefined) {\n element.style.setProperty(name, String(value));\n }\n }\n }\n\n return function unapply() {\n for (const unsubscribe of propUnsubscribers) {\n unsubscribe();\n unsubscribers.splice(unsubscribers.indexOf(unsubscribe), 1);\n }\n };\n }\n\n private applyClasses(element: HTMLElement | SVGElement, classes: unknown, unsubscribers: UnsubscribeFn[]) {\n const classUnsubscribers: UnsubscribeFn[] = [];\n\n if (isFunction(classes)) {\n let unapply: () => void;\n\n const unsubscribe = effect(() => {\n if (isFunction(unapply)) {\n unapply();\n }\n element.removeAttribute(\"class\");\n unapply = this.applyClasses(element, get(classes), unsubscribers);\n });\n\n unsubscribers.push(unsubscribe);\n classUnsubscribers.push(unsubscribe);\n } else {\n const mapped = getClassMap(classes);\n\n for (const name in mapped) {\n const value = mapped[name];\n\n if (isFunction(value)) {\n const unsubscribe = effect(() => {\n if (get(value)) {\n element.classList.add(name);\n } else {\n element.classList.remove(name);\n }\n });\n\n unsubscribers.push(unsubscribe);\n classUnsubscribers.push(unsubscribe);\n } else if (value) {\n element.classList.add(name);\n }\n }\n }\n\n return function unapply() {\n for (const unsubscribe of classUnsubscribers) {\n unsubscribe();\n unsubscribers.splice(unsubscribers.indexOf(unsubscribe), 1);\n }\n };\n }\n}\n\n/**\n * Parse classes into a single object. Classes can be passed as a string, an object with class keys can boolean values, or an array with a mix of both.\n */\nfunction getClassMap(classes: unknown) {\n let mapped: Record<string, boolean> = {};\n\n if (isString(classes)) {\n // Support multiple classes in one string like HTML.\n const names = classes.split(\" \");\n for (const name of names) {\n mapped[name] = true;\n }\n } else if (isObject(classes)) {\n Object.assign(mapped, classes);\n } else if (Array.isArray(classes)) {\n Array.from(classes)\n .filter(Boolean)\n .forEach((item) => {\n Object.assign(mapped, getClassMap(item));\n });\n }\n\n // Delete undefined keys. These are usually the result of a class that is not specified in the stylesheet and would have no effect on appearance.\n delete mapped[\"undefined\"];\n\n return mapped;\n}\n\n/**\n * Parse styles into a single object.\n */\nfunction getStyleMap(styles: unknown) {\n let mapped: Record<string, { value: unknown; priority?: string }> = {};\n\n if (isString(styles)) {\n const lines = styles.split(\";\").filter((line) => line.trim() !== \"\");\n for (const line of lines) {\n const [key, _value] = line.split(\":\");\n const entry: { value: unknown; priority?: string } = {\n value: _value,\n };\n if (_value.includes(\"!important\")) {\n entry.priority = \"important\";\n entry.value = _value.replace(\"!important\", \"\").trim();\n } else {\n entry.value = _value.trim();\n }\n mapped[camelToKebab(key.trim())] = entry;\n }\n }\n if (isObject(styles)) {\n for (const key in styles) {\n if (key.startsWith(\"--\")) {\n // Pass through variable names without processing.\n mapped[key] = { value: styles[key] };\n } else {\n mapped[camelToKebab(key)] = { value: styles[key] };\n }\n }\n } else if (Array.isArray(styles)) {\n Array.from(styles)\n .filter((item) => item != null)\n .forEach((item) => {\n Object.assign(mapped, getStyleMap(item));\n });\n }\n\n return mapped;\n}\n\nfunction getLoggerName(this: HTML) {\n if (this.root == null) return this.tag;\n let name = this.root.tagName.toLowerCase();\n if (this.root.id) {\n name += `#${this.root.id}`;\n }\n if (this.root.classList.length > 0) {\n for (const className of this.root.classList.values()) {\n name += `.${className}`;\n }\n }\n return name;\n}\n\n/**\n * Converts a camelCase string to kebab-case.\n */\nfunction camelToKebab(value: string): string {\n return value.replace(/[A-Z]+(?![a-z])|[A-Z]/g, ($, ofs) => (ofs ? \"-\" : \"\") + $.toLowerCase());\n}\n\n// A list of all known event names. These will be handled as event listeners.\nconst knownEventNames = [\n // Element\n \"animationcancel\",\n \"animationend\",\n \"animationiteration\",\n \"animationstart\",\n \"auxclick\",\n \"beforeinput\",\n \"beforematch\",\n \"beforexrselect\",\n \"blur\",\n \"click\",\n \"compositionend\",\n \"compositionstart\",\n \"compositionupdate\",\n \"contentvisibilityautostatechange\",\n \"contextmenu\",\n \"copy\",\n \"cut\",\n \"dblclick\",\n \"focus\",\n \"focusin\",\n \"focusout\",\n \"fullscreenchange\",\n \"fullscreenerror\",\n \"gotpointercapture\",\n \"input\",\n \"keydown\",\n \"keyup\",\n \"lostpointercapture\",\n \"mousedown\",\n \"mouseenter\",\n \"mouseleave\",\n \"mousemove\",\n \"mouseout\",\n \"mouseover\",\n \"mouseup\",\n \"paste\",\n \"pointercancel\",\n \"pointerdown\",\n \"pointerenter\",\n \"pointerleave\",\n \"pointermove\",\n \"pointerout\",\n \"pointerover\",\n \"pointerrawupdate\",\n \"pointerup\",\n \"scroll\",\n \"scrollend\",\n \"scrollsnapchange\",\n \"scrollsnapchanging\",\n \"securitypolicyviolation\",\n \"touchcancel\",\n \"touchend\",\n \"touchmove\",\n \"touchstart\",\n \"transitioncancel\",\n \"transitionend\",\n \"transitionrun\",\n \"transitionstart\",\n \"webkitmouseforcechanged\",\n \"webkitmouseforcedown\",\n \"webkitmouseforceup\",\n \"webkimouseforcewillbegin\",\n \"wheel\",\n\n // HTMLElement\n \"beforetoggle\",\n \"change\",\n \"command\",\n \"drag\",\n \"dragend\",\n \"dragenter\",\n \"dragleave\",\n \"dragover\",\n \"dragstart\",\n \"drop\",\n \"error\",\n \"load\",\n \"toggle\",\n\n // HTMLInputElement\n \"cancel\",\n \"invalid\",\n \"search\",\n \"select\",\n \"selectionchange\",\n\n // HTMLFormElement\n \"formdata\",\n \"reset\",\n \"submit\",\n];\n","import type { Renderable } from \"../../types.js\";\nimport { Context } from \"../context.js\";\nimport { render, type MarkupNode } from \"../markup.js\";\nimport { TYPE, MARKUP_NODE } from \"../symbols.js\";\n\n/**\n * Renders content into a specified parent node.\n */\nexport class Portal implements MarkupNode {\n [TYPE] = MARKUP_NODE;\n\n private context;\n private content;\n private parent;\n\n private element?: MarkupNode;\n\n constructor(context: Context, content: Renderable, parent: Element) {\n this.context = context;\n this.content = content;\n this.parent = parent;\n }\n\n isMounted() {\n if (!this.element) {\n return false;\n }\n return this.element.isMounted();\n }\n\n mount(_parent: Element, _after?: Node) {\n this.element = render(this.content, this.context);\n this.element.mount(this.parent);\n }\n\n unmount(skipDOM = false) {\n if (this.element?.isMounted()) {\n // Portals MUST unmount DOM nodes because they won't be removed by parents unmounting.\n this.element.unmount(false);\n }\n }\n\n move(_parent: Element, _after?: Node) {}\n}\n","import type { Renderable } from \"../../types.js\";\nimport { deepEqual } from \"../../utils.js\";\nimport type { Context } from \"../context.js\";\nimport type { MarkupNode } from \"../markup.js\";\nimport { $, batch, effect, untracked, type Signal, type Source, type UnsubscribeFn } from \"../signals.js\";\nimport { TYPE, MARKUP_NODE } from \"../symbols.js\";\nimport { ViewInstance } from \"./view.js\";\n\n// ----- Types ----- //\n\nexport type Key = string | number | symbol;\n\nexport type KeyFn<T> = (item: T, index: number) => Key;\nexport type RenderFn<T> = (item: Signal<T>, index: Signal<number>, ctx: Context) => Renderable;\n\ntype ConnectedItem<T> = {\n key: any;\n $item: Source<T>;\n $index: Source<number>;\n node: MarkupNode;\n};\n\n// ----- Code ----- //\n\nexport class Repeat<T> implements MarkupNode {\n [TYPE] = MARKUP_NODE;\n\n root = document.createTextNode(\"\");\n\n private context;\n\n private items: Signal<T[]>;\n private key: KeyFn<T>;\n private render: RenderFn<T>;\n\n private unsubscribe: UnsubscribeFn | null = null;\n private connectedItems: Map<Key, ConnectedItem<T>> = new Map();\n\n constructor(context: Context, items: Signal<T[]>, key: KeyFn<T>, render: RenderFn<T>) {\n this.context = context;\n\n this.items = items;\n this.key = key;\n this.render = render;\n }\n\n isMounted() {\n return this.root.parentElement != null;\n }\n\n mount(parent: Element, after?: Node) {\n if (!this.isMounted()) {\n parent.insertBefore(this.root, after?.nextSibling ?? null);\n\n this.unsubscribe = effect(() => {\n let value = this.items();\n\n if (value == null) {\n value = [];\n this.context.warn(\"repeat() received empty value for items\", value);\n }\n\n untracked(() => {\n this._update(Array.from(value));\n });\n });\n }\n }\n\n unmount(skipDOM = false) {\n if (this.unsubscribe) {\n this.unsubscribe();\n this.unsubscribe = null;\n }\n\n if (!skipDOM && this.isMounted()) {\n this.root.parentNode?.removeChild(this.root);\n }\n\n this._cleanup(skipDOM);\n }\n\n move(parent: Element, after?: Node) {\n // TODO: Implement move\n return this.mount(parent, after);\n }\n\n private _cleanup(parentIsUnmounting: boolean) {\n for (const item of this.connectedItems.values()) {\n item.node.unmount(parentIsUnmounting);\n }\n this.connectedItems.clear();\n }\n\n private _update(value: T[]) {\n if (value.length === 0 || !this.isMounted()) {\n return this._cleanup(false);\n }\n\n type UpdateItem = { key: string | number | symbol; value: T; index: number };\n\n const potentialItems = new Map<string | number | symbol, UpdateItem>();\n let index = 0;\n\n for (const item of value) {\n const key = this.key(item, index);\n potentialItems.set(key, {\n key,\n value: item,\n index: index++,\n });\n }\n\n const newItems: ConnectedItem<T>[] = [];\n\n // Remove views for items that no longer exist in the new list.\n for (const connected of this.connectedItems.values()) {\n if (!potentialItems.has(connected.key) && connected.node.isMounted()) {\n connected.node.unmount(false);\n }\n }\n\n batch(() => {\n // Add new views and update state for existing ones.\n for (const potential of potentialItems.values()) {\n const connected = this.connectedItems.get(potential.key);\n\n if (connected && connected.node.isMounted()) {\n connected.$item(potential.value);\n connected.$index(potential.index);\n\n newItems[potential.index] = connected;\n } else {\n // deepEqual avoids running update code again if the data is equivalent. In list updates this happens a lot.\n const $item = $(potential.value, { equals: deepEqual });\n const $index = $(potential.index);\n\n newItems[potential.index] = {\n key: potential.key,\n $item,\n $index,\n node: new ViewInstance(this.context, RepeatItemView, {\n $item: () => $item(),\n $index: () => $index(),\n render: this.render,\n }),\n };\n }\n }\n });\n\n // Reconnect to ensure order. Lifecycle hooks won't be run again if the view is already connected.\n // TODO: Use a smarter inline reordering method. This causes scrollbars to jump.\n for (let i = 0; i < newItems.length; i++) {\n const item = newItems[i];\n const previous = newItems[i - 1]?.node.root ?? this.root;\n\n const connected = this.connectedItems.get(item.key);\n if (connected && connected.node.isMounted()) {\n item.node.move(this.root.parentElement!, previous);\n } else {\n item.node.mount(this.root.parentElement!, previous);\n }\n }\n\n this.connectedItems.clear();\n for (const item of newItems) {\n this.connectedItems.set(item.key, item);\n }\n\n // Move marker node to end.\n const lastItem = newItems.at(-1)?.node.root ?? this.root;\n this.root.parentNode?.insertBefore(this.root, lastItem.nextSibling);\n }\n}\n\ninterface ListItemProps {\n $item: Signal<any>;\n $index: Signal<number>;\n render: (item: Signal<any>, index: Signal<number>, context: Context) => Renderable;\n}\nconst contextName = \"dolla.RepeatItemView\";\nfunction RepeatItemView(props: ListItemProps, context: Context) {\n context.setName(contextName);\n return props.render.call(context, props.$item, props.$index, context);\n}\n","import { isFunction, isString } from \"../typeChecking.js\";\nimport type { IntrinsicElements, Renderable, View } from \"../types.js\";\nimport { Context } from \"./context.js\";\nimport { DOMNode } from \"./nodes/dom.js\";\nimport { Dynamic } from \"./nodes/dynamic.js\";\nimport { HTML } from \"./nodes/html.js\";\nimport { Portal } from \"./nodes/portal.js\";\nimport { KeyFn, RenderFn, Repeat } from \"./nodes/repeat.js\";\nimport { ViewInstance } from \"./nodes/view.js\";\nimport { $, get, type MaybeSignal, type Signal } from \"./signals.js\";\nimport { TYPE, MARKUP_NODE } from \"./symbols.js\";\n\n/*===========================*\\\n|| Markup ||\n\\*===========================*/\n\n/**\n * Markup is a set of element metadata that hasn't been constructed into a MarkupElement yet.\n */\nexport class Markup<P = any> {\n /**\n * In the case of a view, type will be the View function itself. It can also hold an identifier for special nodes like \"$cond\", \"$repeat\", etc.\n * DOM nodes can be created by name, such as HTML elements like \"div\", \"ul\" or \"span\", SVG elements like \"\"\n */\n type;\n\n /**\n * Data that will be passed to a new MarkupNode instance when it is constructed.\n * Includes a `children` prop if children were passed.\n */\n props;\n\n constructor(type: string | View<P>, props?: P) {\n this.type = type;\n this.props = props;\n }\n}\n\n/**\n * A mountable node that has been constructed from Markup metadata.\n */\nexport interface MarkupNode {\n /**\n *\n */\n readonly root?: Node;\n\n /**\n * Returns true when this MarkupNode is mounted.\n */\n isMounted(): boolean;\n\n /**\n * Mount this MarkupNode to a `parent` element.\n * If passed, this MarkupNode will be mounted as the next sibling of `after`.\n */\n mount(parent: Element, after?: Node): void;\n\n /**\n * Unmount this MarkupNode from its parent element.\n *\n * The `skipDOM` option can be passed as an optimization when unmounting a parent node.\n * A value of `true` indicates that no DOM operations need to happen because the parent is already being unmounted.\n *\n * @param skipDOM - No DOM updates will be performed when true. Lifecycle methods will be called regardless.\n */\n unmount(skipDOM?: boolean): void;\n\n /**\n * Moves a node without unmounting and remounting (if the browser supports Element.moveBefore).\n */\n move(parent: Element, after?: Node): void;\n}\n\nexport function isMarkupNode(value: any): value is MarkupNode {\n return value?.[TYPE] === MARKUP_NODE;\n}\n\nexport enum MarkupType {\n DOM = \"$dom\",\n Dynamic = \"$dynamic\",\n Portal = \"$portal\",\n Repeat = \"$repeat\",\n}\n\nexport interface MarkupNodeProps {\n [MarkupType.DOM]: {\n value: Node;\n };\n [MarkupType.Dynamic]: {\n source: Signal<any>;\n };\n [MarkupType.Portal]: {\n content: Renderable;\n parent: Element;\n };\n [MarkupType.Repeat]: {\n items: Signal<any[]>;\n key: KeyFn<any>;\n render: RenderFn<any>;\n };\n}\n\nexport interface MarkupCustomElementProps {\n /**\n * Custom element tagName pattern (must include a hyphen).\n */\n [tag: `${string}-${string}`]: Record<string, any>;\n}\n\n/**\n * Creates a Markup element that defines an HTML element.\n */\nexport function m<T extends keyof IntrinsicElements>(\n tag: T,\n attrs: IntrinsicElements[T] & { children?: Renderable },\n): Markup;\n\n/**\n * Creates a Markup element that defines an HTML custom element.\n */\nexport function m<T extends keyof MarkupCustomElementProps>(type: T, props: MarkupCustomElementProps[T]): Markup;\n\n/**\n * Creates a Markup element that defines a MarkupNode.\n */\nexport function m<T extends keyof MarkupNodeProps>(type: T, props: MarkupNodeProps[T]): Markup;\n\n/**\n * Creates a Markup element that defines a view.\n */\nexport function m<P extends {}>(type: View<P>, props?: P): Markup;\n\n/**\n * Creates a Markup element that defines a view.\n */\nexport function m<P>(type: View<P>, props: P): Markup;\n\nexport function m(type: string | View<any>, props?: any) {\n return new Markup(type, props ?? {});\n}\n\n/*===========================*\\\n|| View Helpers ||\n\\*===========================*/\n\n/**\n * If `condition` is truthy, displays `thenContent`, otherwise `elseContent`.\n */\nexport function when(condition: MaybeSignal<any>, thenContent?: Renderable, elseContent?: Renderable): Markup {\n return m(MarkupType.Dynamic, {\n source: $<Renderable>(() => {\n const value = get(condition);\n\n if (value && thenContent) {\n return thenContent;\n } else if (!value && elseContent) {\n return elseContent;\n }\n return null;\n }),\n });\n}\n\n/**\n * Inverted `when`. If `condition` is falsy, displays `thenContent`, otherwise `elseContent`.\n */\nexport function unless(condition: MaybeSignal<any>, thenContent?: Renderable, elseContent?: Renderable): Markup {\n return when(condition, elseContent, thenContent);\n}\n\n/**\n * Calls `render` for each item in `items`. Dynamically adds and removes views as items change.\n * The result of `key` is used to compare items and decide if item was added, removed or updated.\n */\nexport function repeat<T>(items: MaybeSignal<T[]>, key: KeyFn<T>, render: RenderFn<T>): Markup {\n return m(MarkupType.Repeat, { items: () => get(items), key, render });\n}\n\n/**\n * Renders `content` into a `parent` node anywhere in the page, rather than its usual position in the view.\n */\nexport function portal(parent: Element, content: Renderable): Markup {\n return m(MarkupType.Portal, { parent, content });\n}\n\n/*===========================*\\\n|| Render ||\n\\*===========================*/\n\nexport function render(content: Renderable, context = new Context(\"$\")): MarkupNode {\n const nodes = toMarkupNodes(context, [content]);\n if (nodes.length === 1) {\n return nodes[0];\n }\n return new Dynamic(context, () => nodes);\n}\n\n/**\n * Convert basically anything into a set of MarkupElements.\n */\nexport function toMarkupNodes(context: Context, ...content: any[]): MarkupNode[] {\n const items = content.flat(Infinity);\n const elements: MarkupNode[] = [];\n\n for (const item of items) {\n if (item === null || item === undefined || item === false) {\n continue;\n }\n\n if (item instanceof Node) {\n elements.push(new DOMNode(item));\n continue;\n }\n\n if (item instanceof Markup) {\n if (isFunction(item.type)) {\n elements.push(new ViewInstance(context, item.type as View<any>, item.props));\n continue;\n } else if (isString(item.type)) {\n switch (item.type) {\n case MarkupType.DOM: {\n const attrs = item.props! as MarkupNodeProps[MarkupType.DOM];\n elements.push(new DOMNode(attrs.value));\n continue;\n }\n case MarkupType.Dynamic: {\n const attrs = item.props! as MarkupNodeProps[MarkupType.Dynamic];\n elements.push(new Dynamic(context, attrs.source));\n continue;\n }\n case MarkupType.Portal: {\n const attrs = item.props! as MarkupNodeProps[MarkupType.Portal];\n elements.push(new Portal(context, attrs.content, attrs.parent));\n continue;\n }\n case MarkupType.Repeat: {\n const attrs = item.props! as MarkupNodeProps[MarkupType.Repeat];\n elements.push(new Repeat(context, attrs.items, attrs.key, attrs.render));\n continue;\n }\n default:\n // Handle type as an HTML tag.\n elements.push(new HTML(context, item.type, item.props));\n continue;\n }\n } else {\n throw new TypeError(`Expected a string or view function. Got: ${item.type}`);\n }\n }\n\n if (isMarkupNode(item)) {\n elements.push(item);\n continue;\n }\n\n if (isFunction(item)) {\n elements.push(new Dynamic(context, item));\n continue;\n }\n\n // fallback to displaying value as text\n elements.push(new DOMNode(document.createTextNode(String(item))));\n }\n\n return elements;\n}\n"],"names":["LifecycleEvent","LifecycleEvent2","NAME","LIFECYCLE","PARENT","STORES","STATE","ContextLifecycle","context","__publicField","event","listener","listeners","_e","_d","_c","_b","_a","_Context","name","options","__privateAdd","_name","__privateSet","untracked","logger","createLogger","get","__privateGet","descriptors","key","state","parent","store","StoreError","getUniqueId","result","error","isFunction","callback","fn","unsubscribe","effect","disposed","defaultValue","value","entries","typeOf","Context","createContext","TYPE","MARKUP_NODE","DOMNode","node","after","skipDOM","Dynamic","$slot","content","toArray","i","element","nodes","toMarkupNodes","previous","lastChildNextSibling","VIEW","ViewInstance","view","props","wasMounted","render","isCamelCaseEventName","IS_SVG","privateProps","HTML","tag","getLoggerName","getEnv","classes","omit","child","e","_key","current","_prev","eventName","knownEventNames","applied","_next","isObject","next","toCamelCase","styles","unsubscribers","propUnsubscribers","unapply","mapped","getStyleMap","priority","classUnsubscribers","getClassMap","isString","names","item","lines","line","_value","entry","camelToKebab","className","$","ofs","Portal","_parent","_after","Repeat","items","parentIsUnmounting","potentialItems","index","newItems","connected","batch","potential","$item","deepEqual","$index","RepeatItemView","lastItem","contextName","Markup","type","isMarkupNode","MarkupType","m","when","condition","thenContent","elseContent","unless","repeat","portal","elements","attrs"],"mappings":"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"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"router-BpuJQ6OA.js","sources":["../src/router/router.utils.ts","../src/router/router.ts"],"sourcesContent":["import { assertString, assertArrayOf, isFunction } from \"../typeChecking.js\";\n\nexport type RouteMatch<T = Record<string, any>> = {\n /**\n * The path string that triggered this match.\n */\n path: string;\n\n /**\n * The pattern satisfied by `path`.\n */\n pattern: string;\n\n /**\n * Named params as parsed from `path`.\n */\n params: Record<string, string>;\n\n /**\n * Query params as parsed from `path`.\n */\n query: Record<string, string>;\n\n /**\n * Metadata registered to this route.\n */\n meta: T;\n};\n\nexport enum FragTypes {\n Literal = 1,\n Param = 2,\n Wildcard = 3,\n NumericParam = 4,\n}\n\nexport type RouteFragment = {\n name: string;\n type: FragTypes;\n value: string | number | null;\n};\n\nexport type ParsedRoute<T> = {\n pattern: string;\n fragments: RouteFragment[];\n meta: T;\n};\n\nexport type RouteMatchOptions<T> = {\n willMatch?: (route: ParsedRoute<T>) => boolean;\n};\n\n/**\n * Separates a URL path into multiple fragments.\n *\n * @param path - A path string (e.g. `\"/api/users/5\"`)\n * @returns an array of fragments (e.g. `[\"api\", \"users\", \"5\"]`)\n */\nexport function splitPath(path: string): string[] {\n assertString(path, \"Expected `path` to be a string. Got type: %t, value: %v\");\n\n return path\n .split(\"/\")\n .map((f) => f.trim())\n .filter((f) => f !== \"\");\n}\n\n/**\n * Joins multiple URL path fragments into a single string.\n *\n * @param parts - One or more URL fragments (e.g. `[\"api\", \"users\", 5]`)\n * @returns a joined path (e.g. `\"api/users/5\"`)\n */\nexport function joinPath(parts: { toString(): string }[]): string {\n assertArrayOf(\n (part) => isFunction(part?.toString),\n parts,\n \"Expected `parts` to be an array of objects with a .toString() method. Got type: %t, value: %v\",\n );\n\n parts = parts.filter((x) => x).flatMap(String);\n\n let joined = parts.shift()?.toString();\n\n if (joined) {\n for (const part of parts.map((p) => p.toString())) {\n if (part.startsWith(\".\")) {\n // Resolve relative path against joined\n joined = resolvePath(joined, part);\n } else if (joined[joined.length - 1] !== \"/\") {\n if (part[0] !== \"/\") {\n joined += \"/\" + part;\n } else {\n joined += part;\n }\n } else {\n if (part[0] === \"/\") {\n joined += part.slice(1);\n } else {\n joined += part;\n }\n }\n }\n\n // Remove trailing slash (unless path is just '/')\n if (joined && joined !== \"/\" && joined.endsWith(\"/\")) {\n joined = joined.slice(0, joined.length - 1);\n }\n }\n\n return joined ?? \"\";\n}\n\nexport function resolvePath(base: string, part: string | null) {\n assertString(base, \"Expected `base` to be a string. Got type: %t, value: %v\");\n\n if (part == null) {\n part = base;\n base = \"\";\n }\n\n if (part.startsWith(\"/\")) {\n return part;\n }\n\n let resolved = base;\n\n while (true) {\n if (part.startsWith(\"..\")) {\n for (let i = resolved.length; i > 0; --i) {\n if (resolved[i] === \"/\" || i === 0) {\n resolved = resolved.slice(0, i);\n part = part.replace(/^\\.\\.\\/?/, \"\");\n break;\n }\n }\n } else if (part.startsWith(\".\")) {\n part = part.replace(/^\\.\\/?/, \"\");\n } else {\n break;\n }\n }\n\n return joinPath([resolved, part]);\n}\n\nexport function parseQueryParams(query: string): Record<string, string> {\n if (!query) return {};\n\n if (query.startsWith(\"?\")) {\n query = query.slice(1);\n }\n\n const entries = query\n .split(\"&\")\n .filter((x) => x.trim() !== \"\")\n .map((entry) =>\n entry\n .split(\"=\")\n .map((x) => x.trim())\n .slice(0, 2),\n );\n\n return Object.fromEntries(entries);\n}\n\n/**\n * Returns the nearest match, or undefined if the path matches no route.\n *\n * @param url - Path to match against routes.\n * @param options - Options to customize how matching operates.\n */\nexport function matchRoutes<T>(\n routes: ParsedRoute<T>[],\n url: string,\n options: RouteMatchOptions<T> = {},\n): RouteMatch<T> | undefined {\n const [path, query] = url.split(\"?\");\n const parts = splitPath(path);\n\n routes: for (const route of routes) {\n const { fragments } = route;\n const hasWildcard = fragments[fragments.length - 1]?.type === FragTypes.Wildcard;\n\n if (!hasWildcard && fragments.length !== parts.length) {\n continue routes;\n }\n\n if (options.willMatch && !options.willMatch(route)) {\n continue routes;\n }\n\n const matched: RouteFragment[] = [];\n\n fragments: for (let i = 0; i < fragments.length; i++) {\n const part = parts[i];\n const frag = fragments[i];\n\n if (part == null && frag.type !== FragTypes.Wildcard) {\n continue routes;\n }\n\n switch (frag.type) {\n case FragTypes.Literal:\n if (frag.name.toLowerCase() === part.toLowerCase()) {\n matched.push(frag);\n break;\n } else {\n continue routes;\n }\n case FragTypes.Param:\n matched.push({ ...frag, value: part });\n break;\n case FragTypes.Wildcard:\n matched.push({ ...frag, value: parts.slice(i).join(\"/\") });\n break fragments;\n case FragTypes.NumericParam:\n if (!isNaN(Number(part))) {\n matched.push({ ...frag, value: part });\n break;\n } else {\n continue routes;\n }\n default:\n throw new Error(`Unknown fragment type: ${frag.type}`);\n }\n }\n\n const params: Record<string, string> = {};\n\n for (const frag of matched) {\n if (frag.type === FragTypes.Param) {\n params[frag.name] = decodeURIComponent(frag.value as string);\n }\n\n if (frag.type === FragTypes.NumericParam) {\n params[frag.name] = String(frag.value);\n }\n\n if (frag.type === FragTypes.Wildcard) {\n params.wildcard = \"/\" + decodeURIComponent(frag.value as string);\n }\n }\n\n return {\n path: \"/\" + matched.map((f) => f.value).join(\"/\"),\n pattern:\n \"/\" +\n fragments\n .map((f) => {\n if (f.type === FragTypes.Param) {\n return `{${f.name}}`;\n }\n\n if (f.type === FragTypes.NumericParam) {\n return `{#${f.name}}`;\n }\n\n return f.name;\n })\n .join(\"/\"),\n params,\n query: parseQueryParams(query),\n meta: route.meta,\n };\n }\n}\n\n/**\n * Sort routes descending by specificity. Guarantees that the most specific route matches first\n * no matter the order in which they were added.\n *\n * Routes without named params and routes with more fragments are weighted more heavily.\n */\nexport function sortRoutes<T>(routes: ParsedRoute<T>[]): ParsedRoute<T>[] {\n const withoutParams = [];\n const withNumericParams = [];\n const withParams = [];\n const wildcard = [];\n\n for (const route of routes) {\n const { fragments } = route;\n\n if (fragments.some((f) => f.type === FragTypes.Wildcard)) {\n wildcard.push(route);\n } else if (fragments.some((f) => f.type === FragTypes.NumericParam)) {\n withNumericParams.push(route);\n } else if (fragments.some((f) => f.type === FragTypes.Param)) {\n withParams.push(route);\n } else {\n withoutParams.push(route);\n }\n }\n\n const bySizeDesc = (a: ParsedRoute<T>, b: ParsedRoute<T>) => {\n if (a.fragments.length > b.fragments.length) {\n return -1;\n } else {\n return 1;\n }\n };\n\n withoutParams.sort(bySizeDesc);\n withNumericParams.sort(bySizeDesc);\n withParams.sort(bySizeDesc);\n wildcard.sort(bySizeDesc);\n\n return [...withoutParams, ...withNumericParams, ...withParams, ...wildcard];\n}\n\n/**\n * Converts a route pattern into a set of matchable fragments.\n *\n * @param route - A route string (e.g. \"/api/users/{id}\")\n */\nexport function patternToFragments(pattern: string): RouteFragment[] {\n const parts = splitPath(pattern);\n const fragments = [];\n\n for (let i = 0; i < parts.length; i++) {\n const part = parts[i];\n\n if (part === \"*\") {\n if (i !== parts.length - 1) {\n throw new Error(`Wildcard must be at the end of a pattern. Received: ${pattern}`);\n }\n fragments.push({\n type: FragTypes.Wildcard,\n name: \"*\",\n value: null,\n });\n } else if (part.at(0) === \"{\" && part.at(-1) === \"}\") {\n fragments.push({\n type: part[1] === \"#\" ? FragTypes.NumericParam : FragTypes.Param,\n name: part[1] === \"#\" ? part.slice(2, -1) : part.slice(1, -1),\n value: null,\n });\n } else {\n fragments.push({\n type: FragTypes.Literal,\n name: part,\n value: part,\n });\n }\n }\n\n return fragments;\n}\n\nconst safeExternalLink = /(noopener|noreferrer) (noopener|noreferrer)/;\nconst protocolLink = /^[\\w-_]+:/;\n\n/**\n * Intercepts links within the root node.\n *\n * This is adapted from https://github.com/choojs/nanohref/blob/master/index.js\n *\n * @param root - Element under which to intercept link clicks\n * @param callback - Function to call when a click event is intercepted\n * @param _window - (optional) Override for global window object\n */\nexport function catchLinks(root: Element, callback: (anchor: HTMLAnchorElement) => void, _window = window) {\n function traverse(node: HTMLElement | null): HTMLAnchorElement | null {\n if (!node || node === root) {\n return null;\n }\n\n if (node.localName !== \"a\" || (node as any).href === undefined) {\n return traverse(node.parentNode as HTMLElement | null);\n }\n\n return node as HTMLAnchorElement;\n }\n\n function handler(e: MouseEvent) {\n if ((e.button && e.button !== 0) || e.ctrlKey || e.metaKey || e.altKey || e.shiftKey || e.defaultPrevented) {\n return;\n }\n\n const anchor = traverse(e.target as HTMLElement);\n\n if (!anchor) {\n return;\n }\n\n if (\n _window.location.protocol !== anchor.protocol ||\n _window.location.hostname !== anchor.hostname ||\n _window.location.port !== anchor.port ||\n anchor.hasAttribute(\"data-router-ignore\") ||\n anchor.hasAttribute(\"download\") ||\n (anchor.getAttribute(\"target\") === \"_blank\" && safeExternalLink.test(anchor.getAttribute(\"rel\")!)) ||\n protocolLink.test(anchor.getAttribute(\"href\")!)\n ) {\n return;\n }\n\n e.preventDefault();\n callback(anchor);\n }\n\n root.addEventListener(\"click\", handler as any);\n\n return function cancel() {\n root.removeEventListener(\"click\", handler as any);\n };\n}\n\n/**\n * Replace route pattern param placeholders with real matched values.\n */\nexport function replaceParams(path: string, params: Record<string, string | number>) {\n for (const key in params) {\n const value = params[key].toString();\n path = path.replace(`{${key}}`, value).replace(`{#${key}}`, value);\n }\n\n return path;\n}\n","import { Context } from \"../core/context.js\";\nimport { createLogger } from \"../core/logger.js\";\nimport { m, MarkupType, type MarkupNode } from \"../core/markup.js\";\nimport { Dynamic } from \"../core/nodes/dynamic.js\";\nimport { ViewInstance } from \"../core/nodes/view.js\";\nimport { $, batch, untracked, type Source } from \"../core/signals.js\";\nimport { assertObject, isArray, isArrayOf, isFunction, isObject, isString } from \"../typeChecking.js\";\nimport type { View } from \"../types.js\";\nimport { shallowEqual } from \"../utils.js\";\nimport {\n catchLinks,\n joinPath,\n matchRoutes,\n patternToFragments,\n replaceParams,\n resolvePath,\n sortRoutes,\n splitPath,\n type ParsedRoute,\n type RouteMatch,\n} from \"./router.utils.js\";\n\n// ----- Types ----- //\n\nexport type Stringable = { toString(): string };\n\nexport interface RouteMatchContext {\n path: string;\n pattern: string;\n params: Record<string, string>;\n query: Record<string, string>;\n\n /**\n * Stores `value` at `key` in this context's state.\n */\n setState<T>(key: any, value: T): void;\n\n /**\n * For each tuple in `entries`, stores `value` at `key` in this context's state.\n */\n setState(entries: [key: any, value: any][]): void;\n\n /**\n * Redirects the user to a different route instead of matching the current one.\n */\n redirect(path: string): void;\n\n /**\n * Triggers the next beforeMatch function, or mounts the route.\n */\n // next(): void;\n}\n\nexport interface Route {\n /**\n * The path or path fragment to match.\n */\n path: string;\n\n /**\n * Path to redirect to when this route is matched, or a callback function that returns such path.\n */\n redirect?: string | ((ctx: RouteRedirectContext) => string) | ((ctx: RouteRedirectContext) => Promise<string>);\n\n /**\n * View to display when this route is matched.\n */\n view?: View<any>;\n\n /**\n * Subroutes.\n */\n routes?: Route[];\n\n /**\n * Called after the match is identified but before it is acted on. Use this to set state, load data, etc.\n */\n beforeMatch?: (ctx: RouteMatchContext) => void | Promise<void>;\n}\n\nexport interface RouteMeta {\n redirect?: string | ((ctx: RouteRedirectContext) => string) | ((ctx: RouteRedirectContext) => Promise<string>);\n pattern?: string;\n layers?: RouteLayer[];\n beforeMatch?: { fn: (ctx: RouteMatchContext) => void | Promise<void>; layerId: number }[];\n}\n\nexport interface RouteConfig {\n pattern: string;\n meta: RouteMeta;\n}\n\nexport interface RouteLayer {\n id: number;\n view: View<{}>;\n}\n\n/**\n * An active route layer whose markup has been initialized into a view.\n */\ninterface ActiveLayer {\n id: number;\n element: MarkupNode;\n context: Context;\n $slot: Source<MarkupNode | undefined>;\n}\n\n/**\n * Object passed to redirect callbacks. Contains information useful for determining how to redirect.\n */\nexport interface RouteRedirectContext {\n /**\n * The path as it appears in the URL bar.\n */\n path: string;\n\n /**\n * The pattern that this path was matched with.\n */\n pattern: string;\n\n /**\n * Named route params parsed from `path`.\n */\n params: Record<string, string>;\n\n /**\n * Query params parsed from `path`.\n */\n query: Record<string, string>;\n}\n\n/**\n * A log for a single step in the route resolution process.\n */\ninterface JourneyStep {\n kind: \"match\" | \"redirect\" | \"miss\";\n message: string;\n}\n\nexport interface NavigateOptions {\n /**\n * Replace the current item in the history stack instead of adding a new one.\n * The back button will send the user to the page they visited before this. Defaults to false.\n */\n replace?: boolean;\n\n /**\n * Preserve existing query params (if any) when navigating. Defaults to false.\n * If true, all existing query params are preserved and merged with new ones.\n * If an array of strings is passed only those keys will be preserved, then merged with any new ones.\n */\n preserveQuery?: boolean | string[];\n}\n\nexport interface RouterOptions {\n routes: Route[];\n\n /**\n * When true, the router will construct routes like \"https://www.example.com/#/sub/route\" which work without any backend intervention.\n */\n hash?: boolean;\n}\n\n// ----- Code ----- //\n\nexport const MOUNT = Symbol();\nexport const UNMOUNT = Symbol();\nexport const ROOT_VIEW = Symbol();\n\nexport class Router {\n #logger = createLogger(\"dolla.router\");\n\n #nextLayerId = 0;\n #activeLayers: ActiveLayer[] = [];\n #routes: ParsedRoute<RouteMeta>[] = [];\n\n #isMounted = false;\n\n #rootLayer!: ActiveLayer;\n\n /**\n * Use hash routing when true. Configured in router options.\n */\n #hash = false;\n\n /**\n * Cleanup functions to call on unmount.\n */\n #cleanup: (() => void)[] = [];\n\n /**\n * The current match object.\n */\n #match = $<RouteMatch>();\n\n /**\n * The currently matched route pattern, if any.\n */\n readonly $pattern = $(() => this.#match()?.pattern);\n\n /**\n * The current URL path.\n */\n readonly $path = $(() => this.#match()?.path ?? window.location.pathname);\n\n /**\n * The current named path params.\n */\n readonly $params = $(() => this.#match()?.params ?? {}, { equals: shallowEqual });\n\n /**\n * The current query params.\n */\n readonly $query = $(() => this.#match()?.query ?? {}, { equals: shallowEqual });\n\n constructor(options: RouterOptions) {\n assertObject(options, \"Options must be an object. Got: %t\");\n\n if (options.hash) {\n this.#hash = true;\n }\n\n // Add routes.\n this.#routes = sortRoutes(\n options.routes\n .flatMap((route) => this.#prepareRoute(route))\n .map((route) => ({\n pattern: route.pattern,\n meta: route.meta,\n fragments: patternToFragments(route.pattern),\n })),\n );\n\n assertValidRedirects(this.#routes);\n }\n\n async [MOUNT](parent: Element, context: Context): Promise<MarkupNode> {\n const $slot = $<MarkupNode>();\n this.#rootLayer = {\n id: this.#nextLayerId++,\n element: new Dynamic(context, $slot),\n context,\n $slot,\n };\n\n // Listen for popstate events and update route accordingly.\n const onPopState = () => this.#updateRoute();\n window.addEventListener(\"popstate\", onPopState);\n this.#cleanup.push(() => window.removeEventListener(\"popstate\", onPopState));\n\n // Listen for clicks on <a> tags within the app.\n this.#cleanup.push(\n catchLinks(parent, (anchor) => {\n this.#logger.info(\"intercepted click on <a> tag\", anchor);\n\n const href = anchor.getAttribute(\"href\")!;\n const preserveQuery = anchor.getAttribute(\"data-router-preserve-query\");\n\n this.go(href, {\n preserveQuery: parsePreserveQueryAttribute(preserveQuery),\n });\n }),\n );\n this.#logger.info(\"will intercept clicks on <a> tags within parent element:\", parent);\n\n this.#isMounted = true;\n\n // Mount initial route.\n await this.#updateRoute();\n\n return this.#rootLayer.element;\n }\n\n async [UNMOUNT]() {\n for (const callback of this.#cleanup) {\n callback();\n }\n this.#cleanup.length = 0;\n }\n\n /**\n * Navigate backward. Pass a number of steps to hit the back button that many times.\n */\n back(steps = 1) {\n window.history.go(-steps);\n }\n\n /**\n * Navigate forward. Pass a number of steps to hit the forward button that many times.\n */\n forward(steps = 1) {\n window.history.go(steps);\n }\n\n /**\n * Navigates to another route.\n *\n * @example\n * router.go(\"/login\"); // navigate to `/login`\n * router.go[\"/users\", 215], { replace: true }); // replace current history entry with `/users/215`\n */\n go(path: Stringable | Stringable[], options: NavigateOptions = {}) {\n let joined: string;\n\n if (Array.isArray(path)) {\n joined = joinPath(path);\n } else {\n joined = path.toString();\n }\n\n joined = resolvePath(window.location.pathname, joined);\n\n if (options.replace) {\n this.#replace(joined, options);\n } else {\n this.#push(joined, options);\n }\n }\n\n /**\n * Updates query params, keeping existing ones and applying new ones. Removes the query param if value is set to `null`.\n */\n updateQuery(values: Record<string, Stringable | null>) {\n const match = untracked(this.#match)!;\n const query = { ...this.$query() };\n\n for (const key in values) {\n const value = values[key];\n if (value === null) {\n delete query[key];\n } else {\n query[key] = value.toString();\n }\n }\n\n let queryParts: string[] = [];\n\n for (const key in query) {\n queryParts.push(`${key}=${query[key]}`);\n }\n const queryString = queryParts.length > 0 ? \"?\" + queryParts.join(\"&\") : \"\";\n\n this.#match({ ...match, query });\n\n window.history.replaceState(null, \"\", this.#hash ? \"/#\" + match.path + queryString : match.path + queryString);\n }\n\n #push(href: string, options: NavigateOptions) {\n this.#logger.info(\"(push)\", href);\n\n window.history.pushState(null, \"\", this.#hash ? \"/#\" + href : href);\n this.#updateRoute(href, options);\n }\n\n #replace(href: string, options: NavigateOptions) {\n this.#logger.info(\"(replace)\", href);\n\n window.history.replaceState(null, \"\", this.#hash ? \"/#\" + href : href);\n this.#updateRoute(href, options);\n }\n\n #getCurrentURL(): URL {\n if (this.#hash) {\n return new URL(window.location.hash.slice(1), window.location.origin);\n } else {\n return new URL(window.location.pathname, window.location.origin);\n }\n }\n\n /**\n * Run when the location changes. Diffs and mounts new routes and updates\n * the $path, $route, $params and $query states accordingly.\n */\n async #updateRoute(href?: string | undefined, options: NavigateOptions = {}) {\n const logger = this.#logger;\n const url = href ? new URL(href, window.location.origin) : this.#getCurrentURL();\n\n const { match, journey, state } = await this.#resolveRoute(url);\n\n for (let i = 0; i < journey.length; i++) {\n const step = journey[i];\n const tag = `(update: step ${i + 1} of ${journey.length})`;\n\n switch (step.kind) {\n case \"match\":\n logger?.info(`${tag} 📍 ${step.message}`);\n break;\n case \"redirect\":\n logger?.info(`${tag} ↩️ ${step.message}`);\n break;\n case \"miss\":\n logger?.info(`${tag} 💀 ${step.message}`);\n break;\n default:\n break;\n }\n }\n\n if (!match) {\n // Only crash if routing has been configured.\n if (this.#isMounted) {\n throw logger.crash(new NoRouteError(`Failed to match route '${url.pathname}'`));\n }\n return;\n }\n\n // Merge query params.\n let query = match.query;\n let queryParts: string[] = [];\n\n if (options.preserveQuery === true) {\n query = Object.assign({}, this.$query(), match.query);\n } else if (isArray(options.preserveQuery)) {\n const preserved: Record<string, any> = {};\n const current = this.$query();\n for (const key in current) {\n if (options.preserveQuery.includes(key)) {\n preserved[key] = current[key];\n }\n }\n query = Object.assign({}, preserved, match.query);\n }\n\n for (const key in query) {\n queryParts.push(`${key}=${query[key]}`);\n }\n const queryString = queryParts.length > 0 ? \"?\" + queryParts.join(\"&\") : \"\";\n\n // Update the URL if matched path differs from navigator path.\n // This happens if route resolution involved redirects.\n if (match.path !== location.pathname || location.search !== queryString) {\n window.history.replaceState(null, \"\", this.#hash ? \"/#\" + match.path + queryString : match.path + queryString);\n }\n\n // Run in batch so all new layers are mounted simultaneously with match signal change.\n // This avoids the old route effects receiving new signal values just before they unmount.\n batch(() => {\n const oldPattern = untracked(this.$pattern);\n\n this.#match({ ...match, query });\n\n if (match.pattern === oldPattern) {\n // If pattern has not changed, update state on current layers.\n for (const layer of this.#activeLayers) {\n const stateEntries = state.get(layer.id);\n if (stateEntries) {\n layer.context.setState(stateEntries);\n }\n }\n return;\n }\n\n const layers = match.meta.layers!;\n logger.info(\"mounting\", match);\n\n // Diff and update route layers.\n for (let i = 0; i < layers.length; i++) {\n const matchedLayer = layers[i];\n const activeLayer = this.#activeLayers[i];\n\n if (activeLayer?.id !== matchedLayer.id) {\n // Discard all previously active layers starting at this depth.\n this.#activeLayers = this.#activeLayers.slice(0, i);\n activeLayer?.element.unmount();\n\n const parentLayer = this.#activeLayers.at(-1) ?? this.#rootLayer;\n\n // Create a $slot and element for this layer.\n const $slot = $<MarkupNode>();\n const element = new ViewInstance(parentLayer.context, matchedLayer.view, {\n children: m(MarkupType.Dynamic, { source: $slot }),\n });\n\n // Set state for new layer.\n const stateEntries = state.get(matchedLayer.id);\n if (stateEntries) {\n element.context.setState(stateEntries);\n }\n\n // Add new layer to activeLayers.\n this.#activeLayers.push({\n id: matchedLayer.id,\n element,\n context: element.context,\n $slot,\n });\n\n // Slot this layer into parent.\n parentLayer.$slot(element);\n } else {\n // Update state for layers that are still active.\n const stateEntries = state.get(activeLayer.id);\n if (stateEntries) {\n activeLayer.context.setState(stateEntries);\n }\n }\n }\n });\n\n return { match, journey };\n }\n\n /**\n * Takes a URL and finds a match, following redirects.\n */\n async #resolveRoute(\n url: URL,\n journey: JourneyStep[] = [],\n state = new Map<number, [any, any][]>(),\n ): Promise<{\n match: RouteMatch<RouteMeta> | null;\n journey: JourneyStep[];\n state: Map<number, [any, any][]>; // map of layerId to state entries\n }> {\n return new Promise((resolve, reject) => {\n const match = matchRoutes(this.#routes, url.pathname);\n\n if (!match) {\n return resolve({\n match: null,\n journey: [...journey, { kind: \"miss\", message: `no match for '${url.pathname}'` }],\n state,\n });\n }\n\n let redirect = match.meta.redirect;\n\n const finalize = async () => {\n if (redirect != null) {\n let path: string;\n\n if (isString(redirect)) {\n path = replaceParams(redirect, match.params);\n } else if (isFunction(redirect)) {\n const redirectContext: RouteRedirectContext = {\n path: match.path,\n pattern: match.pattern,\n params: match.params,\n query: match.query,\n };\n path = await redirect(redirectContext);\n if (!isString(path)) {\n return reject(new Error(`Redirect function must return a path to redirect to.`));\n }\n if (!path.startsWith(\"/\")) {\n // Not absolute. Resolve against matched path.\n path = resolvePath(match.path, path);\n }\n } else {\n return reject(new TypeError(`Redirect must either be a path string or a function.`));\n }\n\n resolve(\n this.#resolveRoute(new URL(path, window.location.origin), [\n ...journey,\n { kind: \"redirect\", message: `redirecting '${match.path}' -> '${path}'` },\n ]),\n );\n } else {\n resolve({ match, journey: [...journey, { kind: \"match\", message: `matched route '${match.path}'` }], state });\n }\n };\n\n if (match.meta.beforeMatch?.length) {\n const callbacks = match.meta.beforeMatch;\n let i = -1;\n const next = () => {\n i++;\n if (i === callbacks.length) {\n // Mount route\n finalize();\n } else {\n // Next callback\n let finalized = false;\n const result = callbacks[i].fn({\n path: match.path,\n pattern: match.pattern,\n params: match.params,\n query: match.query,\n\n setState: (...args: any[]) => {\n const id = callbacks[i].layerId;\n const entries: [any, any][] = [];\n\n if (args.length === 1 && isArrayOf(isArray, args[0])) {\n entries.push(...(args[0] as [any, any][]));\n } else if (args.length === 2) {\n entries.push([args[0], args[1]]);\n } else {\n throw new Error(\"Invalid arguments.\");\n }\n\n const current = state.get(id);\n if (!current) {\n state.set(id, entries);\n } else {\n entries.push(...entries);\n }\n },\n\n redirect: (path) => {\n redirect = path;\n finalized = true;\n finalize();\n },\n\n // next,\n });\n if (!finalized) {\n if (result instanceof Promise) {\n result.then(next);\n } else {\n next();\n }\n }\n }\n };\n\n next();\n\n // TODO: Show warning after timeout if next hasn't been called?\n } else {\n finalize();\n }\n });\n }\n\n /**\n * Parses a route definition object into a set of matchable routes.\n *\n * @param route - Route config object.\n * @param layers - Array of parent layers. Passed when this function calls itself on nested routes.\n */\n #prepareRoute(route: Route, parents: Route[] = [], layers: RouteLayer[] = []) {\n if (!isObject(route) || !isString(route.path)) {\n throw new TypeError(`Route configs must be objects with a 'path' string property. Got: ${route}`);\n }\n\n if (route.redirect && route.routes) {\n throw new Error(`Route cannot have both a 'redirect' and nested 'routes'.`);\n } else if (route.redirect && route.view) {\n throw new Error(`Route cannot have both a 'redirect' and a 'view'.`);\n } else if (!route.view && !route.routes && !route.redirect) {\n throw new Error(`Route must have a 'view', a 'redirect', or a set of nested 'routes'.`);\n }\n\n let parts: string[] = [];\n\n for (const parent of parents) {\n parts.push(...splitPath(parent.path));\n }\n\n parts.push(...splitPath(route.path));\n\n // Remove trailing wildcard for joining with nested routes.\n if (parts[parts.length - 1] === \"*\") {\n parts.pop();\n }\n\n const routes: RouteConfig[] = [];\n\n if (route.redirect) {\n let redirect = route.redirect;\n\n if (isString(redirect)) {\n redirect = resolvePath(joinPath(parts), redirect);\n\n if (!redirect.startsWith(\"/\")) {\n redirect = \"/\" + redirect;\n }\n }\n\n routes.push({\n pattern: \"/\" + joinPath([...parts, ...splitPath(route.path)]),\n meta: {\n redirect,\n },\n });\n\n return routes;\n }\n\n let view: View<any> = (props: any) => props.children;\n\n if (isFunction(route.view)) {\n view = route.view;\n } else if (route.view) {\n throw new TypeError(`Route '${route.path}' expected a view function or undefined. Got: ${route.view}`);\n }\n\n const layer: RouteLayer = { id: this.#nextLayerId++, view };\n\n // Parse nested routes if they exist.\n if (route.routes) {\n for (const subroute of route.routes) {\n routes.push(...this.#prepareRoute(subroute, [...parents, route], [...layers, layer]));\n }\n } else {\n routes.push({\n pattern: parents.length ? joinPath([...parents.map((p) => p.path), route.path]) : route.path,\n meta: {\n pattern: route.path,\n layers: [...layers, layer],\n // Store the layer ID with each beforeMatch so we can correlate which context needs to get any state that is set.\n beforeMatch: parents\n .flatMap((parent, i) => (parent.beforeMatch ? { fn: parent.beforeMatch, layerId: layers[i].id } : null))\n .concat(route.beforeMatch ? { fn: route.beforeMatch, layerId: layer.id } : null)\n .filter((x) => x != null),\n },\n });\n }\n\n return routes;\n }\n}\n\nfunction assertValidRedirects(routes: ParsedRoute<RouteMeta>[]) {\n // Test redirects to make sure all possible redirect targets actually exist.\n for (const route of routes) {\n if (route.meta.redirect) {\n let redirectPath: string;\n\n if (isFunction(route.meta.redirect)) {\n // throw new Error(`Redirect functions are not yet supported.`);\n // Just allow, though it could fail later. Best not to call the function and cause potential side effects.\n } else if (isString(route.meta.redirect)) {\n redirectPath = route.meta.redirect;\n\n const match = matchRoutes(routes, redirectPath, {\n willMatch(r) {\n return r !== route;\n },\n });\n\n if (!match) {\n throw new Error(`Found a redirect to an undefined URL. From '${route.pattern}' to '${route.meta.redirect}'`);\n }\n } else {\n throw new TypeError(`Expected a string or redirect function. Got: ${route.meta.redirect}`);\n }\n }\n }\n}\n\n/**\n * Parses the data-router-preserve-query attribute from a link.\n */\nfunction parsePreserveQueryAttribute(value: null | string | boolean): boolean | string[] {\n if (value === null) {\n return false;\n } else if (value === true || value === false) {\n return value;\n } else if (typeof value === \"string\") {\n value = value.trim();\n if (value === \"\" || value === \"true\") {\n return true;\n } else if (value === \"false\") {\n return false;\n }\n\n return value\n .split(\",\")\n .map((k) => k.trim())\n .filter((k) => k.length > 0);\n } else {\n throw new Error(`Invalid type for data-router-preserve-query attribute: ${typeof value} (value: ${value})`);\n }\n}\n\nclass NoRouteError extends Error {}\n"],"names":["splitPath","path","assertString","f","joinPath","parts","assertArrayOf","part","isFunction","x","joined","_a","p","resolvePath","base","resolved","i","parseQueryParams","query","entries","entry","matchRoutes","routes","url","options","route","fragments","matched","frag","params","sortRoutes","withoutParams","withNumericParams","withParams","wildcard","bySizeDesc","a","b","patternToFragments","pattern","safeExternalLink","protocolLink","catchLinks","root","callback","_window","traverse","node","handler","e","anchor","replaceParams","key","value","MOUNT","UNMOUNT","Router","__privateAdd","_Router_instances","_logger","createLogger","_nextLayerId","_activeLayers","_routes","_isMounted","_rootLayer","_hash","_cleanup","_match","$","__publicField","__privateGet","shallowEqual","assertObject","__privateSet","__privateMethod","prepareRoute_fn","assertValidRedirects","parent","context","$slot","__privateWrapper","Dynamic","onPopState","updateRoute_fn","href","preserveQuery","parsePreserveQueryAttribute","steps","replace_fn","push_fn","values","match","untracked","queryParts","queryString","getCurrentURL_fn","logger","journey","state","resolveRoute_fn","step","tag","NoRouteError","isArray","preserved","current","batch","oldPattern","layer","stateEntries","layers","matchedLayer","activeLayer","parentLayer","element","ViewInstance","m","MarkupType","resolve","reject","redirect","finalize","isString","redirectContext","callbacks","next","finalized","result","args","id","isArrayOf","parents","isObject","view","props","subroute","redirectPath","r","k"],"mappings":"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"}