npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2026 – Pkg Stats / Ryan Hefner

@stonecrop/nuxt

v0.13.0

Published

Nuxt module for Stonecrop

Readme

Nuxt Stonecrop

npm version npm downloads

The official Nuxt module for Stonecrop - a schema-driven UI framework with event-driven workflows and hierarchical state management.

What is Stonecrop?

Stonecrop is a schema-driven UI framework that generates forms, tables, and workflows from JSON schemas. You define your data structure once and Stonecrop handles UI generation, state management, and validation.

Key Benefits:

  • Schema-Driven: Define data models in JSON; form and table rendering follows automatically
  • HST State Management: Hierarchical State Tree for complex, nested application state
  • FSM Workflows: XState-powered finite state machines for predictable business logic
  • Nuxt Native: First-class integration with Nuxt 4's architecture
  • Live Validation: Real-time form validation based on schema rules
  • Excel-like Tables: Rich table component with keyboard navigation and inline editing

Module Features

  • Route Generation: Scans your /doctypes folder and registers routes using your own page components
  • Plugin System: Auto-registers Stonecrop composables and utilities
  • Theme Support: Import and customize Stonecrop themes
  • TypeScript First: Full type safety and IntelliSense support
  • Thin Wrapper: The module is intentionally opinion-free — page rendering, queries, and navigation stay in your application

Quick Setup

Option 1: Interactive Installer (Recommended)

Use the Stonecrop CLI to interactively install features:

npx @stonecrop/nuxt init

This will prompt you to select which features to install:

  • @stonecrop/nuxt - Frontend module with schema-driven UI
  • @stonecrop/graphql-client - GraphQL client with Stonecrop integration
  • @stonecrop/nuxt-grafserv - GraphQL server with Grafserv
  • @stonecrop/casl-middleware - CASL authorization
  • @stonecrop/rockfoil - PostGraphile middleware for database-driven GraphQL
  • Sample doctypes - Example doctype files to get started

You can also use flags for non-interactive installation:

# Install everything
npx @stonecrop/nuxt init --frontend --graphql-client --graphql --casl --rockfoil --doctypes --yes

# Install just the frontend module
npx @stonecrop/nuxt init --frontend

# Add GraphQL server to existing setup
npx @stonecrop/nuxt init --graphql

# Add PostGraphile middleware
npx @stonecrop/nuxt init --rockfoil

Option 2: Manual Installation

npx nuxi module add @stonecrop/nuxt

That's it! You can now use Stonecrop in your Nuxt app.

Basic Usage

Define a DocType Schema

Create a JSON schema in /doctypes/task.json:

{
  "name": "task",
  "label": "Task",
  "schema": [
    {
      "fieldname": "title",
      "label": "Title",
      "fieldtype": "Data",
      "required": true
    },
    {
      "fieldname": "description",
      "label": "Description",
      "fieldtype": "Text"
    },
    {
      "fieldname": "completed",
      "label": "Completed",
      "fieldtype": "Check"
    }
  ]
}

The module picks up this file and, if pageComponent is configured, registers a route at the doctype's slug value (or task if no slug is set), passing the parsed schema into route.meta.

Wire Up Your Data Client

After installing the module, add a client-side plugin to connect your data transport. Use useStonecropSetup() in plugins — it's designed for the initialization context where Stonecrop may not be fully ready yet:

// app/plugins/stonecrop.client.ts
import { StonecropClient } from '@stonecrop/graphql-client'

export default defineNuxtPlugin(() => {
  const { registerClient, registerMeta } = useStonecropSetup()

  const client = new StonecropClient({ endpoint: '/graphql' })

  // Register the data client for record fetching
  registerClient(client)

  // Configure metadata fetching for lazy-loaded doctypes
  // Called by useStonecrop() when it needs doctype metadata
  registerMeta(({ segments }) => {
    const doctype = segments[0] // e.g., "task" from /task/123
    return client.getMeta({ doctype })
  })
})

This wires useStonecrop()'s automatic record loading to your GraphQL (or any other) backend. Without this step, useStonecrop({ doctype, recordId }) falls back to a REST fetch stub that may not exist in your app.

