@shepai/cli 1.149.1-pr470.e87a8eb → 1.150.0-pr472.498e2c5

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 (171) hide show
  1. package/dist/src/presentation/web/app/actions/get-merge-review-data.d.ts.map +1 -1
  2. package/dist/src/presentation/web/app/actions/get-merge-review-data.js +10 -1
  3. package/dist/src/presentation/web/components/common/control-center-drawer/feature-drawer-client.d.ts.map +1 -1
  4. package/dist/src/presentation/web/components/common/control-center-drawer/feature-drawer-client.js +29 -2
  5. package/dist/src/presentation/web/components/common/feature-drawer/use-feature-actions.d.ts +3 -0
  6. package/dist/src/presentation/web/components/common/feature-drawer/use-feature-actions.d.ts.map +1 -1
  7. package/dist/src/presentation/web/components/common/feature-drawer/use-feature-actions.js +32 -1
  8. package/dist/src/presentation/web/components/common/merge-review/merge-review.d.ts.map +1 -1
  9. package/dist/src/presentation/web/components/common/merge-review/merge-review.js +2 -28
  10. package/dist/src/presentation/web/components/common/open-action-menu/open-action-menu.d.ts.map +1 -1
  11. package/dist/src/presentation/web/components/common/open-action-menu/open-action-menu.js +3 -3
  12. package/dist/src/presentation/web/components/common/open-action-menu/open-action-menu.stories.d.ts.map +1 -1
  13. package/dist/src/presentation/web/components/common/open-action-menu/open-action-menu.stories.js +3 -0
  14. package/dist/src/presentation/web/components/features/control-center/control-center-inner.d.ts.map +1 -1
  15. package/dist/src/presentation/web/components/features/control-center/control-center-inner.js +18 -0
  16. package/dist/tsconfig.build.tsbuildinfo +1 -1
  17. package/package.json +1 -1
  18. package/web/.next/BUILD_ID +1 -1
  19. package/web/.next/build-manifest.json +2 -2
  20. package/web/.next/fallback-build-manifest.json +2 -2
  21. package/web/.next/prerender-manifest.json +3 -3
  22. package/web/.next/required-server-files.js +2 -2
  23. package/web/.next/required-server-files.json +2 -2
  24. package/web/.next/server/app/(dashboard)/@drawer/adopt/page/server-reference-manifest.json +28 -28
  25. package/web/.next/server/app/(dashboard)/@drawer/adopt/page.js.nft.json +1 -1
  26. package/web/.next/server/app/(dashboard)/@drawer/adopt/page_client-reference-manifest.js +1 -1
  27. package/web/.next/server/app/(dashboard)/@drawer/create/page/server-reference-manifest.json +28 -28
  28. package/web/.next/server/app/(dashboard)/@drawer/create/page.js.nft.json +1 -1
  29. package/web/.next/server/app/(dashboard)/@drawer/create/page_client-reference-manifest.js +1 -1
  30. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page/server-reference-manifest.json +36 -36
  31. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page.js.nft.json +1 -1
  32. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page_client-reference-manifest.js +1 -1
  33. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page/server-reference-manifest.json +36 -36
  34. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page.js.nft.json +1 -1
  35. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page_client-reference-manifest.js +1 -1
  36. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page/server-reference-manifest.json +26 -26
  37. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page.js.nft.json +1 -1
  38. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page_client-reference-manifest.js +1 -1
  39. package/web/.next/server/app/(dashboard)/create/page/server-reference-manifest.json +28 -28
  40. package/web/.next/server/app/(dashboard)/create/page.js.nft.json +1 -1
  41. package/web/.next/server/app/(dashboard)/create/page_client-reference-manifest.js +1 -1
  42. package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page/server-reference-manifest.json +36 -36
  43. package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page.js.nft.json +1 -1
  44. package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page_client-reference-manifest.js +1 -1
  45. package/web/.next/server/app/(dashboard)/feature/[featureId]/page/server-reference-manifest.json +36 -36
  46. package/web/.next/server/app/(dashboard)/feature/[featureId]/page.js.nft.json +1 -1
  47. package/web/.next/server/app/(dashboard)/feature/[featureId]/page_client-reference-manifest.js +1 -1
  48. package/web/.next/server/app/(dashboard)/page/server-reference-manifest.json +26 -26
  49. package/web/.next/server/app/(dashboard)/page.js.nft.json +1 -1
  50. package/web/.next/server/app/(dashboard)/page_client-reference-manifest.js +1 -1
  51. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page/server-reference-manifest.json +26 -26
  52. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page.js.nft.json +1 -1
  53. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page_client-reference-manifest.js +1 -1
  54. package/web/.next/server/app/_global-error.html +2 -2
  55. package/web/.next/server/app/_global-error.rsc +1 -1
  56. package/web/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +1 -1
  57. package/web/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  58. package/web/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  59. package/web/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  60. package/web/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  61. package/web/.next/server/app/_not-found/page/server-reference-manifest.json +3 -3
  62. package/web/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  63. package/web/.next/server/app/api/attachments/preview/route.js.nft.json +1 -1
  64. package/web/.next/server/app/api/evidence/route.js.nft.json +1 -1
  65. package/web/.next/server/app/api/graph-data/route.js.nft.json +1 -1
  66. package/web/.next/server/app/settings/page/server-reference-manifest.json +8 -8
  67. package/web/.next/server/app/settings/page.js.nft.json +1 -1
  68. package/web/.next/server/app/settings/page_client-reference-manifest.js +1 -1
  69. package/web/.next/server/app/skills/page/server-reference-manifest.json +8 -8
  70. package/web/.next/server/app/skills/page.js.nft.json +1 -1
  71. package/web/.next/server/app/skills/page_client-reference-manifest.js +1 -1
  72. package/web/.next/server/app/tools/page/server-reference-manifest.json +8 -8
  73. package/web/.next/server/app/tools/page.js.nft.json +1 -1
  74. package/web/.next/server/app/tools/page_client-reference-manifest.js +1 -1
  75. package/web/.next/server/app/version/page/server-reference-manifest.json +3 -3
  76. package/web/.next/server/app/version/page_client-reference-manifest.js +1 -1
  77. package/web/.next/server/chunks/[root-of-the-server]__a402b567._.js +1 -1
  78. package/web/.next/server/chunks/ssr/744ca_web_components_common_control-center-drawer_create-drawer-client_tsx_5e26fc0a._.js +1 -1
  79. package/web/.next/server/chunks/ssr/744ca_web_components_common_control-center-drawer_create-drawer-client_tsx_5e26fc0a._.js.map +1 -1
  80. package/web/.next/server/chunks/ssr/[root-of-the-server]__2138fa7e._.js +2 -2
  81. package/web/.next/server/chunks/ssr/[root-of-the-server]__29580090._.js +1 -1
  82. package/web/.next/server/chunks/ssr/[root-of-the-server]__29580090._.js.map +1 -1
  83. package/web/.next/server/chunks/ssr/[root-of-the-server]__357d99f9._.js +1 -1
  84. package/web/.next/server/chunks/ssr/[root-of-the-server]__3ef34e4c._.js +1 -1
  85. package/web/.next/server/chunks/ssr/[root-of-the-server]__43f51aa6._.js +1 -1
  86. package/web/.next/server/chunks/ssr/[root-of-the-server]__43f51aa6._.js.map +1 -1
  87. package/web/.next/server/chunks/ssr/[root-of-the-server]__815546bd._.js +1 -1
  88. package/web/.next/server/chunks/ssr/[root-of-the-server]__815546bd._.js.map +1 -1
  89. package/web/.next/server/chunks/ssr/[root-of-the-server]__aad040c0._.js +3 -3
  90. package/web/.next/server/chunks/ssr/[root-of-the-server]__aad040c0._.js.map +1 -1
  91. package/web/.next/server/chunks/ssr/[root-of-the-server]__c094882b._.js +1 -1
  92. package/web/.next/server/chunks/ssr/[root-of-the-server]__c094882b._.js.map +1 -1
  93. package/web/.next/server/chunks/ssr/[root-of-the-server]__d48c5b11._.js +1 -1
  94. package/web/.next/server/chunks/ssr/[root-of-the-server]__d48c5b11._.js.map +1 -1
  95. package/web/.next/server/chunks/ssr/[root-of-the-server]__dac5dbf1._.js +1 -1
  96. package/web/.next/server/chunks/ssr/[root-of-the-server]__dac5dbf1._.js.map +1 -1
  97. package/web/.next/server/chunks/ssr/[root-of-the-server]__fae8b355._.js +1 -1
  98. package/web/.next/server/chunks/ssr/[root-of-the-server]__fae8b355._.js.map +1 -1
  99. package/web/.next/server/chunks/ssr/_05c23ad9._.js +1 -1
  100. package/web/.next/server/chunks/ssr/_05c23ad9._.js.map +1 -1
  101. package/web/.next/server/chunks/ssr/_0c5f56e3._.js +2 -2
  102. package/web/.next/server/chunks/ssr/_0c5f56e3._.js.map +1 -1
  103. package/web/.next/server/chunks/ssr/_16eb4fec._.js +1 -1
  104. package/web/.next/server/chunks/ssr/_16eb4fec._.js.map +1 -1
  105. package/web/.next/server/chunks/ssr/_1b719e7f._.js +1 -1
  106. package/web/.next/server/chunks/ssr/_1b719e7f._.js.map +1 -1
  107. package/web/.next/server/chunks/ssr/_37e8548b._.js +1 -1
  108. package/web/.next/server/chunks/ssr/_37e8548b._.js.map +1 -1
  109. package/web/.next/server/chunks/ssr/_55d763e2._.js +1 -1
  110. package/web/.next/server/chunks/ssr/_55d763e2._.js.map +1 -1
  111. package/web/.next/server/chunks/ssr/{_2401ea8d._.js → _580ca274._.js} +2 -2
  112. package/web/.next/server/chunks/ssr/_580ca274._.js.map +1 -0
  113. package/web/.next/server/chunks/ssr/_6256a985._.js +1 -1
  114. package/web/.next/server/chunks/ssr/_6256a985._.js.map +1 -1
  115. package/web/.next/server/chunks/ssr/_64bdfc6f._.js +2 -2
  116. package/web/.next/server/chunks/ssr/_64bdfc6f._.js.map +1 -1
  117. package/web/.next/server/chunks/ssr/_8fcc39d4._.js +1 -1
  118. package/web/.next/server/chunks/ssr/{_44be6da8._.js → _9eb24ecf._.js} +2 -2
  119. package/web/.next/server/chunks/ssr/{_44be6da8._.js.map → _9eb24ecf._.js.map} +1 -1
  120. package/web/.next/server/chunks/ssr/_b71645b4._.js +1 -1
  121. package/web/.next/server/chunks/ssr/_b71645b4._.js.map +1 -1
  122. package/web/.next/server/chunks/ssr/_d8575088._.js +1 -1
  123. package/web/.next/server/chunks/ssr/_d8575088._.js.map +1 -1
  124. package/web/.next/server/chunks/ssr/b1a17_presentation_web_components_features_settings_settings-page-client_tsx_6ed9d5f8._.js +1 -1
  125. package/web/.next/server/chunks/ssr/b1a17_presentation_web_components_features_settings_settings-page-client_tsx_6ed9d5f8._.js.map +1 -1
  126. package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_skills_page_actions_1b176e3c.js +1 -1
  127. package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_skills_page_actions_1b176e3c.js.map +1 -1
  128. package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_tools_page_actions_bd9f0dda.js +1 -1
  129. package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_tools_page_actions_bd9f0dda.js.map +1 -1
  130. package/web/.next/server/chunks/ssr/src_presentation_web_app_actions_open-ide_ts_baaca5d5._.js +1 -1
  131. package/web/.next/server/chunks/ssr/src_presentation_web_components_e599bb8c._.js +1 -1
  132. package/web/.next/server/chunks/ssr/src_presentation_web_components_e599bb8c._.js.map +1 -1
  133. package/web/.next/server/chunks/ssr/src_presentation_web_components_features_control-center_7ac3562e._.js +1 -1
  134. package/web/.next/server/chunks/ssr/src_presentation_web_components_features_control-center_7ac3562e._.js.map +1 -1
  135. package/web/.next/server/chunks/ssr/{src_presentation_web_4aadd6df._.js → src_presentation_web_f7c07685._.js} +2 -2
  136. package/web/.next/server/chunks/ssr/{src_presentation_web_4aadd6df._.js.map → src_presentation_web_f7c07685._.js.map} +1 -1
  137. package/web/.next/server/pages/500.html +2 -2
  138. package/web/.next/server/server-reference-manifest.js +1 -1
  139. package/web/.next/server/server-reference-manifest.json +44 -44
  140. package/web/.next/static/chunks/{19c8e86bf510b8b5.js → 27d90addfd3042f4.js} +1 -1
  141. package/web/.next/static/chunks/3488c429220a5428.js +1 -0
  142. package/web/.next/static/chunks/51d6a4e38e055da7.js +1 -0
  143. package/web/.next/static/chunks/{676e80c193fe7921.js → 63a6aef5209ad111.js} +1 -1
  144. package/web/.next/static/chunks/{b08b7d0a9ff0ffdd.js → 672a0665307c4656.js} +1 -1
  145. package/web/.next/static/chunks/8154cf726cf76b22.js +5 -0
  146. package/web/.next/static/chunks/{04b7f4c1ce5ff99f.js → 92c350ffb06ac8c9.js} +1 -1
  147. package/web/.next/static/chunks/9b8678597fa1db84.css +1 -0
  148. package/web/.next/static/chunks/{aef03f1d55f3ab53.js → 9d30850608895a14.js} +2 -2
  149. package/web/.next/static/chunks/d0cf4f41c6473d9e.js +1 -0
  150. package/web/.next/static/chunks/{6a314de2c9310879.js → e1b442e2b4e27e76.js} +1 -1
  151. package/web/.next/static/chunks/f86e6d52f5309915.js +1 -0
  152. package/web/.next/static/chunks/{f821ba3fe0522cfa.js → ff66542d8b6a693b.js} +1 -1
  153. package/dist/src/presentation/web/components/common/evidence-lightbox/evidence-lightbox.d.ts +0 -15
  154. package/dist/src/presentation/web/components/common/evidence-lightbox/evidence-lightbox.d.ts.map +0 -1
  155. package/dist/src/presentation/web/components/common/evidence-lightbox/evidence-lightbox.js +0 -43
  156. package/dist/src/presentation/web/components/common/evidence-lightbox/evidence-lightbox.stories.d.ts +0 -18
  157. package/dist/src/presentation/web/components/common/evidence-lightbox/evidence-lightbox.stories.d.ts.map +0 -1
  158. package/dist/src/presentation/web/components/common/evidence-lightbox/evidence-lightbox.stories.js +0 -125
  159. package/dist/src/presentation/web/components/common/evidence-lightbox/index.d.ts +0 -3
  160. package/dist/src/presentation/web/components/common/evidence-lightbox/index.d.ts.map +0 -1
  161. package/dist/src/presentation/web/components/common/evidence-lightbox/index.js +0 -1
  162. package/web/.next/server/chunks/ssr/_2401ea8d._.js.map +0 -1
  163. package/web/.next/static/chunks/52470bcbe2e8198d.css +0 -1
  164. package/web/.next/static/chunks/6715dda4a1b9ecc3.js +0 -1
  165. package/web/.next/static/chunks/8dffe93d946d1177.js +0 -5
  166. package/web/.next/static/chunks/9bc50fad44bb8029.js +0 -1
  167. package/web/.next/static/chunks/a1f6bb54e8db6bac.js +0 -1
  168. package/web/.next/static/chunks/d3f77935c324dc73.js +0 -1
  169. /package/web/.next/static/{oWoqiR8AqazcpEeXFZleb → atHbG6OJQjDFHjONkKfiS}/_buildManifest.js +0 -0
  170. /package/web/.next/static/{oWoqiR8AqazcpEeXFZleb → atHbG6OJQjDFHjONkKfiS}/_clientMiddlewareManifest.json +0 -0
  171. /package/web/.next/static/{oWoqiR8AqazcpEeXFZleb → atHbG6OJQjDFHjONkKfiS}/_ssgManifest.js +0 -0
