diff --git a/.github/ISSUE_TEMPLATE/bug_report.md b/.github/ISSUE_TEMPLATE/bug_report.md new file mode 100644 index 0000000..ba8e321 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/bug_report.md @@ -0,0 +1,40 @@ +--- +name: Bug report +about: Create a report to help us improve +title: ":bug: " +labels: bug +assignees: '' +--- + +## Bug Description + + +## Steps to Reproduce + +1. +2. +3. + +## Expected Behavior + + +## Actual Behavior + + +## Code Sample + +```js +// Your code here +``` + +## Environment +- Browser and version: +- OS: +- dd version: +- Other relevant details: + +## Screenshots + + +## Additional Context + \ No newline at end of file diff --git a/.github/ISSUE_TEMPLATE/documentation.md b/.github/ISSUE_TEMPLATE/documentation.md new file mode 100644 index 0000000..df30219 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/documentation.md @@ -0,0 +1,22 @@ +--- +name: Documentation improvement +about: Suggest improvements to the documentation +title: ":abc: " +labels: documentation +assignees: '' +--- + +## Documentation Area + + +## Current Issue + + +## Suggested Improvement + + +## Example Content + + +## Additional Context + \ No newline at end of file diff --git a/.github/ISSUE_TEMPLATE/feature_request.md b/.github/ISSUE_TEMPLATE/feature_request.md new file mode 100644 index 0000000..a7db801 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/feature_request.md @@ -0,0 +1,29 @@ +--- +name: Feature request +about: Suggest an idea for this project +title: ":zap: " +labels: enhancement +assignees: '' +--- + + +## Problem Statement + + +## Proposed Solution + + +## Use Cases + + +## Example Implementation + +```js +// Example code +``` + +## Alternatives Considered + + +## Additional Context + diff --git a/.github/pull_request_template.md b/.github/pull_request_template.md new file mode 100644 index 0000000..834c750 --- /dev/null +++ b/.github/pull_request_template.md @@ -0,0 +1,39 @@ + + +## Description + + +## Related Issues + + +## Type of Change +- [ ] Bug fix (non-breaking change that fixes an issue) +- [ ] New feature (non-breaking change that adds functionality) +- [ ] Breaking change (fix or feature that would cause existing functionality to change) +- [ ] Documentation update +- [ ] Code refactoring +- [ ] Performance improvement +- [ ] Test update + +## Testing Performed + + +## Screenshots + + +## Checklist +- [ ] My code follows the code style of this project +- [ ] I have performed a self-review of my own code +- [ ] I have added tests that prove my fix is effective or that my feature works +- [ ] I have updated the documentation accordingly +- [ ] My changes generate no new warnings +- [ ] All existing tests are passing + +## Additional Notes + \ No newline at end of file diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md new file mode 100644 index 0000000..0247d8c --- /dev/null +++ b/CONTRIBUTING.md @@ -0,0 +1,174 @@ +# Contributing to Deka DOM Elements + +Thank you for your interest in contributing to Deka DOM Elements (dd or DDE)! This document provides guidelines and +instructions for contributing to the project. + +## Table of Contents + +- [Code of Conduct](#code-of-conduct) +- [Getting Started](#getting-started) +- [Development Workflow](#development-workflow) +- [Commit Guidelines](#commit-guidelines) +- [Pull Request Process](#pull-request-process) +- [Issue Guidelines](#issue-guidelines) +- [Coding Standards](#coding-standards) +- [Testing](#testing) +- [Documentation](#documentation) + +## Code of Conduct + +Please be respectful and inclusive in your interactions with other contributors. We aim to foster a welcoming community +where everyone feels comfortable participating. + +## Getting Started + +1. **Fork the repository**: + - Click the "Fork" button on the GitHub repository + +2. **Clone your fork**: + ```bash + git clone https://github.com/YOUR-USERNAME/deka-dom-el.git + cd deka-dom-el + ``` + +3. **Set up the development environment**: + ```bash + npm ci + ``` + +4. **Add the upstream repository**: + ```bash + git remote add upstream https://github.com/jaandrle/deka-dom-el.git + ``` + +## Development Workflow + +1. **Create a new branch**: + ```bash + git checkout -b your-feature-branch + ``` + Use descriptive branch names that reflect the changes you're making. + +2. **Make your changes**: + - Write clean, modular code + - Follow the project's coding standards (see [Coding Standards](#coding-standards)) + - Include relevant tests for your changes + +3. ~**Run tests**:~ + ```bash + #npm test + ``` + +4. **Build the project**: + ```bash + npm run build + #or + bs/build.js + ``` + +5. **Preview documentation changes** (if applicable): + ```bash + npm run docs + #or + bs/docs.js + ``` + +…see [BS folder](./bs/README.md) for more info. + +## Commit Guidelines + +We use +[![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/) +for commit messages. This helps keep the commit history clear and consistent. + +``` +:emoji: Short summary of the change +``` +…for example: + +``` +:bug: Fix signal update not triggering on nested properties +:zap: Improve event delegation performance +:abc: Add documentation for custom elements +``` + +## Pull Request Process + +1. **Push your changes**: + ```bash + git push origin your-feature-branch + ``` + +2. **Open a Pull Request**: + - Go to the repository on GitHub + - Click "New Pull Request" + - Select your branch + - Provide a clear description of your changes + +3. **PR Guidelines**: + - Use a clear, descriptive title with the appropriate git3moji + - Reference any related issues + - Explain what the changes do and why they are needed + - List any dependencies that are required for the change + - Include screenshots or examples if applicable + +4. **Code Review**: + - Address any feedback from reviewers + - Make necessary changes and push to your branch + - The PR will be updated automatically + +5. **Merge**: + - Once approved, a maintainer will merge your PR + - The main branch is protected, so you cannot push directly to it + +## Issue Guidelines + +When creating an issue, please use the appropriate template and include as much information as possible: + +### Bug Reports + +- Use the `:bug:` emoji in the title +- Clearly describe the issue +- Include steps to reproduce +- Mention your environment (browser, OS, etc.) +- Add screenshots if applicable + +### Feature Requests + +- Use the `:zap:` emoji in the title +- Describe the feature clearly +- Explain why it would be valuable +- Include examples or mockups if possible + +### Documentation Improvements + +- Use the `:abc:` emoji in the title +- Identify what documentation needs improvement +- Suggest specific changes or additions + +## Coding Standards + +- Follow the existing code style in the project +- Use meaningful variable and function names +- Keep functions small and focused +- Add comments for complex logic +- Use TypeScript types appropriately + + + +## Documentation + +- Update the documentation when you add or modify features +- Document both API usage and underlying concepts +- Use clear, concise language +- Include examples where appropriate + +--- + +Thank you for contributing to Deka DOM Elements! Your efforts help make the project better for everyone. diff --git a/README.md b/README.md index 55a1f73..f808d45 100644 --- a/README.md +++ b/README.md @@ -1,6 +1,7 @@ **WIP** (the experimentation phase) | [source code on GitHub](https://github.com/jaandrle/deka-dom-el) | [*mirrored* on Gitea](https://gitea.jaandrle.cz/jaandrle/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/)