Use the Stonecrop Composable

In your page or component:

<script setup lang="ts">
import taskDoctype from '~/doctypes/task.json'

// HST-reactive form setup — pass doctype + recordId for full integration
const { stonecrop, provideHSTPath, handleHSTChange, formData } = useStonecrop({
  doctype: taskDoctype,
  recordId: 'task-123' // or undefined for new records
})

// Access the hierarchical state tree directly
const store = stonecrop.value?.getStore()
const taskTitle = store?.get('task.task-123.title')
</script>

<template>
  <AForm
    :schema="taskDoctype.fields"
    :data="formData"
    @update="handleHSTChange"
  />
</template>

Understanding Schema-Driven Development

Traditional Approach:

<!-- Manual form creation -->
<template>
  <form>
    <input v-model="task.title" required />
    <textarea v-model="task.description" />
    <input type="checkbox" v-model="task.completed" />
    <button @click="validate">Save</button>
  </form>
</template>

<script setup>
// Manual validation logic
const validate = () => {
  if (!task.title) {
    errors.title = 'Required'
  }
  // ... more validation
}
</script>

Stonecrop Approach:

<!-- Schema generates form automatically -->
<template>
  <AForm :schema="taskSchema" :data="formData" />
</template>

<script setup>
// Validation is automatic from schema
const { formData } = useStonecrop({
  doctype: taskDoctype,
  recordId: taskId
})
</script>

The schema defines:

  • Field types (text input, checkbox, select, etc.)
  • Validation rules (required, patterns, min/max)
  • Labels and help text
  • Relationships between data models

Configuration

Add options to your nuxt.config.ts:

export default defineNuxtConfig({
  modules: ['@stonecrop/nuxt'],

  stonecrop: {
    // Point to your own page component for slug-based routing (one route per doctype)
    pageComponent: 'pages/StonecropPage.vue',

    // Or supply a custom strategy for full control
    // routeStrategy: (doctypes) => [...],

    // Enable DocBuilder for visual schema editing
    docbuilder: false,
  },

  // Import Stonecrop theme
  css: [
    '@stonecrop/themes/default/default.css',
    // or your custom theme
  ]
})

Module Options

| Option | Type | Description | |--------|------|-------------| | pageComponent | string | Path (relative to srcDir) to your page component. The module registers one route per doctype at /<slug>, passing schema and doctype in route.meta. | | routeStrategy | RouteStrategyFn | Custom function receiving all parsed doctypes; returns a NuxtPage[]. Takes priority over pageComponent. | | docbuilder | boolean | Enable the DocBuilder feature at /docbuilder. Defaults to false. | | doctypesDir | string | Override the doctypes directory path. Defaults to doctypes/ inside srcDir. |

If neither pageComponent nor routeStrategy is configured the module logs a warning and skips doctype route registration.

Route Generation

Default: slug-based routing

The module scans your doctypes/ folder and registers one route per JSON file:

doctypes/
  ├── task.json        → /task  (if no slug field)
  ├── user.json        → /user/:id  (if slug is "user/:id")
  └── project.json     → /project

Each route's meta contains the parsed doctype:

route.meta.schema   // ParsedDoctype['fields']
route.meta.doctype  // ParsedDoctype['data']

Your page component receives these via useRoute():

<script setup lang="ts">
const route = useRoute()
const schema = route.meta.schema   // array of field definitions
const doctype = route.meta.doctype // full doctype JSON object
</script>

Custom route strategy

For full control — multiple routes per doctype, conditional skipping, custom meta — provide a RouteStrategyFn:

import type { RouteStrategyFn } from '@stonecrop/nuxt'
import { resolve } from 'path'

const myStrategy: RouteStrategyFn = (doctypes) =>
  doctypes
    .flatMap(({ fileName, data, fields }) => [
      {
        name: `${fileName}-list`,
        path: `/${data.slug ?? fileName.toLowerCase()}`,
        file: resolve('./pages/ListPage.vue'),
        meta: { schema: fields, doctype: data, viewMode: 'list' },
      },
      {
        name: `${fileName}-detail`,
        path: `/${data.slug ?? fileName.toLowerCase()}/:id`,
        file: resolve('./pages/DetailPage.vue'),
        meta: { schema: fields, doctype: data, viewMode: 'detail' },
      },
    ])