@@ -1 +1 @@
1
- {"version":3,"file":"get-merge-review-data.d.ts","sourceRoot":"","sources":["../../../../../../src/presentation/web/app/actions/get-merge-review-data.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EACV,eAAe,EAEhB,MAAM,sDAAsD,CAAC;AAK9D,KAAK,wBAAwB,GAAG,eAAe,GAAG;IAAE,KAAK,EAAE,MAAM,CAAA;CAAE,CAAC;AAqCpE,wBAAsB,kBAAkB,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,wBAAwB,CAAC,CAmG7F"}
1
+ {"version":3,"file":"get-merge-review-data.d.ts","sourceRoot":"","sources":["../../../../../../src/presentation/web/app/actions/get-merge-review-data.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EACV,eAAe,EAEhB,MAAM,sDAAsD,CAAC;AAK9D,KAAK,wBAAwB,GAAG,eAAe,GAAG;IAAE,KAAK,EAAE,MAAM,CAAA;CAAE,CAAC;AAqCpE,wBAAsB,kBAAkB,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,wBAAwB,CAAC,CA2G7F"}
@@ -92,7 +92,16 @@ export async function getMergeReviewData(featureId) {
92
92
  const manifestPath = join(evidenceDir, 'manifest.json');
93
93
  if (existsSync(manifestPath)) {
94
94
  const raw = JSON.parse(readFileSync(manifestPath, 'utf-8'));
95
- evidence = normalizeEvidencePaths(raw, evidenceDir);
95
+ const normalized = normalizeEvidencePaths(raw, evidenceDir);
96
+ // Deduplicate: same type + relativePath means the same evidence entry
97
+ const seen = new Set();
98
+ evidence = normalized.filter((e) => {
99
+ const key = `${e.type}:${e.relativePath}`;
100
+ if (seen.has(key))
101
+ return false;
102
+ seen.add(key);
103
+ return true;
104
+ });
96
105
  }
97
106
  }
98
107
  catch {
@@ -1 +1 @@
1
- {"version":3,"file":"feature-drawer-client.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/control-center-drawer/feature-drawer-client.tsx"],"names":[],"mappings":"AAuCA,OAAO,KAAK,EAAE,UAAU,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAK/D,MAAM,WAAW,wBAAwB;IACvC,IAAI,EAAE,UAAU,CAAC;IACjB,8FAA8F;IAC9F,MAAM,CAAC,EAAE,aAAa,CAAC;CACxB;AAED,wBAAgB,mBAAmB,CAAC,EAAE,IAAI,EAAE,WAAW,EAAE,MAAM,EAAE,EAAE,wBAAwB,2CA4oB1F"}
1
+ {"version":3,"file":"feature-drawer-client.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/control-center-drawer/feature-drawer-client.tsx"],"names":[],"mappings":"AAkDA,OAAO,KAAK,EAAE,UAAU,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAK/D,MAAM,WAAW,wBAAwB;IACvC,IAAI,EAAE,UAAU,CAAC;IACjB,8FAA8F;IAC9F,MAAM,CAAC,EAAE,aAAa,CAAC;CACxB;AAED,wBAAgB,mBAAmB,CAAC,EAAE,IAAI,EAAE,WAAW,EAAE,MAAM,EAAE,EAAE,wBAAwB,2CAsuB1F"}
@@ -3,7 +3,7 @@ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-run
3
3
  import { useState, useCallback, useEffect, useRef } from 'react';
4
4
  import { useRouter, usePathname } from 'next/navigation';
5
5
  import { toast } from 'sonner';
6
- import { Loader2, Trash2, Play, Square, Copy, Check, Code2, ExternalLink } from 'lucide-react';
6
+ import { Loader2, Trash2, Play, Square, Copy, Check, Code2, ExternalLink, Archive, ArchiveRestore, } from 'lucide-react';
7
7
  import { approveFeature } from '../../../app/actions/approve-feature.js';
8
8
  import { resumeFeature } from '../../../app/actions/resume-feature.js';
9
9
  import { startFeature } from '../../../app/actions/start-feature.js';
@@ -106,6 +106,19 @@ export function FeatureDrawerClient({ view: initialView, urlTab }) {
106
106
  // ── Delete state ───────────────────────────────────────────────────────
107
107
  const [isDeleting, setIsDeleting] = useState(false);
108
108
  const [deleteDialogOpen, setDeleteDialogOpen] = useState(false);
109
+ // ── Archive state ─────────────────────────────────────────────────────
110
+ const [isArchiving, setIsArchiving] = useState(false);
111
+ // Reset archive loading spinner when the feature state or feature ID
112
+ // changes (e.g. state flips to 'archived' / 'done' after the server
113
+ // action, or the user navigates to a different feature drawer).
114
+ const archiveResetKey = `${featureNode?.featureId}:${featureNode?.state}`;
115
+ const prevArchiveResetKeyRef = useRef(archiveResetKey);
116
+ useEffect(() => {
117
+ if (archiveResetKey !== prevArchiveResetKeyRef.current) {
118
+ prevArchiveResetKeyRef.current = archiveResetKey;
119
+ setIsArchiving(false);
120
+ }
121
+ }, [archiveResetKey]);
109
122
  // ── Shared reject state ────────────────────────────────────────────────
110
123
  const [isRejecting, setIsRejecting] = useState(false);
111
124
  const isRejectingRef = useRef(false);
@@ -290,6 +303,20 @@ export function FeatureDrawerClient({ view: initialView, urlTab }) {
290
303
  }));
