A React component library for building and editing JSON Schema with a visual UI, a JSON source editor, schema inference, and JSON validation.
Try online: https://json.ophir.dev
npm install jsonjoy-builderJSONJoy Builder expects React, React DOM, and Monaco Editor to be available in your app:
npm install react react-dom monaco-editorImport the stylesheet once, usually in your app entry point:
import "jsonjoy-builder/styles.css";Use SchemaBuilder when you want the full editor: visual editing on one side and editable JSON source on the other.
import "jsonjoy-builder/styles.css";
import { SchemaBuilder, type JsonSchema } from "jsonjoy-builder";
import { useState } from "react";
export function App() {
const [schema, setSchema] = useState<JsonSchema>({
type: "object",
properties: {},
});
return (
<SchemaBuilder
value={schema}
onChange={setSchema}
readOnly={false}
/>
);
}The editor is controlled: pass the current value, then persist updates from onChange.
SchemaBuilder is the best default for application screens. It includes the visual editor, JSON source editor, fullscreen mode, and a draggable split view on desktop.
<SchemaBuilder value={schema} onChange={setSchema} readOnly={false} />SchemaFieldsEditor renders only the visual schema builder. Use it when your app already has its own source preview, tabs, or surrounding layout.
import { SchemaFieldsEditor } from "jsonjoy-builder";
<SchemaFieldsEditor
value={schema}
onChange={setSchema}
readOnly={false}
/>SchemaJsonEditor renders only the Monaco JSON editor. It can be read-only or editable depending on whether you pass readOnly.
import { SchemaJsonEditor } from "jsonjoy-builder";
<SchemaJsonEditor value={schema} onChange={setSchema} />Most apps only need to handle the whole-schema change callback:
const [schema, setSchema] = useState<JsonSchema>({ type: "object" });
<SchemaBuilder
value={schema}
onChange={(nextSchema) => {
setSchema(nextSchema);
saveDraft(nextSchema);
}}
readOnly={false}
/>For visual-editor-only usage, the same pattern is named onChange:
<SchemaFieldsEditor
value={schema}
onChange={setSchema}
readOnly={false}
/>For quick setup, omit value and start with defaultValue:
<SchemaBuilder defaultValue={{ type: "object" }} onChange={setSchema} />Use InferSchemaDialog when you want users to paste example JSON and generate a starting schema.
import {
InferSchemaDialog,
SchemaBuilder,
type JsonSchema,
} from "jsonjoy-builder";
import { useState } from "react";
export function App() {
const [schema, setSchema] = useState<JsonSchema>({ type: "object" });
const [inferOpen, setInferOpen] = useState(false);
return (
<>
<button type="button" onClick={() => setInferOpen(true)}>
Infer from JSON
</button>
<SchemaBuilder
value={schema}
onChange={setSchema}
readOnly={false}
/>
<InferSchemaDialog
open={inferOpen}
onOpenChange={setInferOpen}
onInfer={setSchema}
/>
</>
);
}Inference detects object properties, arrays, strings, numbers, booleans, required fields, and common string formats such as dates, emails, and URIs.
Use ValidateJsonDialog when you want users to test JSON data against the current schema.
import { ValidateJsonDialog, type JsonSchema } from "jsonjoy-builder";
import { useState } from "react";
export function ValidateButton({ schema }: { schema: JsonSchema }) {
const [open, setOpen] = useState(false);
return (
<>
<button type="button" onClick={() => setOpen(true)}>
Validate JSON
</button>
<ValidateJsonDialog
open={open}
onOpenChange={setOpen}
schema={schema}
/>
</>
);
}Validation runs as the user types and reports syntax errors, schema validation errors, and line or column locations when available.
JSONJoy Builder ships with default light and dark theme variables. All exported components include a .jsonjoy wrapper, so you can theme one editor instance or your whole app.
.jsonjoy {
--background: hsl(210 40% 98%);
--foreground: hsl(222.2 84% 4.9%);
--card: hsl(0 0% 100%);
--card-foreground: hsl(222.2 84% 4.9%);
--popover: hsl(0 0% 100%);
--popover-foreground: hsl(222.2 84% 4.9%);
--primary: hsl(210 100% 50%);
--primary-foreground: hsl(210 40% 98%);
--secondary: hsl(210 40% 96.1%);
--secondary-foreground: hsl(222.2 47.4% 11.2%);
--muted: hsl(210 40% 96.1%);
--muted-foreground: hsl(215.4 16.3% 46.9%);
--accent: hsl(210 40% 96.1%);
--accent-foreground: hsl(222.2 47.4% 11.2%);
--destructive: hsl(0 84.2% 60.2%);
--destructive-foreground: hsl(210 40% 98%);
--border: hsl(214.3 31.8% 91.4%);
--input: hsl(214.3 31.8% 91.4%);
--ring: hsl(222.2 84% 4.9%);
--radius: 0.8rem;
--font-sans: "Inter", system-ui, sans-serif;
}For dark mode, add dark to the .jsonjoy element or to one of its ancestors:
<div className="dark">
<SchemaBuilder value={schema} onChange={setSchema} readOnly={false} />
</div>Monaco automatically follows the detected light or dark JSONJoy theme.
The default language is English. Pass a bundled locale directly to the component for simple localization.
import {
SchemaBuilder,
de,
} from "jsonjoy-builder";
<SchemaBuilder value={schema} onChange={setSchema} locale={de} />Bundled locales are exported for English, German, Spanish, French, Polish, Russian, Ukrainian, and Chinese.
For app-wide defaults, wrap a section in SchemaBuilderProvider:
import { SchemaBuilderProvider, fr } from "jsonjoy-builder";
<SchemaBuilderProvider locale={fr} messages={{ schemaEditorTitle: "Schema" }}>
<SchemaBuilder value={schema} onChange={setSchema} />
</SchemaBuilderProvider>You can also provide your own translation object or override individual messages:
import { type Translation } from "jsonjoy-builder";
const customTranslation: Translation = {
// Add every key from the Translation type.
};Use src/i18n/locales/en.ts as the reference for all available keys.
The visual editor covers the common schema authoring flow:
- Object fields and nested objects
- Pattern properties for regex-matched property names
- Required and optional fields
- Strings, numbers, booleans, arrays, objects, and null values
- String length, regex pattern, and format constraints
- Number range, exclusivity, multiple-of, and enum constraints
- Array item type, length, uniqueness, and contains constraints
- Boolean allowed-value constraints
anyOf,oneOf, andallOfcompositionadditionalPropertiescontrols
You can still edit unsupported JSON Schema keywords directly in the JSON source editor.
git clone https://github.com/lovasoa/jsonjoy-builder.git
cd jsonjoy-builder
npm install
npm run devThe demo application runs at http://localhost:5173.
Build the library:
npm run buildCommon scripts:
| Command | Description |
|---|---|
npm run dev |
Start the demo development server |
npm run build |
Build the library for production |
npm run build:demo |
Build the demo app |
npm run preview |
Preview the production demo build |
npm run check |
Run Biome checks |
npm run fix |
Fix Biome issues |
npm run typecheck |
Type check with TypeScript |
npm run test |
Run tests |
MIT
