idea-manager 1.9.0 → 2.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 (185) hide show
  1. package/.next/build-manifest.json +2 -2
  2. package/.next/routes-manifest.json +35 -0
  3. package/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  4. package/.next/server/app/_global-error.html +2 -2
  5. package/.next/server/app/_global-error.rsc +1 -1
  6. package/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  7. package/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
  8. package/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
  9. package/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  10. package/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  11. package/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  12. package/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  13. package/.next/server/app/_not-found.html +2 -2
  14. package/.next/server/app/_not-found.rsc +2 -2
  15. package/.next/server/app/_not-found.segments/_full.segment.rsc +2 -2
  16. package/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  17. package/.next/server/app/_not-found.segments/_index.segment.rsc +2 -2
  18. package/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  19. package/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  20. package/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
  21. package/.next/server/app/api/advisor-actions/route.js +15 -0
  22. package/.next/server/app/api/advisor-actions/route_client-reference-manifest.js +1 -0
  23. package/.next/server/app/api/archive/route.js +1 -122
  24. package/.next/server/app/api/archive/route_client-reference-manifest.js +1 -1
  25. package/.next/server/app/api/filesystem/route_client-reference-manifest.js +1 -1
  26. package/.next/server/app/api/filesystem/tree/route_client-reference-manifest.js +1 -1
  27. package/.next/server/app/api/global-advisor/route.js +37 -0
  28. package/.next/server/app/api/global-advisor/route_client-reference-manifest.js +1 -0
  29. package/.next/server/app/api/global-memo/route.js +8 -0
  30. package/.next/server/app/api/global-memo/route_client-reference-manifest.js +1 -1
  31. package/.next/server/app/api/health/route_client-reference-manifest.js +1 -1
  32. package/.next/server/app/api/maintenance/route.js +130 -0
  33. package/.next/server/app/api/maintenance/route_client-reference-manifest.js +1 -0
  34. package/.next/server/app/api/projects/[id]/advisor/route.js +22 -11
  35. package/.next/server/app/api/projects/[id]/advisor/route_client-reference-manifest.js +1 -1
  36. package/.next/server/app/api/projects/[id]/apply-distribute/route.js +2 -8
  37. package/.next/server/app/api/projects/[id]/apply-distribute/route_client-reference-manifest.js +1 -1
  38. package/.next/server/app/api/projects/[id]/auto-distribute/route.js +126 -3
  39. package/.next/server/app/api/projects/[id]/auto-distribute/route_client-reference-manifest.js +1 -1
  40. package/.next/server/app/api/projects/[id]/brainstorm/route.js +124 -1
  41. package/.next/server/app/api/projects/[id]/brainstorm/route_client-reference-manifest.js +1 -1
  42. package/.next/server/app/api/projects/[id]/git-sync/route.js +124 -1
  43. package/.next/server/app/api/projects/[id]/git-sync/route_client-reference-manifest.js +1 -1
  44. package/.next/server/app/api/projects/[id]/route.js +124 -1
  45. package/.next/server/app/api/projects/[id]/route_client-reference-manifest.js +1 -1
  46. package/.next/server/app/api/projects/[id]/sub-projects/[subId]/route.js +8 -0
  47. package/.next/server/app/api/projects/[id]/sub-projects/[subId]/route_client-reference-manifest.js +1 -1
  48. package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/chat/route.js +1 -7
  49. package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/chat/route_client-reference-manifest.js +1 -1
  50. package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/prompt/route.js +8 -0
  51. package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/prompt/route_client-reference-manifest.js +1 -1
  52. package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/refine/route.js +2 -8
  53. package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/refine/route_client-reference-manifest.js +1 -1
  54. package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/route.js +1 -122
  55. package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/route_client-reference-manifest.js +1 -1
  56. package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/reorder/route.js +124 -0
  57. package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/reorder/route_client-reference-manifest.js +1 -0
  58. package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/route.js +1 -122
  59. package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/route_client-reference-manifest.js +1 -1
  60. package/.next/server/app/api/projects/[id]/sub-projects/route.js +8 -0
  61. package/.next/server/app/api/projects/[id]/sub-projects/route_client-reference-manifest.js +1 -1
  62. package/.next/server/app/api/projects/route.js +124 -1
  63. package/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
  64. package/.next/server/app/api/search/route.js +8 -0
  65. package/.next/server/app/api/search/route_client-reference-manifest.js +1 -1
  66. package/.next/server/app/api/sync/route.js +8 -0
  67. package/.next/server/app/api/sync/route_client-reference-manifest.js +1 -1
  68. package/.next/server/app/api/tasks/[taskId]/move/route.js +15 -0
  69. package/.next/server/app/api/tasks/[taskId]/move/route_client-reference-manifest.js +1 -0
  70. package/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
  71. package/.next/server/app/api/version/route_client-reference-manifest.js +1 -1
  72. package/.next/server/app/index.html +2 -2
  73. package/.next/server/app/index.rsc +3 -3
  74. package/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  75. package/.next/server/app/index.segments/_full.segment.rsc +3 -3
  76. package/.next/server/app/index.segments/_head.segment.rsc +1 -1
  77. package/.next/server/app/index.segments/_index.segment.rsc +2 -2
  78. package/.next/server/app/index.segments/_tree.segment.rsc +2 -2
  79. package/.next/server/app/page.js +12 -12
  80. package/.next/server/app/page_client-reference-manifest.js +1 -1
  81. package/.next/server/app/projects/[id]/page_client-reference-manifest.js +1 -1
  82. package/.next/server/app-paths-manifest.json +18 -13
  83. package/.next/server/chunks/{117.js → 697.js} +16 -2
  84. package/.next/server/pages/404.html +2 -2
  85. package/.next/server/pages/500.html +2 -2
  86. package/.next/static/KREG104cVn2mBTMPTDTvH/_buildManifest.js +1 -0
  87. package/.next/static/chunks/374-23189d7e246ad164.js +1 -0
  88. package/.next/static/chunks/app/_global-error/page-f051f234bea7bddd.js +1 -0
  89. package/.next/static/chunks/app/api/advisor-actions/route-f051f234bea7bddd.js +1 -0
  90. package/.next/static/chunks/app/api/archive/route-f051f234bea7bddd.js +1 -0
  91. package/.next/static/chunks/app/api/filesystem/route-f051f234bea7bddd.js +1 -0
  92. package/.next/static/chunks/app/api/filesystem/tree/route-f051f234bea7bddd.js +1 -0
  93. package/.next/static/chunks/app/api/global-advisor/route-f051f234bea7bddd.js +1 -0
  94. package/.next/static/chunks/app/api/global-memo/route-f051f234bea7bddd.js +1 -0
  95. package/.next/static/chunks/app/api/health/route-f051f234bea7bddd.js +1 -0
  96. package/.next/static/chunks/app/api/maintenance/route-f051f234bea7bddd.js +1 -0
  97. package/.next/static/chunks/app/api/projects/[id]/advisor/route-f051f234bea7bddd.js +1 -0
  98. package/.next/static/chunks/app/api/projects/[id]/apply-distribute/route-f051f234bea7bddd.js +1 -0
  99. package/.next/static/chunks/app/api/projects/[id]/auto-distribute/route-f051f234bea7bddd.js +1 -0
  100. package/.next/static/chunks/app/api/projects/[id]/brainstorm/route-f051f234bea7bddd.js +1 -0
  101. package/.next/static/chunks/app/api/projects/[id]/git-sync/route-f051f234bea7bddd.js +1 -0
  102. package/.next/static/chunks/app/api/projects/[id]/route-f051f234bea7bddd.js +1 -0
  103. package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/route-f051f234bea7bddd.js +1 -0
  104. package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/chat/route-f051f234bea7bddd.js +1 -0
  105. package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/prompt/route-f051f234bea7bddd.js +1 -0
  106. package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/refine/route-f051f234bea7bddd.js +1 -0
  107. package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/route-f051f234bea7bddd.js +1 -0
  108. package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/reorder/route-f051f234bea7bddd.js +1 -0
  109. package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/route-f051f234bea7bddd.js +1 -0
  110. package/.next/static/chunks/app/api/projects/[id]/sub-projects/route-f051f234bea7bddd.js +1 -0
  111. package/.next/static/chunks/app/api/projects/route-f051f234bea7bddd.js +1 -0
  112. package/.next/static/chunks/app/api/search/route-f051f234bea7bddd.js +1 -0
  113. package/.next/static/chunks/app/api/sync/route-f051f234bea7bddd.js +1 -0
  114. package/.next/static/chunks/app/api/tasks/[taskId]/move/route-f051f234bea7bddd.js +1 -0
  115. package/.next/static/chunks/app/api/update/route-f051f234bea7bddd.js +1 -0
  116. package/.next/static/chunks/app/api/version/route-f051f234bea7bddd.js +1 -0
  117. package/.next/static/chunks/app/page-9117037f2947f4f6.js +28 -0
  118. package/.next/static/chunks/next/dist/client/components/builtin/app-error-f051f234bea7bddd.js +1 -0
  119. package/.next/static/chunks/next/dist/client/components/builtin/forbidden-f051f234bea7bddd.js +1 -0
  120. package/.next/static/chunks/next/dist/client/components/builtin/not-found-f051f234bea7bddd.js +1 -0
  121. package/.next/static/chunks/next/dist/client/components/builtin/unauthorized-f051f234bea7bddd.js +1 -0
  122. package/.next/static/css/e9071b58a99b47e4.css +3 -0
  123. package/package.json +1 -1
  124. package/src/app/api/advisor-actions/route.ts +52 -0
  125. package/src/app/api/global-advisor/route.ts +50 -0
  126. package/src/app/api/maintenance/route.ts +36 -0
  127. package/src/app/api/projects/[id]/sub-projects/[subId]/tasks/reorder/route.ts +24 -0
  128. package/src/app/api/tasks/[taskId]/move/route.ts +30 -0
  129. package/src/components/advisor/ActionBlock.tsx +124 -0
  130. package/src/components/advisor/AdvisorChat.tsx +175 -0
  131. package/src/components/advisor/GlobalAdvisorLayer.tsx +38 -0
  132. package/src/components/dashboard/DashboardPanel.tsx +2 -0
  133. package/src/components/memo/GlobalMemoLayer.tsx +81 -0
  134. package/src/components/tabs/TabBar.tsx +2 -0
  135. package/src/components/tabs/TabShell.tsx +6 -0
  136. package/src/components/task/NoteEditor.tsx +137 -0
  137. package/src/components/task/ProjectTree.tsx +105 -57
  138. package/src/components/task/TaskChat.tsx +4 -0
  139. package/src/components/task/TaskDetail.tsx +182 -1
  140. package/src/components/ui/AiActivityIndicator.tsx +66 -0
  141. package/src/components/ui/ShortcutOverlay.tsx +108 -0
  142. package/src/components/workspace/ProjectAdvisor.tsx +17 -181
  143. package/src/components/workspace/WorkspacePanel.tsx +75 -3
  144. package/src/hooks/useAiActivity.ts +6 -0
  145. package/src/lib/advisor-actions/parse.ts +59 -0
  146. package/src/lib/ai/global-context.ts +114 -0
  147. package/src/lib/ai/project-context.ts +22 -2
  148. package/src/lib/ai-activity.ts +33 -0
  149. package/src/lib/db/queries/global-conversations.ts +31 -0
  150. package/src/lib/db/queries/tasks.ts +3 -1
  151. package/src/lib/db/schema.ts +8 -0
  152. package/src/types/advisor-actions.ts +25 -0
  153. package/.next/static/chunks/374-769431701aab500f.js +0 -1
  154. package/.next/static/chunks/app/_global-error/page-3ff8f59aaa75b8f8.js +0 -1
  155. package/.next/static/chunks/app/api/archive/route-3ff8f59aaa75b8f8.js +0 -1
  156. package/.next/static/chunks/app/api/filesystem/route-3ff8f59aaa75b8f8.js +0 -1
  157. package/.next/static/chunks/app/api/filesystem/tree/route-3ff8f59aaa75b8f8.js +0 -1
  158. package/.next/static/chunks/app/api/global-memo/route-3ff8f59aaa75b8f8.js +0 -1
  159. package/.next/static/chunks/app/api/health/route-3ff8f59aaa75b8f8.js +0 -1
  160. package/.next/static/chunks/app/api/projects/[id]/advisor/route-3ff8f59aaa75b8f8.js +0 -1
  161. package/.next/static/chunks/app/api/projects/[id]/apply-distribute/route-3ff8f59aaa75b8f8.js +0 -1
  162. package/.next/static/chunks/app/api/projects/[id]/auto-distribute/route-3ff8f59aaa75b8f8.js +0 -1
  163. package/.next/static/chunks/app/api/projects/[id]/brainstorm/route-3ff8f59aaa75b8f8.js +0 -1
  164. package/.next/static/chunks/app/api/projects/[id]/git-sync/route-3ff8f59aaa75b8f8.js +0 -1
  165. package/.next/static/chunks/app/api/projects/[id]/route-3ff8f59aaa75b8f8.js +0 -1
  166. package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/route-3ff8f59aaa75b8f8.js +0 -1
  167. package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/chat/route-3ff8f59aaa75b8f8.js +0 -1
  168. package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/prompt/route-3ff8f59aaa75b8f8.js +0 -1
  169. package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/refine/route-3ff8f59aaa75b8f8.js +0 -1
  170. package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/route-3ff8f59aaa75b8f8.js +0 -1
  171. package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/route-3ff8f59aaa75b8f8.js +0 -1
  172. package/.next/static/chunks/app/api/projects/[id]/sub-projects/route-3ff8f59aaa75b8f8.js +0 -1
  173. package/.next/static/chunks/app/api/projects/route-3ff8f59aaa75b8f8.js +0 -1
  174. package/.next/static/chunks/app/api/search/route-3ff8f59aaa75b8f8.js +0 -1
  175. package/.next/static/chunks/app/api/sync/route-3ff8f59aaa75b8f8.js +0 -1
  176. package/.next/static/chunks/app/api/update/route-3ff8f59aaa75b8f8.js +0 -1
  177. package/.next/static/chunks/app/api/version/route-3ff8f59aaa75b8f8.js +0 -1
  178. package/.next/static/chunks/app/page-e935ee928da68ca2.js +0 -28
  179. package/.next/static/chunks/next/dist/client/components/builtin/app-error-3ff8f59aaa75b8f8.js +0 -1
  180. package/.next/static/chunks/next/dist/client/components/builtin/forbidden-3ff8f59aaa75b8f8.js +0 -1
  181. package/.next/static/chunks/next/dist/client/components/builtin/not-found-3ff8f59aaa75b8f8.js +0 -1
  182. package/.next/static/chunks/next/dist/client/components/builtin/unauthorized-3ff8f59aaa75b8f8.js +0 -1
  183. package/.next/static/css/e4c7cd5a570312d9.css +0 -3
  184. package/.next/static/pxqzEiwniZAUDOUTb5SnX/_buildManifest.js +0 -1
  185. /package/.next/static/{pxqzEiwniZAUDOUTb5SnX → KREG104cVn2mBTMPTDTvH}/_ssgManifest.js +0 -0
