Awesome-omni-skill settings-ui-patterns
Use when creating or modifying UI components, styling, or visual elements related to Settings in OpenChamber.
git clone https://github.com/diegosouzapw/awesome-omni-skill
T=$(mktemp -d) && git clone --depth=1 https://github.com/diegosouzapw/awesome-omni-skill "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/design/settings-ui-patterns" ~/.claude/skills/diegosouzapw-awesome-omni-skill-settings-ui-patterns && rm -rf "$T"
skills/design/settings-ui-patterns/SKILL.mdSettings UI Patterns Skill
Purpose
This skill provides instructions for creating or redesigning Settings pages, informational panels, and configuration interfaces within the OpenChamber application.
Current Canonical Look (2026)
Use this as source of truth for new settings UI work.
- Flat hierarchy first: Prefer spacing + typography hierarchy over boxed backgrounds.
- No unnecessary wrappers: Avoid extra section wrappers that mix unrelated controls.
- No redundant section titles: Do not add headers like
orTheme Preferences
when controls are already self-explanatory.Scaling & Layout - Compact controls: Option chips and radio rows should be dense, not tall.
- Left-leading state icon: Radio/checkbox state icon appears before text.
- Subtle state contrast: Inactive radio labels should be visibly dimmer than active labels.
- Minimal row chrome: Avoid row hover/background highlighting by default; keep only where explicitly needed.
Typography Guidelines
Always utilize the standard OpenChamber typography classes defined in
packages/ui/src/lib/typography.ts.
- Page Title: Use
for the top-most title of a settings page/dialog.typography-ui-header font-semibold text-foreground - Section Header: Use
for settings sections (e.g.typography-ui-header font-medium text-foreground
,Notification Events
).Session Defaults - Control Group Header: Use
(ortypography-ui-header font-medium text-foreground
if it reads too loud) for grouped controls inside a section (e.g.font-normal
,Default Tool Output
).Diff Layout - Values / Primary Text: Use
. Addtypography-ui-label text-foreground
if displaying numbers or stats to ensure vertical alignment.tabular-nums - Option Labels: Use non-bold label text in compact option controls (
when needed to override).font-normal - Meta / Helper Text: Use
ortypography-meta text-muted-foreground
for supplemental text.typography-small text-muted-foreground
Layout and Spacing Patterns
1. Main Backgrounds
Main wrappers should generally use
bg-background or bg-[var(--surface-background)]. Ensure adequate padding (e.g., px-5 py-6 or p-6).
2. Subsection Grouping
Group related controls with vertical spacing, not mandatory cards.
- Use
between logical subsections.space-y-3 - Use
for subsection internal padding.p-2 - Avoid adding
unless there is a clear reason.bg-[var(--surface-elevated)] - Avoid extra row decorations (
, hover fills) unless there is explicit UX value.rounded-md
3. Header-to-Content Hierarchy (critical)
When removing cards/background wrappers, spacing must be rebalanced so header ownership stays clear.
- Keep section-to-section spacing larger than header-to-own-content spacing.
- Typical pattern:
- header wrapper
mb-1 px-1 - content wrapper
pt-0 pb-2 px-2 - outer section spacing
mb-8
- header wrapper
- Do not leave legacy
style gaps after flattening a section; it makes headers look detached.mb-3
4. Headerless Blocks (when context is obvious)
If the page title already provides enough context, remove redundant local headers and place controls directly below the title.
- Example: project page identity controls can sit directly under project name/path.
- Tighten top gap for this pattern (e.g. top header
instead of larger section spacing).mb-4
<div className="space-y-3"> <section className="p-2">...</section> <section className="p-2">...</section> </div>
Structural Patterns
1. Segmented Option Buttons (compact)
Use for short option sets where button-style segmented choice reads best (e.g. Default Tool Output).
<div className="mt-1 flex flex-wrap items-center gap-1"> <ButtonSmall variant="outline" size="xs" className={cn('!font-normal', isSelected ? 'border-[var(--primary-base)] text-[var(--primary-base)] bg-[var(--primary-base)]/10' : 'text-foreground')} > Collapsed </ButtonSmall> </div>
2. Radio Option Lists (compact rows)
Use for mutually exclusive mode/layout settings (e.g. Diff Layout, Diff View Mode).
- Use shared
component fromRadio
.@/components/ui/radio - Icon first, label second.
- Row container compact:
.py-0.5 - Inactive label can use
.text-foreground/50
<div role="radiogroup" aria-label="Diff layout" className="mt-1 space-y-0"> <div className="flex w-full items-center gap-2 py-0.5"> <Radio checked={selected} onChange={onSelect} ariaLabel="Diff layout: Dynamic" /> <span className={cn('typography-ui-label font-normal', selected ? 'text-foreground' : 'text-foreground/50')}>Dynamic</span> </div> </div>
3. Checkbox Setting Rows
Use shared
Checkbox component from @/components/ui/checkbox for boolean toggles.
- Icon first, text immediately after (
).gap-2 - Typical row spacing for checkbox rows:
.py-1.5 - Keep row click and keyboard toggle support.
- Prefer checkbox over binary show/hide button pairs for pure boolean state.
<div className="group flex cursor-pointer items-center gap-2 py-1.5" role="button" tabIndex={0} > <Checkbox checked={value} onChange={setValue} ariaLabel="Show Dotfiles" /> <span className="typography-ui-label text-foreground">Show Dotfiles</span> </div>
4. Invisible Two-Column Alignment
Use consistent label/control columns across settings rows so controls align on a shared vertical line.
- Desktop row pattern:
flex items-center gap-8 - Label column width:
w-56 shrink-0 - Control cluster:
w-fit
<div className="flex items-center gap-8 py-1.5"> <span className="typography-ui-label text-foreground w-56 shrink-0">Interface Font Size</span> <div className="flex items-center gap-2 w-fit">...</div> </div>
Disabled control rule
If a control is unavailable, disable the control only. Do not dim the label row by default.
Width-matching rule
When matching visual widths across different rows, compare full row footprint (control + adjacent action buttons), not just input width.
5. Theme Row Composition
For theme controls in Appearance:
header on first line; option chips below it.Color Mode
andLight Theme
on one row where possible, wrapping on small widths.Dark Theme- Keep selectors near labels and aligned to existing column rhythm.
- Replace persistent helper text with an info tooltip icon near the related action.
<div className="grid grid-cols-1 gap-2 py-1.5 md:grid-cols-[14rem_auto] md:gap-x-8 md:gap-y-2"> <div className="flex min-w-0 items-center gap-2">Light Theme ...</div> <div className="flex min-w-0 items-center gap-2">Dark Theme ...</div> </div>
6. Numeric Controls in Settings
Use compact stepper input (
- value +) plus reset button.
- Prefer shared
stepper style over slider + numeric combo in dense settings pages.NumberInput - Keep reset button adjacent to control (
).gap-2 - Avoid using Tailwind
on mobile for controls;overflow-hidden
forcespackages/ui/src/styles/mobile.css
. Use.overflow-hidden { overflow-y: auto !important; }
if you truly need clipping.overflow-x-hidden overflow-y-hidden - Touch devices:
enforcespackages/ui/src/styles/mobile.css
onmin-height: 36px
. If you build custom segmented controls withbutton
, ensure the container height can accommodate that (e.g.<button>
).h-9
Optional numeric overrides
For "override unless empty" fields (e.g. agent Temperature/Top P), keep the value optional and provide a fallback for stepping.
<NumberInput value={temperature} fallbackValue={0.7} onValueChange={setTemperature} onClear={() => setTemperature(undefined)} min={0} max={2} step={0.1} inputMode="decimal" emptyLabel="—" />
<div className="flex items-center gap-2 w-fit"> <NumberInput value={fontSize} onValueChange={setFontSize} min={50} max={200} step={5} /> <ButtonSmall variant="ghost" className="h-7 w-7 px-0">...</ButtonSmall> </div>
7. Inputs and Select Triggers (settings density)
Keep form controls in settings compact and aligned.
- Prefer
withInput
in dense settings rows.className="h-7" - Prefer default
sizing (avoidSelectTrigger
in settings).size="lg" - For icon-only actions next to inputs, use
withButtonSmall
.h-7 w-7 p-0
<div className="flex items-center gap-2"> <Input className="h-7" /> <ButtonSmall variant="outline" size="xs" className="h-7 w-7 p-0" aria-label="Browse"> <RiFolderLine className="h-4 w-4" /> </ButtonSmall> </div>
8. Template Grids (text fields)
For template-like settings (title/message pairs), use a simple grid and flat cells.
- Grid:
grid grid-cols-1 gap-2 md:grid-cols-2 md:gap-3 - Cell:
section p-2 - Field:
Input className="h-7"
9. Icon/Color Picker Rows
For dense icon/color pickers in settings:
- Place options under the field label when they are a palette/grid choice.
- Use stable selected-state styling (
/border
/subtle background), avoid transform jumps (ring
).scale-* - Keep chip size compact (
) and spacing consistent (h-7 w-7
).gap-2
Control Selection Rules
- Use compact option buttons for short, chip-like selection groups.
- Use radios for explicit mode/layout choices where list scanning is better.
- Use checkboxes for true/false settings.
- Avoid show/hide button pairs when a checkbox maps directly to the boolean.
- Do not couple unrelated toggles under one synthetic section header; keep hierarchy clear.
Best Practices
- Density: Keep options compact; avoid oversized rows/chips in dense settings pages.
- Consistency: Reuse shared controls (
,Checkbox
,Radio
) instead of inline icon logic.ButtonSmall size="xs" - Reuse via composition: Prefer a single settings component with a
subset (likevisibleSettings
) for multiple tabs (Appearance/Chat) instead of duplicating markup.OpenChamberVisualSettings - Hierarchy: Page title =
; section header =font-semibold
; control group header =font-medium
(orfont-medium
if needed); option labels = non-bold.font-normal - Subsection depth: Nested subgroup headings under a section should usually be one step lighter than parent heading weight.
- Hierarchy sanity check: after flattening UI, verify visual grouping by spacing first (not color).
- Helper blocks: For small notes/errors under a section, use
withmt-1 px-2
(and status token for errors).typography-meta text-muted-foreground/70 - Truncation: Always consider long text. Use
on text containers that sit next to buttons or icons to prevent layout breakage.min-w-0 flex-1 truncate - Theme Variables: Always use CSS variables for colors (e.g.,
) rather than hardcoded hex values or generic Tailwind colors when indicating semantic states.var(--status-success)