generated from coulomb/repo-seed
Add cross-framework analysis to history/ and WP-0017 workplan for the tokens+CSS phase of adopting ~/whynot-design. Component port deferred until upstream ships Lit web components and missing atoms (Card, Modal, Input, Table, Toast). Decisions captured: vendor (not npm), big-bang swap (not pilot), keep off-spec red for btn-danger until upstream defines one.
362 lines
22 KiB
Markdown
362 lines
22 KiB
Markdown
---
|
||
date: 2026-05-23
|
||
topic: whynot-design adoption — cross-framework analysis
|
||
status: research / decision pending
|
||
author: claude (opus-4.7)
|
||
related:
|
||
- ~/whynot-design (the React DS in question)
|
||
- ~/whynot-control (the org control surface — explicitly out of scope)
|
||
- vergabe-teilnahme (the consuming Django app)
|
||
follow-ups:
|
||
- workplan: VERGABE_TEILNAHME-WP-0017 (whynot-design adoption) — not yet drafted, depends on strategy decision below
|
||
- possible new repo: whynot-design-django — see §5
|
||
---
|
||
|
||
# whynot-design — cross-framework adoption analysis
|
||
|
||
> Persisted from a research conversation on 2026-05-23. The conversation was triggered
|
||
> by the question "should vergabe-teilnahme adopt the whynot design system, and how?"
|
||
> The strategic question opened up because whynot-design ships React components
|
||
> while vergabe-teilnahme is Django + HTMX + Tailwind.
|
||
|
||
## 1. The headline insight
|
||
|
||
There is **no community** around "React → Django component porting." That pattern
|
||
doesn't exist as a tool, library, or codified practice. The reason: cross-framework
|
||
UI sharing has, since ~2020, converged on **Web Components** as the standard answer.
|
||
|
||
Every major design system that supports multiple frameworks (Shoelace / Web Awesome,
|
||
IBM Carbon, Adobe Spectrum, Salesforce Lightning, Material Web) either ships Web
|
||
Components or maintains parallel hand-rolled implementations per framework. There is
|
||
no third option that the industry has validated.
|
||
|
||
The W3C Design Tokens Community Group spec reached its **first stable version in
|
||
October 2025**, with Style Dictionary, Figma, Penpot, Tokens Studio, and others as
|
||
reference implementations. Tokens as a cross-framework contract are now genuinely
|
||
portable; *components* still aren't, except via web components.
|
||
|
||
The strategic question therefore reshapes itself: **do we accept the
|
||
parallel-implementations cost, or do we change what `whynot-design` actually ships?**
|
||
|
||
## 2. Research findings — existing approaches and their communities
|
||
|
||
| Approach | What it is | Community / activity | Fit for whynot |
|
||
|---|---|---|---|
|
||
| **Web Components** (Lit, Stencil) | Browser-standard custom elements; works in any framework | Very active; React 19 finally scores perfect on custom-elements-everywhere.com | **High** — best general answer if we're willing to refactor whynot-design |
|
||
| **Shoelace / Web Awesome** | Pre-built Web Components UI kit (Shoelace's successor) | Large, active OSS community | Not direct (competing DS), but proves the model. Has explicit HTMX integration guides |
|
||
| **Stencil JS** | Compiler that emits framework-specific bindings from one source | Mature; IBM/Apple use it; declining mindshare vs. Lit | **Medium** — heavier than Lit but generates per-framework wrappers automatically |
|
||
| **Style Dictionary + DTCG tokens** | One token file → CSS / Tailwind / iOS / Android / Flutter | W3C-backed, stable spec since 2025-10 | **High** — should be our token pipeline regardless of component choice |
|
||
| **Carbon / Spectrum / Lightning / Material pattern** | Single design source, hand-maintained parallel packages per framework | Active but they're 100+ engineer teams | **Low** at our scale — the "expensive but principled" reference path |
|
||
| **Single-spa / micro-frontends** | Multiple frameworks coexisting in one app | Niche, mostly enterprise integration | **Off-topic** — not about sharing components, about hosting heterogeneous apps |
|
||
| **django-cotton / django-components / django-bird** | Django-template-native component libraries; HTMX-friendly | Active, growing in 2024–2026; mature enough to depend on | **High** as the *Django-side runtime* — but doesn't solve cross-framework, just gives Django a real component model |
|
||
| **"Port React to Django" libraries** | (doesn't exist) | None | None |
|
||
|
||
**Read of the field:** the world has settled on "tokens are cross-framework;
|
||
components are framework-native." Most teams either pick web components (one
|
||
implementation, runs anywhere with some friction) or accept parallel implementations
|
||
from a shared design language.
|
||
|
||
## 3. Three viable strategies for whynot's situation
|
||
|
||
**Strategy A — Pivot whynot-design to Web Components.**
|
||
Rewrite `Atoms.jsx` + `Chrome.jsx` as Lit components. Ship one runtime artefact
|
||
(`@whynot/design`) that Django + HTMX templates can drop into a page via plain
|
||
`<whynot-button>` tags. No React dependency anywhere. Tailwind / CSS variables
|
||
continue to drive theming.
|
||
|
||
**Strategy B — Keep whynot-design React-canonical; add a parallel `whynot-design-django` repo.**
|
||
Treat the React JSX as the *visual + API specification*. Hand-port to Django
|
||
partials in a separate, reusable repo so vergabe-teilnahme and future Django
|
||
consumers share one implementation. Tokens stay in whynot-design (single source).
|
||
|
||
**Strategy C — Tokens only; components are framework-local forever.**
|
||
Don't promise component parity. Vergabe re-implements whynot's *look* using the
|
||
tokens but its components live and die in vergabe-teilnahme. Other Django apps
|
||
would do the same fork. Bernd implied this is unsatisfying — he wants a real
|
||
component library for cross-project consistency.
|
||
|
||
The real choice is **A vs. B**.
|
||
|
||
## 4. Pros / cons — A (Web Components) vs. B (parallel Django repo)
|
||
|
||
### Strategy A — Pivot to Web Components (Lit)
|
||
|
||
**Pros**
|
||
- *One* implementation. Zero divergence risk. Bug fixes ship once.
|
||
- Standards-backed; outlives any framework choice. Works in vanilla HTML, Django
|
||
templates, future Vue/Svelte/Solid apps, marketing sites, claude.ai artifacts.
|
||
- HTMX swaps work cleanly — web components are just DOM; HTMX doesn't care.
|
||
- The community is *here*. Tutorials, debugging tools, Storybook integration,
|
||
accessibility testing infrastructure all exist.
|
||
- The Claude atelier in claude.ai can still output JSX prototypes; Lit ports are
|
||
mechanical once the design is decided.
|
||
|
||
**Cons**
|
||
- One-time rewrite cost of the existing 11 React components. Real, but small
|
||
(~267 lines of JSX, not a library of 200 components).
|
||
- Web components have Shadow DOM trade-offs: form-association, deep style scoping,
|
||
and slotting have learning curves. Workable but real friction.
|
||
- React-shaped APIs (`<Button variant="primary">`) translate to attributes /
|
||
properties (`<wn-button variant="primary">`). Slightly less ergonomic in React,
|
||
equally ergonomic everywhere else.
|
||
- Lose the claude.ai → npm symmetry: claude.ai design tool outputs JSX; you'd
|
||
convert each new component.
|
||
- Whoever maintains whynot-design has to learn Lit (small surface area, but new
|
||
vocabulary).
|
||
|
||
### Strategy B — Parallel `whynot-design-django` repo
|
||
|
||
**Pros**
|
||
- Each implementation is idiomatic for its stack. Django templates feel like
|
||
Django; React feels like React.
|
||
- No new technology to learn. Both repos use mature, well-understood patterns.
|
||
- Django consumers get to use `django-cotton` or `django-bird` — real ergonomics,
|
||
not a half-port.
|
||
- claude.ai → JSX → Lit conversion step is avoided. The atelier output is
|
||
*directly* the React reference.
|
||
|
||
**Cons**
|
||
- *Two* implementations forever. Every component change needs both sides updated;
|
||
PRs need cross-repo review discipline.
|
||
- API drift is inevitable. By v0.5 the React `<Button>` and Django `{% button %}`
|
||
will have diverged in props/slots/behavior unless enforced with conformance tests.
|
||
- A *third* consumer stack (Vue, vanilla HTML, native mobile) means a *third* repo.
|
||
Cost scales linearly with the number of stacks; A's cost is roughly constant.
|
||
- The "community" of "people maintaining a React→Django twin DS" is exactly Bernd
|
||
and whoever he recruits. No prior art to lean on.
|
||
- Visual regression testing has to run twice (Playwright against examples in both
|
||
repos) to ensure parity.
|
||
|
||
### Recommendation
|
||
|
||
**Strategy A is the right long-term answer**; B is the right *interim* answer if
|
||
we don't want to halt vergabe-teilnahme to do a Lit rewrite first.
|
||
|
||
A real third path: **A scheduled, B today.** Build B as a reusable Django port now
|
||
to unblock vergabe-teilnahme; treat it as the *bridge* until whynot-design pivots
|
||
to Lit at, say, v0.5. The Django repo becomes obsolete when A lands — that's a
|
||
feature, not a bug; it forces a clear retirement decision instead of accumulating
|
||
tech debt.
|
||
|
||
If we don't want to commit to a future Lit pivot, then B is the right choice on its
|
||
own — just accept the linear-cost-per-stack reality.
|
||
|
||
## 5. Naming, establishing, workflow — if we go with B
|
||
|
||
### Naming proposals
|
||
|
||
| Candidate | Pros | Cons |
|
||
|---|---|---|
|
||
| **`whynot-design-django`** (recommended) | Mirrors `@whynot/design`. Unambiguous about stack. Discoverable. | Slightly long. |
|
||
| `whynot-django` | Short. | Ambiguous — sounds like a Django app, not a DS port. |
|
||
| `whynot-design-py` | Hedges for Jinja2/Flask later. | Vague — Python doesn't pick a template engine. |
|
||
| `whynot-dj` | Compact. | Cryptic. |
|
||
| `whynot-templates` | Honest about what it is. | Doesn't carry the "design" meaning. |
|
||
|
||
**Recommend `whynot-design-django`.** If a Jinja port appears, it gets its own repo
|
||
(`whynot-design-jinja`), not a confusing rename.
|
||
|
||
### Establishing the repo (concrete steps)
|
||
|
||
1. **Gitea org placement:** `gitea-remote:whynot/whynot-design-django.git` — same
|
||
org as `whynot-design`, signaling sibling status.
|
||
2. **Package shape:** a pip-installable Django app (`pyproject.toml`, importable as
|
||
`whynot_design`). Two install paths:
|
||
- `pip install git+ssh://...@v0.1.0` (matches whynot-design's tag-pinned discipline).
|
||
- For dev: editable install (`pip install -e ../whynot-design-django`).
|
||
3. **Layout** (matches Django conventions; mirrors whynot-design's structure):
|
||
```
|
||
whynot-design-django/
|
||
├── README.md ← restates the "match the React spec" contract
|
||
├── CONTRIBUTING.md ← parity rules: any change here must follow a whynot-design change
|
||
├── CHANGELOG.md
|
||
├── pyproject.toml
|
||
├── whynot_design/
|
||
│ ├── __init__.py
|
||
│ ├── apps.py
|
||
│ ├── templates/whynot_design/
|
||
│ │ ├── atoms/eyebrow.html, tag.html, button.html, stage_dot.html, stamp.html, icon.html
|
||
│ │ ├── chrome/top_nav.html, sidebar.html, page_header.html, pipeline_strip.html
|
||
│ │ └── _base.html ← imports the CSS once
|
||
│ ├── templatetags/
|
||
│ │ └── whynot.py ← {% wn_button variant="primary" %}…{% endwn_button %}, etc.
|
||
│ └── static/whynot_design/
|
||
│ └── colors_and_type.css ← vendored from @whynot/design (synced)
|
||
├── tests/
|
||
│ ├── test_components.py ← Django test client, asserts rendered markup matches expected shape
|
||
│ └── parity/ ← Playwright comparison of Django render vs. React render
|
||
└── examples/ ← a Django demo project rendering every component
|
||
```
|
||
4. **Token sync:** a `scripts/sync-from-whynot-design.py` that copies
|
||
`colors_and_type.css` and `tokens/*.json` from a pinned whynot-design ref into
|
||
the static dir + generates a Django settings constant for tokens. Run on every
|
||
whynot-design version bump.
|
||
5. **Underlying component mechanism:** plain `{% include %}` partials (simplest, no
|
||
new dep), `django-cotton` (HTML-like syntax, very ergonomic, active community),
|
||
or `django-components` (most powerful, heaviest). My lean: **`django-cotton`** —
|
||
its `<c-button variant="primary">…</c-button>` syntax parallels React JSX best,
|
||
so the parity contract is more obvious.
|
||
|
||
### What changes in the workflow
|
||
|
||
Before:
|
||
```
|
||
claude.ai atelier ──► whynot-design ──► consumer
|
||
```
|
||
After:
|
||
```
|
||
claude.ai atelier ──► whynot-design ──┬─► whynot-design-django ──► vergabe-teilnahme (+ future Django apps)
|
||
│
|
||
├─► future stacks: whynot-design-jinja / -svelte / -vue / …
|
||
│
|
||
└─► tokens flow into all child repos via a sync script
|
||
```
|
||
|
||
Concretely, this adds three new ongoing obligations:
|
||
|
||
1. **Token sync gate.** When whynot-design ships a token change, the Django repo's
|
||
sync script must run before its next release. CI check: the static CSS in
|
||
`whynot-design-django` must hash-match the CSS at the pinned upstream tag.
|
||
2. **Component parity gate.** When whynot-design adds or modifies a component, an
|
||
issue is opened automatically in `whynot-design-django` referencing the upstream
|
||
PR. A release is blocked until the issue closes. Encode in `CONTRIBUTING.md`
|
||
first; later, a CI cross-repo check.
|
||
3. **Conformance tests.** Each component's Django partial has a Playwright snapshot
|
||
that's compared against the same component rendered in whynot-design's
|
||
`examples/whynot-control/`. Drift = failing build.
|
||
|
||
Vergabe-teilnahme's workplan therefore depends on **whynot-design-django** existing
|
||
first. The vergabe workplan becomes: install the new package, replace partials with
|
||
cotton/include tags, retheme.
|
||
|
||
## 6. Component-level inventory — whynot-design vs. vergabe-teilnahme
|
||
|
||
| whynot-design (React) | vergabe-teilnahme (Django) | Match | Gap action |
|
||
|---|---|---|---|
|
||
| `Eyebrow` (mono uppercase label, fg-3) | — | Missing | New partial `atoms/eyebrow.html` |
|
||
| `Tag` (mono uppercase pill: default / active / draft) | `status_badge.html` (different semantics — domain statuses) | Partial | Replace `status_badge` content with `Tag` variants; keep status→variant mapping |
|
||
| `Button` (3 variants: primary / secondary / ghost; lucide icon support) | CSS classes `.btn-primary`, `.btn-secondary`, `.btn-danger`, `.btn-ghost` | Strong overlap; `btn-danger` not upstream | Adopt 3 base variants; propose `danger` upstream OR retire it (whynot's voice avoids red/destructive emphasis) |
|
||
| `StageDot` (S0–S4 signal levels with grey-ramp + yellow S4) | `phase_nav.html` (numbered phase circles: todo/active/done/warn) | Semantically different — *whynot stages* ≠ *vergabe phases* | Keep both as distinct atoms. Propose `PhaseDot` upstream OR keep vergabe-local |
|
||
| `Stamp` (yellow rotated "DRAFT") | — | Missing | New partial; useful for unfreigegeben items |
|
||
| `Icon` (lucide via `data-lucide`) | inline SVGs / class-based | Missing as a component | New partial that wraps lucide; pull lucide JS into base.html |
|
||
| `TopNav` | `topbar.html` | Both exist; styling and density differ significantly | Reskin vergabe's topbar to whynot's structure (search box + primary action right-aligned) |
|
||
| `Sidebar` | `sidebar.html` | Both exist; nav model differs (vergabe has phase grouping; whynot has Work / Control docs grouping) | Adopt whynot's chrome (item style, eyebrow section headers) but keep vergabe's nav items |
|
||
| `PageHeader` (eyebrow + h1 + lede + actions) | — (inline per template) | Missing | New partial; refactor all page templates to use it |
|
||
| `PipelineStrip` (5-stage horizontal indicator) | `phase_nav.html` (vertical / different) | Semantically related, visually different | Decide: align on whynot's strip OR propose `PipelineStrip` variant upstream |
|
||
| — | `field_row.html` (vergabe-specific) | n/a | Propose upstream — it's a generic key/value display row |
|
||
| — | `breadcrumb.html` | n/a | Propose upstream as `Breadcrumb` |
|
||
| — | `feedback_button.html`, `feedback_modal.html`, `feedback_success.html` | n/a (vergabe-specific UX flow) | Keep vergabe-local; product UI, not DS atoms |
|
||
| — | `freigabe_modal.html`, `freigabe_success.html` | n/a (vergabe-specific) | Keep vergabe-local |
|
||
| — | `search_results.html` (HTMX results target) | n/a | Probably vergabe-local, but a `Listbox`/`Combobox` atom could live upstream — defer |
|
||
|
||
### Gap summary
|
||
|
||
- **5 net-new atoms** in vergabe (Eyebrow, Stamp, PageHeader, Icon wrapper, PipelineStrip).
|
||
- **4 atoms to align** (Tag ↔ status_badge, Button variants, TopNav, Sidebar).
|
||
- **2 atoms to propose upstream** (Breadcrumb, FieldRow).
|
||
- **1 semantic mismatch to resolve** (Stage vs Phase — they're different concepts).
|
||
- **All vergabe-specific flow UI** (feedback, freigabe, search_results) stays
|
||
vergabe-local — that's correct, those are product UI, not design system.
|
||
|
||
### Components missing from whynot-design — full gap list
|
||
|
||
These are components that are absent from `whynot-design` today and would need to
|
||
land there (or in `whynot-design-django`) before vergabe-teilnahme can fully adopt
|
||
the system. Grouped by origin.
|
||
|
||
#### Already exists in vergabe-teilnahme — candidates to promote upstream
|
||
|
||
1. **`Breadcrumb`** — vergabe's `breadcrumb.html`. Trivially generic; no
|
||
vergabe-specific semantics.
|
||
2. **`FieldRow`** — vergabe's `field_row.html` (label + value, 3-column grid). The
|
||
pattern recurs across detail pages; clearly a DS atom.
|
||
3. **`PhaseDot` / `PhaseNav`** *(name TBD)* — vergabe's `phase_nav.html`.
|
||
Semantically distinct from whynot's `StageDot` (numbered phases with
|
||
todo / active / done / warn states, not S0–S4 signal levels). Either upstream
|
||
as a sibling component, or kept vergabe-local. **Open decision.**
|
||
|
||
#### Doesn't exist anywhere yet, but the DS clearly needs them
|
||
|
||
These are components that vergabe-teilnahme has as raw Tailwind classes or ad-hoc
|
||
markup, and whynot-design has no equivalent — but any non-trivial app needs them,
|
||
so they're real DS gaps:
|
||
|
||
4. **`Card`** — whynot's house rules reference cards ("no shadows on cards,"
|
||
"0–4px radii for cards/sheets") but there is no `Card` JSX component. Vergabe
|
||
has `.card` as a Tailwind class. **Surprising omission given the explicit
|
||
design rules.**
|
||
5. **`Input` / `Textarea` / `Select`** — whynot ships no form primitives at all.
|
||
Vergabe has `.form-input` and `.form-label` CSS classes. Required before any
|
||
form-based whynot artefact ships.
|
||
6. **`Modal` / `Dialog`** — whynot ships none. Vergabe has two (`freigabe_modal`,
|
||
`feedback_modal`). Whynot's house rules even prescribe modal radius (8px),
|
||
confirming the intent — just the component is missing.
|
||
7. **`Table`** — whynot ships none. Vergabe has `.table-base`, `.table-header`,
|
||
`.table-row` classes. Any data-heavy view (likely most of vergabe) needs it.
|
||
8. **`Toast` / inline success banner** *(name TBD)* — whynot ships none. Vergabe
|
||
has `freigabe_success.html` and `feedback_success.html` as ad-hoc partials.
|
||
Even with whynot's quiet voice, success / error states need a defined visual.
|
||
|
||
#### Worth flagging but defer until first real need
|
||
|
||
9. **`SearchInput`** — currently inlined inside whynot's `TopNav`. Pulling it out
|
||
as a standalone atom would be useful (vergabe's `search_results.html` is a
|
||
target for an HTMX-driven search box).
|
||
10. **`EmptyState`** — neither codebase has one; vergabe will need it for empty
|
||
list views. Worth designing once before either side fills the gap locally.
|
||
11. **`Tabs`, `Dropdown`, `Tooltip`, `Pagination`, `Avatar`** — common DS atoms;
|
||
neither side has them. Don't add speculatively; add when the first concrete
|
||
use case appears.
|
||
|
||
**Count:** 3 vergabe-existing to promote, 5 genuine DS-level gaps, 5 deferred.
|
||
The 5 "genuine DS-level gaps" — especially `Card`, `Input`, `Modal`, `Table` — are
|
||
**blockers for adoption**: vergabe can't replace its current chrome without them
|
||
existing somewhere shared.
|
||
|
||
### Stylistic gaps to address before "done"
|
||
|
||
- Vergabe currently uses Tailwind utility classes pervasively
|
||
(`.btn-primary { @apply … }`); whynot uses inline-style CSS-variable assignment.
|
||
The Django port should commit to one — recommend CSS variable + utility classes
|
||
(re-derive whynot's component styles as Tailwind `@apply` blocks that read from
|
||
CSS vars), so HTMX-injected fragments don't need their own style scaffolding.
|
||
- Lucide icons aren't wired into vergabe. Need a base.html script tag + an icon partial.
|
||
- Font stack mismatch: vergabe is `ui-sans-serif`; whynot is IBM Plex Sans / Mono /
|
||
Serif. Importing whynot's CSS handles this — but it pulls Google Fonts at runtime.
|
||
Decide self-host vs. CDN.
|
||
- Vergabe's brand blue (`#3b5bdb`) has to go entirely. No mid-state — the whynot
|
||
aesthetic forbids it.
|
||
|
||
## 7. Recommendation and open decision
|
||
|
||
**Recommendation:** Strategy B (parallel `whynot-design-django` repo) as the
|
||
immediate move, with the README explicitly framing it as a **bridge** that becomes
|
||
obsolete if whynot-design eventually pivots to Lit web components. Concretely this
|
||
means **three workplans**, not one:
|
||
|
||
1. **`whynot-design`**: small additions — tag v0.1.0, add upstream candidates
|
||
(`Breadcrumb`, `FieldRow`), document the parity contract for downstream Django port.
|
||
2. **`whynot-design-django`** (new repo): bootstrap, ship v0.1.0 with the 11
|
||
components + 2 vergabe-contributed atoms, set up token-sync + parity tests.
|
||
3. **`vergabe-teilnahme`** (`WP-0017`): consume `whynot-design-django`, retheme,
|
||
pilot on one page first, then full sweep.
|
||
|
||
**Open decision (blocks workplan drafting):**
|
||
|
||
Strategy A (pivot to Lit immediately) vs. Strategy B (parallel Django repo) vs.
|
||
"B now, A later" — Bernd to choose. If "B now, A later," add a workplan stub for
|
||
the Lit pivot in `whynot-design` itself so it doesn't get forgotten. If "A from the
|
||
start," the Django repo doesn't exist and the workplans collapse to two.
|
||
|
||
## Sources
|
||
|
||
- [Shoelace / Web Awesome](https://shoelace.style/) — framework-agnostic web components DS
|
||
- [Lit + Web Components for cross-framework UIs](https://thenewstack.io/how-to-build-framework-agnostic-uis-with-web-components/)
|
||
- [HTMX + Shoelace example](https://binaryigor.com/htmx-with-shoelace-framework-agnostic-components-in-an-example-app.html)
|
||
- [W3C Design Tokens spec first stable version (Oct 2025)](https://www.w3.org/community/design-tokens/2025/10/28/design-tokens-specification-reaches-first-stable-version/)
|
||
- [Style Dictionary cross-platform tokens](https://styledictionary.com/info/tokens/)
|
||
- [Django Cotton — HTML-like component syntax](https://django-cotton.com/)
|
||
- [django-bird](https://github.com/joshuadavidthomas/django-bird)
|
||
- [Salesforce — Beyond Components: multi-framework DS](https://medium.com/salesforce-ux/beyond-components-a-design-system-to-support-multiple-frameworks-cb1e4d511f66)
|
||
- [AgnosticUI post-mortem (rewrite to Lit)](https://frontendmasters.com/blog/post-mortem-rewriting-agnosticui-with-lit-web-components/)
|