Robust-skills modern-css
Proactively apply when creating design systems, component libraries, or any frontend application. Triggers on CSS Grid, Subgrid, Flexbox, Container Queries, :has(), @layer, @scope, CSS nesting, @property, @function, if(), oklch, color-mix, light-dark, relative color, @starting-style, scroll-driven animations, view transitions, anchor positioning, popover, customizable select, content-visibility, logical properties, text-wrap, interpolate-size, clamp, field-sizing, modern CSS, CSS architecture, responsive design, dark mode, theming, design tokens, cascade layers. Use when writing CSS for any web project, choosing layout approaches, building responsive components, implementing dark mode or theming, creating animations or transitions, styling form elements, or modernizing legacy stylesheets. Modern CSS features and best practices for building interfaces with pure native CSS.
git clone https://github.com/ccheney/robust-skills
T=$(mktemp -d) && git clone --depth=1 https://github.com/ccheney/robust-skills "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/modern-css" ~/.claude/skills/ccheney-robust-skills-modern-css && rm -rf "$T"
skills/modern-css/SKILL.mdModern CSS
Pure native CSS for building interfaces — no preprocessors, no frameworks.
When to Use (and When NOT to)
| Use Freely (Baseline) | Feature-Detect First |
|---|---|
| CSS Grid, Subgrid, Flexbox | , (Chrome-only) |
| Container Queries (size + style) | Customizable (Chrome-only) |
, , | Scroll-state queries (Chrome-only) |
CSS Nesting, , | , |
(typed custom props) | , |
, , | Typed beyond |
| Relative color syntax | |
, | (Chrome-only) |
| Scroll-driven animations | Grid Lanes / masonry (experimental) |
| Anchor positioning, Popover API | (Safari TP only) |
, easing | / (no browser yet) |
| View Transitions, logical properties |
CRITICAL: The Modern Cascade
Understanding how styles resolve is the single most important concept in CSS. The additions of
@layer and @scope fundamentally changed the cascade algorithm.
Style Resolution Order (highest priority wins): ┌─────────────────────────────────────────────────┐ │ 1. Transitions (active transition wins) │ │ 2. !important (user-agent > user > author) │ │ 3. @layer order (later layer > earlier layer) │ │ 4. Unlayered styles (beat ALL layers) │ │ 5. Specificity (ID > class > element) │ │ 6. @scope proximity (closer root wins) NEW │ │ 7. Source order (later > earlier) │ └─────────────────────────────────────────────────┘ Unlayered > Last layer > ... > First layer (utilities) (reset)
Cascade layers (
@layer) and scope proximity (@scope) are now more powerful than selector specificity. Define your layer order once (@layer reset, base, components, utilities;) and specificity wars disappear. Unlayered styles always beat layered styles — use this for overrides.
Quick Decision Trees
"How do I lay this out?"
Layout approach? ├─ 2D grid (rows + columns) → CSS Grid │ ├─ Children must align across → Grid + Subgrid │ └─ Waterfall / masonry → grid-lanes (experimental) ├─ 1D row OR column → Flexbox ├─ Component adapts to container → Container Query + Grid/Flex ├─ Viewport-based responsiveness → @media range syntax └─ Element sized to content → fit-content / min-content / stretch
"How do I style this state?"
Style based on what? ├─ Child/descendant presence → :has() ├─ Container size → @container (inline-size) ├─ Container custom property → @container style() ├─ Scroll position (stuck/snapped) → scroll-state() query ├─ Element's own custom property → if(style(...)) ├─ Browser feature support → @supports ├─ User preference (motion/color) → @media (prefers-*) └─ Multiple selectors efficiently → :is() / :where()
"How do I animate this?"
Animation type? ├─ Enter/appear on DOM → @starting-style + transition ├─ Exit/disappear (display:none) → transition-behavior: allow-discrete ├─ Animate to/from auto height → interpolate-size: allow-keywords ├─ Scroll-linked (parallax/reveal) → animation-timeline: scroll()/view() ├─ Page/view navigation → View Transitions API ├─ Custom easing (bounce/spring) → linear() function └─ Always: respect user preference → @media (prefers-reduced-motion)
What CSS Replaced JavaScript For
| JavaScript Pattern | CSS Replacement |
|---|---|
| Scroll position listeners | Scroll-driven animations |
| IntersectionObserver for reveal | |
| Sticky header shadow toggle | |
| Floating UI / Popper.js | Anchor positioning |
| Carousel prev/next/dots | , |
| Auto-expanding textarea | |
| Staggered animation delays | |
hack | |
| Parent element selection | |
| Theme toggle logic | + |
| Tooltip/popover show/hide | Popover API + invoker commands |
| Color manipulation functions | , relative color syntax |
For non-Baseline features, always feature-detect with
or use progressive enhancement. Check MDN or Baseline for current browser support.@supports
Anti-Patterns (CRITICAL)
| Anti-Pattern | Problem | Fix |
|---|---|---|
Overusing | Specificity arms race | Use for cascade control |
Deep nesting () | Fragile, DOM-coupled | Flat selectors, |
IDs for styling () | Too specific to override | Classes () |
for component layout | Viewport-coupled, not reusable | Container queries |
| JS scroll listeners for effects | Janky, expensive | Scroll-driven animations |
| JS for tooltip positioning | Floating UI dependency | Anchor positioning |
| JS for carousel controls | Fragile, a11y issues | , |
| JS for auto-expanding textarea | Unnecessary complexity | |
for animation | Wrong duration, janky | |
/ | Breaks in RTL/vertical | Logical properties () |
with commas | Legacy syntax | space-separated |
on selects | Removes ALL functionality | |
| Preprocessor-only variables | Can't change at runtime | CSS custom properties |
| Preprocessor-only nesting | Extra build step dependency | Native CSS nesting |
| Preprocessor color functions | Can't respond to context | , relative colors |
on paragraphs | Performance-heavy | Only headings/short text |
above fold | Delays LCP rendering | Only off-screen sections |
Overusing | Wastes GPU memory | Apply only to animating elements |
Reference Documentation
| File | Purpose |
|---|---|
| references/CASCADE.md | Nesting, , , cascade control, and CSS architecture |
| references/LAYOUT.md | Grid, Subgrid, Flexbox, Container Queries, and intrinsic sizing |
| references/SELECTORS.md | , , , pseudo-elements, and state-based selection |
| references/COLOR.md | OKLCH, , relative colors, , and theming |
| references/TOKENS.md | , , , math functions, and design tokens |
| references/ANIMATION.md | , , , view transitions |
| references/SCROLL.md | Scroll-driven animations, scroll-state queries, native carousels |
| references/COMPONENTS.md | Customizable , popover, anchor positioning, |
| references/PERFORMANCE.md | , typography, logical properties, accessibility |
| references/CHEATSHEET.md | Quick reference: browser support, legacy→modern patterns, units |
Sources
Official Specifications
- CSS Snapshot 2025 — W3C
- CSS Values and Units Level 5 —
,if()
,random()sibling-index/count() - CSS Functions and Mixins Level 1 —
,@function@mixin - CSS Conditional Rules Level 5 — Scroll-state queries
- CSS Anchor Positioning
- CSS Overflow Level 5 — Scroll markers/buttons
Browser Vendor Blogs
- CSS Wrapped 2025 — Chrome DevRel
- Interop 2025 — WebKit
- What's New in Web UI (I/O 2025)