@@ -0,0 +1,3 @@
1
+ @font-face{font-family:Geist;font-style:normal;font-weight:100 900;font-display:swap;src:url(/_next/static/media/8d697b304b401681-s.woff2) format("woff2");unicode-range:u+0301,u+0400-045f,u+0490-0491,u+04b0-04b1,u+2116}@font-face{font-family:Geist;font-style:normal;font-weight:100 900;font-display:swap;src:url(/_next/static/media/ba015fad6dcf6784-s.woff2) format("woff2");unicode-range:u+0100-02ba,u+02bd-02c5,u+02c7-02cc,u+02ce-02d7,u+02dd-02ff,u+0304,u+0308,u+0329,u+1d00-1dbf,u+1e00-1e9f,u+1ef2-1eff,u+2020,u+20a0-20ab,u+20ad-20c0,u+2113,u+2c60-2c7f,u+a720-a7ff}@font-face{font-family:Geist;font-style:normal;font-weight:100 900;font-display:swap;src:url(/_next/static/media/4cf2300e9c8272f7-s.p.woff2) format("woff2");unicode-range:u+00??,u+0131,u+0152-0153,u+02bb-02bc,u+02c6,u+02da,u+02dc,u+0304,u+0308,u+0329,u+2000-206f,u+20ac,u+2122,u+2191,u+2193,u+2212,u+2215,u+feff,u+fffd}@font-face{font-family:Geist Fallback;src:local("Arial");ascent-override:95.94%;descent-override:28.16%;line-gap-override:0.00%;size-adjust:104.76%}.__className_188709{font-family:Geist,Geist Fallback;font-style:normal}.__variable_188709{--font-geist-sans:"Geist","Geist Fallback"}@font-face{font-family:Geist Mono;font-style:normal;font-weight:100 900;font-display:swap;src:url(/_next/static/media/9610d9e46709d722-s.woff2) format("woff2");unicode-range:u+0301,u+0400-045f,u+0490-0491,u+04b0-04b1,u+2116}@font-face{font-family:Geist Mono;font-style:normal;font-weight:100 900;font-display:swap;src:url(/_next/static/media/747892c23ea88013-s.woff2) format("woff2");unicode-range:u+0100-02ba,u+02bd-02c5,u+02c7-02cc,u+02ce-02d7,u+02dd-02ff,u+0304,u+0308,u+0329,u+1d00-1dbf,u+1e00-1e9f,u+1ef2-1eff,u+2020,u+20a0-20ab,u+20ad-20c0,u+2113,u+2c60-2c7f,u+a720-a7ff}@font-face{font-family:Geist Mono;font-style:normal;font-weight:100 900;font-display:swap;src:url(/_next/static/media/93f479601ee12b01-s.p.woff2) format("woff2");unicode-range:u+00??,u+0131,u+0152-0153,u+02bb-02bc,u+02c6,u+02da,u+02dc,u+0304,u+0308,u+0329,u+2000-206f,u+20ac,u+2122,u+2191,u+2193,u+2212,u+2215,u+feff,u+fffd}@font-face{font-family:Geist Mono Fallback;src:local("Arial");ascent-override:74.67%;descent-override:21.92%;line-gap-override:0.00%;size-adjust:134.59%}.__className_9a8899{font-family:Geist Mono,Geist Mono Fallback;font-style:normal}.__variable_9a8899{--font-geist-mono:"Geist Mono","Geist Mono Fallback"}
2
+
3
+ /*! tailwindcss v4.2.1 | MIT License | https://tailwindcss.com */@layer properties{@supports (((-webkit-hyphens:none)) and (not (margin-trim:inline))) or ((-moz-orient:inline) and (not (color:rgb(from red r g b)))){*,::backdrop,:after,:before{--tw-translate-x:0;--tw-translate-y:0;--tw-translate-z:0;--tw-scale-x:1;--tw-scale-y:1;--tw-scale-z:1;--tw-rotate-x:initial;--tw-rotate-y:initial;--tw-rotate-z:initial;--tw-skew-x:initial;--tw-skew-y:initial;--tw-space-y-reverse:0;--tw-divide-y-reverse:0;--tw-border-style:solid;--tw-leading:initial;--tw-font-weight:initial;--tw-tracking:initial;--tw-ordinal:initial;--tw-slashed-zero:initial;--tw-numeric-figure:initial;--tw-numeric-spacing:initial;--tw-numeric-fraction:initial;--tw-shadow:0 0 #0000;--tw-shadow-color:initial;--tw-shadow-alpha:100%;--tw-inset-shadow:0 0 #0000;--tw-inset-shadow-color:initial;--tw-inset-shadow-alpha:100%;--tw-ring-color:initial;--tw-ring-shadow:0 0 #0000;--tw-inset-ring-color:initial;--tw-inset-ring-shadow:0 0 #0000;--tw-ring-inset:initial;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-offset-shadow:0 0 #0000;--tw-outline-style:solid;--tw-blur:initial;--tw-brightness:initial;--tw-contrast:initial;--tw-grayscale:initial;--tw-hue-rotate:initial;--tw-invert:initial;--tw-opacity:initial;--tw-saturate:initial;--tw-sepia:initial;--tw-drop-shadow:initial;--tw-drop-shadow-color:initial;--tw-drop-shadow-alpha:100%;--tw-drop-shadow-size:initial;--tw-backdrop-blur:initial;--tw-backdrop-brightness:initial;--tw-backdrop-contrast:initial;--tw-backdrop-grayscale:initial;--tw-backdrop-hue-rotate:initial;--tw-backdrop-invert:initial;--tw-backdrop-opacity:initial;--tw-backdrop-saturate:initial;--tw-backdrop-sepia:initial}}}@layer theme{:host,:root{--font-sans:"Pretendard",var(--font-geist-sans),system-ui,sans-serif;--font-mono:var(--font-geist-mono);--color-red-400:oklch(70.4% .191 22.216);--color-amber-400:oklch(82.8% .189 84.429);--color-emerald-400:oklch(76.5% .177 163.223);--color-cyan-400:oklch(78.9% .154 211.53);--color-indigo-400:oklch(67.3% .182 276.935);--color-black:#000;--color-white:#fff;--spacing:.25rem;--container-sm:24rem;--container-md:28rem;--container-lg:32rem;--container-xl:36rem;--container-5xl:64rem;--text-xs:.75rem;--text-xs--line-height:calc(1 / .75);--text-sm:.875rem;--text-sm--line-height:calc(1.25 / .875);--text-base:1rem;--text-base--line-height:calc(1.5 / 1);--text-lg:1.125rem;--text-lg--line-height:calc(1.75 / 1.125);--text-xl:1.25rem;--text-xl--line-height:calc(1.75 / 1.25);--text-2xl:1.5rem;--text-2xl--line-height:calc(2 / 1.5);--font-weight-normal:400;--font-weight-medium:500;--font-weight-semibold:600;--font-weight-bold:700;--tracking-tight:-.025em;--tracking-wide:.025em;--tracking-wider:.05em;--leading-relaxed:1.625;--radius-sm:.25rem;--radius-md:.375rem;--radius-lg:.5rem;--radius-xl:.75rem;--animate-spin:spin 1s linear infinite;--animate-pulse:pulse 2s cubic-bezier(.4,0,.6,1) infinite;--animate-bounce:bounce 1s infinite;--blur-sm:8px;--default-transition-duration:.15s;--default-transition-timing-function:cubic-bezier(.4,0,.2,1);--default-font-family:"Pretendard",var(--font-geist-sans),system-ui,sans-serif;--default-mono-font-family:var(--font-geist-mono)}}@layer base{*,::backdrop,:after,:before{box-sizing:border-box;border:0 solid;margin:0;padding:0}::file-selector-button{box-sizing:border-box;border:0 solid;margin:0;padding:0}:host,html{-webkit-text-size-adjust:100%;tab-size:4;line-height:1.5;font-family:var(--default-font-family,ui-sans-serif,system-ui,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji");font-feature-settings:var(--default-font-feature-settings,normal);font-variation-settings:var(--default-font-variation-settings,normal);-webkit-tap-highlight-color:transparent}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;-webkit-text-decoration:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,pre,samp{font-family:var(--default-mono-font-family,ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace);font-feature-settings:var(--default-mono-font-feature-settings,normal);font-variation-settings:var(--default-mono-font-variation-settings,normal);font-size:1em}small{font-size:80%}sub,sup{vertical-align:baseline;font-size:75%;line-height:0;position:relative}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}:-moz-focusring{outline:auto}progress{vertical-align:baseline}summary{display:list-item}menu,ol,ul{list-style:none}audio,canvas,embed,iframe,img,object,svg,video{vertical-align:middle;display:block}img,video{max-width:100%;height:auto}button,input,optgroup,select,textarea{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}::file-selector-button{font:inherit;font-feature-settings:inherit;font-variation-settings:inherit;letter-spacing:inherit;color:inherit;opacity:1;background-color:#0000;border-radius:0}:where(select:is([multiple],[size])) optgroup{font-weight:bolder}:where(select:is([multiple],[size])) optgroup option{padding-inline-start:20px}::file-selector-button{margin-inline-end:4px}::placeholder{opacity:1}@supports (not ((-webkit-appearance:-apple-pay-button))) or (contain-intrinsic-size:1px){::placeholder{color:currentColor}@supports (color:color-mix(in lab,red,red)){::placeholder{color:color-mix(in oklab,currentcolor 50%,transparent)}}}textarea{resize:vertical}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-date-and-time-value{min-height:1lh;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-datetime-edit,::-webkit-datetime-edit-year-field{padding-block:0}::-webkit-datetime-edit-day-field,::-webkit-datetime-edit-month-field{padding-block:0}::-webkit-datetime-edit-hour-field,::-webkit-datetime-edit-minute-field{padding-block:0}::-webkit-datetime-edit-millisecond-field,::-webkit-datetime-edit-second-field{padding-block:0}::-webkit-datetime-edit-meridiem-field{padding-block:0}::-webkit-calendar-picker-indicator{line-height:1}:-moz-ui-invalid{box-shadow:none}button,input:where([type=button],[type=reset],[type=submit]){appearance:button}::file-selector-button{appearance:button}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}[hidden]:where(:not([hidden=until-found])){display:none!important}}@layer components;@layer utilities{.visible{visibility:visible}.absolute{position:absolute}.fixed{position:fixed}.relative{position:relative}.inset-0{inset:calc(var(--spacing) * 0)}.start{inset-inline-start:var(--spacing)}.end{inset-inline-end:var(--spacing)}.top-full{top:100%}.right-0{right:calc(var(--spacing) * 0)}.right-3{right:calc(var(--spacing) * 3)}.bottom-2{bottom:calc(var(--spacing) * 2)}.bottom-4{bottom:calc(var(--spacing) * 4)}.left-1\/2{left:50%}.z-40{z-index:40}.z-50{z-index:50}.z-\[55\]{z-index:55}.z-\[60\]{z-index:60}.z-\[70\]{z-index:70}.z-\[80\]{z-index:80}.z-\[81\]{z-index:81}.container{width:100%}@media (min-width:40rem){.container{max-width:40rem}}@media (min-width:48rem){.container{max-width:48rem}}@media (min-width:64rem){.container{max-width:64rem}}@media (min-width:80rem){.container{max-width:80rem}}@media (min-width:96rem){.container{max-width:96rem}}.mx-4{margin-inline:calc(var(--spacing) * 4)}.mx-auto{margin-inline:auto}.my-2{margin-block:calc(var(--spacing) * 2)}.mt-0\.5{margin-top:calc(var(--spacing) * .5)}.mt-1{margin-top:calc(var(--spacing) * 1)}.mt-1\.5{margin-top:calc(var(--spacing) * 1.5)}.mt-2{margin-top:calc(var(--spacing) * 2)}.mt-3{margin-top:calc(var(--spacing) * 3)}.mt-4{margin-top:calc(var(--spacing) * 4)}.mr-1{margin-right:calc(var(--spacing) * 1)}.mr-2{margin-right:calc(var(--spacing) * 2)}.mb-0\.5{margin-bottom:calc(var(--spacing) * .5)}.mb-1{margin-bottom:calc(var(--spacing) * 1)}.mb-1\.5{margin-bottom:calc(var(--spacing) * 1.5)}.mb-2{margin-bottom:calc(var(--spacing) * 2)}.mb-3{margin-bottom:calc(var(--spacing) * 3)}.mb-4{margin-bottom:calc(var(--spacing) * 4)}.mb-6{margin-bottom:calc(var(--spacing) * 6)}.ml-1{margin-left:calc(var(--spacing) * 1)}.ml-2{margin-left:calc(var(--spacing) * 2)}.ml-3{margin-left:calc(var(--spacing) * 3)}.ml-auto{margin-left:auto}.contents{display:contents}.flex{display:flex}.grid{display:grid}.inline{display:inline}.inline-block{display:inline-block}.inline-flex{display:inline-flex}.table{display:table}.h-1\.5{height:calc(var(--spacing) * 1.5)}.h-2{height:calc(var(--spacing) * 2)}.h-4{height:calc(var(--spacing) * 4)}.h-6{height:calc(var(--spacing) * 6)}.h-\[38\%\]{height:38%}.h-\[65vh\]{height:65vh}.h-\[80vh\]{height:80vh}.h-full{height:100%}.h-screen{height:100vh}.max-h-40{max-height:calc(var(--spacing) * 40)}.max-h-64{max-height:calc(var(--spacing) * 64)}.max-h-\[50vh\]{max-height:50vh}.max-h-\[55vh\]{max-height:55vh}.max-h-\[70vh\]{max-height:70vh}.max-h-\[80vh\]{max-height:80vh}.max-h-\[85vh\]{max-height:85vh}.max-h-\[200px\]{max-height:200px}.max-h-\[280px\]{max-height:280px}.max-h-\[400px\]{max-height:400px}.max-h-\[550px\]{max-height:550px}.max-h-\[700px\]{max-height:700px}.min-h-0{min-height:calc(var(--spacing) * 0)}.min-h-\[150px\]{min-height:150px}.min-h-\[240px\]{min-height:240px}.min-h-\[300px\]{min-height:300px}.min-h-screen{min-height:100vh}.w-1\.5{width:calc(var(--spacing) * 1.5)}.w-2{width:calc(var(--spacing) * 2)}.w-3{width:calc(var(--spacing) * 3)}.w-4{width:calc(var(--spacing) * 4)}.w-5{width:calc(var(--spacing) * 5)}.w-6{width:calc(var(--spacing) * 6)}.w-8{width:calc(var(--spacing) * 8)}.w-24{width:calc(var(--spacing) * 24)}.w-72{width:calc(var(--spacing) * 72)}.w-\[420px\]{width:420px}.w-\[480px\]{width:480px}.w-\[520px\]{width:520px}.w-\[560px\]{width:560px}.w-\[640px\]{width:640px}.w-\[720px\]{width:720px}.w-full{width:100%}.max-w-5xl{max-width:var(--container-5xl)}.max-w-48{max-width:calc(var(--spacing) * 48)}.max-w-\[50ch\]{max-width:50ch}.max-w-\[70\%\]{max-width:70%}.max-w-\[85vw\]{max-width:85vw}.max-w-\[90\%\]{max-width:90%}.max-w-\[90vw\]{max-width:90vw}.max-w-\[92\%\]{max-width:92%}.max-w-\[200px\]{max-width:200px}.max-w-\[220px\]{max-width:220px}.max-w-\[300px\]{max-width:300px}.max-w-lg{max-width:var(--container-lg)}.max-w-md{max-width:var(--container-md)}.max-w-sm{max-width:var(--container-sm)}.max-w-xl{max-width:var(--container-xl)}.min-w-0{min-width:calc(var(--spacing) * 0)}.flex-1{flex:1}.flex-shrink-0,.shrink-0{flex-shrink:0}.-translate-x-1\/2{--tw-translate-x:calc(calc(1 / 2 * 100%) * -1);translate:var(--tw-translate-x) var(--tw-translate-y)}.scale-110{--tw-scale-x:110%;--tw-scale-y:110%;--tw-scale-z:110%;scale:var(--tw-scale-x) var(--tw-scale-y)}.scale-\[1\.02\]{scale:1.02}.transform{transform:var(--tw-rotate-x,) var(--tw-rotate-y,) var(--tw-rotate-z,) var(--tw-skew-x,) var(--tw-skew-y,)}.animate-bounce{animation:var(--animate-bounce)}.animate-pulse{animation:var(--animate-pulse)}.animate-spin{animation:var(--animate-spin)}.cursor-grab{cursor:grab}.cursor-pointer{cursor:pointer}.cursor-text{cursor:text}.resize-none{resize:none}.grid-cols-1{grid-template-columns:repeat(1,minmax(0,1fr))}.grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}.grid-cols-6{grid-template-columns:repeat(6,minmax(0,1fr))}.grid-cols-\[auto_1fr\]{grid-template-columns:auto 1fr}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.items-center{align-items:center}.items-end{align-items:flex-end}.items-start{align-items:flex-start}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.justify-end{justify-content:flex-end}.gap-0\.5{gap:calc(var(--spacing) * .5)}.gap-1{gap:calc(var(--spacing) * 1)}.gap-1\.5{gap:calc(var(--spacing) * 1.5)}.gap-2{gap:calc(var(--spacing) * 2)}.gap-3{gap:calc(var(--spacing) * 3)}:where(.space-y-1>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 1) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 1) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-1\.5>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 1.5) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 1.5) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-2>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 2) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 2) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-3>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 3) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 3) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-4>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 4) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 4) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-6>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 6) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 6) * calc(1 - var(--tw-space-y-reverse)))}.gap-x-4{column-gap:calc(var(--spacing) * 4)}.gap-y-1{row-gap:calc(var(--spacing) * 1)}:where(.divide-y>:not(:last-child)){--tw-divide-y-reverse:0;border-bottom-style:var(--tw-border-style);border-top-style:var(--tw-border-style);border-top-width:calc(1px * var(--tw-divide-y-reverse));border-bottom-width:calc(1px * calc(1 - var(--tw-divide-y-reverse)))}:where(.divide-border>:not(:last-child)){border-color:hsl(var(--border))}.self-center{align-self:center}.self-end{align-self:flex-end}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-auto{overflow:auto}.overflow-hidden{overflow:hidden}.overflow-y-auto{overflow-y:auto}.rounded{border-radius:.25rem}.rounded-full{border-radius:3.40282e+38px}.rounded-lg{border-radius:var(--radius-lg)}.rounded-md{border-radius:var(--radius-md)}.rounded-xl{border-radius:var(--radius-xl)}.rounded-br-sm{border-bottom-right-radius:var(--radius-sm)}.rounded-bl-sm{border-bottom-left-radius:var(--radius-sm)}.border{border-style:var(--tw-border-style);border-width:1px}.border-2{border-style:var(--tw-border-style);border-width:2px}.border-t{border-top-style:var(--tw-border-style);border-top-width:1px}.border-r{border-right-style:var(--tw-border-style);border-right-width:1px}.border-b{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}.border-l{border-left-style:var(--tw-border-style);border-left-width:1px}.border-l-2{border-left-style:var(--tw-border-style);border-left-width:2px}.border-dashed{--tw-border-style:dashed;border-style:dashed}.border-accent\/30{border-color:hsl(var(--accent))}@supports (color:color-mix(in lab,red,red)){.border-accent\/30{border-color:color-mix(in oklab,hsl(var(--accent)) 30%,transparent)}}.border-border,.border-border\/50{border-color:hsl(var(--border))}@supports (color:color-mix(in lab,red,red)){.border-border\/50{border-color:color-mix(in oklab,hsl(var(--border)) 50%,transparent)}}.border-destructive\/30{border-color:hsl(var(--destructive))}@supports (color:color-mix(in lab,red,red)){.border-destructive\/30{border-color:color-mix(in oklab,hsl(var(--destructive)) 30%,transparent)}}.border-destructive\/40{border-color:hsl(var(--destructive))}@supports (color:color-mix(in lab,red,red)){.border-destructive\/40{border-color:color-mix(in oklab,hsl(var(--destructive)) 40%,transparent)}}.border-muted-foreground\/30{border-color:hsl(var(--muted-foreground))}@supports (color:color-mix(in lab,red,red)){.border-muted-foreground\/30{border-color:color-mix(in oklab,hsl(var(--muted-foreground)) 30%,transparent)}}.border-primary,.border-primary\/30{border-color:hsl(var(--primary))}@supports (color:color-mix(in lab,red,red)){.border-primary\/30{border-color:color-mix(in oklab,hsl(var(--primary)) 30%,transparent)}}.border-success\/30{border-color:hsl(var(--success))}@supports (color:color-mix(in lab,red,red)){.border-success\/30{border-color:color-mix(in oklab,hsl(var(--success)) 30%,transparent)}}.border-success\/40{border-color:hsl(var(--success))}@supports (color:color-mix(in lab,red,red)){.border-success\/40{border-color:color-mix(in oklab,hsl(var(--success)) 40%,transparent)}}.border-transparent{border-color:#0000}.border-warning\/30{border-color:hsl(var(--warning))}@supports (color:color-mix(in lab,red,red)){.border-warning\/30{border-color:color-mix(in oklab,hsl(var(--warning)) 30%,transparent)}}.border-warning\/40{border-color:hsl(var(--warning))}@supports (color:color-mix(in lab,red,red)){.border-warning\/40{border-color:color-mix(in oklab,hsl(var(--warning)) 40%,transparent)}}.border-t-transparent{border-top-color:#0000}.border-l-primary{border-left-color:hsl(var(--primary))}.border-l-transparent{border-left-color:#0000}.bg-accent,.bg-accent\/15{background-color:hsl(var(--accent))}@supports (color:color-mix(in lab,red,red)){.bg-accent\/15{background-color:color-mix(in oklab,hsl(var(--accent)) 15%,transparent)}}.bg-amber-400\/10{background-color:#fcbb001a}@supports (color:color-mix(in lab,red,red)){.bg-amber-400\/10{background-color:color-mix(in oklab,var(--color-amber-400) 10%,transparent)}}.bg-background,.bg-background\/60{background-color:hsl(var(--background))}@supports (color:color-mix(in lab,red,red)){.bg-background\/60{background-color:color-mix(in oklab,hsl(var(--background)) 60%,transparent)}}.bg-black\/40{background-color:#0006}@supports (color:color-mix(in lab,red,red)){.bg-black\/40{background-color:color-mix(in oklab,var(--color-black) 40%,transparent)}}.bg-black\/50{background-color:#00000080}@supports (color:color-mix(in lab,red,red)){.bg-black\/50{background-color:color-mix(in oklab,var(--color-black) 50%,transparent)}}.bg-black\/60{background-color:#0009}@supports (color:color-mix(in lab,red,red)){.bg-black\/60{background-color:color-mix(in oklab,var(--color-black) 60%,transparent)}}.bg-card{background-color:hsl(var(--card))}.bg-card-hover{background-color:hsl(var(--card-hover))}.bg-cyan-400{background-color:var(--color-cyan-400)}.bg-cyan-400\/10{background-color:#00d2ef1a}@supports (color:color-mix(in lab,red,red)){.bg-cyan-400\/10{background-color:color-mix(in oklab,var(--color-cyan-400) 10%,transparent)}}.bg-destructive,.bg-destructive\/5{background-color:hsl(var(--destructive))}@supports (color:color-mix(in lab,red,red)){.bg-destructive\/5{background-color:color-mix(in oklab,hsl(var(--destructive)) 5%,transparent)}}.bg-destructive\/15{background-color:hsl(var(--destructive))}@supports (color:color-mix(in lab,red,red)){.bg-destructive\/15{background-color:color-mix(in oklab,hsl(var(--destructive)) 15%,transparent)}}.bg-destructive\/20{background-color:hsl(var(--destructive))}@supports (color:color-mix(in lab,red,red)){.bg-destructive\/20{background-color:color-mix(in oklab,hsl(var(--destructive)) 20%,transparent)}}.bg-emerald-400{background-color:var(--color-emerald-400)}.bg-emerald-400\/10{background-color:#00d2941a}@supports (color:color-mix(in lab,red,red)){.bg-emerald-400\/10{background-color:color-mix(in oklab,var(--color-emerald-400) 10%,transparent)}}.bg-foreground\/5{background-color:hsl(var(--foreground))}@supports (color:color-mix(in lab,red,red)){.bg-foreground\/5{background-color:color-mix(in oklab,hsl(var(--foreground)) 5%,transparent)}}.bg-indigo-400{background-color:var(--color-indigo-400)}.bg-indigo-400\/10{background-color:#7d87ff1a}@supports (color:color-mix(in lab,red,red)){.bg-indigo-400\/10{background-color:color-mix(in oklab,var(--color-indigo-400) 10%,transparent)}}.bg-input{background-color:hsl(var(--input))}.bg-muted{background-color:hsl(var(--muted))}.bg-muted-foreground,.bg-muted-foreground\/40{background-color:hsl(var(--muted-foreground))}@supports (color:color-mix(in lab,red,red)){.bg-muted-foreground\/40{background-color:color-mix(in oklab,hsl(var(--muted-foreground)) 40%,transparent)}}.bg-muted\/50{background-color:hsl(var(--muted))}@supports (color:color-mix(in lab,red,red)){.bg-muted\/50{background-color:color-mix(in oklab,hsl(var(--muted)) 50%,transparent)}}.bg-muted\/90{background-color:hsl(var(--muted))}@supports (color:color-mix(in lab,red,red)){.bg-muted\/90{background-color:color-mix(in oklab,hsl(var(--muted)) 90%,transparent)}}.bg-primary,.bg-primary\/15{background-color:hsl(var(--primary))}@supports (color:color-mix(in lab,red,red)){.bg-primary\/15{background-color:color-mix(in oklab,hsl(var(--primary)) 15%,transparent)}}.bg-primary\/20{background-color:hsl(var(--primary))}@supports (color:color-mix(in lab,red,red)){.bg-primary\/20{background-color:color-mix(in oklab,hsl(var(--primary)) 20%,transparent)}}.bg-red-400{background-color:var(--color-red-400)}.bg-red-400\/10{background-color:#ff65681a}@supports (color:color-mix(in lab,red,red)){.bg-red-400\/10{background-color:color-mix(in oklab,var(--color-red-400) 10%,transparent)}}.bg-success,.bg-success\/5{background-color:hsl(var(--success))}@supports (color:color-mix(in lab,red,red)){.bg-success\/5{background-color:color-mix(in oklab,hsl(var(--success)) 5%,transparent)}}.bg-success\/15{background-color:hsl(var(--success))}@supports (color:color-mix(in lab,red,red)){.bg-success\/15{background-color:color-mix(in oklab,hsl(var(--success)) 15%,transparent)}}.bg-transparent{background-color:#0000}.bg-warning,.bg-warning\/10{background-color:hsl(var(--warning))}@supports (color:color-mix(in lab,red,red)){.bg-warning\/10{background-color:color-mix(in oklab,hsl(var(--warning)) 10%,transparent)}}.bg-warning\/15{background-color:hsl(var(--warning))}@supports (color:color-mix(in lab,red,red)){.bg-warning\/15{background-color:color-mix(in oklab,hsl(var(--warning)) 15%,transparent)}}.bg-warning\/20{background-color:hsl(var(--warning))}@supports (color:color-mix(in lab,red,red)){.bg-warning\/20{background-color:color-mix(in oklab,hsl(var(--warning)) 20%,transparent)}}.p-1{padding:calc(var(--spacing) * 1)}.p-2{padding:calc(var(--spacing) * 2)}.p-3{padding:calc(var(--spacing) * 3)}.p-4{padding:calc(var(--spacing) * 4)}.p-5{padding:calc(var(--spacing) * 5)}.p-6{padding:calc(var(--spacing) * 6)}.p-8{padding:calc(var(--spacing) * 8)}.px-0\.5{padding-inline:calc(var(--spacing) * .5)}.px-1{padding-inline:calc(var(--spacing) * 1)}.px-1\.5{padding-inline:calc(var(--spacing) * 1.5)}.px-2{padding-inline:calc(var(--spacing) * 2)}.px-3{padding-inline:calc(var(--spacing) * 3)}.px-4{padding-inline:calc(var(--spacing) * 4)}.px-5{padding-inline:calc(var(--spacing) * 5)}.px-6{padding-inline:calc(var(--spacing) * 6)}.py-0{padding-block:calc(var(--spacing) * 0)}.py-0\.5{padding-block:calc(var(--spacing) * .5)}.py-1{padding-block:calc(var(--spacing) * 1)}.py-1\.5{padding-block:calc(var(--spacing) * 1.5)}.py-2{padding-block:calc(var(--spacing) * 2)}.py-2\.5{padding-block:calc(var(--spacing) * 2.5)}.py-3{padding-block:calc(var(--spacing) * 3)}.py-4{padding-block:calc(var(--spacing) * 4)}.py-6{padding-block:calc(var(--spacing) * 6)}.py-8{padding-block:calc(var(--spacing) * 8)}.py-16{padding-block:calc(var(--spacing) * 16)}.py-20{padding-block:calc(var(--spacing) * 20)}.py-\[3px\]{padding-block:3px}.pt-2{padding-top:calc(var(--spacing) * 2)}.pt-\[14vh\]{padding-top:14vh}.pt-\[16vh\]{padding-top:16vh}.pt-\[18vh\]{padding-top:18vh}.pr-2{padding-right:calc(var(--spacing) * 2)}.pr-3{padding-right:calc(var(--spacing) * 3)}.pb-1{padding-bottom:calc(var(--spacing) * 1)}.pb-4{padding-bottom:calc(var(--spacing) * 4)}.pl-1{padding-left:calc(var(--spacing) * 1)}.pl-2{padding-left:calc(var(--spacing) * 2)}.pl-4{padding-left:calc(var(--spacing) * 4)}.text-center{text-align:center}.text-left{text-align:left}.text-right{text-align:right}.font-mono{font-family:var(--font-geist-mono)}.text-2xl{font-size:var(--text-2xl);line-height:var(--tw-leading,var(--text-2xl--line-height))}.text-base{font-size:var(--text-base);line-height:var(--tw-leading,var(--text-base--line-height))}.text-lg{font-size:var(--text-lg);line-height:var(--tw-leading,var(--text-lg--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xl{font-size:var(--text-xl);line-height:var(--tw-leading,var(--text-xl--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.text-\[10px\]{font-size:10px}.leading-none{--tw-leading:1;line-height:1}.leading-relaxed{--tw-leading:var(--leading-relaxed);line-height:var(--leading-relaxed)}.font-bold{--tw-font-weight:var(--font-weight-bold);font-weight:var(--font-weight-bold)}.font-medium{--tw-font-weight:var(--font-weight-medium);font-weight:var(--font-weight-medium)}.font-normal{--tw-font-weight:var(--font-weight-normal);font-weight:var(--font-weight-normal)}.font-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-tight{--tw-tracking:var(--tracking-tight);letter-spacing:var(--tracking-tight)}.tracking-wide{--tw-tracking:var(--tracking-wide);letter-spacing:var(--tracking-wide)}.tracking-wider{--tw-tracking:var(--tracking-wider);letter-spacing:var(--tracking-wider)}.break-words{overflow-wrap:break-word}.break-all{word-break:break-all}.whitespace-nowrap{white-space:nowrap}.whitespace-pre-wrap{white-space:pre-wrap}.text-accent{color:hsl(var(--accent))}.text-amber-400{color:var(--color-amber-400)}.text-border{color:hsl(var(--border))}.text-cyan-400{color:var(--color-cyan-400)}.text-destructive,.text-destructive\/60{color:hsl(var(--destructive))}@supports (color:color-mix(in lab,red,red)){.text-destructive\/60{color:color-mix(in oklab,hsl(var(--destructive)) 60%,transparent)}}.text-emerald-400{color:var(--color-emerald-400)}.text-foreground{color:hsl(var(--foreground))}.text-indigo-400{color:var(--color-indigo-400)}.text-muted-foreground,.text-muted-foreground\/0{color:hsl(var(--muted-foreground))}@supports (color:color-mix(in lab,red,red)){.text-muted-foreground\/0{color:color-mix(in oklab,hsl(var(--muted-foreground)) 0%,transparent)}}.text-muted-foreground\/30{color:hsl(var(--muted-foreground))}@supports (color:color-mix(in lab,red,red)){.text-muted-foreground\/30{color:color-mix(in oklab,hsl(var(--muted-foreground)) 30%,transparent)}}.text-muted-foreground\/40{color:hsl(var(--muted-foreground))}@supports (color:color-mix(in lab,red,red)){.text-muted-foreground\/40{color:color-mix(in oklab,hsl(var(--muted-foreground)) 40%,transparent)}}.text-muted-foreground\/50{color:hsl(var(--muted-foreground))}@supports (color:color-mix(in lab,red,red)){.text-muted-foreground\/50{color:color-mix(in oklab,hsl(var(--muted-foreground)) 50%,transparent)}}.text-muted-foreground\/60{color:hsl(var(--muted-foreground))}@supports (color:color-mix(in lab,red,red)){.text-muted-foreground\/60{color:color-mix(in oklab,hsl(var(--muted-foreground)) 60%,transparent)}}.text-muted-foreground\/70{color:hsl(var(--muted-foreground))}@supports (color:color-mix(in lab,red,red)){.text-muted-foreground\/70{color:color-mix(in oklab,hsl(var(--muted-foreground)) 70%,transparent)}}.text-muted-foreground\/80{color:hsl(var(--muted-foreground))}@supports (color:color-mix(in lab,red,red)){.text-muted-foreground\/80{color:color-mix(in oklab,hsl(var(--muted-foreground)) 80%,transparent)}}.text-primary{color:hsl(var(--primary))}.text-red-400{color:var(--color-red-400)}.text-success{color:hsl(var(--success))}.text-warning{color:hsl(var(--warning))}.text-white{color:var(--color-white)}.uppercase{text-transform:uppercase}.italic{font-style:italic}.tabular-nums{--tw-numeric-spacing:tabular-nums;font-variant-numeric:var(--tw-ordinal,) var(--tw-slashed-zero,) var(--tw-numeric-figure,) var(--tw-numeric-spacing,) var(--tw-numeric-fraction,)}.line-through{text-decoration-line:line-through}.underline{text-decoration-line:underline}.antialiased{-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.opacity-0{opacity:0}.opacity-40{opacity:.4}.opacity-50{opacity:.5}.shadow-2xl{--tw-shadow:0 25px 50px -12px var(--tw-shadow-color,#00000040)}.shadow-2xl,.shadow-lg{box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-lg{--tw-shadow:0 10px 15px -3px var(--tw-shadow-color,#0000001a),0 4px 6px -4px var(--tw-shadow-color,#0000001a)}.shadow-sm{--tw-shadow:0 1px 3px 0 var(--tw-shadow-color,#0000001a),0 1px 2px -1px var(--tw-shadow-color,#0000001a)}.shadow-sm,.shadow-xl{box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-xl{--tw-shadow:0 20px 25px -5px var(--tw-shadow-color,#0000001a),0 8px 10px -6px var(--tw-shadow-color,#0000001a)}.ring-2{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.shadow-black\/40{--tw-shadow-color:#0006}@supports (color:color-mix(in lab,red,red)){.shadow-black\/40{--tw-shadow-color:color-mix(in oklab,color-mix(in oklab,var(--color-black) 40%,transparent) var(--tw-shadow-alpha),transparent)}}.ring-current{--tw-ring-color:currentcolor}.ring-offset-1{--tw-ring-offset-width:1px;--tw-ring-offset-shadow:var(--tw-ring-inset,) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color)}.ring-offset-transparent{--tw-ring-offset-color:transparent}.outline{outline-style:var(--tw-outline-style);outline-width:1px}.filter{filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}.backdrop-blur-\[2px\]{--tw-backdrop-blur:blur(2px)}.backdrop-blur-\[2px\],.backdrop-blur-sm{-webkit-backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,);backdrop-filter:var(--tw-backdrop-blur,) var(--tw-backdrop-brightness,) var(--tw-backdrop-contrast,) var(--tw-backdrop-grayscale,) var(--tw-backdrop-hue-rotate,) var(--tw-backdrop-invert,) var(--tw-backdrop-opacity,) var(--tw-backdrop-saturate,) var(--tw-backdrop-sepia,)}.backdrop-blur-sm{--tw-backdrop-blur:blur(var(--blur-sm))}.transition-all{transition-property:all;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-colors{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}.transition-opacity{transition-property:opacity;transition-timing-function:var(--tw-ease,var(--default-transition-timing-function));transition-duration:var(--tw-duration,var(--default-transition-duration))}@media (hover:hover){.group-hover\:text-muted-foreground:is(:where(.group):hover *){color:hsl(var(--muted-foreground))}.group-hover\:text-primary:is(:where(.group):hover *){color:hsl(var(--primary))}.group-hover\:opacity-100:is(:where(.group):hover *){opacity:1}.group-hover\/task\:text-muted-foreground:is(:where(.group\/task):hover *){color:hsl(var(--muted-foreground))}}.placeholder\:text-muted-foreground\/40::placeholder{color:hsl(var(--muted-foreground))}@supports (color:color-mix(in lab,red,red)){.placeholder\:text-muted-foreground\/40::placeholder{color:color-mix(in oklab,hsl(var(--muted-foreground)) 40%,transparent)}}@media (hover:hover){.hover\:scale-\[1\.02\]:hover{scale:1.02}.hover\:border-muted-foreground:hover,.hover\:border-muted-foreground\/30:hover{border-color:hsl(var(--muted-foreground))}@supports (color:color-mix(in lab,red,red)){.hover\:border-muted-foreground\/30:hover{border-color:color-mix(in oklab,hsl(var(--muted-foreground)) 30%,transparent)}}.hover\:border-primary:hover{border-color:hsl(var(--primary))}.hover\:bg-accent\/25:hover{background-color:hsl(var(--accent))}@supports (color:color-mix(in lab,red,red)){.hover\:bg-accent\/25:hover{background-color:color-mix(in oklab,hsl(var(--accent)) 25%,transparent)}}.hover\:bg-accent\/80:hover{background-color:hsl(var(--accent))}@supports (color:color-mix(in lab,red,red)){.hover\:bg-accent\/80:hover{background-color:color-mix(in oklab,hsl(var(--accent)) 80%,transparent)}}.hover\:bg-background:hover{background-color:hsl(var(--background))}.hover\:bg-card-hover:hover,.hover\:bg-card-hover\/50:hover{background-color:hsl(var(--card-hover))}@supports (color:color-mix(in lab,red,red)){.hover\:bg-card-hover\/50:hover{background-color:color-mix(in oklab,hsl(var(--card-hover)) 50%,transparent)}}.hover\:bg-destructive\/10:hover{background-color:hsl(var(--destructive))}@supports (color:color-mix(in lab,red,red)){.hover\:bg-destructive\/10:hover{background-color:color-mix(in oklab,hsl(var(--destructive)) 10%,transparent)}}.hover\:bg-destructive\/80:hover{background-color:hsl(var(--destructive))}@supports (color:color-mix(in lab,red,red)){.hover\:bg-destructive\/80:hover{background-color:color-mix(in oklab,hsl(var(--destructive)) 80%,transparent)}}.hover\:bg-muted:hover,.hover\:bg-muted\/30:hover{background-color:hsl(var(--muted))}@supports (color:color-mix(in lab,red,red)){.hover\:bg-muted\/30:hover{background-color:color-mix(in oklab,hsl(var(--muted)) 30%,transparent)}}.hover\:bg-primary-hover:hover{background-color:hsl(var(--primary-hover))}.hover\:bg-primary\/10:hover{background-color:hsl(var(--primary))}@supports (color:color-mix(in lab,red,red)){.hover\:bg-primary\/10:hover{background-color:color-mix(in oklab,hsl(var(--primary)) 10%,transparent)}}.hover\:bg-success\/25:hover{background-color:hsl(var(--success))}@supports (color:color-mix(in lab,red,red)){.hover\:bg-success\/25:hover{background-color:color-mix(in oklab,hsl(var(--success)) 25%,transparent)}}.hover\:bg-warning\/25:hover{background-color:hsl(var(--warning))}@supports (color:color-mix(in lab,red,red)){.hover\:bg-warning\/25:hover{background-color:color-mix(in oklab,hsl(var(--warning)) 25%,transparent)}}.hover\:\!text-destructive:hover{color:hsl(var(--destructive))!important}.hover\:text-destructive:hover{color:hsl(var(--destructive))}.hover\:text-foreground:hover{color:hsl(var(--foreground))}.hover\:text-muted-foreground:hover{color:hsl(var(--muted-foreground))}.hover\:text-primary:hover{color:hsl(var(--primary))}.hover\:underline:hover{text-decoration-line:underline}.hover\:opacity-80:hover{opacity:.8}.hover\:opacity-90:hover{opacity:.9}.hover\:shadow-md:hover{--tw-shadow:0 4px 6px -1px var(--tw-shadow-color,#0000001a),0 2px 4px -2px var(--tw-shadow-color,#0000001a);box-shadow:var(--tw-inset-shadow),var(--tw-inset-ring-shadow),var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}.hover\:shadow-black\/20:hover{--tw-shadow-color:#0003}@supports (color:color-mix(in lab,red,red)){.hover\:shadow-black\/20:hover{--tw-shadow-color:color-mix(in oklab,color-mix(in oklab,var(--color-black) 20%,transparent) var(--tw-shadow-alpha),transparent)}}.hover\:brightness-110:hover{--tw-brightness:brightness(110%);filter:var(--tw-blur,) var(--tw-brightness,) var(--tw-contrast,) var(--tw-grayscale,) var(--tw-hue-rotate,) var(--tw-invert,) var(--tw-saturate,) var(--tw-sepia,) var(--tw-drop-shadow,)}}.focus\:border-primary:focus{border-color:hsl(var(--primary))}.focus\:outline-none:focus{--tw-outline-style:none;outline-style:none}.active\:cursor-grabbing:active{cursor:grabbing}.disabled\:opacity-40:disabled{opacity:.4}.disabled\:opacity-50:disabled{opacity:.5}@media (min-width:40rem){.sm\:grid-cols-2{grid-template-columns:repeat(2,minmax(0,1fr))}}@media (min-width:64rem){.lg\:grid-cols-3{grid-template-columns:repeat(3,minmax(0,1fr))}}}:root{--background:224 20% 10%;--foreground:210 40% 98%;--card:224 20% 13%;--card-hover:220 15% 18%;--border:220 15% 28%;--input:220 15% 24%;--primary:210 85% 60%;--primary-hover:210 85% 50%;--accent:265 70% 60%;--success:142 71% 45%;--warning:38 92% 50%;--destructive:0 70% 55%;--muted:220 15% 20%;--muted-foreground:215 15% 60%}body{background:hsl(var(--background));color:hsl(var(--foreground));font-family:var(--font-sans);-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;font-weight:500}::-webkit-scrollbar{width:6px}::-webkit-scrollbar-track{background:hsl(var(--background))}::-webkit-scrollbar-thumb{background:hsl(var(--border));border-radius:3px}::-webkit-scrollbar-thumb:hover{background:hsl(var(--muted-foreground))}input:focus,textarea:focus{box-shadow:0 0 0 2px hsl(var(--primary)/.3);outline:none}.editor-container{flex:1;min-height:0;display:flex;position:relative}.memo-overlay{pointer-events:none;position:absolute;inset:0;overflow:hidden}.memo-pin{pointer-events:auto;cursor:pointer;z-index:10;position:absolute}.memo-pin-icon{filter:drop-shadow(0 1px 2px #0000004d);font-size:14px;transition:transform .15s}.memo-pin:hover .memo-pin-icon{transform:scale(1.3)}.memo-tooltip{background:hsl(var(--card));border:1px solid hsl(var(--border));z-index:20;border-radius:8px;width:240px;padding:8px 12px;position:absolute;top:-4px;right:24px;box-shadow:0 4px 12px #0000004d}.memo-tooltip-anchor{color:hsl(var(--muted-foreground));text-overflow:ellipsis;white-space:nowrap;margin-bottom:4px;font-size:11px;font-style:italic;overflow:hidden}.memo-tooltip-question{color:hsl(var(--foreground));font-size:12px;line-height:1.4}.memo-popover{background:hsl(var(--card));border:1px solid hsl(var(--border));z-index:30;border-radius:10px;flex-direction:column;width:300px;max-height:320px;display:flex;position:absolute;top:-4px;right:24px;box-shadow:0 8px 24px #0006}.memo-popover-anchor{color:hsl(var(--muted-foreground));text-overflow:ellipsis;white-space:nowrap;border-bottom:1px solid hsl(var(--border));padding:8px 12px 4px;font-size:11px;font-style:italic;overflow:hidden}.memo-popover-messages{flex-direction:column;flex:1;gap:6px;min-height:0;padding:8px;display:flex;overflow-y:auto}.memo-popover-bubble{word-break:break-word;border-radius:10px;max-width:90%;padding:6px 10px;font-size:12px;line-height:1.5}.memo-popover-bubble-ai{background:hsl(var(--muted));color:hsl(var(--foreground));border-bottom-left-radius:3px;align-self:flex-start}.memo-popover-bubble-user{background:hsl(var(--accent));color:#fff;border-bottom-right-radius:3px;align-self:flex-end}.memo-popover-input-area{border-top:1px solid hsl(var(--border));gap:6px;padding:6px 8px;display:flex}.memo-popover-input{background:hsl(var(--input));border:1px solid hsl(var(--border));color:hsl(var(--foreground));resize:none;font-size:12px;font-family:var(--font-sans);border-radius:6px;flex:1;padding:5px 10px}.memo-popover-input::placeholder{color:hsl(var(--muted-foreground)/.5)}.memo-popover-send{background:hsl(var(--accent));color:#fff;cursor:pointer;border:none;border-radius:6px;flex-shrink:0;padding:5px 10px;font-size:11px;font-weight:600;transition:opacity .15s}.memo-popover-send:hover:not(:disabled){opacity:.85}.memo-popover-send:disabled{opacity:.4;cursor:not-allowed}.resize-handle{cursor:row-resize;background:hsl(var(--background));border-top:1px solid hsl(var(--border));border-bottom:1px solid hsl(var(--border));flex-shrink:0;justify-content:center;align-items:center;height:6px;transition:background .15s;display:flex}.resize-handle:hover{background:hsl(var(--muted))}.resize-handle-bar{background:hsl(var(--muted-foreground)/.4);border-radius:1px;width:32px;height:2px}.resize-handle:hover .resize-handle-bar{background:hsl(var(--muted-foreground))}.chat-panel{flex-direction:column;height:100%;min-height:0;display:flex}.chat-header{border-bottom:1px solid hsl(var(--border));flex-shrink:0;justify-content:space-between;align-items:center;padding:8px 16px;display:flex}.chat-messages{flex-direction:column;flex:1;gap:8px;min-height:0;padding:12px 16px;display:flex;overflow-y:auto}.chat-empty{height:100%;color:hsl(var(--muted-foreground));justify-content:center;align-items:center;font-size:12px;display:flex}.chat-message{flex-direction:column;max-width:85%;display:flex}.chat-message-ai{align-self:flex-start}.chat-message-user{align-self:flex-end}.chat-bubble{word-break:break-word;border-radius:12px;padding:8px 12px;font-size:13px;line-height:1.5}.chat-bubble-ai{background:hsl(var(--muted));color:hsl(var(--foreground));border-bottom-left-radius:4px}.chat-bubble-user{background:hsl(var(--accent));color:#fff;border-bottom-right-radius:4px}.chat-time{color:hsl(var(--muted-foreground));margin-top:2px;padding:0 4px;font-size:10px}.chat-message-user .chat-time{text-align:right}.chat-loading{gap:4px;padding:12px 16px;display:flex}.chat-loading .dot{background:hsl(var(--muted-foreground));border-radius:50%;width:6px;height:6px;animation:chatDotBounce 1.4s ease-in-out infinite both}.chat-loading .dot:first-child{animation-delay:-.32s}.chat-loading .dot:nth-child(2){animation-delay:-.16s}.chat-loading .dot:nth-child(3){animation-delay:0s}@keyframes chatDotBounce{0%,80%,to{opacity:.4;transform:scale(.6)}40%{opacity:1;transform:scale(1)}}.chat-input-area{border-top:1px solid hsl(var(--border));flex-shrink:0;gap:8px;padding:8px 12px;display:flex}.chat-input{background:hsl(var(--input));border:1px solid hsl(var(--border));color:hsl(var(--foreground));resize:none;font-size:13px;font-family:var(--font-sans);border-radius:8px;flex:1;padding:6px 12px}.chat-input::placeholder{color:hsl(var(--muted-foreground)/.5)}.chat-send-btn{background:hsl(var(--accent));color:#fff;cursor:pointer;border:none;border-radius:8px;flex-shrink:0;padding:6px 14px;font-size:12px;font-weight:600;transition:opacity .15s}.chat-send-btn:hover:not(:disabled){opacity:.85}.chat-send-btn:disabled{opacity:.4;cursor:not-allowed}.pin-toggle{cursor:pointer;background:0 0;border:none;flex-shrink:0;padding:0;font-size:12px;transition:all .15s}.pin-toggle:hover{transform:scale(1.2)}.tree-action-btn{color:hsl(var(--muted-foreground));background:hsl(var(--muted));border:1px solid hsl(var(--border));cursor:pointer;border-radius:4px;flex-shrink:0;padding:1px 6px;font-size:10px;transition:all .15s}.tree-action-btn:hover{color:hsl(var(--foreground));background:hsl(var(--card-hover))}.tree-action-btn-danger:hover{color:hsl(var(--destructive));border-color:hsl(var(--destructive)/.3)}.refine-popover{background:hsl(var(--card));border:1px solid hsl(var(--border));border-radius:10px;flex-direction:column;max-height:300px;margin:4px 0 8px;display:flex;box-shadow:0 4px 16px #0000004d}.refine-header{border-bottom:1px solid hsl(var(--border));color:hsl(var(--foreground));justify-content:space-between;align-items:center;padding:8px 12px;display:flex}.refine-close{color:hsl(var(--muted-foreground));cursor:pointer;background:0 0;border:none;padding:0 4px;font-size:16px;line-height:1}.refine-close:hover{color:hsl(var(--foreground))}.refine-context{color:hsl(var(--muted-foreground));border-bottom:1px solid hsl(var(--border));padding:6px 12px;font-size:11px;line-height:1.4}.refine-messages{flex-direction:column;flex:1;gap:6px;min-height:40px;padding:8px;display:flex;overflow-y:auto}.lock-toggle{cursor:pointer;background:0 0;border:none;flex-shrink:0;padding:0;font-size:14px;transition:transform .15s}.lock-toggle:hover:not(:disabled){transform:scale(1.2)}.lock-toggle:disabled{opacity:.4;cursor:not-allowed}.status-badge{cursor:pointer;background:0 0;border:none;border-radius:4px;flex-shrink:0;align-items:center;gap:2px;padding:1px 4px;font-size:12px;transition:background .15s;display:flex}.status-badge:hover:not(:disabled){background:hsl(var(--muted))}.status-badge-label{color:hsl(var(--muted-foreground));font-size:10px}.item-detail{background:hsl(var(--card));border:1px solid hsl(var(--border));border-radius:8px;margin:0 8px 8px;padding:12px}.prompt-section{border-top:1px solid hsl(var(--border));padding-top:8px}.prompt-header{justify-content:space-between;align-items:center;margin-bottom:6px;display:flex}.prompt-action-btn{color:hsl(var(--muted-foreground));background:hsl(var(--muted));border:1px solid hsl(var(--border));cursor:pointer;border-radius:4px;padding:2px 8px;font-size:10px;transition:all .15s}.prompt-action-btn:hover:not(:disabled){color:hsl(var(--foreground));background:hsl(var(--card-hover))}.prompt-action-btn:disabled{opacity:.4;cursor:not-allowed}.prompt-generate-btn{color:hsl(var(--accent));border-color:hsl(var(--accent)/.3)}.prompt-generate-btn:hover:not(:disabled){background:hsl(var(--accent)/.1);color:hsl(var(--accent))}.prompt-content{border:1px solid hsl(var(--border));white-space:pre-wrap;border-radius:6px;padding:8px 12px}.prompt-content,.prompt-edit-textarea{color:hsl(var(--foreground));background:hsl(var(--background));font-size:14px;line-height:1.7;font-family:var(--font-mono)}.prompt-edit-textarea{width:100%;border:1px solid hsl(var(--accent)/.5);resize:vertical;border-radius:6px;padding:10px 14px}.tree-node-row{cursor:pointer;border-radius:6px;align-items:flex-start;gap:6px;padding:6px 8px;transition:background .15s;display:flex}.tree-node-row:hover{background:hsl(var(--card-hover))}.tree-priority-dot{border-radius:50%;flex-shrink:0;width:6px;height:6px;margin-top:1px}.tree-icon-btn{cursor:pointer;background:0 0;border:none;flex-shrink:0;padding:1px 2px;font-size:11px;transition:transform .15s}.tree-icon-btn:hover{transform:scale(1.2)}.tree-progress-badge{color:hsl(var(--muted-foreground));background:hsl(var(--muted));font-variant-numeric:tabular-nums;border-radius:8px;padding:0 5px;font-size:9px;line-height:16px}.tree-children-group{border-left:1px solid hsl(var(--border)/.5);margin-left:3px}.card-grid{grid-template-columns:repeat(auto-fill,minmax(280px,1fr));gap:12px;padding:12px;display:grid}.project-card{background:hsl(var(--card));border:1px solid hsl(var(--border));border-radius:12px;transition:border-color .2s,box-shadow .2s;overflow:hidden}.project-card:hover{border-color:hsl(var(--muted-foreground)/.3);box-shadow:0 4px 12px #0003}.project-card-header{align-items:flex-start;gap:8px;padding:12px 14px 8px;display:flex}.project-card-icon{flex-shrink:0;margin-top:1px;font-size:20px}.project-card-title{flex:1;min-width:0;font-size:14px;font-weight:600;line-height:1.3}.project-card-desc{color:hsl(var(--muted-foreground));-webkit-line-clamp:2;-webkit-box-orient:vertical;padding:0 14px 8px;font-size:11px;line-height:1.4;display:-webkit-box;overflow:hidden}.project-card-progress{background:hsl(var(--muted));border-radius:2px;height:3px;margin:0 14px;overflow:hidden}.project-card-progress-fill{border-radius:2px;height:100%;transition:width .3s}.project-card-stats{color:hsl(var(--muted-foreground));align-items:center;gap:10px;padding:8px 14px;font-size:10px;display:flex}.project-card-stat{align-items:center;gap:3px;display:flex}.project-card-children{border-top:1px solid hsl(var(--border));max-height:240px;overflow-y:auto}.project-card-child{align-items:center;gap:6px;padding:5px 14px;font-size:12px;transition:background .1s;display:flex}.project-card-child:hover{background:hsl(var(--card-hover))}.project-card-child-done{opacity:.4;text-decoration:line-through}.project-card-expand{border-top:1px solid hsl(var(--border));text-align:center;color:hsl(var(--muted-foreground));cursor:pointer;padding:6px 14px;font-size:11px;transition:background .15s,color .15s}.project-card-expand:hover{background:hsl(var(--card-hover));color:hsl(var(--foreground))}.view-toggle{background:hsl(var(--muted));border-radius:6px;padding:2px;display:flex}.view-toggle-btn{color:hsl(var(--muted-foreground));cursor:pointer;background:0 0;border:none;border-radius:4px;padding:2px 8px;font-size:11px;transition:all .15s}.view-toggle-btn-active,.view-toggle-btn:hover{color:hsl(var(--foreground))}.view-toggle-btn-active{background:hsl(var(--card));box-shadow:0 1px 3px #0003}@keyframes scanIn{0%{opacity:0;transform:translate(-8px)}to{opacity:1;transform:translate(0)}}.animate-scan-in{animation:scanIn .2s ease-out}@keyframes progressIndeterminate{0%{width:0;margin-left:0}50%{width:40%;margin-left:30%}to{width:0;margin-left:100%}}.animate-progress-indeterminate{animation:progressIndeterminate 1.5s ease-in-out infinite}.tab-bar{background:hsl(var(--background));border-bottom:1px solid hsl(var(--border));-webkit-overflow-scrolling:touch;flex-shrink:0;align-items:center;gap:0;height:36px;padding:0 4px;display:flex;overflow-x:auto}.tab-bar::-webkit-scrollbar{height:0}.tab-item{color:hsl(var(--muted-foreground));cursor:pointer;white-space:nowrap;-webkit-user-select:none;user-select:none;border-bottom:2px solid #0000;flex-shrink:0;align-items:center;gap:6px;max-width:180px;padding:6px 12px;font-size:13px;transition:all .15s;display:flex}.tab-item:hover{color:hsl(var(--foreground));background:hsl(var(--muted))}.tab-item-active{color:hsl(var(--foreground));border-bottom-color:hsl(var(--primary))}.tab-close{color:hsl(var(--muted-foreground));opacity:0;border-radius:3px;padding:0 2px;font-size:15px;line-height:1;transition:opacity .1s,color .1s}.tab-item:hover .tab-close{opacity:1}.tab-close:hover{color:hsl(var(--destructive));background:hsl(var(--destructive)/.15)}.tab-bar-spacer{flex:1;min-width:8px}.theme-picker-btn{color:hsl(var(--muted-foreground));border:1px solid hsl(var(--border));background:hsl(var(--card));cursor:pointer;border-radius:6px;flex-shrink:0;align-items:center;gap:5px;margin-right:4px;padding:4px 8px;font-size:12px;transition:all .15s;display:flex}.theme-picker-btn:hover{color:hsl(var(--foreground));border-color:hsl(var(--primary)/.5);background:hsl(var(--card-hover))}.theme-picker-preview{border-radius:50%;flex-shrink:0;width:10px;height:10px}.theme-picker-dropdown{z-index:9999;background:hsl(var(--card));border:1px solid hsl(var(--border));border-radius:10px;min-width:240px;padding:10px;animation:dialogIn .15s ease-out;box-shadow:0 8px 24px #0000004d}.theme-picker-title{color:hsl(var(--muted-foreground));text-transform:uppercase;letter-spacing:.05em;margin-bottom:8px;padding:0 2px;font-size:11px;font-weight:600}.theme-picker-grid{grid-template-columns:1fr 1fr;gap:6px;display:grid}.theme-picker-item{background:hsl(var(--muted));cursor:pointer;text-align:left;border:1.5px solid #0000;border-radius:8px;flex-direction:column;gap:5px;padding:6px;transition:all .15s;display:flex}.theme-picker-item:hover{border-color:hsl(var(--primary)/.4)}.theme-picker-item-active{border-color:hsl(var(--primary));background:hsl(var(--primary)/.1)}.theme-picker-swatch{border-radius:4px;height:20px;display:flex;overflow:hidden}.theme-picker-name{color:hsl(var(--foreground));font-size:11px;font-weight:500}.panel-resize-handle{cursor:col-resize;z-index:10;background:0 0;flex-shrink:0;justify-content:center;align-items:center;width:6px;transition:background .15s;display:flex;position:relative}.panel-resize-handle:active,.panel-resize-handle:hover{background:hsl(var(--primary)/.15)}.panel-resize-handle-bar{background:hsl(var(--muted-foreground)/.2);border-radius:1px;width:2px;height:32px;transition:background .15s}.panel-resize-handle:active .panel-resize-handle-bar,.panel-resize-handle:hover .panel-resize-handle-bar{background:hsl(var(--primary)/.6)}.chat-markdown p{margin:.25em 0}.chat-markdown p:first-child{margin-top:0}.chat-markdown p:last-child{margin-bottom:0}.chat-markdown strong{color:hsl(var(--foreground));font-weight:700}.chat-markdown code{background:hsl(var(--background));font-size:.9em;font-family:var(--font-mono);border-radius:4px;padding:1px 5px}.chat-markdown pre{background:hsl(var(--background));border-radius:6px;margin:.5em 0;padding:8px 12px;overflow-x:auto}.chat-markdown pre code{background:0 0;padding:0}.chat-markdown ol,.chat-markdown ul{margin:.3em 0;padding-left:1.4em}.chat-markdown li{margin:.15em 0}.chat-markdown h1,.chat-markdown h2,.chat-markdown h3,.chat-markdown h4,.chat-markdown h5,.chat-markdown h6{margin:.5em 0 .25em;font-weight:700;line-height:1.3}.chat-markdown h1{font-size:1.2em}.chat-markdown h2{font-size:1.1em}.chat-markdown h3{font-size:1.05em}.chat-markdown blockquote{border-left:3px solid hsl(var(--border));color:hsl(var(--muted-foreground));margin:.4em 0;padding-left:10px}.chat-markdown hr{border:none;border-top:1px solid hsl(var(--border));margin:.5em 0}.chat-markdown table{border-collapse:collapse;width:100%;margin:.5em 0;font-size:.85em}.chat-markdown td,.chat-markdown th{border:1px solid hsl(var(--border));text-align:left;padding:4px 8px}.chat-markdown th{background:hsl(var(--muted));font-weight:600}.chat-markdown tr:nth-child(2n){background:hsl(var(--muted)/.3)}@keyframes dialogIn{0%{opacity:0;transform:scale(.95)translateY(4px)}to{opacity:1;transform:scale(1)translateY(0)}}.animate-dialog-in{animation:dialogIn .15s ease-out}@keyframes drawerIn{0%{transform:translate(100%)}to{transform:translate(0)}}.animate-drawer-in{animation:drawerIn .2s ease-out}@property --tw-translate-x{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-y{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-z{syntax:"*";inherits:false;initial-value:0}@property --tw-scale-x{syntax:"*";inherits:false;initial-value:1}@property --tw-scale-y{syntax:"*";inherits:false;initial-value:1}@property --tw-scale-z{syntax:"*";inherits:false;initial-value:1}@property --tw-rotate-x{syntax:"*";inherits:false}@property --tw-rotate-y{syntax:"*";inherits:false}@property --tw-rotate-z{syntax:"*";inherits:false}@property --tw-skew-x{syntax:"*";inherits:false}@property --tw-skew-y{syntax:"*";inherits:false}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-divide-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-leading{syntax:"*";inherits:false}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false}@property --tw-ordinal{syntax:"*";inherits:false}@property --tw-slashed-zero{syntax:"*";inherits:false}@property --tw-numeric-figure{syntax:"*";inherits:false}@property --tw-numeric-spacing{syntax:"*";inherits:false}@property --tw-numeric-fraction{syntax:"*";inherits:false}@property --tw-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-shadow-color{syntax:"*";inherits:false}@property --tw-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-inset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-shadow-color{syntax:"*";inherits:false}@property --tw-inset-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-ring-color{syntax:"*";inherits:false}@property --tw-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-ring-color{syntax:"*";inherits:false}@property --tw-inset-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-ring-inset{syntax:"*";inherits:false}@property --tw-ring-offset-width{syntax:"<length>";inherits:false;initial-value:0}@property --tw-ring-offset-color{syntax:"*";inherits:false;initial-value:#fff}@property --tw-ring-offset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-outline-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-blur{syntax:"*";inherits:false}@property --tw-brightness{syntax:"*";inherits:false}@property --tw-contrast{syntax:"*";inherits:false}@property --tw-grayscale{syntax:"*";inherits:false}@property --tw-hue-rotate{syntax:"*";inherits:false}@property --tw-invert{syntax:"*";inherits:false}@property --tw-opacity{syntax:"*";inherits:false}@property --tw-saturate{syntax:"*";inherits:false}@property --tw-sepia{syntax:"*";inherits:false}@property --tw-drop-shadow{syntax:"*";inherits:false}@property --tw-drop-shadow-color{syntax:"*";inherits:false}@property --tw-drop-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-drop-shadow-size{syntax:"*";inherits:false}@property --tw-backdrop-blur{syntax:"*";inherits:false}@property --tw-backdrop-brightness{syntax:"*";inherits:false}@property --tw-backdrop-contrast{syntax:"*";inherits:false}@property --tw-backdrop-grayscale{syntax:"*";inherits:false}@property --tw-backdrop-hue-rotate{syntax:"*";inherits:false}@property --tw-backdrop-invert{syntax:"*";inherits:false}@property --tw-backdrop-opacity{syntax:"*";inherits:false}@property --tw-backdrop-saturate{syntax:"*";inherits:false}@property --tw-backdrop-sepia{syntax:"*";inherits:false}@keyframes spin{to{transform:rotate(1turn)}}@keyframes pulse{50%{opacity:.5}}@keyframes bounce{0%,to{animation-timing-function:cubic-bezier(.8,0,1,1);transform:translateY(-25%)}50%{animation-timing-function:cubic-bezier(0,0,.2,1);transform:none}}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "idea-manager",
3
- "version": "1.9.0",
3
+ "version": "2.1.0",
4
4
  "description": "Turn free-form brainstorming into structured task trees with AI-generated prompts. Built-in MCP Server for autonomous AI agent execution. Local-first with SQLite, cross-PC sync via Git.",
5
5
  "keywords": [
6
6
  "brainstorm",
@@ -0,0 +1,52 @@
1
+ import { NextRequest, NextResponse } from 'next/server';
2
+ import { ensureDb } from '@/lib/db';
3
+ import { createTask, updateTask, getTask } from '@/lib/db/queries/tasks';
4
+ import { getSubProject } from '@/lib/db/queries/sub-projects';
5
+ import type { AdvisorAction } from '@/types/advisor-actions';
6
+
7
+ interface ActionResult {
8
+ index: number;
9
+ success: boolean;
10
+ error?: string;
11
+ }
12
+
13
+ export async function POST(request: NextRequest) {
14
+ await ensureDb();
15
+ const body = await request.json();
16
+ const actions: AdvisorAction[] = body.actions;
17
+ if (!Array.isArray(actions) || actions.length === 0) {
18
+ return NextResponse.json({ error: 'actions array required' }, { status: 400 });
19
+ }
20
+
21
+ const results: ActionResult[] = [];
22
+
23
+ for (let i = 0; i < actions.length; i++) {
24
+ const action = actions[i];
25
+ try {
26
+ if (action.type === 'create_task') {
27
+ const sub = getSubProject(action.subProjectId);
28
+ if (!sub) { results.push({ index: i, success: false, error: `Sub-project ${action.subProjectId} not found` }); continue; }
29
+ createTask({
30
+ project_id: action.projectId || sub.project_id,
31
+ sub_project_id: action.subProjectId,
32
+ title: action.title,
33
+ description: action.description,
34
+ priority: action.priority,
35
+ status: action.status,
36
+ });
37
+ results.push({ index: i, success: true });
38
+ } else if (action.type === 'update_task') {
39
+ const task = getTask(action.taskId);
40
+ if (!task) { results.push({ index: i, success: false, error: `Task ${action.taskId} not found` }); continue; }
41
+ updateTask(action.taskId, action.changes);
42
+ results.push({ index: i, success: true });
43
+ } else {
44
+ results.push({ index: i, success: false, error: 'Unknown action type' });
45
+ }
46
+ } catch (err) {
47
+ results.push({ index: i, success: false, error: err instanceof Error ? err.message : String(err) });
48
+ }
49
+ }
50
+
51
+ return NextResponse.json({ results });
52
+ }
@@ -0,0 +1,50 @@
1
+ import { NextRequest, NextResponse } from 'next/server';
2
+ import { ensureDb } from '@/lib/db';
3
+ import { getGlobalConversations, addGlobalConversation, clearGlobalConversations } from '@/lib/db/queries/global-conversations';
4
+ import { buildGlobalAdvisorPrompt, trimHistory } from '@/lib/ai/global-context';
5
+ import { runAgent } from '@/lib/ai/client';
6
+
7
+ export async function GET() {
8
+ await ensureDb();
9
+ const messages = getGlobalConversations();
10
+ return NextResponse.json(messages);
11
+ }
12
+
13
+ export async function POST(request: NextRequest) {
14
+ await ensureDb();
15
+ const body = await request.json();
16
+ if (!body.message || typeof body.message !== 'string') {
17
+ return NextResponse.json({ error: 'message is required' }, { status: 400 });
18
+ }
19
+
20
+ const userMsg = addGlobalConversation('user', body.message);
21
+
22
+ const systemPrompt = buildGlobalAdvisorPrompt();
23
+ const history = getGlobalConversations();
24
+ const trimmed = trimHistory(history.map(m => ({ role: m.role, content: m.content })));
25
+ const conversationText = trimmed
26
+ .map(m => `${m.role === 'user' ? 'User' : m.role === 'system' ? 'System' : 'Assistant'}: ${m.content}`)
27
+ .join('\n');
28
+
29
+ const fullPrompt = `${systemPrompt}\n\n=== CONVERSATION ===\n${conversationText}`;
30
+
31
+ try {
32
+ const aiResponse = await runAgent('claude', fullPrompt, undefined, undefined, { timeoutMs: 120000 });
33
+ const trimmedResponse = aiResponse.trim();
34
+ if (!trimmedResponse) {
35
+ const fallback = addGlobalConversation('assistant', '(AI 응답을 생성하지 못했습니다.)');
36
+ return NextResponse.json({ userMessage: userMsg, aiMessage: fallback });
37
+ }
38
+ const aiMsg = addGlobalConversation('assistant', trimmedResponse);
39
+ return NextResponse.json({ userMessage: userMsg, aiMessage: aiMsg });
40
+ } catch {
41
+ const errorMsg = addGlobalConversation('assistant', '(AI 호출에 실패했습니다.)');
42
+ return NextResponse.json({ userMessage: userMsg, aiMessage: errorMsg });
43
+ }
44
+ }
45
+
46
+ export async function DELETE() {
47
+ await ensureDb();
48
+ clearGlobalConversations();
49
+ return NextResponse.json({ ok: true });
50
+ }
@@ -0,0 +1,36 @@
1
+ import { NextResponse } from 'next/server';
2
+ import { getDb } from '@/lib/db';
3
+ import { ensureDb } from '@/lib/db';
4
+
5
+ export const dynamic = 'force-dynamic';
6
+
7
+ /**
8
+ * Runs daily housekeeping tasks. Called once on app mount (from the dashboard).
9
+ *
10
+ * 1. Today auto-carry: un-done tasks that are still is_today=1 stay marked.
11
+ * Done/archived tasks with is_today=1 get cleared.
12
+ * 2. Done auto-archive: tasks with status='done' older than 14 days get archived.
13
+ */
14
+ export async function POST() {
15
+ await ensureDb();
16
+ const db = getDb();
17
+ const now = new Date().toISOString();
18
+
19
+ // 1. Clear is_today on finished tasks (done/archived stay, but no longer "today")
20
+ const clearedToday = db.prepare(`
21
+ UPDATE tasks SET is_today = 0, updated_at = ?
22
+ WHERE is_today = 1 AND (status = 'done' OR is_archived = 1)
23
+ `).run(now);
24
+
25
+ // 2. Auto-archive done tasks older than 14 days
26
+ const cutoff = new Date(Date.now() - 14 * 24 * 60 * 60 * 1000).toISOString();
27
+ const autoArchived = db.prepare(`
28
+ UPDATE tasks SET is_archived = 1, is_today = 0, updated_at = ?
29
+ WHERE status = 'done' AND is_archived = 0 AND updated_at < ?
30
+ `).run(now, cutoff);
31
+
32
+ return NextResponse.json({
33
+ todayCleared: clearedToday.changes ?? 0,
34
+ autoArchived: autoArchived.changes ?? 0,
35
+ });
36
+ }
@@ -0,0 +1,24 @@
1
+ import { NextRequest, NextResponse } from 'next/server';
2
+ import { getDb } from '@/lib/db';
3
+ import { ensureDb } from '@/lib/db';
4
+
5
+ export async function POST(
6
+ request: NextRequest,
7
+ { params }: { params: Promise<{ id: string; subId: string }> },
8
+ ) {
9
+ await ensureDb();
10
+ await params; // validate route
11
+ const body = await request.json() as { orderedIds?: string[] };
12
+ if (!Array.isArray(body.orderedIds) || body.orderedIds.length === 0) {
13
+ return NextResponse.json({ error: 'orderedIds array required' }, { status: 400 });
14
+ }
15
+
16
+ const db = getDb();
17
+ const now = new Date().toISOString();
18
+ const stmt = db.prepare('UPDATE tasks SET sort_order = ?, updated_at = ? WHERE id = ?');
19
+ for (let i = 0; i < body.orderedIds.length; i++) {
20
+ stmt.run(i, now, body.orderedIds[i]);
21
+ }
22
+
23
+ return NextResponse.json({ ok: true });
24
+ }
@@ -0,0 +1,30 @@
1
+ import { NextRequest, NextResponse } from 'next/server';
2
+ import { ensureDb } from '@/lib/db';
3
+ import { getTask, updateTask } from '@/lib/db/queries/tasks';
4
+ import { getSubProject } from '@/lib/db/queries/sub-projects';
5
+
6
+ export async function POST(
7
+ request: NextRequest,
8
+ { params }: { params: Promise<{ taskId: string }> },
9
+ ) {
10
+ await ensureDb();
11
+ const { taskId } = await params;
12
+ const body = await request.json() as { projectId?: string; subProjectId?: string };
13
+
14
+ if (!body.subProjectId) {
15
+ return NextResponse.json({ error: 'subProjectId is required' }, { status: 400 });
16
+ }
17
+
18
+ const task = getTask(taskId);
19
+ if (!task) return NextResponse.json({ error: 'Task not found' }, { status: 404 });
20
+
21
+ const targetSub = getSubProject(body.subProjectId);
22
+ if (!targetSub) return NextResponse.json({ error: 'Target sub-project not found' }, { status: 404 });
23
+
24
+ const updated = updateTask(taskId, {
25
+ project_id: targetSub.project_id,
26
+ sub_project_id: targetSub.id,
27
+ });
28
+
29
+ return NextResponse.json(updated);
30
+ }
@@ -0,0 +1,124 @@
1
+ 'use client';
2
+
3
+ import { useState, useCallback } from 'react';
4
+ import type { AdvisorAction, CreateTaskAction, UpdateTaskAction } from '@/types/advisor-actions';
5
+
6
+ function CreateRow({ action }: { action: CreateTaskAction }) {
7
+ return (
8
+ <div className="flex items-center gap-2">
9
+ <span className="text-success text-xs font-bold">+</span>
10
+ <span className="text-xs text-foreground truncate flex-1">{action.title}</span>
11
+ {action.priority && action.priority !== 'medium' && (
12
+ <span className={`text-[10px] px-1 rounded ${action.priority === 'high' ? 'bg-destructive/20 text-destructive' : 'bg-muted text-muted-foreground'}`}>
13
+ {action.priority}
14
+ </span>
15
+ )}
16
+ </div>
17
+ );
18
+ }
19
+
20
+ function UpdateRow({ action }: { action: UpdateTaskAction }) {
21
+ const changes = Object.entries(action.changes)
22
+ .filter(([, v]) => v !== undefined)
23
+ .map(([k, v]) => `${k}: ${v}`)
24
+ .join(', ');
25
+ return (
26
+ <div className="flex items-center gap-2">
27
+ <span className="text-warning text-xs font-bold">~</span>
28
+ <span className="text-xs text-muted-foreground truncate">task:{action.taskId.slice(0, 8)}</span>
29
+ <span className="text-xs text-foreground truncate flex-1">→ {changes}</span>
30
+ </div>
31
+ );
32
+ }
33
+
34
+ export default function ActionBlock({
35
+ actions,
36
+ onApplied,
37
+ }: {
38
+ actions: AdvisorAction[];
39
+ onApplied?: () => void;
40
+ }) {
41
+ const [status, setStatus] = useState<'pending' | 'applying' | 'applied' | 'error'>('pending');
42
+ const [error, setError] = useState<string | null>(null);
43
+ const [resultSummary, setResultSummary] = useState<string | null>(null);
44
+
45
+ const apply = useCallback(async () => {
46
+ setStatus('applying');
47
+ setError(null);
48
+ try {
49
+ const res = await fetch('/api/advisor-actions', {
50
+ method: 'POST',
51
+ headers: { 'Content-Type': 'application/json' },
52
+ body: JSON.stringify({ actions }),
53
+ });
54
+ const data = await res.json();
55
+ if (!res.ok) throw new Error(data.error || `HTTP ${res.status}`);
56
+ const results = data.results as { index: number; success: boolean; error?: string }[];
57
+ const ok = results.filter(r => r.success).length;
58
+ const fail = results.filter(r => !r.success).length;
59
+ if (fail > 0) {
60
+ setResultSummary(`${ok} 성공, ${fail} 실패`);
61
+ setStatus(ok > 0 ? 'applied' : 'error');
62
+ } else {
63
+ setResultSummary(`${ok}개 적용 완료`);
64
+ setStatus('applied');
65
+ }
66
+ onApplied?.();
67
+ window.dispatchEvent(new Event('advisor-action-applied'));
68
+ } catch (err) {
69
+ setError(err instanceof Error ? err.message : '적용 실패');
70
+ setStatus('error');
71
+ }
72
+ }, [actions, onApplied]);
73
+
74
+ const createCount = actions.filter(a => a.type === 'create_task').length;
75
+ const updateCount = actions.filter(a => a.type === 'update_task').length;
76
+
77
+ return (
78
+ <div className={`my-2 border rounded-lg text-xs ${
79
+ status === 'applied' ? 'border-success/30 bg-success/5' :
80
+ status === 'error' ? 'border-destructive/30 bg-destructive/5' :
81
+ 'border-border bg-card'
82
+ }`}>
83
+ <div className="px-3 py-2 border-b border-border flex items-center justify-between">
84
+ <div className="flex items-center gap-2 text-muted-foreground">
85
+ <span>Proposed Actions</span>
86
+ {createCount > 0 && <span className="text-success">+{createCount} create</span>}
87
+ {updateCount > 0 && <span className="text-warning">~{updateCount} update</span>}
88
+ </div>
89
+ {status === 'applied' && <span className="text-success">✓ {resultSummary}</span>}
90
+ {status === 'error' && <span className="text-destructive">⚠ {error || resultSummary}</span>}
91
+ </div>
92
+ <div className="px-3 py-2 space-y-1.5 max-h-[200px] overflow-y-auto">
93
+ {actions.map((a, i) => (
94
+ <div key={i}>
95
+ {a.type === 'create_task' ? <CreateRow action={a} /> : <UpdateRow action={a} />}
96
+ </div>
97
+ ))}
98
+ </div>
99
+ {status !== 'applied' && (
100
+ <div className="px-3 py-2 border-t border-border flex justify-end gap-2">
101
+ {status === 'error' && (
102
+ <button onClick={apply} className="px-2 py-1 text-foreground hover:text-primary transition-colors">
103
+ 재시도
104
+ </button>
105
+ )}
106
+ {status === 'pending' && (
107
+ <button
108
+ onClick={apply}
109
+ className="px-3 py-1 bg-primary text-primary-foreground rounded hover:opacity-90 transition-opacity"
110
+ >
111
+ 적용
112
+ </button>
113
+ )}
114
+ {status === 'applying' && (
115
+ <span className="flex items-center gap-1.5 text-muted-foreground">
116
+ <span className="inline-block w-1.5 h-1.5 rounded-full bg-warning animate-pulse" />
117
+ 적용 중…
118
+ </span>
119
+ )}
120
+ </div>
121
+ )}
122
+ </div>
123
+ );
124
+ }
@@ -0,0 +1,175 @@
1
+ 'use client';
2
+
3
+ import { useState, useEffect, useRef, useCallback } from 'react';
4
+ import type { IProjectConversation } from '@/types';
5
+ import { registerAiActivity, unregisterAiActivity } from '@/lib/ai-activity';
6
+ import { parseAdvisorContent } from '@/lib/advisor-actions/parse';
7
+ import ActionBlock from './ActionBlock';
8
+ import ReactMarkdown from 'react-markdown';
9
+ import remarkGfm from 'remark-gfm';
10
+
11
+ export default function AdvisorChat({
12
+ basePath,
13
+ title,
14
+ shortcutHint,
15
+ placeholder,
16
+ emptyIcon,
17
+ emptyHints,
18
+ activityType,
19
+ activityLabel,
20
+ onClose,
21
+ }: {
22
+ basePath: string;
23
+ title: string;
24
+ shortcutHint: string;
25
+ placeholder: string;
26
+ emptyIcon: string;
27
+ emptyHints: string[];
28
+ activityType: 'project-advisor' | 'global-advisor';
29
+ activityLabel: string;
30
+ onClose: () => void;
31
+ }) {
32
+ const [messages, setMessages] = useState<IProjectConversation[]>([]);
33
+ const [input, setInput] = useState('');
34
+ const [loading, setLoading] = useState(false);
35
+ const messagesEndRef = useRef<HTMLDivElement>(null);
36
+ const inputRef = useRef<HTMLTextAreaElement>(null);
37
+
38
+ const fetchMessages = useCallback(async () => {
39
+ try {
40
+ const res = await fetch(basePath);
41
+ if (!res.ok) return;
42
+ const data = await res.json();
43
+ if (Array.isArray(data)) setMessages(data);
44
+ } catch { /* silent */ }
45
+ }, [basePath]);
46
+
47
+ useEffect(() => { fetchMessages(); }, [fetchMessages]);
48
+ useEffect(() => { messagesEndRef.current?.scrollIntoView({ behavior: 'smooth' }); }, [messages]);
49
+ useEffect(() => { inputRef.current?.focus(); }, []);
50
+
51
+ useEffect(() => {
52
+ const onKey = (e: KeyboardEvent) => {
53
+ if (e.key === 'Escape') { e.preventDefault(); onClose(); }
54
+ };
55
+ window.addEventListener('keydown', onKey);
56
+ return () => window.removeEventListener('keydown', onKey);
57
+ }, [onClose]);
58
+
59
+ const send = useCallback(async () => {
60
+ const text = input.trim();
61
+ if (!text || loading) return;
62
+ setInput('');
63
+ setLoading(true);
64
+ const actId = `${activityType}-${Date.now()}`;
65
+ registerAiActivity({ id: actId, type: activityType, label: activityLabel, startedAt: Date.now() });
66
+ const tempId = `temp-${Date.now()}`;
67
+ setMessages(prev => [...prev, { id: tempId, project_id: '', role: 'user', content: text, created_at: new Date().toISOString() }]);
68
+ try {
69
+ const res = await fetch(basePath, {
70
+ method: 'POST',
71
+ headers: { 'Content-Type': 'application/json' },
72
+ body: JSON.stringify({ message: text }),
73
+ });
74
+ if (res.ok) {
75
+ const data = await res.json();
76
+ setMessages(prev => [...prev.filter(m => m.id !== tempId), data.userMessage, data.aiMessage]);
77
+ }
78
+ } catch { /* silent */ }
79
+ unregisterAiActivity(actId);
80
+ setLoading(false);
81
+ inputRef.current?.focus();
82
+ }, [input, loading, basePath, activityType, activityLabel]);
83
+
84
+ const handleClear = useCallback(async () => {
85
+ await fetch(basePath, { method: 'DELETE' });
86
+ setMessages([]);
87
+ inputRef.current?.focus();
88
+ }, [basePath]);
89
+
90
+ return (
91
+ <div className="fixed inset-0 z-[55] flex items-center justify-center" onClick={onClose}>
92
+ <div className="absolute inset-0 bg-black/40" style={{ backdropFilter: 'blur(2px)' }} />
93
+ <div
94
+ onClick={(e) => e.stopPropagation()}
95
+ className="relative w-[560px] max-w-[90vw] h-[80vh] max-h-[700px] bg-card border border-border rounded-xl shadow-2xl flex flex-col animate-dialog-in"
96
+ >
97
+ <div className="px-4 py-3 border-b border-border flex items-center justify-between flex-shrink-0">
98
+ <div className="flex items-center gap-2">
99
+ <span className="text-sm font-semibold text-foreground">{title}</span>
100
+ <span className="text-[10px] text-muted-foreground/60">{shortcutHint}</span>
101
+ </div>
102
+ <div className="flex items-center gap-2">
103
+ {messages.length > 0 && (
104
+ <button onClick={handleClear} className="text-xs text-muted-foreground hover:text-foreground transition-colors px-1">Clear</button>
105
+ )}
106
+ <button onClick={onClose} className="text-muted-foreground hover:text-foreground transition-colors text-lg leading-none">×</button>
107
+ </div>
108
+ </div>
109
+
110
+ <div className="flex-1 overflow-y-auto px-4 py-3 space-y-3 min-h-0">
111
+ {messages.length === 0 && !loading && (
112
+ <div className="flex flex-col items-center justify-center h-full text-center gap-3 text-muted-foreground">
113
+ <div className="text-2xl">{emptyIcon}</div>
114
+ {emptyHints.map((h, i) => (
115
+ <div key={i} className={i === 0 ? 'text-sm' : 'text-xs text-muted-foreground/70 max-w-[300px] leading-relaxed'}>{h}</div>
116
+ ))}
117
+ </div>
118
+ )}
119
+ {messages.filter(m => m.role !== 'system').map((msg) => (
120
+ <div key={msg.id} className={`flex flex-col ${msg.role === 'user' ? 'items-end' : 'items-start'}`}>
121
+ {msg.role === 'user' ? (
122
+ <div className="max-w-[92%] px-3 py-2 rounded-lg text-sm leading-relaxed bg-accent text-white rounded-br-sm whitespace-pre-wrap">
123
+ {msg.content}
124
+ </div>
125
+ ) : (
126
+ <div className="max-w-[92%] w-full">
127
+ {parseAdvisorContent(msg.content).segments.map((seg, i) =>
128
+ seg.type === 'markdown' ? (
129
+ <div key={i} className="px-3 py-2 rounded-lg bg-muted text-foreground rounded-bl-sm chat-markdown text-sm leading-relaxed">
130
+ <ReactMarkdown remarkPlugins={[remarkGfm]}>{seg.text}</ReactMarkdown>
131
+ </div>
132
+ ) : (
133
+ <ActionBlock key={i} actions={seg.actions} />
134
+ )
135
+ )}
136
+ </div>
137
+ )}
138
+ </div>
139
+ ))}
140
+ {loading && (
141
+ <div className="flex items-start">
142
+ <div className="px-3 py-2 rounded-lg bg-muted text-foreground rounded-bl-sm">
143
+ <div className="flex gap-1">
144
+ <div className="w-1.5 h-1.5 rounded-full bg-muted-foreground animate-bounce" style={{ animationDelay: '0ms' }} />
145
+ <div className="w-1.5 h-1.5 rounded-full bg-muted-foreground animate-bounce" style={{ animationDelay: '150ms' }} />
146
+ <div className="w-1.5 h-1.5 rounded-full bg-muted-foreground animate-bounce" style={{ animationDelay: '300ms' }} />
147
+ </div>
148
+ </div>
149
+ </div>
150
+ )}
151
+ <div ref={messagesEndRef} />
152
+ </div>
153
+
154
+ <div className="flex gap-1.5 px-3 py-3 border-t border-border flex-shrink-0">
155
+ <textarea
156
+ ref={inputRef}
157
+ value={input}
158
+ onChange={(e) => setInput(e.target.value)}
159
+ onKeyDown={(e) => { if (e.key === 'Enter' && !e.shiftKey) { e.preventDefault(); send(); } }}
160
+ placeholder={placeholder}
161
+ rows={2}
162
+ className="flex-1 bg-input border border-border rounded-md px-3 py-2 text-sm text-foreground resize-none focus:border-primary focus:outline-none"
163
+ />
164
+ <button
165
+ onClick={send}
166
+ disabled={!input.trim() || loading}
167
+ className="px-3 py-2 bg-accent text-white text-sm rounded-md disabled:opacity-40 hover:bg-accent/80 transition-colors flex-shrink-0 self-end"
168
+ >
169
+ Send
170
+ </button>
171
+ </div>
172
+ </div>
173
+ </div>
174
+ );
175
+ }