Loading source
Pulling the file list, source metadata, and syntax-aware rendering for this listing.
Source from repo
Design engineering principles for polished UI details: concentric border radius, interruptible animations, shadows, and font smoothing.
Files
Skill
Size
Entrypoint
Format
Open file
Syntax-highlighted preview of this file as included in the skill package.
SKILL.md
1---2name: make-interfaces-feel-better3description: Design engineering principles for making interfaces feel polished. Use when building UI components, reviewing frontend code, implementing animations, hover states, shadows, borders, typography, micro-interactions, enter/exit animations, or any visual detail work. Triggers on UI polish, design details, "make it feel better", "feels off", stagger animations, border radius, optical alignment, font smoothing, tabular numbers, image outlines, box shadows.4---56# Details that make interfaces feel better78Great interfaces rarely come from a single thing. It's usually a collection of small details that compound into a great experience. Apply these principles when building or reviewing UI code.910## Quick Reference1112| Category | When to Use |13| --- | --- |14| [Typography](typography.md) | Text wrapping, font smoothing, tabular numbers |15| [Surfaces](surfaces.md) | Border radius, optical alignment, shadows, image outlines, hit areas |16| [Animations](animations.md) | Interruptible animations, enter/exit transitions, icon animations, scale on press |17| [Performance](performance.md) | Transition specificity, `will-change` usage |1819## Core Principles2021### 1. Concentric Border Radius2223Outer radius = inner radius + padding. Mismatched radii on nested elements is the most common thing that makes interfaces feel off.2425### 2. Optical Over Geometric Alignment2627When geometric centering looks off, align optically. Buttons with icons, play triangles, and asymmetric icons all need manual adjustment.2829### 3. Shadows Over Borders3031Layer multiple transparent `box-shadow` values for natural depth. Shadows adapt to any background; solid borders don't.3233### 4. Interruptible Animations3435Use CSS transitions for interactive state changes — they can be interrupted mid-animation. Reserve keyframes for staged sequences that run once.3637### 5. Split and Stagger Enter Animations3839Don't animate a single container. Break content into semantic chunks and stagger each with ~100ms delay.4041### 6. Subtle Exit Animations4243Use a small fixed `translateY` instead of full height. Exits should be softer than enters.4445### 7. Contextual Icon Animations4647Animate icons with `opacity`, `scale`, and `blur` instead of toggling visibility. Use exactly these values: scale from `0.25` to `1`, opacity from `0` to `1`, blur from `4px` to `0px`. If the project has `motion` or `framer-motion` in `package.json`, use `transition: { type: "spring", duration: 0.3, bounce: 0 }` — bounce must always be `0`. If no motion library is installed, keep both icons in the DOM (one absolute-positioned) and cross-fade with CSS transitions using `cubic-bezier(0.2, 0, 0, 1)` — this gives both enter and exit animations without any dependency.4849### 8. Font Smoothing5051Apply `-webkit-font-smoothing: antialiased` to the root layout on macOS for crisper text.5253### 9. Tabular Numbers5455Use `font-variant-numeric: tabular-nums` for any dynamically updating numbers to prevent layout shift.5657### 10. Text Wrapping5859Use `text-wrap: balance` on headings. Use `text-wrap: pretty` for body text to avoid orphans.6061### 11. Image Outlines6263Add a subtle `1px` outline with low opacity to images for consistent depth. The color must be pure black in light mode (`rgba(0, 0, 0, 0.1)`) and pure white in dark mode (`rgba(255, 255, 255, 0.1)`) — never a near-black like slate, zinc, or any tinted neutral. A tinted outline picks up the surface color underneath it and reads as dirt on the image edge.6465### 12. Scale on Press6667A subtle `scale(0.96)` on click gives buttons tactile feedback. Always use `0.96`. Never use a value smaller than `0.95` — anything below feels exaggerated. Add a `static` prop to disable it when motion would be distracting.6869### 13. Skip Animation on Page Load7071Use `initial={false}` on `AnimatePresence` to prevent enter animations on first render. Verify it doesn't break intentional entrance animations.7273### 14. Never Use `transition: all`7475Always specify exact properties: `transition-property: scale, opacity`. Tailwind's `transition-transform` covers `transform, translate, scale, rotate`.7677### 15. Use `will-change` Sparingly7879Only for `transform`, `opacity`, `filter` — properties the GPU can composite. Never use `will-change: all`. Only add when you notice first-frame stutter.8081### 16. Minimum Hit Area8283Interactive elements need at least 40×40px hit area. Extend with a pseudo-element if the visible element is smaller. Never let hit areas of two elements overlap.8485## Common Mistakes8687| Mistake | Fix |88| --- | --- |89| Same border radius on parent and child | Calculate `outerRadius = innerRadius + padding` |90| Icons look off-center | Adjust optically with padding or fix SVG directly |91| Hard borders between sections | Use layered `box-shadow` with transparency |92| Jarring enter/exit animations | Split, stagger, and keep exits subtle |93| Numbers cause layout shift | Apply `tabular-nums` |94| Heavy text on macOS | Apply `antialiased` to root |95| Animation plays on page load | Add `initial={false}` to `AnimatePresence` |96| `transition: all` on elements | Specify exact properties |97| First-frame animation stutter | Add `will-change: transform` (sparingly) |98| Tiny hit areas on small controls | Extend with pseudo-element to 40×40px |99100## Review Output Format101102Always present changes as a markdown table with **Before** and **After** columns. Include every change you made — not just a subset. Never list findings as separate "Before:" / "After:" lines outside of a table. Group changes by principle using a heading above each table, and keep each row focused on a single diff so the reader can scan the whole list quickly.103104### Example105106#### Concentric border radius107| Before | After |108| --- | --- |109| `rounded-xl` on card + `rounded-xl` on inner button (`p-2`) | `rounded-2xl` on card (`12 + 8`), `rounded-lg` on inner button |110| `border-radius: 16px` on both nested surfaces | Outer `24px`, inner `16px` with `8px` padding |111112#### Tabular numbers113| Before | After |114| --- | --- |115| `<span>{count}</span>` on animated counter | `<span className="tabular-nums">{count}</span>` |116| Default numerals on timer | Added `font-variant-numeric: tabular-nums` to root |117118#### Scale on press119| Before | After |120| --- | --- |121| `<button className="...">` | Added `active:scale-[0.96] transition-transform` |122| `scale(0.9)` on press | Raised to `scale(0.96)` — anything below `0.95` feels exaggerated |123124Rows should cite the specific file and the specific property that changed when it isn't obvious from the snippet. If a principle was reviewed but nothing needed to change, omit that table entirely — empty tables add noise.125126## Review Checklist127128- [ ] Nested rounded elements use concentric border radius129- [ ] Icons are optically centered, not just geometrically130- [ ] Shadows used instead of borders where appropriate131- [ ] Enter animations are split and staggered132- [ ] Exit animations are subtle133- [ ] Dynamic numbers use tabular-nums134- [ ] Font smoothing is applied135- [ ] Headings use text-wrap: balance136- [ ] Images have subtle outlines137- [ ] Buttons use scale on press where appropriate138- [ ] AnimatePresence uses `initial={false}` for default-state elements139- [ ] No `transition: all` — only specific properties140- [ ] `will-change` only on transform/opacity/filter, never `all`141- [ ] Interactive elements have at least 40×40px hit area142143## Reference Files144145- [typography.md](typography.md) — Text wrapping, font smoothing, tabular numbers146- [surfaces.md](surfaces.md) — Border radius, optical alignment, shadows, image outlines147- [animations.md](animations.md) — Interruptible animations, enter/exit transitions, icon animations, scale on press148- [performance.md](performance.md) — Transition specificity, `will-change` usage149