291
304
  router.push('/');
292
305
  }, [router]);
306
+ const handleArchive = useCallback((featureId) => {
307
+ setIsArchiving(true);
308
+ window.dispatchEvent(new CustomEvent('shep:feature-archive-requested', {
309
+ detail: { featureId },
310
+ }));
311
+ router.push('/');
312
+ }, [router]);
313
+ const handleUnarchive = useCallback((featureId) => {
314
+ setIsArchiving(true);
315
+ window.dispatchEvent(new CustomEvent('shep:feature-unarchive-requested', {
316
+ detail: { featureId },
317
+ }));
318
+ router.push('/');
319
+ }, [router]);
293
320
  const handleRetry = useCallback(async (featureId) => {
294
321
  const result = await resumeFeature(featureId);
295
322
  if (result.error) {
@@ -376,7 +403,7 @@ export function FeatureDrawerClient({ view: initialView, urlTab }) {
376
403
  const repoName = featureNode.repositoryName ??
377
404
  featureNode.repositoryPath.split('/').filter(Boolean).at(-1) ??
378
405
  '';
379
- header = (_jsxs(_Fragment, { children: [_jsxs("div", { "data-testid": "feature-drawer-header", children: [_jsx(DrawerTitle, { children: featureNode.name }), repoName ? (_jsxs("div", { className: "flex items-center gap-1.5 pt-0.5", children: [_jsx(Code2, { className: "text-muted-foreground size-3.5 shrink-0" }), featureNode.remoteUrl ? (_jsxs("a", { href: featureNode.remoteUrl, target: "_blank", rel: "noopener noreferrer", className: "text-muted-foreground hover:text-foreground inline-flex items-center gap-1 text-xs transition-colors", "data-testid": "feature-drawer-repo-link", children: [repoName, _jsx(ExternalLink, { className: "size-3" })] })) : (_jsx("span", { className: "text-muted-foreground text-xs", children: repoName }))] })) : null, _jsx(DrawerDescription, { className: "sr-only", children: featureNode.name })] }), featureActionsInput ? (_jsxs("div", { className: "flex items-center gap-2 pt-2", "data-testid": "feature-drawer-actions", children: [_jsx(OpenActionMenu, { actions: featureActions, repositoryPath: featureActionsInput.repositoryPath, worktreePath: featureActionsInput.worktreePath, showSpecs: !!featureActionsInput.specPath }), featureFlags.envDeploy && featureDeployTarget ? (_jsxs(_Fragment, { children: [_jsx(TooltipProvider, { children: _jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { asChild: true, children: _jsx("span", { children: _jsx(ActionButton, { label: isFeatureDeployActive ? 'Stop Dev Server' : 'Start Dev Server', onClick: isFeatureDeployActive ? deployAction.stop : deployAction.deploy, loading: deployAction.deployLoading || deployAction.stopLoading, error: !!deployAction.deployError, icon: isFeatureDeployActive ? Square : Play, iconOnly: true, variant: "outline", size: "icon-sm" }) }) }), _jsx(TooltipContent, { children: isFeatureDeployActive ? 'Stop Dev Server' : 'Start Dev Server' })] }) }), isFeatureDeployActive ? (_jsx(DeploymentStatusBadge, { status: deployAction.status, url: deployAction.url, targetId: featureDeployTarget?.targetId })) : null] })) : null, _jsxs("div", { className: "ml-auto flex items-center gap-1.5", children: [_jsx("code", { className: "bg-muted text-muted-foreground rounded px-1.5 py-0.5 font-mono text-xs", children: shortId }), _jsx("button", { type: "button", onClick: handleCopyId, className: "text-muted-foreground hover:text-foreground inline-flex items-center rounded p-0.5 transition-colors", "aria-label": "Copy feature ID", "data-testid": "feature-drawer-copy-id", children: idCopied ? (_jsx(Check, { className: "size-3.5 text-green-600" })) : (_jsx(Copy, { className: "size-3.5" })) })] }), featureNode.featureId ? (_jsxs(_Fragment, { children: [_jsx(Button, { variant: "ghost", size: "icon-sm", "aria-label": "Delete feature", disabled: isDeleting, className: "text-muted-foreground hover:text-destructive", "data-testid": "feature-drawer-delete", onClick: () => setDeleteDialogOpen(true), children: isDeleting ? (_jsx(Loader2, { className: "size-4 animate-spin" })) : (_jsx(Trash2, { className: "size-4" })) }), _jsx(DeleteFeatureDialog, { open: deleteDialogOpen, onOpenChange: setDeleteDialogOpen, onConfirm: (cleanup, cascadeDelete, closePr) => handleDelete(featureNode.featureId, cleanup, cascadeDelete, closePr), isDeleting: isDeleting, featureName: featureNode.name, featureId: featureNode.featureId, hasChildren: featureNode.hasChildren, hasOpenPr: !!featureNode.pr && featureNode.pr.status === 'Open' })] })) : null] })) : null] }));
406
+ header = (_jsxs(_Fragment, { children: [_jsxs("div", { "data-testid": "feature-drawer-header", children: [_jsx(DrawerTitle, { children: featureNode.name }), repoName ? (_jsxs("div", { className: "flex items-center gap-1.5 pt-0.5", children: [_jsx(Code2, { className: "text-muted-foreground size-3.5 shrink-0" }), featureNode.remoteUrl ? (_jsxs("a", { href: featureNode.remoteUrl, target: "_blank", rel: "noopener noreferrer", className: "text-muted-foreground hover:text-foreground inline-flex items-center gap-1 text-xs transition-colors", "data-testid": "feature-drawer-repo-link", children: [repoName, _jsx(ExternalLink, { className: "size-3" })] })) : (_jsx("span", { className: "text-muted-foreground text-xs", children: repoName }))] })) : null, _jsx(DrawerDescription, { className: "sr-only", children: featureNode.name })] }), featureActionsInput ? (_jsxs("div", { className: "flex items-center gap-2 pt-2", "data-testid": "feature-drawer-actions", children: [featureNode?.state !== 'done' ? (_jsx(OpenActionMenu, { actions: featureActions, repositoryPath: featureActionsInput.repositoryPath, worktreePath: featureActionsInput.worktreePath, showSpecs: !!featureActionsInput.specPath })) : null, featureNode?.state !== 'done' && featureFlags.envDeploy && featureDeployTarget ? (_jsxs(_Fragment, { children: [_jsx(TooltipProvider, { children: _jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { asChild: true, children: _jsx("span", { children: _jsx(ActionButton, { label: isFeatureDeployActive ? 'Stop Dev Server' : 'Start Dev Server', onClick: isFeatureDeployActive ? deployAction.stop : deployAction.deploy, loading: deployAction.deployLoading || deployAction.stopLoading, error: !!deployAction.deployError, icon: isFeatureDeployActive ? Square : Play, iconOnly: true, variant: "outline", size: "icon-sm" }) }) }), _jsx(TooltipContent, { children: isFeatureDeployActive ? 'Stop Dev Server' : 'Start Dev Server' })] }) }), isFeatureDeployActive ? (_jsx(DeploymentStatusBadge, { status: deployAction.status, url: deployAction.url, targetId: featureDeployTarget?.targetId })) : null] })) : null, _jsxs("div", { className: "ml-auto flex items-center gap-1.5", children: [_jsx("code", { className: "bg-muted text-muted-foreground rounded px-1.5 py-0.5 font-mono text-xs", children: shortId }), _jsx("button", { type: "button", onClick: handleCopyId, className: "text-muted-foreground hover:text-foreground inline-flex items-center rounded p-0.5 transition-colors", "aria-label": "Copy feature ID", "data-testid": "feature-drawer-copy-id", children: idCopied ? (_jsx(Check, { className: "size-3.5 text-green-600" })) : (_jsx(Copy, { className: "size-3.5" })) })] }), featureNode.featureId ? (_jsxs(_Fragment, { children: [featureNode.state === 'archived' ? (_jsx(TooltipProvider, { children: _jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { asChild: true, children: _jsx(Button, { variant: "ghost", size: "icon-sm", "aria-label": "Unarchive feature", disabled: isArchiving, className: "text-muted-foreground hover:text-primary", "data-testid": "feature-drawer-unarchive", onClick: () => handleUnarchive(featureNode.featureId), children: isArchiving ? (_jsx(Loader2, { className: "size-4 animate-spin" })) : (_jsx(ArchiveRestore, { className: "size-4" })) }) }), _jsx(TooltipContent, { children: "Unarchive feature" })] }) })) : featureNode.state !== 'deleting' ? (_jsx(TooltipProvider, { children: _jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { asChild: true, children: _jsx(Button, { variant: "ghost", size: "icon-sm", "aria-label": "Archive feature", disabled: isArchiving, className: "text-muted-foreground hover:text-foreground", "data-testid": "feature-drawer-archive", onClick: () => handleArchive(featureNode.featureId), children: isArchiving ? (_jsx(Loader2, { className: "size-4 animate-spin" })) : (_jsx(Archive, { className: "size-4" })) }) }), _jsx(TooltipContent, { children: "Archive feature" })] }) })) : null, _jsx(Button, { variant: "ghost", size: "icon-sm", "aria-label": "Delete feature", disabled: isDeleting, className: "text-muted-foreground hover:text-destructive", "data-testid": "feature-drawer-delete", onClick: () => setDeleteDialogOpen(true), children: isDeleting ? (_jsx(Loader2, { className: "size-4 animate-spin" })) : (_jsx(Trash2, { className: "size-4" })) }), _jsx(DeleteFeatureDialog, { open: deleteDialogOpen, onOpenChange: setDeleteDialogOpen, onConfirm: (cleanup, cascadeDelete, closePr) => handleDelete(featureNode.featureId, cleanup, cascadeDelete, closePr), isDeleting: isDeleting, featureName: featureNode.name, featureId: featureNode.featureId, hasChildren: featureNode.hasChildren, hasOpenPr: !!featureNode.pr && featureNode.pr.status === 'Open' })] })) : null] })) : null] }));
380
407
  }
