UI Builder solves the fundamental problem of UI creation tools: they ignore your existing React component library and force you to rebuild from scratch.
UI Builder is a shadcn/ui package that adds a Figma‑style editor to your own product, letting non‑developers compose pages, emails, dashboards, and white‑label views with the exact React components you already ship.
Layouts are saved as plain JSON for easy versioning and can be rendered instantly with dynamic data allowing:
- your marketing team to update a landing page without waiting on engineering.
- a customer to tweak a branded portal with their own content and branding.
- a product manager to modify email templates but parts of the content is dynamic for each user.
- add a visual "head" to your headless CMS, connecting your content API with your component library.
How it unlocks novel product features:
- Give users no‑code superpowers — add a full visual builder to your SaaS with one install.
- Design with components you already ship — nothing new to build or maintain.
- Store layouts as human‑readable JSON — render inside your product to ship changes immediately.
- Create dynamic, data-driven interfaces — bind component properties to variables for personalized content.
See the docs site for more information.
Tailwind 4 + React 19: Migration coming soon. Currently blocked by 3rd party component compatibility. If using latest shadcn/ui CLI fails, try: npx shadcn@2.1.8 add ...
Server Components: Not supported. RSC can't be re-rendered client-side for live preview. A separate RSC renderer for final page rendering is possible — open an issue if you have a use case.
If you are using shadcn/ui in your project, you can install the component directly from the registry.
npx shadcn@latest add https://raw.githubusercontent.com/olliethedev/ui-builder/main/registry/block-registry.json
Or you can start a new project with the UI Builder:
npx shadcn@latest init https://raw.githubusercontent.com/olliethedev/ui-builder/main/registry/block-registry.json
Note: You need to use style variables to have page theming working correctly.
If you are not using shadcn/ui, you can install the component simply by copying the files in this repo into your project.
Add dev dependencies, since there currently seems to be an issue with shadcn/ui not installing them from the registry:
npm install -D @types/lodash.template @tailwindcss/typography @types/react-syntax-highlighter tailwindcss-animate @types/object-hash
And that's it! You have a UI Builder that you can use to build your UI.
To use the UI Builder, you just need to provide a component registry.
import z from "zod"; import UIBuilder from "@/components/ui/ui-builder"; import { Button } from "@/components/ui/button"; import { ComponentRegistry, ComponentLayer } from "@/components/ui/ui-builder/types"; import { commonFieldOverrides, classNameFieldOverrides, childrenAsTextareaFieldOverrides } from "@/lib/ui-builder/registry/form-field-overrides"; // Define your component registry const myComponentRegistry: ComponentRegistry = { Button: { component: Button, schema: z.object({ className: z.string().optional(), children: z.any().optional(), variant: z .enum([ "default", "destructive", "outline", "secondary", "ghost", "link", ]) .default("default"), size: z.enum(["default", "sm", "lg", "icon"]).default("default"), }), from: "@/components/ui/button", defaultChildren: [ { id: "button-text", type: "span", name: "span", props: {}, children: "Hello World", } satisfies ComponentLayer, ], fieldOverrides: commonFieldOverrides() }, span: { schema: z.object({ className: z.string().optional(), children: z.any().optional(), }), fieldOverrides: { className:(layer)=> classNameFieldOverrides(layer), children: (layer)=> childrenAsTextareaFieldOverrides(layer) }, defaultChildren: "Text" }, }; export function App() { return ( <UIBuilder componentRegistry={myComponentRegistry} /> ); }
Note: fieldOverrides is optional, they provide the ability to customize the auto-generated form fields for the component's properties in the editor's sidebar. In the above example we are using the commonFieldOverrides to provide custom form fields for children, className props. Read more about fieldOverrides in the Add your custom components to the registry section.
Important: Make sure to include definitions for all component types referenced in your defaultChildren
or initialLayers
. In this example, the Button component's defaultChildren
references a span
component, so we need to include span
in our component registry. For a more complete setup, you can use the pre-built component definitions:
import { primitiveComponentDefinitions } from "@/lib/ui-builder/registry/primitive-component-definitions"; import { complexComponentDefinitions } from "@/lib/ui-builder/registry/complex-component-definitions"; const myComponentRegistry: ComponentRegistry = { ...primitiveComponentDefinitions, // Includes span, div, img, etc. ...complexComponentDefinitions, // Includes Button, Badge, Card, etc. // Your custom components... };
You can initialize the UI with initial layers from a database and use the onChange callback to persist the state to a database. Variables make your pages dynamic by allowing you to bind component properties to data that can change at runtime.
import React from "react"; import UIBuilder from "@/components/ui/ui-builder"; import { ComponentLayer, ComponentRegistry } from "@/components/ui/ui-builder/types"; import { Variable } from '@/components/ui/ui-builder/types'; const myComponentRegistry: ComponentRegistry = {...}; // Your component registry // Static initial layers or you can fetch from database const initialLayers: ComponentLayer[] = [ { id: "page1", type: "div", // Ensure div is in your componentRegistry name: "Page 1", props: { className: "p-4 flex flex-col gap-2", }, children: [ { id: "qCTIIed", type: "Button", // Ensure Button is in your componentRegistry name: "Button", props: { variant: "default", size: "default", className: "w-full items-center gap-2 max-w-sm", }, children: [ { id: "UzZY6Dp", type: "span", // Ensure span is in your componentRegistry name: "span", props: {}, children: "Hello World", }, { id: "hn3PF6A", type: "Icon", // Ensure Icon is in your componentRegistry name: "Icon", props: { size: "medium", color: "secondary", rotate: "none", iconName: "Github", className: "", }, children: [], }, ], }, ], }, ]; // Define variables to make your pages dynamic and reusable // Variables can be bound to component properties for personalized content const initialVariables: Variable[] = [ { id: "var1", name: "userName", type: "string", defaultValue: "John Doe" }, { id: "var2", name: "userAge", type: "number", defaultValue: 25 }, { id: "var3", name: "isActive", type: "boolean", defaultValue: true } ]; const App = () => { const handleLayersChange = (updatedLayers: ComponentLayer[]) => { // Here you can send the updated layers to the backend console.log("Layers updated:", updatedLayers); }; const handleVariablesChange = (updatedVariables: Variable[]) => { // Here you can send the updated variables to the backend console.log("Variables updated:", updatedVariables); }; return ( <div> <UIBuilder initialLayers={initialLayers} onChange={handleLayersChange} initialVariables={initialVariables} onVariablesChange={handleVariablesChange} componentRegistry={myComponentRegistry} /> </div> ); }; export default App;
Note: Variables are optional, but they are a powerful way to make your pages dynamic and reusable. Read more about variables in the Understanding Variables section.
componentRegistry
: Required. An object mapping component type names to their definitions (component, schema, import path, etc.).initialLayers
: Optional prop to set up initial pages and layers. Useful for setting the initial state of the builder, from a database for example.onChange
: Optional callback triggered when the editor state changes, providing the updated pages. Can be used to persist the state to a database.initialVariables
: Optional prop to set up initial variables. Useful for setting the initial variable state of the builder, from a database for example. Variables enable dynamic content binding.onVariablesChange
: Optional callback triggered when the variables change, providing the updated variables. Can be used to persist the variable state to a database.panelConfig
: Optional. An object to customize the different panels of the UI Builder (e.g., nav bar, editor panel, props panel). If not provided, a default configuration is used. This allows for fine-grained control over the editor's appearance and layout.persistLayerStore
: Optional boolean (defaults totrue
). Determines whether the editor's state (layers and their configurations) is persisted in the browser's local storage across sessions. Set tofalse
to disable local storage persistence, useful if you are managing state entirely throughinitialLayers
andonChange
.allowVariableEditing
: Optional boolean (defaults totrue
). Controls whether users can edit variables in the Variables panel. Whentrue
, users can add, edit, and delete variables. Whenfalse
, the Variables panel becomes read-only, hiding the "Add Variable" button and the edit/delete buttons on individual variables. Useful when you want to provide a read-only variables experience or manage variables entirely throughinitialVariables
.allowPagesCreation
: Optional boolean (defaults totrue
). Controls whether users can create new pages in the editor. Whentrue
, users can add new pages to the editor. Whenfalse
, the Pages panel becomes read-only, hiding the "Add Page" button. Useful when you want to provide a read-only pages experience or manage pages entirely throughinitialLayers
.allowPagesDeletion
: Optional boolean (defaults totrue
). Controls whether users can delete pages in the editor. Whentrue
, users can delete pages from the editor. Whenfalse
, the Pages panel becomes read-only, hiding the "Delete Page" button. Useful when you want to provide a read-only pages experience or manage pages entirely throughinitialLayers
.
You can also render the page layer without editor functionality by using the LayerRenderer component. This is where variables become powerful - you can override variable values at runtime to create personalized, dynamic content.
import LayerRenderer from "@/components/ui/ui-builder/layer-renderer"; import { ComponentLayer } from "@/components/ui/ui-builder/types"; import { ComponentRegistry } from "@/components/ui/ui-builder/types"; import { Variable } from '@/components/ui/ui-builder/types'; const myComponentRegistry: ComponentRegistry = {...}; // Your component registry const page: ComponentLayer = {...} // Fetch or define your page layer // Define variables for the page const variables: Variable[] = [ { id: "var1", name: "userName", type: "string", defaultValue: "John Doe" }, { id: "var2", name: "userAge", type: "number", defaultValue: 25 } ]; // Override variable values at runtime for personalized content // This is where the magic happens - same page structure, different data! const variableValues = { var1: "Jane Smith", // Override userName var2: 30 // Override userAge }; export function MyPage() { return ( <LayerRenderer page={page} componentRegistry={myComponentRegistry} variables={variables} variableValues={variableValues} /> ); }
page
: Required. The page layer to render.componentRegistry
: Required. An object mapping component type names to their definitions.className
: Optional. CSS class name for the root container.editorConfig
: Optional. Configuration for editor-specific functionality (usually not needed for rendering).variables
: Optional. Array of variable definitions available for binding in the page. These define the structure and types of dynamic data.variableValues
: Optional. Object mapping variable IDs to their runtime values, overriding default values. This is how you inject real data into your templates.
You add custom components by defining them in a ComponentRegistry
object and passing that object to the UIBuilder
component via the componentRegistry
prop. The registry maps a unique component type name (string) to its definition.
Here is an example of how to define a custom component within the registry object:
import { z } from 'zod'; import { FancyComponent } from '@/components/ui/fancy-component'; // Example custom component with className, children, title, count, disabled, timestamp, mode props import { classNameFieldOverrides, childrenFieldOverrides } from "@/lib/ui-builder/registry/form-field-overrides"; import { ComponentRegistry } from "@/components/ui/ui-builder/types"; // Define your custom component's schema and configuration const fancyComponentDefinition = { // The React component itself component: FancyComponent, // The Zod schema defining the component's props. This will be used to generate the properties form in the editor schema: z.object({ className: z.string().optional(), children: z.any().optional(), // Use z.any() for children prop and provide custom handling in fieldOverrides title: z.string().default("Default Title"), count: z.coerce.number().default(1), disabled: z.boolean().optional(), timestamp: z.coerce.date().optional(), mode: z .enum([ "fancy", "boring" ]) .default("fancy"), }), // The import path for code generation from: "@/components/ui/fancy-component", // Customizations for the properties auto-form fieldOverrides: { className: (layer) => classNameFieldOverrides(layer), // Handle className selection with a custom component that is not a default auto-form component children: (layer) => childrenFieldOverrides(layer), // Handle children selection with a custom component that is not a default auto-form component // You can add your own component overrides for other fields here } }; // Create the full registry, potentially combining with other component definitions export const myComponentRegistry: ComponentRegistry = { FancyComponent: fancyComponentDefinition, }; // Then pass `myComponentRegistry` to the UIBuilder prop: // <UIBuilder componentRegistry={myComponentRegistry} />
Here's a practical example showing how to use defaultVariableBindings
to create components that automatically bind to system variables:
import { z } from 'zod'; import { UserProfile } from '@/components/ui/user-profile'; import { BrandedButton } from '@/components/ui/branded-button'; import { ComponentRegistry } from "@/components/ui/ui-builder/types"; // First, define your variables (these would typically come from your app's state) const systemVariables = [ { id: 'user-id-var', name: 'currentUserId', type: 'string', defaultValue: 'user123' }, { id: 'user-name-var', name: 'currentUserName', type: 'string', defaultValue: 'John Doe' }, { id: 'brand-color-var', name: 'primaryBrandColor', type: 'string', defaultValue: '#3b82f6' }, { id: 'company-name-var', name: 'companyName', type: 'string', defaultValue: 'Acme Corp' }, ]; // Define components with automatic variable bindings const myComponentRegistry: ComponentRegistry = { UserProfile: { component: UserProfile, schema: z.object({ userId: z.string().default(''), displayName: z.string().default('Anonymous'), showAvatar: z.boolean().default(true), }), from: "@/components/ui/user-profile", // Automatically bind user data when component is added defaultVariableBindings: [ { propName: 'userId', variableId: 'user-id-var', immutable: true }, // System data - can't be changed { propName: 'displayName', variableId: 'user-name-var', immutable: false }, // Can be overridden ], }, BrandedButton: { component: BrandedButton, schema: z.object({ text: z.string().default('Click me'), brandColor: z.string().default('#000000'), companyName: z.string().default('Company'), }), from: "@/components/ui/branded-button", // Automatically apply branding when component is added defaultVariableBindings: [ { propName: 'brandColor', variableId: 'brand-color-var', immutable: true }, // Brand consistency { propName: 'companyName', variableId: 'company-name-var', immutable: true }, // Brand consistency // 'text' is not bound, allowing content editors to customize button text ], }, }; // Usage in your app const App = () => { return ( <UIBuilder componentRegistry={myComponentRegistry} initialVariables={systemVariables} // When users add UserProfile or BrandedButton components, // they'll automatically be bound to the appropriate variables /> ); };
In this example:
- UserProfile components automatically bind to current user data, with
userId
locked (immutable) for security - BrandedButton components automatically inherit brand colors and company name, ensuring visual consistency
- Content editors can still customize the button text, but can't accidentally break branding
- The immutable bindings prevent accidental unbinding of critical system or brand data
Component Definition Fields:
component
: Required. The React component function or class.schema
: Required. A Zod schema defining the component's props, their types, and validation rules.- This schema powers the automatic generation of a properties form in the editor using Auto-Form.
- Currently supported Zod types for auto-form generation include:
boolean
,date
,number
,string
,enum
(of supported types),object
(with supported property types), andarray
(of objects with supported property types).
from
: Required. The source import path for the component. This is used when generating React code.isFromDefaultExport
: Optional. If true, indicates the component should be imported as a default export when generating React code.fieldOverrides
: Optional. An object to customize the auto-generated form fields for the component's properties in the editor's sidebar.- The keys of this object correspond to the prop names defined in the Zod schema.
- The values are typically functions that receive the current
layer
object and return configuration options for theAutoForm
field. These options can control the field's label, input type (fieldType
), visibility (isHidden
), placeholder text, render logic, and more. See the AutoForm documentation for available options. - This is useful for:
- Providing more descriptive labels or help text.
- Using specific input components (e.g., a color picker, a custom slider).
- Hiding props that shouldn't be user-editable (like internal state).
- Implementing conditional logic (e.g., showing/hiding a field based on another prop's value).
- The example uses
classNameFieldOverrides
andchildrenFieldOverrides
from@/lib/ui-builder/registry/form-field-overrides
to provide standardized handling for common props likeclassName
(using a auto suggest text input) andchildren
(using a custom component). You can create your own override functions or objects.
defaultChildren
: Optional. Default children to use when a new instance of this component is added to the canvas. For example setting initial text on a span component.defaultVariableBindings
: Optional. An array of default variable bindings to apply when a new instance of this component is added to the canvas. This enables automatic binding of component properties to variables, with optional immutability controls.- Each binding object contains:
propName
: The name of the component property to bindvariableId
: The ID of the variable to bind to this propertyimmutable
: Optional boolean (defaults tofalse
). Whentrue
, prevents users from unbinding this variable in the UI, ensuring the binding remains intact
- Use cases for immutable bindings:
- System-level data: Bind user ID, tenant ID, or other system variables that shouldn't be changed by content editors
- Branding consistency: Lock brand colors, logos, or company names to maintain visual consistency
- Security: Prevent modification of security-related variables like permissions or access levels
- Template integrity: Ensure critical template variables remain bound in white-label or multi-tenant scenarios
- Example:
defaultVariableBindings: [ { propName: 'userEmail', variableId: 'user-email-var', immutable: true }, { propName: 'welcomeMessage', variableId: 'welcome-msg-var', immutable: false } ]
- Each binding object contains:
You can customize the tabs in the left "Page Config" panel (by default, "Layers" and "Appearance") by providing a pageConfigPanelTabsContent
property inside the panelConfig
prop. This allows you to change the tab titles or provide custom content for each tab. For example, you might want to provide your own appearance tab if you not using tailwind or a custom subset of tailwind like in React-Email .
import UIBuilder, { defaultConfigTabsContent } from "@/components/ui/ui-builder"; // Example: Customizing the tabs const myTabsContent = { layers: { title: "My Layers", content: <MyCustomLayersPanel /> }, appearance: { title: "Theme", content: <MyCustomAppearancePanel /> }, }; <UIBuilder componentRegistry={myComponentRegistry} panelConfig={{ pageConfigPanelTabsContent: myTabsContent, // ...other panel overrides }} />
If you do not provide this, the default tabs ("Layers" and "Appearance") will be used. You can also override other panels in the panelConfig
like editorPanel
, propsPanel
, and navBar
.
UI Builder empowers you to visually construct and modify user interfaces by leveraging your own React components. Here's a breakdown of its core principles:
-
Component-Driven Foundation: At its heart, UI Builder operates on your existing React components. You provide a
componentRegistry
detailing the components you want to make available in the editor. This registry includes the component itself, its Zod schema for props, its import path for code generation, and optional UI customizations for the editor. -
Layer-Based Canvas: The user interface is constructed as a tree of "layers." Each layer represents an instance of a component from your registry. Users can visually add, remove, reorder, and nest these layers on an interactive canvas, offering a direct manipulation experience. Any component type, like
div
,main
, or custom containers, can serve as the root page layer. -
Dynamic Props Editing: Each component in the registry is defined with a Zod schema that outlines its props, their types, and default values. UI Builder utilizes this schema to automatically generate a properties panel (using Auto-Form). This allows users to configure component instances in real-time, with changes immediately reflected on the canvas.
-
Variable-Driven Dynamic Content: Variables transform static designs into dynamic, data-driven interfaces. Users can define typed variables (string, number, boolean) and bind them to component properties. This enables:
- Personalized content that adapts to user data
- Reusable templates that work across different contexts
- Multi-tenant applications with customized branding per client
- A/B testing and feature flags through boolean variables
- Content management where non-technical users can update dynamic content
-
Flexible State Management: By default, the editor's state (the arrangement and configuration of layers, plus variables) is persisted in the browser's local storage for convenience across sessions. For more robust or backend-integrated solutions, you can provide
initialLayers
andinitialVariables
to seed the editor (e.g., from a database) and use theonChange
andonVariablesChange
callbacks to capture state changes and persist them as needed. -
React Code Generation: A key feature is the ability to export the visually designed page structure as clean, readable React code. This process uses the import paths (
from
property) specified in your component registry, ensuring the generated code correctly references your components. Variable bindings are converted to proper TypeScript interfaces and prop references. -
Runtime Variable Resolution: When rendering pages with the
LayerRenderer
, you can providevariableValues
to override default variable values with real data from APIs, databases, or user input. This separation of structure and content enables powerful use cases like personalized dashboards, white-label applications, and dynamic email templates. -
Extensibility and Customization: The system is designed for deep integration with your project:
- Customize the properties form for each component using
fieldOverrides
to enhance the editing experience (e.g., custom labels, input types, or conditional visibility). - Enable variable binding for specific component properties to make them dynamic.
- Provide custom React components via the
panelConfig
prop to provide custom components for the editor panels. - If you only need to display a UI Builder page without the editor, the
LayerRenderer
component can render it using the samecomponentRegistry
, serialized page data, and variable values.
- Customize the properties form for each component using
Variables are the key to creating dynamic, data-driven interfaces with UI Builder. Instead of hardcoding static values into your components, variables allow you to bind component properties to dynamic data that can change at runtime.
Variables in UI Builder are typed data containers that can be bound to component properties. They support three types:
- String: For text content, names, descriptions, etc.
- Number: For counts, ages, prices, quantities, etc.
- Boolean: For feature flags, visibility toggles, active states, etc.
Variables unlock powerful use cases that transform static designs into dynamic applications:
Bind user-specific data to create personalized experiences:
// Variable: userName = "Jane Smith" // Button text becomes "Welcome, Jane Smith!"
Create one design that works for multiple contexts:
- Marketing teams can create landing page templates with variables for product names, pricing, and features
- Customer portals can use the same layout with different branding variables per client
- Email templates can have variables for recipient names, company info, and dynamic content
Build white-label solutions where each customer gets customized interfaces:
// Variables: companyName, brandColor, logoUrl // Same page structure, different branding per tenant
Use boolean variables to control feature visibility:
// Variable: showNewFeature = true/false // Conditionally show/hide components based on feature flags
Separate content from structure, allowing non-technical users to update dynamic content without touching the layout.
- Internal structure of the editor directory has changed. When calling shadcn add, you will need to manually delete old files in /ui-builder and /lib/ui-builder directories.
- Many components have been refactored to support drag and drop functionality and the new canvas.
- Removed page layer type in favor of using any component type (like
div
,main
, or custom containers) as the root page layer. This enhances flexibility, enabling use cases like building react-email templates directly. You should migrate any layers stored in the database to use a standard component type as the root. The migrateV2ToV3 function inlayer-utils.ts
can assist with this migration. - The
componentRegistry
is now passed as a prop to theUIBuilder
component instead of being defined in a standalone file. - Removed the script used to generate component schema definitions. This approach proved problematic to maintain and didn't function correctly for complex components or varying project setups. Component schema definitions should now be manually created or generated using project-specific tooling if desired.
panelConfig
prop added toUIBuilder
to allow customization of the page config panel tabs.- Made
layer-store
local storage persistence optional and configurable via props.
- Removed text layer type in favor of using span and Markdown components. You should migrate any layers stored in the database to use the new components. You can use the migrateV1ToV2 function in layer-utils.ts to help with the migration.
Build component registry after updating lib or ui:
npm run build-registry
Host the registry locally:
npm run host-registry
Use the local registry in a local project:
npx shadcn@latest add http://127.0.0.1:8080/block-registry.json -o
npm run test
- Add variable binding to layer children and not just props
- Update to React 19
- Update to latest Shadcn/ui + Tailwind CSS v4
- Add Blocks. Reusable component blocks that can be used in multiple pages
- Move component schemas to separate shadcn registry to keep main registry light
- Move prop form field components (overrides) to separate shadcn registry to keep main registry light
- Add visual data model editor + code gen for backend code for CRUD operations. (ex Zenstack schema editor)
- Add event handlers to component layers (onClick, onSubmit, etc)
- Update to new AutoForm when stable
- Update to Zod v4 (when stable) for native json schema conversion to enforce safety in layer props
- React Native support
- VS Code plugin integration for UI Builder
Contributions are welcome! Please feel free to submit issues and pull requests.
MIT