export default defineNuxtConfig({
  stonecrop: { routeStrategy: myStrategy },
})

Plugin Registration

The module auto-registers:

  • useStonecropRegistry() - Composable for wiring data clients and getMeta after plugin install
  • useStonecrop() - Main composable for HST integration (from @stonecrop/stonecrop)
  • Pinia store configuration
  • AForm and ATable component registration (from @stonecrop/aform)

Why Schema-Driven?

Problem: Building CRUD applications is repetitive. Every data model needs:

  • Forms for creating/editing
  • Tables for listing
  • Validation logic
  • State management
  • API integration

Solution: Define the structure once, generate everything automatically.

Benefits:

  • Faster Development: Write less boilerplate code
  • Consistency: All forms/tables follow the same patterns
  • Fewer Bugs: Validation and state management are centralized
  • Self-Documenting: Schemas serve as data model documentation
  • Easy Updates: Change schema, UI updates automatically

useStonecropSetup() — Configuring the Framework in Plugins

When you need to configure Stonecrop during Nuxt plugin initialization (before components mount), use useStonecropSetup(). This composable is designed specifically for the plugin context and returns values that may be undefined if Stonecrop hasn't finished initializing.

// app/plugins/stonecrop.client.ts
import { StonecropClient } from '@stonecrop/graphql-client'

export default defineNuxtPlugin(() => {
  const { isReady, registerClient, registerMeta, registerDoctype } = useStonecropSetup()

  // Check if Stonecrop is ready (module plugins run before project plugins)
  if (!isReady) {
    console.warn('Stonecrop not ready - ensure @stonecrop/nuxt module is installed')
    return
  }

  const client = new StonecropClient({ endpoint: '/graphql' })

  // Register the data client
  registerClient(client)

  // Configure metadata fetching for lazy-loaded doctypes
  registerMeta(({ segments }) => {
    const doctype = segments[0]
    return client.getMeta({ doctype })
  })

  // Optionally pre-load doctypes into the Registry
  const planDoctype = Doctype.fromObject({ name: 'plan', fields: [...] })
  registerDoctype(planDoctype)
})

useStonecropSetup() vs useStonecropRegistry()

| Context | Use | Behavior | |---------|-----|----------| | Plugin/initialization | useStonecropSetup() | Returns values that may be undefined; provides isReady check | | Component/runtime | useStonecropRegistry() | Throws if Stonecrop isn't initialized (expected to be ready) |

useStonecropSetup() API

| Property/Method | Type | Description | |-----------------|------|-------------| | registry | Registry \| undefined | The Registry instance (undefined if not ready) | | stonecrop | Stonecrop \| undefined | The Stonecrop instance (undefined if not ready) | | isReady | boolean | true when both registry and stonecrop are available | | registerClient(client) | (client: DataClient) => void | Set the data client for record fetching. Throws if stonecrop not available. | | getClient() | () => DataClient \| undefined | Get the currently configured client. | | registerMeta(fn) | (fn: (ctx) => Doctype) => void | Set the getMeta function on the Registry. Throws if registry not available. | | registerDoctype(doctype) | (doctype: Doctype) => void | Pre-load a doctype into the Registry. Throws if registry not available. | | dispatchAction(...) | Promise<{ success, data, error }> | Dispatch an action via the configured client. |

useStonecropRegistry() — Using the Framework in Components

useStonecropRegistry() is for component/runtime context where the framework is expected to be fully initialized. Use it in components to access the configured registry and stonecrop instances.

// In a component or composable
const { registry, stonecrop, dispatchAction } = useStonecropRegistry()

// Access doctype metadata
const plan = registry.getDoctype('plan')

// Dispatch actions
await dispatchAction({ name: 'plan' }, 'SUBMIT', [recordId])

Note: If called before Stonecrop is initialized (e.g., in a plugin), this throws with guidance to use useStonecropSetup() instead.

useStonecropRegistry() API

