name: app-specific-patterns description: GROWI main application (apps/app) specific patterns for Next.js, Jotai, SWR, and testing. Auto-invoked when working in apps/app.
For general testing patterns, see the global testing-patterns-with-vitest skill.
Pages must use .page.tsx suffix:
pages/
├── _app.page.tsx # App wrapper
├── [[...path]]/index.page.tsx # Catch-all wiki pages
└── admin/index.page.tsx # Admin pages
// pages/admin/index.page.tsx
import type { NextPageWithLayout } from '~/interfaces/next-page';
const AdminPage: NextPageWithLayout = () => <AdminDashboard />;
AdminPage.getLayout = (page) => <AdminLayout>{page}</AdminLayout>;
export default AdminPage;
src/states/
├── ui/
│ ├── sidebar/ # Multi-file feature
│ ├── device.ts # Single-file feature
│ └── modal/ # 1 modal = 1 file
│ ├── page-create.ts
│ └── page-delete.ts
├── page/ # Page data state
├── server-configurations/
└── context.ts
features/{name}/client/states/ # Feature-scoped atoms
| Category | Location |
|---|---|
| UI state | states/ui/ |
| Modal state | states/ui/modal/ (1 file per modal) |
| Page data | states/page/ |
| Feature-specific | features/{name}/client/states/ |
import { atom } from 'jotai';
export const currentPageAtom = atom<Page | null>(null);
// Derived (read-only)
export const currentPagePathAtom = atom((get) => {
return get(currentPageAtom)?.path ?? null;
});
src/stores-universal/
├── pages.ts # Page hooks
├── users.ts # User hooks
└── admin/settings.ts
import useSWR from 'swr';
import useSWRImmutable from 'swr/immutable';
// Auto-revalidation
export const usePageList = () => useSWR<Page[]>('/api/v3/pages', fetcher);
// No auto-revalidation (static data)
export const usePageById = (id: string | null) =>
useSWRImmutable<Page>(id ? `/api/v3/pages/${id}` : null, fetcher);
import { mockDeep } from 'vitest-mock-extended';
import type { NextRouter } from 'next/router';
const createMockRouter = (overrides = {}) => {
const mock = mockDeep<NextRouter>();
mock.pathname = '/test';
mock.push.mockResolvedValue(true);
return Object.assign(mock, overrides);
};
vi.mock('next/router', () => ({
useRouter: () => createMockRouter(),
}));
import { Provider } from 'jotai';
import { useHydrateAtoms } from 'jotai/utils';
const HydrateAtoms = ({ initialValues, children }) => {
useHydrateAtoms(initialValues);
return children;
};
const renderWithJotai = (ui, initialValues = []) => render(
<Provider>
<HydrateAtoms initialValues={initialValues}>{ui}</HydrateAtoms>
</Provider>
);
// Usage
renderWithJotai(<PageHeader />, [[currentPageAtom, mockPage]]);
import { SWRConfig } from 'swr';
const wrapper = ({ children }) => (
<SWRConfig value={{ dedupingInterval: 0, provider: () => new Map() }}>
{children}
</SWRConfig>
);
const { result } = renderHook(() => usePageById('123'), { wrapper });
Always use ~/ for imports:
import { PageService } from '~/server/services/PageService';
import { currentPageAtom } from '~/states/page/page-atoms';
.page.tsx suffix, getLayout for layoutsstates/ global, features/*/client/states/ feature-scopedstores-universal/, null key for conditional fetch~/ path alias