Build local-first apps with Evolu and Next.js for offline-first operation, end-to-end encryption, and cross-device sync. Covers branded types, reactive queries, CRUD, and mnemonic recovery. Use when avoiding backend infrastructure or prioritizing privacy.
Resources
1Install
npx skillscat add kaladivo/davenov-cc-collection/davenov-cc-expert-evolu-nextjs Install via the SkillsCat registry.
Enable TypeScript strict mode and exactOptionalPropertyTypes. Define schema with branded types using id() and NonEmptyString. Create Evolu instance and wrap app with EvoluProvider. Use useQuery for reactive data and create/update for mutations.
</quick_start>
- Set up Evolu in a new Next.js project
- Add features to existing Evolu app
- Debug sync, encryption, or hydration issues
- Understand Evolu concepts (schemas, queries, mutations)
Wait for response before proceeding.
- TypeScript 5.7 or later
- strict: true in tsconfig.json
- exactOptionalPropertyTypes: true in tsconfig.json
- Next.js 13+ with App Router
Packages: @evolu/common, @evolu/react, @evolu/react-web
Quick install:
npm install @evolu/common @evolu/react @evolu/react-webSee: [references/installation.md] for platform variants (React Native, Expo, Svelte) and full TypeScript configuration.
Key patterns:
id("TableName")- branded ID types prevent mixing IDs across tablesNonEmptyString+maxLength()- validated stringsnullOr()- optional fields (Evolu uses null, not undefined)SqliteBoolean- booleans stored as 0/1
See: [references/schema-definition.md] for complete examples and automatic system columns.
Key files:
lib/evolu.ts- Schema, instance creation, typed hooksapp/providers.tsx- EvoluProvider + Suspense wrapperapp/layout.tsx- Import providers
Important: All Evolu code must be in Client Components ("use client").
See: [references/instance-setup.md] for complete setup code.
Key patterns:
const todosQuery = evolu.createQuery((db) =>
db.selectFrom("todo")
.select(["id", "title", "isCompleted"])
.where("isDeleted", "is not", evolu.sqliteTrue)
.orderBy("createdAt", "desc")
);
const { rows } = useQuery(todosQuery);Always filter: .where("isDeleted", "is not", evolu.sqliteTrue)
See: [references/queries.md] for relationships, conditional queries, and useQueries for parallel loading.
Key rules:
- Create:
create(tableName, data)- ID auto-generated - Update:
update(tableName, { id, ...changes })- ID required - Delete: Soft delete with
isDeleted: evolu.sqliteTrue - Validate: Always use
.from()before mutations
See: [references/mutations.md] for complete CRUD examples and batch operations.
Key operations:
getMnemonic()- Get 12-word recovery phrase (handle securely!)restoreAppOwner(mnemonic)- Restore on new deviceresetAppOwner()- Delete all local data (irreversible)exportDatabase()- Export for backup
Security: Mnemonic = master key to ALL encrypted data. Never log in production.
See: [references/owner-management.md] for complete examples and security checklist.
Architecture:
- Server Components can render Client Components that use Evolu
- Evolu runs entirely client-side (local-first)
- Use Suspense for loading states
- Consider
next/dynamicwithssr: falseif hydration issues occur
See: [references/nextjs-integration.md] for complete setup and hydration troubleshooting.
See: [references/troubleshooting.md] for detailed solutions.
</error_handling>
- TypeScript compiles:
npx tsc --noEmit - Schema valid: All tables have branded ID types
- Validation used:
.from()before mutations - Soft delete filtered: Queries filter
isDeletedrows - Client Component: All Evolu code has
"use client" - Suspense wrapped: Components using
useQueryhave boundary - App builds:
npm run build