@knotpad/app 0.1.0

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.
Files changed (342) hide show
  1. package/README.md +167 -0
  2. package/app/(app)/calendar/page.tsx +57 -0
  3. package/app/(app)/error.tsx +35 -0
  4. package/app/(app)/graph/page.tsx +32 -0
  5. package/app/(app)/guide/page.tsx +21 -0
  6. package/app/(app)/kanban/loading.tsx +24 -0
  7. package/app/(app)/kanban/page.tsx +59 -0
  8. package/app/(app)/layout.tsx +122 -0
  9. package/app/(app)/list/loading.tsx +21 -0
  10. package/app/(app)/list/page.tsx +137 -0
  11. package/app/(app)/loading.tsx +18 -0
  12. package/app/(app)/notes/[noteId]/page.tsx +84 -0
  13. package/app/(app)/notes/layout.tsx +30 -0
  14. package/app/(app)/notes/page.tsx +39 -0
  15. package/app/(app)/page.tsx +5 -0
  16. package/app/(app)/settings/agent-token/page.tsx +59 -0
  17. package/app/(app)/settings/backup/page.tsx +49 -0
  18. package/app/(app)/settings/billing/page.tsx +53 -0
  19. package/app/(app)/settings/calendar/page.tsx +41 -0
  20. package/app/(app)/settings/layout.test.tsx +39 -0
  21. package/app/(app)/settings/layout.tsx +71 -0
  22. package/app/(app)/settings/page.tsx +4 -0
  23. package/app/(app)/settings/security/page.tsx +43 -0
  24. package/app/(app)/settings/team/page.tsx +74 -0
  25. package/app/(app)/settings/workspace/page.tsx +27 -0
  26. package/app/(app)/tasks/[taskId]/page.tsx +79 -0
  27. package/app/(auth)/forgot-password/page.tsx +106 -0
  28. package/app/(auth)/guest/page.tsx +56 -0
  29. package/app/(auth)/layout.tsx +13 -0
  30. package/app/(auth)/login/page.tsx +14 -0
  31. package/app/(auth)/register/page.tsx +193 -0
  32. package/app/(auth)/reset-password/page.tsx +138 -0
  33. package/app/api/account/claim/route.tsx +135 -0
  34. package/app/api/admin/backfill-encryption/route.tsx +43 -0
  35. package/app/api/admin/license/route.tsx +42 -0
  36. package/app/api/auth/2fa/route.tsx +148 -0
  37. package/app/api/auth/[...nextauth]/route.tsx +3 -0
  38. package/app/api/auth/change-password/route.tsx +61 -0
  39. package/app/api/auth/check-2fa/route.tsx +19 -0
  40. package/app/api/auth/forgot-password/route.tsx +65 -0
  41. package/app/api/auth/reset-password/route.tsx +52 -0
  42. package/app/api/auth/verify-2fa/route.tsx +88 -0
  43. package/app/api/backup/download/db/route.ts +29 -0
  44. package/app/api/backup/download/notes/route.ts +25 -0
  45. package/app/api/backup/settings/route.ts +92 -0
  46. package/app/api/billing/checkout/route.tsx +81 -0
  47. package/app/api/billing/migrate/route.tsx +163 -0
  48. package/app/api/billing/portal/route.tsx +24 -0
  49. package/app/api/billing/setup-intent/route.tsx +55 -0
  50. package/app/api/billing/status/route.tsx +36 -0
  51. package/app/api/billing/subscribe/route.tsx +85 -0
  52. package/app/api/billing/webhook/route.tsx +199 -0
  53. package/app/api/calendar-feeds/[feedId]/route.tsx +67 -0
  54. package/app/api/calendar-feeds/[feedId]/sync/route.tsx +37 -0
  55. package/app/api/calendar-feeds/events/route.tsx +82 -0
  56. package/app/api/calendar-feeds/route.tsx +52 -0
  57. package/app/api/calendar-feeds/sync-all/route.tsx +34 -0
  58. package/app/api/cron/calendar-feeds/route.tsx +31 -0
  59. package/app/api/cron/stale-tasks/route.tsx +51 -0
  60. package/app/api/cron/sync/route.tsx +34 -0
  61. package/app/api/devices/[deviceId]/route.tsx +25 -0
  62. package/app/api/devices/route.tsx +41 -0
  63. package/app/api/export/route.tsx +40 -0
  64. package/app/api/feedback/route.tsx +54 -0
  65. package/app/api/folders/[folderId]/route.tsx +51 -0
  66. package/app/api/folders/route.tsx +37 -0
  67. package/app/api/graph/route.tsx +242 -0
  68. package/app/api/guest/route.tsx +58 -0
  69. package/app/api/health/route.tsx +10 -0
  70. package/app/api/holidays/countries/route.tsx +14 -0
  71. package/app/api/holidays/route.tsx +49 -0
  72. package/app/api/holidays/states/route.tsx +21 -0
  73. package/app/api/invites/[token]/route.tsx +131 -0
  74. package/app/api/invites/route.tsx +74 -0
  75. package/app/api/mcp/generate-token/route.tsx +55 -0
  76. package/app/api/mcp/revoke-token/[tokenId]/route.tsx +30 -0
  77. package/app/api/mcp/update-alias/[tokenId]/route.tsx +22 -0
  78. package/app/api/notes/[noteId]/export/route.tsx +45 -0
  79. package/app/api/notes/[noteId]/route.tsx +360 -0
  80. package/app/api/notes/route.tsx +112 -0
  81. package/app/api/notifications/route.tsx +44 -0
  82. package/app/api/register/route.tsx +67 -0
  83. package/app/api/restore/route.tsx +148 -0
  84. package/app/api/sync/conflicts/[conflictId]/route.tsx +134 -0
  85. package/app/api/sync/conflicts/route.tsx +48 -0
  86. package/app/api/sync/status/route.tsx +49 -0
  87. package/app/api/sync/trigger/route.tsx +15 -0
  88. package/app/api/tasks/[taskId]/detail/route.tsx +68 -0
  89. package/app/api/tasks/[taskId]/route.tsx +259 -0
  90. package/app/api/tasks/bulk/route.tsx +133 -0
  91. package/app/api/tasks/route.tsx +36 -0
  92. package/app/api/workspace/active/route.tsx +39 -0
  93. package/app/api/workspace/create-team/route.tsx +42 -0
  94. package/app/api/workspace/kanban-statuses/route.tsx +71 -0
  95. package/app/api/workspace/members/[memberId]/route.tsx +69 -0
  96. package/app/api/workspace/route.tsx +24 -0
  97. package/app/download/page.tsx +170 -0
  98. package/app/favicon.ico +0 -0
  99. package/app/generated/prisma/client.d.ts +1 -0
  100. package/app/generated/prisma/client.js +5 -0
  101. package/app/generated/prisma/default.d.ts +1 -0
  102. package/app/generated/prisma/default.js +5 -0
  103. package/app/generated/prisma/edge.d.ts +1 -0
  104. package/app/generated/prisma/edge.js +497 -0
  105. package/app/generated/prisma/index-browser.js +523 -0
  106. package/app/generated/prisma/index.d.ts +46376 -0
  107. package/app/generated/prisma/index.js +497 -0
  108. package/app/generated/prisma/package.json +144 -0
  109. package/app/generated/prisma/query_compiler_fast_bg.js +2 -0
  110. package/app/generated/prisma/query_compiler_fast_bg.wasm +0 -0
  111. package/app/generated/prisma/query_compiler_fast_bg.wasm-base64.js +2 -0
  112. package/app/generated/prisma/runtime/client.d.ts +3386 -0
  113. package/app/generated/prisma/runtime/client.js +86 -0
  114. package/app/generated/prisma/runtime/index-browser.d.ts +90 -0
  115. package/app/generated/prisma/runtime/index-browser.js +6 -0
  116. package/app/generated/prisma/runtime/wasm-compiler-edge.js +76 -0
  117. package/app/generated/prisma/schema.prisma +456 -0
  118. package/app/generated/prisma/wasm-edge-light-loader.mjs +5 -0
  119. package/app/generated/prisma/wasm-worker-loader.mjs +5 -0
  120. package/app/globals.css +54 -0
  121. package/app/invite/[token]/page.tsx +52 -0
  122. package/app/layout.tsx +90 -0
  123. package/app/mcp/route.tsx +430 -0
  124. package/app/opengraph-image.tsx +120 -0
  125. package/app/page.tsx +398 -0
  126. package/app/privacy/page.tsx +69 -0
  127. package/app/robots.tsx +25 -0
  128. package/app/sitemap.tsx +36 -0
  129. package/app/terms/page.tsx +69 -0
  130. package/app/upgrade/page.tsx +75 -0
  131. package/auth.config.ts +33 -0
  132. package/auth.ts +79 -0
  133. package/bin/brief.js +224 -0
  134. package/components/auth/login-form.tsx +302 -0
  135. package/components/auth/password-checklist.tsx +31 -0
  136. package/components/auth/password-input.tsx +36 -0
  137. package/components/auth/switch-account-button.test.tsx +22 -0
  138. package/components/auth/switch-account-button.tsx +19 -0
  139. package/components/auth/two-factor-input.tsx +116 -0
  140. package/components/billing/billing-dashboard.tsx +265 -0
  141. package/components/billing/card-form.tsx +210 -0
  142. package/components/billing/claim-account-form.tsx +99 -0
  143. package/components/branding/app-logo.test.tsx +20 -0
  144. package/components/branding/app-logo.tsx +25 -0
  145. package/components/calendar/calendar-agenda.tsx +150 -0
  146. package/components/calendar/calendar-drag.test.tsx +177 -0
  147. package/components/calendar/calendar-grid.tsx +357 -0
  148. package/components/calendar/calendar-hooks.test.tsx +27 -0
  149. package/components/calendar/calendar-hooks.ts +351 -0
  150. package/components/calendar/calendar-toolbar.test.tsx +68 -0
  151. package/components/calendar/calendar-toolbar.tsx +291 -0
  152. package/components/calendar/calendar-types.ts +148 -0
  153. package/components/calendar/calendar-view.test.tsx +295 -0
  154. package/components/calendar/calendar-view.tsx +307 -0
  155. package/components/calendar/day-detail-popover.tsx +174 -0
  156. package/components/calendar/task-chip.tsx +86 -0
  157. package/components/command/command-palette.test.tsx +33 -0
  158. package/components/command/command-palette.tsx +310 -0
  159. package/components/download-cta.tsx +87 -0
  160. package/components/feedback/feedback-popup.tsx +207 -0
  161. package/components/graph/graph-draw.ts +337 -0
  162. package/components/graph/graph-overlays.tsx +160 -0
  163. package/components/graph/graph-page.test.tsx +131 -0
  164. package/components/graph/graph-page.tsx +263 -0
  165. package/components/graph/graph-types.ts +47 -0
  166. package/components/graph/graph-view.tsx +322 -0
  167. package/components/guide/guide-view.tsx +522 -0
  168. package/components/kanban/kanban-board.test.tsx +128 -0
  169. package/components/kanban/kanban-board.tsx +361 -0
  170. package/components/kanban/kanban-card-menu.tsx +102 -0
  171. package/components/kanban/kanban-card.tsx +227 -0
  172. package/components/kanban/kanban-column.tsx +49 -0
  173. package/components/kanban/kanban-status-context.tsx +28 -0
  174. package/components/landing/calendar-sandbox.test.tsx +15 -0
  175. package/components/landing/calendar-sandbox.tsx +107 -0
  176. package/components/landing/graph-sandbox.test.tsx +27 -0
  177. package/components/landing/graph-sandbox.tsx +80 -0
  178. package/components/landing/kanban-sandbox.test.tsx +24 -0
  179. package/components/landing/kanban-sandbox.tsx +101 -0
  180. package/components/landing/landing-showcase.test.tsx +21 -0
  181. package/components/landing/landing-showcase.tsx +54 -0
  182. package/components/landing/list-sandbox.tsx +86 -0
  183. package/components/landing/mock-workspace.ts +168 -0
  184. package/components/landing/notes-sandbox.test.tsx +14 -0
  185. package/components/landing/notes-sandbox.tsx +88 -0
  186. package/components/layout/app-shell.tsx +83 -0
  187. package/components/layout/backup-scheduler.tsx +122 -0
  188. package/components/layout/bottom-nav.tsx +43 -0
  189. package/components/layout/icon-bar.test.tsx +29 -0
  190. package/components/layout/icon-bar.tsx +118 -0
  191. package/components/layout/mobile-top-bar.tsx +68 -0
  192. package/components/layout/notes-panel-folder.tsx +127 -0
  193. package/components/layout/notes-panel-note-item.tsx +140 -0
  194. package/components/layout/notes-panel-task-tab.tsx +63 -0
  195. package/components/layout/notes-panel-types.ts +44 -0
  196. package/components/layout/notes-panel.tsx +476 -0
  197. package/components/layout/notification-bell.tsx +251 -0
  198. package/components/layout/paywall-screen.tsx +41 -0
  199. package/components/layout/pro-banner.tsx +76 -0
  200. package/components/layout/sw-register.tsx +27 -0
  201. package/components/layout/workspace-switcher.tsx +90 -0
  202. package/components/notes/mobile-bottom-sheet.tsx +99 -0
  203. package/components/notes/note-editor-context-menu.tsx +47 -0
  204. package/components/notes/note-editor-dom.ts +33 -0
  205. package/components/notes/note-editor-dropdowns.tsx +484 -0
  206. package/components/notes/note-editor-hooks.ts +692 -0
  207. package/components/notes/note-editor-keyboard.ts +305 -0
  208. package/components/notes/note-editor-overlay.tsx +90 -0
  209. package/components/notes/note-editor.test.tsx +372 -0
  210. package/components/notes/note-editor.tsx +662 -0
  211. package/components/notes/note-preview-pane.tsx +156 -0
  212. package/components/notes/note-tabs.tsx +120 -0
  213. package/components/notes/note-types.tsx +157 -0
  214. package/components/settings/accept-invite.tsx +108 -0
  215. package/components/settings/agent-token-settings.tsx +369 -0
  216. package/components/settings/backup-restore-settings.test.tsx +25 -0
  217. package/components/settings/backup-restore-settings.tsx +327 -0
  218. package/components/settings/calendar-feeds-settings.tsx +489 -0
  219. package/components/settings/calendar-general-settings.tsx +174 -0
  220. package/components/settings/confirm-danger-action.test.tsx +215 -0
  221. package/components/settings/confirm-danger-action.tsx +65 -0
  222. package/components/settings/security-settings.tsx +252 -0
  223. package/components/settings/settings-guidance.test.tsx +98 -0
  224. package/components/settings/team-settings.tsx +319 -0
  225. package/components/settings/two-factor-auth.tsx +296 -0
  226. package/components/settings/workspace-settings-client.tsx +363 -0
  227. package/components/settings/workspace-settings-form.tsx +73 -0
  228. package/components/sync/conflict-viewer.tsx +247 -0
  229. package/components/sync/sync-indicator.tsx +171 -0
  230. package/components/tasks/snippet-thread.tsx +119 -0
  231. package/components/tasks/status-dot.tsx +47 -0
  232. package/components/tasks/task-badge.tsx +43 -0
  233. package/components/tasks/task-detail.test.tsx +187 -0
  234. package/components/tasks/task-detail.tsx +458 -0
  235. package/components/tasks/task-list-filters.test.tsx +75 -0
  236. package/components/tasks/task-list-filters.tsx +163 -0
  237. package/components/tasks/task-list-types.ts +20 -0
  238. package/components/tasks/task-list.test.tsx +175 -0
  239. package/components/tasks/task-list.tsx +481 -0
  240. package/components/tasks/task-row.tsx +85 -0
  241. package/components/tasks/task-table-row.tsx +259 -0
  242. package/components/ui/skeleton.tsx +3 -0
  243. package/components/ui/toast.test.tsx +42 -0
  244. package/components/ui/toast.tsx +70 -0
  245. package/electron/main.ts +251 -0
  246. package/electron/preload.ts +56 -0
  247. package/instrumentation.tsx +23 -0
  248. package/lib/api-error.ts +50 -0
  249. package/lib/backup/backup-runner.test.ts +32 -0
  250. package/lib/backup/backup-runner.ts +19 -0
  251. package/lib/backup/backup-schedule.test.ts +23 -0
  252. package/lib/backup/backup-schedule.ts +55 -0
  253. package/lib/backup/backup-settings.test.ts +30 -0
  254. package/lib/backup/backup-settings.ts +27 -0
  255. package/lib/backup/export-notes-zip.test.ts +26 -0
  256. package/lib/backup/export-notes-zip.ts +82 -0
  257. package/lib/backup/export-workspace-backup.test.ts +17 -0
  258. package/lib/backup/export-workspace-backup.ts +77 -0
  259. package/lib/backup/restore-workspace-from-export.test.ts +18 -0
  260. package/lib/backup/restore-workspace-from-export.ts +183 -0
  261. package/lib/backup/types.ts +14 -0
  262. package/lib/brand-icons.ts +1 -0
  263. package/lib/calendar-feed-crypto.ts +38 -0
  264. package/lib/calendar-feed.ts +239 -0
  265. package/lib/client/online-status.ts +47 -0
  266. package/lib/conflict-resolver.test.ts +57 -0
  267. package/lib/conflict-resolver.ts +240 -0
  268. package/lib/db-init.ts +79 -0
  269. package/lib/email.ts +159 -0
  270. package/lib/encryption.test.ts +41 -0
  271. package/lib/encryption.ts +98 -0
  272. package/lib/extract-snippet.test.ts +123 -0
  273. package/lib/extract-snippet.ts +69 -0
  274. package/lib/kanban-status.ts +55 -0
  275. package/lib/license.ts +21 -0
  276. package/lib/limits.ts +31 -0
  277. package/lib/mcp-auth.test.ts +58 -0
  278. package/lib/mcp-auth.ts +65 -0
  279. package/lib/mcp-contract.test.ts +25 -0
  280. package/lib/mcp-contract.ts +210 -0
  281. package/lib/mcp-handler.ts +31 -0
  282. package/lib/mcp-url.test.ts +12 -0
  283. package/lib/mcp-url.ts +7 -0
  284. package/lib/mentions.test.ts +45 -0
  285. package/lib/mentions.ts +73 -0
  286. package/lib/note-crypto.ts +108 -0
  287. package/lib/note-sync.ts +201 -0
  288. package/lib/note-title.ts +93 -0
  289. package/lib/prisma.ts +193 -0
  290. package/lib/pro-flush.ts +292 -0
  291. package/lib/rate-limit.ts +57 -0
  292. package/lib/stripe.ts +38 -0
  293. package/lib/sync-worker.ts +388 -0
  294. package/lib/task-parser.test.ts +91 -0
  295. package/lib/task-parser.ts +81 -0
  296. package/lib/task-utils.ts +52 -0
  297. package/lib/use-is-electron.ts +19 -0
  298. package/lib/use-is-mobile.ts +22 -0
  299. package/lib/validation/calendar-feed.ts +31 -0
  300. package/lib/validation/note.ts +27 -0
  301. package/lib/validation/task.ts +26 -0
  302. package/lib/view-preferences.test.ts +54 -0
  303. package/lib/view-preferences.ts +28 -0
  304. package/lib/workspace.ts +66 -0
  305. package/next.config.ts +21 -0
  306. package/package.json +99 -0
  307. package/postcss.config.mjs +7 -0
  308. package/prisma/migrations/20260519021916_init/migration.sql +388 -0
  309. package/prisma/migrations/20260519061113_drop_sync_password/migration.sql +8 -0
  310. package/prisma/migrations/20260520065016_add_task_start_date/migration.sql +2 -0
  311. package/prisma/migrations/20260529010600_remove_encryption_fields/migration.sql +12 -0
  312. package/prisma/migrations/20260529020000_restore_encryption_salt/migration.sql +3 -0
  313. package/prisma/migrations/20260529030000_add_folders/migration.sql +17 -0
  314. package/prisma/migrations/20260605000000_deferred_fixes/migration.sql +31 -0
  315. package/prisma/migrations/20260605020806_add_pending_sync_to_note_and_task/migration.sql +5 -0
  316. package/prisma/migrations/20260605063634_add_stripe_webhook_event_sync_lock/migration.sql +14 -0
  317. package/prisma/migrations/20260605100000_add_prod_indexes/migration.sql +26 -0
  318. package/prisma/migrations/20260608081404_add_kanban_statuses/migration.sql +23 -0
  319. package/prisma/migrations/20260611032723_add_calendar_feeds/migration.sql +43 -0
  320. package/prisma/migrations/20260611040000_add_calendar_feed_color/migration.sql +2 -0
  321. package/prisma/migrations/20260611050000_add_task_priority/migration.sql +14 -0
  322. package/prisma/migrations/20260612060000_add_critical_priority/migration.sql +2 -0
  323. package/prisma/migrations/20260613090000_add_backup_settings/migration.sql +25 -0
  324. package/prisma/migrations/20260614160000_add_feedback/migration.sql +20 -0
  325. package/prisma/migrations/20260614210000_add_2fa/migration.sql +4 -0
  326. package/prisma/migrations/migration_lock.toml +3 -0
  327. package/prisma/schema.prisma +457 -0
  328. package/public/Logo_icon.svg +1 -0
  329. package/public/file.svg +1 -0
  330. package/public/globe.svg +1 -0
  331. package/public/icon-192.png +0 -0
  332. package/public/icon-512.png +0 -0
  333. package/public/icon.svg +4 -0
  334. package/public/icon_dark.svg +1 -0
  335. package/public/knotpad_icon.svg +1 -0
  336. package/public/knotpad_logo_full.svg +1 -0
  337. package/public/manifest.json +14 -0
  338. package/public/next.svg +1 -0
  339. package/public/sw.js +137 -0
  340. package/public/vercel.svg +1 -0
  341. package/public/window.svg +1 -0
  342. package/tsconfig.json +35 -0
@@ -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
+ }
@@ -0,0 +1,5 @@
1
+
2
+ /* !!! This is code generated by Prisma. Do not edit directly. !!!
3
+ /* eslint-disable */
4
+ // biome-ignore-all lint: generated file
5
+ export default import('./query_compiler_fast_bg.wasm?module')
@@ -0,0 +1,5 @@
1
+
2
+ /* !!! This is code generated by Prisma. Do not edit directly. !!!
3
+ /* eslint-disable */
4
+ // biome-ignore-all lint: generated file
5
+ export default import('./query_compiler_fast_bg.wasm')
@@ -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
+ }