The missing css
prop API for š¼ PandaCSS.
Radipan is an all-in-JS CSS-in-JS engine. With Radipan, you can write all your styles in JavaScript without the need for HTML or CSS.
- Real Zero-Runtime:
- Built-in transpiler that transforms
css
prop into its resultingclassName
prop at build time, and generates transpiled.lite.ts
files for you to import and use to completely rid of the dependence on Radipan (or PandaCSS)
- Built-in transpiler that transforms
- Flexible API:
- JSX-compatible API
- HyperScript-compatible API
h(tag, attrs, [text?, Elements?,...])
- Recipe Shaking Optimization (coming soon...)
Radipan runs your code during build time and scan the entry component and its child nodes for inline css
props and transforms these props into intermediate PandaCSS code (css
, cva
, etc.), which generates static CSS code.
Radipan supports the widely adopted inline css
prop syntax that made popular by libraries like Emotion.js, Stitches.js and Mitosis, etc...
Radipan works with various frameworks and tools, such as React, Preact, Vite, Next.js, etc.
- Radipan + Vite + React + TypeScript Starter (radipan-vite-react-ts)
- Radipan + Vite + React + TypeScript + JSX Starter (radipan-vite-react-tsx)
- Radipan + Vite + Solid + TypeScript Starter (radipan-vite-solid-ts)
- Radipan + Vite + Solid + TypeScript + JSX Starter (radipan-vite-solid-tsx)
(More coming soon...)
Radipan comes with HyperScript-compatible API, which you can import and use like h(tag, attrs, [text?, Elements?,...])
:
import { h } from 'radipan';
function App() {
return h('div', { css: { color: 'red' } }, 'whee!')); // Red whee!
}
You can also use Radipan's own withCreate
function, which comes with a more verbose create
method:
import { withCreate } from 'radipan';
function App() {
return withCreate('div').create({ css: { color: 'red' } }, 'whee!')); // Red whee!
}
You can import HTML tags from radipan/tags
for convenience:
import { div, form, input, span } from 'radipan/tags';
function App() {
return form.create({css: { display: 'flex' }}, [
div.create({ css: { color: 'green' } }, 'green div'));
span.create({ css: { color: 'red' } }, 'red span'));
input.create({ css: { color: 'blue' }, placeholder: 'blue input' }));
]);
}
You can use recipes directly within the css
prop. Recipes are a way to define styles for different variants of a component. They offer better performance, developer experience, and composability.
A recipe consists of four properties:
base
: The base styles for the componentvariants
: The different visual styles for the componentcompoundVariants
: The different combinations of variants for the componentdefaultVariants
: The default variant values for the component
A recipe must have at least variants
to be recognized as a recipe. Use a nested object to specify the variant name and the corresponding style in variants. For example:
import { withCreate } from "radipan";
const Badge = ({
as = "span",
size = "md", // 'size' is a recipe variant
variant = "solid", // 'variant' is also a recipe variant
children = undefined,
...props
} = {}) => {
return withCreate(as).create(
{
size,
variant,
...props,
css: badgeRecipe,
children
);
};
const badgeRecipe = {
base: {
borderRadius: "xs",
textTransform: "uppercase",
},
variants: {
size: {
sm: { fontSize: "xs", padding: "0 2px" },
md: { fontSize: "sm", padding: "0 var(--spacing-1x)" },
lg: { fontSize: "md", padding: "1px var(--spacing-1x)" },
},
variant: {
solid: {
/* solid style CSS code */
},
subtle: {
/* subtle style CSS code */
},
outline: {
/* outline style CSS code */
},
},
},
},
};
export default withCreate(Badge);
Use your preferred package manager to install Radipan:
npm install --save-dev radipan@latest
Create a file named radipan.config.ts
(or .js
) file in the root of your project and add these configuration options for Radipan:
import { defineConfig } from "radipan/config";
export default defineConfig({
include: ["src"], // Source paths to include for CSS processing
exclude: [""], // Source paths to exclude from scanning
includeNames: ["*.ts", "*.tsx"], // Source files to include for CSS processing
excludeNames: [
"main.tsx",
"*.init.ts",
"*.init.tsx",
"*.lite.ts",
"*.lite.tsx",
], // Source files to exclude from scanning
jsxFramework: "react", // "react" | "solid" | "preact" | "vue" | "qwik"
preflight: true, // Whether to use css reset
recipeShaking: true, // Whether to trim unused recipe variants
sourceTranspile: {
// Whether to transform `css` prop to corresponding `className`
enabled: true,
// File extension for the transpiled output files
extension: ".lite.tsx",
},
outdir: "@design-system", // The output directory in /node_modules
theme: {
// Useful for theme customization
extend: {},
},
});
Update your package.json
scripts to work with Radipan:
"scripts": {
"cssgen": "npx radipan cssgen",
"prepare": "npx radipan prepare",
"dev": "npx radipan prepare --watch & next dev",
"build": "npx radipan cssgen && tsc && next build",
}
The prepare
script runs codegen after dependency installation and regenerates the output directory.
Create an entry CSS file or replace the existing one named index.css
(or global.css
), usually under the src
or src/app
folder, with this code:
@layer reset, base, tokens, recipes, utilities;
/* Import generated Radipan static CSS */
@import "radipan/styles.css";
When you run npx radipan cssgen
, it scans all css
props in your app and statically generates the corresponding CSS code at build time.
Optionally, you can follow the below steps to use Radipan with JSX in TypeScript if you prefer to use JSX syntax. Feel free to skip this step if you don't use JSX.
Configure the compilerOptions
in TSConfig (usually tsconfig.json
) with the following settings:
"jsx": "react-jsx",
"jsxImportSource": "radipan"
This requires the new JSX transform and the jsxImportSource
option, which are available in TS 4.1 or later.
If you are using a UI framework other than React, you also need to change the jsxFramework
setting in your radipan.config.ts
file to match your framework. For example, if you are using Solid.js, you should set jsxFramework: 'solid'
. This ensures that your JSX code is transpiled correctly for your framework.
With this configuration, you can now use the object syntax to define styles and pass them to your components using the css
prop!
function App() {
return (
<main
css={{
width: "100%",
height: "100vh",
color: { base: "black", _osDark: "white" },
background: { base: "white", _osDark: "black" },
}}
>
<div css={{ fontSize: "2xl", fontWeight: "bold" }}>Hello Radipš¼n!</div>
</main>
);
}
If you're on an older version of React / TypeScript and unable to use the newer react-jsx
transform, you will need to set the jsxFactory
TSConfig option to "radipan.h"
, or specify the JSX factory at the top of every file:
/** @jsx h */
import { h } from "radipan";
- The transpiler uses Babel to handle the
css
-to-className
transformation. It then runs your code at build time, and turns any dynamic references into static values. If you need real dynamic styles, you might need to move any runtime references (e.g., reference of values from hooks) intostyles
prop.
const Widget = ({ width: number }) => {
return (
<div
// ā ļø Static: The value of `width`` is turned into static at build-time as per usage
// Any change to `width` will not update the width during runtime
css={{ width, color: "red" }}
>
Hello Radipš¼n!
</div>
);
};
function Widget() {
const [color, setColor] = useState("red.300");
return (
// ā ļø Static: This will work statically because `red.300` is the value at build time
// Any change to `color` will not update the color during runtime
<div css={{ ...divCss, color }}>Hello Radipš¼n!</div>
);
const divCss = { fontSize: "2xl", fontWeight: "bold" };
}
function Widget() {
const [color, setColor] = useState("red.300");
return (
// ā
Good: `color` is in `styles` because it's dynamically updated after user interactions
<>
<div css={divCss} styles={{ color }}>
Hello Radipš¼n!
</div>
<button type="button" onClick={() => setColor("green.600")}>
Change Color
</button>
</>
);
const divCss = { fontSize: "2xl", fontWeight: "bold" };
}
To enable debug logging, run the Radipan CLI with DEBUG=true
.