idea-manager 1.7.0 → 1.9.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.
- package/.next/build-manifest.json +2 -2
- package/.next/routes-manifest.json +8 -0
- package/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
- package/.next/server/app/_global-error.html +2 -2
- package/.next/server/app/_global-error.rsc +1 -1
- package/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/.next/server/app/_global-error.segments/_global-error/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/_global-error.segments/_global-error.segment.rsc +1 -1
- package/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/.next/server/app/_not-found.html +2 -2
- package/.next/server/app/_not-found.rsc +2 -2
- package/.next/server/app/_not-found.segments/_full.segment.rsc +2 -2
- package/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/.next/server/app/_not-found.segments/_index.segment.rsc +2 -2
- package/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/api/archive/route.js +12 -3
- package/.next/server/app/api/archive/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/filesystem/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/filesystem/tree/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/global-memo/route.js +9 -0
- package/.next/server/app/api/global-memo/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/health/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/projects/[id]/advisor/route.js +34 -0
- package/.next/server/app/api/projects/[id]/advisor/route_client-reference-manifest.js +1 -0
- package/.next/server/app/api/projects/[id]/apply-distribute/route.js +5 -5
- package/.next/server/app/api/projects/[id]/apply-distribute/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/projects/[id]/auto-distribute/route.js +3 -3
- package/.next/server/app/api/projects/[id]/auto-distribute/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/projects/[id]/brainstorm/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/projects/[id]/git-sync/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/projects/[id]/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/projects/[id]/sub-projects/[subId]/route.js +12 -3
- package/.next/server/app/api/projects/[id]/sub-projects/[subId]/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/chat/route.js +3 -3
- package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/chat/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/prompt/route.js +9 -0
- package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/prompt/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/refine/route.js +3 -3
- package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/refine/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/route.js +12 -3
- package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/route.js +12 -3
- package/.next/server/app/api/projects/[id]/sub-projects/[subId]/tasks/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/projects/[id]/sub-projects/route.js +12 -3
- package/.next/server/app/api/projects/[id]/sub-projects/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/projects/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/search/route.js +9 -0
- package/.next/server/app/api/search/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/sync/route.js +9 -0
- package/.next/server/app/api/sync/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/update/route_client-reference-manifest.js +1 -1
- package/.next/server/app/api/version/route_client-reference-manifest.js +1 -1
- package/.next/server/app/index.html +2 -2
- package/.next/server/app/index.rsc +3 -3
- package/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
- package/.next/server/app/index.segments/_full.segment.rsc +3 -3
- package/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/.next/server/app/index.segments/_index.segment.rsc +2 -2
- package/.next/server/app/index.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/page.js +6 -6
- package/.next/server/app/page_client-reference-manifest.js +1 -1
- package/.next/server/app/projects/[id]/page_client-reference-manifest.js +1 -1
- package/.next/server/app-paths-manifest.json +12 -11
- package/.next/server/chunks/117.js +9 -0
- package/.next/server/pages/404.html +2 -2
- package/.next/server/pages/500.html +2 -2
- package/.next/static/chunks/374-769431701aab500f.js +1 -0
- package/.next/static/chunks/app/_global-error/page-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/archive/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/filesystem/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/filesystem/tree/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/global-memo/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/health/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/projects/[id]/advisor/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/projects/[id]/apply-distribute/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/projects/[id]/auto-distribute/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/projects/[id]/brainstorm/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/projects/[id]/git-sync/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/projects/[id]/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/chat/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/prompt/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/refine/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/projects/[id]/sub-projects/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/projects/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/search/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/sync/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/update/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/api/version/route-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/app/{page-9a1dc101e82c397c.js → page-e935ee928da68ca2.js} +7 -7
- package/.next/static/chunks/next/dist/client/components/builtin/app-error-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/next/dist/client/components/builtin/forbidden-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/next/dist/client/components/builtin/not-found-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/chunks/next/dist/client/components/builtin/unauthorized-3ff8f59aaa75b8f8.js +1 -0
- package/.next/static/css/e4c7cd5a570312d9.css +3 -0
- package/README.ja.md +2 -0
- package/README.ko.md +3 -1
- package/README.md +3 -1
- package/README.zh.md +2 -0
- package/package.json +1 -1
- package/src/app/api/projects/[id]/advisor/route.ts +71 -0
- package/src/cli.ts +10 -0
- package/src/components/workspace/ProjectAdvisor.tsx +194 -0
- package/src/components/workspace/WorkspacePanel.tsx +23 -0
- package/src/lib/ai/project-context.ts +111 -0
- package/src/lib/auto-update.ts +136 -0
- package/src/lib/db/queries/project-conversations.ts +29 -0
- package/src/lib/db/schema.ts +11 -0
- package/src/types/index.ts +8 -0
- package/.next/static/chunks/374-23189d7e246ad164.js +0 -1
- package/.next/static/chunks/app/_global-error/page-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/archive/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/filesystem/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/filesystem/tree/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/global-memo/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/health/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/projects/[id]/apply-distribute/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/projects/[id]/auto-distribute/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/projects/[id]/brainstorm/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/projects/[id]/git-sync/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/projects/[id]/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/chat/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/prompt/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/refine/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/[taskId]/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/projects/[id]/sub-projects/[subId]/tasks/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/projects/[id]/sub-projects/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/projects/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/search/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/sync/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/update/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/app/api/version/route-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/next/dist/client/components/builtin/app-error-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/next/dist/client/components/builtin/forbidden-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/next/dist/client/components/builtin/not-found-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/chunks/next/dist/client/components/builtin/unauthorized-e6a77f238d2cdbb9.js +0 -1
- package/.next/static/css/eab748b03f49c43a.css +0 -3
- /package/.next/static/{mxrEVQX3r5YlDPZgpDvSp → pxqzEiwniZAUDOUTb5SnX}/_buildManifest.js +0 -0
- /package/.next/static/{mxrEVQX3r5YlDPZgpDvSp → pxqzEiwniZAUDOUTb5SnX}/_ssgManifest.js +0 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
(self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[2,64,86,94,123,167,175,177,178,202,212,307,363,437,460,514,595,711,770,772,819,822,851,860,896,922,980,988],{4441:()=>{}},_=>{_.O(0,[441,794,358],()=>_(_.s=4441)),_N_E=_.O()}]);
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
(self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[2,64,86,94,123,167,175,177,178,202,212,307,363,437,460,514,595,711,770,772,819,822,851,860,896,922,980,988],{4441:()=>{}},_=>{_.O(0,[441,794,358],()=>_(_.s=4441)),_N_E=_.O()}]);
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
(self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[2,64,86,94,123,167,175,177,178,202,212,307,363,437,460,514,595,711,770,772,819,822,851,860,896,922,980,988],{4441:()=>{}},_=>{_.O(0,[441,794,358],()=>_(_.s=4441)),_N_E=_.O()}]);
|
package/.next/static/chunks/next/dist/client/components/builtin/unauthorized-3ff8f59aaa75b8f8.js
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
(self.webpackChunk_N_E=self.webpackChunk_N_E||[]).push([[2,64,86,94,123,167,175,177,178,202,212,307,363,437,460,514,595,711,770,772,819,822,851,860,896,922,980,988],{4441:()=>{}},_=>{_.O(0,[441,794,358],()=>_(_.s=4441)),_N_E=_.O()}]);
|
|
@@ -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-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;--tw-ease: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-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;--ease-out:cubic-bezier(0,0,.2,1);--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)}.right-3{right:calc(var(--spacing) * 3)}.bottom-2{bottom:calc(var(--spacing) * 2)}.z-50{z-index:50}.z-\[60\]{z-index:60}.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}.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)}.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-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}.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-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}.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-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-\[420px\]{width:420px}.w-\[480px\]{width:480px}.w-\[520px\]{width:520px}.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-\[92\%\]{max-width:92%}.max-w-\[220px\]{max-width:220px}.max-w-\[300px\]{max-width:300px}.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}.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))}.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)))}: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-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\/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-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\/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\/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\/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)}.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-4{padding-left:calc(var(--spacing) * 4)}.text-center{text-align:center}.text-left{text-align:left}.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-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\/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))}.ease-out{--tw-ease:var(--ease-out);transition-timing-function:var(--ease-out)}@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\:\!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-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}@property --tw-ease{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/README.ja.md
CHANGED
|
@@ -2,6 +2,8 @@
|
|
|
2
2
|
|
|
3
3
|
[English](README.md) | [한국어](README.ko.md) | **日本語** | [中文](README.zh.md)
|
|
4
4
|
|
|
5
|
+
> **v1.8+**: `im start` を実行するだけで自動アップデートされます。旧バージョンの方は `npm install -g idea-manager@latest` を一度実行してください。
|
|
6
|
+
|
|
5
7
|
> 自由なブレインストーミングを構造化されたタスクツリーとAIプロンプトに変換
|
|
6
8
|
|
|
7
9
|
開発者向けのローカルファーストタスク管理ツール。アイデアをサブプロジェクトとタスクに整理し、各タスクにプロンプトを作成してAIエージェントに渡せます。内蔵MCPサーバーでAIエージェントの自律実行をサポート。Git経由のPC間同期。
|
package/README.ko.md
CHANGED
|
@@ -2,6 +2,8 @@
|
|
|
2
2
|
|
|
3
3
|
[English](README.md) | **한국어** | [日本語](README.ja.md) | [中文](README.zh.md)
|
|
4
4
|
|
|
5
|
+
> **v1.8+ 사용자**: `im start`만 실행하면 부팅 시 자동으로 최신 버전이 설치됩니다. 이전 버전을 쓰고 있다면 `npm install -g idea-manager@latest`를 한 번만 실행하세요 — 이후 모든 릴리즈는 자동 반영됩니다.
|
|
6
|
+
|
|
5
7
|
> 자유로운 브레인스토밍을 구조화된 태스크 트리와 AI 프롬프트로 변환
|
|
6
8
|
|
|
7
9
|
개발자를 위한 로컬 기반 태스크 관리 도구. 아이디어를 워크스페이스와 프로젝트로 정리하고, 각 태스크에 프롬프트를 작성한 뒤 AI 에이전트에게 넘길 수 있습니다. 내장 MCP 서버로 AI 에이전트의 자율 실행을 지원합니다. Git을 통한 PC 간 동기화.
|
|
@@ -15,7 +17,7 @@ npm install -g idea-manager
|
|
|
15
17
|
im start
|
|
16
18
|
```
|
|
17
19
|
|
|
18
|
-
네이티브 앱처럼 독립 윈도우로 열립니다 (Chrome/Edge `--app` 모드). 첫 실행 시 자동 빌드됩니다.
|
|
20
|
+
네이티브 앱처럼 독립 윈도우로 열립니다 (Chrome/Edge `--app` 모드). 첫 실행 시 자동 빌드됩니다. v1.8.0부터 `im start`가 부팅 시 npm 최신 확인 → 자동 설치합니다 (`IM_NO_AUTO_UPDATE=1`로 비활성화 가능).
|
|
19
21
|
|
|
20
22
|
## 핵심 워크플로우
|
|
21
23
|
|
package/README.md
CHANGED
|
@@ -2,6 +2,8 @@
|
|
|
2
2
|
|
|
3
3
|
**English** | [한국어](README.ko.md) | [日本語](README.ja.md) | [中文](README.zh.md)
|
|
4
4
|
|
|
5
|
+
> **v1.8+ users**: `im start` now auto-updates on boot. If you're on an older version, run `npm install -g idea-manager@latest` once — after that, every future release is applied automatically.
|
|
6
|
+
|
|
5
7
|
> Turn free-form brainstorming into structured task trees with AI-generated prompts.
|
|
6
8
|
|
|
7
9
|
A local-first task management tool for developers. Organize ideas into workspaces and projects, refine prompts for each task, and hand them off to AI agents. Built-in MCP Server enables autonomous AI agent execution. Cross-PC sync via Git.
|
|
@@ -15,7 +17,7 @@ npm install -g idea-manager
|
|
|
15
17
|
im start
|
|
16
18
|
```
|
|
17
19
|
|
|
18
|
-
Opens a native-like app window (Chrome/Edge `--app` mode). First run builds automatically.
|
|
20
|
+
Opens a native-like app window (Chrome/Edge `--app` mode). First run builds automatically. From v1.8.0, `im start` checks npm on boot and auto-upgrades when a newer version is available (`IM_NO_AUTO_UPDATE=1` to opt out).
|
|
19
21
|
|
|
20
22
|
## Core Workflow
|
|
21
23
|
|
package/README.zh.md
CHANGED
|
@@ -2,6 +2,8 @@
|
|
|
2
2
|
|
|
3
3
|
[English](README.md) | [한국어](README.ko.md) | [日本語](README.ja.md) | **中文**
|
|
4
4
|
|
|
5
|
+
> **v1.8+**: 运行 `im start` 时自动检查并安装最新版本。旧版本用户请先执行 `npm install -g idea-manager@latest`。
|
|
6
|
+
|
|
5
7
|
> 将自由头脑风暴转化为结构化任务树和AI提示词
|
|
6
8
|
|
|
7
9
|
面向开发者的本地优先任务管理工具。将想法组织成子项目和任务,为每个任务编写提示词,然后交给AI代理执行。内置MCP服务器支持AI代理自主执行。通过Git实现跨PC同步。
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "idea-manager",
|
|
3
|
-
"version": "1.
|
|
3
|
+
"version": "1.9.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,71 @@
|
|
|
1
|
+
import { NextRequest, NextResponse } from 'next/server';
|
|
2
|
+
import { ensureDb } from '@/lib/db';
|
|
3
|
+
import { getProject } from '@/lib/db/queries/projects';
|
|
4
|
+
import { getProjectConversations, addProjectConversation, clearProjectConversations } from '@/lib/db/queries/project-conversations';
|
|
5
|
+
import { buildProjectAdvisorPrompt, trimConversationHistory } from '@/lib/ai/project-context';
|
|
6
|
+
import { runAgent } from '@/lib/ai/client';
|
|
7
|
+
|
|
8
|
+
export async function GET(
|
|
9
|
+
_request: NextRequest,
|
|
10
|
+
{ params }: { params: Promise<{ id: string }> },
|
|
11
|
+
) {
|
|
12
|
+
await ensureDb();
|
|
13
|
+
const { id } = await params;
|
|
14
|
+
const messages = getProjectConversations(id);
|
|
15
|
+
return NextResponse.json(messages);
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
export async function POST(
|
|
19
|
+
request: NextRequest,
|
|
20
|
+
{ params }: { params: Promise<{ id: string }> },
|
|
21
|
+
) {
|
|
22
|
+
await ensureDb();
|
|
23
|
+
const { id } = await params;
|
|
24
|
+
const body = await request.json();
|
|
25
|
+
|
|
26
|
+
if (!body.message || typeof body.message !== 'string') {
|
|
27
|
+
return NextResponse.json({ error: 'message is required' }, { status: 400 });
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
const project = getProject(id);
|
|
31
|
+
if (!project) {
|
|
32
|
+
return NextResponse.json({ error: 'Project not found' }, { status: 404 });
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
const userMsg = addProjectConversation(id, 'user', body.message);
|
|
36
|
+
|
|
37
|
+
const systemPrompt = buildProjectAdvisorPrompt(id);
|
|
38
|
+
const history = getProjectConversations(id);
|
|
39
|
+
const trimmed = trimConversationHistory(history.map(m => ({ role: m.role, content: m.content })));
|
|
40
|
+
const conversationText = trimmed
|
|
41
|
+
.map(m => `${m.role === 'user' ? 'User' : m.role === 'system' ? 'System' : 'Assistant'}: ${m.content}`)
|
|
42
|
+
.join('\n');
|
|
43
|
+
|
|
44
|
+
const fullPrompt = `${systemPrompt}\n\n=== CONVERSATION ===\n${conversationText}`;
|
|
45
|
+
|
|
46
|
+
try {
|
|
47
|
+
const agentType = project.agent_type || 'claude';
|
|
48
|
+
const cwd = project.project_path || undefined;
|
|
49
|
+
const aiResponse = await runAgent(agentType, fullPrompt, undefined, undefined, { cwd, timeoutMs: 120000 });
|
|
50
|
+
const trimmedResponse = aiResponse.trim();
|
|
51
|
+
if (!trimmedResponse) {
|
|
52
|
+
const fallback = addProjectConversation(id, 'assistant', '(AI 응답을 생성하지 못했습니다. 다시 시도해주세요.)');
|
|
53
|
+
return NextResponse.json({ userMessage: userMsg, aiMessage: fallback });
|
|
54
|
+
}
|
|
55
|
+
const aiMsg = addProjectConversation(id, 'assistant', trimmedResponse);
|
|
56
|
+
return NextResponse.json({ userMessage: userMsg, aiMessage: aiMsg });
|
|
57
|
+
} catch {
|
|
58
|
+
const errorMsg = addProjectConversation(id, 'assistant', '(AI 호출에 실패했습니다. 다시 시도해주세요.)');
|
|
59
|
+
return NextResponse.json({ userMessage: userMsg, aiMessage: errorMsg });
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
export async function DELETE(
|
|
64
|
+
_request: NextRequest,
|
|
65
|
+
{ params }: { params: Promise<{ id: string }> },
|
|
66
|
+
) {
|
|
67
|
+
await ensureDb();
|
|
68
|
+
const { id } = await params;
|
|
69
|
+
clearProjectConversations(id);
|
|
70
|
+
return NextResponse.json({ ok: true });
|
|
71
|
+
}
|
package/src/cli.ts
CHANGED
|
@@ -10,6 +10,7 @@ import { getTaskPrompt } from './lib/db/queries/task-prompts';
|
|
|
10
10
|
import type { McpToolContext } from './lib/mcp/tools';
|
|
11
11
|
import { startWatcher } from './lib/watcher';
|
|
12
12
|
import { syncInit, syncPush, syncPull, syncStatus } from './lib/sync/index';
|
|
13
|
+
import { maybeAutoUpdate, respawnSelf } from './lib/auto-update';
|
|
13
14
|
import { spawn } from 'child_process';
|
|
14
15
|
import path from 'path';
|
|
15
16
|
import { fileURLToPath } from 'url';
|
|
@@ -118,6 +119,15 @@ program
|
|
|
118
119
|
const port = opts.port;
|
|
119
120
|
const fs = await import('fs');
|
|
120
121
|
|
|
122
|
+
// Auto-update on boot — fetches npm latest, installs if newer, respawns.
|
|
123
|
+
// Only runs for `im start`; `im mcp` / `im watch` stay untouched because
|
|
124
|
+
// they may be long-running integrations we shouldn't disrupt.
|
|
125
|
+
const upd = await maybeAutoUpdate(PKG_ROOT);
|
|
126
|
+
if (upd.upgraded) {
|
|
127
|
+
respawnSelf();
|
|
128
|
+
return;
|
|
129
|
+
}
|
|
130
|
+
|
|
121
131
|
// Resolve next CLI
|
|
122
132
|
let nextCli: string;
|
|
123
133
|
try {
|
|
@@ -0,0 +1,194 @@
|
|
|
1
|
+
'use client';
|
|
2
|
+
|
|
3
|
+
import { useState, useEffect, useRef, useCallback } from 'react';
|
|
4
|
+
import type { IProjectConversation } from '@/types';
|
|
5
|
+
import ReactMarkdown from 'react-markdown';
|
|
6
|
+
import remarkGfm from 'remark-gfm';
|
|
7
|
+
|
|
8
|
+
export default function ProjectAdvisor({
|
|
9
|
+
projectId,
|
|
10
|
+
onClose,
|
|
11
|
+
}: {
|
|
12
|
+
projectId: string;
|
|
13
|
+
onClose: () => void;
|
|
14
|
+
}) {
|
|
15
|
+
const [messages, setMessages] = useState<IProjectConversation[]>([]);
|
|
16
|
+
const [input, setInput] = useState('');
|
|
17
|
+
const [loading, setLoading] = useState(false);
|
|
18
|
+
const messagesEndRef = useRef<HTMLDivElement>(null);
|
|
19
|
+
const inputRef = useRef<HTMLTextAreaElement>(null);
|
|
20
|
+
const basePath = `/api/projects/${projectId}/advisor`;
|
|
21
|
+
|
|
22
|
+
const fetchMessages = useCallback(async () => {
|
|
23
|
+
try {
|
|
24
|
+
const res = await fetch(basePath);
|
|
25
|
+
if (!res.ok) return;
|
|
26
|
+
const data = await res.json();
|
|
27
|
+
if (Array.isArray(data)) setMessages(data);
|
|
28
|
+
} catch { /* silent */ }
|
|
29
|
+
}, [basePath]);
|
|
30
|
+
|
|
31
|
+
useEffect(() => {
|
|
32
|
+
fetchMessages();
|
|
33
|
+
}, [fetchMessages]);
|
|
34
|
+
|
|
35
|
+
useEffect(() => {
|
|
36
|
+
messagesEndRef.current?.scrollIntoView({ behavior: 'smooth' });
|
|
37
|
+
}, [messages]);
|
|
38
|
+
|
|
39
|
+
useEffect(() => {
|
|
40
|
+
inputRef.current?.focus();
|
|
41
|
+
}, []);
|
|
42
|
+
|
|
43
|
+
useEffect(() => {
|
|
44
|
+
const onKey = (e: KeyboardEvent) => {
|
|
45
|
+
if (e.key === 'Escape') { e.preventDefault(); onClose(); }
|
|
46
|
+
};
|
|
47
|
+
window.addEventListener('keydown', onKey);
|
|
48
|
+
return () => window.removeEventListener('keydown', onKey);
|
|
49
|
+
}, [onClose]);
|
|
50
|
+
|
|
51
|
+
const send = useCallback(async () => {
|
|
52
|
+
const text = input.trim();
|
|
53
|
+
if (!text || loading) return;
|
|
54
|
+
setInput('');
|
|
55
|
+
setLoading(true);
|
|
56
|
+
|
|
57
|
+
const tempId = `temp-${Date.now()}`;
|
|
58
|
+
const userMsg: IProjectConversation = {
|
|
59
|
+
id: tempId, project_id: projectId, role: 'user', content: text,
|
|
60
|
+
created_at: new Date().toISOString(),
|
|
61
|
+
};
|
|
62
|
+
setMessages(prev => [...prev, userMsg]);
|
|
63
|
+
|
|
64
|
+
try {
|
|
65
|
+
const res = await fetch(basePath, {
|
|
66
|
+
method: 'POST',
|
|
67
|
+
headers: { 'Content-Type': 'application/json' },
|
|
68
|
+
body: JSON.stringify({ message: text }),
|
|
69
|
+
});
|
|
70
|
+
if (res.ok) {
|
|
71
|
+
const data = await res.json();
|
|
72
|
+
setMessages(prev => {
|
|
73
|
+
const withoutTemp = prev.filter(m => m.id !== tempId);
|
|
74
|
+
return [...withoutTemp, data.userMessage, data.aiMessage];
|
|
75
|
+
});
|
|
76
|
+
}
|
|
77
|
+
} catch { /* silent */ }
|
|
78
|
+
setLoading(false);
|
|
79
|
+
inputRef.current?.focus();
|
|
80
|
+
}, [input, loading, basePath, projectId]);
|
|
81
|
+
|
|
82
|
+
const handleClear = useCallback(async () => {
|
|
83
|
+
await fetch(basePath, { method: 'DELETE' });
|
|
84
|
+
setMessages([]);
|
|
85
|
+
inputRef.current?.focus();
|
|
86
|
+
}, [basePath]);
|
|
87
|
+
|
|
88
|
+
const handleKeyDown = (e: React.KeyboardEvent) => {
|
|
89
|
+
if (e.key === 'Enter' && !e.shiftKey) {
|
|
90
|
+
e.preventDefault();
|
|
91
|
+
send();
|
|
92
|
+
}
|
|
93
|
+
};
|
|
94
|
+
|
|
95
|
+
return (
|
|
96
|
+
<div className="fixed inset-0 z-50 flex justify-end" onClick={onClose}>
|
|
97
|
+
<div className="absolute inset-0 bg-black/40" style={{ backdropFilter: 'blur(2px)' }} />
|
|
98
|
+
|
|
99
|
+
<div
|
|
100
|
+
onClick={(e) => e.stopPropagation()}
|
|
101
|
+
className="relative w-[480px] max-w-[85vw] h-full bg-card border-l border-border shadow-2xl flex flex-col"
|
|
102
|
+
style={{ animation: 'dialog-in 0.15s ease-out' }}
|
|
103
|
+
>
|
|
104
|
+
{/* Header */}
|
|
105
|
+
<div className="px-4 py-3 border-b border-border flex items-center justify-between flex-shrink-0">
|
|
106
|
+
<div className="flex items-center gap-2">
|
|
107
|
+
<span className="text-sm font-semibold text-foreground">Project Advisor</span>
|
|
108
|
+
<span className="text-[10px] text-muted-foreground/60">⌘L</span>
|
|
109
|
+
</div>
|
|
110
|
+
<div className="flex items-center gap-2">
|
|
111
|
+
{messages.length > 0 && (
|
|
112
|
+
<button
|
|
113
|
+
onClick={handleClear}
|
|
114
|
+
className="text-xs text-muted-foreground hover:text-foreground transition-colors px-1"
|
|
115
|
+
title="대화 초기화"
|
|
116
|
+
>
|
|
117
|
+
Clear
|
|
118
|
+
</button>
|
|
119
|
+
)}
|
|
120
|
+
<button
|
|
121
|
+
onClick={onClose}
|
|
122
|
+
className="text-muted-foreground hover:text-foreground transition-colors text-lg leading-none"
|
|
123
|
+
>
|
|
124
|
+
×
|
|
125
|
+
</button>
|
|
126
|
+
</div>
|
|
127
|
+
</div>
|
|
128
|
+
|
|
129
|
+
{/* Messages */}
|
|
130
|
+
<div className="flex-1 overflow-y-auto px-4 py-3 space-y-3 min-h-0">
|
|
131
|
+
{messages.length === 0 && !loading && (
|
|
132
|
+
<div className="flex flex-col items-center justify-center h-full text-center gap-3 text-muted-foreground">
|
|
133
|
+
<div className="text-2xl">🧭</div>
|
|
134
|
+
<div className="text-sm">프로젝트 전체 맥락을 보고 답합니다</div>
|
|
135
|
+
<div className="text-xs text-muted-foreground/70 max-w-[300px] leading-relaxed">
|
|
136
|
+
"다음 뭐부터 하면 좋겠어?"<br />
|
|
137
|
+
"빠진 작업 없나?"<br />
|
|
138
|
+
"이번 주 진행 상황 정리해줘"<br />
|
|
139
|
+
"이 방향이 맞는지 검토해줘"
|
|
140
|
+
</div>
|
|
141
|
+
</div>
|
|
142
|
+
)}
|
|
143
|
+
{messages.filter(m => m.role !== 'system').map((msg) => (
|
|
144
|
+
<div key={msg.id} className={`flex flex-col ${msg.role === 'user' ? 'items-end' : 'items-start'}`}>
|
|
145
|
+
<div className={`max-w-[92%] px-3 py-2 rounded-lg text-sm leading-relaxed ${
|
|
146
|
+
msg.role === 'user'
|
|
147
|
+
? 'bg-accent text-white rounded-br-sm whitespace-pre-wrap'
|
|
148
|
+
: 'bg-muted text-foreground rounded-bl-sm chat-markdown'
|
|
149
|
+
}`}>
|
|
150
|
+
{msg.role === 'assistant'
|
|
151
|
+
? <ReactMarkdown remarkPlugins={[remarkGfm]}>{msg.content}</ReactMarkdown>
|
|
152
|
+
: msg.content}
|
|
153
|
+
</div>
|
|
154
|
+
</div>
|
|
155
|
+
))}
|
|
156
|
+
{loading && (
|
|
157
|
+
<div className="flex items-start">
|
|
158
|
+
<div className="px-3 py-2 rounded-lg bg-muted text-foreground rounded-bl-sm">
|
|
159
|
+
<div className="flex gap-1">
|
|
160
|
+
<div className="w-1.5 h-1.5 rounded-full bg-muted-foreground animate-bounce" style={{ animationDelay: '0ms' }} />
|
|
161
|
+
<div className="w-1.5 h-1.5 rounded-full bg-muted-foreground animate-bounce" style={{ animationDelay: '150ms' }} />
|
|
162
|
+
<div className="w-1.5 h-1.5 rounded-full bg-muted-foreground animate-bounce" style={{ animationDelay: '300ms' }} />
|
|
163
|
+
</div>
|
|
164
|
+
</div>
|
|
165
|
+
</div>
|
|
166
|
+
)}
|
|
167
|
+
<div ref={messagesEndRef} />
|
|
168
|
+
</div>
|
|
169
|
+
|
|
170
|
+
{/* Input */}
|
|
171
|
+
<div className="flex gap-1.5 px-3 py-3 border-t border-border flex-shrink-0">
|
|
172
|
+
<textarea
|
|
173
|
+
ref={inputRef}
|
|
174
|
+
value={input}
|
|
175
|
+
onChange={(e) => setInput(e.target.value)}
|
|
176
|
+
onKeyDown={handleKeyDown}
|
|
177
|
+
placeholder="프로젝트에 대해 무엇이든 물어보세요…"
|
|
178
|
+
rows={2}
|
|
179
|
+
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"
|
|
180
|
+
/>
|
|
181
|
+
<button
|
|
182
|
+
onClick={send}
|
|
183
|
+
disabled={!input.trim() || loading}
|
|
184
|
+
className="px-3 py-2 bg-accent text-white text-sm rounded-md
|
|
185
|
+
disabled:opacity-40 hover:bg-accent/80 transition-colors flex-shrink-0 self-end"
|
|
186
|
+
>
|
|
187
|
+
Send
|
|
188
|
+
</button>
|
|
189
|
+
</div>
|
|
190
|
+
</div>
|
|
191
|
+
|
|
192
|
+
</div>
|
|
193
|
+
);
|
|
194
|
+
}
|
|
@@ -11,6 +11,7 @@ import AiPolicyModal from '@/components/ui/AiPolicyModal';
|
|
|
11
11
|
import GitSyncResultsModal from '@/components/dashboard/GitSyncResultsModal';
|
|
12
12
|
import FileTreeDrawer from '@/components/ui/FileTreeDrawer';
|
|
13
13
|
import AutoDistributeModal from '@/components/ui/AutoDistributeModal';
|
|
14
|
+
import ProjectAdvisor from '@/components/workspace/ProjectAdvisor';
|
|
14
15
|
import type { ISubProject, ITask, TaskStatus, ISubProjectWithStats, IGitSyncResult } from '@/types';
|
|
15
16
|
|
|
16
17
|
interface IProject {
|
|
@@ -62,6 +63,7 @@ export default function WorkspacePanel({
|
|
|
62
63
|
const [showBrainstorm, setShowBrainstorm] = useState(true);
|
|
63
64
|
const [newSubName, setNewSubName] = useState('');
|
|
64
65
|
const [showAiPolicy, setShowAiPolicy] = useState(false);
|
|
66
|
+
const [showAdvisor, setShowAdvisor] = useState(false);
|
|
65
67
|
const [syncing, setSyncing] = useState(false);
|
|
66
68
|
const [syncResults, setSyncResults] = useState<IGitSyncResult[] | null>(null);
|
|
67
69
|
const [lastSyncTime, setLastSyncTime] = useState<Date | null>(null);
|
|
@@ -359,6 +361,12 @@ export default function WorkspacePanel({
|
|
|
359
361
|
addBtn?.click();
|
|
360
362
|
return;
|
|
361
363
|
}
|
|
364
|
+
// ⌘L — toggle project advisor (works even from input/editor)
|
|
365
|
+
if ((e.metaKey || e.ctrlKey) && e.code === 'KeyL') {
|
|
366
|
+
e.preventDefault();
|
|
367
|
+
setShowAdvisor(prev => !prev);
|
|
368
|
+
return;
|
|
369
|
+
}
|
|
362
370
|
if (selectedTaskId && selectedSubId && !isInput) {
|
|
363
371
|
const statusMap: Record<string, TaskStatus> = {
|
|
364
372
|
'Digit1': 'idea', 'Digit2': 'doing', 'Digit3': 'done', 'Digit4': 'problem',
|
|
@@ -441,6 +449,15 @@ export default function WorkspacePanel({
|
|
|
441
449
|
}`}>
|
|
442
450
|
AI Policy{project.ai_context ? ' *' : ''}
|
|
443
451
|
</button>
|
|
452
|
+
<button onClick={() => setShowAdvisor(true)}
|
|
453
|
+
className={`px-3 py-1.5 text-xs border rounded-md transition-colors ${
|
|
454
|
+
showAdvisor
|
|
455
|
+
? 'bg-primary/15 text-primary border-primary/30'
|
|
456
|
+
: 'bg-muted hover:bg-card-hover text-muted-foreground border-border'
|
|
457
|
+
}`}
|
|
458
|
+
title="프로젝트 어드바이저 (⌘L)">
|
|
459
|
+
Advisor
|
|
460
|
+
</button>
|
|
444
461
|
{project.project_path ? (
|
|
445
462
|
<div className="flex items-center gap-1.5">
|
|
446
463
|
<button
|
|
@@ -603,6 +620,12 @@ export default function WorkspacePanel({
|
|
|
603
620
|
onClose={() => setShowAutoDistribute(false)}
|
|
604
621
|
onApplied={() => { loadSubProjects(); }}
|
|
605
622
|
/>
|
|
623
|
+
{showAdvisor && (
|
|
624
|
+
<ProjectAdvisor
|
|
625
|
+
projectId={id}
|
|
626
|
+
onClose={() => setShowAdvisor(false)}
|
|
627
|
+
/>
|
|
628
|
+
)}
|
|
606
629
|
</div>
|
|
607
630
|
);
|
|
608
631
|
}
|