frontend

Scaffold frontend CRUD pages - queries hook, route, page, table, and create dialog. Use after data_model scaffold when building UI for an entity.

$ Instalar

git clone https://github.com/promobase/openpromo /tmp/openpromo && cp -r /tmp/openpromo/.claude/skills/frontend ~/.claude/skills/openpromo

// tip: Run this command in your terminal to install the skill


name: frontend description: Scaffold frontend CRUD pages - queries hook, route, page, table, and create dialog. Use after data_model scaffold when building UI for an entity.

Frontend Scaffolding

Generates React components for CRUD operations on an entity.

Quick Scaffold

python .claude/skills/frontend/scripts/scaffold.py <EntityName>

# Examples:
python .claude/skills/frontend/scripts/scaffold.py Campaign
python .claude/skills/frontend/scripts/scaffold.py BlogPost --dry-run

Prerequisites: Run data_model scaffold first to create oRPC routes.

Generated Files

FilePath
Queries hookpackages/dash/ui/src/queries/{entity-name}.ts
Routepackages/dash/ui/src/routes/_authenticated/workspaces/$workspaceSlug/{entity-name}.tsx
Pagepackages/dash/ui/src/components/{entity-name}/{EntityName}Page.tsx
Tablepackages/dash/ui/src/components/{entity-name}/{EntityName}Table.tsx
Create dialogpackages/dash/ui/src/components/{entity-name}/{EntityName}CreateDialog.tsx

Post-Scaffold Steps

  1. Add route to sidebar (if needed)
  2. Customize table columns
  3. Add form fields to create dialog
  4. Run pnpm lint

Patterns

Grouped Query Hook

All CRUD operations in one hook:

const { list, create, update, remove } = useCampaign();

// List data
const { data, isLoading } = list;

// Mutations
await create.mutateAsync({ workspaceSlug, name });
await update.mutateAsync({ workspaceSlug, id, name });
await remove.mutateAsync({ workspaceSlug, id });

oRPC Direct Usage

For one-off queries, use oRPC directly:

import { orpc } from "@/lib/orpc-client";

// In component
const { data } = useQuery(
  orpc.campaign.list.queryOptions({
    input: { workspaceSlug },
  })
);

// Direct call (in handlers, loaders)
const result = await orpc.campaign.create.call({
  workspaceSlug,
  name: "New Campaign",
});

// Query key for invalidation
queryClient.invalidateQueries({
  queryKey: orpc.campaign.list.key({ input: { workspaceSlug } }),
});

TanStack Router Pattern

import { createFileRoute } from "@tanstack/react-router";
import * as z from "zod";

const searchSchema = z.object({
  status: z.enum(["active", "paused"]).optional(),
  page: z.coerce.number().positive().optional(),
});

export const Route = createFileRoute(
  "/_authenticated/workspaces/$workspaceSlug/campaign",
)({
  validateSearch: searchSchema,
  pendingComponent: WorkspaceLoading,
  component: CampaignPage,
});

Reference Files

ComponentExample
Queriespackages/dash/ui/src/queries/radar.ts
Routepackages/dash/ui/src/routes/_authenticated/workspaces/$workspaceSlug/radar.tsx
Pagepackages/dash/ui/src/components/radar/RadarPage.tsx
Tablepackages/dash/ui/src/components/radar/RadarSourceTable.tsx