sinwan 0.0.0 → 1.0.0
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/LICENSE +21 -0
- package/README.md +94 -1
- package/dist/cjs/index.development.js +2288 -0
- package/dist/cjs/index.development.js.map +35 -0
- package/dist/cjs/index.production.min.js +3 -0
- package/dist/cjs/index.production.min.js.map +35 -0
- package/dist/cjs/jsx/jsx-dev-runtime.development.js +124 -0
- package/dist/cjs/jsx/jsx-dev-runtime.development.js.map +10 -0
- package/dist/cjs/jsx/jsx-dev-runtime.production.min.js +3 -0
- package/dist/cjs/jsx/jsx-dev-runtime.production.min.js.map +10 -0
- package/dist/cjs/jsx/jsx-runtime.development.js +116 -0
- package/dist/cjs/jsx/jsx-runtime.development.js.map +10 -0
- package/dist/cjs/jsx/jsx-runtime.production.min.js +3 -0
- package/dist/cjs/jsx/jsx-runtime.production.min.js.map +10 -0
- package/dist/cjs/package.json +4 -0
- package/dist/cjs/renderer/index.development.js +1175 -0
- package/dist/cjs/renderer/index.development.js.map +24 -0
- package/dist/cjs/renderer/index.production.min.js +3 -0
- package/dist/cjs/renderer/index.production.min.js.map +24 -0
- package/dist/cjs/server/index.development.js +1265 -0
- package/dist/cjs/server/index.development.js.map +23 -0
- package/dist/cjs/server/index.production.min.js +3 -0
- package/dist/cjs/server/index.production.min.js.map +23 -0
- package/dist/component/control-flow.d.ts +18 -0
- package/dist/component/control-flow.d.ts.map +1 -0
- package/dist/component/create.d.ts +75 -0
- package/dist/component/create.d.ts.map +1 -0
- package/dist/component/index.d.ts +12 -0
- package/dist/component/index.d.ts.map +1 -0
- package/dist/component/instance.d.ts +84 -0
- package/dist/component/instance.d.ts.map +1 -0
- package/dist/component/lifecycle.d.ts +41 -0
- package/dist/component/lifecycle.d.ts.map +1 -0
- package/dist/component/provide-inject.d.ts +45 -0
- package/dist/component/provide-inject.d.ts.map +1 -0
- package/dist/escaper.d.ts +26 -0
- package/dist/escaper.d.ts.map +1 -0
- package/dist/esm/index.development.js +2201 -0
- package/dist/esm/index.development.js.map +35 -0
- package/dist/esm/index.production.min.js +4 -0
- package/dist/esm/index.production.min.js.map +35 -0
- package/dist/esm/jsx/jsx-dev-runtime.development.js +73 -0
- package/dist/esm/jsx/jsx-dev-runtime.development.js.map +10 -0
- package/dist/esm/jsx/jsx-dev-runtime.production.min.js +4 -0
- package/dist/esm/jsx/jsx-dev-runtime.production.min.js.map +10 -0
- package/dist/esm/jsx/jsx-runtime.development.js +77 -0
- package/dist/esm/jsx/jsx-runtime.development.js.map +10 -0
- package/dist/esm/jsx/jsx-runtime.production.min.js +4 -0
- package/dist/esm/jsx/jsx-runtime.production.min.js.map +10 -0
- package/dist/esm/package.json +4 -0
- package/dist/esm/renderer/index.development.js +1124 -0
- package/dist/esm/renderer/index.development.js.map +24 -0
- package/dist/esm/renderer/index.production.min.js +4 -0
- package/dist/esm/renderer/index.production.min.js.map +24 -0
- package/dist/esm/server/index.development.js +1214 -0
- package/dist/esm/server/index.development.js.map +23 -0
- package/dist/esm/server/index.production.min.js +4 -0
- package/dist/esm/server/index.production.min.js.map +23 -0
- package/dist/hydration/hydrate.d.ts +29 -0
- package/dist/hydration/hydrate.d.ts.map +1 -0
- package/dist/hydration/index.d.ts +7 -0
- package/dist/hydration/index.d.ts.map +1 -0
- package/dist/hydration/markers.d.ts +48 -0
- package/dist/hydration/markers.d.ts.map +1 -0
- package/dist/hydration/walk.d.ts +32 -0
- package/dist/hydration/walk.d.ts.map +1 -0
- package/dist/index.d.ts +14 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +7 -0
- package/dist/index.mjs +4 -0
- package/dist/jsx/jsx-dev-runtime.d.ts +10 -0
- package/dist/jsx/jsx-dev-runtime.d.ts.map +1 -0
- package/dist/jsx/jsx-runtime.d.ts +67 -0
- package/dist/jsx/jsx-runtime.d.ts.map +1 -0
- package/dist/jsx/jsx-types.d.ts +702 -0
- package/dist/jsx/jsx-types.d.ts.map +1 -0
- package/dist/jsx-dev-runtime.d.ts +1 -0
- package/dist/jsx-dev-runtime.js +7 -0
- package/dist/jsx-dev-runtime.mjs +4 -0
- package/dist/jsx-runtime.d.ts +1 -0
- package/dist/jsx-runtime.js +7 -0
- package/dist/jsx-runtime.mjs +4 -0
- package/dist/reactivity/batch.d.ts +27 -0
- package/dist/reactivity/batch.d.ts.map +1 -0
- package/dist/reactivity/computed.d.ts +41 -0
- package/dist/reactivity/computed.d.ts.map +1 -0
- package/dist/reactivity/effect.d.ts +79 -0
- package/dist/reactivity/effect.d.ts.map +1 -0
- package/dist/reactivity/index.d.ts +15 -0
- package/dist/reactivity/index.d.ts.map +1 -0
- package/dist/reactivity/scheduler.d.ts +35 -0
- package/dist/reactivity/scheduler.d.ts.map +1 -0
- package/dist/reactivity/signal.d.ts +36 -0
- package/dist/reactivity/signal.d.ts.map +1 -0
- package/dist/renderer/attributes.d.ts +8 -0
- package/dist/renderer/attributes.d.ts.map +1 -0
- package/dist/renderer/dom-ops.d.ts +30 -0
- package/dist/renderer/dom-ops.d.ts.map +1 -0
- package/dist/renderer/events.d.ts +21 -0
- package/dist/renderer/events.d.ts.map +1 -0
- package/dist/renderer/index.d.ts +12 -0
- package/dist/renderer/index.d.ts.map +1 -0
- package/dist/renderer/mount.d.ts +27 -0
- package/dist/renderer/mount.d.ts.map +1 -0
- package/dist/renderer/render-children.d.ts +19 -0
- package/dist/renderer/render-children.d.ts.map +1 -0
- package/dist/renderer/render-control-flow.d.ts +13 -0
- package/dist/renderer/render-control-flow.d.ts.map +1 -0
- package/dist/renderer/render-element.d.ts +13 -0
- package/dist/renderer/render-element.d.ts.map +1 -0
- package/dist/renderer/types.d.ts +65 -0
- package/dist/renderer/types.d.ts.map +1 -0
- package/dist/renderer/unmount.d.ts +20 -0
- package/dist/renderer/unmount.d.ts.map +1 -0
- package/dist/renderer.d.ts +1 -0
- package/dist/renderer.js +7 -0
- package/dist/renderer.mjs +4 -0
- package/dist/server/hydration-markers.d.ts +23 -0
- package/dist/server/hydration-markers.d.ts.map +1 -0
- package/dist/server/index.d.ts +9 -0
- package/dist/server/index.d.ts.map +1 -0
- package/dist/server/renderer.d.ts +33 -0
- package/dist/server/renderer.d.ts.map +1 -0
- package/dist/server/stream.d.ts +20 -0
- package/dist/server/stream.d.ts.map +1 -0
- package/dist/server.d.ts +1 -0
- package/dist/server.js +7 -0
- package/dist/server.mjs +4 -0
- package/dist/types.d.ts +48 -0
- package/dist/types.d.ts.map +1 -0
- package/package.json +116 -4
- package/bun.lock +0 -26
- package/src/index.ts +0 -1
- package/tsconfig.json +0 -29
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../src/jsx/jsx-runtime.ts", "../../src/escaper.ts", "../../src/reactivity/scheduler.ts", "../../src/reactivity/effect.ts", "../../src/reactivity/signal.ts", "../../src/reactivity/computed.ts", "../../src/renderer/dom-ops.ts", "../../src/renderer/events.ts", "../../src/component/control-flow.ts", "../../src/server/renderer.ts", "../../src/component/instance.ts", "../../src/hydration/markers.ts", "../../src/server/stream.ts", "../../src/server/hydration-markers.ts"],
|
|
4
|
+
"sourcesContent": [
|
|
5
|
+
"/**\n * SinwanJS View Module — JSX Runtime\n *\n * JSX factory that returns SinwanElement structures for the view renderer.\n * Imported automatically when using JSX syntax.\n */\n\nimport type { SinwanElement, SinwanNode } from \"../types.ts\";\nimport type { SinwanIntrinsicElements } from \"./jsx-types\";\n\nexport const Fragment = Symbol(\"Fragment\");\n\n/**\n * A string that has been marked as safe HTML (already escaped).\n * Used for raw HTML injection with explicit trust marking.\n */\nexport class HtmlEscapedString extends String {\n constructor(public readonly value: string) {\n super(value);\n }\n override toString() {\n return this.value;\n }\n}\n\nexport const raw = (str: string) => new HtmlEscapedString(str);\n\n// Fast path for intrinsic elements\nconst VOID_ELEMENTS = new Set([\n \"area\",\n \"base\",\n \"br\",\n \"col\",\n \"embed\",\n \"hr\",\n \"img\",\n \"input\",\n \"link\",\n \"meta\",\n \"param\",\n \"source\",\n \"track\",\n \"wbr\",\n]);\n\n/**\n * Normalize children into an array of SinwanNode.\n */\nfunction normalizeChildren(children: any): SinwanNode[] {\n if (children == null || typeof children === \"boolean\") return [];\n if (Array.isArray(children)) return children.flat(Infinity);\n return [children];\n}\n\n/**\n * Build an SinwanElement from a component/tag invocation.\n * Shared logic for jsx, jsxs, and jsxDEV.\n */\nfunction buildElement(\n type: any,\n props: any,\n children: SinwanNode[],\n): SinwanElement {\n // Handle Fragment\n if (type === Fragment) {\n return { tag: \"\", props: {}, children };\n }\n\n // Functional components and intrinsic HTML elements both pass the type\n // through as `tag`. The renderer (client / server / hydration) is the\n // single owner of component-instance creation and lifecycle dispatch.\n // Calling the function eagerly here would bypass instance management\n // and break `onMounted`, `provide`/`inject`, and the parent/child tree.\n if (typeof type === \"function\" || typeof type === \"string\") {\n const finalProps = props ?? {};\n // Mirror children into props.children so component setup can read it.\n if (children.length > 0 && finalProps.children === undefined) {\n finalProps.children = children.length === 1 ? children[0] : children;\n }\n return { tag: type, props: finalProps, children };\n }\n\n // Fallback\n return { tag: \"\", props: {}, children };\n}\n\n/**\n * JSX factory — called for elements with 0 or 1 child.\n * TypeScript auto-imports this in production mode (`react-jsx`).\n */\nexport function jsx(type: any, props: any, key?: any): SinwanElement {\n return buildElement(type, props, normalizeChildren(props?.children));\n}\n\n/**\n * JSX static factory — called for elements with 2+ children.\n * `props.children` is **already an array**, so we skip normalizeChildren.\n */\nexport function jsxs(type: any, props: any, key?: any): SinwanElement {\n const children = props?.children;\n // Children is guaranteed to be an array by the compiler\n return buildElement(\n type,\n props,\n Array.isArray(children)\n ? children.flat(Infinity)\n : normalizeChildren(children),\n );\n}\n\n/**\n * Source location metadata attached by the compiler in dev mode.\n */\nexport interface JSXSource {\n fileName: string;\n lineNumber: number;\n columnNumber: number;\n}\n\n/**\n * JSX dev factory — called in dev mode (`react-jsxdev`).\n * Receives extra source/debug info for better error messages.\n */\nexport function jsxDEV(\n type: any,\n props: any,\n key: any,\n isStaticChildren: boolean,\n source?: JSXSource,\n self?: unknown,\n): SinwanElement {\n const children = isStaticChildren\n ? Array.isArray(props?.children)\n ? props.children.flat(Infinity)\n : normalizeChildren(props?.children)\n : normalizeChildren(props?.children);\n\n const element = buildElement(type, props, children);\n\n // Attach debug metadata (useful for dev-tools / error traces)\n if (source) {\n (element as any).__source = source;\n }\n\n return element;\n}\n\nexport namespace JSX {\n export type Element = SinwanElement | Promise<SinwanElement>;\n export interface IntrinsicAttributes {\n key?: string | number;\n }\n export interface ElementChildrenAttribute {\n children: {};\n }\n export interface IntrinsicElements extends SinwanIntrinsicElements {}\n}\n\ndeclare global {\n namespace JSX {\n type Element = SinwanElement | Promise<SinwanElement>;\n interface IntrinsicAttributes {\n key?: string | number;\n }\n interface ElementChildrenAttribute {\n children: {};\n }\n interface IntrinsicElements extends SinwanIntrinsicElements {}\n }\n}\n",
|
|
6
|
+
"/**\n * Sinwan — HTML Escaping\n *\n * Security utilities for sanitizing interpolated values.\n * Runtime-agnostic: prefers `Bun.escapeHTML` when running on Bun (native\n * speed), and falls back to a portable implementation everywhere else\n * (Node, Deno, Cloudflare Workers, browsers).\n */\n\nimport { HtmlEscapedString, raw } from \"./jsx/jsx-runtime\";\n\nexport { HtmlEscapedString, raw };\n\n// ─── Native fast-path detection ─────────────────────────────\n// `Bun` is only defined inside the Bun runtime. We probe it via globalThis\n// so module evaluation never throws on other runtimes.\nconst _bun = (globalThis as any).Bun as\n | { escapeHTML?: (s: string) => string }\n | undefined;\n\nconst _nativeEscape: ((s: string) => string) | undefined =\n typeof _bun?.escapeHTML === \"function\"\n ? _bun.escapeHTML.bind(_bun)\n : undefined;\n\n// ─── Portable fallback ──────────────────────────────────────\n// Covers the five characters that matter inside HTML text and attributes.\nconst HTML_ESCAPE_RE = /[&<>\"']/g;\nconst HTML_ESCAPE_MAP: Record<string, string> = {\n \"&\": \"&\",\n \"<\": \"<\",\n \">\": \">\",\n '\"': \""\",\n \"'\": \"'\",\n};\n\nfunction portableEscape(str: string): string {\n // Avoid a regex pass when nothing to escape (very common case).\n HTML_ESCAPE_RE.lastIndex = 0;\n if (!HTML_ESCAPE_RE.test(str)) return str;\n return str.replace(HTML_ESCAPE_RE, (c) => HTML_ESCAPE_MAP[c]!);\n}\n\n/**\n * Escape HTML entities in any value.\n * Returns \"\" for null/undefined/boolean and pass-through for numbers.\n * Pre-escaped strings (`HtmlEscapedString`) are returned untouched.\n */\nexport function escapeHtml(value: unknown): string {\n if (value == null || typeof value === \"boolean\") return \"\";\n if (typeof value === \"number\") return String(value);\n if (value instanceof HtmlEscapedString) return value.value;\n const s = String(value);\n return _nativeEscape ? _nativeEscape(s) : portableEscape(s);\n}\n\n/**\n * Mark a string as safe HTML (pre-escaped).\n * USE WITH CAUTION - only for trusted content!\n */\nexport function safeHtml(html: string): HtmlEscapedString {\n return raw(html);\n}\n\n/**\n * Check if a value is already escaped HTML.\n */\nexport function isSafeHtml(value: unknown): value is HtmlEscapedString {\n return value instanceof HtmlEscapedString;\n}\n",
|
|
7
|
+
"/**\n * SinwanJS Reactivity — Scheduler\n *\n * Microtask-based flush queue for batching reactive updates.\n * Effects are NOT run synchronously on signal write — they are\n * queued and flushed in a microtask (like Vue's nextTick).\n */\n\nexport interface EffectNode {\n id: number;\n run(): void;\n active: boolean;\n}\n\nconst pendingEffects = new Set<EffectNode>();\nlet flushScheduled = false;\nlet isFlushing = false;\n\n// Pending nextTick callbacks\nconst pendingCallbacks: (() => void)[] = [];\n\n/**\n * Schedule an effect for the next microtask flush.\n */\nexport function scheduleEffect(effect: EffectNode): void {\n if (!effect.active) return;\n pendingEffects.add(effect);\n scheduleFlush();\n}\n\n/**\n * Remove a scheduled effect (e.g. when disposed before flush).\n */\nexport function unscheduleEffect(effect: EffectNode): void {\n pendingEffects.delete(effect);\n}\n\n/**\n * Schedule the microtask flush if not already scheduled.\n */\nfunction scheduleFlush(): void {\n if (!flushScheduled) {\n flushScheduled = true;\n queueMicrotask(flush);\n }\n}\n\n/**\n * Flush all pending effects. Effects added during flush are\n * processed in the same pass (convergence loop with a safety limit).\n */\nfunction flush(): void {\n isFlushing = true;\n\n // Sort by id to guarantee parent-before-child execution order\n const sorted = [...pendingEffects].sort((a, b) => a.id - b.id);\n pendingEffects.clear();\n\n for (const effect of sorted) {\n if (effect.active) {\n effect.run();\n }\n }\n\n // If new effects were queued during the flush, drain them too\n // (safety limit to prevent infinite loops)\n let safety = 10;\n while (pendingEffects.size > 0 && safety-- > 0) {\n const next = [...pendingEffects].sort((a, b) => a.id - b.id);\n pendingEffects.clear();\n for (const effect of next) {\n if (effect.active) {\n effect.run();\n }\n }\n }\n\n flushScheduled = false;\n isFlushing = false;\n\n // Run nextTick callbacks after all effects\n const cbs = pendingCallbacks.splice(0);\n for (const cb of cbs) {\n cb();\n }\n}\n\n/**\n * Returns true if the scheduler is currently flushing effects.\n */\nexport function isFlushingEffects(): boolean {\n return isFlushing;\n}\n\n/**\n * Queue a callback that runs after the next reactive flush completes.\n * Similar to Vue's nextTick().\n */\nexport function nextTick(fn?: () => void): Promise<void> {\n return new Promise<void>((resolve) => {\n const callback = () => {\n fn?.();\n resolve();\n };\n if (flushScheduled || isFlushing) {\n // Effects are pending — run after they flush\n pendingCallbacks.push(callback);\n } else {\n // No pending effects — run on next microtask\n queueMicrotask(callback);\n }\n });\n}\n\n/**\n * Force synchronous flush of all pending effects.\n * Primarily for testing and batch().\n */\nexport function flushSync(): void {\n if (flushScheduled) {\n flushScheduled = false;\n flush();\n }\n}\n",
|
|
8
|
+
"/**\n * SinwanJS Reactivity — Effect\n *\n * Fine-grained effect system with automatic dependency tracking.\n * When an effect runs, any signal reads are tracked as dependencies.\n * When those signals change, the effect is re-scheduled.\n */\n\nimport { type EffectNode, scheduleEffect, unscheduleEffect } from \"./scheduler.ts\";\n\n// ─── Global tracking state ─────────────────────────────────\n\nlet activeEffect: ReactiveEffect | null = null;\nconst effectStack: ReactiveEffect[] = [];\nlet effectIdCounter = 0;\n\n// ─── Subscription interface ────────────────────────────────\n\n/**\n * A Dep is any object that can track subscribers.\n * Signals and computeds implement this internally.\n */\nexport interface Dep {\n subscribers: Set<ReactiveEffect>;\n}\n\n// ─── ReactiveEffect ────────────────────────────────────────\n\nexport type CleanupFn = () => void;\nexport type EffectFn = () => CleanupFn | void;\n\nexport class ReactiveEffect implements EffectNode {\n id: number;\n active = true;\n\n /** The user-supplied function */\n private fn: EffectFn;\n\n /** Cleanup returned from the last run */\n private cleanup: CleanupFn | void = undefined;\n\n /** All deps this effect is subscribed to (for bidirectional cleanup) */\n deps: Set<Dep> = new Set();\n\n constructor(fn: EffectFn) {\n this.id = effectIdCounter++;\n this.fn = fn;\n }\n\n /**\n * Execute the effect function while tracking dependencies.\n */\n run(): void {\n if (!this.active) return;\n\n // Prevent infinite re-entry\n if (effectStack.includes(this)) return;\n\n // Clean up previous dependencies\n this.cleanupDeps();\n\n // Run user cleanup from previous execution\n if (this.cleanup) {\n this.cleanup();\n this.cleanup = undefined;\n }\n\n // Push onto the tracking stack\n effectStack.push(this);\n const prevEffect = activeEffect;\n activeEffect = this;\n\n try {\n const result = this.fn();\n if (typeof result === \"function\") {\n this.cleanup = result;\n }\n } finally {\n activeEffect = prevEffect;\n effectStack.pop();\n }\n }\n\n /**\n * Unsubscribe from all current deps so stale deps don't trigger this effect.\n */\n private cleanupDeps(): void {\n for (const dep of this.deps) {\n dep.subscribers.delete(this);\n }\n this.deps.clear();\n }\n\n /**\n * Notify the scheduler that this effect should re-run.\n */\n notify(): void {\n scheduleEffect(this);\n }\n\n /**\n * Permanently dispose this effect — stop tracking & unsubscribe.\n */\n dispose(): void {\n if (!this.active) return;\n this.active = false;\n\n // Run user cleanup\n if (this.cleanup) {\n this.cleanup();\n this.cleanup = undefined;\n }\n\n this.cleanupDeps();\n unscheduleEffect(this);\n }\n}\n\n// ─── Public API ────────────────────────────────────────────\n\n/**\n * Create a reactive effect.\n *\n * The effect function runs immediately to establish initial dependencies.\n * It re-runs whenever any tracked signal changes.\n * Returns a dispose function to stop the effect.\n *\n * @example\n * const count = signal(0);\n * const dispose = effect(() => {\n * console.log(\"count is\", count.value);\n * });\n * // logs \"count is 0\" immediately\n *\n * count.value = 1;\n * // logs \"count is 1\" on next microtask\n *\n * dispose(); // stops tracking\n */\nexport function effect(fn: EffectFn): CleanupFn {\n const e = new ReactiveEffect(fn);\n // Run immediately (synchronous first run for initial tracking)\n e.run();\n return () => e.dispose();\n}\n\n// ─── Tracking helpers (used by signals/computed) ───────────\n\n/**\n * Track a dependency from the currently active effect.\n * Called by signal.value getters.\n */\nexport function track(dep: Dep): void {\n if (activeEffect) {\n dep.subscribers.add(activeEffect);\n activeEffect.deps.add(dep);\n }\n}\n\n/**\n * Trigger all subscribers of a dependency.\n * Called by signal.value setters.\n */\nexport function trigger(dep: Dep): void {\n // Copy to avoid modification during iteration\n const effects = [...dep.subscribers];\n for (const effect of effects) {\n effect.notify();\n }\n}\n\n/**\n * Returns the currently active effect (for advanced usage).\n */\nexport function getActiveEffect(): ReactiveEffect | null {\n return activeEffect;\n}\n",
|
|
9
|
+
"/**\n * SinwanJS Reactivity — Signal\n *\n * A signal is a reactive container for a single value.\n * Reading `.value` tracks the current effect as a subscriber.\n * Writing `.value` notifies all subscribers.\n *\n * Inspired by Vue 3 ref(), Solid signals, Preact signals.\n */\n\nimport { type Dep, track, trigger } from \"./effect.ts\";\n\n// ─── Signal interface ──────────────────────────────────────\n\nexport interface Signal<T> {\n /** Get or set the reactive value. Reading tracks; writing notifies. */\n value: T;\n\n /** Read the value without tracking dependencies. */\n peek(): T;\n\n /** Manually subscribe to changes. Returns an unsubscribe function. */\n subscribe(fn: (value: T) => void): () => void;\n}\n\n// Brand for type-checking\nconst SIGNAL_BRAND = Symbol(\"Sinwan:signal\");\n\n// ─── Implementation ────────────────────────────────────────\n\nclass SignalImpl<T> implements Signal<T>, Dep {\n [SIGNAL_BRAND] = true;\n\n subscribers = new Set<import(\"./effect.ts\").ReactiveEffect>();\n private _value: T;\n private _manualSubs = new Set<(value: T) => void>();\n\n constructor(initial: T) {\n this._value = initial;\n }\n\n get value(): T {\n track(this);\n return this._value;\n }\n\n set value(newValue: T) {\n if (Object.is(this._value, newValue)) return;\n this._value = newValue;\n trigger(this);\n\n // Notify manual subscribers\n for (const fn of this._manualSubs) {\n fn(newValue);\n }\n }\n\n peek(): T {\n return this._value;\n }\n\n subscribe(fn: (value: T) => void): () => void {\n this._manualSubs.add(fn);\n return () => {\n this._manualSubs.delete(fn);\n };\n }\n\n /**\n * toString() for interpolation in templates.\n */\n toString(): string {\n return String(this.value);\n }\n\n /**\n * valueOf() for numeric operations.\n */\n valueOf(): T {\n return this.value;\n }\n}\n\n// ─── Public API ────────────────────────────────────────────\n\n/**\n * Create a reactive signal.\n *\n * @example\n * const count = signal(0);\n * console.log(count.value); // 0\n *\n * effect(() => {\n * console.log(count.value); // re-runs when count changes\n * });\n *\n * count.value = 5; // triggers the effect\n */\nexport function signal<T>(initial: T): Signal<T> {\n return new SignalImpl(initial);\n}\n\n/**\n * Type guard: check if a value is a Signal.\n */\nexport function isSignal(value: unknown): value is Signal<unknown> {\n return (\n value != null && typeof value === \"object\" && SIGNAL_BRAND in (value as any)\n );\n}\n",
|
|
10
|
+
"/**\n * SinwanJS Reactivity — Computed\n *\n * A computed is a derived reactive value that lazily re-evaluates\n * when its dependencies change. It caches the result and only\n * recomputes when actually read after a dependency has changed.\n *\n * Design: The computed does NOT use the scheduler for its own\n * re-evaluation. When a dependency changes, it marks itself dirty\n * and triggers downstream subscribers (which ARE scheduled).\n * The actual re-evaluation happens lazily on `.value` access.\n *\n * Inspired by Vue 3 computed(), Solid createMemo().\n */\n\nimport { type Dep, track, trigger, ReactiveEffect } from \"./effect.ts\";\n\n// ─── Computed interface ────────────────────────────────────\n\nexport interface Computed<T> {\n /** Read the computed value (lazy evaluation, cached). */\n readonly value: T;\n\n /** Read without tracking. */\n peek(): T;\n}\n\n// Brand for type-checking\nconst COMPUTED_BRAND = Symbol(\"Sinwan:computed\");\n\n// ─── Implementation ────────────────────────────────────────\n\nclass ComputedImpl<T> implements Computed<T>, Dep {\n [COMPUTED_BRAND] = true;\n\n subscribers = new Set<ReactiveEffect>();\n\n _value!: T;\n _dirty = true;\n _effect: ReactiveEffect;\n\n constructor(getter: () => T) {\n const self = this;\n\n // Internal effect solely for dependency tracking.\n // The fn wraps the getter — it writes to self._value as a side effect.\n this._effect = new ReactiveEffect(() => {\n self._value = getter();\n });\n\n // Override notify: don't schedule this effect via the scheduler.\n // Instead mark dirty and propagate to our own subscribers.\n this._effect.notify = function () {\n if (!self._dirty) {\n self._dirty = true;\n trigger(self);\n }\n };\n\n // Run once synchronously to establish deps and cache initial value\n this._effect.run();\n this._dirty = false;\n }\n\n get value(): T {\n track(this);\n\n if (this._dirty) {\n this._effect.run();\n this._dirty = false;\n }\n\n return this._value;\n }\n\n peek(): T {\n if (this._dirty) {\n this._effect.run();\n this._dirty = false;\n }\n return this._value;\n }\n\n /**\n * toString() for template interpolation.\n */\n toString(): string {\n return String(this.value);\n }\n\n /**\n * valueOf() for numeric operations.\n */\n valueOf(): T {\n return this.value;\n }\n}\n\n// ─── Public API ────────────────────────────────────────────\n\n/**\n * Create a computed reactive value.\n *\n * The getter function is tracked — when any signal it reads changes,\n * the computed is marked dirty and will re-evaluate on next `.value` read.\n *\n * @example\n * const count = signal(2);\n * const doubled = computed(() => count.value * 2);\n *\n * console.log(doubled.value); // 4\n *\n * count.value = 5;\n * console.log(doubled.value); // 10\n */\nexport function computed<T>(getter: () => T): Computed<T> {\n return new ComputedImpl(getter);\n}\n\n/**\n * Type guard: check if a value is a Computed.\n */\nexport function isComputed(value: unknown): value is Computed<unknown> {\n return (\n value != null &&\n typeof value === \"object\" &&\n COMPUTED_BRAND in (value as any)\n );\n}\n",
|
|
11
|
+
"/// <reference lib=\"dom\" />\n\n/**\n * SinwanJS Client Renderer — DOM Operations\n *\n * Thin abstraction over native DOM APIs for testability\n * and potential future server-side DOM (e.g., happy-dom, linkedom).\n */\n\nexport interface DOMOps {\n createElement(tag: string): Element;\n createElementNS(namespace: string, tag: string): Element;\n createTextNode(text: string): Text;\n createComment(text: string): Comment;\n setAttribute(el: Element, key: string, value: string): void;\n removeAttribute(el: Element, key: string): void;\n setProperty(el: Element, key: string, value: unknown): void;\n insertBefore(parent: Node, child: Node, anchor: Node | null): void;\n appendChild(parent: Node, child: Node): void;\n remove(node: Node): void;\n setTextContent(node: Text, text: string): void;\n addEventListener(el: Element, event: string, handler: EventListener): void;\n removeEventListener(el: Element, event: string, handler: EventListener): void;\n parentNode(node: Node): Node | null;\n nextSibling(node: Node): Node | null;\n}\n\nfunction createDefaultDOMOps(): DOMOps {\n return {\n createElement(tag: string): Element {\n return document.createElement(tag);\n },\n\n createElementNS(namespace: string, tag: string): Element {\n return document.createElementNS(namespace, tag);\n },\n\n createTextNode(text: string): Text {\n return document.createTextNode(text);\n },\n\n createComment(text: string): Comment {\n return document.createComment(text);\n },\n\n setAttribute(el: Element, key: string, value: string): void {\n el.setAttribute(key, value);\n },\n\n removeAttribute(el: Element, key: string): void {\n el.removeAttribute(key);\n },\n\n setProperty(el: Element, key: string, value: unknown): void {\n (el as any)[key] = value;\n },\n\n insertBefore(parent: Node, child: Node, anchor: Node | null): void {\n parent.insertBefore(child, anchor);\n },\n\n appendChild(parent: Node, child: Node): void {\n parent.appendChild(child);\n },\n\n remove(node: Node): void {\n node.parentNode?.removeChild(node);\n },\n\n setTextContent(node: Text, text: string): void {\n node.data = text;\n },\n\n addEventListener(el: Element, event: string, handler: EventListener): void {\n el.addEventListener(event, handler);\n },\n\n removeEventListener(el: Element, event: string, handler: EventListener): void {\n el.removeEventListener(event, handler);\n },\n\n parentNode(node: Node): Node | null {\n return node.parentNode;\n },\n\n nextSibling(node: Node): Node | null {\n return node.nextSibling;\n },\n };\n}\n\nconst defaultDOMOps = createDefaultDOMOps();\n\n/**\n * Live DOM operations object used by the renderer.\n */\nexport const domOps: DOMOps = { ...defaultDOMOps };\n\nexport function setDOMOps(overrides: Partial<DOMOps>): void {\n Object.assign(domOps, overrides);\n}\n\nexport function resetDOMOps(): void {\n for (const key of Object.keys(domOps) as (keyof DOMOps)[]) {\n delete (domOps as any)[key];\n }\n Object.assign(domOps, defaultDOMOps);\n}\n",
|
|
12
|
+
"/// <reference lib=\"dom\" />\n\n/**\n * SinwanJS Client Renderer — Event Binding\n *\n * Direct event binding (not delegation). Each handler is attached\n * directly to its target element for simplicity and easy hydration.\n *\n * Design decision: direct binding like Solid.js, not delegation like React.\n */\n\nimport { domOps } from \"./dom-ops.ts\";\nimport type { CleanupFn } from \"../reactivity/index.ts\";\n\n/**\n * Check if a prop key is an event handler (starts with \"on\").\n */\nexport function isEventProp(key: string): boolean {\n return key.length > 2 && key[0] === \"o\" && key[1] === \"n\" && key[2]! >= \"A\" && key[2]! <= \"Z\";\n}\n\n/**\n * Extract the DOM event name from a prop key.\n * e.g., \"onClick\" → \"click\", \"onMouseEnter\" → \"mouseenter\"\n */\nexport function toEventName(key: string): string {\n return key.slice(2).toLowerCase();\n}\n\n/**\n * Bind an event handler to an element.\n * Returns a cleanup function to remove the listener.\n */\nexport function bindEvent(\n el: Element,\n eventName: string,\n handler: EventListener,\n): CleanupFn {\n domOps.addEventListener(el, eventName, handler);\n return () => {\n domOps.removeEventListener(el, eventName, handler);\n };\n}\n\n/**\n * Bind all event props from an element's props object.\n * Returns an array of cleanup functions.\n */\nexport function bindEvents(\n el: Element,\n props: Record<string, unknown>,\n): CleanupFn[] {\n const cleanups: CleanupFn[] = [];\n\n for (const key of Object.keys(props)) {\n if (isEventProp(key)) {\n const handler = props[key];\n if (typeof handler === \"function\") {\n const eventName = toEventName(key);\n cleanups.push(bindEvent(el, eventName, handler as EventListener));\n }\n }\n }\n\n return cleanups;\n}\n",
|
|
13
|
+
"import type { Reactive, SinwanElement, SinwanNode } from \"../types.ts\";\n\nexport const SHOW_TYPE = Symbol.for(\"Sinwan.Show\");\nexport const FOR_TYPE = Symbol.for(\"Sinwan.For\");\n\nexport interface ShowProps<T> {\n when: Reactive<T | false | null | undefined>;\n fallback?: SinwanNode;\n children?: SinwanNode | ((value: NonNullable<T>) => SinwanNode);\n}\n\nexport interface ForProps<T> {\n each: Reactive<readonly T[]>;\n key?: (item: T, index: number) => string | number | symbol;\n children?: (item: T, index: () => number) => SinwanNode;\n}\n\nexport function Show<T>(props: ShowProps<T>): SinwanElement {\n return {\n tag: SHOW_TYPE,\n props: props as unknown as Record<string, unknown>,\n children: [],\n };\n}\n\nexport function For<T>(props: ForProps<T>): SinwanElement {\n return {\n tag: FOR_TYPE,\n props: props as unknown as Record<string, unknown>,\n children: [],\n };\n}\n\nexport function isShowElement(element: SinwanElement): boolean {\n return element.tag === SHOW_TYPE;\n}\n\nexport function isForElement(element: SinwanElement): boolean {\n return element.tag === FOR_TYPE;\n}\n",
|
|
14
|
+
"/**\n * SinwanJS View Module — Renderer & Component Registry\n *\n * Renders Sinwan component trees to HTML strings.\n * Supports async components, caching, and streaming.\n */\n\nimport type {\n SinwanNode,\n SinwanElement,\n SinwanComponent,\n SinwanPage,\n SinwanSlots,\n} from \"../types.ts\";\nimport { HtmlEscapedString, escapeHtml } from \"../escaper.ts\";\nimport { isSignal } from \"../reactivity/signal.ts\";\nimport { isComputed } from \"../reactivity/computed.ts\";\nimport { isEventProp } from \"../renderer/events.ts\";\nimport { isForElement, isShowElement } from \"../component/control-flow.ts\";\n\n// Component cache - maps component identity to render function\nconst componentCache = new WeakMap<SinwanComponent<any>, boolean>();\n\n// Page registry\nconst pageRegistry = new Map<string, SinwanPage<any>>();\n\n/**\n * Register a page renderer by name.\n */\nexport function registerPage<D extends object = {}>(\n name: string,\n page: SinwanPage<D>,\n): void {\n pageRegistry.set(name, page);\n}\n\n/**\n * Get a registered page by name.\n */\nexport function getPage<D extends object = {}>(\n name: string,\n): SinwanPage<D> | undefined {\n return pageRegistry.get(name);\n}\n\n/**\n * Check if a page is registered.\n */\nexport function hasPage(name: string): boolean {\n return pageRegistry.has(name);\n}\n\n/**\n * Render a registered page to an HTML string.\n */\nexport async function renderPage<D extends object = {}>(\n name: string,\n data: D,\n): Promise<string> {\n const page = getPage<D>(name);\n if (!page) {\n throw new Error(`Page \"${name}\" not found in registry`);\n }\n\n const element = await page(data);\n return renderToString(element);\n}\n\n/**\n * Render a node tree to an HTML string.\n * Handles primitives, elements, components, and arrays.\n */\nexport async function renderToString(node: SinwanNode): Promise<string> {\n // Handle null/undefined/boolean\n if (node == null || typeof node === \"boolean\") {\n return \"\";\n }\n\n // Handle strings (escape them)\n if (typeof node === \"string\") {\n return escapeHtml(node);\n }\n\n // Handle numbers\n if (typeof node === \"number\") {\n return String(node);\n }\n\n // Handle pre-escaped HTML\n if (node instanceof HtmlEscapedString) {\n return node.value;\n }\n\n // Handle reactive containers — read current value and render as text\n if (isSignal(node) || isComputed(node)) {\n return escapeHtml(String((node as any).value));\n }\n\n // Handle arrays - render each child and concatenate\n if (Array.isArray(node)) {\n const results = await Promise.all(\n node.map((child) => renderToString(child)),\n );\n return results.join(\"\");\n }\n\n // Handle promises (async components)\n if (node instanceof Promise) {\n return renderElement(await node);\n }\n\n // Handle elements\n return renderElement(node as SinwanElement);\n}\n\n/**\n * Render an element to HTML string.\n */\nasync function renderElement(element: SinwanElement): Promise<string> {\n const { tag, props, children } = element;\n\n if (isShowElement(element)) {\n const when = readReactive(props.when);\n return renderToString(\n when ? resolveShowChildren(element, when) : props.fallback as SinwanNode,\n );\n }\n\n if (isForElement(element)) {\n return renderForElement(element);\n }\n\n // Handle functional components\n if (typeof tag === \"function\") {\n const result = await tag(props);\n return renderToString(result);\n }\n\n // Handle intrinsic HTML elements\n if (typeof tag === \"string\") {\n return renderIntrinsicElement(tag, props, children);\n }\n\n // Fallback - shouldn't happen with valid JSX\n return renderToString(children);\n}\n\n// Void elements that don't have closing tags\nconst VOID_ELEMENTS = new Set([\n \"area\",\n \"base\",\n \"br\",\n \"col\",\n \"embed\",\n \"hr\",\n \"img\",\n \"input\",\n \"link\",\n \"meta\",\n \"param\",\n \"source\",\n \"track\",\n \"wbr\",\n]);\n\n/**\n * Render an intrinsic HTML element.\n */\nasync function renderIntrinsicElement(\n tag: string,\n props: Record<string, unknown>,\n children: SinwanNode[],\n): Promise<string> {\n const attrs = renderAttributes(props);\n\n // Void elements have no children and no closing tag\n if (VOID_ELEMENTS.has(tag)) {\n return attrs ? `<${tag}${attrs}>` : `<${tag}>`;\n }\n\n // Render children (handles dangerouslySetInnerHTML)\n const childrenHtml = await renderChildren(children, props);\n\n // Build element\n return attrs\n ? `<${tag}${attrs}>${childrenHtml}</${tag}>`\n : `<${tag}>${childrenHtml}</${tag}>`;\n}\n\n/**\n * Render HTML attributes from props.\n */\nfunction renderAttributes(props: Record<string, unknown>): string {\n let attrs = \"\";\n\n for (const [key, value] of Object.entries(props)) {\n // Skip children and special props\n if (\n key === \"children\" ||\n key === \"key\" ||\n key === \"ref\" ||\n key === \"dangerouslySetInnerHTML\" ||\n isEventProp(key)\n ) {\n continue;\n }\n\n const resolvedValue = readReactive(value);\n\n // Skip null/undefined/false values\n if (resolvedValue == null || resolvedValue === false) continue;\n\n // Handle className -> class\n const attrName = key === \"className\" ? \"class\" : key;\n\n // Handle htmlFor -> for\n const finalName = attrName === \"htmlFor\" ? \"for\" : attrName;\n\n // Handle boolean true (just the attribute name)\n if (resolvedValue === true) {\n attrs += ` ${finalName}`;\n continue;\n }\n\n // Escape the attribute value\n const attrValue = escapeHtml(String(resolvedValue));\n attrs += ` ${finalName}=\"${attrValue}\"`;\n }\n\n return attrs;\n}\n\n/**\n * Render children, with special handling for dangerouslySetInnerHTML.\n */\nasync function renderChildren(\n children: SinwanNode[],\n props: Record<string, unknown>,\n): Promise<string> {\n // Check for dangerous inner HTML\n const dangerous = props.dangerouslySetInnerHTML as\n | { __html?: string }\n | undefined;\n if (dangerous && typeof dangerous.__html === \"string\") {\n return dangerous.__html; // Trust the HTML (user explicitly marked safe)\n }\n\n return renderToString(children);\n}\n\n// Wire up dangerouslySetInnerHTML handling by patching renderIntrinsicElement\nconst originalRenderIntrinsic = renderIntrinsicElement;\n\n/**\n * Check if children is a slots object (named slots).\n */\nexport function isSlots(children: unknown): children is SinwanSlots {\n return (\n children != null &&\n typeof children === \"object\" &&\n !Array.isArray(children) &&\n !(children instanceof HtmlEscapedString)\n );\n}\n\nasync function renderForElement(element: SinwanElement): Promise<string> {\n const props = element.props as {\n each?: unknown;\n children?: (item: unknown, index: () => number) => SinwanNode;\n };\n const each = readReactive(props.each);\n if (!Array.isArray(each) || typeof props.children !== \"function\") {\n return \"\";\n }\n\n const rendered = await Promise.all(\n each.map((item, index) => renderToString(props.children!(item, () => index))),\n );\n return rendered.join(\"\");\n}\n\nfunction resolveShowChildren(element: SinwanElement, value: unknown): SinwanNode {\n const children = (element.props as any).children ?? element.children;\n if (typeof children === \"function\") {\n return children(value);\n }\n return children as SinwanNode;\n}\n\nfunction readReactive(value: unknown): unknown {\n return isSignal(value) || isComputed(value) ? (value as any).value : value;\n}\n",
|
|
15
|
+
"/**\n * SinwanJS Component Runtime — Instance Management\n *\n * Each component rendered on the client gets a ComponentInstance\n * that tracks its lifecycle hooks, effects, parent/child tree,\n * and provide/inject context.\n *\n * A global `currentInstance` stack lets lifecycle hooks (onMounted, etc.)\n * register themselves during setup or synchronous lifecycle callbacks —\n * same pattern as Vue's getCurrentInstance.\n */\n\nimport type { SinwanComponent } from \"../types.ts\";\nimport type { MountedNode } from \"../renderer/types.ts\";\nimport type { CleanupFn } from \"../reactivity/index.ts\";\nimport { nextTick } from \"../reactivity/scheduler.ts\";\n\n// ─── ComponentInstance ─────────────────────────────────────\n\nlet uidCounter = 0;\n\nexport interface ComponentInstance {\n /** Unique identifier for this instance. */\n uid: number;\n\n /** The component definition (setup function). */\n component: SinwanComponent<any>;\n\n /** Props passed to this component. */\n props: Record<string, any>;\n\n /** The rendered DOM subtree (set after render). */\n element: MountedNode | null;\n\n /** Parent instance in the component tree. */\n parent: ComponentInstance | null;\n\n /** Child component instances. */\n children: ComponentInstance[];\n\n /** All effect dispose functions owned by this component. */\n effects: CleanupFn[];\n\n // ─── Lifecycle hook queues ────────────────────────────\n\n /** Callbacks to fire after the component is mounted to DOM. */\n _mountedHooks: (() => void)[];\n\n /** Callbacks to fire when the component is unmounted. */\n _unmountedHooks: (() => void)[];\n\n /** Callbacks to fire after any reactive update in this component. */\n _updatedHooks: (() => void)[];\n\n /** Error handler callbacks. */\n _errorHooks: ((err: Error) => void)[];\n\n // ─── Provide/Inject context ───────────────────────────\n\n /** Values provided by this instance (for inject in children). */\n provides: Record<string | symbol, unknown>;\n\n // ─── State flags ──────────────────────────────────────\n\n isMounted: boolean;\n isUnmounted: boolean;\n}\n\n/**\n * Create a fresh ComponentInstance.\n */\nexport function createComponentInstance(\n component: SinwanComponent<any>,\n props: Record<string, any>,\n parent: ComponentInstance | null,\n): ComponentInstance {\n return {\n uid: uidCounter++,\n component,\n props,\n element: null,\n parent,\n children: [],\n effects: [],\n _mountedHooks: [],\n _unmountedHooks: [],\n _updatedHooks: [],\n _errorHooks: [],\n // Inherit parent's provides (prototype chain for lookup)\n provides: parent ? Object.create(parent.provides) : Object.create(null),\n isMounted: false,\n isUnmounted: false,\n };\n}\n\n// ─── Current instance stack ────────────────────────────────\n\nlet currentInstance: ComponentInstance | null = null;\n\n/**\n * Get the currently active component instance.\n * Used by lifecycle hooks to register themselves.\n */\nexport function getCurrentInstance(): ComponentInstance | null {\n return currentInstance;\n}\n\n/**\n * Set the current instance (called by renderer before setup).\n * Returns the previous instance for restoration.\n */\nexport function setCurrentInstance(\n instance: ComponentInstance | null,\n): ComponentInstance | null {\n const prev = currentInstance;\n currentInstance = instance;\n return prev;\n}\n\n/**\n * Run a function with `instance` as the current component instance.\n * Automatically restores the previous instance when done.\n */\nexport function withInstance<T>(instance: ComponentInstance, fn: () => T): T {\n const prev = setCurrentInstance(instance);\n try {\n return fn();\n } finally {\n setCurrentInstance(prev);\n }\n}\n\n// ─── Lifecycle execution ───────────────────────────────────\n\n/**\n * Fire all onMounted hooks for an instance and its children (depth-first).\n */\nexport function fireMountedHooks(instance: ComponentInstance): void {\n if (instance.isUnmounted) {\n return;\n }\n\n // Children first (bottom-up, like Vue)\n for (const child of instance.children) {\n fireMountedHooks(child);\n }\n\n if (!instance.isMounted) {\n instance.isMounted = true;\n for (const hook of instance._mountedHooks) {\n hook();\n }\n }\n}\n\n/**\n * Fire all onUnmounted hooks and dispose all effects for an instance\n * and its children (depth-first, children first).\n */\nexport function fireUnmountedHooks(instance: ComponentInstance): void {\n // Children first\n for (const child of instance.children) {\n fireUnmountedHooks(child);\n }\n\n if (instance.isMounted && !instance.isUnmounted) {\n instance.isUnmounted = true;\n instance.isMounted = false;\n\n // Fire unmounted hooks\n for (const hook of instance._unmountedHooks) {\n hook();\n }\n\n // Dispose all effects owned by this component\n for (const dispose of instance.effects) {\n dispose();\n }\n instance.effects.length = 0;\n }\n}\n\n/**\n * Fire onUpdated hooks for the current instance.\n */\nexport function fireUpdatedHooks(instance: ComponentInstance): void {\n for (const hook of instance._updatedHooks) {\n hook();\n }\n}\n\nconst queuedUpdatedHooks = new Set<ComponentInstance>();\n\n/**\n * Queue onUpdated hooks to run after the current reactive flush.\n * Multiple DOM effects in the same flush produce one updated callback.\n */\nexport function queueUpdatedHooks(instance: ComponentInstance | null): void {\n if (\n !instance ||\n !instance.isMounted ||\n instance.isUnmounted ||\n instance._updatedHooks.length === 0 ||\n queuedUpdatedHooks.has(instance)\n ) {\n return;\n }\n\n queuedUpdatedHooks.add(instance);\n nextTick(() => {\n queuedUpdatedHooks.delete(instance);\n if (instance.isMounted && !instance.isUnmounted) {\n fireUpdatedHooks(instance);\n }\n });\n}\n\n/**\n * Handle an error in the component tree — walks up to find an error handler.\n */\nexport function handleComponentError(\n instance: ComponentInstance,\n err: Error,\n): void {\n let current: ComponentInstance | null = instance;\n while (current) {\n if (current._errorHooks.length > 0) {\n for (const hook of current._errorHooks) {\n hook(err);\n }\n return;\n }\n current = current.parent;\n }\n // No handler found — re-throw\n console.error(\"[Sinwan] Unhandled component error:\", err);\n}\n",
|
|
16
|
+
"/// <reference lib=\"dom\" />\n\n/**\n * SinwanJS Hydration — Marker Protocol\n *\n * Constants and helpers for the hydration marker format:\n *\n * data-sinwan-id=\"c0\" — component boundary\n * <!--sinwan-t:0-->val<!--/sinwan-t--> — reactive text boundary\n * data-sinwan-ev=\"click:0\" — event binding reference\n */\n\n// ─── Constants ─────────────────────────────────────────────\n\n/** Attribute on the root element of each component instance. */\nexport const COMP_ID_ATTR = \"data-sinwan-id\";\n\n/** Prefix for component IDs. */\nexport const COMP_ID_PREFIX = \"c\";\n\n/** Opening comment prefix for reactive text slots: `Sinwan-t:N` */\nexport const TEXT_MARKER_OPEN = \"sinwan-t:\";\n\n/** Closing comment for reactive text slots. */\nexport const TEXT_MARKER_CLOSE = \"/sinwan-t\";\n\n/** Attribute for event binding references. */\nexport const EVENT_ATTR = \"data-sinwan-ev\";\n\n// ─── Server-side marker generation ────────────────────────\n\n/** Build a component ID string, e.g. `\"c0\"`. */\nexport function compId(index: number): string {\n return `${COMP_ID_PREFIX}${index}`;\n}\n\n/** Build an opening text marker comment string. */\nexport function textMarkerOpen(index: number): string {\n return `<!--${TEXT_MARKER_OPEN}${index}-->`;\n}\n\n/** Build a closing text marker comment string. */\nexport function textMarkerCloseStr(): string {\n return `<!--${TEXT_MARKER_CLOSE}-->`;\n}\n\n/** Build an event attribute value, e.g. `\"click:0\"`. */\nexport function eventAttrValue(event: string, index: number): string {\n return `${event}:${index}`;\n}\n\n// ─── Client-side marker parsing ───────────────────────────\n\n/**\n * Check if a comment node is a reactive text opening marker.\n * Returns the slot index, or -1 if not a marker.\n */\nexport function parseTextOpenMarker(node: Comment): number {\n const data = node.data;\n if (data.startsWith(TEXT_MARKER_OPEN)) {\n const idx = parseInt(data.slice(TEXT_MARKER_OPEN.length), 10);\n return Number.isNaN(idx) ? -1 : idx;\n }\n return -1;\n}\n\n/**\n * Check if a comment node is a reactive text closing marker.\n */\nexport function isTextCloseMarker(node: Comment): boolean {\n return node.data === TEXT_MARKER_CLOSE;\n}\n\n/**\n * Parse `data-sinwan-ev` attribute value into event entries.\n * Format: `\"click:0\"` or `\"click:0,input:1\"` for multiple.\n * Returns array of `[eventName, handlerIndex]` tuples.\n */\nexport function parseEventAttr(value: string): [string, number][] {\n return value.split(\",\").map((pair) => {\n const [event, idx] = pair.split(\":\");\n return [event!, parseInt(idx!, 10)];\n });\n}\n\n/**\n * Parse `data-sinwan-id` into the component index.\n * e.g., `\"c3\"` → `3`\n */\nexport function parseCompId(value: string): number {\n return parseInt(value.slice(COMP_ID_PREFIX.length), 10);\n}\n",
|
|
17
|
+
"/**\n * SinwanJS View Module — Streaming SSR\n *\n * Progressive HTML streaming using Bun's native ReadableStream.\n * Streams chunks as they resolve without waiting for full tree.\n */\n\nimport type {\n SinwanNode,\n SinwanElement,\n SinwanPage,\n SinwanComponent,\n} from \"../types.ts\";\nimport { HtmlEscapedString, escapeHtml } from \"../escaper.ts\";\nimport { isSignal } from \"../reactivity/signal.ts\";\nimport { isComputed } from \"../reactivity/computed.ts\";\nimport { isEventProp, toEventName } from \"../renderer/events.ts\";\nimport {\n For,\n Show,\n isForElement,\n isShowElement,\n} from \"../component/control-flow.ts\";\nimport {\n createComponentInstance,\n getCurrentInstance,\n setCurrentInstance,\n} from \"../component/instance.ts\";\nimport {\n compId,\n textMarkerOpen,\n textMarkerCloseStr,\n COMP_ID_ATTR,\n EVENT_ATTR,\n} from \"../hydration/markers.ts\";\n\ninterface HydratableStreamContext {\n componentIndex: number;\n textIndex: number;\n eventIndex: number;\n}\n\nfunction createHydratableStreamContext(): HydratableStreamContext {\n return { componentIndex: 0, textIndex: 0, eventIndex: 0 };\n}\n\n// Void elements that don't have closing tags\nconst VOID_ELEMENTS = new Set([\n \"area\",\n \"base\",\n \"br\",\n \"col\",\n \"embed\",\n \"hr\",\n \"img\",\n \"input\",\n \"link\",\n \"meta\",\n \"param\",\n \"source\",\n \"track\",\n \"wbr\",\n]);\n\n/**\n * Stream a page to a ReadableStream.\n */\nexport function streamPage<D extends object = {}>(\n page: SinwanPage<D>,\n data: D,\n): ReadableStream<Uint8Array> {\n const encoder = new TextEncoder();\n\n return new ReadableStream({\n async start(controller) {\n try {\n // Resolve the page component\n const element = await page(data);\n\n // Stream the element tree\n await streamNode(element, controller, encoder);\n\n // Close the stream\n controller.close();\n } catch (error) {\n controller.error(error);\n }\n },\n });\n}\n\n/**\n * Stream a component with hydration markers.\n */\nexport function streamHydratablePage(\n component: SinwanComponent<any>,\n props?: Record<string, unknown>,\n): ReadableStream<Uint8Array> {\n const encoder = new TextEncoder();\n const ctx = createHydratableStreamContext();\n\n return new ReadableStream({\n async start(controller) {\n try {\n await streamHydratableComponent(\n component,\n props ?? {},\n controller,\n encoder,\n ctx,\n true,\n );\n controller.close();\n } catch (error) {\n controller.error(error);\n }\n },\n });\n}\n\n/**\n * Stream a raw node tree with hydration markers.\n */\nexport function streamHydratableNode(\n node: SinwanNode,\n): ReadableStream<Uint8Array> {\n const encoder = new TextEncoder();\n const ctx = createHydratableStreamContext();\n\n return new ReadableStream({\n async start(controller) {\n try {\n await streamHydratableNodeToController(node, controller, encoder, ctx);\n controller.close();\n } catch (error) {\n controller.error(error);\n }\n },\n });\n}\n\n/**\n * Stream a node tree to a controller.\n */\nasync function streamNode(\n node: SinwanNode,\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n): Promise<void> {\n // Handle null/undefined/boolean\n if (node == null || typeof node === \"boolean\") {\n return;\n }\n\n // Handle strings (escape them)\n if (typeof node === \"string\") {\n controller.enqueue(encoder.encode(escapeHtml(node)));\n return;\n }\n\n // Handle numbers\n if (typeof node === \"number\") {\n controller.enqueue(encoder.encode(String(node)));\n return;\n }\n\n // Handle pre-escaped HTML\n if (node instanceof HtmlEscapedString) {\n controller.enqueue(encoder.encode(node.value));\n return;\n }\n\n // Handle reactive containers — stream the escaped current value\n if (isSignal(node) || isComputed(node)) {\n controller.enqueue(encoder.encode(escapeHtml(String((node as any).value))));\n return;\n }\n\n // Handle arrays - stream each child\n if (Array.isArray(node)) {\n for (const child of node) {\n await streamNode(child, controller, encoder);\n }\n return;\n }\n\n // Handle async elements (Promise<SinwanElement>)\n if (node instanceof Promise) {\n const resolved = await node;\n await streamElement(resolved, controller, encoder);\n return;\n }\n\n // Handle elements\n await streamElement(node as SinwanElement, controller, encoder);\n}\n\n/**\n * Stream an element to the controller.\n */\nasync function streamElement(\n element: SinwanElement,\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n): Promise<void> {\n const { tag, props, children } = element;\n\n if (isShowElement(element)) {\n const when = readReactive(props.when);\n await streamNode(\n when ? resolveShowChildren(element, when) : (props.fallback as SinwanNode),\n controller,\n encoder,\n );\n return;\n }\n\n if (isForElement(element)) {\n await streamForElement(element, controller, encoder);\n return;\n }\n\n // Handle functional components\n if (typeof tag === \"function\") {\n const result = await tag(props);\n await streamNode(result, controller, encoder);\n return;\n }\n\n // Handle intrinsic HTML elements\n if (typeof tag === \"string\") {\n await streamIntrinsicElement(tag, props, children, controller, encoder);\n return;\n }\n\n // Fallback\n await streamNode(children, controller, encoder);\n}\n\n/**\n * Stream an intrinsic HTML element.\n */\nasync function streamIntrinsicElement(\n tag: string,\n props: Record<string, unknown>,\n children: SinwanNode[],\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n): Promise<void> {\n const attrs = renderAttributes(props);\n\n // Check for dangerous inner HTML\n const dangerous = props.dangerouslySetInnerHTML as\n | { __html?: string }\n | undefined;\n\n // Void elements have no children and no closing tag\n if (VOID_ELEMENTS.has(tag)) {\n const html = attrs ? `<${tag}${attrs}>` : `<${tag}>`;\n controller.enqueue(encoder.encode(html));\n return;\n }\n\n // Opening tag\n const openTag = attrs ? `<${tag}${attrs}>` : `<${tag}>`;\n controller.enqueue(encoder.encode(openTag));\n\n // Children or dangerous HTML\n if (dangerous && typeof dangerous.__html === \"string\") {\n controller.enqueue(encoder.encode(dangerous.__html));\n } else {\n await streamNode(children, controller, encoder);\n }\n\n // Closing tag\n controller.enqueue(encoder.encode(`</${tag}>`));\n}\n\n/**\n * Render HTML attributes from props.\n */\nfunction renderAttributes(props: Record<string, unknown>): string {\n let attrs = \"\";\n\n for (const [key, value] of Object.entries(props)) {\n if (\n key === \"children\" ||\n key === \"key\" ||\n key === \"ref\" ||\n key === \"dangerouslySetInnerHTML\" ||\n isEventProp(key)\n ) {\n continue;\n }\n\n const resolvedValue = readReactive(value);\n if (resolvedValue == null || resolvedValue === false) continue;\n\n const attrName =\n key === \"className\" ? \"class\" : key === \"htmlFor\" ? \"for\" : key;\n\n if (resolvedValue === true) {\n attrs += ` ${attrName}`;\n continue;\n }\n\n const attrValue = escapeHtml(String(resolvedValue));\n attrs += ` ${attrName}=\"${attrValue}\"`;\n }\n\n return attrs;\n}\n\nasync function streamHydratableNodeToController(\n node: SinwanNode,\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n ctx: HydratableStreamContext,\n isComponentRoot = false,\n): Promise<void> {\n if (node == null || typeof node === \"boolean\") {\n return;\n }\n\n if (typeof node === \"string\") {\n enqueue(controller, encoder, escapeHtml(node));\n return;\n }\n\n if (typeof node === \"number\") {\n enqueue(controller, encoder, String(node));\n return;\n }\n\n if (node instanceof HtmlEscapedString) {\n enqueue(controller, encoder, node.value);\n return;\n }\n\n if (isSignal(node) || isComputed(node)) {\n const idx = ctx.textIndex++;\n enqueue(\n controller,\n encoder,\n `${textMarkerOpen(idx)}${escapeHtml(String((node as any).value))}${textMarkerCloseStr()}`,\n );\n return;\n }\n\n if (Array.isArray(node)) {\n for (const child of node) {\n await streamHydratableNodeToController(child, controller, encoder, ctx);\n }\n return;\n }\n\n if (node instanceof Promise) {\n await streamHydratableElement(\n await node,\n controller,\n encoder,\n ctx,\n isComponentRoot,\n );\n return;\n }\n\n await streamHydratableElement(\n node as SinwanElement,\n controller,\n encoder,\n ctx,\n isComponentRoot,\n );\n}\n\nasync function streamHydratableElement(\n element: SinwanElement,\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n ctx: HydratableStreamContext,\n isComponentRoot: boolean,\n): Promise<void> {\n const { tag, props, children } = element;\n\n if (tag === \"\") {\n for (const child of children) {\n await streamHydratableNodeToController(child, controller, encoder, ctx);\n }\n return;\n }\n\n if (tag === Show || tag === For) {\n await streamHydratableElement(\n (tag as Function)(props),\n controller,\n encoder,\n ctx,\n isComponentRoot,\n );\n return;\n }\n\n if (isShowElement(element)) {\n const when = readReactive(props.when);\n await streamHydratableNodeToController(\n when ? resolveShowChildren(element, when) : (props.fallback as SinwanNode),\n controller,\n encoder,\n ctx,\n isComponentRoot,\n );\n return;\n }\n\n if (isForElement(element)) {\n await streamHydratableForElement(element, controller, encoder, ctx);\n return;\n }\n\n if (typeof tag === \"function\") {\n await streamHydratableComponent(\n tag as SinwanComponent<any>,\n props,\n controller,\n encoder,\n ctx,\n true,\n );\n return;\n }\n\n if (typeof tag === \"string\") {\n await streamHydratableIntrinsic(\n tag,\n props,\n children,\n controller,\n encoder,\n ctx,\n isComponentRoot,\n );\n return;\n }\n\n await streamHydratableNodeToController(children, controller, encoder, ctx);\n}\n\nasync function streamHydratableComponent(\n component: SinwanComponent<any>,\n props: Record<string, unknown>,\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n ctx: HydratableStreamContext,\n isComponentRoot: boolean,\n): Promise<void> {\n const parentInstance = getCurrentInstance();\n const instance = createComponentInstance(component, props, parentInstance);\n if (parentInstance) {\n parentInstance.children.push(instance);\n }\n\n const prev = setCurrentInstance(instance);\n try {\n const result = await component(props);\n await streamHydratableNodeToController(\n result,\n controller,\n encoder,\n ctx,\n isComponentRoot,\n );\n } finally {\n setCurrentInstance(prev);\n }\n}\n\nasync function streamHydratableIntrinsic(\n tag: string,\n props: Record<string, unknown>,\n children: SinwanNode[],\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n ctx: HydratableStreamContext,\n isComponentRoot: boolean,\n): Promise<void> {\n const attrs = renderHydratableAttributes(props, ctx, isComponentRoot);\n const dangerous = props.dangerouslySetInnerHTML as\n | { __html?: string }\n | undefined;\n\n enqueue(controller, encoder, attrs ? `<${tag}${attrs}>` : `<${tag}>`);\n\n if (VOID_ELEMENTS.has(tag)) {\n return;\n }\n\n if (dangerous && typeof dangerous.__html === \"string\") {\n enqueue(controller, encoder, dangerous.__html);\n } else {\n for (const child of children) {\n await streamHydratableNodeToController(child, controller, encoder, ctx);\n }\n }\n\n enqueue(controller, encoder, `</${tag}>`);\n}\n\nasync function streamHydratableForElement(\n element: SinwanElement,\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n ctx: HydratableStreamContext,\n): Promise<void> {\n const props = element.props as {\n each?: unknown;\n children?: (item: unknown, index: () => number) => SinwanNode;\n };\n const each = readReactive(props.each);\n if (!Array.isArray(each) || typeof props.children !== \"function\") {\n return;\n }\n\n for (let index = 0; index < each.length; index++) {\n await streamHydratableNodeToController(\n props.children(each[index], () => index),\n controller,\n encoder,\n ctx,\n );\n }\n}\n\nfunction renderHydratableAttributes(\n props: Record<string, unknown>,\n ctx: HydratableStreamContext,\n isComponentRoot: boolean,\n): string {\n let attrs = \"\";\n\n if (isComponentRoot) {\n attrs += ` ${COMP_ID_ATTR}=\"${compId(ctx.componentIndex++)}\"`;\n }\n\n const eventParts: string[] = [];\n\n for (const [key, value] of Object.entries(props)) {\n if (\n key === \"children\" ||\n key === \"key\" ||\n key === \"ref\" ||\n key === \"dangerouslySetInnerHTML\"\n ) {\n continue;\n }\n\n if (isEventProp(key)) {\n eventParts.push(`${toEventName(key)}:${ctx.eventIndex++}`);\n continue;\n }\n\n const resolvedValue = readReactive(value);\n if (resolvedValue == null || resolvedValue === false) continue;\n\n const attrName =\n key === \"className\" ? \"class\" : key === \"htmlFor\" ? \"for\" : key;\n\n if (resolvedValue === true) {\n attrs += ` ${attrName}`;\n continue;\n }\n\n attrs += ` ${attrName}=\"${escapeHtml(String(resolvedValue))}\"`;\n }\n\n if (eventParts.length > 0) {\n attrs += ` ${EVENT_ATTR}=\"${eventParts.join(\",\")}\"`;\n }\n\n return attrs;\n}\n\nfunction enqueue(\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n html: string,\n): void {\n controller.enqueue(encoder.encode(html));\n}\n\nasync function streamForElement(\n element: SinwanElement,\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n): Promise<void> {\n const props = element.props as {\n each?: unknown;\n children?: (item: unknown, index: () => number) => SinwanNode;\n };\n const each = readReactive(props.each);\n if (!Array.isArray(each) || typeof props.children !== \"function\") {\n return;\n }\n\n for (let index = 0; index < each.length; index++) {\n await streamNode(props.children(each[index], () => index), controller, encoder);\n }\n}\n\nfunction resolveShowChildren(element: SinwanElement, value: unknown): SinwanNode {\n const children = (element.props as any).children ?? element.children;\n if (typeof children === \"function\") {\n return children(value);\n }\n return children as SinwanNode;\n}\n\nfunction readReactive(value: unknown): unknown {\n return isSignal(value) || isComputed(value) ? (value as any).value : value;\n}\n",
|
|
18
|
+
"/**\n * SinwanJS Server — Hydration-Aware SSR Renderer\n *\n * Enhanced `renderToString` that injects hydration markers:\n *\n * data-sinwan-id=\"c0\" — component boundary\n * <!--sinwan-t:0-->val<!--/sinwan-t--> — reactive text boundary\n * data-sinwan-ev=\"click:0\" — event binding reference\n *\n * Usage:\n * const html = await renderToHydratableString(App, { name: \"World\" });\n * // → '<div data-sinwan-id=\"c0\"><p>Count: <!--sinwan-t:0-->5<!--/sinwan-t--></p>...</div>'\n */\n\nimport type { SinwanElement, SinwanNode, SinwanComponent } from \"../types.ts\";\nimport { HtmlEscapedString, escapeHtml } from \"../escaper.ts\";\nimport { isSignal } from \"../reactivity/signal.ts\";\nimport { isComputed } from \"../reactivity/computed.ts\";\nimport {\n compId,\n textMarkerOpen,\n textMarkerCloseStr,\n COMP_ID_ATTR,\n EVENT_ATTR,\n} from \"../hydration/markers.ts\";\nimport { isEventProp, toEventName } from \"../renderer/events.ts\";\nimport {\n createComponentInstance,\n getCurrentInstance,\n setCurrentInstance,\n} from \"../component/instance.ts\";\nimport { For, Show, isForElement, isShowElement } from \"../component/control-flow.ts\";\n\n// ─── Hydration context ─────────────────────────────────────\n\ninterface HydrationContext {\n componentIndex: number;\n textIndex: number;\n eventIndex: number;\n}\n\nfunction createHydrationContext(): HydrationContext {\n return { componentIndex: 0, textIndex: 0, eventIndex: 0 };\n}\n\n// ─── Public API ────────────────────────────────────────────\n\n/**\n * Render a component to an HTML string with hydration markers.\n */\nexport async function renderToHydratableString(\n component: SinwanComponent<any>,\n props?: Record<string, unknown>,\n): Promise<string> {\n const ctx = createHydrationContext();\n const mergedProps = props ?? {};\n\n // Create a temporary instance so lifecycle hooks register silently\n const instance = createComponentInstance(component, mergedProps, null);\n const prev = setCurrentInstance(instance);\n\n try {\n // Call the component to get the element tree\n const result = await component(mergedProps);\n if (result && typeof result === \"object\" && \"tag\" in result) {\n return renderElementH(result, ctx, true /* isComponentRoot */);\n }\n\n return renderNodeH(result as SinwanNode, ctx);\n } finally {\n setCurrentInstance(prev);\n }\n}\n\n/**\n * Render a raw SinwanNode tree with hydration markers.\n */\nexport async function renderNodeToHydratableString(\n node: SinwanNode,\n): Promise<string> {\n const ctx = createHydrationContext();\n return renderNodeH(node, ctx);\n}\n\n// ─── Internal rendering ────────────────────────────────────\n\nconst VOID_ELEMENTS = new Set([\n \"area\",\n \"base\",\n \"br\",\n \"col\",\n \"embed\",\n \"hr\",\n \"img\",\n \"input\",\n \"link\",\n \"meta\",\n \"param\",\n \"source\",\n \"track\",\n \"wbr\",\n]);\n\n/**\n * Render a node with hydration markers.\n */\nfunction renderNodeH(node: SinwanNode, ctx: HydrationContext): string {\n if (node == null || typeof node === \"boolean\") return \"\";\n\n if (typeof node === \"string\") return escapeHtml(node);\n if (typeof node === \"number\") return String(node);\n\n if (node instanceof HtmlEscapedString) return node.value;\n\n // Signal or Computed → wrap with text markers\n if (isSignal(node) || isComputed(node)) {\n const value = (node as any).value;\n const idx = ctx.textIndex++;\n return `${textMarkerOpen(idx)}${escapeHtml(String(value))}${textMarkerCloseStr()}`;\n }\n\n if (Array.isArray(node)) {\n return node.map((child) => renderNodeH(child, ctx)).join(\"\");\n }\n\n if (node instanceof Promise) {\n // Sync-only for hydration SSR — await should be handled at top level\n return \"\";\n }\n\n if (typeof node === \"object\" && \"tag\" in node) {\n return renderElementH(node, ctx, false);\n }\n\n return escapeHtml(String(node));\n}\n\n/**\n * Render an element with hydration markers.\n */\nfunction renderElementH(\n element: SinwanElement,\n ctx: HydrationContext,\n isComponentRoot: boolean,\n): string {\n const { tag, props, children } = element;\n\n // Fragment\n if (tag === \"\") {\n return children.map((child) => renderNodeH(child, ctx)).join(\"\");\n }\n\n if (tag === Show || tag === For) {\n return renderElementH((tag as Function)(props), ctx, isComponentRoot);\n }\n\n if (isShowElement(element)) {\n const when = readReactive(props.when);\n const content = when\n ? resolveShowChildren(element, when)\n : (props.fallback as SinwanNode);\n return renderNodeMaybeRoot(content, ctx, isComponentRoot);\n }\n\n if (isForElement(element)) {\n return renderForElementH(element, ctx);\n }\n\n // Functional component\n if (typeof tag === \"function\") {\n return renderComponentH(tag, props, ctx);\n }\n\n // Intrinsic HTML element\n if (typeof tag === \"string\") {\n return renderIntrinsicH(tag, props, children, ctx, isComponentRoot);\n }\n\n return children.map((child) => renderNodeH(child, ctx)).join(\"\");\n}\n\n/**\n * Render a functional component — calls it and marks the root element.\n */\nfunction renderComponentH(\n component: Function,\n props: Record<string, unknown>,\n ctx: HydrationContext,\n): string {\n // Set a temporary instance for lifecycle hooks\n const parentInstance = getCurrentInstance();\n const instance = createComponentInstance(component as any, props, parentInstance);\n if (parentInstance) {\n parentInstance.children.push(instance);\n }\n const prev = setCurrentInstance(instance);\n\n try {\n const result = component(props);\n\n if (result && typeof result === \"object\" && \"tag\" in result) {\n return renderElementH(\n result as SinwanElement,\n ctx,\n true /* mark as component root */,\n );\n }\n\n return renderNodeH(result as SinwanNode, ctx);\n } finally {\n setCurrentInstance(prev);\n }\n}\n\n/**\n * Render an intrinsic element with hydration markers.\n */\nfunction renderIntrinsicH(\n tag: string,\n props: Record<string, unknown>,\n children: SinwanNode[],\n ctx: HydrationContext,\n isComponentRoot: boolean,\n): string {\n let attrs = \"\";\n\n // Component boundary marker\n if (isComponentRoot) {\n attrs += ` ${COMP_ID_ATTR}=\"${compId(ctx.componentIndex++)}\"`;\n }\n\n // Event markers + regular attributes\n const eventParts: string[] = [];\n\n for (const [key, value] of Object.entries(props)) {\n if (\n key === \"children\" ||\n key === \"key\" ||\n key === \"ref\" ||\n key === \"dangerouslySetInnerHTML\"\n ) {\n continue;\n }\n\n if (isEventProp(key)) {\n // Collect event markers\n const eventName = toEventName(key);\n eventParts.push(`${eventName}:${ctx.eventIndex++}`);\n continue;\n }\n\n if (value == null || value === false) continue;\n\n // Resolve signal/computed values to their current values for SSR\n let resolvedValue = value;\n if (isSignal(value) || isComputed(value)) {\n resolvedValue = (value as any).value;\n }\n\n if (resolvedValue === true) {\n const attrName =\n key === \"className\" ? \"class\" : key === \"htmlFor\" ? \"for\" : key;\n attrs += ` ${attrName}`;\n continue;\n }\n\n const attrName =\n key === \"className\" ? \"class\" : key === \"htmlFor\" ? \"for\" : key;\n attrs += ` ${attrName}=\"${escapeHtml(String(resolvedValue))}\"`;\n }\n\n // Add event attribute\n if (eventParts.length > 0) {\n attrs += ` ${EVENT_ATTR}=\"${eventParts.join(\",\")}\"`;\n }\n\n // Void elements\n if (VOID_ELEMENTS.has(tag)) {\n return `<${tag}${attrs}>`;\n }\n\n // Dangerous inner HTML\n const dangerous = props.dangerouslySetInnerHTML as\n | { __html?: string }\n | undefined;\n if (dangerous && typeof dangerous.__html === \"string\") {\n return `<${tag}${attrs}>${dangerous.__html}</${tag}>`;\n }\n\n // Render children with markers\n const childrenHtml = children\n .map((child) => renderNodeH(child, ctx))\n .join(\"\");\n\n return `<${tag}${attrs}>${childrenHtml}</${tag}>`;\n}\n\nfunction renderNodeMaybeRoot(\n node: SinwanNode,\n ctx: HydrationContext,\n isComponentRoot: boolean,\n): string {\n if (\n isComponentRoot &&\n node &&\n typeof node === \"object\" &&\n !Array.isArray(node) &&\n \"tag\" in node\n ) {\n return renderElementH(node as SinwanElement, ctx, true);\n }\n return renderNodeH(node, ctx);\n}\n\nfunction renderForElementH(element: SinwanElement, ctx: HydrationContext): string {\n const props = element.props as {\n each?: unknown;\n children?: (item: unknown, index: () => number) => SinwanNode;\n };\n const each = readReactive(props.each);\n if (!Array.isArray(each) || typeof props.children !== \"function\") {\n return \"\";\n }\n\n return each\n .map((item, index) => renderNodeH(props.children!(item, () => index), ctx))\n .join(\"\");\n}\n\nfunction resolveShowChildren(element: SinwanElement, value: unknown): SinwanNode {\n const children = (element.props as any).children ?? element.children;\n if (typeof children === \"function\") {\n return children(value);\n }\n return children as SinwanNode;\n}\n\nfunction readReactive(value: unknown): unknown {\n return isSignal(value) || isComputed(value) ? (value as any).value : value;\n}\n"
|
|
19
|
+
],
|
|
20
|
+
"mappings": "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",
|
|
21
|
+
"debugId": "107ADB344D9D30F764756E2164756E21",
|
|
22
|
+
"names": []
|
|
23
|
+
}
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
var{defineProperty:a,getOwnPropertyNames:hY,getOwnPropertyDescriptor:uY}=Object,mY=Object.prototype.hasOwnProperty;function iY(Y){return this[Y]}var ZY=(Y)=>{var W=(XY??=new WeakMap).get(Y),J;if(W)return W;if(W=a({},"__esModule",{value:!0}),Y&&typeof Y==="object"||typeof Y==="function"){for(var K of hY(Y))if(!mY.call(W,K))a(W,K,{get:iY.bind(Y,K),enumerable:!(J=uY(Y,K))||J.enumerable})}return XY.set(Y,W),W},XY;var pY=(Y)=>Y;function sY(Y,W){this[Y]=pY.bind(null,W)}var $Y=(Y,W)=>{for(var J in W)a(Y,J,{get:W[J],enumerable:!0,configurable:!0,set:sY.bind(W,J)})};var lY={};$Y(lY,{raw:()=>d,jsxs:()=>dY,jsxDEV:()=>cY,jsx:()=>aY,HtmlEscapedString:()=>B,Fragment:()=>GY});module.exports=ZY(lY);var GY=Symbol("Fragment");class B extends String{value;constructor(Y){super(Y);this.value=Y}toString(){return this.value}}var d=(Y)=>new B(Y);function k(Y){if(Y==null||typeof Y==="boolean")return[];if(Array.isArray(Y))return Y.flat(1/0);return[Y]}function c(Y,W,J){if(Y===GY)return{tag:"",props:{},children:J};if(typeof Y==="function"||typeof Y==="string"){let K=W??{};if(J.length>0&&K.children===void 0)K.children=J.length===1?J[0]:J;return{tag:Y,props:K,children:J}}return{tag:"",props:{},children:J}}function aY(Y,W,J){return c(Y,W,k(W?.children))}function dY(Y,W,J){let K=W?.children;return c(Y,W,Array.isArray(K)?K.flat(1/0):k(K))}function cY(Y,W,J,K,Q,X){let Z=K?Array.isArray(W?.children)?W.children.flat(1/0):k(W?.children):k(W?.children),$=c(Y,W,Z);if(Q)$.__source=Q;return $}var _W={};$Y(_W,{streamPage:()=>HY,streamHydratablePage:()=>EY,streamHydratableNode:()=>kY,renderToString:()=>F,renderToHydratableString:()=>fY,renderPage:()=>VY,renderNodeToHydratableString:()=>gY,registerPage:()=>_Y,isSlots:()=>SY,hasPage:()=>bY,getPage:()=>e});module.exports=ZY(_W);var l=globalThis.Bun,jY=typeof l?.escapeHTML==="function"?l.escapeHTML.bind(l):void 0,r=/[&<>"']/g,rY={"&":"&","<":"<",">":">",'"':""","'":"'"};function oY(Y){if(r.lastIndex=0,!r.test(Y))return Y;return Y.replace(r,(W)=>rY[W])}function z(Y){if(Y==null||typeof Y==="boolean")return"";if(typeof Y==="number")return String(Y);if(Y instanceof B)return Y.value;let W=String(Y);return jY?jY(W):oY(W)}function IW(Y){return d(Y)}function NW(Y){return Y instanceof B}var A=new Set,S=!1,o=!1,zY=[];function BY(Y){if(!Y.active)return;A.add(Y),tY()}function wY(Y){A.delete(Y)}function tY(){if(!S)S=!0,queueMicrotask(UY)}function UY(){o=!0;let Y=[...A].sort((K,Q)=>K.id-Q.id);A.clear();for(let K of Y)if(K.active)K.run();let W=10;while(A.size>0&&W-- >0){let K=[...A].sort((Q,X)=>Q.id-X.id);A.clear();for(let Q of K)if(Q.active)Q.run()}S=!1,o=!1;let J=zY.splice(0);for(let K of J)K()}function qY(Y){return new Promise((W)=>{let J=()=>{Y?.(),W()};if(S||o)zY.push(J);else queueMicrotask(J)})}function HW(){if(S)S=!1,UY()}var I=null,t=[],nY=0;class C{id;active=!0;fn;cleanup=void 0;deps=new Set;constructor(Y){this.id=nY++,this.fn=Y}run(){if(!this.active)return;if(t.includes(this))return;if(this.cleanupDeps(),this.cleanup)this.cleanup(),this.cleanup=void 0;t.push(this);let Y=I;I=this;try{let W=this.fn();if(typeof W==="function")this.cleanup=W}finally{I=Y,t.pop()}}cleanupDeps(){for(let Y of this.deps)Y.subscribers.delete(this);this.deps.clear()}notify(){BY(this)}dispose(){if(!this.active)return;if(this.active=!1,this.cleanup)this.cleanup(),this.cleanup=void 0;this.cleanupDeps(),wY(this)}}function CW(Y){let W=new C(Y);return W.run(),()=>W.dispose()}function y(Y){if(I)Y.subscribers.add(I),I.deps.add(Y)}function x(Y){let W=[...Y.subscribers];for(let J of W)J.notify()}var FY=Symbol("Sinwan:signal");class LY{[FY]=!0;subscribers=new Set;_value;_manualSubs=new Set;constructor(Y){this._value=Y}get value(){return y(this),this._value}set value(Y){if(Object.is(this._value,Y))return;this._value=Y,x(this);for(let W of this._manualSubs)W(Y)}peek(){return this._value}subscribe(Y){return this._manualSubs.add(Y),()=>{this._manualSubs.delete(Y)}}toString(){return String(this.value)}valueOf(){return this.value}}function fW(Y){return new LY(Y)}function w(Y){return Y!=null&&typeof Y==="object"&&FY in Y}var PY=Symbol("Sinwan:computed");class DY{[PY]=!0;subscribers=new Set;_value;_dirty=!0;_effect;constructor(Y){let W=this;this._effect=new C(()=>{W._value=Y()}),this._effect.notify=function(){if(!W._dirty)W._dirty=!0,x(W)},this._effect.run(),this._dirty=!1}get value(){if(y(this),this._dirty)this._effect.run(),this._dirty=!1;return this._value}peek(){if(this._dirty)this._effect.run(),this._dirty=!1;return this._value}toString(){return String(this.value)}valueOf(){return this.value}}function hW(Y){return new DY(Y)}function U(Y){return Y!=null&&typeof Y==="object"&&PY in Y}function eY(){return{createElement(Y){return document.createElement(Y)},createElementNS(Y,W){return document.createElementNS(Y,W)},createTextNode(Y){return document.createTextNode(Y)},createComment(Y){return document.createComment(Y)},setAttribute(Y,W,J){Y.setAttribute(W,J)},removeAttribute(Y,W){Y.removeAttribute(W)},setProperty(Y,W,J){Y[W]=J},insertBefore(Y,W,J){Y.insertBefore(W,J)},appendChild(Y,W){Y.appendChild(W)},remove(Y){Y.parentNode?.removeChild(Y)},setTextContent(Y,W){Y.data=W},addEventListener(Y,W,J){Y.addEventListener(W,J)},removeEventListener(Y,W,J){Y.removeEventListener(W,J)},parentNode(Y){return Y.parentNode},nextSibling(Y){return Y.nextSibling}}}var AY=eY(),O={...AY};function mW(Y){Object.assign(O,Y)}function iW(){for(let Y of Object.keys(O))delete O[Y];Object.assign(O,AY)}function P(Y){return Y.length>2&&Y[0]==="o"&&Y[1]==="n"&&Y[2]>="A"&&Y[2]<="Z"}function R(Y){return Y.slice(2).toLowerCase()}function YW(Y,W,J){return O.addEventListener(Y,W,J),()=>{O.removeEventListener(Y,W,J)}}function aW(Y,W){let J=[];for(let K of Object.keys(W))if(P(K)){let Q=W[K];if(typeof Q==="function"){let X=R(K);J.push(YW(Y,X,Q))}}return J}var OY=Symbol.for("Sinwan.Show"),MY=Symbol.for("Sinwan.For");function f(Y){return{tag:OY,props:Y,children:[]}}function g(Y){return{tag:MY,props:Y,children:[]}}function M(Y){return Y.tag===OY}function T(Y){return Y.tag===MY}var n=new Map;function _Y(Y,W){n.set(Y,W)}function e(Y){return n.get(Y)}function bY(Y){return n.has(Y)}async function VY(Y,W){let J=e(Y);if(!J)throw Error(`Page "${Y}" not found in registry`);let K=await J(W);return F(K)}async function F(Y){if(Y==null||typeof Y==="boolean")return"";if(typeof Y==="string")return z(Y);if(typeof Y==="number")return String(Y);if(Y instanceof B)return Y.value;if(w(Y)||U(Y))return z(String(Y.value));if(Array.isArray(Y))return(await Promise.all(Y.map((J)=>F(J)))).join("");if(Y instanceof Promise)return TY(await Y);return TY(Y)}async function TY(Y){let{tag:W,props:J,children:K}=Y;if(M(Y)){let Q=YY(J.when);return F(Q?ZW(Y,Q):J.fallback)}if(T(Y))return XW(Y);if(typeof W==="function"){let Q=await W(J);return F(Q)}if(typeof W==="string")return JW(W,J,K);return F(K)}var WW=new Set(["area","base","br","col","embed","hr","img","input","link","meta","param","source","track","wbr"]);async function JW(Y,W,J){let K=KW(W);if(WW.has(Y))return K?`<${Y}${K}>`:`<${Y}>`;let Q=await QW(J,W);return K?`<${Y}${K}>${Q}</${Y}>`:`<${Y}>${Q}</${Y}>`}function KW(Y){let W="";for(let[J,K]of Object.entries(Y)){if(J==="children"||J==="key"||J==="ref"||J==="dangerouslySetInnerHTML"||P(J))continue;let Q=YY(K);if(Q==null||Q===!1)continue;let X=J==="className"?"class":J,Z=X==="htmlFor"?"for":X;if(Q===!0){W+=` ${Z}`;continue}let $=z(String(Q));W+=` ${Z}="${$}"`}return W}async function QW(Y,W){let J=W.dangerouslySetInnerHTML;if(J&&typeof J.__html==="string")return J.__html;return F(Y)}function SY(Y){return Y!=null&&typeof Y==="object"&&!Array.isArray(Y)&&!(Y instanceof B)}async function XW(Y){let W=Y.props,J=YY(W.each);if(!Array.isArray(J)||typeof W.children!=="function")return"";return(await Promise.all(J.map((Q,X)=>F(W.children(Q,()=>X))))).join("")}function ZW(Y,W){let J=Y.props.children??Y.children;if(typeof J==="function")return J(W);return J}function YY(Y){return w(Y)||U(Y)?Y.value:Y}var $W=0;function H(Y,W,J){return{uid:$W++,component:Y,props:W,element:null,parent:J,children:[],effects:[],_mountedHooks:[],_unmountedHooks:[],_updatedHooks:[],_errorHooks:[],provides:J?Object.create(J.provides):Object.create(null),isMounted:!1,isUnmounted:!1}}var JY=null;function v(){return JY}function q(Y){let W=JY;return JY=Y,W}function WJ(Y,W){let J=q(Y);try{return W()}finally{q(J)}}function GW(Y){if(Y.isUnmounted)return;for(let W of Y.children)GW(W);if(!Y.isMounted){Y.isMounted=!0;for(let W of Y._mountedHooks)W()}}function jW(Y){for(let W of Y.children)jW(W);if(Y.isMounted&&!Y.isUnmounted){Y.isUnmounted=!0,Y.isMounted=!1;for(let W of Y._unmountedHooks)W();for(let W of Y.effects)W();Y.effects.length=0}}function zW(Y){for(let W of Y._updatedHooks)W()}var WY=new Set;function JJ(Y){if(!Y||!Y.isMounted||Y.isUnmounted||Y._updatedHooks.length===0||WY.has(Y))return;WY.add(Y),qY(()=>{if(WY.delete(Y),Y.isMounted&&!Y.isUnmounted)zW(Y)})}function KJ(Y,W){let J=Y;while(J){if(J._errorHooks.length>0){for(let K of J._errorHooks)K(W);return}J=J.parent}console.error("[Sinwan] Unhandled component error:",W)}var h="data-sinwan-id";var u="data-sinwan-ev";function m(Y){return`c${Y}`}function i(Y){return`<!--sinwan-t:${Y}-->`}function p(){return"<!--/sinwan-t-->"}function XJ(Y){let W=Y.data;if(W.startsWith("sinwan-t:")){let J=parseInt(W.slice(9),10);return Number.isNaN(J)?-1:J}return-1}function ZJ(Y){return Y.data==="/sinwan-t"}function NY(){return{componentIndex:0,textIndex:0,eventIndex:0}}var RY=new Set(["area","base","br","col","embed","hr","img","input","link","meta","param","source","track","wbr"]);function HY(Y,W){let J=new TextEncoder;return new ReadableStream({async start(K){try{let Q=await Y(W);await b(Q,K,J),K.close()}catch(Q){K.error(Q)}}})}function EY(Y,W){let J=new TextEncoder,K=NY();return new ReadableStream({async start(Q){try{await CY(Y,W??{},Q,J,K,!0),Q.close()}catch(X){Q.error(X)}}})}function kY(Y){let W=new TextEncoder,J=NY();return new ReadableStream({async start(K){try{await D(Y,K,W,J),K.close()}catch(Q){K.error(Q)}}})}async function b(Y,W,J){if(Y==null||typeof Y==="boolean")return;if(typeof Y==="string"){W.enqueue(J.encode(z(Y)));return}if(typeof Y==="number"){W.enqueue(J.encode(String(Y)));return}if(Y instanceof B){W.enqueue(J.encode(Y.value));return}if(w(Y)||U(Y)){W.enqueue(J.encode(z(String(Y.value))));return}if(Array.isArray(Y)){for(let K of Y)await b(K,W,J);return}if(Y instanceof Promise){let K=await Y;await IY(K,W,J);return}await IY(Y,W,J)}async function IY(Y,W,J){let{tag:K,props:Q,children:X}=Y;if(M(Y)){let Z=N(Q.when);await b(Z?yY(Y,Z):Q.fallback,W,J);return}if(T(Y)){await LW(Y,W,J);return}if(typeof K==="function"){let Z=await K(Q);await b(Z,W,J);return}if(typeof K==="string"){await BW(K,Q,X,W,J);return}await b(X,W,J)}async function BW(Y,W,J,K,Q){let X=wW(W),Z=W.dangerouslySetInnerHTML;if(RY.has(Y)){let j=X?`<${Y}${X}>`:`<${Y}>`;K.enqueue(Q.encode(j));return}let $=X?`<${Y}${X}>`:`<${Y}>`;if(K.enqueue(Q.encode($)),Z&&typeof Z.__html==="string")K.enqueue(Q.encode(Z.__html));else await b(J,K,Q);K.enqueue(Q.encode(`</${Y}>`))}function wW(Y){let W="";for(let[J,K]of Object.entries(Y)){if(J==="children"||J==="key"||J==="ref"||J==="dangerouslySetInnerHTML"||P(J))continue;let Q=N(K);if(Q==null||Q===!1)continue;let X=J==="className"?"class":J==="htmlFor"?"for":J;if(Q===!0){W+=` ${X}`;continue}let Z=z(String(Q));W+=` ${X}="${Z}"`}return W}async function D(Y,W,J,K,Q=!1){if(Y==null||typeof Y==="boolean")return;if(typeof Y==="string"){_(W,J,z(Y));return}if(typeof Y==="number"){_(W,J,String(Y));return}if(Y instanceof B){_(W,J,Y.value);return}if(w(Y)||U(Y)){let X=K.textIndex++;_(W,J,`${i(X)}${z(String(Y.value))}${p()}`);return}if(Array.isArray(Y)){for(let X of Y)await D(X,W,J,K);return}if(Y instanceof Promise){await KY(await Y,W,J,K,Q);return}await KY(Y,W,J,K,Q)}async function KY(Y,W,J,K,Q){let{tag:X,props:Z,children:$}=Y;if(X===""){for(let j of $)await D(j,W,J,K);return}if(X===f||X===g){await KY(X(Z),W,J,K,Q);return}if(M(Y)){let j=N(Z.when);await D(j?yY(Y,j):Z.fallback,W,J,K,Q);return}if(T(Y)){await qW(Y,W,J,K);return}if(typeof X==="function"){await CY(X,Z,W,J,K,!0);return}if(typeof X==="string"){await UW(X,Z,$,W,J,K,Q);return}await D($,W,J,K)}async function CY(Y,W,J,K,Q,X){let Z=v(),$=H(Y,W,Z);if(Z)Z.children.push($);let j=q($);try{let G=await Y(W);await D(G,J,K,Q,X)}finally{q(j)}}async function UW(Y,W,J,K,Q,X,Z){let $=FW(W,X,Z),j=W.dangerouslySetInnerHTML;if(_(K,Q,$?`<${Y}${$}>`:`<${Y}>`),RY.has(Y))return;if(j&&typeof j.__html==="string")_(K,Q,j.__html);else for(let G of J)await D(G,K,Q,X);_(K,Q,`</${Y}>`)}async function qW(Y,W,J,K){let Q=Y.props,X=N(Q.each);if(!Array.isArray(X)||typeof Q.children!=="function")return;for(let Z=0;Z<X.length;Z++)await D(Q.children(X[Z],()=>Z),W,J,K)}function FW(Y,W,J){let K="";if(J)K+=` ${h}="${m(W.componentIndex++)}"`;let Q=[];for(let[X,Z]of Object.entries(Y)){if(X==="children"||X==="key"||X==="ref"||X==="dangerouslySetInnerHTML")continue;if(P(X)){Q.push(`${R(X)}:${W.eventIndex++}`);continue}let $=N(Z);if($==null||$===!1)continue;let j=X==="className"?"class":X==="htmlFor"?"for":X;if($===!0){K+=` ${j}`;continue}K+=` ${j}="${z(String($))}"`}if(Q.length>0)K+=` ${u}="${Q.join(",")}"`;return K}function _(Y,W,J){Y.enqueue(W.encode(J))}async function LW(Y,W,J){let K=Y.props,Q=N(K.each);if(!Array.isArray(Q)||typeof K.children!=="function")return;for(let X=0;X<Q.length;X++)await b(K.children(Q[X],()=>X),W,J)}function yY(Y,W){let J=Y.props.children??Y.children;if(typeof J==="function")return J(W);return J}function N(Y){return w(Y)||U(Y)?Y.value:Y}function xY(){return{componentIndex:0,textIndex:0,eventIndex:0}}async function fY(Y,W){let J=xY(),K=W??{},Q=H(Y,K,null),X=q(Q);try{let Z=await Y(K);if(Z&&typeof Z==="object"&&"tag"in Z)return E(Z,J,!0);return L(Z,J)}finally{q(X)}}async function gY(Y){let W=xY();return L(Y,W)}var PW=new Set(["area","base","br","col","embed","hr","img","input","link","meta","param","source","track","wbr"]);function L(Y,W){if(Y==null||typeof Y==="boolean")return"";if(typeof Y==="string")return z(Y);if(typeof Y==="number")return String(Y);if(Y instanceof B)return Y.value;if(w(Y)||U(Y)){let J=Y.value,K=W.textIndex++;return`${i(K)}${z(String(J))}${p()}`}if(Array.isArray(Y))return Y.map((J)=>L(J,W)).join("");if(Y instanceof Promise)return"";if(typeof Y==="object"&&"tag"in Y)return E(Y,W,!1);return z(String(Y))}function E(Y,W,J){let{tag:K,props:Q,children:X}=Y;if(K==="")return X.map((Z)=>L(Z,W)).join("");if(K===f||K===g)return E(K(Q),W,J);if(M(Y)){let Z=vY(Q.when),$=Z?TW(Y,Z):Q.fallback;return OW($,W,J)}if(T(Y))return MW(Y,W);if(typeof K==="function")return DW(K,Q,W);if(typeof K==="string")return AW(K,Q,X,W,J);return X.map((Z)=>L(Z,W)).join("")}function DW(Y,W,J){let K=v(),Q=H(Y,W,K);if(K)K.children.push(Q);let X=q(Q);try{let Z=Y(W);if(Z&&typeof Z==="object"&&"tag"in Z)return E(Z,J,!0);return L(Z,J)}finally{q(X)}}function AW(Y,W,J,K,Q){let X="";if(Q)X+=` ${h}="${m(K.componentIndex++)}"`;let Z=[];for(let[G,V]of Object.entries(W)){if(G==="children"||G==="key"||G==="ref"||G==="dangerouslySetInnerHTML")continue;if(P(G)){let QY=R(G);Z.push(`${QY}:${K.eventIndex++}`);continue}if(V==null||V===!1)continue;let s=V;if(w(V)||U(V))s=V.value;if(s===!0){X+=` ${G==="className"?"class":G==="htmlFor"?"for":G}`;continue}X+=` ${G==="className"?"class":G==="htmlFor"?"for":G}="${z(String(s))}"`}if(Z.length>0)X+=` ${u}="${Z.join(",")}"`;if(PW.has(Y))return`<${Y}${X}>`;let $=W.dangerouslySetInnerHTML;if($&&typeof $.__html==="string")return`<${Y}${X}>${$.__html}</${Y}>`;let j=J.map((G)=>L(G,K)).join("");return`<${Y}${X}>${j}</${Y}>`}function OW(Y,W,J){if(J&&Y&&typeof Y==="object"&&!Array.isArray(Y)&&"tag"in Y)return E(Y,W,!0);return L(Y,W)}function MW(Y,W){let J=Y.props,K=vY(J.each);if(!Array.isArray(K)||typeof J.children!=="function")return"";return K.map((Q,X)=>L(J.children(Q,()=>X),W)).join("")}function TW(Y,W){let J=Y.props.children??Y.children;if(typeof J==="function")return J(W);return J}function vY(Y){return w(Y)||U(Y)?Y.value:Y}
|
|
2
|
+
|
|
3
|
+
//# debugId=27BC7EF2F2F881D864756E2164756E21
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../src/jsx/jsx-runtime.ts", "../../src/escaper.ts", "../../src/reactivity/scheduler.ts", "../../src/reactivity/effect.ts", "../../src/reactivity/signal.ts", "../../src/reactivity/computed.ts", "../../src/renderer/dom-ops.ts", "../../src/renderer/events.ts", "../../src/component/control-flow.ts", "../../src/server/renderer.ts", "../../src/component/instance.ts", "../../src/hydration/markers.ts", "../../src/server/stream.ts", "../../src/server/hydration-markers.ts"],
|
|
4
|
+
"sourcesContent": [
|
|
5
|
+
"/**\n * SinwanJS View Module — JSX Runtime\n *\n * JSX factory that returns SinwanElement structures for the view renderer.\n * Imported automatically when using JSX syntax.\n */\n\nimport type { SinwanElement, SinwanNode } from \"../types.ts\";\nimport type { SinwanIntrinsicElements } from \"./jsx-types\";\n\nexport const Fragment = Symbol(\"Fragment\");\n\n/**\n * A string that has been marked as safe HTML (already escaped).\n * Used for raw HTML injection with explicit trust marking.\n */\nexport class HtmlEscapedString extends String {\n constructor(public readonly value: string) {\n super(value);\n }\n override toString() {\n return this.value;\n }\n}\n\nexport const raw = (str: string) => new HtmlEscapedString(str);\n\n// Fast path for intrinsic elements\nconst VOID_ELEMENTS = new Set([\n \"area\",\n \"base\",\n \"br\",\n \"col\",\n \"embed\",\n \"hr\",\n \"img\",\n \"input\",\n \"link\",\n \"meta\",\n \"param\",\n \"source\",\n \"track\",\n \"wbr\",\n]);\n\n/**\n * Normalize children into an array of SinwanNode.\n */\nfunction normalizeChildren(children: any): SinwanNode[] {\n if (children == null || typeof children === \"boolean\") return [];\n if (Array.isArray(children)) return children.flat(Infinity);\n return [children];\n}\n\n/**\n * Build an SinwanElement from a component/tag invocation.\n * Shared logic for jsx, jsxs, and jsxDEV.\n */\nfunction buildElement(\n type: any,\n props: any,\n children: SinwanNode[],\n): SinwanElement {\n // Handle Fragment\n if (type === Fragment) {\n return { tag: \"\", props: {}, children };\n }\n\n // Functional components and intrinsic HTML elements both pass the type\n // through as `tag`. The renderer (client / server / hydration) is the\n // single owner of component-instance creation and lifecycle dispatch.\n // Calling the function eagerly here would bypass instance management\n // and break `onMounted`, `provide`/`inject`, and the parent/child tree.\n if (typeof type === \"function\" || typeof type === \"string\") {\n const finalProps = props ?? {};\n // Mirror children into props.children so component setup can read it.\n if (children.length > 0 && finalProps.children === undefined) {\n finalProps.children = children.length === 1 ? children[0] : children;\n }\n return { tag: type, props: finalProps, children };\n }\n\n // Fallback\n return { tag: \"\", props: {}, children };\n}\n\n/**\n * JSX factory — called for elements with 0 or 1 child.\n * TypeScript auto-imports this in production mode (`react-jsx`).\n */\nexport function jsx(type: any, props: any, key?: any): SinwanElement {\n return buildElement(type, props, normalizeChildren(props?.children));\n}\n\n/**\n * JSX static factory — called for elements with 2+ children.\n * `props.children` is **already an array**, so we skip normalizeChildren.\n */\nexport function jsxs(type: any, props: any, key?: any): SinwanElement {\n const children = props?.children;\n // Children is guaranteed to be an array by the compiler\n return buildElement(\n type,\n props,\n Array.isArray(children)\n ? children.flat(Infinity)\n : normalizeChildren(children),\n );\n}\n\n/**\n * Source location metadata attached by the compiler in dev mode.\n */\nexport interface JSXSource {\n fileName: string;\n lineNumber: number;\n columnNumber: number;\n}\n\n/**\n * JSX dev factory — called in dev mode (`react-jsxdev`).\n * Receives extra source/debug info for better error messages.\n */\nexport function jsxDEV(\n type: any,\n props: any,\n key: any,\n isStaticChildren: boolean,\n source?: JSXSource,\n self?: unknown,\n): SinwanElement {\n const children = isStaticChildren\n ? Array.isArray(props?.children)\n ? props.children.flat(Infinity)\n : normalizeChildren(props?.children)\n : normalizeChildren(props?.children);\n\n const element = buildElement(type, props, children);\n\n // Attach debug metadata (useful for dev-tools / error traces)\n if (source) {\n (element as any).__source = source;\n }\n\n return element;\n}\n\nexport namespace JSX {\n export type Element = SinwanElement | Promise<SinwanElement>;\n export interface IntrinsicAttributes {\n key?: string | number;\n }\n export interface ElementChildrenAttribute {\n children: {};\n }\n export interface IntrinsicElements extends SinwanIntrinsicElements {}\n}\n\ndeclare global {\n namespace JSX {\n type Element = SinwanElement | Promise<SinwanElement>;\n interface IntrinsicAttributes {\n key?: string | number;\n }\n interface ElementChildrenAttribute {\n children: {};\n }\n interface IntrinsicElements extends SinwanIntrinsicElements {}\n }\n}\n",
|
|
6
|
+
"/**\n * Sinwan — HTML Escaping\n *\n * Security utilities for sanitizing interpolated values.\n * Runtime-agnostic: prefers `Bun.escapeHTML` when running on Bun (native\n * speed), and falls back to a portable implementation everywhere else\n * (Node, Deno, Cloudflare Workers, browsers).\n */\n\nimport { HtmlEscapedString, raw } from \"./jsx/jsx-runtime\";\n\nexport { HtmlEscapedString, raw };\n\n// ─── Native fast-path detection ─────────────────────────────\n// `Bun` is only defined inside the Bun runtime. We probe it via globalThis\n// so module evaluation never throws on other runtimes.\nconst _bun = (globalThis as any).Bun as\n | { escapeHTML?: (s: string) => string }\n | undefined;\n\nconst _nativeEscape: ((s: string) => string) | undefined =\n typeof _bun?.escapeHTML === \"function\"\n ? _bun.escapeHTML.bind(_bun)\n : undefined;\n\n// ─── Portable fallback ──────────────────────────────────────\n// Covers the five characters that matter inside HTML text and attributes.\nconst HTML_ESCAPE_RE = /[&<>\"']/g;\nconst HTML_ESCAPE_MAP: Record<string, string> = {\n \"&\": \"&\",\n \"<\": \"<\",\n \">\": \">\",\n '\"': \""\",\n \"'\": \"'\",\n};\n\nfunction portableEscape(str: string): string {\n // Avoid a regex pass when nothing to escape (very common case).\n HTML_ESCAPE_RE.lastIndex = 0;\n if (!HTML_ESCAPE_RE.test(str)) return str;\n return str.replace(HTML_ESCAPE_RE, (c) => HTML_ESCAPE_MAP[c]!);\n}\n\n/**\n * Escape HTML entities in any value.\n * Returns \"\" for null/undefined/boolean and pass-through for numbers.\n * Pre-escaped strings (`HtmlEscapedString`) are returned untouched.\n */\nexport function escapeHtml(value: unknown): string {\n if (value == null || typeof value === \"boolean\") return \"\";\n if (typeof value === \"number\") return String(value);\n if (value instanceof HtmlEscapedString) return value.value;\n const s = String(value);\n return _nativeEscape ? _nativeEscape(s) : portableEscape(s);\n}\n\n/**\n * Mark a string as safe HTML (pre-escaped).\n * USE WITH CAUTION - only for trusted content!\n */\nexport function safeHtml(html: string): HtmlEscapedString {\n return raw(html);\n}\n\n/**\n * Check if a value is already escaped HTML.\n */\nexport function isSafeHtml(value: unknown): value is HtmlEscapedString {\n return value instanceof HtmlEscapedString;\n}\n",
|
|
7
|
+
"/**\n * SinwanJS Reactivity — Scheduler\n *\n * Microtask-based flush queue for batching reactive updates.\n * Effects are NOT run synchronously on signal write — they are\n * queued and flushed in a microtask (like Vue's nextTick).\n */\n\nexport interface EffectNode {\n id: number;\n run(): void;\n active: boolean;\n}\n\nconst pendingEffects = new Set<EffectNode>();\nlet flushScheduled = false;\nlet isFlushing = false;\n\n// Pending nextTick callbacks\nconst pendingCallbacks: (() => void)[] = [];\n\n/**\n * Schedule an effect for the next microtask flush.\n */\nexport function scheduleEffect(effect: EffectNode): void {\n if (!effect.active) return;\n pendingEffects.add(effect);\n scheduleFlush();\n}\n\n/**\n * Remove a scheduled effect (e.g. when disposed before flush).\n */\nexport function unscheduleEffect(effect: EffectNode): void {\n pendingEffects.delete(effect);\n}\n\n/**\n * Schedule the microtask flush if not already scheduled.\n */\nfunction scheduleFlush(): void {\n if (!flushScheduled) {\n flushScheduled = true;\n queueMicrotask(flush);\n }\n}\n\n/**\n * Flush all pending effects. Effects added during flush are\n * processed in the same pass (convergence loop with a safety limit).\n */\nfunction flush(): void {\n isFlushing = true;\n\n // Sort by id to guarantee parent-before-child execution order\n const sorted = [...pendingEffects].sort((a, b) => a.id - b.id);\n pendingEffects.clear();\n\n for (const effect of sorted) {\n if (effect.active) {\n effect.run();\n }\n }\n\n // If new effects were queued during the flush, drain them too\n // (safety limit to prevent infinite loops)\n let safety = 10;\n while (pendingEffects.size > 0 && safety-- > 0) {\n const next = [...pendingEffects].sort((a, b) => a.id - b.id);\n pendingEffects.clear();\n for (const effect of next) {\n if (effect.active) {\n effect.run();\n }\n }\n }\n\n flushScheduled = false;\n isFlushing = false;\n\n // Run nextTick callbacks after all effects\n const cbs = pendingCallbacks.splice(0);\n for (const cb of cbs) {\n cb();\n }\n}\n\n/**\n * Returns true if the scheduler is currently flushing effects.\n */\nexport function isFlushingEffects(): boolean {\n return isFlushing;\n}\n\n/**\n * Queue a callback that runs after the next reactive flush completes.\n * Similar to Vue's nextTick().\n */\nexport function nextTick(fn?: () => void): Promise<void> {\n return new Promise<void>((resolve) => {\n const callback = () => {\n fn?.();\n resolve();\n };\n if (flushScheduled || isFlushing) {\n // Effects are pending — run after they flush\n pendingCallbacks.push(callback);\n } else {\n // No pending effects — run on next microtask\n queueMicrotask(callback);\n }\n });\n}\n\n/**\n * Force synchronous flush of all pending effects.\n * Primarily for testing and batch().\n */\nexport function flushSync(): void {\n if (flushScheduled) {\n flushScheduled = false;\n flush();\n }\n}\n",
|
|
8
|
+
"/**\n * SinwanJS Reactivity — Effect\n *\n * Fine-grained effect system with automatic dependency tracking.\n * When an effect runs, any signal reads are tracked as dependencies.\n * When those signals change, the effect is re-scheduled.\n */\n\nimport { type EffectNode, scheduleEffect, unscheduleEffect } from \"./scheduler.ts\";\n\n// ─── Global tracking state ─────────────────────────────────\n\nlet activeEffect: ReactiveEffect | null = null;\nconst effectStack: ReactiveEffect[] = [];\nlet effectIdCounter = 0;\n\n// ─── Subscription interface ────────────────────────────────\n\n/**\n * A Dep is any object that can track subscribers.\n * Signals and computeds implement this internally.\n */\nexport interface Dep {\n subscribers: Set<ReactiveEffect>;\n}\n\n// ─── ReactiveEffect ────────────────────────────────────────\n\nexport type CleanupFn = () => void;\nexport type EffectFn = () => CleanupFn | void;\n\nexport class ReactiveEffect implements EffectNode {\n id: number;\n active = true;\n\n /** The user-supplied function */\n private fn: EffectFn;\n\n /** Cleanup returned from the last run */\n private cleanup: CleanupFn | void = undefined;\n\n /** All deps this effect is subscribed to (for bidirectional cleanup) */\n deps: Set<Dep> = new Set();\n\n constructor(fn: EffectFn) {\n this.id = effectIdCounter++;\n this.fn = fn;\n }\n\n /**\n * Execute the effect function while tracking dependencies.\n */\n run(): void {\n if (!this.active) return;\n\n // Prevent infinite re-entry\n if (effectStack.includes(this)) return;\n\n // Clean up previous dependencies\n this.cleanupDeps();\n\n // Run user cleanup from previous execution\n if (this.cleanup) {\n this.cleanup();\n this.cleanup = undefined;\n }\n\n // Push onto the tracking stack\n effectStack.push(this);\n const prevEffect = activeEffect;\n activeEffect = this;\n\n try {\n const result = this.fn();\n if (typeof result === \"function\") {\n this.cleanup = result;\n }\n } finally {\n activeEffect = prevEffect;\n effectStack.pop();\n }\n }\n\n /**\n * Unsubscribe from all current deps so stale deps don't trigger this effect.\n */\n private cleanupDeps(): void {\n for (const dep of this.deps) {\n dep.subscribers.delete(this);\n }\n this.deps.clear();\n }\n\n /**\n * Notify the scheduler that this effect should re-run.\n */\n notify(): void {\n scheduleEffect(this);\n }\n\n /**\n * Permanently dispose this effect — stop tracking & unsubscribe.\n */\n dispose(): void {\n if (!this.active) return;\n this.active = false;\n\n // Run user cleanup\n if (this.cleanup) {\n this.cleanup();\n this.cleanup = undefined;\n }\n\n this.cleanupDeps();\n unscheduleEffect(this);\n }\n}\n\n// ─── Public API ────────────────────────────────────────────\n\n/**\n * Create a reactive effect.\n *\n * The effect function runs immediately to establish initial dependencies.\n * It re-runs whenever any tracked signal changes.\n * Returns a dispose function to stop the effect.\n *\n * @example\n * const count = signal(0);\n * const dispose = effect(() => {\n * console.log(\"count is\", count.value);\n * });\n * // logs \"count is 0\" immediately\n *\n * count.value = 1;\n * // logs \"count is 1\" on next microtask\n *\n * dispose(); // stops tracking\n */\nexport function effect(fn: EffectFn): CleanupFn {\n const e = new ReactiveEffect(fn);\n // Run immediately (synchronous first run for initial tracking)\n e.run();\n return () => e.dispose();\n}\n\n// ─── Tracking helpers (used by signals/computed) ───────────\n\n/**\n * Track a dependency from the currently active effect.\n * Called by signal.value getters.\n */\nexport function track(dep: Dep): void {\n if (activeEffect) {\n dep.subscribers.add(activeEffect);\n activeEffect.deps.add(dep);\n }\n}\n\n/**\n * Trigger all subscribers of a dependency.\n * Called by signal.value setters.\n */\nexport function trigger(dep: Dep): void {\n // Copy to avoid modification during iteration\n const effects = [...dep.subscribers];\n for (const effect of effects) {\n effect.notify();\n }\n}\n\n/**\n * Returns the currently active effect (for advanced usage).\n */\nexport function getActiveEffect(): ReactiveEffect | null {\n return activeEffect;\n}\n",
|
|
9
|
+
"/**\n * SinwanJS Reactivity — Signal\n *\n * A signal is a reactive container for a single value.\n * Reading `.value` tracks the current effect as a subscriber.\n * Writing `.value` notifies all subscribers.\n *\n * Inspired by Vue 3 ref(), Solid signals, Preact signals.\n */\n\nimport { type Dep, track, trigger } from \"./effect.ts\";\n\n// ─── Signal interface ──────────────────────────────────────\n\nexport interface Signal<T> {\n /** Get or set the reactive value. Reading tracks; writing notifies. */\n value: T;\n\n /** Read the value without tracking dependencies. */\n peek(): T;\n\n /** Manually subscribe to changes. Returns an unsubscribe function. */\n subscribe(fn: (value: T) => void): () => void;\n}\n\n// Brand for type-checking\nconst SIGNAL_BRAND = Symbol(\"Sinwan:signal\");\n\n// ─── Implementation ────────────────────────────────────────\n\nclass SignalImpl<T> implements Signal<T>, Dep {\n [SIGNAL_BRAND] = true;\n\n subscribers = new Set<import(\"./effect.ts\").ReactiveEffect>();\n private _value: T;\n private _manualSubs = new Set<(value: T) => void>();\n\n constructor(initial: T) {\n this._value = initial;\n }\n\n get value(): T {\n track(this);\n return this._value;\n }\n\n set value(newValue: T) {\n if (Object.is(this._value, newValue)) return;\n this._value = newValue;\n trigger(this);\n\n // Notify manual subscribers\n for (const fn of this._manualSubs) {\n fn(newValue);\n }\n }\n\n peek(): T {\n return this._value;\n }\n\n subscribe(fn: (value: T) => void): () => void {\n this._manualSubs.add(fn);\n return () => {\n this._manualSubs.delete(fn);\n };\n }\n\n /**\n * toString() for interpolation in templates.\n */\n toString(): string {\n return String(this.value);\n }\n\n /**\n * valueOf() for numeric operations.\n */\n valueOf(): T {\n return this.value;\n }\n}\n\n// ─── Public API ────────────────────────────────────────────\n\n/**\n * Create a reactive signal.\n *\n * @example\n * const count = signal(0);\n * console.log(count.value); // 0\n *\n * effect(() => {\n * console.log(count.value); // re-runs when count changes\n * });\n *\n * count.value = 5; // triggers the effect\n */\nexport function signal<T>(initial: T): Signal<T> {\n return new SignalImpl(initial);\n}\n\n/**\n * Type guard: check if a value is a Signal.\n */\nexport function isSignal(value: unknown): value is Signal<unknown> {\n return (\n value != null && typeof value === \"object\" && SIGNAL_BRAND in (value as any)\n );\n}\n",
|
|
10
|
+
"/**\n * SinwanJS Reactivity — Computed\n *\n * A computed is a derived reactive value that lazily re-evaluates\n * when its dependencies change. It caches the result and only\n * recomputes when actually read after a dependency has changed.\n *\n * Design: The computed does NOT use the scheduler for its own\n * re-evaluation. When a dependency changes, it marks itself dirty\n * and triggers downstream subscribers (which ARE scheduled).\n * The actual re-evaluation happens lazily on `.value` access.\n *\n * Inspired by Vue 3 computed(), Solid createMemo().\n */\n\nimport { type Dep, track, trigger, ReactiveEffect } from \"./effect.ts\";\n\n// ─── Computed interface ────────────────────────────────────\n\nexport interface Computed<T> {\n /** Read the computed value (lazy evaluation, cached). */\n readonly value: T;\n\n /** Read without tracking. */\n peek(): T;\n}\n\n// Brand for type-checking\nconst COMPUTED_BRAND = Symbol(\"Sinwan:computed\");\n\n// ─── Implementation ────────────────────────────────────────\n\nclass ComputedImpl<T> implements Computed<T>, Dep {\n [COMPUTED_BRAND] = true;\n\n subscribers = new Set<ReactiveEffect>();\n\n _value!: T;\n _dirty = true;\n _effect: ReactiveEffect;\n\n constructor(getter: () => T) {\n const self = this;\n\n // Internal effect solely for dependency tracking.\n // The fn wraps the getter — it writes to self._value as a side effect.\n this._effect = new ReactiveEffect(() => {\n self._value = getter();\n });\n\n // Override notify: don't schedule this effect via the scheduler.\n // Instead mark dirty and propagate to our own subscribers.\n this._effect.notify = function () {\n if (!self._dirty) {\n self._dirty = true;\n trigger(self);\n }\n };\n\n // Run once synchronously to establish deps and cache initial value\n this._effect.run();\n this._dirty = false;\n }\n\n get value(): T {\n track(this);\n\n if (this._dirty) {\n this._effect.run();\n this._dirty = false;\n }\n\n return this._value;\n }\n\n peek(): T {\n if (this._dirty) {\n this._effect.run();\n this._dirty = false;\n }\n return this._value;\n }\n\n /**\n * toString() for template interpolation.\n */\n toString(): string {\n return String(this.value);\n }\n\n /**\n * valueOf() for numeric operations.\n */\n valueOf(): T {\n return this.value;\n }\n}\n\n// ─── Public API ────────────────────────────────────────────\n\n/**\n * Create a computed reactive value.\n *\n * The getter function is tracked — when any signal it reads changes,\n * the computed is marked dirty and will re-evaluate on next `.value` read.\n *\n * @example\n * const count = signal(2);\n * const doubled = computed(() => count.value * 2);\n *\n * console.log(doubled.value); // 4\n *\n * count.value = 5;\n * console.log(doubled.value); // 10\n */\nexport function computed<T>(getter: () => T): Computed<T> {\n return new ComputedImpl(getter);\n}\n\n/**\n * Type guard: check if a value is a Computed.\n */\nexport function isComputed(value: unknown): value is Computed<unknown> {\n return (\n value != null &&\n typeof value === \"object\" &&\n COMPUTED_BRAND in (value as any)\n );\n}\n",
|
|
11
|
+
"/// <reference lib=\"dom\" />\n\n/**\n * SinwanJS Client Renderer — DOM Operations\n *\n * Thin abstraction over native DOM APIs for testability\n * and potential future server-side DOM (e.g., happy-dom, linkedom).\n */\n\nexport interface DOMOps {\n createElement(tag: string): Element;\n createElementNS(namespace: string, tag: string): Element;\n createTextNode(text: string): Text;\n createComment(text: string): Comment;\n setAttribute(el: Element, key: string, value: string): void;\n removeAttribute(el: Element, key: string): void;\n setProperty(el: Element, key: string, value: unknown): void;\n insertBefore(parent: Node, child: Node, anchor: Node | null): void;\n appendChild(parent: Node, child: Node): void;\n remove(node: Node): void;\n setTextContent(node: Text, text: string): void;\n addEventListener(el: Element, event: string, handler: EventListener): void;\n removeEventListener(el: Element, event: string, handler: EventListener): void;\n parentNode(node: Node): Node | null;\n nextSibling(node: Node): Node | null;\n}\n\nfunction createDefaultDOMOps(): DOMOps {\n return {\n createElement(tag: string): Element {\n return document.createElement(tag);\n },\n\n createElementNS(namespace: string, tag: string): Element {\n return document.createElementNS(namespace, tag);\n },\n\n createTextNode(text: string): Text {\n return document.createTextNode(text);\n },\n\n createComment(text: string): Comment {\n return document.createComment(text);\n },\n\n setAttribute(el: Element, key: string, value: string): void {\n el.setAttribute(key, value);\n },\n\n removeAttribute(el: Element, key: string): void {\n el.removeAttribute(key);\n },\n\n setProperty(el: Element, key: string, value: unknown): void {\n (el as any)[key] = value;\n },\n\n insertBefore(parent: Node, child: Node, anchor: Node | null): void {\n parent.insertBefore(child, anchor);\n },\n\n appendChild(parent: Node, child: Node): void {\n parent.appendChild(child);\n },\n\n remove(node: Node): void {\n node.parentNode?.removeChild(node);\n },\n\n setTextContent(node: Text, text: string): void {\n node.data = text;\n },\n\n addEventListener(el: Element, event: string, handler: EventListener): void {\n el.addEventListener(event, handler);\n },\n\n removeEventListener(el: Element, event: string, handler: EventListener): void {\n el.removeEventListener(event, handler);\n },\n\n parentNode(node: Node): Node | null {\n return node.parentNode;\n },\n\n nextSibling(node: Node): Node | null {\n return node.nextSibling;\n },\n };\n}\n\nconst defaultDOMOps = createDefaultDOMOps();\n\n/**\n * Live DOM operations object used by the renderer.\n */\nexport const domOps: DOMOps = { ...defaultDOMOps };\n\nexport function setDOMOps(overrides: Partial<DOMOps>): void {\n Object.assign(domOps, overrides);\n}\n\nexport function resetDOMOps(): void {\n for (const key of Object.keys(domOps) as (keyof DOMOps)[]) {\n delete (domOps as any)[key];\n }\n Object.assign(domOps, defaultDOMOps);\n}\n",
|
|
12
|
+
"/// <reference lib=\"dom\" />\n\n/**\n * SinwanJS Client Renderer — Event Binding\n *\n * Direct event binding (not delegation). Each handler is attached\n * directly to its target element for simplicity and easy hydration.\n *\n * Design decision: direct binding like Solid.js, not delegation like React.\n */\n\nimport { domOps } from \"./dom-ops.ts\";\nimport type { CleanupFn } from \"../reactivity/index.ts\";\n\n/**\n * Check if a prop key is an event handler (starts with \"on\").\n */\nexport function isEventProp(key: string): boolean {\n return key.length > 2 && key[0] === \"o\" && key[1] === \"n\" && key[2]! >= \"A\" && key[2]! <= \"Z\";\n}\n\n/**\n * Extract the DOM event name from a prop key.\n * e.g., \"onClick\" → \"click\", \"onMouseEnter\" → \"mouseenter\"\n */\nexport function toEventName(key: string): string {\n return key.slice(2).toLowerCase();\n}\n\n/**\n * Bind an event handler to an element.\n * Returns a cleanup function to remove the listener.\n */\nexport function bindEvent(\n el: Element,\n eventName: string,\n handler: EventListener,\n): CleanupFn {\n domOps.addEventListener(el, eventName, handler);\n return () => {\n domOps.removeEventListener(el, eventName, handler);\n };\n}\n\n/**\n * Bind all event props from an element's props object.\n * Returns an array of cleanup functions.\n */\nexport function bindEvents(\n el: Element,\n props: Record<string, unknown>,\n): CleanupFn[] {\n const cleanups: CleanupFn[] = [];\n\n for (const key of Object.keys(props)) {\n if (isEventProp(key)) {\n const handler = props[key];\n if (typeof handler === \"function\") {\n const eventName = toEventName(key);\n cleanups.push(bindEvent(el, eventName, handler as EventListener));\n }\n }\n }\n\n return cleanups;\n}\n",
|
|
13
|
+
"import type { Reactive, SinwanElement, SinwanNode } from \"../types.ts\";\n\nexport const SHOW_TYPE = Symbol.for(\"Sinwan.Show\");\nexport const FOR_TYPE = Symbol.for(\"Sinwan.For\");\n\nexport interface ShowProps<T> {\n when: Reactive<T | false | null | undefined>;\n fallback?: SinwanNode;\n children?: SinwanNode | ((value: NonNullable<T>) => SinwanNode);\n}\n\nexport interface ForProps<T> {\n each: Reactive<readonly T[]>;\n key?: (item: T, index: number) => string | number | symbol;\n children?: (item: T, index: () => number) => SinwanNode;\n}\n\nexport function Show<T>(props: ShowProps<T>): SinwanElement {\n return {\n tag: SHOW_TYPE,\n props: props as unknown as Record<string, unknown>,\n children: [],\n };\n}\n\nexport function For<T>(props: ForProps<T>): SinwanElement {\n return {\n tag: FOR_TYPE,\n props: props as unknown as Record<string, unknown>,\n children: [],\n };\n}\n\nexport function isShowElement(element: SinwanElement): boolean {\n return element.tag === SHOW_TYPE;\n}\n\nexport function isForElement(element: SinwanElement): boolean {\n return element.tag === FOR_TYPE;\n}\n",
|
|
14
|
+
"/**\n * SinwanJS View Module — Renderer & Component Registry\n *\n * Renders Sinwan component trees to HTML strings.\n * Supports async components, caching, and streaming.\n */\n\nimport type {\n SinwanNode,\n SinwanElement,\n SinwanComponent,\n SinwanPage,\n SinwanSlots,\n} from \"../types.ts\";\nimport { HtmlEscapedString, escapeHtml } from \"../escaper.ts\";\nimport { isSignal } from \"../reactivity/signal.ts\";\nimport { isComputed } from \"../reactivity/computed.ts\";\nimport { isEventProp } from \"../renderer/events.ts\";\nimport { isForElement, isShowElement } from \"../component/control-flow.ts\";\n\n// Component cache - maps component identity to render function\nconst componentCache = new WeakMap<SinwanComponent<any>, boolean>();\n\n// Page registry\nconst pageRegistry = new Map<string, SinwanPage<any>>();\n\n/**\n * Register a page renderer by name.\n */\nexport function registerPage<D extends object = {}>(\n name: string,\n page: SinwanPage<D>,\n): void {\n pageRegistry.set(name, page);\n}\n\n/**\n * Get a registered page by name.\n */\nexport function getPage<D extends object = {}>(\n name: string,\n): SinwanPage<D> | undefined {\n return pageRegistry.get(name);\n}\n\n/**\n * Check if a page is registered.\n */\nexport function hasPage(name: string): boolean {\n return pageRegistry.has(name);\n}\n\n/**\n * Render a registered page to an HTML string.\n */\nexport async function renderPage<D extends object = {}>(\n name: string,\n data: D,\n): Promise<string> {\n const page = getPage<D>(name);\n if (!page) {\n throw new Error(`Page \"${name}\" not found in registry`);\n }\n\n const element = await page(data);\n return renderToString(element);\n}\n\n/**\n * Render a node tree to an HTML string.\n * Handles primitives, elements, components, and arrays.\n */\nexport async function renderToString(node: SinwanNode): Promise<string> {\n // Handle null/undefined/boolean\n if (node == null || typeof node === \"boolean\") {\n return \"\";\n }\n\n // Handle strings (escape them)\n if (typeof node === \"string\") {\n return escapeHtml(node);\n }\n\n // Handle numbers\n if (typeof node === \"number\") {\n return String(node);\n }\n\n // Handle pre-escaped HTML\n if (node instanceof HtmlEscapedString) {\n return node.value;\n }\n\n // Handle reactive containers — read current value and render as text\n if (isSignal(node) || isComputed(node)) {\n return escapeHtml(String((node as any).value));\n }\n\n // Handle arrays - render each child and concatenate\n if (Array.isArray(node)) {\n const results = await Promise.all(\n node.map((child) => renderToString(child)),\n );\n return results.join(\"\");\n }\n\n // Handle promises (async components)\n if (node instanceof Promise) {\n return renderElement(await node);\n }\n\n // Handle elements\n return renderElement(node as SinwanElement);\n}\n\n/**\n * Render an element to HTML string.\n */\nasync function renderElement(element: SinwanElement): Promise<string> {\n const { tag, props, children } = element;\n\n if (isShowElement(element)) {\n const when = readReactive(props.when);\n return renderToString(\n when ? resolveShowChildren(element, when) : props.fallback as SinwanNode,\n );\n }\n\n if (isForElement(element)) {\n return renderForElement(element);\n }\n\n // Handle functional components\n if (typeof tag === \"function\") {\n const result = await tag(props);\n return renderToString(result);\n }\n\n // Handle intrinsic HTML elements\n if (typeof tag === \"string\") {\n return renderIntrinsicElement(tag, props, children);\n }\n\n // Fallback - shouldn't happen with valid JSX\n return renderToString(children);\n}\n\n// Void elements that don't have closing tags\nconst VOID_ELEMENTS = new Set([\n \"area\",\n \"base\",\n \"br\",\n \"col\",\n \"embed\",\n \"hr\",\n \"img\",\n \"input\",\n \"link\",\n \"meta\",\n \"param\",\n \"source\",\n \"track\",\n \"wbr\",\n]);\n\n/**\n * Render an intrinsic HTML element.\n */\nasync function renderIntrinsicElement(\n tag: string,\n props: Record<string, unknown>,\n children: SinwanNode[],\n): Promise<string> {\n const attrs = renderAttributes(props);\n\n // Void elements have no children and no closing tag\n if (VOID_ELEMENTS.has(tag)) {\n return attrs ? `<${tag}${attrs}>` : `<${tag}>`;\n }\n\n // Render children (handles dangerouslySetInnerHTML)\n const childrenHtml = await renderChildren(children, props);\n\n // Build element\n return attrs\n ? `<${tag}${attrs}>${childrenHtml}</${tag}>`\n : `<${tag}>${childrenHtml}</${tag}>`;\n}\n\n/**\n * Render HTML attributes from props.\n */\nfunction renderAttributes(props: Record<string, unknown>): string {\n let attrs = \"\";\n\n for (const [key, value] of Object.entries(props)) {\n // Skip children and special props\n if (\n key === \"children\" ||\n key === \"key\" ||\n key === \"ref\" ||\n key === \"dangerouslySetInnerHTML\" ||\n isEventProp(key)\n ) {\n continue;\n }\n\n const resolvedValue = readReactive(value);\n\n // Skip null/undefined/false values\n if (resolvedValue == null || resolvedValue === false) continue;\n\n // Handle className -> class\n const attrName = key === \"className\" ? \"class\" : key;\n\n // Handle htmlFor -> for\n const finalName = attrName === \"htmlFor\" ? \"for\" : attrName;\n\n // Handle boolean true (just the attribute name)\n if (resolvedValue === true) {\n attrs += ` ${finalName}`;\n continue;\n }\n\n // Escape the attribute value\n const attrValue = escapeHtml(String(resolvedValue));\n attrs += ` ${finalName}=\"${attrValue}\"`;\n }\n\n return attrs;\n}\n\n/**\n * Render children, with special handling for dangerouslySetInnerHTML.\n */\nasync function renderChildren(\n children: SinwanNode[],\n props: Record<string, unknown>,\n): Promise<string> {\n // Check for dangerous inner HTML\n const dangerous = props.dangerouslySetInnerHTML as\n | { __html?: string }\n | undefined;\n if (dangerous && typeof dangerous.__html === \"string\") {\n return dangerous.__html; // Trust the HTML (user explicitly marked safe)\n }\n\n return renderToString(children);\n}\n\n// Wire up dangerouslySetInnerHTML handling by patching renderIntrinsicElement\nconst originalRenderIntrinsic = renderIntrinsicElement;\n\n/**\n * Check if children is a slots object (named slots).\n */\nexport function isSlots(children: unknown): children is SinwanSlots {\n return (\n children != null &&\n typeof children === \"object\" &&\n !Array.isArray(children) &&\n !(children instanceof HtmlEscapedString)\n );\n}\n\nasync function renderForElement(element: SinwanElement): Promise<string> {\n const props = element.props as {\n each?: unknown;\n children?: (item: unknown, index: () => number) => SinwanNode;\n };\n const each = readReactive(props.each);\n if (!Array.isArray(each) || typeof props.children !== \"function\") {\n return \"\";\n }\n\n const rendered = await Promise.all(\n each.map((item, index) => renderToString(props.children!(item, () => index))),\n );\n return rendered.join(\"\");\n}\n\nfunction resolveShowChildren(element: SinwanElement, value: unknown): SinwanNode {\n const children = (element.props as any).children ?? element.children;\n if (typeof children === \"function\") {\n return children(value);\n }\n return children as SinwanNode;\n}\n\nfunction readReactive(value: unknown): unknown {\n return isSignal(value) || isComputed(value) ? (value as any).value : value;\n}\n",
|
|
15
|
+
"/**\n * SinwanJS Component Runtime — Instance Management\n *\n * Each component rendered on the client gets a ComponentInstance\n * that tracks its lifecycle hooks, effects, parent/child tree,\n * and provide/inject context.\n *\n * A global `currentInstance` stack lets lifecycle hooks (onMounted, etc.)\n * register themselves during setup or synchronous lifecycle callbacks —\n * same pattern as Vue's getCurrentInstance.\n */\n\nimport type { SinwanComponent } from \"../types.ts\";\nimport type { MountedNode } from \"../renderer/types.ts\";\nimport type { CleanupFn } from \"../reactivity/index.ts\";\nimport { nextTick } from \"../reactivity/scheduler.ts\";\n\n// ─── ComponentInstance ─────────────────────────────────────\n\nlet uidCounter = 0;\n\nexport interface ComponentInstance {\n /** Unique identifier for this instance. */\n uid: number;\n\n /** The component definition (setup function). */\n component: SinwanComponent<any>;\n\n /** Props passed to this component. */\n props: Record<string, any>;\n\n /** The rendered DOM subtree (set after render). */\n element: MountedNode | null;\n\n /** Parent instance in the component tree. */\n parent: ComponentInstance | null;\n\n /** Child component instances. */\n children: ComponentInstance[];\n\n /** All effect dispose functions owned by this component. */\n effects: CleanupFn[];\n\n // ─── Lifecycle hook queues ────────────────────────────\n\n /** Callbacks to fire after the component is mounted to DOM. */\n _mountedHooks: (() => void)[];\n\n /** Callbacks to fire when the component is unmounted. */\n _unmountedHooks: (() => void)[];\n\n /** Callbacks to fire after any reactive update in this component. */\n _updatedHooks: (() => void)[];\n\n /** Error handler callbacks. */\n _errorHooks: ((err: Error) => void)[];\n\n // ─── Provide/Inject context ───────────────────────────\n\n /** Values provided by this instance (for inject in children). */\n provides: Record<string | symbol, unknown>;\n\n // ─── State flags ──────────────────────────────────────\n\n isMounted: boolean;\n isUnmounted: boolean;\n}\n\n/**\n * Create a fresh ComponentInstance.\n */\nexport function createComponentInstance(\n component: SinwanComponent<any>,\n props: Record<string, any>,\n parent: ComponentInstance | null,\n): ComponentInstance {\n return {\n uid: uidCounter++,\n component,\n props,\n element: null,\n parent,\n children: [],\n effects: [],\n _mountedHooks: [],\n _unmountedHooks: [],\n _updatedHooks: [],\n _errorHooks: [],\n // Inherit parent's provides (prototype chain for lookup)\n provides: parent ? Object.create(parent.provides) : Object.create(null),\n isMounted: false,\n isUnmounted: false,\n };\n}\n\n// ─── Current instance stack ────────────────────────────────\n\nlet currentInstance: ComponentInstance | null = null;\n\n/**\n * Get the currently active component instance.\n * Used by lifecycle hooks to register themselves.\n */\nexport function getCurrentInstance(): ComponentInstance | null {\n return currentInstance;\n}\n\n/**\n * Set the current instance (called by renderer before setup).\n * Returns the previous instance for restoration.\n */\nexport function setCurrentInstance(\n instance: ComponentInstance | null,\n): ComponentInstance | null {\n const prev = currentInstance;\n currentInstance = instance;\n return prev;\n}\n\n/**\n * Run a function with `instance` as the current component instance.\n * Automatically restores the previous instance when done.\n */\nexport function withInstance<T>(instance: ComponentInstance, fn: () => T): T {\n const prev = setCurrentInstance(instance);\n try {\n return fn();\n } finally {\n setCurrentInstance(prev);\n }\n}\n\n// ─── Lifecycle execution ───────────────────────────────────\n\n/**\n * Fire all onMounted hooks for an instance and its children (depth-first).\n */\nexport function fireMountedHooks(instance: ComponentInstance): void {\n if (instance.isUnmounted) {\n return;\n }\n\n // Children first (bottom-up, like Vue)\n for (const child of instance.children) {\n fireMountedHooks(child);\n }\n\n if (!instance.isMounted) {\n instance.isMounted = true;\n for (const hook of instance._mountedHooks) {\n hook();\n }\n }\n}\n\n/**\n * Fire all onUnmounted hooks and dispose all effects for an instance\n * and its children (depth-first, children first).\n */\nexport function fireUnmountedHooks(instance: ComponentInstance): void {\n // Children first\n for (const child of instance.children) {\n fireUnmountedHooks(child);\n }\n\n if (instance.isMounted && !instance.isUnmounted) {\n instance.isUnmounted = true;\n instance.isMounted = false;\n\n // Fire unmounted hooks\n for (const hook of instance._unmountedHooks) {\n hook();\n }\n\n // Dispose all effects owned by this component\n for (const dispose of instance.effects) {\n dispose();\n }\n instance.effects.length = 0;\n }\n}\n\n/**\n * Fire onUpdated hooks for the current instance.\n */\nexport function fireUpdatedHooks(instance: ComponentInstance): void {\n for (const hook of instance._updatedHooks) {\n hook();\n }\n}\n\nconst queuedUpdatedHooks = new Set<ComponentInstance>();\n\n/**\n * Queue onUpdated hooks to run after the current reactive flush.\n * Multiple DOM effects in the same flush produce one updated callback.\n */\nexport function queueUpdatedHooks(instance: ComponentInstance | null): void {\n if (\n !instance ||\n !instance.isMounted ||\n instance.isUnmounted ||\n instance._updatedHooks.length === 0 ||\n queuedUpdatedHooks.has(instance)\n ) {\n return;\n }\n\n queuedUpdatedHooks.add(instance);\n nextTick(() => {\n queuedUpdatedHooks.delete(instance);\n if (instance.isMounted && !instance.isUnmounted) {\n fireUpdatedHooks(instance);\n }\n });\n}\n\n/**\n * Handle an error in the component tree — walks up to find an error handler.\n */\nexport function handleComponentError(\n instance: ComponentInstance,\n err: Error,\n): void {\n let current: ComponentInstance | null = instance;\n while (current) {\n if (current._errorHooks.length > 0) {\n for (const hook of current._errorHooks) {\n hook(err);\n }\n return;\n }\n current = current.parent;\n }\n // No handler found — re-throw\n console.error(\"[Sinwan] Unhandled component error:\", err);\n}\n",
|
|
16
|
+
"/// <reference lib=\"dom\" />\n\n/**\n * SinwanJS Hydration — Marker Protocol\n *\n * Constants and helpers for the hydration marker format:\n *\n * data-sinwan-id=\"c0\" — component boundary\n * <!--sinwan-t:0-->val<!--/sinwan-t--> — reactive text boundary\n * data-sinwan-ev=\"click:0\" — event binding reference\n */\n\n// ─── Constants ─────────────────────────────────────────────\n\n/** Attribute on the root element of each component instance. */\nexport const COMP_ID_ATTR = \"data-sinwan-id\";\n\n/** Prefix for component IDs. */\nexport const COMP_ID_PREFIX = \"c\";\n\n/** Opening comment prefix for reactive text slots: `Sinwan-t:N` */\nexport const TEXT_MARKER_OPEN = \"sinwan-t:\";\n\n/** Closing comment for reactive text slots. */\nexport const TEXT_MARKER_CLOSE = \"/sinwan-t\";\n\n/** Attribute for event binding references. */\nexport const EVENT_ATTR = \"data-sinwan-ev\";\n\n// ─── Server-side marker generation ────────────────────────\n\n/** Build a component ID string, e.g. `\"c0\"`. */\nexport function compId(index: number): string {\n return `${COMP_ID_PREFIX}${index}`;\n}\n\n/** Build an opening text marker comment string. */\nexport function textMarkerOpen(index: number): string {\n return `<!--${TEXT_MARKER_OPEN}${index}-->`;\n}\n\n/** Build a closing text marker comment string. */\nexport function textMarkerCloseStr(): string {\n return `<!--${TEXT_MARKER_CLOSE}-->`;\n}\n\n/** Build an event attribute value, e.g. `\"click:0\"`. */\nexport function eventAttrValue(event: string, index: number): string {\n return `${event}:${index}`;\n}\n\n// ─── Client-side marker parsing ───────────────────────────\n\n/**\n * Check if a comment node is a reactive text opening marker.\n * Returns the slot index, or -1 if not a marker.\n */\nexport function parseTextOpenMarker(node: Comment): number {\n const data = node.data;\n if (data.startsWith(TEXT_MARKER_OPEN)) {\n const idx = parseInt(data.slice(TEXT_MARKER_OPEN.length), 10);\n return Number.isNaN(idx) ? -1 : idx;\n }\n return -1;\n}\n\n/**\n * Check if a comment node is a reactive text closing marker.\n */\nexport function isTextCloseMarker(node: Comment): boolean {\n return node.data === TEXT_MARKER_CLOSE;\n}\n\n/**\n * Parse `data-sinwan-ev` attribute value into event entries.\n * Format: `\"click:0\"` or `\"click:0,input:1\"` for multiple.\n * Returns array of `[eventName, handlerIndex]` tuples.\n */\nexport function parseEventAttr(value: string): [string, number][] {\n return value.split(\",\").map((pair) => {\n const [event, idx] = pair.split(\":\");\n return [event!, parseInt(idx!, 10)];\n });\n}\n\n/**\n * Parse `data-sinwan-id` into the component index.\n * e.g., `\"c3\"` → `3`\n */\nexport function parseCompId(value: string): number {\n return parseInt(value.slice(COMP_ID_PREFIX.length), 10);\n}\n",
|
|
17
|
+
"/**\n * SinwanJS View Module — Streaming SSR\n *\n * Progressive HTML streaming using Bun's native ReadableStream.\n * Streams chunks as they resolve without waiting for full tree.\n */\n\nimport type {\n SinwanNode,\n SinwanElement,\n SinwanPage,\n SinwanComponent,\n} from \"../types.ts\";\nimport { HtmlEscapedString, escapeHtml } from \"../escaper.ts\";\nimport { isSignal } from \"../reactivity/signal.ts\";\nimport { isComputed } from \"../reactivity/computed.ts\";\nimport { isEventProp, toEventName } from \"../renderer/events.ts\";\nimport {\n For,\n Show,\n isForElement,\n isShowElement,\n} from \"../component/control-flow.ts\";\nimport {\n createComponentInstance,\n getCurrentInstance,\n setCurrentInstance,\n} from \"../component/instance.ts\";\nimport {\n compId,\n textMarkerOpen,\n textMarkerCloseStr,\n COMP_ID_ATTR,\n EVENT_ATTR,\n} from \"../hydration/markers.ts\";\n\ninterface HydratableStreamContext {\n componentIndex: number;\n textIndex: number;\n eventIndex: number;\n}\n\nfunction createHydratableStreamContext(): HydratableStreamContext {\n return { componentIndex: 0, textIndex: 0, eventIndex: 0 };\n}\n\n// Void elements that don't have closing tags\nconst VOID_ELEMENTS = new Set([\n \"area\",\n \"base\",\n \"br\",\n \"col\",\n \"embed\",\n \"hr\",\n \"img\",\n \"input\",\n \"link\",\n \"meta\",\n \"param\",\n \"source\",\n \"track\",\n \"wbr\",\n]);\n\n/**\n * Stream a page to a ReadableStream.\n */\nexport function streamPage<D extends object = {}>(\n page: SinwanPage<D>,\n data: D,\n): ReadableStream<Uint8Array> {\n const encoder = new TextEncoder();\n\n return new ReadableStream({\n async start(controller) {\n try {\n // Resolve the page component\n const element = await page(data);\n\n // Stream the element tree\n await streamNode(element, controller, encoder);\n\n // Close the stream\n controller.close();\n } catch (error) {\n controller.error(error);\n }\n },\n });\n}\n\n/**\n * Stream a component with hydration markers.\n */\nexport function streamHydratablePage(\n component: SinwanComponent<any>,\n props?: Record<string, unknown>,\n): ReadableStream<Uint8Array> {\n const encoder = new TextEncoder();\n const ctx = createHydratableStreamContext();\n\n return new ReadableStream({\n async start(controller) {\n try {\n await streamHydratableComponent(\n component,\n props ?? {},\n controller,\n encoder,\n ctx,\n true,\n );\n controller.close();\n } catch (error) {\n controller.error(error);\n }\n },\n });\n}\n\n/**\n * Stream a raw node tree with hydration markers.\n */\nexport function streamHydratableNode(\n node: SinwanNode,\n): ReadableStream<Uint8Array> {\n const encoder = new TextEncoder();\n const ctx = createHydratableStreamContext();\n\n return new ReadableStream({\n async start(controller) {\n try {\n await streamHydratableNodeToController(node, controller, encoder, ctx);\n controller.close();\n } catch (error) {\n controller.error(error);\n }\n },\n });\n}\n\n/**\n * Stream a node tree to a controller.\n */\nasync function streamNode(\n node: SinwanNode,\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n): Promise<void> {\n // Handle null/undefined/boolean\n if (node == null || typeof node === \"boolean\") {\n return;\n }\n\n // Handle strings (escape them)\n if (typeof node === \"string\") {\n controller.enqueue(encoder.encode(escapeHtml(node)));\n return;\n }\n\n // Handle numbers\n if (typeof node === \"number\") {\n controller.enqueue(encoder.encode(String(node)));\n return;\n }\n\n // Handle pre-escaped HTML\n if (node instanceof HtmlEscapedString) {\n controller.enqueue(encoder.encode(node.value));\n return;\n }\n\n // Handle reactive containers — stream the escaped current value\n if (isSignal(node) || isComputed(node)) {\n controller.enqueue(encoder.encode(escapeHtml(String((node as any).value))));\n return;\n }\n\n // Handle arrays - stream each child\n if (Array.isArray(node)) {\n for (const child of node) {\n await streamNode(child, controller, encoder);\n }\n return;\n }\n\n // Handle async elements (Promise<SinwanElement>)\n if (node instanceof Promise) {\n const resolved = await node;\n await streamElement(resolved, controller, encoder);\n return;\n }\n\n // Handle elements\n await streamElement(node as SinwanElement, controller, encoder);\n}\n\n/**\n * Stream an element to the controller.\n */\nasync function streamElement(\n element: SinwanElement,\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n): Promise<void> {\n const { tag, props, children } = element;\n\n if (isShowElement(element)) {\n const when = readReactive(props.when);\n await streamNode(\n when ? resolveShowChildren(element, when) : (props.fallback as SinwanNode),\n controller,\n encoder,\n );\n return;\n }\n\n if (isForElement(element)) {\n await streamForElement(element, controller, encoder);\n return;\n }\n\n // Handle functional components\n if (typeof tag === \"function\") {\n const result = await tag(props);\n await streamNode(result, controller, encoder);\n return;\n }\n\n // Handle intrinsic HTML elements\n if (typeof tag === \"string\") {\n await streamIntrinsicElement(tag, props, children, controller, encoder);\n return;\n }\n\n // Fallback\n await streamNode(children, controller, encoder);\n}\n\n/**\n * Stream an intrinsic HTML element.\n */\nasync function streamIntrinsicElement(\n tag: string,\n props: Record<string, unknown>,\n children: SinwanNode[],\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n): Promise<void> {\n const attrs = renderAttributes(props);\n\n // Check for dangerous inner HTML\n const dangerous = props.dangerouslySetInnerHTML as\n | { __html?: string }\n | undefined;\n\n // Void elements have no children and no closing tag\n if (VOID_ELEMENTS.has(tag)) {\n const html = attrs ? `<${tag}${attrs}>` : `<${tag}>`;\n controller.enqueue(encoder.encode(html));\n return;\n }\n\n // Opening tag\n const openTag = attrs ? `<${tag}${attrs}>` : `<${tag}>`;\n controller.enqueue(encoder.encode(openTag));\n\n // Children or dangerous HTML\n if (dangerous && typeof dangerous.__html === \"string\") {\n controller.enqueue(encoder.encode(dangerous.__html));\n } else {\n await streamNode(children, controller, encoder);\n }\n\n // Closing tag\n controller.enqueue(encoder.encode(`</${tag}>`));\n}\n\n/**\n * Render HTML attributes from props.\n */\nfunction renderAttributes(props: Record<string, unknown>): string {\n let attrs = \"\";\n\n for (const [key, value] of Object.entries(props)) {\n if (\n key === \"children\" ||\n key === \"key\" ||\n key === \"ref\" ||\n key === \"dangerouslySetInnerHTML\" ||\n isEventProp(key)\n ) {\n continue;\n }\n\n const resolvedValue = readReactive(value);\n if (resolvedValue == null || resolvedValue === false) continue;\n\n const attrName =\n key === \"className\" ? \"class\" : key === \"htmlFor\" ? \"for\" : key;\n\n if (resolvedValue === true) {\n attrs += ` ${attrName}`;\n continue;\n }\n\n const attrValue = escapeHtml(String(resolvedValue));\n attrs += ` ${attrName}=\"${attrValue}\"`;\n }\n\n return attrs;\n}\n\nasync function streamHydratableNodeToController(\n node: SinwanNode,\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n ctx: HydratableStreamContext,\n isComponentRoot = false,\n): Promise<void> {\n if (node == null || typeof node === \"boolean\") {\n return;\n }\n\n if (typeof node === \"string\") {\n enqueue(controller, encoder, escapeHtml(node));\n return;\n }\n\n if (typeof node === \"number\") {\n enqueue(controller, encoder, String(node));\n return;\n }\n\n if (node instanceof HtmlEscapedString) {\n enqueue(controller, encoder, node.value);\n return;\n }\n\n if (isSignal(node) || isComputed(node)) {\n const idx = ctx.textIndex++;\n enqueue(\n controller,\n encoder,\n `${textMarkerOpen(idx)}${escapeHtml(String((node as any).value))}${textMarkerCloseStr()}`,\n );\n return;\n }\n\n if (Array.isArray(node)) {\n for (const child of node) {\n await streamHydratableNodeToController(child, controller, encoder, ctx);\n }\n return;\n }\n\n if (node instanceof Promise) {\n await streamHydratableElement(\n await node,\n controller,\n encoder,\n ctx,\n isComponentRoot,\n );\n return;\n }\n\n await streamHydratableElement(\n node as SinwanElement,\n controller,\n encoder,\n ctx,\n isComponentRoot,\n );\n}\n\nasync function streamHydratableElement(\n element: SinwanElement,\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n ctx: HydratableStreamContext,\n isComponentRoot: boolean,\n): Promise<void> {\n const { tag, props, children } = element;\n\n if (tag === \"\") {\n for (const child of children) {\n await streamHydratableNodeToController(child, controller, encoder, ctx);\n }\n return;\n }\n\n if (tag === Show || tag === For) {\n await streamHydratableElement(\n (tag as Function)(props),\n controller,\n encoder,\n ctx,\n isComponentRoot,\n );\n return;\n }\n\n if (isShowElement(element)) {\n const when = readReactive(props.when);\n await streamHydratableNodeToController(\n when ? resolveShowChildren(element, when) : (props.fallback as SinwanNode),\n controller,\n encoder,\n ctx,\n isComponentRoot,\n );\n return;\n }\n\n if (isForElement(element)) {\n await streamHydratableForElement(element, controller, encoder, ctx);\n return;\n }\n\n if (typeof tag === \"function\") {\n await streamHydratableComponent(\n tag as SinwanComponent<any>,\n props,\n controller,\n encoder,\n ctx,\n true,\n );\n return;\n }\n\n if (typeof tag === \"string\") {\n await streamHydratableIntrinsic(\n tag,\n props,\n children,\n controller,\n encoder,\n ctx,\n isComponentRoot,\n );\n return;\n }\n\n await streamHydratableNodeToController(children, controller, encoder, ctx);\n}\n\nasync function streamHydratableComponent(\n component: SinwanComponent<any>,\n props: Record<string, unknown>,\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n ctx: HydratableStreamContext,\n isComponentRoot: boolean,\n): Promise<void> {\n const parentInstance = getCurrentInstance();\n const instance = createComponentInstance(component, props, parentInstance);\n if (parentInstance) {\n parentInstance.children.push(instance);\n }\n\n const prev = setCurrentInstance(instance);\n try {\n const result = await component(props);\n await streamHydratableNodeToController(\n result,\n controller,\n encoder,\n ctx,\n isComponentRoot,\n );\n } finally {\n setCurrentInstance(prev);\n }\n}\n\nasync function streamHydratableIntrinsic(\n tag: string,\n props: Record<string, unknown>,\n children: SinwanNode[],\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n ctx: HydratableStreamContext,\n isComponentRoot: boolean,\n): Promise<void> {\n const attrs = renderHydratableAttributes(props, ctx, isComponentRoot);\n const dangerous = props.dangerouslySetInnerHTML as\n | { __html?: string }\n | undefined;\n\n enqueue(controller, encoder, attrs ? `<${tag}${attrs}>` : `<${tag}>`);\n\n if (VOID_ELEMENTS.has(tag)) {\n return;\n }\n\n if (dangerous && typeof dangerous.__html === \"string\") {\n enqueue(controller, encoder, dangerous.__html);\n } else {\n for (const child of children) {\n await streamHydratableNodeToController(child, controller, encoder, ctx);\n }\n }\n\n enqueue(controller, encoder, `</${tag}>`);\n}\n\nasync function streamHydratableForElement(\n element: SinwanElement,\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n ctx: HydratableStreamContext,\n): Promise<void> {\n const props = element.props as {\n each?: unknown;\n children?: (item: unknown, index: () => number) => SinwanNode;\n };\n const each = readReactive(props.each);\n if (!Array.isArray(each) || typeof props.children !== \"function\") {\n return;\n }\n\n for (let index = 0; index < each.length; index++) {\n await streamHydratableNodeToController(\n props.children(each[index], () => index),\n controller,\n encoder,\n ctx,\n );\n }\n}\n\nfunction renderHydratableAttributes(\n props: Record<string, unknown>,\n ctx: HydratableStreamContext,\n isComponentRoot: boolean,\n): string {\n let attrs = \"\";\n\n if (isComponentRoot) {\n attrs += ` ${COMP_ID_ATTR}=\"${compId(ctx.componentIndex++)}\"`;\n }\n\n const eventParts: string[] = [];\n\n for (const [key, value] of Object.entries(props)) {\n if (\n key === \"children\" ||\n key === \"key\" ||\n key === \"ref\" ||\n key === \"dangerouslySetInnerHTML\"\n ) {\n continue;\n }\n\n if (isEventProp(key)) {\n eventParts.push(`${toEventName(key)}:${ctx.eventIndex++}`);\n continue;\n }\n\n const resolvedValue = readReactive(value);\n if (resolvedValue == null || resolvedValue === false) continue;\n\n const attrName =\n key === \"className\" ? \"class\" : key === \"htmlFor\" ? \"for\" : key;\n\n if (resolvedValue === true) {\n attrs += ` ${attrName}`;\n continue;\n }\n\n attrs += ` ${attrName}=\"${escapeHtml(String(resolvedValue))}\"`;\n }\n\n if (eventParts.length > 0) {\n attrs += ` ${EVENT_ATTR}=\"${eventParts.join(\",\")}\"`;\n }\n\n return attrs;\n}\n\nfunction enqueue(\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n html: string,\n): void {\n controller.enqueue(encoder.encode(html));\n}\n\nasync function streamForElement(\n element: SinwanElement,\n controller: ReadableStreamDefaultController<Uint8Array>,\n encoder: TextEncoder,\n): Promise<void> {\n const props = element.props as {\n each?: unknown;\n children?: (item: unknown, index: () => number) => SinwanNode;\n };\n const each = readReactive(props.each);\n if (!Array.isArray(each) || typeof props.children !== \"function\") {\n return;\n }\n\n for (let index = 0; index < each.length; index++) {\n await streamNode(props.children(each[index], () => index), controller, encoder);\n }\n}\n\nfunction resolveShowChildren(element: SinwanElement, value: unknown): SinwanNode {\n const children = (element.props as any).children ?? element.children;\n if (typeof children === \"function\") {\n return children(value);\n }\n return children as SinwanNode;\n}\n\nfunction readReactive(value: unknown): unknown {\n return isSignal(value) || isComputed(value) ? (value as any).value : value;\n}\n",
|
|
18
|
+
"/**\n * SinwanJS Server — Hydration-Aware SSR Renderer\n *\n * Enhanced `renderToString` that injects hydration markers:\n *\n * data-sinwan-id=\"c0\" — component boundary\n * <!--sinwan-t:0-->val<!--/sinwan-t--> — reactive text boundary\n * data-sinwan-ev=\"click:0\" — event binding reference\n *\n * Usage:\n * const html = await renderToHydratableString(App, { name: \"World\" });\n * // → '<div data-sinwan-id=\"c0\"><p>Count: <!--sinwan-t:0-->5<!--/sinwan-t--></p>...</div>'\n */\n\nimport type { SinwanElement, SinwanNode, SinwanComponent } from \"../types.ts\";\nimport { HtmlEscapedString, escapeHtml } from \"../escaper.ts\";\nimport { isSignal } from \"../reactivity/signal.ts\";\nimport { isComputed } from \"../reactivity/computed.ts\";\nimport {\n compId,\n textMarkerOpen,\n textMarkerCloseStr,\n COMP_ID_ATTR,\n EVENT_ATTR,\n} from \"../hydration/markers.ts\";\nimport { isEventProp, toEventName } from \"../renderer/events.ts\";\nimport {\n createComponentInstance,\n getCurrentInstance,\n setCurrentInstance,\n} from \"../component/instance.ts\";\nimport { For, Show, isForElement, isShowElement } from \"../component/control-flow.ts\";\n\n// ─── Hydration context ─────────────────────────────────────\n\ninterface HydrationContext {\n componentIndex: number;\n textIndex: number;\n eventIndex: number;\n}\n\nfunction createHydrationContext(): HydrationContext {\n return { componentIndex: 0, textIndex: 0, eventIndex: 0 };\n}\n\n// ─── Public API ────────────────────────────────────────────\n\n/**\n * Render a component to an HTML string with hydration markers.\n */\nexport async function renderToHydratableString(\n component: SinwanComponent<any>,\n props?: Record<string, unknown>,\n): Promise<string> {\n const ctx = createHydrationContext();\n const mergedProps = props ?? {};\n\n // Create a temporary instance so lifecycle hooks register silently\n const instance = createComponentInstance(component, mergedProps, null);\n const prev = setCurrentInstance(instance);\n\n try {\n // Call the component to get the element tree\n const result = await component(mergedProps);\n if (result && typeof result === \"object\" && \"tag\" in result) {\n return renderElementH(result, ctx, true /* isComponentRoot */);\n }\n\n return renderNodeH(result as SinwanNode, ctx);\n } finally {\n setCurrentInstance(prev);\n }\n}\n\n/**\n * Render a raw SinwanNode tree with hydration markers.\n */\nexport async function renderNodeToHydratableString(\n node: SinwanNode,\n): Promise<string> {\n const ctx = createHydrationContext();\n return renderNodeH(node, ctx);\n}\n\n// ─── Internal rendering ────────────────────────────────────\n\nconst VOID_ELEMENTS = new Set([\n \"area\",\n \"base\",\n \"br\",\n \"col\",\n \"embed\",\n \"hr\",\n \"img\",\n \"input\",\n \"link\",\n \"meta\",\n \"param\",\n \"source\",\n \"track\",\n \"wbr\",\n]);\n\n/**\n * Render a node with hydration markers.\n */\nfunction renderNodeH(node: SinwanNode, ctx: HydrationContext): string {\n if (node == null || typeof node === \"boolean\") return \"\";\n\n if (typeof node === \"string\") return escapeHtml(node);\n if (typeof node === \"number\") return String(node);\n\n if (node instanceof HtmlEscapedString) return node.value;\n\n // Signal or Computed → wrap with text markers\n if (isSignal(node) || isComputed(node)) {\n const value = (node as any).value;\n const idx = ctx.textIndex++;\n return `${textMarkerOpen(idx)}${escapeHtml(String(value))}${textMarkerCloseStr()}`;\n }\n\n if (Array.isArray(node)) {\n return node.map((child) => renderNodeH(child, ctx)).join(\"\");\n }\n\n if (node instanceof Promise) {\n // Sync-only for hydration SSR — await should be handled at top level\n return \"\";\n }\n\n if (typeof node === \"object\" && \"tag\" in node) {\n return renderElementH(node, ctx, false);\n }\n\n return escapeHtml(String(node));\n}\n\n/**\n * Render an element with hydration markers.\n */\nfunction renderElementH(\n element: SinwanElement,\n ctx: HydrationContext,\n isComponentRoot: boolean,\n): string {\n const { tag, props, children } = element;\n\n // Fragment\n if (tag === \"\") {\n return children.map((child) => renderNodeH(child, ctx)).join(\"\");\n }\n\n if (tag === Show || tag === For) {\n return renderElementH((tag as Function)(props), ctx, isComponentRoot);\n }\n\n if (isShowElement(element)) {\n const when = readReactive(props.when);\n const content = when\n ? resolveShowChildren(element, when)\n : (props.fallback as SinwanNode);\n return renderNodeMaybeRoot(content, ctx, isComponentRoot);\n }\n\n if (isForElement(element)) {\n return renderForElementH(element, ctx);\n }\n\n // Functional component\n if (typeof tag === \"function\") {\n return renderComponentH(tag, props, ctx);\n }\n\n // Intrinsic HTML element\n if (typeof tag === \"string\") {\n return renderIntrinsicH(tag, props, children, ctx, isComponentRoot);\n }\n\n return children.map((child) => renderNodeH(child, ctx)).join(\"\");\n}\n\n/**\n * Render a functional component — calls it and marks the root element.\n */\nfunction renderComponentH(\n component: Function,\n props: Record<string, unknown>,\n ctx: HydrationContext,\n): string {\n // Set a temporary instance for lifecycle hooks\n const parentInstance = getCurrentInstance();\n const instance = createComponentInstance(component as any, props, parentInstance);\n if (parentInstance) {\n parentInstance.children.push(instance);\n }\n const prev = setCurrentInstance(instance);\n\n try {\n const result = component(props);\n\n if (result && typeof result === \"object\" && \"tag\" in result) {\n return renderElementH(\n result as SinwanElement,\n ctx,\n true /* mark as component root */,\n );\n }\n\n return renderNodeH(result as SinwanNode, ctx);\n } finally {\n setCurrentInstance(prev);\n }\n}\n\n/**\n * Render an intrinsic element with hydration markers.\n */\nfunction renderIntrinsicH(\n tag: string,\n props: Record<string, unknown>,\n children: SinwanNode[],\n ctx: HydrationContext,\n isComponentRoot: boolean,\n): string {\n let attrs = \"\";\n\n // Component boundary marker\n if (isComponentRoot) {\n attrs += ` ${COMP_ID_ATTR}=\"${compId(ctx.componentIndex++)}\"`;\n }\n\n // Event markers + regular attributes\n const eventParts: string[] = [];\n\n for (const [key, value] of Object.entries(props)) {\n if (\n key === \"children\" ||\n key === \"key\" ||\n key === \"ref\" ||\n key === \"dangerouslySetInnerHTML\"\n ) {\n continue;\n }\n\n if (isEventProp(key)) {\n // Collect event markers\n const eventName = toEventName(key);\n eventParts.push(`${eventName}:${ctx.eventIndex++}`);\n continue;\n }\n\n if (value == null || value === false) continue;\n\n // Resolve signal/computed values to their current values for SSR\n let resolvedValue = value;\n if (isSignal(value) || isComputed(value)) {\n resolvedValue = (value as any).value;\n }\n\n if (resolvedValue === true) {\n const attrName =\n key === \"className\" ? \"class\" : key === \"htmlFor\" ? \"for\" : key;\n attrs += ` ${attrName}`;\n continue;\n }\n\n const attrName =\n key === \"className\" ? \"class\" : key === \"htmlFor\" ? \"for\" : key;\n attrs += ` ${attrName}=\"${escapeHtml(String(resolvedValue))}\"`;\n }\n\n // Add event attribute\n if (eventParts.length > 0) {\n attrs += ` ${EVENT_ATTR}=\"${eventParts.join(\",\")}\"`;\n }\n\n // Void elements\n if (VOID_ELEMENTS.has(tag)) {\n return `<${tag}${attrs}>`;\n }\n\n // Dangerous inner HTML\n const dangerous = props.dangerouslySetInnerHTML as\n | { __html?: string }\n | undefined;\n if (dangerous && typeof dangerous.__html === \"string\") {\n return `<${tag}${attrs}>${dangerous.__html}</${tag}>`;\n }\n\n // Render children with markers\n const childrenHtml = children\n .map((child) => renderNodeH(child, ctx))\n .join(\"\");\n\n return `<${tag}${attrs}>${childrenHtml}</${tag}>`;\n}\n\nfunction renderNodeMaybeRoot(\n node: SinwanNode,\n ctx: HydrationContext,\n isComponentRoot: boolean,\n): string {\n if (\n isComponentRoot &&\n node &&\n typeof node === \"object\" &&\n !Array.isArray(node) &&\n \"tag\" in node\n ) {\n return renderElementH(node as SinwanElement, ctx, true);\n }\n return renderNodeH(node, ctx);\n}\n\nfunction renderForElementH(element: SinwanElement, ctx: HydrationContext): string {\n const props = element.props as {\n each?: unknown;\n children?: (item: unknown, index: () => number) => SinwanNode;\n };\n const each = readReactive(props.each);\n if (!Array.isArray(each) || typeof props.children !== \"function\") {\n return \"\";\n }\n\n return each\n .map((item, index) => renderNodeH(props.children!(item, () => index), ctx))\n .join(\"\");\n}\n\nfunction resolveShowChildren(element: SinwanElement, value: unknown): SinwanNode {\n const children = (element.props as any).children ?? element.children;\n if (typeof children === \"function\") {\n return children(value);\n }\n return children as SinwanNode;\n}\n\nfunction readReactive(value: unknown): unknown {\n return isSignal(value) || isComputed(value) ? (value as any).value : value;\n}\n"
|
|
19
|
+
],
|
|
20
|
+
"mappings": "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",
|
|
21
|
+
"debugId": "27BC7EF2F2F881D864756E2164756E21",
|
|
22
|
+
"names": []
|
|
23
|
+
}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import type { Reactive, SinwanElement, SinwanNode } from "../types.ts";
|
|
2
|
+
export declare const SHOW_TYPE: unique symbol;
|
|
3
|
+
export declare const FOR_TYPE: unique symbol;
|
|
4
|
+
export interface ShowProps<T> {
|
|
5
|
+
when: Reactive<T | false | null | undefined>;
|
|
6
|
+
fallback?: SinwanNode;
|
|
7
|
+
children?: SinwanNode | ((value: NonNullable<T>) => SinwanNode);
|
|
8
|
+
}
|
|
9
|
+
export interface ForProps<T> {
|
|
10
|
+
each: Reactive<readonly T[]>;
|
|
11
|
+
key?: (item: T, index: number) => string | number | symbol;
|
|
12
|
+
children?: (item: T, index: () => number) => SinwanNode;
|
|
13
|
+
}
|
|
14
|
+
export declare function Show<T>(props: ShowProps<T>): SinwanElement;
|
|
15
|
+
export declare function For<T>(props: ForProps<T>): SinwanElement;
|
|
16
|
+
export declare function isShowElement(element: SinwanElement): boolean;
|
|
17
|
+
export declare function isForElement(element: SinwanElement): boolean;
|
|
18
|
+
//# sourceMappingURL=control-flow.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"control-flow.d.ts","sourceRoot":"","sources":["../../src/component/control-flow.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,aAAa,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAEvE,eAAO,MAAM,SAAS,eAA4B,CAAC;AACnD,eAAO,MAAM,QAAQ,eAA2B,CAAC;AAEjD,MAAM,WAAW,SAAS,CAAC,CAAC;IAC1B,IAAI,EAAE,QAAQ,CAAC,CAAC,GAAG,KAAK,GAAG,IAAI,GAAG,SAAS,CAAC,CAAC;IAC7C,QAAQ,CAAC,EAAE,UAAU,CAAC;IACtB,QAAQ,CAAC,EAAE,UAAU,GAAG,CAAC,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC,CAAC,KAAK,UAAU,CAAC,CAAC;CACjE;AAED,MAAM,WAAW,QAAQ,CAAC,CAAC;IACzB,IAAI,EAAE,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;IAC7B,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,KAAK,MAAM,GAAG,MAAM,GAAG,MAAM,CAAC;IAC3D,QAAQ,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,MAAM,KAAK,UAAU,CAAC;CACzD;AAED,wBAAgB,IAAI,CAAC,CAAC,EAAE,KAAK,EAAE,SAAS,CAAC,CAAC,CAAC,GAAG,aAAa,CAM1D;AAED,wBAAgB,GAAG,CAAC,CAAC,EAAE,KAAK,EAAE,QAAQ,CAAC,CAAC,CAAC,GAAG,aAAa,CAMxD;AAED,wBAAgB,aAAa,CAAC,OAAO,EAAE,aAAa,GAAG,OAAO,CAE7D;AAED,wBAAgB,YAAY,CAAC,OAAO,EAAE,aAAa,GAAG,OAAO,CAE5D"}
|