Styles and Assets

Table of Contents

CSS Files

Global CSS can live under src/frontend/styles/** and be imported by pages, layouts, or generated entries.

/* src/frontend/styles/app.css */
:root {
  color-scheme: light dark;
}

body {
  margin: 0;
}

Use global CSS for reset, base typography, and design tokens. Prefer components or CSS Modules for local component styles.

CSS Modules

CSS Modules are enabled for .module.css.

/* src/frontend/styles/card.module.css */
.card {
  border: 1px solid var(--border);
  border-radius: 8px;
  padding: 16px;
}
import styles from "@styles/card.module.css";

export function Card(props: { children: React.ReactNode }) {
  return <section className={styles.card}>{props.children}</section>;
}

Vext JSCSS

JSCSS is Vext's default dynamic style facade. It is designed for component-level dynamic styles and build-time extraction.

// src/frontend/styles/button.style.ts
import { recipe, style, vars } from "vextjs/style";

export const button = recipe({
  base: style({
    borderRadius: 8,
    padding: "8px 12px",
    color: vars.color.text,
  }),
  variants: {
    intent: {
      primary: style({ background: vars.color.primary }),
      danger: style({ background: vars.color.danger }),
    },
  },
});

The extractor writes CSS into the build output. You do not need Emotion or styled-components as default runtime dependencies.

CSS Variables

Use CSS variables when values need to be changed by theme, tenant, or runtime state.

import { createVar, setVar, style } from "vextjs/style";

export const accent = createVar("accent");

export const panel = style({
  [setVar(accent)]: "#4f46e5",
  borderColor: accent,
});

Keep variables semantic, such as color.surface, color.text, space.md, or radius.sm.

Imported Assets

Imported assets go through esbuild and the Vext manifest.

import logoUrl from "@assets/logo.png";

export function Logo() {
  return <img src={logoUrl} alt="Company" />;
}

Production builds content-hash imported assets and include them in manifest.json and deploy-manifest.json.

Public Assets

Files under public/** keep URL-style access.

public/
  favicon.svg
  robots.txt
  images/social-card.png

Use them as:

<img src="/images/social-card.png" alt="" />

public/** files are copied into frontend output and included in deploy planning.

CDN URLs

Set frontend.deploy.assetBaseUrl when production assets are served from a CDN.

export default {
  frontend: {
    deploy: {
      assetBaseUrl: "https://cdn.example.com/my-app/",
      crossOrigin: "anonymous",
      integrity: true,
    },
  },
};

assetBaseUrl affects generated asset URLs. Upload is controlled separately by frontend.deploy.upload, vext build --upload-assets, or vext deploy assets.