This document describes the complete UI and styling approach for the template storefront: Tailwind CSS, shadcn/ui, design tokens, and related development practices.
The project uses Tailwind CSS v4 with a utility-first approach.
- Use Tailwind utility classes in component JSX for layout, spacing, typography, and colors.
- Use the
cn()utility for conditional or combined class names:import { cn } from '@/lib/utils'. Example:cn('rounded p-4', isActive && 'ring-2'). - Follow mobile-first responsive patterns using breakpoint prefixes:
sm:,md:,lg:,xl:,2xl:. - Do not use inline styles (
style={{ ... }}) for styling. - Do not use CSS modules (
.module.css) or separate CSS files for component-level styles. - Global and theme styles belong in
src/theme/only. The entry point issrc/theme/index.css, with tokens split acrosssrc/theme/tokens/, base resets insrc/theme/base.css, and component overrides insrc/theme/overrides/.
Colors and theme values are defined as CSS variables (design tokens). Use semantic token-based classes instead of hard-coded colors:
- Backgrounds:
bg-background,bg-muted,bg-card - Text:
text-foreground,text-muted-foreground,text-primary - Borders:
border-border - Interactive:
bg-primary,text-primary-foreground,hover:bg-primary/90
Avoid raw color utilities (e.g. bg-[#hex]) so the app stays consistent with the theme and supports dark mode.
Use breakpoints consistently:
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4">
{/* Content */}
</div>Presentational UI components are built on Radix UI primitives with shadcn/ui as the styling layer. They live in src/components/ui/.
Add new components only via the official CLI so they are ejected with the correct config and Tailwind setup:
npx shadcn@latest add <component-name>This ejects the component into src/components/ui/ with the right dependencies and styles.
- Do add and customize shadcn components by editing the files in
src/components/ui/. - Do not create custom components inside
src/components/ui/; keep that directory for ejected shadcn components only. - Do not manually copy components from the shadcn docs; always use the CLI so configuration (e.g.
components.json) stays in sync.
Keeping src/components/ui/ limited to ejected shadcn components makes upgrades and maintenance predictable. For custom UI, use src/components/ (or another feature directory) and compose or wrap shadcn components as needed.
See src/components/ui/README.md for more detail.
Tailwind's utility-first approach means most styling lives inline in JSX. Before extracting a reusable abstraction, read the official guide on managing reuse — it covers multi-cursor editing, loops, and component extraction as the preferred strategies before reaching for CSS abstractions.
Use a React component (the default choice) when:
- The pattern involves markup structure — multiple elements, slots, children
- There is logic, state, or event handling
- It accepts props that change behavior or content
- It composes other components (shadcn, Radix, etc.)
Use a CSS component class (@layer components in src/theme/base.css) only when:
- The pattern is pure layout/styling — padding, max-width, centering, typography presets
- There is no logic, state, or props — just a bag of CSS properties
- It needs to be applied to many different HTML elements across the codebase (divs, sections, wrappers)
- Utilities need to override it in specific contexts (the components layer is lower specificity than utilities)
Example: section-container — consolidates px-4 sm:px-8 lg:px-16 max-w-screen-2xl mx-auto into one class, used by 30+ files. A page can add max-w-4xl alongside it and the utility wins.
Rule of thumb: if you can express it as a single className string with no JSX children, it's a CSS class. If it renders elements or accepts props, it's a React component.
/* src/theme/base.css — CSS component class */
@layer components {
.section-container {
@apply px-4 sm:px-8 lg:px-16 max-w-screen-2xl mx-auto;
}
}/* React component — has structure, props, and children */
function CategoryBanner({ title, image }: CategoryBannerProps) {
return (
<div className="section-container">
<img src={image} alt="" />
<h1>{title}</h1>
</div>
);
}Do not use
@utilityfor multi-property compositions that need to be overridable. The utility layer has the highest specificity, so any override attempt (e.g., addingmax-w-4xlalongside a@utilityclass) would lose. Use@layer componentsinstead.
- Radix UI: Use Radix primitives for accessible behavior (focus, keyboard, ARIA).
- Icons: Use Lucide React and React Simple Icons for iconography.
Dark mode is supported via CSS variables and the .dark class on a parent (e.g. root). Theme variables switch automatically:
<div className="bg-background text-foreground border-border">
<button className="bg-primary text-primary-foreground">Click me</button>
</div>No extra class changes are needed for dark mode when using design tokens.
- Use semantic HTML (
<button>,<nav>,<main>, etc.) and appropriate ARIA where needed. - Ensure keyboard navigation and visible focus states for interactive elements.
- Aim for WCAG compliance (contrast, focus order, labels).
- Keep spacing and typography consistent with the design system defined in
src/theme/and Tailwind config.
Quick reference:
| Do | Don't |
|---|---|
| Tailwind utility classes | Inline styles, CSS modules, component-level .css files |
cn() for conditional classes |
Manual string concatenation for className |
Design tokens (bg-background, text-muted-foreground) |
Hard-coded colors |
npx shadcn@latest add <name> |
Manually copying or creating components in src/components/ui/ |
Global/theme styles in src/theme/ |
Scattered or duplicate global CSS |
For a short checklist, see the styling section in the Storefront Next development guidelines (e.g. the storefront_next_development_guidelines tool).