8b2b03d276
* 🎨 feat(web/default): add shadcn-style theme presets, radius prefs, and fix selection badges Integrate the qn-platform–style OKLCH color system into the default frontend while keeping the existing blue-tinted dark tokens for the default theme. Add [data-theme-preset] palettes for seven named presets plus the default zinc-like scale, define [data-theme-radius] overrides so user radius beats preset --radius, and align the Tailwind @custom-variant dark helper with .dark usage. Introduce ThemeCustomizationProvider to own preset and radius state, persist choices in cookies (theme-preset, theme-radius), and sync data-theme-preset / data-theme-radius on <html>. Wrap the tree in main.tsx. Extend ConfigDrawer with theme preset swatches (scoped data-theme-preset) and radius previews wired to context; refactor swatch/card markup so selected CircleCheck badges sit outside clipped rows (remove outer overflow-hidden that hid the centered checkmark). Add i18n keys for preset names, radius, and accessibility labels across en, zh, fr, ja, ru, vi. * 🎨 fix(web): align segmented controls with theme radius tokens - Replace hard-coded inner pill radii (rounded-[5px]) on dashboard chart toolbars with radius-md so the active state follows --radius when users change Radius in Theme Settings. - Use nested radii consistent with TabsList/TabsTrigger: outer rounded-lg (var(--radius)) and inner rounded-md (calc(var(--radius) - 2px)) so the track and active thumb stay concentric at small scales (e.g. 0.3rem) instead of a squared “focus” block inside a rounded shell. - Apply the same pattern to pricing SegmentedControl and the segmented groups in consumption-distribution-chart, model-charts, and user-charts. Verified: bun run typecheck (web/default) * ✨ feat(pricing): enrich model details with uptime sparkline and API documentation Add a compact 30-day uptime sparkline (OpenRouter-style bars + aggregate %) with per-day tooltips, surface it in a status row under quick stats and in the per-group performance table, and extend mock data so uptime series are stable and optionally scoped by group. Introduce an API tab with Shiki-highlighted code samples (cURL, Python, TypeScript, JavaScript), endpoint-type switching, authentication guidance, a supported-parameters table, and mock per-group RPM/TPM/RPD limits. Infer vendor, tokenizer, license, and data-retention hints for a provider & data privacy card on the Overview tab (capabilities/modalities stay with model identity; rate limits stay with the API tab). Update i18n for all new user-facing strings across en, zh, fr, ja, ru, and vi. * 🏆 feat(rankings): add comprehensive rankings dashboard Add a mock-data powered rankings experience with period tabs, model, app, and vendor leaderboards, market share and history charts, movers, new releases, and per-category sections while backend analytics are pending. Link ranked models to pricing details and ranked vendors to filtered pricing results, and include localized copy for all supported frontend locales. * fix(theme): correct theme preset selection state - update Base UI Radio selectors to use data-checked/data-unchecked states. - fix unchecked theme options still showing selected indicators. - isolate the default theme preview tokens to prevent preset changes from leaking into it. * fix(setup): correct usage mode radio state - use Base UI data-checked/data-unchecked states for RadioGroup styling. - hide radio indicators when options are unchecked to avoid setup page display issues. - drive usage mode card and icon selection styles from Base UI state. * fix(auth): submit sign-in and sign-up forms * 🎨 refactor: Align default theme with shadcn Base Nova and prune legacy customization Migrate shadcn UI to Base UI primitives via CLI (`base-nova` / `components.json`) and reinstall full component registry with `--overwrite`, including Hugeicons-backed widgets and newly added registry components. - Remove custom multi-preset/theme-radius system (`ThemeCustomizationProvider`, cookies, preset UI from config drawer); rely on official semantic CSS tokens + light/dark only. - Replace `theme.css` with shadcn’s documented neutral `:root`/`.dark` palette and `@theme inline` mappings (plus skeleton token vars for existing shimmer usage). - Update global styles for Base UI: collapsible animation uses `--collapsible-panel-height`; clarify scroll-lock override comment. Application compatibility: - Keep minimal shims where app code diverged from official APIs (popover collision props, combobox legacy `options` callers, Spinner prop typing). - Switch interactive styling from Radix-era `data-state` / `--radix-*` selectors to Base UI semantics (`data-open`, `data-popup-open`, `data-panel-open`, `--anchor-width`, etc.) Tooling / docs / build: - Rename Rsbuild vendor chunk grouping to `@base-ui` + transitive `@radix-ui`. - Refresh AGENTS.md / CLAUDE.md / classic→default sync skill for Base UI stack. - Bump `package.json` / lockfile for shadcn-postinstall deps (Hugeicons, chart stack, themes, etc.) Verified: `bun run typecheck` passes. Note: `bun run lint` still reports pre-existing hooks rule violations elsewhere; not addressed in this change. * 🎨 chore(web/default): unify table toolbar, relocate usage stats, refine filters - Refactor DataTableToolbar to a single wrapping flex row with a right-aligned action cluster (Reset / Search / View / Expand) for a cleaner Ant Design Pro–style filter bar; remove the dedicated stats row and the toolbar `stats` prop. - Move Common Logs summary badges (Usage / RPM / TPM) and the sensitive- data visibility toggle into the page header via CommonLogsHeaderActions and SectionPageLayout.Actions so the toolbar stays focused on filters. - Slim CommonLogsFilterBar props (no stats / preActions eye control). - Improve CompactDateTimeRangePicker: show minute-precision labels on the trigger (seconds omitted; aligns with datetime-local inputs); widen the trigger on sm+ breakpoints so the full range is visible without truncation; apply the same width in task logs filters. - Simplify DataTableViewOptions: text-only “View” trigger, no sliders icon. - Earlier layout tweak: extra top padding on SectionPageLayout scroll content so control focus rings are not clipped by overflow. * feat(web/default): Base UI migration and component foundation Migrate from Radix UI to Base UI, rewrite core UI primitives, update dependencies (recharts, date-fns, next-themes), add shadcn agent skill documentation, and refresh AI element components. This is the foundational work from the v2/localmain lineage that was not covered by the individual feature commits above. --------- Co-authored-by: t0ng7u <dev@aiass.cc> Co-authored-by: QuentinHsu <xuquentinyang@gmail.com>
210 lines
5.9 KiB
Markdown
210 lines
5.9 KiB
Markdown
# Customization & Theming
|
||
|
||
Components reference semantic CSS variable tokens. Change the variables to change every component.
|
||
|
||
## Contents
|
||
|
||
- How it works (CSS variables → Tailwind utilities → components)
|
||
- Color variables and OKLCH format
|
||
- Dark mode setup
|
||
- Changing the theme (presets, CSS variables)
|
||
- Adding custom colors (Tailwind v3 and v4)
|
||
- Border radius
|
||
- Customizing components (variants, className, wrappers)
|
||
- Checking for updates
|
||
|
||
---
|
||
|
||
## How It Works
|
||
|
||
1. CSS variables defined in `:root` (light) and `.dark` (dark mode).
|
||
2. Tailwind maps them to utilities: `bg-primary`, `text-muted-foreground`, etc.
|
||
3. Components use these utilities — changing a variable changes all components that reference it.
|
||
|
||
---
|
||
|
||
## Color Variables
|
||
|
||
Every color follows the `name` / `name-foreground` convention. The base variable is for backgrounds, `-foreground` is for text/icons on that background.
|
||
|
||
| Variable | Purpose |
|
||
| -------------------------------------------- | -------------------------------- |
|
||
| `--background` / `--foreground` | Page background and default text |
|
||
| `--card` / `--card-foreground` | Card surfaces |
|
||
| `--primary` / `--primary-foreground` | Primary buttons and actions |
|
||
| `--secondary` / `--secondary-foreground` | Secondary actions |
|
||
| `--muted` / `--muted-foreground` | Muted/disabled states |
|
||
| `--accent` / `--accent-foreground` | Hover and accent states |
|
||
| `--destructive` / `--destructive-foreground` | Error and destructive actions |
|
||
| `--border` | Default border color |
|
||
| `--input` | Form input borders |
|
||
| `--ring` | Focus ring color |
|
||
| `--chart-1` through `--chart-5` | Chart/data visualization |
|
||
| `--sidebar-*` | Sidebar-specific colors |
|
||
| `--surface` / `--surface-foreground` | Secondary surface |
|
||
|
||
Colors use OKLCH: `--primary: oklch(0.205 0 0)` where values are lightness (0–1), chroma (0 = gray), and hue (0–360).
|
||
|
||
---
|
||
|
||
## Dark Mode
|
||
|
||
Class-based toggle via `.dark` on the root element. In Next.js, use `next-themes`:
|
||
|
||
```tsx
|
||
import { ThemeProvider } from "next-themes"
|
||
|
||
<ThemeProvider attribute="class" defaultTheme="system" enableSystem>
|
||
{children}
|
||
</ThemeProvider>
|
||
```
|
||
|
||
---
|
||
|
||
## Changing the Theme
|
||
|
||
```bash
|
||
# Apply a preset code from ui.shadcn.com.
|
||
npx shadcn@latest apply --preset a2r6bw
|
||
|
||
# Positional shorthand also works.
|
||
npx shadcn@latest apply a2r6bw
|
||
|
||
# Switch to a named preset and overwrite existing components.
|
||
npx shadcn@latest apply --preset nova
|
||
|
||
# Preserve existing components instead.
|
||
npx shadcn@latest init --preset nova --force --no-reinstall
|
||
|
||
# Use a custom theme URL.
|
||
npx shadcn@latest apply --preset "https://ui.shadcn.com/init?base=radix&style=nova&theme=blue&..."
|
||
```
|
||
|
||
Or edit CSS variables directly in `globals.css`.
|
||
|
||
---
|
||
|
||
## Adding Custom Colors
|
||
|
||
Add variables to the file at `tailwindCssFile` from `npx shadcn@latest info` (typically `globals.css`). Never create a new CSS file for this.
|
||
|
||
```css
|
||
/* 1. Define in the global CSS file. */
|
||
:root {
|
||
--warning: oklch(0.84 0.16 84);
|
||
--warning-foreground: oklch(0.28 0.07 46);
|
||
}
|
||
.dark {
|
||
--warning: oklch(0.41 0.11 46);
|
||
--warning-foreground: oklch(0.99 0.02 95);
|
||
}
|
||
```
|
||
|
||
```css
|
||
/* 2a. Register with Tailwind v4 (@theme inline). */
|
||
@theme inline {
|
||
--color-warning: var(--warning);
|
||
--color-warning-foreground: var(--warning-foreground);
|
||
}
|
||
```
|
||
|
||
When `tailwindVersion` is `"v3"` (check via `npx shadcn@latest info`), register in `tailwind.config.js` instead:
|
||
|
||
```js
|
||
// 2b. Register with Tailwind v3 (tailwind.config.js).
|
||
module.exports = {
|
||
theme: {
|
||
extend: {
|
||
colors: {
|
||
warning: "oklch(var(--warning) / <alpha-value>)",
|
||
"warning-foreground":
|
||
"oklch(var(--warning-foreground) / <alpha-value>)",
|
||
},
|
||
},
|
||
},
|
||
}
|
||
```
|
||
|
||
```tsx
|
||
// 3. Use in components.
|
||
<div className="bg-warning text-warning-foreground">Warning</div>
|
||
```
|
||
|
||
---
|
||
|
||
## Border Radius
|
||
|
||
`--radius` controls border radius globally. Components derive values from it (`rounded-lg` = `var(--radius)`, `rounded-md` = `calc(var(--radius) - 2px)`).
|
||
|
||
---
|
||
|
||
## Customizing Components
|
||
|
||
See also: [rules/styling.md](./rules/styling.md) for Incorrect/Correct examples.
|
||
|
||
Prefer these approaches in order:
|
||
|
||
### 1. Built-in variants
|
||
|
||
```tsx
|
||
<Button variant="outline" size="sm">
|
||
Click
|
||
</Button>
|
||
```
|
||
|
||
### 2. Tailwind classes via `className`
|
||
|
||
```tsx
|
||
<Card className="mx-auto max-w-md">...</Card>
|
||
```
|
||
|
||
### 3. Add a new variant
|
||
|
||
Edit the component source to add a variant via `cva`:
|
||
|
||
```tsx
|
||
// components/ui/button.tsx
|
||
warning: "bg-warning text-warning-foreground hover:bg-warning/90",
|
||
```
|
||
|
||
### 4. Wrapper components
|
||
|
||
Compose shadcn/ui primitives into higher-level components:
|
||
|
||
```tsx
|
||
export function ConfirmDialog({ title, description, onConfirm, children }) {
|
||
return (
|
||
<AlertDialog>
|
||
<AlertDialogTrigger asChild>{children}</AlertDialogTrigger>
|
||
<AlertDialogContent>
|
||
<AlertDialogHeader>
|
||
<AlertDialogTitle>{title}</AlertDialogTitle>
|
||
<AlertDialogDescription>{description}</AlertDialogDescription>
|
||
</AlertDialogHeader>
|
||
<AlertDialogFooter>
|
||
<AlertDialogCancel>Cancel</AlertDialogCancel>
|
||
<AlertDialogAction onClick={onConfirm}>Confirm</AlertDialogAction>
|
||
</AlertDialogFooter>
|
||
</AlertDialogContent>
|
||
</AlertDialog>
|
||
)
|
||
}
|
||
```
|
||
|
||
---
|
||
|
||
## Checking for Updates
|
||
|
||
```bash
|
||
npx shadcn@latest add button --diff
|
||
```
|
||
|
||
To preview exactly what would change before updating, use `--dry-run` and `--diff`:
|
||
|
||
```bash
|
||
npx shadcn@latest add button --dry-run # see all affected files
|
||
npx shadcn@latest add button --diff button.tsx # see the diff for a specific file
|
||
```
|
||
|
||
See [Updating Components in SKILL.md](./SKILL.md#updating-components) for the full smart merge workflow.
|