mirror of
https://github.com/jaandrle/deka-dom-el
synced 2025-04-04 12:45:54 +02:00
🔤 ⚡
This commit is contained in:
parent
8756dabc55
commit
9251e70015
@ -26,28 +26,23 @@ function Todos(){
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// Setup hash change listener
|
|
||||||
window.addEventListener("hashchange", () => {
|
|
||||||
const hash = location.hash.replace("#", "") || "all";
|
|
||||||
S.action(pageS, "set", /** @type {"all"|"active"|"completed"} */(hash));
|
|
||||||
});
|
|
||||||
|
|
||||||
/** @type {ddeElementAddon<HTMLInputElement>} */
|
/** @type {ddeElementAddon<HTMLInputElement>} */
|
||||||
const onToggleAll = on("change", event => {
|
const onToggleAll = on("change", event => {
|
||||||
const checked = /** @type {HTMLInputElement} */ (event.target).checked;
|
const checked = /** @type {HTMLInputElement} */ (event.target).checked;
|
||||||
S.action(todosS, "completeAll", checked);
|
S.action(todosS, "completeAll", checked);
|
||||||
});
|
});
|
||||||
|
const formNewTodo = "newTodo";
|
||||||
/** @type {ddeElementAddon<HTMLFormElement>} */
|
/** @type {ddeElementAddon<HTMLFormElement>} */
|
||||||
const onSubmitNewTodo = on("submit", event => {
|
const onSubmitNewTodo = on("submit", event => {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
const input = /** @type {HTMLInputElement} */(
|
const input = /** @type {HTMLInputElement} */(
|
||||||
/** @type {HTMLFormElement} */(event.target).elements.namedItem("newTodo")
|
/** @type {HTMLFormElement} */(event.target).elements.namedItem(formNewTodo)
|
||||||
);
|
);
|
||||||
const title = input.value.trim();
|
const title = input.value.trim();
|
||||||
if (title) {
|
if (!title) return;
|
||||||
S.action(todosS, "add", title);
|
|
||||||
input.value = "";
|
S.action(todosS, "add", title);
|
||||||
}
|
input.value = "";
|
||||||
});
|
});
|
||||||
const onClearCompleted = on("click", () => S.action(todosS, "clearCompleted"));
|
const onClearCompleted = on("click", () => S.action(todosS, "clearCompleted"));
|
||||||
const onDelete = on("todo:delete", ev =>
|
const onDelete = on("todo:delete", ev =>
|
||||||
@ -61,15 +56,16 @@ function Todos(){
|
|||||||
el("form", null, onSubmitNewTodo).append(
|
el("form", null, onSubmitNewTodo).append(
|
||||||
el("input", {
|
el("input", {
|
||||||
className: "new-todo",
|
className: "new-todo",
|
||||||
name: "newTodo",
|
name: formNewTodo,
|
||||||
placeholder: "What needs to be done?",
|
placeholder: "What needs to be done?",
|
||||||
autocomplete: "off",
|
autocomplete: "off",
|
||||||
autofocus: true
|
autofocus: true
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
),
|
),
|
||||||
S.el(todosS, todos => todos.length
|
S.el(todosS, todos => !todos.length
|
||||||
? el("main", { className: "main" }).append(
|
? el()
|
||||||
|
: el("main", { className: "main" }).append(
|
||||||
el("input", {
|
el("input", {
|
||||||
id: "toggle-all",
|
id: "toggle-all",
|
||||||
className: "toggle-all",
|
className: "toggle-all",
|
||||||
@ -82,50 +78,40 @@ function Todos(){
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
: el()
|
|
||||||
),
|
),
|
||||||
S.el(todosS, todos => memo(todos.length, length=> length
|
S.el(todosS, todos => !todos.length
|
||||||
? el("footer", { className: "footer" }).append(
|
? el()
|
||||||
|
: el("footer", { className: "footer" }).append(
|
||||||
el("span", { className: "todo-count" }).append(
|
el("span", { className: "todo-count" }).append(
|
||||||
S.el(S(() => todosS.get().filter(todo => !todo.completed).length),
|
noOfLeft(todos)
|
||||||
length=> el("strong").append(
|
),
|
||||||
length + " ",
|
memo("filters", ()=>
|
||||||
length === 1 ? "item left" : "items left"
|
el("ul", { className: "filters" }).append(
|
||||||
|
...[ "All", "Active", "Completed" ].map(textContent =>
|
||||||
|
el("li").append(
|
||||||
|
el("a", {
|
||||||
|
textContent,
|
||||||
|
classList: { selected: S(()=> pageS.get() === textContent.toLowerCase()) },
|
||||||
|
href: `#${textContent.toLowerCase()}`
|
||||||
|
})
|
||||||
|
)
|
||||||
)
|
)
|
||||||
)
|
|
||||||
),
|
|
||||||
el("ul", { className: "filters" }).append(
|
|
||||||
el("li").append(
|
|
||||||
el("a", {
|
|
||||||
textContent: "All",
|
|
||||||
className: S(()=> pageS.get() === "all" ? "selected" : ""),
|
|
||||||
href: "#"
|
|
||||||
}),
|
|
||||||
),
|
),
|
||||||
el("li").append(
|
|
||||||
el("a", {
|
|
||||||
textContent: "Active",
|
|
||||||
className: S(()=> pageS.get() === "active" ? "selected" : ""),
|
|
||||||
href: "#active"
|
|
||||||
}),
|
|
||||||
),
|
|
||||||
el("li").append(
|
|
||||||
el("a", {
|
|
||||||
textContent: "Completed",
|
|
||||||
className: S(()=> pageS.get() === "completed" ? "selected" : ""),
|
|
||||||
href: "#completed"
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
),
|
),
|
||||||
S.el(S(() => todosS.get().some(todo => todo.completed)),
|
!todos.some(todo => todo.completed)
|
||||||
hasTodosCompleted=> hasTodosCompleted
|
? el()
|
||||||
? el("button", { textContent: "Clear completed", className: "clear-completed" }, onClearCompleted)
|
: el("button", { textContent: "Clear completed", className: "clear-completed" }, onClearCompleted)
|
||||||
: el()
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
: el()
|
)
|
||||||
))
|
|
||||||
);
|
);
|
||||||
|
/** @param {Todo[]} todos */
|
||||||
|
function noOfLeft(todos){
|
||||||
|
const { length }= todos.filter(todo => !todo.completed);
|
||||||
|
return el("strong").append(
|
||||||
|
length + " ",
|
||||||
|
length === 1 ? "item left" : "items left"
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -177,10 +163,11 @@ function TodoItem({ id, title, completed }) {
|
|||||||
}
|
}
|
||||||
isEditing.set(false);
|
isEditing.set(false);
|
||||||
});
|
});
|
||||||
|
const formEdit = "edit";
|
||||||
/** @type {ddeElementAddon<HTMLFormElement>} */
|
/** @type {ddeElementAddon<HTMLFormElement>} */
|
||||||
const onSubmitEdit = on("submit", event => {
|
const onSubmitEdit = on("submit", event => {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
const input = /** @type {HTMLFormElement} */(event.target).elements.namedItem("edit");
|
const input = /** @type {HTMLFormElement} */(event.target).elements.namedItem(formEdit);
|
||||||
const value = /** @type {HTMLInputElement} */(input).value.trim();
|
const value = /** @type {HTMLInputElement} */(input).value.trim();
|
||||||
if (value) {
|
if (value) {
|
||||||
dispatchEdit({ id, title: value });
|
dispatchEdit({ id, title: value });
|
||||||
@ -209,16 +196,15 @@ function TodoItem({ id, title, completed }) {
|
|||||||
el("label", { textContent: title }, onStartEdit),
|
el("label", { textContent: title }, onStartEdit),
|
||||||
el("button", { className: "destroy" }, onDelete)
|
el("button", { className: "destroy" }, onDelete)
|
||||||
),
|
),
|
||||||
S.el(isEditing, editing => editing
|
S.el(isEditing, editing => !editing
|
||||||
? el("form", null, onSubmitEdit).append(
|
? el()
|
||||||
|
: el("form", null, onSubmitEdit).append(
|
||||||
el("input", {
|
el("input", {
|
||||||
className: "edit",
|
className: "edit",
|
||||||
name: "edit",
|
name: formEdit,
|
||||||
value: title,
|
value: title,
|
||||||
"data-id": id
|
|
||||||
}, onBlurEdit, onKeyDown, addFocus)
|
}, onBlurEdit, onKeyDown, addFocus)
|
||||||
)
|
)
|
||||||
: el()
|
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -354,7 +340,7 @@ function todosSignal(){
|
|||||||
*/
|
*/
|
||||||
function routerSignal(signal){
|
function routerSignal(signal){
|
||||||
const initial = location.hash.replace("#", "") || "all";
|
const initial = location.hash.replace("#", "") || "all";
|
||||||
return signal(initial, {
|
const out = signal(initial, {
|
||||||
/**
|
/**
|
||||||
* Set the current route
|
* Set the current route
|
||||||
* @param {"all"|"active"|"completed"} hash - The route to set
|
* @param {"all"|"active"|"completed"} hash - The route to set
|
||||||
@ -364,6 +350,14 @@ function routerSignal(signal){
|
|||||||
this.value = hash;
|
this.value = hash;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Setup hash change listener
|
||||||
|
window.addEventListener("hashchange", () => {
|
||||||
|
const hash = location.hash.replace("#", "") || "all";
|
||||||
|
S.action(out, "set", /** @type {"all"|"active"|"completed"} */(hash));
|
||||||
|
});
|
||||||
|
|
||||||
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -177,6 +177,13 @@ export function page({ pkg, info }){
|
|||||||
clearCompleted() {
|
clearCompleted() {
|
||||||
this.value = this.value.filter(todo => !todo.completed);
|
this.value = this.value.filter(todo => !todo.completed);
|
||||||
},
|
},
|
||||||
|
/**
|
||||||
|
* Mark all todos as completed or active
|
||||||
|
* @param {boolean} state - Whether to mark todos as completed or active
|
||||||
|
*/
|
||||||
|
completeAll(state = true) {
|
||||||
|
this.value.forEach(todo => todo.completed = state);
|
||||||
|
},
|
||||||
/**
|
/**
|
||||||
* Handle cleanup when signal is cleared
|
* Handle cleanup when signal is cleared
|
||||||
*/
|
*/
|
||||||
@ -237,8 +244,32 @@ export function page({ pkg, info }){
|
|||||||
The derived signal automatically recalculates whenever either the todos list or the current filter changes,
|
The derived signal automatically recalculates whenever either the todos list or the current filter changes,
|
||||||
ensuring the UI always shows the correct filtered todos.
|
ensuring the UI always shows the correct filtered todos.
|
||||||
`),
|
`),
|
||||||
|
|
||||||
|
el("h4", t`2. Toggle All Functionality`),
|
||||||
|
el(code, { content: `
|
||||||
|
/** @type {ddeElementAddon<HTMLInputElement>} */
|
||||||
|
const onToggleAll = on("change", event => {
|
||||||
|
const checked = /** @type {HTMLInputElement} */ (event.target).checked;
|
||||||
|
S.action(todosS, "completeAll", checked);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Using the toggle-all functionality in the UI
|
||||||
|
el("input", {
|
||||||
|
id: "toggle-all",
|
||||||
|
className: "toggle-all",
|
||||||
|
type: "checkbox"
|
||||||
|
}, onToggleAll),
|
||||||
|
el("label", { htmlFor: "toggle-all", title: "Mark all as complete" }),
|
||||||
|
`, page_id }),
|
||||||
|
|
||||||
|
el("p").append(T`
|
||||||
|
The "toggle all" checkbox allows users to mark all todos as completed or active. When the checkbox
|
||||||
|
is toggled, it calls the ${el("code", "completeAll")} action on the todos signal, passing the current
|
||||||
|
checked state. This is a good example of how signals and actions can be used to manage application
|
||||||
|
state in a clean, declarative way.
|
||||||
|
`),
|
||||||
|
|
||||||
el("h4", t`2. Local Component State`),
|
el("h4", t`3. Local Component State`),
|
||||||
el(code, { content: `
|
el(code, { content: `
|
||||||
function TodoItem({ id, title, completed }) {
|
function TodoItem({ id, title, completed }) {
|
||||||
const { host }= scope;
|
const { host }= scope;
|
||||||
@ -273,7 +304,7 @@ export function page({ pkg, info }){
|
|||||||
UI feedback while still communicating changes to the parent via events.
|
UI feedback while still communicating changes to the parent via events.
|
||||||
`),
|
`),
|
||||||
|
|
||||||
el("h4", t`3. Reactive Properties`),
|
el("h4", t`4. Reactive Properties`),
|
||||||
el(code, { content: `
|
el(code, { content: `
|
||||||
// Dynamic class attributes
|
// Dynamic class attributes
|
||||||
el("a", {
|
el("a", {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user