diff --git a/README.md b/README.md index cb63fd0..12e1221 100644 --- a/README.md +++ b/README.md @@ -3,7 +3,7 @@ | [*mirrored* on Gitea](https://gitea.jaandrle.cz/jaandrle/deka-dom-el)

- Deka DOM Elements Logo + Deka DOM Elements Logo

# Deka DOM Elements @@ -15,40 +15,40 @@ ```javascript // 🌟 Reactive component with clear separation of concerns document.body.append( - el(EmojiCounter, { initial: "🚀" }) + el(EmojiCounter, { initial: "🚀" }) ); function EmojiCounter({ initial }) { - // ✨ State - Define reactive data - const count = S(0); - const emoji = S(initial); + // ✨ State - Define reactive data + const count = S(0); + const emoji = S(initial); - /** @param {HTMLOptionElement} el */ - const isSelected= el=> (el.selected= el.value===initial); - - // 🔄 View - UI updates automatically when signals change - return el().append( - el("p", { - className: "output", - textContent: S(() => - `Hello World ${emoji.get().repeat(clicks.get())}`), - }), - - // 🎮 Controls - Update state on events - el("button", { textContent: "Add Emoji" }, - on("click", () => count.set(count.get() + 1)) - ), - - el("select", null, on("change", e => emoji.set(e.target.value))) + /** @param {HTMLOptionElement} el */ + const isSelected= el=> (el.selected= el.value===initial); + + // 🔄 View - UI updates automatically when signals change + return el().append( + el("p", { + className: "output", + textContent: S(() => + `Hello World ${emoji.get().repeat(clicks.get())}`), + }), + + // 🎮 Controls - Update state on events + el("button", { textContent: "Add Emoji" }, + on("click", () => count.set(count.get() + 1)) + ), + + el("select", null, on("change", e => emoji.set(e.target.value))) .append( - el(Option, "🎉", isSelected), - el(Option, "🚀", isSelected), - el(Option, "💖", isSelected), - ) - ); + el(Option, "🎉", isSelected), + el(Option, "🚀", isSelected), + el(Option, "💖", isSelected), + ) + ); } function Option({ textContent }){ - return Ol("option", { value: textContent, textContent }); + return Ol("option", { value: textContent, textContent }); } ``` @@ -68,11 +68,16 @@ Creating reactive elements, components, and Web Components using the native ## Why Another Library? -This library bridges the gap between minimal solutions like van/hyperscript and more comprehensive frameworks like [solid-js](https://github.com/solidjs/solid), offering a balanced trade-off between size, complexity, and usability. +This library bridges the gap between minimal solutions like van/hyperscript and more comprehensive frameworks like +[solid-js](https://github.com/solidjs/solid), offering a balanced trade-off between size, complexity, and usability. -Following functional programming principles, Deka DOM Elements starts with pure JavaScript (DOM API) and gradually adds auxiliary functions. These range from minor improvements to advanced features for building complete declarative reactive UI templates. +Following functional programming principles, Deka DOM Elements starts with pure JavaScript (DOM API) and gradually adds +auxiliary functions. These range from minor improvements to advanced features for building complete declarative +reactive UI templates. -A key advantage: any internal function (`assign`, `classListDeclarative`, `on`, `dispatchEvent`, `S`, etc.) can be used independently while also working seamlessly together. This modular approach makes it easier to integrate the library into existing projects. +A key advantage: any internal function (`assign`, `classListDeclarative`, `on`, `dispatchEvent`, `S`, etc.) can be used +independently while also working seamlessly together. This modular approach makes it easier to integrate the library +into existing projects. ## Getting Started @@ -88,7 +93,7 @@ A key advantage: any internal function (`assign`, `classListDeclarative`, `on`, ```html ``` @@ -112,4 +117,5 @@ Signals are the reactive backbone of Deka DOM Elements: - [adamhaile/S](https://github.com/adamhaile/S) - Simple, clean, fast reactive programming - [hyperhype/hyperscript](https://github.com/hyperhype/hyperscript) - Create HyperText with JavaScript - [potch/signals](https://github.com/potch/signals) - A small reactive signals library -- [jaandrle/dollar_dom_component](https://github.com/jaandrle/dollar_dom_component) - Functional DOM components without JSX/virtual DOM +- [jaandrle/dollar_dom_component](https://github.com/jaandrle/dollar_dom_component) - + Functional DOM components without JSX/virtual DOM diff --git a/bs/docs.js b/bs/docs.js index f9bd3e7..76d6501 100755 --- a/bs/docs.js +++ b/bs/docs.js @@ -42,7 +42,7 @@ s.echo(styles.content).to(path_target.css+styles.name); // Copy assets echo("Copying assets…"); if(s.test("-d", "docs/assets")) { - s.cp("-r", "docs/assets/*", path_target.assets); + s.cp("-r", "docs/assets/*", path_target.assets); } dispatchEvent("onssrend"); diff --git a/dist/esm-with-signals.d.min.ts b/dist/esm-with-signals.d.min.ts index fdf18cf..49490b4 100644 --- a/dist/esm-with-signals.d.min.ts +++ b/dist/esm-with-signals.d.min.ts @@ -72,6 +72,20 @@ export function assignAttribute[ATT] type ExtendedHTMLElementTagNameMap= HTMLElementTagNameMap & CustomElementTagNameMap; +export namespace el { + /** + * Creates a marker comment for elements + * + * @param attrs - Marker attributes + * @param [is_open=false] - Whether the marker is open-ended + * @returns Comment node marker + */ + export function mark( + attrs: { type: "component"|"reactive"|"later", name?: string, host?: "this"|"parentElement" }, + is_open?: boolean + ): Comment; +} + export function el< A extends ddeComponentAttributes, EL extends SupportedElement | ddeDocumentFragment @@ -251,6 +265,27 @@ export function customElementWithDDE HTMLElement)>(custom_e export function lifecyclesToEvents HTMLElement)>(custom_element: EL): EL export function observedAttributes(custom_element: HTMLElement): Record +/** + * This is used primarly for server side rendering. To be sure that all async operations + * are finished before the page is sent to the client. + * ``` + * // on component + * function component(){ + * … + * queue(fetch(...).then(...)); + * } + * + * // building the page + * async function build(){ + * const { component }= await import("./component.js"); + * document.body.append(el(component)); + * await queue(); + * retutn document.body.innerHTML; + * } + * ``` + * */ +export function queue(promise?: Promise): Promise; + /* TypeScript MEH */ declare global{ type ddeAppend= (...nodes: (Node | string)[])=> el; diff --git a/dist/esm-with-signals.d.ts b/dist/esm-with-signals.d.ts index fdf18cf..49490b4 100644 --- a/dist/esm-with-signals.d.ts +++ b/dist/esm-with-signals.d.ts @@ -72,6 +72,20 @@ export function assignAttribute[ATT] type ExtendedHTMLElementTagNameMap= HTMLElementTagNameMap & CustomElementTagNameMap; +export namespace el { + /** + * Creates a marker comment for elements + * + * @param attrs - Marker attributes + * @param [is_open=false] - Whether the marker is open-ended + * @returns Comment node marker + */ + export function mark( + attrs: { type: "component"|"reactive"|"later", name?: string, host?: "this"|"parentElement" }, + is_open?: boolean + ): Comment; +} + export function el< A extends ddeComponentAttributes, EL extends SupportedElement | ddeDocumentFragment @@ -251,6 +265,27 @@ export function customElementWithDDE HTMLElement)>(custom_e export function lifecyclesToEvents HTMLElement)>(custom_element: EL): EL export function observedAttributes(custom_element: HTMLElement): Record +/** + * This is used primarly for server side rendering. To be sure that all async operations + * are finished before the page is sent to the client. + * ``` + * // on component + * function component(){ + * … + * queue(fetch(...).then(...)); + * } + * + * // building the page + * async function build(){ + * const { component }= await import("./component.js"); + * document.body.append(el(component)); + * await queue(); + * retutn document.body.innerHTML; + * } + * ``` + * */ +export function queue(promise?: Promise): Promise; + /* TypeScript MEH */ declare global{ type ddeAppend= (...nodes: (Node | string)[])=> el; diff --git a/dist/esm.d.min.ts b/dist/esm.d.min.ts index 78497fd..9ac112a 100644 --- a/dist/esm.d.min.ts +++ b/dist/esm.d.min.ts @@ -72,6 +72,20 @@ export function assignAttribute[ATT] type ExtendedHTMLElementTagNameMap= HTMLElementTagNameMap & CustomElementTagNameMap; +export namespace el { + /** + * Creates a marker comment for elements + * + * @param attrs - Marker attributes + * @param [is_open=false] - Whether the marker is open-ended + * @returns Comment node marker + */ + export function mark( + attrs: { type: "component"|"reactive"|"later", name?: string, host?: "this"|"parentElement" }, + is_open?: boolean + ): Comment; +} + export function el< A extends ddeComponentAttributes, EL extends SupportedElement | ddeDocumentFragment @@ -251,6 +265,27 @@ export function customElementWithDDE HTMLElement)>(custom_e export function lifecyclesToEvents HTMLElement)>(custom_element: EL): EL export function observedAttributes(custom_element: HTMLElement): Record +/** + * This is used primarly for server side rendering. To be sure that all async operations + * are finished before the page is sent to the client. + * ``` + * // on component + * function component(){ + * … + * queue(fetch(...).then(...)); + * } + * + * // building the page + * async function build(){ + * const { component }= await import("./component.js"); + * document.body.append(el(component)); + * await queue(); + * retutn document.body.innerHTML; + * } + * ``` + * */ +export function queue(promise?: Promise): Promise; + /* TypeScript MEH */ declare global{ type ddeAppend= (...nodes: (Node | string)[])=> el; diff --git a/dist/esm.d.ts b/dist/esm.d.ts index 78497fd..9ac112a 100644 --- a/dist/esm.d.ts +++ b/dist/esm.d.ts @@ -72,6 +72,20 @@ export function assignAttribute[ATT] type ExtendedHTMLElementTagNameMap= HTMLElementTagNameMap & CustomElementTagNameMap; +export namespace el { + /** + * Creates a marker comment for elements + * + * @param attrs - Marker attributes + * @param [is_open=false] - Whether the marker is open-ended + * @returns Comment node marker + */ + export function mark( + attrs: { type: "component"|"reactive"|"later", name?: string, host?: "this"|"parentElement" }, + is_open?: boolean + ): Comment; +} + export function el< A extends ddeComponentAttributes, EL extends SupportedElement | ddeDocumentFragment @@ -251,6 +265,27 @@ export function customElementWithDDE HTMLElement)>(custom_e export function lifecyclesToEvents HTMLElement)>(custom_element: EL): EL export function observedAttributes(custom_element: HTMLElement): Record +/** + * This is used primarly for server side rendering. To be sure that all async operations + * are finished before the page is sent to the client. + * ``` + * // on component + * function component(){ + * … + * queue(fetch(...).then(...)); + * } + * + * // building the page + * async function build(){ + * const { component }= await import("./component.js"); + * document.body.append(el(component)); + * await queue(); + * retutn document.body.innerHTML; + * } + * ``` + * */ +export function queue(promise?: Promise): Promise; + /* TypeScript MEH */ declare global{ type ddeAppend= (...nodes: (Node | string)[])=> el; diff --git a/docs/assets/favicon.svg b/docs/assets/favicon.svg index e52cd62..9110dc5 100644 --- a/docs/assets/favicon.svg +++ b/docs/assets/favicon.svg @@ -1,28 +1,28 @@ - - - - - - - - - - - - - - - - - - - - - - - - - dde + + + + + + + + + + + + + + + + + + + + + + + + + dde \ No newline at end of file diff --git a/docs/assets/logo.svg b/docs/assets/logo.svg index 690e248..73d0587 100644 --- a/docs/assets/logo.svg +++ b/docs/assets/logo.svg @@ -1,33 +1,33 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - dde + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + dde \ No newline at end of file diff --git a/docs/components/examples/elements/dde-dom-create.js b/docs/components/examples/elements/dde-dom-create.js index 846c24b..b9edb55 100644 --- a/docs/components/examples/elements/dde-dom-create.js +++ b/docs/components/examples/elements/dde-dom-create.js @@ -2,9 +2,9 @@ import { el } from "deka-dom-el"; // Create element with properties const button = el("button", { - textContent: "Click me", - className: "primary", - disabled: true + textContent: "Click me", + className: "primary", + disabled: true }); // Shorter and more expressive diff --git a/docs/components/examples/elements/native-dom-create.js b/docs/components/examples/elements/native-dom-create.js index fd717c6..8b7affc 100644 --- a/docs/components/examples/elements/native-dom-create.js +++ b/docs/components/examples/elements/native-dom-create.js @@ -6,12 +6,12 @@ button.disabled = true; // Or using Object.assign() const button2 = Object.assign( - document.createElement('button'), - { - textContent: "Click me", - className: "primary", - disabled: true - } + document.createElement('button'), + { + textContent: "Click me", + className: "primary", + disabled: true + } ); // Add to DOM diff --git a/docs/components/examples/events/append-event.js b/docs/components/examples/events/append-event.js index 271c0d7..86aa442 100644 --- a/docs/components/examples/events/append-event.js +++ b/docs/components/examples/events/append-event.js @@ -2,7 +2,7 @@ import { el, on } from "deka-dom-el"; // Third approach - append with on addon el("button", { - textContent: "click me" + textContent: "click me" }).append( - on("click", (e) => console.log("Clicked!", e)) + on("click", (e) => console.log("Clicked!", e)) ); \ No newline at end of file diff --git a/docs/components/examples/events/attribute-event.js b/docs/components/examples/events/attribute-event.js index fa6906c..fb55ecb 100644 --- a/docs/components/examples/events/attribute-event.js +++ b/docs/components/examples/events/attribute-event.js @@ -2,6 +2,6 @@ import { el } from "deka-dom-el"; // Using events with HTML attribute style el("button", { - textContent: "click me", - "=onclick": "console.log(event)" + textContent: "click me", + "=onclick": "console.log(event)" }); \ No newline at end of file diff --git a/docs/components/examples/events/chain-event.js b/docs/components/examples/events/chain-event.js index f537a14..471d981 100644 --- a/docs/components/examples/events/chain-event.js +++ b/docs/components/examples/events/chain-event.js @@ -2,7 +2,7 @@ import { el, on } from "deka-dom-el"; // Using events as addons - chainable approach el("button", { - textContent: "click me", -}, - on("click", (e) => console.log("Clicked!", e)) + textContent: "click me", +}, + on("click", (e) => console.log("Clicked!", e)) ); \ No newline at end of file diff --git a/docs/components/examples/events/property-event.js b/docs/components/examples/events/property-event.js index 63f8a0d..acfb229 100644 --- a/docs/components/examples/events/property-event.js +++ b/docs/components/examples/events/property-event.js @@ -2,6 +2,6 @@ import { el } from "deka-dom-el"; // Using events with property assignment el("button", { - textContent: "click me", - onclick: console.log + textContent: "click me", + onclick: console.log }); \ No newline at end of file diff --git a/docs/components/examples/introducing/3ps-before.js b/docs/components/examples/introducing/3ps-before.js index 4a0a669..babc75f 100644 --- a/docs/components/examples/introducing/3ps-before.js +++ b/docs/components/examples/introducing/3ps-before.js @@ -4,11 +4,11 @@ const button = document.querySelector('button'); let count = 0; button.addEventListener('click', () => { - count++; - document.querySelector('p').textContent = - 'Clicked ' + count + ' times'; + count++; + document.querySelector('p').textContent = + 'Clicked ' + count + ' times'; - if (count > 10) { - button.disabled = true; - } + if (count > 10) { + button.disabled = true; + } }); diff --git a/docs/components/examples/introducing/3ps.js b/docs/components/examples/introducing/3ps.js index 93d7bd8..d78ede6 100644 --- a/docs/components/examples/introducing/3ps.js +++ b/docs/components/examples/introducing/3ps.js @@ -4,11 +4,11 @@ const count = S(0); // 2. React to state changes S.on(count, value => { - updateUI(value); - if (value > 10) disableButton(); + updateUI(value); + if (value > 10) disableButton(); }); // 3. Update state on events button.addEventListener('click', () => { - count.set(count.get() + 1); + count.set(count.get() + 1); }); diff --git a/docs/components/examples/introducing/helloWorld.js b/docs/components/examples/introducing/helloWorld.js index b1ca151..a13692d 100644 --- a/docs/components/examples/introducing/helloWorld.js +++ b/docs/components/examples/introducing/helloWorld.js @@ -3,28 +3,28 @@ import { S } from "deka-dom-el/signals"; // A HelloWorld component using the 3PS pattern function HelloWorld({ emoji = "🚀" }) { - // PART 1: Create reactive state - const clicks = S(0); - - return el().append( - // PART 2: Bind state to UI elements - el("p", { - className: "greeting", - // This paragraph automatically updates when clicks changes - textContent: S(() => `Hello World ${emoji.repeat(clicks.get())}`) - }), - - // PART 3: Update state in response to events - el("button", { - type: "button", - textContent: "Add emoji", - // When clicked, update the state - onclick: () => clicks.set(clicks.get() + 1) - }) - ); + // PART 1: Create reactive state + const clicks = S(0); + + return el().append( + // PART 2: Bind state to UI elements + el("p", { + className: "greeting", + // This paragraph automatically updates when clicks changes + textContent: S(() => `Hello World ${emoji.repeat(clicks.get())}`) + }), + + // PART 3: Update state in response to events + el("button", { + type: "button", + textContent: "Add emoji", + // When clicked, update the state + onclick: () => clicks.set(clicks.get() + 1) + }) + ); } // Use the component in your app document.body.append( - el(HelloWorld, { emoji: "🎉" }) + el(HelloWorld, { emoji: "🎉" }) ); \ No newline at end of file diff --git a/docs/components/examples/scopes/custom-scope.js b/docs/components/examples/scopes/custom-scope.js index ed659da..d1c74f8 100644 --- a/docs/components/examples/scopes/custom-scope.js +++ b/docs/components/examples/scopes/custom-scope.js @@ -2,13 +2,13 @@ import { scope } from "deka-dom-el"; import { S } from "deka-dom-el/signals"; function customSignalLogic() { - // Create an isolated scope for a specific operation - scope.push(); // Start new scope + // Create an isolated scope for a specific operation + scope.push(); // Start new scope - // These signals are in the new scope - const isolatedCount = S(0); - const isolatedDerived = S(() => isolatedCount.get() * 2); + // These signals are in the new scope + const isolatedCount = S(0); + const isolatedDerived = S(() => isolatedCount.get() * 2); - // Clean up by returning to previous scope - scope.pop(); + // Clean up by returning to previous scope + scope.pop(); } diff --git a/docs/components/examples/scopes/with-scope.js b/docs/components/examples/scopes/with-scope.js index 850f947..fbbe5c9 100644 --- a/docs/components/examples/scopes/with-scope.js +++ b/docs/components/examples/scopes/with-scope.js @@ -2,44 +2,44 @@ import { el, scope } from "deka-dom-el"; import { S } from "deka-dom-el/signals"; function CounterWithIsolatedTimer() { - const { host } = scope; + const { host } = scope; - // Main component state - const count = S(0); + // Main component state + const count = S(0); - // Create a timer in an isolated scope - scope.isolate(() => { - // These subscriptions won't be tied to the component lifecycle - // They would continue to run even if the component was removed - const timer = S(0); + // Create a timer in an isolated scope + scope.isolate(() => { + // These subscriptions won't be tied to the component lifecycle + // They would continue to run even if the component was removed + const timer = S(0); - // Not recommended for real applications! - // Just demonstrating scope isolation - setInterval(() => { - timer.set(timer.get() + 1); - console.log(`Timer: ${timer.get()}`); - }, 1000); - }); + // Not recommended for real applications! + // Just demonstrating scope isolation + setInterval(() => { + timer.set(timer.get() + 1); + console.log(`Timer: ${timer.get()}`); + }, 1000); + }); - // Normal component functionality within main scope - function increment() { - count.set(count.get() + 1); - } + // Normal component functionality within main scope + function increment() { + count.set(count.get() + 1); + } - return el("div").append( - el("p").append( - "Count: ", - el("#text", S(() => count.get())) - ), - el("button", { - textContent: "Increment", - onclick: increment - }), - el("p", "An isolated timer runs in console") - ); + return el("div").append( + el("p").append( + "Count: ", + el("#text", S(() => count.get())) + ), + el("button", { + textContent: "Increment", + onclick: increment + }), + el("p", "An isolated timer runs in console") + ); } // Usage document.body.append( - el(CounterWithIsolatedTimer) + el(CounterWithIsolatedTimer) ); diff --git a/docs/components/ireland.html.js b/docs/components/ireland.html.js index 1b2dbc1..8387664 100644 --- a/docs/components/ireland.html.js +++ b/docs/components/ireland.html.js @@ -20,7 +20,7 @@ export function ireland({ src, exportName = "default", props = {} }) { // relative src against the current directory const path= "./"+relative(dir, src.pathname); const id = "ireland-" + generateComponentId(src); - const element = el.mark({ type: "ireland", name: ireland.name }); + const element = el.mark({ type: "later", name: ireland.name }); queue(import(path).then(module => { const component = module[exportName]; element.replaceWith(el(component, props, mark(id))); diff --git a/docs/index.html.js b/docs/index.html.js index 1de9a71..d01d56a 100644 --- a/docs/index.html.js +++ b/docs/index.html.js @@ -104,7 +104,8 @@ export function page({ pkg, info }){ el("li").append(...T`${el("strong", "Custom Elements")} — Building web components`), el("li").append(...T`${el("strong", "Debugging")} — Tools to help you build and fix your apps`), el("li").append(...T`${el("strong", "Extensions")} — Integrating third-party functionalities`), - el("li").append(...T`${el("strong", "Ireland Components")} — Creating interactive demos with server-side pre-rendering`), + el("li").append(...T`${el("strong", "Ireland Components")} — + Creating interactive demos with server-side pre-rendering`), el("li").append(...T`${el("strong", "SSR")} — Server-side rendering with DDE`) ), el("p").append(...T` @@ -112,4 +113,4 @@ export function page({ pkg, info }){ Let's get started with the basics of creating elements! `), ); -} \ No newline at end of file +} diff --git a/docs/p05-scopes.html.js b/docs/p05-scopes.html.js index e35230e..fb118f1 100644 --- a/docs/p05-scopes.html.js +++ b/docs/p05-scopes.html.js @@ -55,21 +55,21 @@ export function page({ pkg, info }){ el(MyComponent); function MyComponent() { - // 2. access the host element - const { host } = scope; +  // 2. access the host element +  const { host } = scope; - // 3. Add behavior to host - host( - on.click(handleClick) - ); +  // 3. Add behavior to host +  host( +  on.click(handleClick) +  ); - // 4. Return the host element - return el("div", { - className: "my-component" - }).append( - el("h2", "Title"), - el("p", "Content") - ); +  // 4. Return the host element +  return el("div", { +  className: "my-component" +  }).append( +  el("h2", "Title"), +  el("p", "Content") +  ); } `.trim())) ), @@ -118,9 +118,9 @@ function MyComponent() { 3. Component interactions happen 4. Component removed from DOM → disconnected event 5. Automatic cleanup of: - - Event listeners - - Signal subscriptions - - Custom cleanup code +  - Event listeners +  - Signal subscriptions +  - Custom cleanup code `)) ), el(example, { src: fileURL("./components/examples/scopes/cleaning.js"), page_id }), @@ -167,6 +167,9 @@ function MyComponent() { el("li").append(...T` ${el("strong", "Capture host early:")} Use ${el("code", "const { host } = scope")} at component start `), + el("li").append(...T` + ${el("strong", "Define signals as constants:")} ${el("code", "const counter = S(0);")} + `), el("li").append(...T` ${el("strong", "Prefer declarative patterns:")} Use signals to drive UI updates rather than manual DOM manipulation `), diff --git a/docs/p06-customElement.html.js b/docs/p06-customElement.html.js index f4e3d02..f81d957 100644 --- a/docs/p06-customElement.html.js +++ b/docs/p06-customElement.html.js @@ -202,15 +202,14 @@ export function page({ pkg, info }){ el("h4", t`Shadow DOM Encapsulation`), el("pre").append(el("code", ` +  ┌─────────────────────────┐ +    #shadow-root - ┌─────────────────────────┐ - #shadow-root - - Created with DDE: - ┌──────────────────┐ -
-

Title

-

Content

+      Created with DDE: +    ┌──────────────────┐ +      
+       

Title

+       

Content

`)) ), el(example, { src: fileURL("./components/examples/customElement/shadowRoot.js"), page_id }), diff --git a/docs/p08-extensions.html.js b/docs/p08-extensions.html.js index b72755d..826dd77 100644 --- a/docs/p08-extensions.html.js +++ b/docs/p08-extensions.html.js @@ -42,8 +42,8 @@ export function page({ pkg, info }){ // Basic structure of an addon function myAddon(config) { return function(element) { - // Apply functionality to element - element.dataset.myAddon = config.option; + // Apply functionality to element + element.dataset.myAddon = config.option; }; } @@ -67,18 +67,15 @@ el("div", { id: "example" }, myAddon({ option: "value" })); // Third-party library addon with proper cleanup function externalLibraryAddon(config, signal) { return function(element) { - // Get an abort signal that triggers on element disconnection - const signal = on.disconnectedAsAbort(element); + // Initialize the third-party library + const instance = new ExternalLibrary(element, config); - // Initialize the third-party library - const instance = new ExternalLibrary(element, config); + // Set up cleanup when the element is removed + signal.addEventListener('abort', () => { + instance.destroy(); + }); - // Set up cleanup when the element is removed - signal.addEventListener('abort', () => { - instance.destroy(); - }); - - return element; + return element; }; } // dde component @@ -157,8 +154,8 @@ function createEnhancedSignal(initialValue) { // Return the original signal with added methods return Object.assign(signal, { - increment, - decrement + increment, + decrement }); } @@ -240,10 +237,12 @@ console.log(doubled.get()); // 10`, page_id }), el("h4", t`Common Extension Pitfalls`), el("dl").append( el("dt", t`Leaking event listeners or resources`), - el("dd", t`Always use AbortSignal-based cleanup to automatically remove listeners when elements are disconnected`), + el("dd", t`Always use AbortSignal-based cleanup to automatically remove listeners when elements + are disconnected`), el("dt", t`Tight coupling with library internals`), - el("dd", t`Focus on standard DOM APIs and clean interfaces rather than depending on deka-dom-el implementation details`), + el("dd", t`Focus on standard DOM APIs and clean interfaces rather than depending on deka-dom-el + implementation details`), el("dt", t`Mutating element prototypes`), el("dd", t`Prefer compositional approaches with addons over modifying element prototypes`), diff --git a/docs/p09-ssr.html.js b/docs/p09-ssr.html.js index 843783e..4a20fb1 100644 --- a/docs/p09-ssr.html.js +++ b/docs/p09-ssr.html.js @@ -18,10 +18,11 @@ export function page({ pkg, info }){ return el(simplePage, { info, pkg }).append( el("div", { className: "warning" }).append( el("p").append(...T` - This part of the documentation is primarily intended for technical enthusiasts and documentation - authors. It describes an advanced feature, not a core part of the library. Most users will not need to - implement this functionality directly in their applications. This capability will hopefully be covered - by third-party libraries or frameworks that provide simpler SSR integration using deka-dom-el. + This part of the documentation is primarily intended for technical enthusiasts and authors of + 3rd-party libraries. It describes an advanced feature, not a core part of the library. Most users will + not need to implement this functionality directly in their applications. This capability will hopefully + be covered by third-party libraries or frameworks that provide simpler SSR integration using + deka-dom-el. `) ), el("p").append(...T` diff --git a/docs/p10-ireland.html.js b/docs/p10-ireland.html.js index b46fd9a..e59fc53 100644 --- a/docs/p10-ireland.html.js +++ b/docs/p10-ireland.html.js @@ -2,7 +2,8 @@ import { T, t } from "./utils/index.js"; export const info= { title: t`Ireland Components`, fullTitle: t`Interactive Demo Components with Server-Side Pre-Rendering`, - description: t`Creating live, interactive component examples in documentation with server-side rendering and client-side hydration.`, + description: t`Creating live, interactive component examples in documentation with server-side + rendering and client-side hydration.`, }; import { el } from "deka-dom-el"; @@ -19,10 +20,11 @@ export function page({ pkg, info }){ return el(simplePage, { info, pkg }).append( el("div", { className: "warning" }).append( el("p").append(...T` - This part of the documentation is primarily intended for technical enthusiasts and documentation - authors. It describes an advanced feature, not a core part of the library. Most users will not need to - implement this functionality directly in their applications. This capability will hopefully be covered - by third-party libraries or frameworks that provide simpler SSR integration using deka-dom-el. + This part of the documentation is primarily intended for technical enthusiasts and authors of + 3rd-party libraries. It describes an advanced feature, not a core part of the library. Most users will + not need to implement this functionality directly in their applications. This capability will hopefully + be covered by third-party libraries or frameworks that provide simpler SSR integration using + deka-dom-el. `) ), @@ -61,14 +63,15 @@ export function page({ pkg, info }){ el(h3, t`Implementation Architecture`), el("p").append(...T` The core of the Ireland system is implemented in ${el("code", "docs/components/ireland.html.js")}. - It integrates with the SSR build process using the ${el("code", "registerClientFile")} function from ${el("code", "docs/ssr.js")}. + It integrates with the SSR build process using the ${el("code", "registerClientFile")} function + from ${el("code", "docs/ssr.js")}. `), el(code, { content: ` // Basic usage of an ireland component el(ireland, { - src: fileURL("./components/examples/path/to/component.js"), - exportName: "NamedExport", // optional, defaults to "default", + src: fileURL("./components/examples/path/to/component.js"), + exportName: "NamedExport", // optional, defaults to "default", })`, page_id }), el("p").append(...T` @@ -96,27 +99,27 @@ import { path_target, dispatchEvent } from "../docs/ssr.js"; // For each page, render it on the server for(const { id, info } of pages) { - // Create a virtual DOM environment for server-side rendering - const serverDOM = createHTMl(""); - serverDOM.registerGlobally("HTMLScriptElement"); - - // Register deka-dom-el with the virtual DOM - const { el } = await register(serverDOM.dom); - - // Import and render the page component - const { page } = await import(\`../docs/\${id}.html.js\`); - serverDOM.document.body.append( - el(page, { pkg, info }), - ); - - // Process the queue of asynchronous operations - await queue(); - - // Trigger render event handlers - dispatchEvent("oneachrender", document); - - // Write the HTML to the output file - s.echo(serverDOM.serialize()).to(path_target.root+id+".html"); + // Create a virtual DOM environment for server-side rendering + const serverDOM = createHTMl(""); + serverDOM.registerGlobally("HTMLScriptElement"); + + // Register deka-dom-el with the virtual DOM + const { el } = await register(serverDOM.dom); + + // Import and render the page component + const { page } = await import(\`../docs/\${id}.html.js\`); + serverDOM.document.body.append( + el(page, { pkg, info }), + ); + + // Process the queue of asynchronous operations + await queue(); + + // Trigger render event handlers + dispatchEvent("oneachrender", document); + + // Write the HTML to the output file + s.echo(serverDOM.serialize()).to(path_target.root+id+".html"); } // Final build step - trigger SSR end event @@ -126,107 +129,107 @@ dispatchEvent("onssrend"); el(code, { content: ` // From docs/ssr.js - File registration system export function registerClientFile(url, { head, folder = "", replacer } = {}) { - // Ensure folder path ends with a slash - if(folder && !folder.endsWith("/")) folder += "/"; - - // Extract filename from URL - const file_name = url.pathname.split("/").pop(); - - // Create target directory if needed - s.mkdir("-p", path_target.root+folder); - - // Get file content and apply optional replacer function - let content = s.cat(url); - if(replacer) content = s.echo(replacer(content.toString())); - - // Write content to the output directory - content.to(path_target.root+folder+file_name); + // Ensure folder path ends with a slash + if(folder && !folder.endsWith("/")) folder += "/"; - // If a head element was provided, add it to the document - if(!head) return; - head[head instanceof HTMLScriptElement ? "src" : "href"] = file_name; - document.head.append(head); + // Extract filename from URL + const file_name = url.pathname.split("/").pop(); + + // Create target directory if needed + s.mkdir("-p", path_target.root+folder); + + // Get file content and apply optional replacer function + let content = s.cat(url); + if(replacer) content = s.echo(replacer(content.toString())); + + // Write content to the output directory + content.to(path_target.root+folder+file_name); + + // If a head element was provided, add it to the document + if(!head) return; + head[head instanceof HTMLScriptElement ? "src" : "href"] = file_name; + document.head.append(head); } `, page_id }), el("h4", t`Server-Side Rendering`), el(code, { content: ` // From docs/components/ireland.html.js - Server-side component implementation export function ireland({ src, exportName = "default", props = {} }) { - // Calculate relative path for imports - const path = "./"+relative(dir, src.pathname); - - // Generate unique ID for this component instance - const id = "ireland-" + generateComponentId(src); - - // Create placeholder element - const element = el.mark({ type: "ireland", name: ireland.name }); - - // Import and render the component during SSR - queue(import(path).then(module => { - const component = module[exportName]; - element.replaceWith(el(component, props, mark(id))); - })); + // Calculate relative path for imports + const path = "./"+relative(dir, src.pathname); - // Register client-side hydration on first component - if(!componentsRegistry.size) - addEventListener("oneachrender", registerClientPart); - - // Store component info for client-side hydration - componentsRegistry.set(id, { - src, - path: dirFE+"/"+path.split("/").pop(), - exportName, - props, - }); + // Generate unique ID for this component instance + const id = "ireland-" + generateComponentId(src); - return element; + // Create placeholder element + const element = el.mark({ type: "later", name: ireland.name }); + + // Import and render the component during SSR + queue(import(path).then(module => { + const component = module[exportName]; + element.replaceWith(el(component, props, mark(id))); + })); + + // Register client-side hydration on first component + if(!componentsRegistry.size) + addEventListener("oneachrender", registerClientPart); + + // Store component info for client-side hydration + componentsRegistry.set(id, { + src, + path: dirFE+"/"+path.split("/").pop(), + exportName, + props, + }); + + return element; } // Register client-side resources function registerClientPart() { - // Process all component registrations - const todo = Array.from(componentsRegistry.entries()) - .map(([ id, d ]) => { - // Copy the component source file to output directory - registerClientFile(d.src, { - folder: dirFE, - // Replace bare imports for browser compatibility - replacer(file) { - return file.replaceAll( - / from "deka-dom-el(\/signals)?";/g, - \` from "./esm-with-signals.js";\` - ); - } - }); - return [ id, d ]; - }); - - // Serialize the component registry for client-side use - const store = JSON.stringify(JSON.stringify(todo)); - - // Copy client-side scripts to output - registerClientFile(new URL("./ireland.js.js", import.meta.url)); - registerClientFile(new URL("../../dist/esm-with-signals.js", import.meta.url), { folder: dirFE }); - - // Add import map for package resolution - document.head.append( - el("script", { type: "importmap" }).append(\` + // Process all component registrations + const todo = Array.from(componentsRegistry.entries()) + .map(([ id, d ]) => { + // Copy the component source file to output directory + registerClientFile(d.src, { + folder: dirFE, + // Replace bare imports for browser compatibility + replacer(file) { + return file.replaceAll( + / from "deka-dom-el(\/signals)?";/g, + \` from "./esm-with-signals.js";\` + ); + } + }); + return [ id, d ]; + }); + + // Serialize the component registry for client-side use + const store = JSON.stringify(JSON.stringify(todo)); + + // Copy client-side scripts to output + registerClientFile(new URL("./ireland.js.js", import.meta.url)); + registerClientFile(new URL("../../dist/esm-with-signals.js", import.meta.url), { folder: dirFE }); + + // Add import map for package resolution + document.head.append( + el("script", { type: "importmap" }).append(\` { - "imports": { - "deka-dom-el": "./\${dirFE}/esm-with-signals.js", - "deka-dom-el/signals": "./\${dirFE}/esm-with-signals.js" - } + "imports": { + "deka-dom-el": "./\${dirFE}/esm-with-signals.js", + "deka-dom-el/signals": "./\${dirFE}/esm-with-signals.js" + } } - \`.trim()) - ); - - // Add bootstrap script to load components - document.body.append( - el("script", { type: "module" }).append(\` + \`.trim()) + ); + + // Add bootstrap script to load components + document.body.append( + el("script", { type: "module" }).append(\` import { loadIrelands } from "./ireland.js.js"; loadIrelands(new Map(JSON.parse(\${store}))); - \`.trim()) - ); + \`.trim()) + ); } `, page_id }), el("h4", t`Client-Side Hydration`), @@ -235,22 +238,22 @@ loadIrelands(new Map(JSON.parse(\${store}))); import { el } from "./irelands/esm-with-signals.js"; export function loadIrelands(store) { - // Find all marked components in the DOM - document.body.querySelectorAll("[data-dde-mark]").forEach(ireland => { - const { ddeMark } = ireland.dataset; - - // Skip if this component isn't in our registry - if(!store.has(ddeMark)) return; - - // Get component information - const { path, exportName, props } = store.get(ddeMark); - - // Dynamically import the component module - import("./" + path).then(module => { - // Replace the server-rendered element with the client-side version - ireland.replaceWith(el(module[exportName], props)); - }); - }); + // Find all marked components in the DOM + document.body.querySelectorAll("[data-dde-mark]").forEach(ireland => { + const { ddeMark } = ireland.dataset; + + // Skip if this component isn't in our registry + if(!store.has(ddeMark)) return; + + // Get component information + const { path, exportName, props } = store.get(ddeMark); + + // Dynamically import the component module + import("./" + path).then(module => { + // Replace the server-rendered element with the client-side version + ireland.replaceWith(el(module[exportName], props)); + }); + }); } `, page_id }), @@ -287,7 +290,7 @@ export function loadIrelands(store) { ${el("strong", "Export a function:")} Components should be exported as named or default functions `), el("li").append(...T` - ${el("strong", "Return a DOM element:")} The function should return a valid DOM element created with ${el("code", "el()")} + `), el("li").append(...T` ${el("strong", "Accept props:")} Components should accept a props object, even if not using it diff --git a/index.d.ts b/index.d.ts index 011cbdb..187ae54 100644 --- a/index.d.ts +++ b/index.d.ts @@ -73,14 +73,17 @@ export function assignAttribute