Best Practices for Structuring Your Zap.ts Project
Zap.ts is built to help you create apps quickly with a clear and organized structure. This page explains the recommended way to set up your project in the src
folder and why it works well for building modern apps.
Project Structure Overview
The src
folder in Zap.ts is organized to keep your code clean, easy to find, and ready to grow. Here’s what each folder does:
- actions: Holds server actions for handling backend tasks, like form submissions (more on this below).
- app: Contains Next.js app router files, such as pages and layouts.
- components: Stores reusable UI components, split into
common
andui
subfolders:- common: General components used across the app, like a user profile card (e.g.,
ProfileCard.tsx
). - ui: Components styled with shadcn/ui, focusing on design consistency (e.g.,
Card.tsx
).
- common: General components used across the app, like a user profile card (e.g.,
- data: Keeps static data, like JSON files, and feature flags to turn features on or off at build time without a database.
- db: Includes database-related code, such as Drizzle ORM migrations and queries.
- features: Organizes app features into separate folders (e.g.,
push-notifications
), each with its ownhooks
andcomponents
folder. - hooks: Holds custom React hooks that can be shared across features.
- lib: Stores shared utilities, like helper functions or API clients.
- providers: Contains React context providers for app-wide state (but use Zustand stores when possible—see below).
- rpc: Manages oRPC code for type-safe API communication.
- schemas: Defines Drizzle ORM schemas and API validation schemas for type safety.
- stores: Holds Zustand stores for lightweight state management.
- styles: Includes global styles, such as Tailwind CSS configurations.
Focus on the Features Folder
The features
folder organizes your app’s logic. Each feature (like push-notifications
) gets its own folder with this structure:
- hooks: Custom hooks for the feature, keeping logic reusable.
- components: UI components tied to the feature, making code easy to manage.
Example: User Profile Feature
Here’s how the user-profile
feature might look:
features/
user-profile/
hooks/
use-user-profile.ts # Fetches user data
components/
user-card.tsx # Displays user info
In use-user-profile.ts
:
import useSWR from "swr";
const fetcher = (url: string) => fetch(url).then((res) => res.json());
export function useUserProfile(userId: string) {
return useSWR(`/api/users/${userId}`, fetcher);
}
In user-card.tsx
:
import { useUserProfile } from "./hooks/use-user-profile";
export function UserCard({ userId }: { userId: string }) {
const { data: user, isLoading } = useUserProfile(userId);
if (isLoading) return <div>Loading...</div>;
if (error) return <>An error occured</>;
return <div>{user.name}</div>;
}
This keeps the feature self-contained and easy to update.
Differences Between Components Subfolders
The components
folder splits into common
and ui
to organize your UI code:
- common: For general components that don’t rely on specific styling libraries. Example: a
ProfileCard.tsx
that you style manually. - ui: For components built with shadcn/ui, ensuring consistent design. Example: a
Card.tsx
using shadcn/ui’s styles.
This split keeps your components organized and makes it clear which ones follow shadcn/ui’s design system.
Server Actions vs. API Routes with oRPC
Server Actions and API Routes both handle backend logic, but they work differently:
- Server Actions: Server-side functions for tasks like form submissions. They run on the server and are good for one-off operations. They don’t need a separate endpoint.
- API Routes: Next.js endpoints (e.g.,
/api/users
) for reusable APIs. They’re better for parallel requests and external access. With oRPC, they become type-safe.
Example with oRPC
Here’s how they differ in practice:
Server Action (in actions/update-user.action.ts
):
"use server";
export async function updateUser(userId: string, name: string) {
// Update user in database
return { success: true };
}
API Route with oRPC (in rpc/user.rpc.ts
):
import { procedure, router } from "@orpc/server";
export const userRouter = router({
getUser: procedure
.input(z.object({ id: z.string() }))
.query(async ({ input }) => {
return { id: input.id, name: "Alex" }; // Fetch from DB
}),
});
Usage (in a component):
import { userRouter } from "@/rpc/user.rpc";
import { updateUser } from "@/actions/update-user";
async function UserProfile({ userId }: { userId: string }) {
const user = await userRouter.getUser({ id: userId }); // API call with oRPC
const handleUpdate = async () => {
await updateUser(userId, "New Name"); // Server action
};
return <button onClick={handleUpdate}>Update Name</button>;
}
- Parallelism: API routes with oRPC can handle multiple requests at once, ideal for fetching data. Actions are sequential, better for single tasks like updates.
- Usage with oRPC: oRPC makes API routes type-safe, ensuring the frontend and backend match. oRPC is compatible with server actions.
Using the Data Folder
The data
folder holds static data and feature flags, so you don’t need a database for everything. Feature flags let you turn features on or off at build time.
Example: Feature Flags
In data/settings.ts
:
export const FEATURE_FLAGS = {
enablePushNotifications: false,
showNewDashboard: true,
};
In a component:
import { FEATURE_FLAGS } from "@/data/feature-flags";
function Dashboard() {
if (!FEATURE_FLAGS.showNewDashboard) {
return <div>Old Dashboard</div>;
}
return <div>New Dashboard</div>;
}
This lets you control features without changing code or using a database.
Providers vs. Zustand Stores
The providers
folder is for React context providers, but we recommend using Zustand stores for state management whenever possible.
TIP
Use providers
only when you have no other choice (e.g., for third-party libraries requiring context). For app state, use Zustand stores in the stores
folder—they’re lighter and easier to manage.
Example Zustand store (in stores/user.store.ts
):
"use client";
import { create } from "zustand";
export const useUserStore = create((set) => ({
user: null,
setUser: (user) => set({ user }),
}));
Using Schemas for Type-Safe APIs
The schemas
folder defines schemas for Drizzle ORM and API validation. With oRPC, you can use these schemas to ensure type safety and infer types for better app-wide type management.
Example: Type-Safe API
In schemas/user.schema.ts
:
import { z } from "zod";
export const UserSchema = z.object({
id: z.string(),
name: z.string(),
email: z.string().email(),
});
export type User = z.infer<typeof UserSchema>;
In rpc/user.rpc.ts
:
import { procedure, router } from "@orpc/server";
import { userSchema } from "@/schemas/user.schema";
export const userRouter = router({
getUser: procedure
.input(z.object({ id: z.string() }))
.output(userSchema)
.query(async ({ input }) => {
return { id: input.id, name: "Alex", email: "alex@example.com" };
}),
});
This ensures the API response matches the User
type, and your IDE will catch type errors early.
Naming Files
Follow these naming rules for clarity:
- For hooks: Use
use-hook.ts
(e.g.,use-user-profile.ts
). - For components: Use
feature-name.tsx
(e.g.,user-card.tsx
). - For stores: Use
your-store.store.ts
(e.g.,user.store.ts
). - For server actions: Use
your-action.action.ts
(e.g.,update-user.action.ts
). - For Zod schemas: Use
your-schema.schema.ts
(e.g.,user.schema.ts
) and prefer PascalCase for schema names (e.g.,UserSchema
overuserSchema
). - For RPC procedures: Use
your-procedure.rpc.ts
(e.g.,user.rpc.ts
). - For constants in the
data
folder: Use uppercase syntax likeFLAGS
orBASE_URL
(e.g.,export const FLAGS = { ... }
). - Otherwise, follow React naming rules (e.g., PascalCase for components).
Navigate Easily with Your IDE
It may sound cliché, but take advantage of your IDE’s search function to navigate between files quickly.
TIP
In VSCode, hold CMD
(or Ctrl
on Windows) and left-click a file name (e.g., use-user-profile.ts
) to open it quickly.
Why This Structure?
This setup makes building apps faster and easier:
- Find Things Quickly: Folders like
features
andcomponents
keep code organized. - Grow Without Mess: Add new features without cluttering the project.
- Work Together Better: Teams can focus on their own features without conflicts.
- Build Faster: Pre-organized folders mean less setup time.
Follow this structure to get the most out of Zap.ts—happy coding!