| Property/Method | Type | Description | |-----------------|------|-------------| | registry | Registry | The Registry instance for doctype management. | | stonecrop | Stonecrop | The Stonecrop instance for HST and operation log access. Throws if not initialized. | | setMeta(fn) | (fn: (ctx) => Doctype \| Promise<Doctype>) => void | Sets the getMeta function on the Registry. Called by useStonecrop() to lazy-load doctype metadata for the current route. ctx = { path, segments }. | | setClient(client) | (client: DataClient) => void | Set the data client for record fetching. Throws if stonecrop not available. | | getClient() | () => DataClient \| undefined | Get the currently configured client. | | dispatchAction(doctype, action, args) | Promise<{ success, data, error }> | Dispatch an action via the configured client. Returns error if doctype not found in registry. |

Advanced Features

Hierarchical State Tree (HST)

HST provides path-based state addressing:

const store = stonecrop.getStore()

// Set nested values with dot notation
store.set('project.proj-1.tasks.task-1.completed', true)

// Get values anywhere in the tree
const completed = store.get('project.proj-1.tasks.task-1.completed')

// Navigate the tree hierarchy
const taskNode = store.getNode('project.proj-1.tasks.task-1')
const ancestor = taskNode.getAncestor() // Returns project node
const breadcrumbs = taskNode.getBreadcrumbs()

XState Integration

Define workflows as finite state machines:

import { createMachine } from 'xstate'

const taskMachine = createMachine({
  id: 'task',
  initial: 'draft',
  states: {
    draft: {
      on: { SUBMIT: 'pending' }
    },
    pending: {
      on: {
        APPROVE: 'completed',
        REJECT: 'draft'
      }
    },
    completed: {
      type: 'final'
    }
  }
})

// Stonecrop persists state machine data in HST

Examples

Check out the playground for an example featuring:

  • Permission management system (RBAC)
  • DocType builder with visual state machine editor
  • Complex nested forms with relationships
  • Table views with inline editing
  • HST state visualization

Contribution

# Install dependencies
npm install

# Generate type stubs
npm run dev:prepare

# Develop with the playground
npm run dev

# Build the playground
npm run dev:build

# Run ESLint
npm run lint

# Run Vitest
npm run test
npm run test:watch

Testing the CLI Locally

To test the npx @stonecrop/nuxt init command from another directory outside this project:

1. Build the monorepo (from stonecrop root):

cd /path/to/stonecrop
rush update
rush build

2. Create a test Nuxt project (in a separate directory):

cd /tmp  # or any directory outside stonecrop
npx nuxi init my-test-app
cd my-test-app
npm install

3. Run the CLI using the local package:

# Option A: Run from within the nuxt package directory (simplest)
# This ensures Node can find the dependencies
cd /path/to/stonecrop/nuxt
node bin/init.mjs init --cwd /tmp/my-test-app

# Option B: Use pnpm link (from the test project)
cd /path/to/stonecrop/nuxt
pnpm link --global
cd /tmp/my-test-app
pnpm link --global @stonecrop/nuxt
npx stonecrop-nuxt init

# Option C: Use npm pack to create a tarball (simulates real npm install)
cd /path/to/stonecrop/nuxt
npm pack
cd /tmp/my-test-app
npm install /path/to/stonecrop/nuxt/stonecrop-nuxt-0.6.3.tgz
npx stonecrop-nuxt init

Note: Option A uses --cwd to specify the target directory while running from within the nuxt package where dependencies are available. Options B and C install the package into the test project so dependencies are resolved correctly.

4. Interactive testing:

The CLI will detect that you're in a Nuxt project and prompt for features:

🌱 Stonecrop Nuxt Installer

✔ Nuxt project detected

? Select features to install
  ◉ @stonecrop/nuxt - Frontend module
  ◯ @stonecrop/nuxt-grafserv - GraphQL server
  ◯ @stonecrop/casl-middleware - Authorization
  ◉ Sample doctypes

5. Verify the installation:

After running the installer, check:

  • package.json has the new dependencies
  • nuxt.config.ts has the module configuration
  • doctypes/ folder contains sample schemas (if selected)
  • server/ folder contains GraphQL files (if selected)