381
408
  // ── Body ──────────────────────────────────────────────────────────────
382
409
  let body = null;
@@ -8,14 +8,17 @@ export interface FeatureActionsInput {
8
8
  export interface FeatureActionsState {
9
9
  openInIde: () => Promise<void>;
10
10
  openInShell: () => Promise<void>;
11
+ openFolder: () => Promise<void>;
11
12
  openSpecsFolder: () => Promise<void>;
12
13
  rebaseOnMain: () => Promise<void>;
13
14
  ideLoading: boolean;
14
15
  shellLoading: boolean;
16
+ folderLoading: boolean;
15
17
  specsLoading: boolean;
16
18
  rebaseLoading: boolean;
17
19
  ideError: string | null;
18
20
  shellError: string | null;
21
+ folderError: string | null;
19
22
  specsError: string | null;
20
23
  rebaseError: string | null;
21
24
  }
@@ -1 +1 @@
1
- {"version":3,"file":"use-feature-actions.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/feature-drawer/use-feature-actions.ts"],"names":[],"mappings":"AAQA,MAAM,WAAW,mBAAmB;IAClC,SAAS,EAAE,MAAM,CAAC;IAClB,cAAc,EAAE,MAAM,CAAC;IACvB,MAAM,EAAE,MAAM,CAAC;IACf,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,mBAAmB;IAClC,SAAS,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAC/B,WAAW,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IACjC,eAAe,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IACrC,YAAY,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAClC,UAAU,EAAE,OAAO,CAAC;IACpB,YAAY,EAAE,OAAO,CAAC;IACtB,YAAY,EAAE,OAAO,CAAC;IACtB,aAAa,EAAE,OAAO,CAAC;IACvB,QAAQ,EAAE,MAAM,GAAG,IAAI,CAAC;IACxB,UAAU,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,UAAU,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,WAAW,EAAE,MAAM,GAAG,IAAI,CAAC;CAC5B;AAID,wBAAgB,iBAAiB,CAAC,KAAK,EAAE,mBAAmB,GAAG,IAAI,GAAG,mBAAmB,CA6IxF"}
1
+ {"version":3,"file":"use-feature-actions.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/feature-drawer/use-feature-actions.ts"],"names":[],"mappings":"AAQA,MAAM,WAAW,mBAAmB;IAClC,SAAS,EAAE,MAAM,CAAC;IAClB,cAAc,EAAE,MAAM,CAAC;IACvB,MAAM,EAAE,MAAM,CAAC;IACf,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,mBAAmB;IAClC,SAAS,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAC/B,WAAW,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IACjC,UAAU,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAChC,eAAe,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IACrC,YAAY,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAClC,UAAU,EAAE,OAAO,CAAC;IACpB,YAAY,EAAE,OAAO,CAAC;IACtB,aAAa,EAAE,OAAO,CAAC;IACvB,YAAY,EAAE,OAAO,CAAC;IACtB,aAAa,EAAE,OAAO,CAAC;IACvB,QAAQ,EAAE,MAAM,GAAG,IAAI,CAAC;IACxB,UAAU,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,WAAW,EAAE,MAAM,GAAG,IAAI,CAAC;IAC3B,UAAU,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,WAAW,EAAE,MAAM,GAAG,IAAI,CAAC;CAC5B;AAID,wBAAgB,iBAAiB,CAAC,KAAK,EAAE,mBAAmB,GAAG,IAAI,GAAG,mBAAmB,CA6KxF"}
@@ -8,20 +8,23 @@ const ERROR_CLEAR_DELAY = 5000;
8
8
  export function useFeatureActions(input) {
9
9
  const [ideLoading, setIdeLoading] = useState(false);
10
10
  const [shellLoading, setShellLoading] = useState(false);
11
+ const [folderLoading, setFolderLoading] = useState(false);
11
12
  const [specsLoading, setSpecsLoading] = useState(false);
12
13
  const [rebaseLoading, setRebaseLoading] = useState(false);
13
14
  const [ideError, setIdeError] = useState(null);
14
15
  const [shellError, setShellError] = useState(null);
16
+ const [folderError, setFolderError] = useState(null);
15
17
  const [specsError, setSpecsError] = useState(null);
16
18
  const [rebaseError, setRebaseError] = useState(null);
17
19
  const ideTimerRef = useRef(null);
18
20
  const shellTimerRef = useRef(null);
21
+ const folderTimerRef = useRef(null);
19
22
  const specsTimerRef = useRef(null);
20
23
  const rebaseTimerRef = useRef(null);
21
24
  // Clear timers on unmount — read .current inside cleanup so we get the
22
25
  // actual timer value at teardown time, not the always-null value at mount.
23
26
  useEffect(() => {
24
- const refs = [ideTimerRef, shellTimerRef, specsTimerRef, rebaseTimerRef];
27
+ const refs = [ideTimerRef, shellTimerRef, folderTimerRef, specsTimerRef, rebaseTimerRef];
25
28
  return () => {
26
29
  for (const ref of refs) {
27
30
  if (ref.current)
@@ -59,6 +62,31 @@ export function useFeatureActions(input) {
59
62
  }, [input]);
60
63
  const handleOpenIde = useCallback(() => performAction(openIde, setIdeLoading, setIdeError, ideTimerRef, ideLoading), [performAction, ideLoading]);
61
64
  const handleOpenShell = useCallback(() => performAction(openShell, setShellLoading, setShellError, shellTimerRef, shellLoading), [performAction, shellLoading]);
65
+ const handleOpenFolder = useCallback(async () => {
66
+ if (!input || folderLoading)
67
+ return;
68
+ if (folderTimerRef.current)
69
+ clearTimeout(folderTimerRef.current);
70
+ setFolderLoading(true);
71
+ setFolderError(null);
72
+ try {
73
+ const folderPath = input.worktreePath ?? input.repositoryPath;
74
+ const result = await openFolder(folderPath);
75
+ if (!result.success) {
76
+ const errorMessage = result.error ?? 'An unexpected error occurred';
77
+ setFolderError(errorMessage);
78
+ folderTimerRef.current = setTimeout(() => setFolderError(null), ERROR_CLEAR_DELAY);
79
+ }
80
+ }
81
+ catch (err) {
82
+ const errorMessage = err instanceof Error ? err.message : 'An unexpected error occurred';
83
+ setFolderError(errorMessage);
84
+ folderTimerRef.current = setTimeout(() => setFolderError(null), ERROR_CLEAR_DELAY);
85
+ }
86
+ finally {
87
+ setFolderLoading(false);
88
+ }
89
+ }, [input, folderLoading]);
62
90
  const handleOpenSpecsFolder = useCallback(async () => {
63
91
  if (!input?.specPath || specsLoading)
64
92
  return;
@@ -110,14 +138,17 @@ export function useFeatureActions(input) {
110
138
  return {
111
139
  openInIde: handleOpenIde,
112
140
  openInShell: handleOpenShell,
141
+ openFolder: handleOpenFolder,
113
142
  openSpecsFolder: handleOpenSpecsFolder,
114
143
  rebaseOnMain: handleRebaseOnMain,
115
144
  ideLoading,
116
145
  shellLoading,
146
+ folderLoading,
117
147
  specsLoading,
118
148
  rebaseLoading,
119
149
  ideError,
120
150
  shellError,
151
+ folderError,
121
152
  specsError,
122
153
  rebaseError,
123
154
  };
@@ -1 +1 @@
1
- {"version":3,"file":"merge-review.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/merge-review/merge-review.tsx"],"names":[],"mappings":"AAuBA,OAAO,KAAK,EAAE,gBAAgB,EAAuB,MAAM,uBAAuB,CAAC;AAmNnF,wBAAgB,WAAW,CAAC,EAC1B,IAAI,EACJ,QAAgB,EAChB,SAAS,EACT,QAAQ,EACR,YAAoB,EACpB,WAAmB,EACnB,SAAS,EACT,iBAAiB,GAClB,EAAE,gBAAgB,2CA+JlB"}
1
+ {"version":3,"file":"merge-review.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/merge-review/merge-review.tsx"],"names":[],"mappings":"AAsBA,OAAO,KAAK,EAAE,gBAAgB,EAAuB,MAAM,uBAAuB,CAAC;AAuJnF,wBAAgB,WAAW,CAAC,EAC1B,IAAI,EACJ,QAAgB,EAChB,SAAS,EACT,QAAQ,EACR,YAAoB,EACpB,WAAmB,EACnB,SAAS,EACT,iBAAiB,GAClB,EAAE,gBAAgB,2CA+JlB"}
@@ -1,11 +1,10 @@
1
1
  'use client';
2
2
  import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
3
- import { useState, useMemo } from 'react';
3
+ import { useState } from 'react';
4
4
  import { ExternalLink, AlertTriangle, FileDiff, GitCommitHorizontal, GitBranch, ArrowRight, Camera, FileText, MonitorPlay, Terminal, Download, ChevronDown, ChevronRight, } from 'lucide-react';
5
5
  import { Badge } from '../../ui/badge.js';
6
6
  import { CiStatusBadge } from '../../common/ci-status-badge/index.js';
7
7
  import { DrawerActionBar } from '../../common/drawer-action-bar/index.js';
8
- import { EvidenceLightbox } from '../../common/evidence-lightbox/index.js';
9
8
  import { DiffView } from './diff-view.js';
10
9
  const EVIDENCE_ICONS = {
11
10
  Screenshot: Camera,
@@ -55,33 +54,8 @@ function EvidenceTextPreview({ url }) {
55
54
  }
56
55
  return (_jsx("pre", { className: "bg-muted/50 max-h-60 overflow-auto rounded-md p-3 font-mono text-[11px] leading-relaxed", children: content }));
57
56
  }
58
- function isImageEvidence(evidence) {
59
- return (evidence.type === 'Screenshot' || IMAGE_EXTENSIONS.has(getExtension(evidence.relativePath)));
60
- }
61
57
  function EvidenceList({ evidence }) {
62
- const [lightboxOpen, setLightboxOpen] = useState(false);
63
- const [selectedIndex, setSelectedIndex] = useState(0);
64
- const { imageEvidence, nonImageEvidence } = useMemo(() => {
65
- const imgs = [];
66
- const nonImgs = [];
67
- for (const e of evidence) {
68
- if (isImageEvidence(e)) {
69
- imgs.push(e);
70
- }
71
- else {
72
- nonImgs.push(e);
73
- }
74
- }
75
- return { imageEvidence: imgs, nonImageEvidence: nonImgs };
76
- }, [evidence]);
77
- function handleThumbnailClick(index) {
78
- setSelectedIndex(index);
79
- setLightboxOpen(true);
80
- }
81
- return (_jsxs("div", { className: "border-border rounded-lg border", children: [_jsxs("div", { className: "px-4 py-3", children: [_jsxs("div", { className: "mb-3 flex items-center gap-2", children: [_jsx(Camera, { className: "text-muted-foreground h-4 w-4" }), _jsx("span", { className: "text-foreground text-xs font-semibold", children: "Evidence" }), _jsx(Badge, { variant: "secondary", className: "text-[10px]", children: evidence.length })] }), imageEvidence.length > 0 ? (
82
- /* eslint-disable @next/next/no-img-element -- Local evidence files require raw <img>, not next/image */
83
- _jsx("div", { className: "mb-3 grid grid-cols-3 gap-2", children: imageEvidence.map((e, i) => (_jsx("button", { type: "button", className: "cursor-pointer overflow-hidden rounded-md border", onClick: () => handleThumbnailClick(i), children: _jsx("img", { src: buildEvidenceUrl(e.relativePath), alt: e.description, className: "aspect-square w-full object-cover", loading: "lazy" }) }, `thumb-${e.relativePath}`))) })) : /* eslint-enable @next/next/no-img-element */
84
- null, nonImageEvidence.length > 0 ? (_jsx("ul", { className: "space-y-2", children: nonImageEvidence.map((e) => (_jsx(EvidenceItem, { evidence: e }, `${e.type}-${e.relativePath}`))) })) : null] }), _jsx(EvidenceLightbox, { images: imageEvidence, open: lightboxOpen, onOpenChange: setLightboxOpen, selectedIndex: selectedIndex, onSelectedIndexChange: setSelectedIndex })] }));
58
+ return (_jsx("div", { className: "border-border rounded-lg border", children: _jsxs("div", { className: "px-4 py-3", children: [_jsxs("div", { className: "mb-3 flex items-center gap-2", children: [_jsx(Camera, { className: "text-muted-foreground h-4 w-4" }), _jsx("span", { className: "text-foreground text-xs font-semibold", children: "Evidence" }), _jsx(Badge, { variant: "secondary", className: "text-[10px]", children: evidence.length })] }), _jsx("ul", { className: "space-y-2", children: evidence.map((e) => (_jsx(EvidenceItem, { evidence: e }, `${e.type}-${e.relativePath}`))) })] }) }));
85
59
  }
86
60
  export function MergeReview({ data, readOnly = false, onApprove, onReject, isProcessing = false, isRejecting = false, chatInput, onChatInputChange, }) {
87
61
  const { pr, diffSummary, fileDiffs, branch, warning, evidence, hideCiStatus } = data;
@@ -1 +1 @@
1
- {"version":3,"file":"open-action-menu.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/open-action-menu/open-action-menu.tsx"],"names":[],"mappings":"AAsBA,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,UAAU,CAAC;AAIpD,wBAAgB,cAAc,CAAC,EAC7B,OAAO,EACP,cAAc,EACd,YAAY,EACZ,SAAS,GACV,EAAE,mBAAmB,2CAoFrB"}
1
+ {"version":3,"file":"open-action-menu.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/open-action-menu/open-action-menu.tsx"],"names":[],"mappings":"AAsBA,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,UAAU,CAAC;AAIpD,wBAAgB,cAAc,CAAC,EAC7B,OAAO,EACP,cAAc,EACd,YAAY,EACZ,SAAS,GACV,EAAE,mBAAmB,2CAqGrB"}
@@ -12,7 +12,7 @@ export function OpenActionMenu({ actions, repositoryPath, worktreePath, showSpec
12
12
  setCopied(true);
13
13
  setTimeout(() => setCopied(false), COPY_FEEDBACK_DELAY);
14
14
  };
15
- const anyLoading = actions.ideLoading || actions.shellLoading || actions.specsLoading;
16
- const anyError = actions.ideError ?? actions.shellError ?? actions.specsError;
17
- return (_jsxs(DropdownMenu, { modal: false, children: [_jsx(DropdownMenuTrigger, { asChild: true, children: _jsxs(Button, { variant: "outline", size: "sm", className: "gap-1.5", disabled: anyLoading, children: [anyLoading ? (_jsx(Loader2, { className: "size-4 animate-spin" })) : anyError ? (_jsx(CircleAlert, { className: "text-destructive size-4" })) : (_jsx(FolderOpen, { className: "size-4" })), "Open", _jsx(ChevronDown, { className: "size-3 opacity-60" })] }) }), _jsxs(DropdownMenuContent, { align: "start", className: "w-48", children: [_jsx(DropdownMenuLabel, { children: "Open in" }), _jsxs(DropdownMenuItem, { onClick: actions.openInIde, disabled: actions.ideLoading, className: "gap-2", children: [actions.ideLoading ? (_jsx(Loader2, { className: "size-4 animate-spin" })) : actions.ideError ? (_jsx(CircleAlert, { className: "text-destructive size-4" })) : (_jsx(Code2, { className: "size-4" })), "IDE"] }), _jsxs(DropdownMenuItem, { onClick: actions.openInShell, disabled: actions.shellLoading, className: "gap-2", children: [actions.shellLoading ? (_jsx(Loader2, { className: "size-4 animate-spin" })) : actions.shellError ? (_jsx(CircleAlert, { className: "text-destructive size-4" })) : (_jsx(Terminal, { className: "size-4" })), "Terminal"] }), _jsxs(DropdownMenuItem, { onClick: actions.openSpecsFolder, disabled: actions.specsLoading || !showSpecs, className: "gap-2", children: [actions.specsLoading ? (_jsx(Loader2, { className: "size-4 animate-spin" })) : actions.specsError ? (_jsx(CircleAlert, { className: "text-destructive size-4" })) : (_jsx(FolderOpen, { className: "size-4" })), "Specs Folder"] }), _jsx(DropdownMenuSeparator, {}), _jsxs(DropdownMenuItem, { onClick: handleCopyPath, className: "gap-2", children: [copied ? _jsx(Check, { className: "size-4 text-green-600" }) : _jsx(Copy, { className: "size-4" }), copied ? 'Copied!' : 'Copy path'] })] })] }));
15
+ const anyLoading = actions.ideLoading || actions.shellLoading || actions.folderLoading || actions.specsLoading;
16
+ const anyError = actions.ideError ?? actions.shellError ?? actions.folderError ?? actions.specsError;
17
+ return (_jsxs(DropdownMenu, { modal: false, children: [_jsx(DropdownMenuTrigger, { asChild: true, children: _jsxs(Button, { variant: "outline", size: "sm", className: "gap-1.5", disabled: anyLoading, children: [anyLoading ? (_jsx(Loader2, { className: "size-4 animate-spin" })) : anyError ? (_jsx(CircleAlert, { className: "text-destructive size-4" })) : (_jsx(FolderOpen, { className: "size-4" })), "Open", _jsx(ChevronDown, { className: "size-3 opacity-60" })] }) }), _jsxs(DropdownMenuContent, { align: "start", className: "w-48", children: [_jsx(DropdownMenuLabel, { children: "Open in" }), _jsxs(DropdownMenuItem, { onClick: actions.openInIde, disabled: actions.ideLoading, className: "gap-2", children: [actions.ideLoading ? (_jsx(Loader2, { className: "size-4 animate-spin" })) : actions.ideError ? (_jsx(CircleAlert, { className: "text-destructive size-4" })) : (_jsx(Code2, { className: "size-4" })), "IDE"] }), _jsxs(DropdownMenuItem, { onClick: actions.openInShell, disabled: actions.shellLoading, className: "gap-2", children: [actions.shellLoading ? (_jsx(Loader2, { className: "size-4 animate-spin" })) : actions.shellError ? (_jsx(CircleAlert, { className: "text-destructive size-4" })) : (_jsx(Terminal, { className: "size-4" })), "Terminal"] }), _jsxs(DropdownMenuItem, { onClick: actions.openFolder, disabled: actions.folderLoading, className: "gap-2", children: [actions.folderLoading ? (_jsx(Loader2, { className: "size-4 animate-spin" })) : actions.folderError ? (_jsx(CircleAlert, { className: "text-destructive size-4" })) : (_jsx(FolderOpen, { className: "size-4" })), "Folder"] }), _jsxs(DropdownMenuItem, { onClick: actions.openSpecsFolder, disabled: actions.specsLoading || !showSpecs, className: "gap-2", children: [actions.specsLoading ? (_jsx(Loader2, { className: "size-4 animate-spin" })) : actions.specsError ? (_jsx(CircleAlert, { className: "text-destructive size-4" })) : (_jsx(FolderOpen, { className: "size-4" })), "Specs Folder"] }), _jsx(DropdownMenuSeparator, {}), _jsxs(DropdownMenuItem, { onClick: handleCopyPath, className: "gap-2", children: [copied ? _jsx(Check, { className: "size-4 text-green-600" }) : _jsx(Copy, { className: "size-4" }), copied ? 'Copied!' : 'Copy path'] })] })] }));
18
18
  }
@@ -1 +1 @@
1
- {"version":3,"file":"open-action-menu.stories.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/open-action-menu/open-action-menu.stories.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,kBAAkB,CAAC;AAEvD,OAAO,EAAE,cAAc,EAAE,MAAM,oBAAoB,CAAC;AAkBpD,QAAA,MAAM,IAAI,EAAE,IAAI,CAAC,OAAO,cAAc,CAOrC,CAAC;AAEF,eAAe,IAAI,CAAC;AACpB,KAAK,KAAK,GAAG,QAAQ,CAAC,OAAO,cAAc,CAAC,CAAC;AAE7C,4DAA4D;AAC5D,eAAO,MAAM,OAAO,EAAE,KAMrB,CAAC;AAEF,wCAAwC;AACxC,eAAO,MAAM,YAAY,EAAE,KAM1B,CAAC;AAEF,wCAAwC;AACxC,eAAO,MAAM,OAAO,EAAE,KAMrB,CAAC;AAEF,4CAA4C;AAC5C,eAAO,MAAM,SAAS,EAAE,KAMvB,CAAC;AAEF,sEAAsE;AACtE,eAAO,MAAM,gBAAgB,EAAE,KAO9B,CAAC"}
1
+ {"version":3,"file":"open-action-menu.stories.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/open-action-menu/open-action-menu.stories.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAE,QAAQ,EAAE,MAAM,kBAAkB,CAAC;AAEvD,OAAO,EAAE,cAAc,EAAE,MAAM,oBAAoB,CAAC;AAqBpD,QAAA,MAAM,IAAI,EAAE,IAAI,CAAC,OAAO,cAAc,CAOrC,CAAC;AAEF,eAAe,IAAI,CAAC;AACpB,KAAK,KAAK,GAAG,QAAQ,CAAC,OAAO,cAAc,CAAC,CAAC;AAE7C,4DAA4D;AAC5D,eAAO,MAAM,OAAO,EAAE,KAMrB,CAAC;AAEF,wCAAwC;AACxC,eAAO,MAAM,YAAY,EAAE,KAM1B,CAAC;AAEF,wCAAwC;AACxC,eAAO,MAAM,OAAO,EAAE,KAMrB,CAAC;AAEF,4CAA4C;AAC5C,eAAO,MAAM,SAAS,EAAE,KAMvB,CAAC;AAEF,sEAAsE;AACtE,eAAO,MAAM,gBAAgB,EAAE,KAO9B,CAAC"}
@@ -3,14 +3,17 @@ import { OpenActionMenu } from './open-action-menu.js';
3
3
  const defaultActions = {
4
4
  openInIde: fn().mockName('openInIde'),
5
5
  openInShell: fn().mockName('openInShell'),
6
+ openFolder: fn().mockName('openFolder'),
6
7
  openSpecsFolder: fn().mockName('openSpecsFolder'),
7
8
  rebaseOnMain: fn().mockName('rebaseOnMain'),
8
9
  ideLoading: false,
9
10
  shellLoading: false,
11
+ folderLoading: false,
10
12
  specsLoading: false,
11
13
  rebaseLoading: false,
12
14
  ideError: null,
13
15
  shellError: null,
16
+ folderError: null,
14
17
  specsError: null,
15
18
  rebaseError: null,
16
19
  };
@@ -1 +1 @@
1
- {"version":3,"file":"control-center-inner.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/features/control-center/control-center-inner.tsx"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,IAAI,EAAY,MAAM,eAAe,CAAC;AAKpD,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,uCAAuC,CAAC;AAwB5E,UAAU,uBAAuB;IAC/B,YAAY,EAAE,cAAc,EAAE,CAAC;IAC/B,YAAY,EAAE,IAAI,EAAE,CAAC;CACtB;AAED,wBAAgB,kBAAkB,CAAC,EAAE,YAAY,EAAE,YAAY,EAAE,EAAE,uBAAuB,2CAsYzF"}
1
+ {"version":3,"file":"control-center-inner.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/features/control-center/control-center-inner.tsx"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,IAAI,EAAY,MAAM,eAAe,CAAC;AAKpD,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,uCAAuC,CAAC;AAwB5E,UAAU,uBAAuB;IAC/B,YAAY,EAAE,cAAc,EAAE,CAAC;IAC/B,YAAY,EAAE,IAAI,EAAE,CAAC;CACtB;AAED,wBAAgB,kBAAkB,CAAC,EAAE,YAAY,EAAE,YAAY,EAAE,EAAE,uBAAuB,2CA0ZzF"}
@@ -197,6 +197,24 @@ export function ControlCenterInner({ initialNodes, initialEdges }) {
197
197
  window.addEventListener('shep:feature-delete-requested', handler);
198
198
  return () => window.removeEventListener('shep:feature-delete-requested', handler);
199
199
  }, [handleDeleteFeature]);
200
+ // Listen for archive requests from the feature drawer.
201
+ useEffect(() => {
202
+ const handler = (e) => {
203
+ const { featureId } = e.detail;
204
+ handleArchiveFeature(featureId);
205
+ };
206
+ window.addEventListener('shep:feature-archive-requested', handler);
207
+ return () => window.removeEventListener('shep:feature-archive-requested', handler);
208
+ }, [handleArchiveFeature]);
209
+ // Listen for unarchive requests from the feature drawer.
210
+ useEffect(() => {
211
+ const handler = (e) => {
212
+ const { featureId } = e.detail;
213
+ handleUnarchiveFeature(featureId);
214
+ };
215
+ window.addEventListener('shep:feature-unarchive-requested', handler);
216
+ return () => window.removeEventListener('shep:feature-unarchive-requested', handler);
217
+ }, [handleUnarchiveFeature]);
200
218
  // Wire callbacks into derived node data (via ref — no re-render).
201
219
  useEffect(() => {
202
220
  setCallbacks({