@knotpad/app 0.1.5 → 0.1.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/app/(app)/calendar/page.tsx +57 -0
- package/app/(app)/error.tsx +35 -0
- package/app/(app)/graph/page.tsx +32 -0
- package/app/(app)/guide/page.tsx +21 -0
- package/app/(app)/kanban/loading.tsx +24 -0
- package/app/(app)/kanban/page.tsx +59 -0
- package/app/(app)/layout.tsx +122 -0
- package/app/(app)/list/loading.tsx +21 -0
- package/app/(app)/list/page.tsx +137 -0
- package/app/(app)/loading.tsx +18 -0
- package/app/(app)/notes/[noteId]/page.tsx +84 -0
- package/app/(app)/notes/layout.tsx +30 -0
- package/app/(app)/notes/page.tsx +39 -0
- package/app/(app)/page.tsx +5 -0
- package/app/(app)/settings/agent-token/page.tsx +59 -0
- package/app/(app)/settings/backup/page.tsx +49 -0
- package/app/(app)/settings/billing/page.tsx +53 -0
- package/app/(app)/settings/calendar/page.tsx +41 -0
- package/app/(app)/settings/layout.test.tsx +39 -0
- package/app/(app)/settings/layout.tsx +71 -0
- package/app/(app)/settings/page.tsx +4 -0
- package/app/(app)/settings/security/page.tsx +43 -0
- package/app/(app)/settings/team/page.tsx +74 -0
- package/app/(app)/settings/workspace/page.tsx +27 -0
- package/app/(app)/tasks/[taskId]/page.tsx +79 -0
- package/app/(auth)/forgot-password/page.tsx +106 -0
- package/app/(auth)/guest/page.tsx +56 -0
- package/app/(auth)/layout.tsx +13 -0
- package/app/(auth)/login/page.tsx +14 -0
- package/app/(auth)/register/page.tsx +193 -0
- package/app/(auth)/reset-password/page.tsx +138 -0
- package/app/api/account/claim/route.tsx +135 -0
- package/app/api/admin/backfill-encryption/route.tsx +43 -0
- package/app/api/admin/license/route.tsx +42 -0
- package/app/api/auth/2fa/route.tsx +148 -0
- package/app/api/auth/[...nextauth]/route.tsx +3 -0
- package/app/api/auth/change-password/route.tsx +61 -0
- package/app/api/auth/check-2fa/route.tsx +19 -0
- package/app/api/auth/forgot-password/route.tsx +65 -0
- package/app/api/auth/reset-password/route.tsx +52 -0
- package/app/api/auth/verify-2fa/route.tsx +88 -0
- package/app/api/backup/download/db/route.ts +29 -0
- package/app/api/backup/download/notes/route.ts +25 -0
- package/app/api/backup/settings/route.ts +92 -0
- package/app/api/billing/checkout/route.tsx +81 -0
- package/app/api/billing/migrate/route.tsx +163 -0
- package/app/api/billing/portal/route.tsx +24 -0
- package/app/api/billing/setup-intent/route.tsx +55 -0
- package/app/api/billing/status/route.tsx +36 -0
- package/app/api/billing/subscribe/route.tsx +85 -0
- package/app/api/billing/webhook/route.tsx +199 -0
- package/app/api/calendar-feeds/[feedId]/route.tsx +67 -0
- package/app/api/calendar-feeds/[feedId]/sync/route.tsx +37 -0
- package/app/api/calendar-feeds/events/route.tsx +82 -0
- package/app/api/calendar-feeds/route.tsx +52 -0
- package/app/api/calendar-feeds/sync-all/route.tsx +34 -0
- package/app/api/cron/calendar-feeds/route.tsx +31 -0
- package/app/api/cron/stale-tasks/route.tsx +51 -0
- package/app/api/cron/sync/route.tsx +34 -0
- package/app/api/devices/[deviceId]/route.tsx +25 -0
- package/app/api/devices/route.tsx +41 -0
- package/app/api/export/route.tsx +40 -0
- package/app/api/feedback/route.tsx +54 -0
- package/app/api/folders/[folderId]/route.tsx +51 -0
- package/app/api/folders/route.tsx +37 -0
- package/app/api/graph/route.tsx +242 -0
- package/app/api/guest/route.tsx +58 -0
- package/app/api/health/route.tsx +10 -0
- package/app/api/holidays/countries/route.tsx +14 -0
- package/app/api/holidays/route.tsx +49 -0
- package/app/api/holidays/states/route.tsx +21 -0
- package/app/api/invites/[token]/route.tsx +131 -0
- package/app/api/invites/route.tsx +74 -0
- package/app/api/mcp/generate-token/route.tsx +55 -0
- package/app/api/mcp/revoke-token/[tokenId]/route.tsx +30 -0
- package/app/api/mcp/update-alias/[tokenId]/route.tsx +22 -0
- package/app/api/notes/[noteId]/export/route.tsx +45 -0
- package/app/api/notes/[noteId]/route.tsx +360 -0
- package/app/api/notes/route.tsx +112 -0
- package/app/api/notifications/route.tsx +44 -0
- package/app/api/register/route.tsx +67 -0
- package/app/api/restore/route.tsx +148 -0
- package/app/api/sync/conflicts/[conflictId]/route.tsx +134 -0
- package/app/api/sync/conflicts/route.tsx +48 -0
- package/app/api/sync/status/route.tsx +49 -0
- package/app/api/sync/trigger/route.tsx +15 -0
- package/app/api/tasks/[taskId]/detail/route.tsx +68 -0
- package/app/api/tasks/[taskId]/route.tsx +259 -0
- package/app/api/tasks/bulk/route.tsx +133 -0
- package/app/api/tasks/route.tsx +36 -0
- package/app/api/workspace/active/route.tsx +39 -0
- package/app/api/workspace/create-team/route.tsx +42 -0
- package/app/api/workspace/kanban-statuses/route.tsx +71 -0
- package/app/api/workspace/members/[memberId]/route.tsx +69 -0
- package/app/api/workspace/route.tsx +24 -0
- package/app/download/page.tsx +170 -0
- package/app/favicon.ico +0 -0
- package/app/generated/prisma/client.d.ts +1 -0
- package/app/generated/prisma/client.js +5 -0
- package/app/generated/prisma/default.d.ts +1 -0
- package/app/generated/prisma/default.js +5 -0
- package/app/generated/prisma/edge.d.ts +1 -0
- package/app/generated/prisma/edge.js +497 -0
- package/app/generated/prisma/index-browser.js +523 -0
- package/app/generated/prisma/index.d.ts +46376 -0
- package/app/generated/prisma/index.js +497 -0
- package/app/generated/prisma/package.json +144 -0
- package/app/generated/prisma/query_compiler_fast_bg.js +2 -0
- package/app/generated/prisma/query_compiler_fast_bg.wasm +0 -0
- package/app/generated/prisma/query_compiler_fast_bg.wasm-base64.js +2 -0
- package/app/generated/prisma/runtime/client.d.ts +3386 -0
- package/app/generated/prisma/runtime/client.js +86 -0
- package/app/generated/prisma/runtime/index-browser.d.ts +90 -0
- package/app/generated/prisma/runtime/index-browser.js +6 -0
- package/app/generated/prisma/runtime/wasm-compiler-edge.js +76 -0
- package/app/generated/prisma/schema.prisma +456 -0
- package/app/generated/prisma/wasm-edge-light-loader.mjs +5 -0
- package/app/generated/prisma/wasm-worker-loader.mjs +5 -0
- package/app/globals.css +54 -0
- package/app/invite/[token]/page.tsx +52 -0
- package/app/layout.tsx +90 -0
- package/app/mcp/route.tsx +430 -0
- package/app/opengraph-image.tsx +120 -0
- package/app/page.tsx +398 -0
- package/app/privacy/page.tsx +69 -0
- package/app/robots.tsx +25 -0
- package/app/sitemap.tsx +36 -0
- package/app/terms/page.tsx +69 -0
- package/app/upgrade/page.tsx +75 -0
- package/auth.config.ts +33 -0
- package/auth.ts +79 -0
- package/bin/brief.js +229 -0
- package/components/auth/login-form.tsx +302 -0
- package/components/auth/password-checklist.tsx +31 -0
- package/components/auth/password-input.tsx +36 -0
- package/components/auth/switch-account-button.test.tsx +22 -0
- package/components/auth/switch-account-button.tsx +19 -0
- package/components/auth/two-factor-input.tsx +116 -0
- package/components/billing/billing-dashboard.tsx +265 -0
- package/components/billing/card-form.tsx +210 -0
- package/components/billing/claim-account-form.tsx +99 -0
- package/components/branding/app-logo.test.tsx +20 -0
- package/components/branding/app-logo.tsx +25 -0
- package/components/calendar/calendar-agenda.tsx +150 -0
- package/components/calendar/calendar-drag.test.tsx +177 -0
- package/components/calendar/calendar-grid.tsx +357 -0
- package/components/calendar/calendar-hooks.test.tsx +27 -0
- package/components/calendar/calendar-hooks.ts +351 -0
- package/components/calendar/calendar-toolbar.test.tsx +68 -0
- package/components/calendar/calendar-toolbar.tsx +291 -0
- package/components/calendar/calendar-types.ts +148 -0
- package/components/calendar/calendar-view.test.tsx +295 -0
- package/components/calendar/calendar-view.tsx +307 -0
- package/components/calendar/day-detail-popover.tsx +174 -0
- package/components/calendar/task-chip.tsx +86 -0
- package/components/command/command-palette.test.tsx +33 -0
- package/components/command/command-palette.tsx +310 -0
- package/components/download-cta.tsx +87 -0
- package/components/feedback/feedback-popup.tsx +207 -0
- package/components/graph/graph-draw.ts +337 -0
- package/components/graph/graph-overlays.tsx +160 -0
- package/components/graph/graph-page.test.tsx +131 -0
- package/components/graph/graph-page.tsx +263 -0
- package/components/graph/graph-types.ts +47 -0
- package/components/graph/graph-view.tsx +322 -0
- package/components/guide/guide-view.tsx +522 -0
- package/components/kanban/kanban-board.test.tsx +128 -0
- package/components/kanban/kanban-board.tsx +361 -0
- package/components/kanban/kanban-card-menu.tsx +102 -0
- package/components/kanban/kanban-card.tsx +227 -0
- package/components/kanban/kanban-column.tsx +49 -0
- package/components/kanban/kanban-status-context.tsx +28 -0
- package/components/landing/calendar-sandbox.test.tsx +15 -0
- package/components/landing/calendar-sandbox.tsx +107 -0
- package/components/landing/graph-sandbox.test.tsx +27 -0
- package/components/landing/graph-sandbox.tsx +80 -0
- package/components/landing/kanban-sandbox.test.tsx +24 -0
- package/components/landing/kanban-sandbox.tsx +101 -0
- package/components/landing/landing-showcase.test.tsx +21 -0
- package/components/landing/landing-showcase.tsx +54 -0
- package/components/landing/list-sandbox.tsx +86 -0
- package/components/landing/mock-workspace.ts +168 -0
- package/components/landing/notes-sandbox.test.tsx +14 -0
- package/components/landing/notes-sandbox.tsx +88 -0
- package/components/layout/app-shell.tsx +83 -0
- package/components/layout/backup-scheduler.tsx +122 -0
- package/components/layout/bottom-nav.tsx +43 -0
- package/components/layout/icon-bar.test.tsx +29 -0
- package/components/layout/icon-bar.tsx +118 -0
- package/components/layout/mobile-top-bar.tsx +68 -0
- package/components/layout/notes-panel-folder.tsx +127 -0
- package/components/layout/notes-panel-note-item.tsx +140 -0
- package/components/layout/notes-panel-task-tab.tsx +63 -0
- package/components/layout/notes-panel-types.ts +44 -0
- package/components/layout/notes-panel.tsx +476 -0
- package/components/layout/notification-bell.tsx +251 -0
- package/components/layout/paywall-screen.tsx +41 -0
- package/components/layout/pro-banner.tsx +76 -0
- package/components/layout/sw-register.tsx +27 -0
- package/components/layout/workspace-switcher.tsx +90 -0
- package/components/notes/mobile-bottom-sheet.tsx +99 -0
- package/components/notes/note-editor-context-menu.tsx +47 -0
- package/components/notes/note-editor-dom.ts +33 -0
- package/components/notes/note-editor-dropdowns.tsx +484 -0
- package/components/notes/note-editor-hooks.ts +692 -0
- package/components/notes/note-editor-keyboard.ts +305 -0
- package/components/notes/note-editor-overlay.tsx +90 -0
- package/components/notes/note-editor.test.tsx +372 -0
- package/components/notes/note-editor.tsx +662 -0
- package/components/notes/note-preview-pane.tsx +156 -0
- package/components/notes/note-tabs.tsx +120 -0
- package/components/notes/note-types.tsx +157 -0
- package/components/settings/accept-invite.tsx +108 -0
- package/components/settings/agent-token-settings.tsx +369 -0
- package/components/settings/backup-restore-settings.test.tsx +25 -0
- package/components/settings/backup-restore-settings.tsx +327 -0
- package/components/settings/calendar-feeds-settings.tsx +489 -0
- package/components/settings/calendar-general-settings.tsx +174 -0
- package/components/settings/confirm-danger-action.test.tsx +215 -0
- package/components/settings/confirm-danger-action.tsx +65 -0
- package/components/settings/security-settings.tsx +252 -0
- package/components/settings/settings-guidance.test.tsx +98 -0
- package/components/settings/team-settings.tsx +319 -0
- package/components/settings/two-factor-auth.tsx +296 -0
- package/components/settings/workspace-settings-client.tsx +363 -0
- package/components/settings/workspace-settings-form.tsx +73 -0
- package/components/sync/conflict-viewer.tsx +247 -0
- package/components/sync/sync-indicator.tsx +171 -0
- package/components/tasks/snippet-thread.tsx +119 -0
- package/components/tasks/status-dot.tsx +47 -0
- package/components/tasks/task-badge.tsx +43 -0
- package/components/tasks/task-detail.test.tsx +187 -0
- package/components/tasks/task-detail.tsx +458 -0
- package/components/tasks/task-list-filters.test.tsx +75 -0
- package/components/tasks/task-list-filters.tsx +163 -0
- package/components/tasks/task-list-types.ts +20 -0
- package/components/tasks/task-list.test.tsx +175 -0
- package/components/tasks/task-list.tsx +481 -0
- package/components/tasks/task-row.tsx +85 -0
- package/components/tasks/task-table-row.tsx +259 -0
- package/components/ui/skeleton.tsx +3 -0
- package/components/ui/toast.test.tsx +42 -0
- package/components/ui/toast.tsx +70 -0
- package/instrumentation.tsx +23 -0
- package/lib/api-error.ts +50 -0
- package/lib/backup/backup-runner.test.ts +32 -0
- package/lib/backup/backup-runner.ts +19 -0
- package/lib/backup/backup-schedule.test.ts +23 -0
- package/lib/backup/backup-schedule.ts +55 -0
- package/lib/backup/backup-settings.test.ts +30 -0
- package/lib/backup/backup-settings.ts +27 -0
- package/lib/backup/export-notes-zip.test.ts +26 -0
- package/lib/backup/export-notes-zip.ts +82 -0
- package/lib/backup/export-workspace-backup.test.ts +17 -0
- package/lib/backup/export-workspace-backup.ts +77 -0
- package/lib/backup/restore-workspace-from-export.test.ts +18 -0
- package/lib/backup/restore-workspace-from-export.ts +183 -0
- package/lib/backup/types.ts +14 -0
- package/lib/brand-icons.ts +1 -0
- package/lib/calendar-feed-crypto.ts +38 -0
- package/lib/calendar-feed.ts +239 -0
- package/lib/client/online-status.ts +47 -0
- package/lib/conflict-resolver.test.ts +57 -0
- package/lib/conflict-resolver.ts +240 -0
- package/lib/db-init.ts +79 -0
- package/lib/email.ts +159 -0
- package/lib/encryption.test.ts +41 -0
- package/lib/encryption.ts +98 -0
- package/lib/extract-snippet.test.ts +123 -0
- package/lib/extract-snippet.ts +69 -0
- package/lib/kanban-status.ts +55 -0
- package/lib/license.ts +21 -0
- package/lib/limits.ts +31 -0
- package/lib/mcp-auth.test.ts +58 -0
- package/lib/mcp-auth.ts +65 -0
- package/lib/mcp-contract.test.ts +25 -0
- package/lib/mcp-contract.ts +210 -0
- package/lib/mcp-handler.ts +31 -0
- package/lib/mcp-url.test.ts +12 -0
- package/lib/mcp-url.ts +7 -0
- package/lib/mentions.test.ts +45 -0
- package/lib/mentions.ts +73 -0
- package/lib/note-crypto.ts +108 -0
- package/lib/note-sync.ts +201 -0
- package/lib/note-title.ts +93 -0
- package/lib/prisma.ts +193 -0
- package/lib/pro-flush.ts +292 -0
- package/lib/rate-limit.ts +57 -0
- package/lib/stripe.ts +38 -0
- package/lib/sync-worker.ts +388 -0
- package/lib/task-parser.test.ts +91 -0
- package/lib/task-parser.ts +81 -0
- package/lib/task-utils.ts +52 -0
- package/lib/use-is-electron.ts +19 -0
- package/lib/use-is-mobile.ts +22 -0
- package/lib/validation/calendar-feed.ts +31 -0
- package/lib/validation/note.ts +27 -0
- package/lib/validation/task.ts +26 -0
- package/lib/view-preferences.test.ts +54 -0
- package/lib/view-preferences.ts +28 -0
- package/lib/workspace.ts +66 -0
- package/next.config.ts +21 -0
- package/package.json +54 -3
- package/postcss.config.mjs +7 -0
- package/prisma/migrations/20260519021916_init/migration.sql +388 -0
- package/prisma/migrations/20260519061113_drop_sync_password/migration.sql +8 -0
- package/prisma/migrations/20260520065016_add_task_start_date/migration.sql +2 -0
- package/prisma/migrations/20260529010600_remove_encryption_fields/migration.sql +12 -0
- package/prisma/migrations/20260529020000_restore_encryption_salt/migration.sql +3 -0
- package/prisma/migrations/20260529030000_add_folders/migration.sql +17 -0
- package/prisma/migrations/20260605000000_deferred_fixes/migration.sql +31 -0
- package/prisma/migrations/20260605020806_add_pending_sync_to_note_and_task/migration.sql +5 -0
- package/prisma/migrations/20260605063634_add_stripe_webhook_event_sync_lock/migration.sql +14 -0
- package/prisma/migrations/20260605100000_add_prod_indexes/migration.sql +26 -0
- package/prisma/migrations/20260608081404_add_kanban_statuses/migration.sql +23 -0
- package/prisma/migrations/20260611032723_add_calendar_feeds/migration.sql +43 -0
- package/prisma/migrations/20260611040000_add_calendar_feed_color/migration.sql +2 -0
- package/prisma/migrations/20260611050000_add_task_priority/migration.sql +14 -0
- package/prisma/migrations/20260612060000_add_critical_priority/migration.sql +2 -0
- package/prisma/migrations/20260613090000_add_backup_settings/migration.sql +25 -0
- package/prisma/migrations/20260614160000_add_feedback/migration.sql +20 -0
- package/prisma/migrations/20260614210000_add_2fa/migration.sql +4 -0
- package/prisma/migrations/migration_lock.toml +3 -0
- package/prisma/schema.prisma +457 -0
- package/public/Logo_icon.svg +1 -0
- package/public/file.svg +1 -0
- package/public/globe.svg +1 -0
- package/public/icon_dark.svg +1 -0
- package/public/knotpad_icon.svg +1 -0
- package/public/knotpad_logo_full.svg +1 -0
- package/public/manifest.json +14 -0
- package/public/next.svg +1 -0
- package/public/sw.js +137 -0
- package/public/vercel.svg +1 -0
- package/public/window.svg +1 -0
- package/tsconfig.json +35 -0
- package/brief.js +0 -311
|
@@ -0,0 +1,456 @@
|
|
|
1
|
+
generator client {
|
|
2
|
+
provider = "prisma-client-js"
|
|
3
|
+
output = "../app/generated/prisma"
|
|
4
|
+
}
|
|
5
|
+
|
|
6
|
+
datasource db {
|
|
7
|
+
provider = "postgresql"
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
enum Role {
|
|
11
|
+
OWNER
|
|
12
|
+
ADMIN
|
|
13
|
+
MEMBER
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
enum TaskStatus {
|
|
17
|
+
OPEN
|
|
18
|
+
CLAIMED
|
|
19
|
+
IN_PROGRESS
|
|
20
|
+
REVIEW
|
|
21
|
+
DONE
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
enum AssigneeType {
|
|
25
|
+
HUMAN
|
|
26
|
+
AGENT
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
enum Priority {
|
|
30
|
+
CRITICAL
|
|
31
|
+
HIGH
|
|
32
|
+
MEDIUM
|
|
33
|
+
LOW
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
enum WorkspaceType {
|
|
37
|
+
PERSONAL
|
|
38
|
+
TEAM
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
enum PlanType {
|
|
42
|
+
FREE
|
|
43
|
+
PERSONAL_PRO
|
|
44
|
+
TEAM_PRO
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
enum LicenseType {
|
|
48
|
+
STANDARD
|
|
49
|
+
COMPLIMENTARY
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
enum BackupCadence {
|
|
53
|
+
DAILY
|
|
54
|
+
WEEKLY
|
|
55
|
+
MONTHLY
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
model User {
|
|
59
|
+
id String @id @default(cuid())
|
|
60
|
+
email String? @unique
|
|
61
|
+
name String?
|
|
62
|
+
passwordHash String?
|
|
63
|
+
image String?
|
|
64
|
+
isPro Boolean @default(false)
|
|
65
|
+
role Role @default(MEMBER)
|
|
66
|
+
createdAt DateTime @default(now())
|
|
67
|
+
workspaces WorkspaceMember[]
|
|
68
|
+
mcpTokens McpToken[]
|
|
69
|
+
tasksAssigned Task[] @relation("AssignedTo")
|
|
70
|
+
auditLogs AuditLog[]
|
|
71
|
+
deviceSessions DeviceSession[]
|
|
72
|
+
accounts Account[]
|
|
73
|
+
sessions Session[]
|
|
74
|
+
invitesSent InviteToken[] @relation("InvitedBy")
|
|
75
|
+
notifications Notification[]
|
|
76
|
+
passwordResetTokens PasswordResetToken[]
|
|
77
|
+
calendarFeeds CalendarFeed[]
|
|
78
|
+
feedback Feedback[]
|
|
79
|
+
twoFactorSecret String?
|
|
80
|
+
twoFactorEnabled Boolean @default(false)
|
|
81
|
+
twoFactorVerified Boolean @default(false)
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
model Workspace {
|
|
85
|
+
id String @id @default(cuid())
|
|
86
|
+
name String
|
|
87
|
+
slug String @unique
|
|
88
|
+
type WorkspaceType @default(PERSONAL)
|
|
89
|
+
planType PlanType @default(FREE)
|
|
90
|
+
licenseType LicenseType @default(STANDARD)
|
|
91
|
+
isCloud Boolean @default(false) // derived: planType != FREE || licenseType == COMPLIMENTARY
|
|
92
|
+
isPro Boolean @default(false) // derived: planType != FREE || licenseType == COMPLIMENTARY
|
|
93
|
+
seatCount Int @default(1) // mirrors Stripe subscription quantity
|
|
94
|
+
stripeId String? // Stripe customer ID
|
|
95
|
+
stripeSubId String? // Stripe subscription ID
|
|
96
|
+
encryptionSalt String? // per-workspace salt for at-rest note encryption (non-sensitive)
|
|
97
|
+
createdAt DateTime @default(now())
|
|
98
|
+
members WorkspaceMember[]
|
|
99
|
+
notes Note[]
|
|
100
|
+
folders Folder[]
|
|
101
|
+
tasks Task[]
|
|
102
|
+
mcpTokens McpToken[]
|
|
103
|
+
syncState SyncState?
|
|
104
|
+
conflictLogs ConflictLog[]
|
|
105
|
+
tombstones Tombstone[]
|
|
106
|
+
invites InviteToken[]
|
|
107
|
+
kanbanStatuses KanbanStatus[]
|
|
108
|
+
backupSettings WorkspaceBackupSettings?
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
model WorkspaceBackupSettings {
|
|
112
|
+
id String @id @default(cuid())
|
|
113
|
+
workspaceId String @unique
|
|
114
|
+
scheduleEnabled Boolean @default(false)
|
|
115
|
+
scheduleCadence BackupCadence @default(WEEKLY)
|
|
116
|
+
destinationPath String?
|
|
117
|
+
includeMarkdownZip Boolean @default(false)
|
|
118
|
+
lastBackupAt DateTime?
|
|
119
|
+
lastBackupStatus String @default("idle")
|
|
120
|
+
lastBackupError String?
|
|
121
|
+
createdAt DateTime @default(now())
|
|
122
|
+
updatedAt DateTime @updatedAt
|
|
123
|
+
workspace Workspace @relation(fields: [workspaceId], references: [id], onDelete: Cascade)
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
model KanbanStatus {
|
|
127
|
+
id String @id @default(cuid())
|
|
128
|
+
workspaceId String
|
|
129
|
+
key String // e.g. OPEN, CLAIMED, IN_PROGRESS, REVIEW, DONE
|
|
130
|
+
label String
|
|
131
|
+
color String // Tailwind border class, e.g. border-zinc-700
|
|
132
|
+
order Int @default(0)
|
|
133
|
+
isVisible Boolean @default(true)
|
|
134
|
+
createdAt DateTime @default(now())
|
|
135
|
+
updatedAt DateTime @updatedAt
|
|
136
|
+
workspace Workspace @relation(fields: [workspaceId], references: [id], onDelete: Cascade)
|
|
137
|
+
|
|
138
|
+
@@unique([workspaceId, key])
|
|
139
|
+
@@index([workspaceId])
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
model WorkspaceMember {
|
|
143
|
+
id String @id @default(cuid())
|
|
144
|
+
userId String
|
|
145
|
+
workspaceId String
|
|
146
|
+
role Role @default(MEMBER)
|
|
147
|
+
mcpAlias String?
|
|
148
|
+
joinedAt DateTime @default(now())
|
|
149
|
+
revokedAt DateTime?
|
|
150
|
+
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
|
151
|
+
workspace Workspace @relation(fields: [workspaceId], references: [id], onDelete: Cascade)
|
|
152
|
+
|
|
153
|
+
@@unique([userId, workspaceId])
|
|
154
|
+
@@index([userId])
|
|
155
|
+
@@index([workspaceId])
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
model Note {
|
|
159
|
+
id String @id @default(cuid())
|
|
160
|
+
title String
|
|
161
|
+
content String @db.Text
|
|
162
|
+
workspaceId String
|
|
163
|
+
folderId String?
|
|
164
|
+
isLocked Boolean @default(false)
|
|
165
|
+
cloudOnly Boolean @default(false)
|
|
166
|
+
pendingSync Boolean @default(false)
|
|
167
|
+
version Int @default(0)
|
|
168
|
+
deviceId String?
|
|
169
|
+
createdAt DateTime @default(now())
|
|
170
|
+
updatedAt DateTime @updatedAt
|
|
171
|
+
workspace Workspace @relation(fields: [workspaceId], references: [id], onDelete: Cascade)
|
|
172
|
+
folder Folder? @relation(fields: [folderId], references: [id], onDelete: SetNull)
|
|
173
|
+
tasks Task[]
|
|
174
|
+
references TaskReference[]
|
|
175
|
+
|
|
176
|
+
@@index([workspaceId])
|
|
177
|
+
@@index([workspaceId, updatedAt])
|
|
178
|
+
@@index([folderId])
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
model Folder {
|
|
182
|
+
id String @id @default(cuid())
|
|
183
|
+
name String
|
|
184
|
+
workspaceId String
|
|
185
|
+
createdAt DateTime @default(now())
|
|
186
|
+
workspace Workspace @relation(fields: [workspaceId], references: [id], onDelete: Cascade)
|
|
187
|
+
notes Note[]
|
|
188
|
+
|
|
189
|
+
@@index([workspaceId])
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
model Task {
|
|
193
|
+
id String @id @default(cuid())
|
|
194
|
+
title String
|
|
195
|
+
status TaskStatus @default(OPEN)
|
|
196
|
+
priority Priority @default(MEDIUM)
|
|
197
|
+
noteId String
|
|
198
|
+
workspaceId String
|
|
199
|
+
assigneeId String?
|
|
200
|
+
assigneeType AssigneeType @default(HUMAN)
|
|
201
|
+
claimedBy String?
|
|
202
|
+
claimedByAlias String?
|
|
203
|
+
claimedAt DateTime?
|
|
204
|
+
lastHeartbeat DateTime?
|
|
205
|
+
fileRefs String[]
|
|
206
|
+
startDate DateTime?
|
|
207
|
+
dueDate DateTime?
|
|
208
|
+
syncLocal Boolean @default(true) // false = cloud-only, won't be pulled to local
|
|
209
|
+
pendingSync Boolean @default(false)
|
|
210
|
+
version Int @default(0)
|
|
211
|
+
deviceId String?
|
|
212
|
+
createdAt DateTime @default(now())
|
|
213
|
+
updatedAt DateTime @updatedAt
|
|
214
|
+
note Note @relation(fields: [noteId], references: [id], onDelete: Cascade)
|
|
215
|
+
workspace Workspace @relation(fields: [workspaceId], references: [id], onDelete: Cascade)
|
|
216
|
+
assignee User? @relation("AssignedTo", fields: [assigneeId], references: [id])
|
|
217
|
+
auditLogs AuditLog[]
|
|
218
|
+
references TaskReference[]
|
|
219
|
+
|
|
220
|
+
@@index([workspaceId])
|
|
221
|
+
@@index([noteId])
|
|
222
|
+
@@index([workspaceId, status])
|
|
223
|
+
@@index([workspaceId, assigneeType, status])
|
|
224
|
+
@@index([workspaceId, priority])
|
|
225
|
+
@@index([claimedBy])
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
model TaskReference {
|
|
229
|
+
id String @id @default(cuid())
|
|
230
|
+
taskId String
|
|
231
|
+
noteId String
|
|
232
|
+
snippet String @db.Text
|
|
233
|
+
task Task @relation(fields: [taskId], references: [id], onDelete: Cascade)
|
|
234
|
+
note Note @relation(fields: [noteId], references: [id], onDelete: Cascade)
|
|
235
|
+
|
|
236
|
+
@@index([taskId])
|
|
237
|
+
@@index([noteId])
|
|
238
|
+
}
|
|
239
|
+
|
|
240
|
+
model McpToken {
|
|
241
|
+
id String @id @default(cuid())
|
|
242
|
+
token String @unique
|
|
243
|
+
userId String
|
|
244
|
+
workspaceId String
|
|
245
|
+
alias String?
|
|
246
|
+
lastUsed DateTime?
|
|
247
|
+
revokedAt DateTime?
|
|
248
|
+
createdAt DateTime @default(now())
|
|
249
|
+
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
|
250
|
+
workspace Workspace @relation(fields: [workspaceId], references: [id], onDelete: Cascade)
|
|
251
|
+
|
|
252
|
+
@@index([userId])
|
|
253
|
+
@@index([workspaceId])
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
model DeviceSession {
|
|
257
|
+
id String @id @default(cuid())
|
|
258
|
+
userId String
|
|
259
|
+
deviceName String
|
|
260
|
+
deviceId String @unique
|
|
261
|
+
lastActive DateTime @default(now())
|
|
262
|
+
revokedAt DateTime?
|
|
263
|
+
createdAt DateTime @default(now())
|
|
264
|
+
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
model AuditLog {
|
|
268
|
+
id String @id @default(cuid())
|
|
269
|
+
taskId String
|
|
270
|
+
userId String?
|
|
271
|
+
action String
|
|
272
|
+
detail String?
|
|
273
|
+
deviceId String?
|
|
274
|
+
createdAt DateTime @default(now())
|
|
275
|
+
task Task @relation(fields: [taskId], references: [id], onDelete: Cascade)
|
|
276
|
+
user User? @relation(fields: [userId], references: [id])
|
|
277
|
+
|
|
278
|
+
@@index([taskId])
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
model SyncState {
|
|
282
|
+
id String @id @default(cuid())
|
|
283
|
+
workspaceId String @unique
|
|
284
|
+
lastSyncedAt DateTime?
|
|
285
|
+
localVersion Int @default(0)
|
|
286
|
+
cloudVersion Int @default(0)
|
|
287
|
+
noteSnapshots String? @db.Text
|
|
288
|
+
snapshotVersion Int @default(0)
|
|
289
|
+
/// Distributed sync lock. Set to now() when a flush starts; cleared when done.
|
|
290
|
+
/// Any instance that finds a lock < 120s old will skip the flush to avoid races.
|
|
291
|
+
syncLockedAt DateTime?
|
|
292
|
+
workspace Workspace @relation(fields: [workspaceId], references: [id], onDelete: Cascade)
|
|
293
|
+
}
|
|
294
|
+
|
|
295
|
+
model ConflictLog {
|
|
296
|
+
id String @id @default(cuid())
|
|
297
|
+
workspaceId String
|
|
298
|
+
entityType String
|
|
299
|
+
entityId String
|
|
300
|
+
localValue String @db.Text
|
|
301
|
+
cloudValue String @db.Text
|
|
302
|
+
resolvedBy String?
|
|
303
|
+
resolvedAt DateTime?
|
|
304
|
+
createdAt DateTime @default(now())
|
|
305
|
+
workspace Workspace @relation(fields: [workspaceId], references: [id], onDelete: Cascade)
|
|
306
|
+
|
|
307
|
+
@@unique([workspaceId, entityType, entityId])
|
|
308
|
+
@@index([workspaceId, resolvedAt])
|
|
309
|
+
}
|
|
310
|
+
|
|
311
|
+
model Tombstone {
|
|
312
|
+
id String @id @default(cuid())
|
|
313
|
+
workspaceId String
|
|
314
|
+
entityType String
|
|
315
|
+
entityId String
|
|
316
|
+
deletedAt DateTime @default(now())
|
|
317
|
+
workspace Workspace @relation(fields: [workspaceId], references: [id], onDelete: Cascade)
|
|
318
|
+
|
|
319
|
+
@@index([workspaceId, entityType, entityId])
|
|
320
|
+
}
|
|
321
|
+
|
|
322
|
+
model Account {
|
|
323
|
+
id String @id @default(cuid())
|
|
324
|
+
userId String
|
|
325
|
+
type String
|
|
326
|
+
provider String
|
|
327
|
+
providerAccountId String
|
|
328
|
+
refresh_token String? @db.Text
|
|
329
|
+
access_token String? @db.Text
|
|
330
|
+
expires_at Int?
|
|
331
|
+
token_type String?
|
|
332
|
+
scope String?
|
|
333
|
+
id_token String? @db.Text
|
|
334
|
+
session_state String?
|
|
335
|
+
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
|
336
|
+
|
|
337
|
+
@@unique([provider, providerAccountId])
|
|
338
|
+
}
|
|
339
|
+
|
|
340
|
+
model Session {
|
|
341
|
+
id String @id @default(cuid())
|
|
342
|
+
sessionToken String @unique
|
|
343
|
+
userId String
|
|
344
|
+
expires DateTime
|
|
345
|
+
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
|
346
|
+
}
|
|
347
|
+
|
|
348
|
+
model VerificationToken {
|
|
349
|
+
identifier String
|
|
350
|
+
token String @unique
|
|
351
|
+
expires DateTime
|
|
352
|
+
|
|
353
|
+
@@unique([identifier, token])
|
|
354
|
+
}
|
|
355
|
+
|
|
356
|
+
model PasswordResetToken {
|
|
357
|
+
id String @id @default(cuid())
|
|
358
|
+
token String @unique @default(cuid())
|
|
359
|
+
userId String
|
|
360
|
+
expiresAt DateTime
|
|
361
|
+
usedAt DateTime?
|
|
362
|
+
createdAt DateTime @default(now())
|
|
363
|
+
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
|
364
|
+
}
|
|
365
|
+
|
|
366
|
+
model InviteToken {
|
|
367
|
+
id String @id @default(cuid())
|
|
368
|
+
token String @unique @default(cuid())
|
|
369
|
+
email String
|
|
370
|
+
workspaceId String
|
|
371
|
+
role Role @default(MEMBER)
|
|
372
|
+
invitedById String
|
|
373
|
+
expiresAt DateTime
|
|
374
|
+
acceptedAt DateTime?
|
|
375
|
+
createdAt DateTime @default(now())
|
|
376
|
+
workspace Workspace @relation(fields: [workspaceId], references: [id], onDelete: Cascade)
|
|
377
|
+
invitedBy User @relation("InvitedBy", fields: [invitedById], references: [id])
|
|
378
|
+
}
|
|
379
|
+
|
|
380
|
+
model Notification {
|
|
381
|
+
id String @id @default(cuid())
|
|
382
|
+
userId String
|
|
383
|
+
type String
|
|
384
|
+
title String
|
|
385
|
+
body String?
|
|
386
|
+
read Boolean @default(false)
|
|
387
|
+
taskId String?
|
|
388
|
+
noteId String?
|
|
389
|
+
createdAt DateTime @default(now())
|
|
390
|
+
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
|
391
|
+
|
|
392
|
+
@@index([userId])
|
|
393
|
+
@@index([userId, read])
|
|
394
|
+
}
|
|
395
|
+
|
|
396
|
+
/// A user's personal read-only calendar subscription (Google/Apple "secret
|
|
397
|
+
/// address in iCal format"). The URL is a credential — anyone with it can read
|
|
398
|
+
/// the calendar — so it's encrypted at rest with a per-row salt.
|
|
399
|
+
model CalendarFeed {
|
|
400
|
+
id String @id @default(cuid())
|
|
401
|
+
userId String
|
|
402
|
+
label String
|
|
403
|
+
encryptedUrl String @db.Text
|
|
404
|
+
urlSalt String
|
|
405
|
+
color String @default("sky")
|
|
406
|
+
enabled Boolean @default(true)
|
|
407
|
+
lastFetchedAt DateTime?
|
|
408
|
+
lastError String?
|
|
409
|
+
createdAt DateTime @default(now())
|
|
410
|
+
updatedAt DateTime @updatedAt
|
|
411
|
+
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
|
412
|
+
events CalendarFeedEvent[]
|
|
413
|
+
|
|
414
|
+
@@index([userId])
|
|
415
|
+
}
|
|
416
|
+
|
|
417
|
+
/// Cached events parsed from a CalendarFeed's .ics source, including expanded
|
|
418
|
+
/// recurring-event occurrences within the sync window.
|
|
419
|
+
model CalendarFeedEvent {
|
|
420
|
+
id String @id @default(cuid())
|
|
421
|
+
feedId String
|
|
422
|
+
uid String
|
|
423
|
+
title String
|
|
424
|
+
start DateTime
|
|
425
|
+
end DateTime
|
|
426
|
+
allDay Boolean @default(false)
|
|
427
|
+
feed CalendarFeed @relation(fields: [feedId], references: [id], onDelete: Cascade)
|
|
428
|
+
|
|
429
|
+
@@unique([feedId, uid])
|
|
430
|
+
@@index([feedId, start])
|
|
431
|
+
}
|
|
432
|
+
|
|
433
|
+
/// Idempotency log for Stripe webhook events.
|
|
434
|
+
/// Stripe retries events on non-2xx responses for up to 3 days; storing the
|
|
435
|
+
/// event ID here ensures each event is processed exactly once.
|
|
436
|
+
model StripeWebhookEvent {
|
|
437
|
+
id String @id // Stripe event ID (evt_...)
|
|
438
|
+
type String // e.g. "checkout.session.completed"
|
|
439
|
+
processedAt DateTime @default(now())
|
|
440
|
+
|
|
441
|
+
@@index([processedAt])
|
|
442
|
+
}
|
|
443
|
+
|
|
444
|
+
/// User feedback and issue reports
|
|
445
|
+
model Feedback {
|
|
446
|
+
id String @id @default(cuid())
|
|
447
|
+
userId String
|
|
448
|
+
type String // "feedback", "bug", "feature_request"
|
|
449
|
+
subject String
|
|
450
|
+
message String @db.Text
|
|
451
|
+
createdAt DateTime @default(now())
|
|
452
|
+
user User @relation(fields: [userId], references: [id], onDelete: Cascade)
|
|
453
|
+
|
|
454
|
+
@@index([userId])
|
|
455
|
+
@@index([createdAt])
|
|
456
|
+
}
|
package/app/globals.css
ADDED
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
@import "tailwindcss";
|
|
2
|
+
|
|
3
|
+
:root {
|
|
4
|
+
--background: #09090b;
|
|
5
|
+
--foreground: #fafafa;
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
@theme inline {
|
|
9
|
+
--color-background: var(--background);
|
|
10
|
+
--color-foreground: var(--foreground);
|
|
11
|
+
--font-sans: var(--font-geist-sans);
|
|
12
|
+
--font-mono: var(--font-geist-mono);
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
body {
|
|
16
|
+
background: var(--background);
|
|
17
|
+
color: var(--foreground);
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
* {
|
|
21
|
+
scrollbar-width: thin;
|
|
22
|
+
scrollbar-color: #3f3f46 transparent;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
/* Shared accessible focus ring. Apply to interactive inputs/buttons that
|
|
26
|
+
otherwise strip their outline. Uses :focus-visible so it only shows for
|
|
27
|
+
keyboard users, not on mouse click. */
|
|
28
|
+
/* Prevent outer scroll on mobile when editor is focused */
|
|
29
|
+
@media (max-width: 767.98px) {
|
|
30
|
+
html, body {
|
|
31
|
+
overflow: hidden;
|
|
32
|
+
overscroll-behavior: none;
|
|
33
|
+
position: fixed;
|
|
34
|
+
width: 100%;
|
|
35
|
+
height: 100%;
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
/* Kanban cards: prevent iOS Safari touch interference during drag */
|
|
40
|
+
[data-kanban-card] {
|
|
41
|
+
touch-action: none;
|
|
42
|
+
-webkit-touch-callout: none;
|
|
43
|
+
-webkit-user-select: none;
|
|
44
|
+
user-select: none;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
@utility ring-focus {
|
|
48
|
+
&:focus-visible {
|
|
49
|
+
outline: 2px solid transparent;
|
|
50
|
+
outline-offset: 2px;
|
|
51
|
+
box-shadow: 0 0 0 2px rgba(59, 130, 246, 0.6);
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
import { auth } from "@/auth";
|
|
2
|
+
import { redirect } from "next/navigation";
|
|
3
|
+
import { prisma } from "@/lib/prisma";
|
|
4
|
+
import { AcceptInvite } from "@/components/settings/accept-invite";
|
|
5
|
+
|
|
6
|
+
export default async function InvitePage({
|
|
7
|
+
params,
|
|
8
|
+
}: {
|
|
9
|
+
params: Promise<{ token: string }>;
|
|
10
|
+
}) {
|
|
11
|
+
const { token } = await params;
|
|
12
|
+
const session = await auth();
|
|
13
|
+
|
|
14
|
+
const invite = await prisma.inviteToken.findUnique({
|
|
15
|
+
where: { token },
|
|
16
|
+
include: {
|
|
17
|
+
workspace: { select: { name: true } },
|
|
18
|
+
invitedBy: { select: { name: true, email: true } },
|
|
19
|
+
},
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
if (!invite || invite.acceptedAt || invite.expiresAt < new Date()) {
|
|
23
|
+
return (
|
|
24
|
+
<div className="flex min-h-full items-center justify-center">
|
|
25
|
+
<div className="text-center space-y-2">
|
|
26
|
+
<h1 className="text-xl font-semibold text-zinc-300">Invite not valid</h1>
|
|
27
|
+
<p className="text-sm text-zinc-500">
|
|
28
|
+
This invite link has expired or already been used.
|
|
29
|
+
</p>
|
|
30
|
+
</div>
|
|
31
|
+
</div>
|
|
32
|
+
);
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
// If the current user matches the invite email, auto-accept is possible
|
|
36
|
+
const isLoggedInAsInvitee =
|
|
37
|
+
session?.user?.email?.toLowerCase() === invite.email.toLowerCase();
|
|
38
|
+
|
|
39
|
+
return (
|
|
40
|
+
<div className="flex min-h-full items-center justify-center">
|
|
41
|
+
<AcceptInvite
|
|
42
|
+
token={token}
|
|
43
|
+
email={invite.email}
|
|
44
|
+
workspaceName={invite.workspace.name}
|
|
45
|
+
invitedBy={invite.invitedBy.name ?? invite.invitedBy.email ?? "Someone"}
|
|
46
|
+
role={invite.role}
|
|
47
|
+
isLoggedIn={!!session}
|
|
48
|
+
isLoggedInAsInvitee={isLoggedInAsInvitee}
|
|
49
|
+
/>
|
|
50
|
+
</div>
|
|
51
|
+
);
|
|
52
|
+
}
|
package/app/layout.tsx
ADDED
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
import type { Metadata, Viewport } from "next";
|
|
2
|
+
import { Geist, Geist_Mono } from "next/font/google";
|
|
3
|
+
import { WEB_ICON_PATH } from "@/lib/brand-icons";
|
|
4
|
+
import { SwRegister } from "@/components/layout/sw-register";
|
|
5
|
+
import { ToastProvider } from "@/components/ui/toast";
|
|
6
|
+
import "./globals.css";
|
|
7
|
+
|
|
8
|
+
const geistSans = Geist({
|
|
9
|
+
variable: "--font-geist-sans",
|
|
10
|
+
subsets: ["latin"],
|
|
11
|
+
});
|
|
12
|
+
|
|
13
|
+
const geistMono = Geist_Mono({
|
|
14
|
+
variable: "--font-geist-mono",
|
|
15
|
+
subsets: ["latin"],
|
|
16
|
+
});
|
|
17
|
+
|
|
18
|
+
const SITE_URL = "https://knotpad.app";
|
|
19
|
+
const SITE_DESCRIPTION =
|
|
20
|
+
"Note-first project management with AI agent task routing. Write the note — the tasks come with it.";
|
|
21
|
+
|
|
22
|
+
export const metadata: Metadata = {
|
|
23
|
+
metadataBase: new URL(SITE_URL),
|
|
24
|
+
title: {
|
|
25
|
+
default: "Knotpad — Write the note. The tasks come with it.",
|
|
26
|
+
template: "%s | Knotpad",
|
|
27
|
+
},
|
|
28
|
+
description: SITE_DESCRIPTION,
|
|
29
|
+
keywords: [
|
|
30
|
+
"project management",
|
|
31
|
+
"note-taking",
|
|
32
|
+
"kanban",
|
|
33
|
+
"AI tasks",
|
|
34
|
+
"MCP",
|
|
35
|
+
"local-first",
|
|
36
|
+
"task management",
|
|
37
|
+
"team collaboration",
|
|
38
|
+
],
|
|
39
|
+
authors: [{ name: "Knotpad", url: SITE_URL }],
|
|
40
|
+
creator: "Knotpad",
|
|
41
|
+
publisher: "Nexstrive Services",
|
|
42
|
+
category: "productivity",
|
|
43
|
+
robots: "index, follow",
|
|
44
|
+
manifest: "/manifest.json",
|
|
45
|
+
icons: {
|
|
46
|
+
icon: WEB_ICON_PATH,
|
|
47
|
+
shortcut: WEB_ICON_PATH,
|
|
48
|
+
},
|
|
49
|
+
alternates: {
|
|
50
|
+
canonical: SITE_URL,
|
|
51
|
+
},
|
|
52
|
+
openGraph: {
|
|
53
|
+
title: "Knotpad — Write the note. The tasks come with it.",
|
|
54
|
+
description: SITE_DESCRIPTION,
|
|
55
|
+
url: SITE_URL,
|
|
56
|
+
siteName: "Knotpad",
|
|
57
|
+
locale: "en_US",
|
|
58
|
+
type: "website",
|
|
59
|
+
},
|
|
60
|
+
twitter: {
|
|
61
|
+
card: "summary_large_image",
|
|
62
|
+
title: "Knotpad — Write the note. The tasks come with it.",
|
|
63
|
+
description: SITE_DESCRIPTION,
|
|
64
|
+
},
|
|
65
|
+
};
|
|
66
|
+
|
|
67
|
+
export const viewport: Viewport = {
|
|
68
|
+
width: "device-width",
|
|
69
|
+
initialScale: 1,
|
|
70
|
+
maximumScale: 1,
|
|
71
|
+
themeColor: "#09090b",
|
|
72
|
+
colorScheme: "dark",
|
|
73
|
+
viewportFit: "cover",
|
|
74
|
+
interactiveWidget: "resizes-content",
|
|
75
|
+
};
|
|
76
|
+
|
|
77
|
+
export default function RootLayout({
|
|
78
|
+
children,
|
|
79
|
+
}: Readonly<{
|
|
80
|
+
children: React.ReactNode;
|
|
81
|
+
}>) {
|
|
82
|
+
return (
|
|
83
|
+
<html lang="en" className={`${geistSans.variable} ${geistMono.variable} h-full`}>
|
|
84
|
+
<body className="h-full bg-zinc-950 text-zinc-100 antialiased">
|
|
85
|
+
<SwRegister />
|
|
86
|
+
<ToastProvider>{children}</ToastProvider>
|
|
87
|
+
</body>
|
|
88
|
+
</html>
|
|
89
|
+
);
|
|
90
|
+
}
|