1
0
mirror of https://github.com/jaandrle/deka-dom-el synced 2025-04-03 20:35:53 +02:00

Compare commits

..

No commits in common. "a459c0d7861d4b2bb54edce5b61b3a20959522d8" and "8415f5cf6fe96169014cbc466a905cce0d6f43ee" have entirely different histories.

7 changed files with 44 additions and 48 deletions

View File

@ -1,7 +1,7 @@
**Alpha** **Alpha**
| [source code on GitHub](https://github.com/jaandrle/deka-dom-el) | [source code on GitHub](https://github.com/jaandrle/deka-dom-el)
| [*mirrored* on Gitea](https://gitea.jaandrle.cz/jaandrle/deka-dom-el) | [*mirrored* on Gitea](https://gitea.jaandrle.cz/jaandrle/deka-dom-el)
| [npm package](https://www.npmjs.com/package/deka-dom-el) | [![git3moji](https://img.shields.io/badge/git3moji%E2%80%93v1-%E2%9A%A1%EF%B8%8F%F0%9F%90%9B%F0%9F%93%BA%F0%9F%91%AE%F0%9F%94%A4-fffad8.svg?style=flat-square)](https://robinpokorny.github.io/git3moji/) <!-- editorconfig-checker-disable-line -->
<p align="center"> <p align="center">
<img src="docs/assets/logo.svg" alt="Deka DOM Elements Logo" width="180" height="180"> <img src="docs/assets/logo.svg" alt="Deka DOM Elements Logo" width="180" height="180">
@ -24,6 +24,9 @@ function EmojiCounter({ initial }) {
const count = S(0); const count = S(0);
const emoji = S(initial); const emoji = S(initial);
/** @param {HTMLOptionElement} el */
const isSelected= el=> (el.selected= el.value===initial);
// 🔄 View - UI updates automatically when signals change // 🔄 View - UI updates automatically when signals change
return el().append( return el().append(
el("p", { el("p", {
@ -84,7 +87,7 @@ npm install deka-dom-el --save
…or via CDN / Direct Script: …or via CDN / Direct Script:
For CDN links and various build formats (ESM/IIFE, with/without signals, minified/unminified), see the [interactive For CDN links and various build formats (ESM/IIFE, with/without signals, minified/unminified), see the [interactive
format selector](https://jaandrle.github.io/deka-dom-el/#h-getting-started) on the documentation site. format selector](https://jaandrle.github.io/deka-dom-el/) on the documentation site.
```html ```html
<!-- Example with IIFE build (creates a global DDE object) --> <!-- Example with IIFE build (creates a global DDE object) -->

View File

@ -81,11 +81,11 @@ function Todos(){
) )
) )
), ),
S.el(todosS, ({ length }) => !length S.el(todosS, todos => !todos.length
? el() ? el()
: el("footer", { className: "footer" }).append( : el("footer", { className: "footer" }).append(
el("span", { className: "todo-count" }).append( el("span", { className: "todo-count" }).append(
el("strong", length + " " + (length === 1 ? "item" : "items")), noOfLeft()
), ),
memo("filters", ()=> memo("filters", ()=>
el("ul", { className: "filters" }).append( el("ul", { className: "filters" }).append(
@ -100,7 +100,7 @@ function Todos(){
) )
), ),
), ),
length - todosRemainingS.get() === 0 todos.length - todosRemainingS.get() === 0
? el() ? el()
: memo("delete", () => : memo("delete", () =>
el("button", el("button",
@ -110,6 +110,13 @@ function Todos(){
) )
) )
); );
function noOfLeft(){
const length = todosRemainingS.get();
return el("strong").append(
length + " ",
length === 1 ? "item left" : "items left"
)
}
} }
/** /**

View File

@ -16,11 +16,7 @@ import { getLibraryUrl } from "./components/getLibraryUrl.html.js";
/** @param {string} url */ /** @param {string} url */
const fileURL= url=> new URL(url, import.meta.url); const fileURL= url=> new URL(url, import.meta.url);
const references= { const references= {
npm: { w_mvv:{
title: t`NPM package page for dd<el>`,
href: "https://www.npmjs.com/package/deka-dom-el",
},
w_mvv: {
title: t`Wikipedia: Modelviewviewmodel`, title: t`Wikipedia: Modelviewviewmodel`,
href: "https://en.wikipedia.org/wiki/Model%E2%80%93view%E2%80%93viewmodel", href: "https://en.wikipedia.org/wiki/Model%E2%80%93view%E2%80%93viewmodel",
}, },
@ -110,10 +106,6 @@ export function page({ pkg, info }){
`), `),
el("h4", "npm installation"), el("h4", "npm installation"),
el(code, { content: "npm install deka-dom-el --save", language: "shell", page_id }), el(code, { content: "npm install deka-dom-el --save", language: "shell", page_id }),
el("p").append(T`
see ${el("a", { textContent: "package page", ...references.npm, target: "_blank" })}.
`),
el("h4", "CDN / Direct Script Usage"), el("h4", "CDN / Direct Script Usage"),
el("p").append(T` el("p").append(T`
Use the interactive selector below to choose your preferred format: Use the interactive selector below to choose your preferred format:

View File

@ -311,9 +311,9 @@ export function page({ pkg, info }){
el(code, { content: ` el(code, { content: `
// Dynamic class attributes // Dynamic class attributes
el("a", { el("a", {
textContent, textContent: "All",
classList: { selected: S(()=> pageS.get() === textContent.toLowerCase()) }, className: S(()=> pageS.get() === "all" ? "selected" : ""),
href: \`#\${textContent.toLowerCase()}\` href: "#"
}) })
// Reactive classList // Reactive classList
@ -355,25 +355,18 @@ export function page({ pkg, info }){
el("h4", t`Memoizing UI Sections`), el("h4", t`Memoizing UI Sections`),
el(code, { content: ` el(code, { content: `
S.el(todosS, ({ length }) => !length S.el(todosS, todos => memo(todos.length, length=> length
? el() ? el("footer", { className: "footer" }).append(
: el("footer", { className: "footer" }).append( // Footer content...
// …
memo("filters", ()=>
// …
el("a", {
textContent,
classList: { selected: S(()=> pageS.get() === textContent.toLowerCase()) },
href: \`#\${textContent.toLowerCase()}\`
})
// …
) )
: el()
)) ))
`, page_id }), `, page_id }),
el("p").append(T` el("p").append(T`
We memoize the UI section and uses derived signal for the classList. Re-rendering this part is therefore By memoizing based on the todos length, the entire footer component is only re-rendered
unnecessary when the number of todos changes. when todos are added or removed, not when their properties change. This improves performance
by avoiding unnecessary DOM operations.
`), `),
el("div", { className: "tip" }).append( el("div", { className: "tip" }).append(
@ -396,10 +389,8 @@ export function page({ pkg, info }){
`), `),
el(code, { content: ` el(code, { content: `
// Event handlers in the main component // Event handlers in the main component
const onDelete = on("todo:delete", ev => const onDelete = on("todo:delete", ev => S.action(todosS, "delete", ev.detail));
S.action(todosS, "delete", /** @type {{ detail: Todo["id"] }} */(ev).detail)); const onEdit = on("todo:edit", ev => S.action(todosS, "edit", ev.detail));
const onEdit = on("todo:edit", ev =>
S.action(todosS, "edit", /** @type {{ detail: Partial<Todo> & { id: Todo["id"] } }} */(ev).detail));
`, page_id }), `, page_id }),
el("h4", t`2. The TodoItem Component with Scopes and Local State`), el("h4", t`2. The TodoItem Component with Scopes and Local State`),
@ -531,24 +522,25 @@ export function page({ pkg, info }){
el("h4", t`Conditional Todo List`), el("h4", t`Conditional Todo List`),
el(code, { content: ` el(code, { content: `
S.el(todosS, todos => todos.length S.el(todosS, todos => todos.length
? el() ? el("main", { className: "main" }).append(
: el("main", { className: "main" }).append(
// Main content with toggle all and todo list // Main content with toggle all and todo list
) )
: el()
) )
`, page_id }), `, page_id }),
el("h4", t`Conditional Edit Form`), el("h4", t`Conditional Edit Form`),
el(code, { content: ` el(code, { content: `
S.el(isEditing, editing => !editing S.el(isEditing, editing => editing
? el() ? el("form", null, onSubmitEdit).append(
: el("form", null, onSubmitEdit).append(
el("input", { el("input", {
className: "edit", className: "edit",
name: formEdit, name: "edit",
value: title, value: title,
"data-id": id
}, onBlurEdit, onKeyDown, addFocus) }, onBlurEdit, onKeyDown, addFocus)
) )
: el()
) )
`, page_id }), `, page_id }),
@ -638,7 +630,7 @@ export function page({ pkg, info }){
${el("strong", "Declarative Class Management:")} Using the classList property for cleaner class handling ${el("strong", "Declarative Class Management:")} Using the classList property for cleaner class handling
`), `),
el("li").append(T` el("li").append(T`
${el("strong", "Focus Management:")} Reliable input focus with requestAnimationFrame ${el("strong", "Focus Management:")} Reliable input focus with setTimeout
`), `),
el("li").append(T` el("li").append(T`
${el("strong", "Persistent Storage:")} Automatically saving application state with signal listeners ${el("strong", "Persistent Storage:")} Automatically saving application state with signal listeners

View File

@ -1,8 +1,9 @@
import { T, t } from "./utils/index.js"; import { T, t } from "./utils/index.js";
export const info= { export const info= {
title: t`Ireland Components`, title: t`Ireland Components`,
fullTitle: t`Server-Side Pre-Rendering and Client-Side Rehydration`, fullTitle: t`Interactive Demo Components with Server-Side Pre-Rendering`,
description: t`Using Ireland components for server-side pre-rendering and client-side rehydration`, description: t`Creating live, interactive component examples in documentation with server-side
rendering and client-side hydration.`,
}; };
import { el } from "deka-dom-el"; import { el } from "deka-dom-el";

4
package-lock.json generated
View File

@ -1,12 +1,12 @@
{ {
"name": "deka-dom-el", "name": "deka-dom-el",
"version": "0.9.4-alpha", "version": "0.9.3-alpha",
"lockfileVersion": 3, "lockfileVersion": 3,
"requires": true, "requires": true,
"packages": { "packages": {
"": { "": {
"name": "deka-dom-el", "name": "deka-dom-el",
"version": "0.9.4-alpha", "version": "0.9.3-alpha",
"license": "MIT", "license": "MIT",
"devDependencies": { "devDependencies": {
"@size-limit/preset-small-lib": "~11.2", "@size-limit/preset-small-lib": "~11.2",

View File

@ -1,10 +1,10 @@
{ {
"name": "deka-dom-el", "name": "deka-dom-el",
"version": "0.9.4-alpha", "version": "0.9.3-alpha",
"description": "A low-code library that simplifies the creation of native DOM elements/components using small wrappers and tweaks.", "description": "A low-code library that simplifies the creation of native DOM elements/components using small wrappers and tweaks.",
"author": "Jan Andrle <andrle.jan@centrum.cz>", "author": "Jan Andrle <andrle.jan@centrum.cz>",
"license": "MIT", "license": "MIT",
"homepage": "https://jaandrle.github.io/deka-dom-el/", "homepage": "https://github.com/jaandrle/deka-dom-el",
"repository": { "repository": {
"type": "git", "type": "git",
"url": "git+ssh://git@github.com/jaandrle/deka-dom-el.git" "url": "git+ssh://git@github.com/jaandrle/deka-dom-el.git"
@ -52,6 +52,7 @@
"maxdepth": 3, "maxdepth": 3,
"maxcomplexity": 14, "maxcomplexity": 14,
"globals": { "globals": {
"requestIdleCallback": false,
"AbortController": false, "AbortController": false,
"AbortSignal": false, "AbortSignal": false,
"FinalizationRegistry": false "FinalizationRegistry": false