Staffa
A small, opinionated TypeScript component library for the Aberdeen reactive UI library.
import A from "aberdeen";import S from "staffa";
const $user = A.proxy({ name: "", email: "" });
A.mount(document.body, () => { S.main({ title: "Sign up", maxWidth: "40rem", content: () => { S.form({ submit: () => console.log(A.unproxy($user)), content: () => { S.textline({ label: "Name", required: true, bind: A.ref($user, "name") }); S.textline({ label: "Email", type: "email", bind: A.ref($user, "email") }); }, actions: () => S.button({ text: "Create account", type: "submit" }), }); }, });});Staffa is made to look decent out of the box, but easily customizable at runtime.
Install
Section titled “Install”npm install staffa aberdeenAberdeen is a peer dependency. Staffa is published as ESM with TypeScript types.
How it works
Section titled “How it works”Components are functions
Section titled “Components are functions”Every component takes a single typed options object and draws DOM via Aberdeen. No classes, no web components. The S object collects all component functions:
S.button({ text: "Save", disabled: false });S.box({ header: "Settings", content: () => { ... } });Options objects are typed and can be reactive
Section titled “Options objects are typed and can be reactive”All components get their options in a typed object. The object may be an Aberdeen proxy, if you want to update the component in-place.
const $btn = A.proxy({ text: "Save", disabled: false });S.button($btn);// ...later:$btn.disabled = true; // button updates instantlyRich text slots
Section titled “Rich text slots”Anywhere a component takes content, a label, header, button text, dialog body, etc, you can pass either a string or a () => void draw function. Strings render as rich text: *italic*, **bold**, `code`, [link](https://github.com/vanviegen/staffa/blob/main/path). All text is safely escaped.
S.button({ text: "Save **now**" });S.box({ header: "See the [docs](https://github.com/vanviegen/staffa/blob/main/docs)", content: () => { ... } });Surfaces
Section titled “Surfaces”Staffa builds on surfaces: elements marked with .s-s that have their own background and derived text/border tokens. Add modifier classes to colour them:
- level:
.base,.panel,.raised - role:
.primary,.secondary,.gradient,.neutral,.danger,.success,.warning - variant:
.filled,.tonal,.outlined
Components are built from these (S.button is a .s-s.primary.filled, S.box a .s-s.panel, etc.). Because component options include an optional attrs string, which has Aberdeen A() string semantics, you can easily override it:
S.button({ text: "Delete", attrs: ".danger" });S.box({ attrs: ".raised.outlined", content: () => { ... } });Inside any surface, CSS variables are defined for suitable foreground colors ($s-fg, $s-bg, $s-fg-muted, $s-border, $s-accent, …), with color defaulting to $s-fg. By using these, components has access to various foreground colors that will look regardless of the surface it is drawing on.
Dark and light modes
Section titled “Dark and light modes”Dark/light mode is detected from OS preference by default. If you want to override this (based on user preferences), use:
S.setDarkMode(true); // force darkS.setDarkMode(false); // force lightS.setDarkMode(undefined); // follow OSHint: A buttonChooser is probably the right component for a color scheme selector.
CSS reset
Section titled “CSS reset”Staffa includes a lightweight CSS reset that makes bare semantic HTML look a bit better but unsurprising without additional styling.
Theming
Section titled “Theming”The first step in theming is just setting some CSS variables, most commonly the primary and secondary color. This can be done through CSS directly, or using Aberdeen:
A.cssVars["s-primary"] = "#fdda58";A.cssVars["s-secondary"] = "#cc5624";A.cssVars["s-danger"] = "#ee4422";A.cssVars["s-radius"] = "4px";See src/theme.ts for what other CSS variables are being used.
If you need further customization, just add some CSS to override the default styling. For instance, to add your own surface type:
// In filled mode, 's-a' is the foreground and 's-b' is the background. "outlined" and "tonal" use the colors in different ways.A.insertGlobalCss({".s-s.my-surface": "--s-a:white --s-b:#ef6b00"});
S.button({ text: "You'll want to click me", attrs: ".my-surface", click: () => S.alert("Good work!", {attrs: ".my-surface"})});Note that when changing CSS like this, things may break if you upgrade Staffa. The recommended update strategy is therefore: don’t!
If you want to make changes that are dependent upon the current light/dark mode setting, rely on Aberdeen reactivity:
A(() => { if (S.getDarkMode()) { A.cssVars["s-primary"] = "#aa9944"; A.insertGlobalCss({".s-s.my-surface": "--s-a:white --s-b:#444444"}); } else { A.cssVars["s-primary"] = "#fdda58"; A.insertGlobalCss({".s-s.my-surface": "--s-a:black --s-b:#cccccc"}); }});Components
Section titled “Components”Components share naming conventions for options: attrs (outermost element), contentAttrs (children-holding element), inputAttrs (form control element), and <region>Attrs (sub-regions like headerAttrs/footerAttrs). Form components consistently support label, help, error, disabled, required, name through the drawField() helper.
Layout & containers
Section titled “Layout & containers”S.main(opts): app shell, a sticky header withicon,title,subtitle,menu; scrollable content area; footer. SetmaxWidthto center the content.S.box(opts | content): surface with optionalheader/footerand padded body. Pass a function for shorthand{ content }.S.tabs(opts): tablist with live panels and keyboard navigation.S.form(opts | content): form aligning fields in a column or responsive grid, with anactionsbar. Prevents the default page reload.
Form fields
Section titled “Form fields”S.textline(opts): single-line input (text,password,email,number,tel,url,search, dates, …).S.textarea(opts): multi-line input.S.checkbox(opts): labelled checkbox.S.select(opts): single-select dropdown backed by native<select>(styled control, OS dropdown).S.autocomplete(opts): type-ahead combobox withmulti(chips),allowCustom(free text),required, and dynamicoptions.
Dialogs
Section titled “Dialogs”S.dialog(opts): modal dialog with backdrop and fade transition. Thecontentslot receives aclose()function. Lifecycle is tied to the calling scope (disappears when cleaned up). Nesting stacks correctly.S.alert(msg)/S.confirm(msg)/S.prompt(msg, initial?): promise-returning shortcuts.
Actions
Section titled “Actions”S.button(opts | text): button surface; restyle viaattrs(e.g..danger,.outlined), plussize,disabled,icon,href(renders<a role=button>). Defaults to filled.primary.S.buttonGroup(opts): groups buttons,attached(segmented) orspaced.S.buttonChooser(opts): single-select segmented control bound to a value.
S.menuButton(opts)/S.showFloatingMenu(opts): menu actions and floating menus, with keyboard navigation and submenus.S.toast(opts): transient notification at the bottom of the viewport.S.addTooltip(el, opts): tooltip on hover, attached to an existing element.
Two-way binding uses Aberdeen proxies: pass bind: A.ref($obj, "key") to form fields.
Browser (no bundler)
Section titled “Browser (no bundler)”staffa/all.js is a pre-built ESM bundle. Use an import map:
<script type="importmap">{ "imports": { "aberdeen": "https://cdn.jsdelivr.net/npm/aberdeen/dist/src/aberdeen.js", "staffa/all.js": "https://cdn.jsdelivr.net/npm/staffa/dist/staffa.esm.js" }}</script><script type="module"> import A from "aberdeen"; import S from "staffa/all.js"; // ...</script>It includes all components, but not the icons.
Extending Staffa
Section titled “Extending Staffa”Staffa is designed for extension. A component is simply a plain function taking a typed options object and drawing Aberdeen DOM. This section explains the philosophy so extensions follow the same patterns.
Design principles
Section titled “Design principles”-
Components are functions. They take one typed options object, emit Aberdeen DOM, and usually return nothing.
-
Reuse option types. Define options by extending
ContentOptions(for layout components) orFieldOptions(for form controls) fromsrc/core.tsandsrc/components/field.ts. Don’t reinvent fields likeattrs,label,help, etc. -
Reach for reactivity deliberately. Pass option strings straight to
Aas positional args (the caller’s scope). Only wrap a dedicatedA(() => ...)scope where it matters: input elements (recreation loses focus), or large subtrees you don’t want to redraw. UseA.peek(() => ...)when you need a value but must not subscribe. -
Build on surfaces. Mark elements
.s-sand add level/role/variant modifiers. Inside them, use the contextual foreground color CSS variables ($s-fg,$s-bg,$s-border, …) so components adapt to wherever they’re nested. Hard-coding colors in components shouldn’t be needed, but if you must, make sure you set both foreground and background. -
No outer margins. Components don’t margin themselves; spacing is the parent’s job. Content components set default
paddingon the content element;contentAttrsoverrides it. -
Make everything styleable. Provide
attrs,contentAttrs,inputAttrs, and<region>Attrshooks so callers can customize. Applyattrslast so it can override component classes. -
Use semantic HTML and ARIA. Prefer native elements (
<button>,<label>,<form>,<section>) and native behaviour. Add ARIA only where semantics fall short (e.g. tabs, combobox). -
Use CSS. Use
A.insertGlobalCss({...})at module top level to provide (nested) CSS styling for your component. Give your top-level element thes-<component-name>class. Avoid inventing further classes; lean on nesting (&for the element, bare key for descendants) and element/structural selectors. -
Reuse form controls. Use
drawField()and callapplyControlAttrs(). -
Function over form. Provide enough contrast. Stick to UI conventions to help users; buttons have a rounded border, links are underlined, text input background is white, etc.
Adding a component to Staffa
Section titled “Adding a component to Staffa”The previous section is good advice for any project-specific custom, but should definitely be followed for any new components to be included in Staffa. In addition, you’d want to:
- Create
src/components/<name>.ts. - Define
<Name>OptionsextendingContentOptions,FieldOptions, or a plain interface. Add TSDoc on every option. - Add a TSDoc
@exampleon the function. - Register in
src/index.ts(theSobject + type re-export). - Extend
smoke.mjsto render it. Runnpm run smokeandnpm run build.
See src/components/button.ts and src/components/dialog.ts for examples.
Commands
Section titled “Commands”npm run build # compile TypeScript to dist/npm run typecheck # check typesnpm run smoke # render every component in jsdomnpx http-server # allows demo to be viewed at http://localhost:8080/demo