Deka DOM Elements Logo @@ -127,6 +128,11 @@ Signals are the reactive backbone of Deka DOM Elements: - [TC39 Signals Proposal](https://github.com/tc39/proposal-signals) (future standard) - [Observer pattern](https://en.wikipedia.org/wiki/Observer_pattern) (underlying concept) +## Contributing + +We welcome contributions from the community! Please see our [Contributing Guide](CONTRIBUTING.md) for details on how to +get started, coding standards, commit guidelines, and the pull request process. + ## Inspiration and Alternatives - [vanjs-org/van](https://github.com/vanjs-org/van) — World's smallest reactive UI framework diff --git a/bs/README.md b/bs/README.md index 33ac7c4..d607e6f 100644 --- a/bs/README.md +++ b/bs/README.md @@ -2,15 +2,18 @@ This project uses [jaandrle/bs: The simplest possible build system using executable/bash scripts]( https://github.com/jaandrle/bs). -#### bs/build.js [--minify|--help] +#### bs/build.js [main|signals] [--no-types|--help] Generates alternative versions of the project (other than native ESM code). Also generates typescript definitions. #### bs/docs.js Generates documentation, from `docs/`. Uses “SSR” technique, using deka-dom-el itself. +For running use `npx serve dist/docs`. + #### bs/lint.sh Lints size of the project, jshint. See configs: - `package.json`: key `size-limit` - `package.json`: key `jshintConfig` +- `.editorconfig` diff --git a/docs/components/mnemonic/elements-init.js b/docs/components/mnemonic/elements-init.js index 0893128..3ae5570 100644 --- a/docs/components/mnemonic/elements-init.js +++ b/docs/components/mnemonic/elements-init.js @@ -9,7 +9,7 @@ export function mnemonic(){ ), el("li").append( el("code", "el(, )[.append(...)]: "), - " — simple element containing only text", + " — simple element containing only text (accepts string, number or signal)", ), el("li").append( el("code", "el(, )[.append(...)]: "), @@ -26,6 +26,6 @@ export function mnemonic(){ el("li").append( el("code", "elNS()()[.append(...)]: "), " — typically SVG elements", - ) + ), ); } diff --git a/docs/components/mnemonic/signals-init.js b/docs/components/mnemonic/signals-init.js index 6e4429e..b9181e7 100644 --- a/docs/components/mnemonic/signals-init.js +++ b/docs/components/mnemonic/signals-init.js @@ -29,7 +29,7 @@ export function mnemonic(){ el("li").append( el("code", "S.clear(...)"), " — off and clear signals (most of the time it is not needed as reactive ", - "attributes and elements are cleared automatically)", + "attributes and elements are handled automatically)", ), ); } diff --git a/docs/index.html.js b/docs/index.html.js index b0bec30..8863209 100644 --- a/docs/index.html.js +++ b/docs/index.html.js @@ -82,7 +82,7 @@ export function page({ pkg, info }){ el("p").append(T` By separating these concerns, your code becomes more modular, testable, and easier to maintain. This - approach ${el("strong", "is not")} something new and/or special to dd. It’s based on ${el("a", { + approach ${el("strong", "is not something new and/or special to dd")}. It’s based on ${el("a", { textContent: "MVC", ...references.w_mvc })} (${el("a", { textContent: "MVVM", ...references.w_mvv })}), but is there presented in simpler form. `), @@ -154,10 +154,14 @@ export function page({ pkg, info }){ Interactive demos with server-side pre-rendering`), el("li").append(T`${el("a", { href: "p13-appendix.html" }).append(el("strong", "Appendix & Summary"))} — Comprehensive reference and best practices`), + el("li").append(T`${el("a", { href: "p14-converter.html" }).append(el("strong", "HTML Converter"))} — + Convert HTML to dd JavaScript code`), + el("li").append(T`${el("a", { href: "p15-examples.html" }).append(el("strong", "Examples Gallery"))} — + Real-world application examples and case studies`), ), el("p").append(T` Each section builds on the previous ones, so we recommend following them in order. Let’s get started with the basics of creating elements! `), ); -} +} \ No newline at end of file diff --git a/docs/p02-elements.html.js b/docs/p02-elements.html.js index bda75bf..ff47db2 100644 --- a/docs/p02-elements.html.js +++ b/docs/p02-elements.html.js @@ -208,6 +208,6 @@ export function page({ pkg, info }){ `), ), - el(mnemonic) + el(mnemonic), ); } diff --git a/docs/p03-events.html.js b/docs/p03-events.html.js index acbd302..9e1605b 100644 --- a/docs/p03-events.html.js +++ b/docs/p03-events.html.js @@ -136,7 +136,7 @@ export function page({ pkg, info }){ el("li", t`Set up lifecycle behaviors`), el("li", t`Integrate third-party libraries`), el("li", t`Create reusable element behaviors`), - el("li", t`Capture element references`) + el("li", t`Capture element references`), // TODO: add example? ) ), el("p").append(T` diff --git a/docs/p04-signals.html.js b/docs/p04-signals.html.js index f1f1849..06560e9 100644 --- a/docs/p04-signals.html.js +++ b/docs/p04-signals.html.js @@ -277,7 +277,72 @@ export function page({ pkg, info }){ `), el("li").append(T` ${el("strong", "Avoid infinite loops")}: Be careful when one signal updates another in a subscription - `) + `), + ), + el("p").append(T` + While signals provide powerful reactivity for complex UI interactions, they’re not always necessary. + A good approach is to started with variables/constants and when necessary, convert them to signals. + `), + + el("div", { className: "tabs" }).append( + el("div", { className: "tab", dataTab: "events" }).append( + el("h4", t`We can process form events without signals`), + el("p", t`This can be used when the form data doesn’t need to be reactive and we just waiting for + results.`), + el(code, { page_id, content: ` + const onFormSubmit = on("submit", e => { + e.preventDefault(); + const formData = new FormData(e.currentTarget); + // this can be sent to a server + // or processed locally + // e.g.: console.log(Object.fromEntries(formData)) + }); + // … + return el("form", null, onFormSubmit).append( + // … + ); + ` }) + ), + + el("div", { className: "tab", dataTab: "variables" }).append( + el("h4", t`We can use variables without signals`), + el("p", t`We use this when we dont’t need to reflect changes in the elsewhere (UI).`), + el(code, { page_id, content: ` + let canSubmit = false; + + const onFormSubmit = on("submit", e => { + e.preventDefault(); + if(!canSubmit) return; // some message + // … + }); + const onAllowSubmit = on("click", e => { + canSubmit = true; + }); + `}), + ), + + el("div", { className: "tab", dataTab: "state" }).append( + el("h4", t`Using signals`), + el("p", t`We use this when we need to reflect changes for example in the UI (e.g. enable/disable + buttons).`), + el(code, { page_id, content: ` + const canSubmit = S(false); + + const onFormSubmit = on("submit", e => { + e.preventDefault(); + // … + }); + const onAllowSubmit = on("click", e => { + canSubmit.set(true); + }); + + return el("form", null, onFormSubmit).append( + // ... + el("button", { textContent: "Allow Submit", type: "button" }, onAllowSubmit), + el("button", { disabled: S(()=> !canSubmit), textContent: "Submit" }) + ); + `}), + ), ), el("div", { className: "troubleshooting" }).append( @@ -298,6 +363,6 @@ export function page({ pkg, info }){ ) ), - el(mnemonic) + el(mnemonic), ); } diff --git a/docs/p05-scopes.html.js b/docs/p05-scopes.html.js index 81b6df9..418933e 100644 --- a/docs/p05-scopes.html.js +++ b/docs/p05-scopes.html.js @@ -55,7 +55,7 @@ export function page({ pkg, info }){ el(MyComponent); function MyComponent() { - // 2. access the host element + // 2. access the host element (or other scope related values) const { host } = scope; // 3. Add behavior to host diff --git a/docs/p07-debugging.html.js b/docs/p07-debugging.html.js index dd430a9..44f7517 100644 --- a/docs/p07-debugging.html.js +++ b/docs/p07-debugging.html.js @@ -80,7 +80,7 @@ export function page({ pkg, info }){ el("li", t`listeners: A Set of functions called when the signal value changes`), el("li", t`actions: Custom actions that can be performed on the signal`), el("li", t`onclear: Functions to run when the signal is cleared`), - el("li", t`host: Reference to the host element/scope`), + el("li", t`host: Reference to the host element/scope in which the signal was created`), el("li", t`defined: Stack trace information for debugging`), el("li", t`readonly: Boolean flag indicating if the signal is read-only`) ), @@ -114,7 +114,13 @@ export function page({ pkg, info }){ el("ul").append( el("li", t`That you’re using signal.set() to update the value, not modifying objects/arrays directly`), el("li", t`For mutable objects, ensure you’re using actions or making proper copies before updating`), - el("li", t`That the signal is actually connected to the DOM element (check your S.el or attribute binding code)`) + el("li", t`That the signal is actually connected to the DOM element (check your S.el or attribute binding + code)`), + el("li").append(T` + That you’re passing signal corecctly (without using ${el("code", "*.get()")}) and for ${el("code", + "S.el")} that you passing (derived) signals not a function (use ${el("code", + "S.el(S(()=> count.get() % 2), odd=> …)")}). + `), ), el(code, { src: fileURL("./components/examples/debugging/mutations.js"), page_id }), diff --git a/docs/p13-appendix.html.js b/docs/p13-appendix.html.js index 7cf5c2b..5841fbe 100644 --- a/docs/p13-appendix.html.js +++ b/docs/p13-appendix.html.js @@ -129,75 +129,6 @@ export function page({ pkg, info }){ `) ), - el("h4", t`Using Signals Appropriately`), - el("p").append(T` - While signals provide powerful reactivity for complex UI interactions, they’re not always necessary. - `), - - el("div", { className: "tabs" }).append( - el("div", { className: "tab", dataTab: "events" }).append( - el("h4", t`We can process form events without signals`), - el("p", t`This can be used when the form data doesn’t need to be reactive and we just waiting for - results.`), - el(code, { page_id, content: ` - const onFormSubmit = on("submit", e => { - e.preventDefault(); - const formData = new FormData(e.currentTarget); - // this can be sent to a server - // or processed locally - // e.g.: console.log(Object.fromEntries(formData)) - }); - // … - return el("form", null, onFormSubmit).append( - // … - ); - ` }) - ), - - el("div", { className: "tab", dataTab: "variables" }).append( - el("h4", t`We can use variables without signals`), - el("p", t`We use this when we dont’t need to reflect changes in the elsewhere (UI).`), - el(code, { page_id, content: ` - let canSubmit = false; - - const onFormSubmit = on("submit", e => { - e.preventDefault(); - if(!canSubmit) return; // some message - // … - }); - const onAllowSubmit = on("click", e => { - canSubmit = true; - }); - `}), - ), - - el("div", { className: "tab", dataTab: "state" }).append( - el("h4", t`Using signals`), - el("p", t`We use this when we need to reflect changes for example in the UI (e.g. enable/disable - buttons).`), - el(code, { page_id, content: ` - const canSubmit = S(false); - - const onFormSubmit = on("submit", e => { - e.preventDefault(); - // … - }); - const onAllowSubmit = on("click", e => { - canSubmit.set(true); - }); - - return el("form", null, onFormSubmit).append( - // ... - el("button", { textContent: "Allow Submit", type: "button" }, onAllowSubmit), - el("button", { disabled: S(()=> !canSubmit), textContent: "Submit" }) - ); - `}), - ), - el("p").append(T` - A good approach is to started with variables/constants and when necessary, convert them to signals. - `), - ), - el("h4", t`Migrating from Traditional Approaches`), el("p").append(T` When migrating from traditional DOM manipulation or other frameworks to dd: @@ -394,46 +325,46 @@ export function page({ pkg, info }){ el("tr").append( el("th", "Feature"), el("th", "dd"), - el("th", "React"), - el("th", "Vue"), - el("th", "Svelte") + el("th", "VanJS"), + el("th", "Solid"), + el("th", "Alpine") ) ), el("tbody").append( el("tr").append( el("td", "No Build Step Required"), el("td", "✅"), + el("td", "✅"), el("td", "⚠️ JSX needs transpilation"), - el("td", "⚠️ SFC needs compilation"), - el("td", "❌ Requires compilation") + el("td", "✅") ), el("tr").append( - el("td", "Bundle Size (minimal)"), - el("td", "~10-15kb"), - el("td", "~40kb+"), - el("td", "~33kb+"), - el("td", "Minimal runtime") + el("td", "Bundle Size (minified)"), + el("td", "~14kb"), + el("td", "~3kb"), + el("td", "~20kb"), + el("td", "~43kb") ), el("tr").append( el("td", "Reactivity Model"), el("td", "Signal-based"), - el("td", "Virtual DOM diffing"), - el("td", "Proxy-based"), - el("td", "Compile-time reactivity") + el("td", "Signal-based (basics only)"), + el("td", "Signal-based"), + el("td", "MVVM + Proxy") ), el("tr").append( el("td", "DOM Interface"), el("td", "Direct DOM API"), - el("td", "Virtual DOM"), - el("td", "Virtual DOM"), - el("td", "Compiled DOM updates") + el("td", "Direct DOM API"), + el("td", "Compiled DOM updates"), + el("td", "Directive-based") ), el("tr").append( el("td", "Server-Side Rendering"), el("td", "✅ Basic Support"), + el("td", "✅ Basic Support"), el("td", "✅ Advanced"), - el("td", "✅ Advanced"), - el("td", "✅ Advanced") + el("td", "❌") ) ) ), diff --git a/package.json b/package.json index 04020bf..4c8b585 100644 --- a/package.json +++ b/package.json @@ -85,7 +85,12 @@ "modifyEsbuildConfig": { "platform": "browser" }, - "scripts": {}, + "scripts": { + "test": "echo \"Error: no tests yet\"", + "build": "bs/build.js", + "lint": "bs/lint.sh", + "docs": "bs/docs.sh" + }, "keywords": [ "dom", "javascript",