@shepai/cli 1.162.0 → 1.163.0-pr499.0ce2086

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 (394) hide show
  1. package/README.md +12 -0
  2. package/apis/json-schema/FabLayoutConfig.yaml +11 -0
  3. package/apis/json-schema/Settings.yaml +3 -0
  4. package/dist/packages/core/src/application/ports/output/services/github-repository-service.interface.d.ts +20 -0
  5. package/dist/packages/core/src/application/ports/output/services/github-repository-service.interface.d.ts.map +1 -1
  6. package/dist/packages/core/src/application/ports/output/services/index.d.ts +1 -1
  7. package/dist/packages/core/src/application/ports/output/services/index.d.ts.map +1 -1
  8. package/dist/packages/core/src/application/use-cases/repositories/list-github-organizations.use-case.d.ts +13 -0
  9. package/dist/packages/core/src/application/use-cases/repositories/list-github-organizations.use-case.d.ts.map +1 -0
  10. package/dist/packages/core/src/application/use-cases/repositories/list-github-organizations.use-case.js +35 -0
  11. package/dist/packages/core/src/domain/generated/output.d.ts +13 -0
  12. package/dist/packages/core/src/domain/generated/output.d.ts.map +1 -1
  13. package/dist/packages/core/src/infrastructure/di/container.d.ts.map +1 -1
  14. package/dist/packages/core/src/infrastructure/di/container.js +5 -0
  15. package/dist/packages/core/src/infrastructure/persistence/sqlite/mappers/settings.mapper.d.ts +1 -0
  16. package/dist/packages/core/src/infrastructure/persistence/sqlite/mappers/settings.mapper.d.ts.map +1 -1
  17. package/dist/packages/core/src/infrastructure/persistence/sqlite/mappers/settings.mapper.js +6 -0
  18. package/dist/packages/core/src/infrastructure/persistence/sqlite/migrations/050-add-fab-position-swapped.d.ts +15 -0
  19. package/dist/packages/core/src/infrastructure/persistence/sqlite/migrations/050-add-fab-position-swapped.d.ts.map +1 -0
  20. package/dist/packages/core/src/infrastructure/persistence/sqlite/migrations/050-add-fab-position-swapped.js +20 -0
  21. package/dist/packages/core/src/infrastructure/services/external/github-repository.service.d.ts +2 -1
  22. package/dist/packages/core/src/infrastructure/services/external/github-repository.service.d.ts.map +1 -1
  23. package/dist/packages/core/src/infrastructure/services/external/github-repository.service.js +20 -0
  24. package/dist/src/presentation/cli/commands/repo/add.command.d.ts.map +1 -1
  25. package/dist/src/presentation/cli/commands/repo/add.command.js +3 -1
  26. package/dist/src/presentation/tui/wizards/github-import.wizard.d.ts +5 -2
  27. package/dist/src/presentation/tui/wizards/github-import.wizard.d.ts.map +1 -1
  28. package/dist/src/presentation/tui/wizards/github-import.wizard.js +38 -6
  29. package/dist/src/presentation/web/app/actions/get-git-log.d.ts +44 -0
  30. package/dist/src/presentation/web/app/actions/get-git-log.d.ts.map +1 -0
  31. package/dist/src/presentation/web/app/actions/get-git-log.js +125 -0
  32. package/dist/src/presentation/web/app/actions/list-github-organizations.d.ts +6 -0
  33. package/dist/src/presentation/web/app/actions/list-github-organizations.d.ts.map +1 -0
  34. package/dist/src/presentation/web/app/actions/list-github-organizations.js +17 -0
  35. package/dist/src/presentation/web/app/actions/list-github-repositories.d.ts +1 -0
  36. package/dist/src/presentation/web/app/actions/list-github-repositories.d.ts.map +1 -1
  37. package/dist/src/presentation/web/app/actions/open-folder.d.ts.map +1 -1
  38. package/dist/src/presentation/web/app/actions/open-folder.js +5 -2
  39. package/dist/src/presentation/web/app/layout.d.ts.map +1 -1
  40. package/dist/src/presentation/web/app/layout.js +3 -1
  41. package/dist/src/presentation/web/components/common/base-drawer/base-drawer.js +1 -1
  42. package/dist/src/presentation/web/components/common/control-center-drawer/feature-drawer-client.d.ts.map +1 -1
  43. package/dist/src/presentation/web/components/common/control-center-drawer/feature-drawer-client.js +46 -15
  44. package/dist/src/presentation/web/components/common/control-center-drawer/repository-drawer-client.d.ts.map +1 -1
  45. package/dist/src/presentation/web/components/common/control-center-drawer/repository-drawer-client.js +77 -5
  46. package/dist/src/presentation/web/components/common/control-center-drawer/use-drawer-sync.d.ts.map +1 -1
  47. package/dist/src/presentation/web/components/common/control-center-drawer/use-drawer-sync.js +11 -0
  48. package/dist/src/presentation/web/components/common/deployment-status-badge/deployment-status-badge.d.ts.map +1 -1
  49. package/dist/src/presentation/web/components/common/deployment-status-badge/deployment-status-badge.js +10 -9
  50. package/dist/src/presentation/web/components/common/feature-drawer-tabs/feature-drawer-tabs.d.ts +5 -1
  51. package/dist/src/presentation/web/components/common/feature-drawer-tabs/feature-drawer-tabs.d.ts.map +1 -1
  52. package/dist/src/presentation/web/components/common/feature-drawer-tabs/feature-drawer-tabs.js +24 -13
  53. package/dist/src/presentation/web/components/common/feature-drawer-tabs/overview-tab.d.ts.map +1 -1
  54. package/dist/src/presentation/web/components/common/feature-drawer-tabs/overview-tab.js +56 -81
  55. package/dist/src/presentation/web/components/common/github-import-dialog/github-repo-browser.d.ts +8 -1
  56. package/dist/src/presentation/web/components/common/github-import-dialog/github-repo-browser.d.ts.map +1 -1
  57. package/dist/src/presentation/web/components/common/github-import-dialog/github-repo-browser.js +37 -7
  58. package/dist/src/presentation/web/components/common/github-import-dialog/github-repo-browser.stories.d.ts +1 -0
  59. package/dist/src/presentation/web/components/common/github-import-dialog/github-repo-browser.stories.d.ts.map +1 -1
  60. package/dist/src/presentation/web/components/common/github-import-dialog/github-repo-browser.stories.js +35 -0
  61. package/dist/src/presentation/web/components/common/open-action-menu/open-action-menu.d.ts.map +1 -1
  62. package/dist/src/presentation/web/components/common/open-action-menu/open-action-menu.js +11 -6
  63. package/dist/src/presentation/web/components/common/repository-node/repository-node.d.ts.map +1 -1
  64. package/dist/src/presentation/web/components/features/chat/ChatSheet.d.ts.map +1 -1
  65. package/dist/src/presentation/web/components/features/chat/ChatSheet.js +22 -2
  66. package/dist/src/presentation/web/components/features/control-center/control-center-inner.d.ts.map +1 -1
  67. package/dist/src/presentation/web/components/features/control-center/control-center-inner.js +15 -1
  68. package/dist/src/presentation/web/components/features/settings/settings-page-client.d.ts.map +1 -1
  69. package/dist/src/presentation/web/components/features/settings/settings-page-client.js +9 -2
  70. package/dist/src/presentation/web/components/layouts/app-sidebar/app-sidebar.d.ts.map +1 -1
  71. package/dist/src/presentation/web/components/layouts/app-sidebar/app-sidebar.js +7 -2
  72. package/dist/src/presentation/web/hooks/fab-layout-context.d.ts +18 -0
  73. package/dist/src/presentation/web/hooks/fab-layout-context.d.ts.map +1 -0
  74. package/dist/src/presentation/web/hooks/fab-layout-context.js +19 -0
  75. package/dist/src/presentation/web/hooks/use-animations-enabled.d.ts +6 -0
  76. package/dist/src/presentation/web/hooks/use-animations-enabled.d.ts.map +1 -0
  77. package/dist/src/presentation/web/hooks/use-animations-enabled.js +30 -0
  78. package/dist/src/presentation/web/lib/fab-layout.d.ts +11 -0
  79. package/dist/src/presentation/web/lib/fab-layout.d.ts.map +1 -0
  80. package/dist/src/presentation/web/lib/fab-layout.js +21 -0
  81. package/dist/translations/ar/web.json +8 -0
  82. package/dist/translations/de/web.json +8 -0
  83. package/dist/translations/en/web.json +8 -0
  84. package/dist/translations/es/web.json +8 -0
  85. package/dist/translations/fr/web.json +8 -0
  86. package/dist/translations/he/web.json +8 -0
  87. package/dist/translations/pt/web.json +8 -0
  88. package/dist/translations/ru/web.json +8 -0
  89. package/dist/tsconfig.build.tsbuildinfo +1 -1
  90. package/package.json +1 -1
  91. package/web/.next/BUILD_ID +1 -1
  92. package/web/.next/build-manifest.json +2 -2
  93. package/web/.next/fallback-build-manifest.json +2 -2
  94. package/web/.next/prerender-manifest.json +3 -3
  95. package/web/.next/required-server-files.js +3 -3
  96. package/web/.next/required-server-files.json +3 -3
  97. package/web/.next/server/app/(dashboard)/@drawer/adopt/page/server-reference-manifest.json +71 -56
  98. package/web/.next/server/app/(dashboard)/@drawer/adopt/page.js +2 -2
  99. package/web/.next/server/app/(dashboard)/@drawer/adopt/page.js.nft.json +1 -1
  100. package/web/.next/server/app/(dashboard)/@drawer/adopt/page_client-reference-manifest.js +1 -1
  101. package/web/.next/server/app/(dashboard)/@drawer/chat/page/server-reference-manifest.json +67 -52
  102. package/web/.next/server/app/(dashboard)/@drawer/chat/page.js +2 -2
  103. package/web/.next/server/app/(dashboard)/@drawer/chat/page.js.nft.json +1 -1
  104. package/web/.next/server/app/(dashboard)/@drawer/chat/page_client-reference-manifest.js +1 -1
  105. package/web/.next/server/app/(dashboard)/@drawer/create/page/server-reference-manifest.json +73 -58
  106. package/web/.next/server/app/(dashboard)/@drawer/create/page.js +2 -2
  107. package/web/.next/server/app/(dashboard)/@drawer/create/page.js.nft.json +1 -1
  108. package/web/.next/server/app/(dashboard)/@drawer/create/page_client-reference-manifest.js +1 -1
  109. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page/server-reference-manifest.json +87 -72
  110. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page.js +2 -2
  111. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page.js.nft.json +1 -1
  112. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page_client-reference-manifest.js +1 -1
  113. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page/server-reference-manifest.json +87 -72
  114. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page.js +2 -2
  115. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page.js.nft.json +1 -1
  116. package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page_client-reference-manifest.js +1 -1
  117. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/[tab]/page/server-reference-manifest.json +82 -52
  118. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/[tab]/page.js +2 -2
  119. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/[tab]/page.js.nft.json +1 -1
  120. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/[tab]/page_client-reference-manifest.js +1 -1
  121. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page/server-reference-manifest.json +82 -52
  122. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page.js +2 -2
  123. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page.js.nft.json +1 -1
  124. package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page_client-reference-manifest.js +1 -1
  125. package/web/.next/server/app/(dashboard)/chat/page/server-reference-manifest.json +67 -52
  126. package/web/.next/server/app/(dashboard)/chat/page.js +2 -2
  127. package/web/.next/server/app/(dashboard)/chat/page.js.nft.json +1 -1
  128. package/web/.next/server/app/(dashboard)/chat/page_client-reference-manifest.js +1 -1
  129. package/web/.next/server/app/(dashboard)/create/page/server-reference-manifest.json +73 -58
  130. package/web/.next/server/app/(dashboard)/create/page.js +2 -2
  131. package/web/.next/server/app/(dashboard)/create/page.js.nft.json +1 -1
  132. package/web/.next/server/app/(dashboard)/create/page_client-reference-manifest.js +1 -1
  133. package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page/server-reference-manifest.json +87 -72
  134. package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page.js +2 -2
  135. package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page.js.nft.json +1 -1
  136. package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page_client-reference-manifest.js +1 -1
  137. package/web/.next/server/app/(dashboard)/feature/[featureId]/page/server-reference-manifest.json +87 -72
  138. package/web/.next/server/app/(dashboard)/feature/[featureId]/page.js +2 -2
  139. package/web/.next/server/app/(dashboard)/feature/[featureId]/page.js.nft.json +1 -1
  140. package/web/.next/server/app/(dashboard)/feature/[featureId]/page_client-reference-manifest.js +1 -1
  141. package/web/.next/server/app/(dashboard)/page/server-reference-manifest.json +67 -52
  142. package/web/.next/server/app/(dashboard)/page.js +2 -2
  143. package/web/.next/server/app/(dashboard)/page.js.nft.json +1 -1
  144. package/web/.next/server/app/(dashboard)/page_client-reference-manifest.js +1 -1
  145. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/[tab]/page/server-reference-manifest.json +82 -52
  146. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/[tab]/page.js +2 -2
  147. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/[tab]/page.js.nft.json +1 -1
  148. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/[tab]/page_client-reference-manifest.js +1 -1
  149. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page/server-reference-manifest.json +82 -52
  150. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page.js +2 -2
  151. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page.js.nft.json +1 -1
  152. package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page_client-reference-manifest.js +1 -1
  153. package/web/.next/server/app/_global-error/page_client-reference-manifest.js +1 -1
  154. package/web/.next/server/app/_global-error.html +2 -2
  155. package/web/.next/server/app/_global-error.rsc +1 -1
  156. package/web/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +1 -1
  157. package/web/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  158. package/web/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  159. package/web/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  160. package/web/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  161. package/web/.next/server/app/_not-found/page/server-reference-manifest.json +25 -10
  162. package/web/.next/server/app/_not-found/page.js +2 -2
  163. package/web/.next/server/app/_not-found/page.js.nft.json +1 -1
  164. package/web/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  165. package/web/.next/server/app/api/attachments/preview/route.js.nft.json +1 -1
  166. package/web/.next/server/app/api/evidence/route.js.nft.json +1 -1
  167. package/web/.next/server/app/api/graph-data/route.js +1 -1
  168. package/web/.next/server/app/api/graph-data/route.js.nft.json +1 -1
  169. package/web/.next/server/app/api/interactive/chat/[featureId]/messages/route.js.nft.json +1 -1
  170. package/web/.next/server/app/settings/page/server-reference-manifest.json +31 -16
  171. package/web/.next/server/app/settings/page.js +2 -2
  172. package/web/.next/server/app/settings/page.js.nft.json +1 -1
  173. package/web/.next/server/app/settings/page_client-reference-manifest.js +1 -1
  174. package/web/.next/server/app/skills/page/server-reference-manifest.json +35 -20
  175. package/web/.next/server/app/skills/page.js +2 -2
  176. package/web/.next/server/app/skills/page.js.nft.json +1 -1
  177. package/web/.next/server/app/skills/page_client-reference-manifest.js +1 -1
  178. package/web/.next/server/app/tools/page/server-reference-manifest.json +35 -20
  179. package/web/.next/server/app/tools/page.js +2 -2
  180. package/web/.next/server/app/tools/page.js.nft.json +1 -1
  181. package/web/.next/server/app/tools/page_client-reference-manifest.js +1 -1
  182. package/web/.next/server/app/version/page/server-reference-manifest.json +25 -10
  183. package/web/.next/server/app/version/page.js +2 -2
  184. package/web/.next/server/app/version/page.js.nft.json +1 -1
  185. package/web/.next/server/app/version/page_client-reference-manifest.js +1 -1
  186. package/web/.next/server/chunks/[root-of-the-server]__a402b567._.js +1 -1
  187. package/web/.next/server/chunks/{[root-of-the-server]__beda892a._.js → [root-of-the-server]__c78383b1._.js} +2 -2
  188. package/web/.next/server/chunks/[root-of-the-server]__c78383b1._.js.map +1 -0
  189. package/web/.next/server/chunks/[root-of-the-server]__cd67a84c._.js.map +1 -1
  190. package/web/.next/server/chunks/ssr/403f9_next_dist_esm_ceb2fa1e._.js +1 -1
  191. package/web/.next/server/chunks/ssr/744ca_web_components_common_control-center-drawer_create-drawer-client_tsx_5e26fc0a._.js +1 -1
  192. package/web/.next/server/chunks/ssr/744ca_web_components_common_control-center-drawer_create-drawer-client_tsx_5e26fc0a._.js.map +1 -1
  193. package/web/.next/server/chunks/ssr/744ca_web_components_common_control-center-drawer_feature-drawer-client_tsx_e9755fc8._.js +3 -3
  194. package/web/.next/server/chunks/ssr/744ca_web_components_common_control-center-drawer_feature-drawer-client_tsx_e9755fc8._.js.map +1 -1
  195. package/web/.next/server/chunks/ssr/{7f428_lucide-react_dist_esm_icons_4b319ae6._.js → 7f428_lucide-react_dist_esm_icons_281e0ef8._.js} +2 -2
  196. package/web/.next/server/chunks/ssr/7f428_lucide-react_dist_esm_icons_281e0ef8._.js.map +1 -0
  197. package/web/.next/server/chunks/ssr/7f428_lucide-react_dist_esm_icons_a593f310._.js +3 -0
  198. package/web/.next/server/chunks/ssr/7f428_lucide-react_dist_esm_icons_a593f310._.js.map +1 -0
  199. package/web/.next/server/chunks/ssr/[root-of-the-server]__1abe77bb._.js +4 -0
  200. package/web/.next/server/chunks/ssr/[root-of-the-server]__1abe77bb._.js.map +1 -0
  201. package/web/.next/server/chunks/ssr/[root-of-the-server]__1f389e5d._.js +4 -0
  202. package/web/.next/server/chunks/ssr/[root-of-the-server]__1f389e5d._.js.map +1 -0
  203. package/web/.next/server/chunks/ssr/[root-of-the-server]__23b5ca2c._.js +1 -1
  204. package/web/.next/server/chunks/ssr/[root-of-the-server]__357d99f9._.js +1 -1
  205. package/web/.next/server/chunks/ssr/[root-of-the-server]__563e4faf._.js +4 -0
  206. package/web/.next/server/chunks/ssr/[root-of-the-server]__563e4faf._.js.map +1 -0
  207. package/web/.next/server/chunks/ssr/[root-of-the-server]__6eb2a2b0._.js +1 -1
  208. package/web/.next/server/chunks/ssr/[root-of-the-server]__7528eb6f._.js +1 -1
  209. package/web/.next/server/chunks/ssr/[root-of-the-server]__7562afc6._.js +4 -0
  210. package/web/.next/server/chunks/ssr/[root-of-the-server]__7562afc6._.js.map +1 -0
  211. package/web/.next/server/chunks/ssr/[root-of-the-server]__821a11c1._.js +4 -0
  212. package/web/.next/server/chunks/ssr/[root-of-the-server]__821a11c1._.js.map +1 -0
  213. package/web/.next/server/chunks/ssr/[root-of-the-server]__86ff0bc5._.js +7 -0
  214. package/web/.next/server/chunks/ssr/[root-of-the-server]__86ff0bc5._.js.map +1 -0
  215. package/web/.next/server/chunks/ssr/[root-of-the-server]__8b0aac03._.js +1 -1
  216. package/web/.next/server/chunks/ssr/[root-of-the-server]__8b0aac03._.js.map +1 -1
  217. package/web/.next/server/chunks/ssr/[root-of-the-server]__98740ee4._.js +4 -0
  218. package/web/.next/server/chunks/ssr/[root-of-the-server]__98740ee4._.js.map +1 -0
  219. package/web/.next/server/chunks/ssr/[root-of-the-server]__b7b96453._.js +4 -0
  220. package/web/.next/server/chunks/ssr/[root-of-the-server]__b7b96453._.js.map +1 -0
  221. package/web/.next/server/chunks/ssr/[root-of-the-server]__ba9f9e11._.js +4 -0
  222. package/web/.next/server/chunks/ssr/[root-of-the-server]__ba9f9e11._.js.map +1 -0
  223. package/web/.next/server/chunks/ssr/[root-of-the-server]__e0be67c7._.js +3 -0
  224. package/web/.next/server/chunks/ssr/[root-of-the-server]__e0be67c7._.js.map +1 -0
  225. package/web/.next/server/chunks/ssr/_02e01240._.js +4 -0
  226. package/web/.next/server/chunks/ssr/_02e01240._.js.map +1 -0
  227. package/web/.next/server/chunks/ssr/_05c23ad9._.js +1 -1
  228. package/web/.next/server/chunks/ssr/_05c23ad9._.js.map +1 -1
  229. package/web/.next/server/chunks/ssr/{_9495d50b._.js → _0727935d._.js} +2 -2
  230. package/web/.next/server/chunks/ssr/{_9495d50b._.js.map → _0727935d._.js.map} +1 -1
  231. package/web/.next/server/chunks/ssr/_0dc06d07._.js.map +1 -1
  232. package/web/.next/server/chunks/ssr/_16eb4fec._.js +1 -1
  233. package/web/.next/server/chunks/ssr/_16eb4fec._.js.map +1 -1
  234. package/web/.next/server/chunks/ssr/{_bcaea66b._.js → _1879404a._.js} +2 -2
  235. package/web/.next/server/chunks/ssr/_1879404a._.js.map +1 -0
  236. package/web/.next/server/chunks/ssr/_18886033._.js +4 -0
  237. package/web/.next/server/chunks/ssr/_18886033._.js.map +1 -0
  238. package/web/.next/server/chunks/ssr/{_b9ba2473._.js → _1e08a336._.js} +2 -2
  239. package/web/.next/server/chunks/ssr/_1e08a336._.js.map +1 -0
  240. package/web/.next/server/chunks/ssr/_22e00a14._.js +4 -0
  241. package/web/.next/server/chunks/ssr/_22e00a14._.js.map +1 -0
  242. package/web/.next/server/chunks/ssr/{_07635b7c._.js → _3625b35c._.js} +2 -2
  243. package/web/.next/server/chunks/ssr/_3625b35c._.js.map +1 -0
  244. package/web/.next/server/chunks/ssr/_4d49a312._.js +3 -0
  245. package/web/.next/server/chunks/ssr/_4d49a312._.js.map +1 -0
  246. package/web/.next/server/chunks/ssr/_5119a3df._.js.map +1 -1
  247. package/web/.next/server/chunks/ssr/_56b9d60f._.js +1 -1
  248. package/web/.next/server/chunks/ssr/_56b9d60f._.js.map +1 -1
  249. package/web/.next/server/chunks/ssr/_9215e9ec._.js +3 -0
  250. package/web/.next/server/chunks/ssr/_9215e9ec._.js.map +1 -0
  251. package/web/.next/server/chunks/ssr/_954c5b3f._.js +3 -0
  252. package/web/.next/server/chunks/ssr/_954c5b3f._.js.map +1 -0
  253. package/web/.next/server/chunks/ssr/_a5a5901d._.js +4 -0
  254. package/web/.next/server/chunks/ssr/_a5a5901d._.js.map +1 -0
  255. package/web/.next/server/chunks/ssr/_ad09f271._.js +4 -0
  256. package/web/.next/server/chunks/ssr/_ad09f271._.js.map +1 -0
  257. package/web/.next/server/chunks/ssr/_b7a43c05._.js +3 -0
  258. package/web/.next/server/chunks/ssr/_b7a43c05._.js.map +1 -0
  259. package/web/.next/server/chunks/ssr/_c3f595c6._.js +4 -0
  260. package/web/.next/server/chunks/ssr/_c3f595c6._.js.map +1 -0
  261. package/web/.next/server/chunks/ssr/_c45aee16._.js +3 -0
  262. package/web/.next/server/chunks/ssr/_c45aee16._.js.map +1 -0
  263. package/web/.next/server/chunks/ssr/{_1c309165._.js → _cc654b75._.js} +3 -3
  264. package/web/.next/server/chunks/ssr/_cc654b75._.js.map +1 -0
  265. package/web/.next/server/chunks/ssr/_ea9e1556._.js +4 -0
  266. package/web/.next/server/chunks/ssr/_ea9e1556._.js.map +1 -0
  267. package/web/.next/server/chunks/ssr/{_73e1eb5f._.js → _ea9f5467._.js} +2 -2
  268. package/web/.next/server/chunks/ssr/{_73e1eb5f._.js.map → _ea9f5467._.js.map} +1 -1
  269. package/web/.next/server/chunks/ssr/_f1ba9be6._.js +6 -0
  270. package/web/.next/server/chunks/ssr/_f1ba9be6._.js.map +1 -0
  271. package/web/.next/server/chunks/ssr/_f33cd07e._.js +6 -0
  272. package/web/.next/server/chunks/ssr/_f33cd07e._.js.map +1 -0
  273. package/web/.next/server/chunks/ssr/_f8b45233._.js +4 -0
  274. package/web/.next/server/chunks/ssr/_f8b45233._.js.map +1 -0
  275. package/web/.next/server/chunks/ssr/b1a17_presentation_web_components_features_settings_settings-page-client_tsx_6ed9d5f8._.js +1 -1
  276. package/web/.next/server/chunks/ssr/b1a17_presentation_web_components_features_settings_settings-page-client_tsx_6ed9d5f8._.js.map +1 -1
  277. package/web/.next/server/chunks/ssr/f3a1f_components_common_control-center-drawer_repository-drawer-client_tsx_39a00c03._.js +3 -0
  278. package/web/.next/server/chunks/ssr/f3a1f_components_common_control-center-drawer_repository-drawer-client_tsx_39a00c03._.js.map +1 -0
  279. package/web/.next/server/chunks/ssr/node_modules__pnpm_1300ae39._.js +3 -0
  280. package/web/.next/server/chunks/ssr/node_modules__pnpm_1300ae39._.js.map +1 -0
  281. package/web/.next/server/chunks/ssr/src_presentation_web_app_actions_open-ide_ts_baaca5d5._.js +1 -1
  282. package/web/.next/server/chunks/ssr/src_presentation_web_app_actions_open-ide_ts_baaca5d5._.js.map +1 -1
  283. package/web/.next/server/chunks/ssr/src_presentation_web_components_895e5bfa._.js +1 -1
  284. package/web/.next/server/chunks/ssr/src_presentation_web_components_895e5bfa._.js.map +1 -1
  285. package/web/.next/server/chunks/ssr/src_presentation_web_components_features_control-center_7ac3562e._.js +1 -1
  286. package/web/.next/server/chunks/ssr/src_presentation_web_components_features_control-center_7ac3562e._.js.map +1 -1
  287. package/web/.next/server/chunks/ssr/src_presentation_web_components_features_tools_tools-page-client_tsx_3d0aa70c._.js +1 -1
  288. package/web/.next/server/chunks/ssr/src_presentation_web_components_features_tools_tools-page-client_tsx_3d0aa70c._.js.map +1 -1
  289. package/web/.next/server/chunks/ssr/src_presentation_web_components_ui_select_tsx_45d6b8ae._.js +3 -0
  290. package/web/.next/server/chunks/ssr/src_presentation_web_components_ui_select_tsx_45d6b8ae._.js.map +1 -0
  291. package/web/.next/server/chunks/ssr/translations_23dd5e7e._.js +1 -1
  292. package/web/.next/server/chunks/ssr/translations_23dd5e7e._.js.map +1 -1
  293. package/web/.next/server/pages/500.html +2 -2
  294. package/web/.next/server/server-reference-manifest.js +1 -1
  295. package/web/.next/server/server-reference-manifest.json +672 -495
  296. package/web/.next/static/chunks/039b43423300c5fb.js +7 -0
  297. package/web/.next/static/chunks/{04b3e8b2c0b2a310.js → 16faea5f198d1c15.js} +2 -2
  298. package/web/.next/static/chunks/28ae5046faef6cec.js +1 -0
  299. package/web/.next/static/chunks/29f0d874b1fde3d6.js +5 -0
  300. package/web/.next/static/chunks/2c498b3dc30a33d4.js +1 -0
  301. package/web/.next/static/chunks/3b9e9bcb040b302b.js +1 -0
  302. package/web/.next/static/chunks/400e93efac983a76.css +1 -0
  303. package/web/.next/static/chunks/5a5455d111b22e62.js +1 -0
  304. package/web/.next/static/chunks/76e2b554ca427213.js +1 -0
  305. package/web/.next/static/chunks/779b5c49587b074e.js +1 -0
  306. package/web/.next/static/chunks/89f0ff47b59fcd7b.js +1 -0
  307. package/web/.next/static/chunks/{1db4dfa35a1637ed.js → 91ff36bca978b7c0.js} +1 -1
  308. package/web/.next/static/chunks/966eb2688300b7f5.js +1 -0
  309. package/web/.next/static/chunks/971e52f3f386ccfd.js +1 -0
  310. package/web/.next/static/chunks/9d6b4bfb854588a9.js +1 -0
  311. package/web/.next/static/chunks/{2770aabb38d76609.js → a76c0a69840530a0.js} +1 -1
  312. package/web/.next/static/chunks/a8243f8d06bdcef0.js +5 -0
  313. package/web/.next/static/chunks/b63e6727c84f30e2.js +1 -0
  314. package/web/.next/static/chunks/c744b25661a572d6.js +5 -0
  315. package/web/.next/static/chunks/c89c71e1d2e4e3f6.js +1 -0
  316. package/web/.next/static/chunks/{192dd30a464572ef.js → fc52571f411eb0c7.js} +2 -2
  317. package/web/.next/server/chunks/[root-of-the-server]__beda892a._.js.map +0 -1
  318. package/web/.next/server/chunks/ssr/7f428_lucide-react_dist_esm_icons_4b319ae6._.js.map +0 -1
  319. package/web/.next/server/chunks/ssr/[root-of-the-server]__209c9597._.js +0 -4
  320. package/web/.next/server/chunks/ssr/[root-of-the-server]__209c9597._.js.map +0 -1
  321. package/web/.next/server/chunks/ssr/[root-of-the-server]__27301e38._.js +0 -4
  322. package/web/.next/server/chunks/ssr/[root-of-the-server]__27301e38._.js.map +0 -1
  323. package/web/.next/server/chunks/ssr/[root-of-the-server]__56b70465._.js +0 -4
  324. package/web/.next/server/chunks/ssr/[root-of-the-server]__56b70465._.js.map +0 -1
  325. package/web/.next/server/chunks/ssr/[root-of-the-server]__6fecf886._.js +0 -4
  326. package/web/.next/server/chunks/ssr/[root-of-the-server]__6fecf886._.js.map +0 -1
  327. package/web/.next/server/chunks/ssr/[root-of-the-server]__8d27866c._.js +0 -4
  328. package/web/.next/server/chunks/ssr/[root-of-the-server]__8d27866c._.js.map +0 -1
  329. package/web/.next/server/chunks/ssr/[root-of-the-server]__9f3504c7._.js +0 -4
  330. package/web/.next/server/chunks/ssr/[root-of-the-server]__9f3504c7._.js.map +0 -1
  331. package/web/.next/server/chunks/ssr/[root-of-the-server]__bd6ed91a._.js +0 -4
  332. package/web/.next/server/chunks/ssr/[root-of-the-server]__bd6ed91a._.js.map +0 -1
  333. package/web/.next/server/chunks/ssr/[root-of-the-server]__c30f1f82._.js +0 -4
  334. package/web/.next/server/chunks/ssr/[root-of-the-server]__c30f1f82._.js.map +0 -1
  335. package/web/.next/server/chunks/ssr/[root-of-the-server]__e91ffd5e._.js +0 -7
  336. package/web/.next/server/chunks/ssr/[root-of-the-server]__e91ffd5e._.js.map +0 -1
  337. package/web/.next/server/chunks/ssr/[root-of-the-server]__ee0eaaf9._.js +0 -3
  338. package/web/.next/server/chunks/ssr/[root-of-the-server]__ee0eaaf9._.js.map +0 -1
  339. package/web/.next/server/chunks/ssr/_07635b7c._.js.map +0 -1
  340. package/web/.next/server/chunks/ssr/_1c309165._.js.map +0 -1
  341. package/web/.next/server/chunks/ssr/_3a0b989f._.js +0 -6
  342. package/web/.next/server/chunks/ssr/_3a0b989f._.js.map +0 -1
  343. package/web/.next/server/chunks/ssr/_573caadf._.js +0 -3
  344. package/web/.next/server/chunks/ssr/_573caadf._.js.map +0 -1
  345. package/web/.next/server/chunks/ssr/_5f69c13f._.js +0 -4
  346. package/web/.next/server/chunks/ssr/_5f69c13f._.js.map +0 -1
  347. package/web/.next/server/chunks/ssr/_7c5b97c6._.js +0 -4
  348. package/web/.next/server/chunks/ssr/_7c5b97c6._.js.map +0 -1
  349. package/web/.next/server/chunks/ssr/_82c57f10._.js +0 -4
  350. package/web/.next/server/chunks/ssr/_82c57f10._.js.map +0 -1
  351. package/web/.next/server/chunks/ssr/_9c5bd84a._.js +0 -3
  352. package/web/.next/server/chunks/ssr/_9c5bd84a._.js.map +0 -1
  353. package/web/.next/server/chunks/ssr/_a0e3f7e4._.js +0 -4
  354. package/web/.next/server/chunks/ssr/_a0e3f7e4._.js.map +0 -1
  355. package/web/.next/server/chunks/ssr/_ac4a3873._.js +0 -3
  356. package/web/.next/server/chunks/ssr/_ac4a3873._.js.map +0 -1
  357. package/web/.next/server/chunks/ssr/_b9ba2473._.js.map +0 -1
  358. package/web/.next/server/chunks/ssr/_bcaea66b._.js.map +0 -1
  359. package/web/.next/server/chunks/ssr/_ca0aa7f0._.js +0 -4
  360. package/web/.next/server/chunks/ssr/_ca0aa7f0._.js.map +0 -1
  361. package/web/.next/server/chunks/ssr/_cb5a021e._.js +0 -4
  362. package/web/.next/server/chunks/ssr/_cb5a021e._.js.map +0 -1
  363. package/web/.next/server/chunks/ssr/_cfbd1d7e._.js +0 -3
  364. package/web/.next/server/chunks/ssr/_cfbd1d7e._.js.map +0 -1
  365. package/web/.next/server/chunks/ssr/_d86175ae._.js +0 -4
  366. package/web/.next/server/chunks/ssr/_d86175ae._.js.map +0 -1
  367. package/web/.next/server/chunks/ssr/_d8bedf13._.js +0 -4
  368. package/web/.next/server/chunks/ssr/_d8bedf13._.js.map +0 -1
  369. package/web/.next/server/chunks/ssr/_d90b0a06._.js +0 -3
  370. package/web/.next/server/chunks/ssr/_d90b0a06._.js.map +0 -1
  371. package/web/.next/server/chunks/ssr/_fa7efce3._.js +0 -6
  372. package/web/.next/server/chunks/ssr/_fa7efce3._.js.map +0 -1
  373. package/web/.next/server/chunks/ssr/node_modules__pnpm_8ec2c790._.js +0 -3
  374. package/web/.next/server/chunks/ssr/node_modules__pnpm_8ec2c790._.js.map +0 -1
  375. package/web/.next/static/chunks/0a79dfbb8486b66e.js +0 -5
  376. package/web/.next/static/chunks/14d9d9181bb4e337.js +0 -1
  377. package/web/.next/static/chunks/1d980f961818ff68.js +0 -1
  378. package/web/.next/static/chunks/2c9404cf96b0a5a0.js +0 -1
  379. package/web/.next/static/chunks/47477ed4c5871747.js +0 -1
  380. package/web/.next/static/chunks/4864ff06a738d58b.js +0 -1
  381. package/web/.next/static/chunks/5281c5a2011132bc.js +0 -5
  382. package/web/.next/static/chunks/63b4954715fc3423.js +0 -1
  383. package/web/.next/static/chunks/74db65fa7bfb80bd.js +0 -1
  384. package/web/.next/static/chunks/7cd7aef397d53b94.js +0 -1
  385. package/web/.next/static/chunks/83928a921868a809.js +0 -1
  386. package/web/.next/static/chunks/86c476f04ff32e7b.css +0 -1
  387. package/web/.next/static/chunks/a81350e847797ff8.js +0 -1
  388. package/web/.next/static/chunks/cb41e1da2ce3a710.js +0 -7
  389. package/web/.next/static/chunks/d0613380bbaa9a8e.js +0 -1
  390. package/web/.next/static/chunks/d2cbeefbc8967b16.js +0 -1
  391. package/web/.next/static/chunks/d6e702c209c413ce.js +0 -5
  392. /package/web/.next/static/{GXMtXdKTjklQ1IkrOohzE → SFr_DcWGpl2mSNLzCZet_}/_buildManifest.js +0 -0
  393. /package/web/.next/static/{GXMtXdKTjklQ1IkrOohzE → SFr_DcWGpl2mSNLzCZet_}/_clientMiddlewareManifest.json +0 -0
  394. /package/web/.next/static/{GXMtXdKTjklQ1IkrOohzE → SFr_DcWGpl2mSNLzCZet_}/_ssgManifest.js +0 -0
@@ -1,24 +1,28 @@
1
1
  'use client';
2
2
  import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
3
- import { useState, useCallback } from 'react';
3
+ import { useState, useCallback, useEffect } from 'react';
4
4
  import { useRouter, usePathname } from 'next/navigation';
5
- import { Code2, Terminal, FolderOpen, RefreshCw, Play, Square } from 'lucide-react';
5
+ import { Code2, Terminal, FolderOpen, RefreshCw, Play, Square, Copy, Check, Loader2, LayoutDashboard, MessageSquare, GitBranch, GitCommitHorizontal, AlertTriangle, Globe, Tag, Archive, FileEdit, FilePlus, FileCheck2, } from 'lucide-react';
6
+ import { cn } from '../../../lib/utils.js';
6
7
  import { BaseDrawer } from '../../common/base-drawer/index.js';
7
- import { ActionButton } from '../../common/action-button/index.js';
8
8
  import { DeploymentStatusBadge } from '../../common/deployment-status-badge/index.js';
9
- import { Separator } from '../../ui/separator.js';
10
9
  import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '../../ui/tooltip.js';
11
10
  import { Tabs, TabsList, TabsTrigger, TabsContent } from '../../ui/tabs.js';
12
11
  import { useRepositoryActions } from '../../common/repository-node/use-repository-actions.js';
13
12
  import { useDeployAction } from '../../../hooks/use-deploy-action.js';
14
13
  import { useFeatureFlags } from '../../../hooks/feature-flags-context.js';
15
14
  import { ChatTab } from '../../features/chat/ChatTab.js';
15
+ import { getGitRepoInfo } from '../../../app/actions/get-git-log.js';
16
+ const COPY_FEEDBACK_DELAY = 2000;
17
+ const tbBtn = 'text-muted-foreground hover:bg-foreground/8 hover:text-foreground inline-flex size-8 items-center justify-center rounded-[3px] disabled:opacity-40';
18
+ const tbSep = 'bg-border/60 mx-1.5 h-5 w-px shrink-0';
16
19
  export function RepositoryDrawerClient({ data, initialTab }) {
17
20
  const featureFlags = useFeatureFlags();
18
21
  const router = useRouter();
19
22
  const pathname = usePathname();
20
23
  const isOpen = pathname.startsWith('/repository/');
21
24
  const [activeTab, setActiveTab] = useState(initialTab ?? 'overview');
25
+ const [pathCopied, setPathCopied] = useState(false);
22
26
  const repoActions = useRepositoryActions(data.repositoryPath ? { repositoryId: data.id, repositoryPath: data.repositoryPath } : null);
23
27
  const onClose = useCallback(() => {
24
28
  router.push('/');
@@ -31,7 +35,75 @@ export function RepositoryDrawerClient({ data, initialTab }) {
31
35
  }
32
36
  : null);
33
37
  const isDeployActive = deployAction.status === 'Booting' || deployAction.status === 'Ready';
38
+ const handleCopyPath = useCallback(() => {
39
+ if (!data.repositoryPath)
40
+ return;
41
+ void navigator.clipboard.writeText(data.repositoryPath);
42
+ setPathCopied(true);
43
+ setTimeout(() => setPathCopied(false), COPY_FEEDBACK_DELAY);
44
+ }, [data.repositoryPath]);
34
45
  // Session ID for repo chat — deterministic per repo
35
46
  const repoSessionId = data.id ? `repo-${data.id}` : `repo-${data.name}`;
36
- return (_jsx(BaseDrawer, { open: isOpen, onClose: onClose, size: "lg", modal: false, "data-testid": "repository-drawer", children: _jsxs(Tabs, { value: activeTab, onValueChange: setActiveTab, className: "flex min-h-0 flex-1 flex-col", children: [_jsxs("div", { className: "shrink-0 px-4 pt-4 pb-3", "data-testid": "repository-drawer-header", children: [_jsxs("div", { className: "flex items-baseline gap-4 pe-6", children: [_jsx("h2", { className: "text-foreground min-w-0 shrink truncate text-base font-semibold tracking-tight", children: data.name }), _jsxs(TabsList, { className: "h-auto shrink-0 gap-0.5 rounded-none border-0 bg-transparent p-0", children: [_jsx(TabsTrigger, { value: "overview", className: "text-muted-foreground hover:text-foreground data-[state=active]:text-foreground data-[state=active]:border-primary h-auto rounded-none border-b-2 border-transparent bg-transparent px-2 py-0.5 text-[12px] font-medium shadow-none transition-colors data-[state=active]:bg-transparent data-[state=active]:shadow-none", children: "Overview" }), _jsx(TabsTrigger, { value: "chat", className: "text-muted-foreground hover:text-foreground data-[state=active]:text-foreground data-[state=active]:border-primary h-auto rounded-none border-b-2 border-transparent bg-transparent px-2 py-0.5 text-[12px] font-medium shadow-none transition-colors data-[state=active]:bg-transparent data-[state=active]:shadow-none", children: "Chat" })] })] }), _jsxs("div", { className: "mt-1 flex items-center gap-2", children: [data.repositoryPath ? (_jsx("p", { className: "text-muted-foreground min-w-0 truncate font-mono text-xs", children: data.repositoryPath })) : null, featureFlags.envDeploy && data.repositoryPath ? (_jsxs("div", { className: "ml-auto flex shrink-0 items-center gap-2", children: [_jsx(TooltipProvider, { children: _jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { asChild: true, children: _jsx("span", { children: _jsx(ActionButton, { label: isDeployActive ? 'Stop Dev Server' : 'Start Dev Server', onClick: isDeployActive ? deployAction.stop : deployAction.deploy, loading: deployAction.deployLoading || deployAction.stopLoading, error: !!deployAction.deployError, icon: isDeployActive ? Square : Play, iconOnly: true, variant: "outline", size: "icon-sm" }) }) }), _jsx(TooltipContent, { children: isDeployActive ? 'Stop Dev Server' : 'Start Dev Server' })] }) }), isDeployActive ? (_jsx(DeploymentStatusBadge, { status: deployAction.status, url: deployAction.url, targetId: data.repositoryPath })) : null] })) : null] })] }), _jsx(Separator, {}), _jsx(TabsContent, { value: "overview", className: "mt-0 flex-1 overflow-y-auto", children: data.repositoryPath ? (_jsxs("div", { className: "flex-1 overflow-y-auto", children: [_jsxs("div", { className: "flex flex-col gap-3 p-4", children: [_jsx("div", { className: "text-muted-foreground text-xs font-semibold tracking-wider", children: "OPEN WITH" }), _jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(ActionButton, { label: "Open in IDE", onClick: repoActions.openInIde, loading: repoActions.ideLoading, error: !!repoActions.ideError, icon: Code2, variant: "outline", size: "sm" }), _jsx(ActionButton, { label: "Open in Shell", onClick: repoActions.openInShell, loading: repoActions.shellLoading, error: !!repoActions.shellError, icon: Terminal, variant: "outline", size: "sm" }), _jsx(ActionButton, { label: "Open Folder", onClick: repoActions.openFolder, loading: repoActions.folderLoading, error: !!repoActions.folderError, icon: FolderOpen, variant: "outline", size: "sm" })] })] }), data.id && featureFlags.gitRebaseSync ? (_jsxs(_Fragment, { children: [_jsx(Separator, {}), _jsxs("div", { className: "flex flex-col gap-3 p-4", children: [_jsx("div", { className: "text-muted-foreground text-xs font-semibold tracking-wider", children: "GIT OPERATIONS" }), _jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(ActionButton, { label: "Sync Main", onClick: repoActions.syncMain, loading: repoActions.syncLoading, error: !!repoActions.syncError, icon: RefreshCw, variant: "outline", size: "sm" }), repoActions.syncError ? (_jsx("p", { className: "text-destructive text-xs", children: repoActions.syncError })) : null] })] })] })) : null] })) : null }), _jsx(TabsContent, { value: "chat", className: "mt-0 flex min-h-0 flex-1 flex-col overflow-hidden", children: _jsx(ChatTab, { featureId: repoSessionId, worktreePath: data.repositoryPath }) })] }) }));
47
+ return (_jsx(BaseDrawer, { open: isOpen, onClose: onClose, size: "lg", modal: false, "data-testid": "repository-drawer", children: _jsxs(Tabs, { value: activeTab, onValueChange: setActiveTab, className: "flex min-h-0 flex-1 flex-col", children: [_jsxs(TabsList, { className: "bg-muted/50 h-auto w-full shrink-0 justify-start gap-0 rounded-none border-b p-0", children: [_jsxs(TabsTrigger, { value: "overview", className: "text-muted-foreground hover:bg-muted hover:text-foreground data-[state=active]:bg-background data-[state=active]:text-foreground data-[state=active]:border-t-primary [&:not([data-state=active])]:border-r-border relative h-auto rounded-none border-t-2 border-r border-t-transparent border-r-transparent bg-transparent px-3.5 py-2.5 text-[13px] font-normal shadow-none transition-none last:border-r-transparent data-[state=active]:shadow-none", children: [_jsx(LayoutDashboard, { className: "mr-1.5 size-4" }), "Overview"] }), _jsxs(TabsTrigger, { value: "chat", className: "text-muted-foreground hover:bg-muted hover:text-foreground data-[state=active]:bg-background data-[state=active]:text-foreground data-[state=active]:border-t-primary [&:not([data-state=active])]:border-r-border relative h-auto rounded-none border-t-2 border-r border-t-transparent border-r-transparent bg-transparent px-3.5 py-2.5 text-[13px] font-normal shadow-none transition-none last:border-r-transparent data-[state=active]:shadow-none", children: [_jsx(MessageSquare, { className: "mr-1.5 size-4" }), "Chat"] })] }), _jsxs("div", { className: "bg-muted/40 shrink-0 border-b", children: [_jsxs("div", { className: "flex items-center gap-2 px-4 pe-10 pt-2.5 pb-2", "data-testid": "repository-drawer-header", children: [_jsx("h2", { className: "text-foreground min-w-0 truncate text-base font-semibold tracking-tight", children: data.name }), data.repositoryPath ? (_jsxs(_Fragment, { children: [_jsx("span", { className: "text-muted-foreground/40 text-xs", children: "/" }), _jsx("span", { className: "text-muted-foreground min-w-0 truncate font-mono text-[11px]", children: data.repositoryPath })] })) : null] }), data.repositoryPath ? (_jsx("div", { "data-testid": "repository-drawer-toolbar", children: _jsxs("div", { className: "flex h-9 items-center px-1.5", children: [_jsxs(TooltipProvider, { delayDuration: 300, children: [_jsxs("div", { className: "flex items-center", children: [_jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { asChild: true, children: _jsx("button", { type: "button", className: tbBtn, onClick: repoActions.openInIde, disabled: repoActions.ideLoading, "aria-label": "Open in IDE", children: repoActions.ideLoading ? (_jsx(Loader2, { className: "size-3.5 animate-spin" })) : (_jsx(Code2, { className: "size-4" })) }) }), _jsx(TooltipContent, { side: "bottom", className: "text-xs", children: "Open in IDE" })] }), _jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { asChild: true, children: _jsx("button", { type: "button", className: tbBtn, onClick: repoActions.openInShell, disabled: repoActions.shellLoading, "aria-label": "Open terminal", children: repoActions.shellLoading ? (_jsx(Loader2, { className: "size-3.5 animate-spin" })) : (_jsx(Terminal, { className: "size-4" })) }) }), _jsx(TooltipContent, { side: "bottom", className: "text-xs", children: "Open terminal" })] }), _jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { asChild: true, children: _jsx("button", { type: "button", className: tbBtn, onClick: repoActions.openFolder, disabled: repoActions.folderLoading, "aria-label": "Open folder", children: repoActions.folderLoading ? (_jsx(Loader2, { className: "size-3.5 animate-spin" })) : (_jsx(FolderOpen, { className: "size-4" })) }) }), _jsx(TooltipContent, { side: "bottom", className: "text-xs", children: "Open folder" })] }), _jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { asChild: true, children: _jsx("button", { type: "button", className: tbBtn, onClick: handleCopyPath, "aria-label": "Copy path", children: pathCopied ? (_jsx(Check, { className: "size-3.5 text-green-500" })) : (_jsx(Copy, { className: "size-4" })) }) }), _jsx(TooltipContent, { side: "bottom", className: "text-xs", children: pathCopied ? 'Copied!' : 'Copy path' })] })] }), data.id && featureFlags.gitRebaseSync ? (_jsxs(_Fragment, { children: [_jsx("div", { className: tbSep }), _jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { asChild: true, children: _jsx("button", { type: "button", className: tbBtn, onClick: repoActions.syncMain, disabled: repoActions.syncLoading, "aria-label": "Sync main", children: repoActions.syncLoading ? (_jsx(Loader2, { className: "size-3.5 animate-spin" })) : (_jsx(RefreshCw, { className: "size-4" })) }) }), _jsx(TooltipContent, { side: "bottom", className: "text-xs", children: "Sync main" })] })] })) : null, featureFlags.envDeploy ? (_jsxs(_Fragment, { children: [_jsx("div", { className: tbSep }), _jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { asChild: true, children: _jsx("button", { type: "button", disabled: deployAction.deployLoading || deployAction.stopLoading, onClick: isDeployActive ? deployAction.stop : deployAction.deploy, className: cn('inline-flex size-7 items-center justify-center rounded-[3px] disabled:opacity-40', isDeployActive
48
+ ? 'text-red-500 hover:bg-red-500/10 hover:text-red-400'
49
+ : 'text-green-500 hover:bg-green-500/10 hover:text-green-400'), "aria-label": isDeployActive ? 'Stop dev server' : 'Start dev server', children: deployAction.deployLoading || deployAction.stopLoading ? (_jsx(Loader2, { className: "size-3.5 animate-spin" })) : isDeployActive ? (_jsx(Square, { className: "size-4" })) : (_jsx(Play, { className: "size-4" })) }) }), _jsx(TooltipContent, { side: "bottom", className: "text-xs", children: isDeployActive ? 'Stop dev server' : 'Start dev server' })] }), isDeployActive ? (_jsx(DeploymentStatusBadge, { status: deployAction.status, url: deployAction.url, targetId: data.repositoryPath })) : null] })) : null] }), _jsx("div", { className: "flex-1" })] }) })) : null] }), _jsx(TabsContent, { value: "overview", className: "mt-0 flex-1 overflow-y-auto", children: _jsx(RepoOverview, { data: data, syncError: repoActions.syncError }) }), _jsx(TabsContent, { value: "chat", className: "mt-0 flex min-h-0 flex-1 flex-col overflow-hidden", children: _jsx(ChatTab, { featureId: repoSessionId, worktreePath: data.repositoryPath }) })] }) }));
50
+ }
51
+ // ── Repo Overview ───────────────────────────────────────────────────
52
+ function Section({ icon: Icon, title, children, }) {
53
+ return (_jsxs("div", { className: "px-3 pt-4 pb-1", children: [_jsxs("div", { className: "text-foreground mb-2 flex items-center gap-1.5 text-sm font-semibold tracking-wider uppercase", children: [_jsx(Icon, { className: "size-4 opacity-50" }), title] }), children] }));
54
+ }
55
+ function Card({ children, className }) {
56
+ return (_jsx("div", { className: cn('bg-muted/60 rounded-md border border-transparent p-3', className), children: children }));
57
+ }
58
+ function KV({ label, children }) {
59
+ return (_jsxs("div", { className: "flex flex-col gap-0.5", children: [_jsx("span", { className: "text-foreground/40 text-[11px] font-medium tracking-wider uppercase", children: label }), _jsx("span", { className: "text-sm leading-snug", children: children })] }));
60
+ }
61
+ function RepoOverview({ data, syncError, }) {
62
+ const [repoInfo, setRepoInfo] = useState(null);
63
+ const [loading, setLoading] = useState(false);
64
+ useEffect(() => {
65
+ if (!data.repositoryPath)
66
+ return;
67
+ let cancelled = false;
68
+ setLoading(true);
69
+ getGitRepoInfo(data.repositoryPath, 8)
70
+ .then((result) => {
71
+ if (!cancelled)
72
+ setRepoInfo(result);
73
+ })
74
+ .catch(() => {
75
+ // Silently handle — repo info is non-critical
76
+ })
77
+ .finally(() => {
78
+ if (!cancelled)
79
+ setLoading(false);
80
+ });
81
+ return () => {
82
+ cancelled = true;
83
+ };
84
+ }, [data.repositoryPath]);
85
+ if (!data.repositoryPath)
86
+ return null;
87
+ const wt = repoInfo?.workingTree;
88
+ const isDirty = wt && (wt.staged > 0 || wt.modified > 0 || wt.untracked > 0);
89
+ const ds = repoInfo?.diffStats;
90
+ return (_jsxs("div", { className: "pb-4", children: [loading ? (_jsxs("div", { className: "text-foreground/40 flex items-center gap-2 px-4 py-8 text-sm", children: [_jsx(Loader2, { className: "size-4 animate-spin" }), " Loading repository info..."] })) : null, repoInfo ? (_jsxs("div", { className: "grid grid-cols-2 gap-2 px-3 pt-3", children: [_jsx(Card, { children: _jsxs(KV, { label: "Branch", children: [_jsxs("span", { className: "inline-flex items-center gap-1.5", children: [_jsx(GitBranch, { className: "text-foreground/30 size-3.5 shrink-0" }), _jsx("code", { className: "font-mono text-sm", children: repoInfo.currentBranch ?? data.branch ?? '—' })] }), data.behindCount != null && data.behindCount > 0 ? (_jsxs("span", { className: "mt-0.5 block text-xs text-orange-600 dark:text-orange-400", children: [data.behindCount, " behind default"] })) : null] }) }), _jsx(Card, { children: _jsx(KV, { label: "Working Tree", children: isDirty ? (_jsxs("div", { className: "flex flex-wrap gap-x-3 gap-y-0.5", children: [wt.staged > 0 ? (_jsxs("span", { className: "inline-flex items-center gap-1 text-sm text-green-600 dark:text-green-400", children: [_jsx(FileCheck2, { className: "size-3.5" }), " ", wt.staged, " staged"] })) : null, wt.modified > 0 ? (_jsxs("span", { className: "inline-flex items-center gap-1 text-sm text-amber-600 dark:text-amber-400", children: [_jsx(FileEdit, { className: "size-3.5" }), " ", wt.modified, " modified"] })) : null, wt.untracked > 0 ? (_jsxs("span", { className: "text-foreground/50 inline-flex items-center gap-1 text-sm", children: [_jsx(FilePlus, { className: "size-3.5" }), " ", wt.untracked, " untracked"] })) : null] })) : (_jsxs("span", { className: "inline-flex items-center gap-1.5 text-sm text-green-600 dark:text-green-400", children: [_jsx(Check, { className: "size-3.5" }), " Clean"] })) }) }), ds ? (_jsx(Card, { children: _jsx(KV, { label: "Uncommitted Changes", children: _jsxs("div", { className: "flex items-center gap-3 text-sm", children: [_jsxs("span", { children: [ds.filesChanged, " file", ds.filesChanged !== 1 ? 's' : ''] }), _jsxs("span", { className: "text-green-600 dark:text-green-400", children: ["+", ds.insertions] }), _jsxs("span", { className: "text-red-500 dark:text-red-400", children: ["-", ds.deletions] })] }) }) })) : null, repoInfo.remotes.length > 0 ? (_jsx(Card, { children: _jsx(KV, { label: "Remote", children: repoInfo.remotes.map((r) => (_jsxs("span", { className: "inline-flex items-center gap-1.5 text-sm", children: [_jsx(Globe, { className: "text-foreground/30 size-3.5 shrink-0" }), _jsx("span", { className: "truncate", children: r.url
91
+ .replace(/\.git$/, '')
92
+ .replace(/^https?:\/\/([^@]+@)?/, '')
93
+ .replace(/x-access-token:[^@]+@/, '') })] }, r.name))) }) })) : null, repoInfo.stashCount > 0 ? (_jsx(Card, { children: _jsx(KV, { label: "Stashes", children: _jsxs("span", { className: "inline-flex items-center gap-1.5 text-sm", children: [_jsx(Archive, { className: "text-foreground/30 size-3.5 shrink-0" }), repoInfo.stashCount, " stash", repoInfo.stashCount !== 1 ? 'es' : ''] }) }) })) : null, repoInfo.tags.length > 0 ? (_jsx(Card, { children: _jsx(KV, { label: "Tags", children: _jsx("div", { className: "flex flex-wrap gap-1.5", children: repoInfo.tags.map((t) => (_jsxs("span", { className: "bg-foreground/[0.04] inline-flex items-center gap-0.5 rounded px-1.5 py-0.5 text-xs", children: [_jsx(Tag, { className: "text-foreground/30 size-2.5" }), t] }, t))) }) }) })) : null] })) : null, repoInfo && repoInfo.commits.length > 0 ? (_jsx(Section, { icon: GitCommitHorizontal, title: "Recent Commits", children: _jsxs("div", { className: "relative ml-3", children: [_jsx("div", { className: "bg-border absolute top-2 bottom-2 left-[5px] w-px" }), repoInfo.commits.map((c, i) => (_jsxs("div", { className: "group relative flex gap-3 py-1.5", children: [_jsx("div", { className: cn('relative z-10 mt-1.5 size-[11px] shrink-0 rounded-full border-2', i === 0
94
+ ? 'border-primary bg-primary'
95
+ : 'border-border bg-background group-hover:border-foreground/30') }), _jsxs("div", { className: "min-w-0 flex-1", children: [_jsxs("div", { className: "flex items-baseline gap-2", children: [_jsx("p", { className: "min-w-0 truncate text-sm leading-snug", children: c.subject }), _jsx("code", { className: "text-foreground/30 shrink-0 font-mono text-[11px]", children: c.shortHash })] }), _jsxs("div", { className: "text-foreground/40 mt-0.5 flex items-center gap-2 text-xs", children: [_jsx("span", { children: c.author }), _jsx("span", { children: "\u00B7" }), _jsx("span", { children: c.relativeDate }), c.branch ? (_jsxs(_Fragment, { children: [_jsx("span", { children: "\u00B7" }), _jsxs("span", { className: "inline-flex items-center gap-0.5", children: [_jsx(GitBranch, { className: "size-3" }), c.branch] })] })) : null] })] })] }, c.hash)))] }) })) : null, repoInfo && repoInfo.branches.length > 1 ? (_jsx(Section, { icon: GitBranch, title: "Branches", children: _jsx("div", { className: "flex flex-col", children: [...repoInfo.branches]
96
+ .sort((a, b) => {
97
+ const isDefault = (n) => /^(main|master)$/.test(n);
98
+ if (a.isCurrent && !b.isCurrent)
99
+ return -1;
100
+ if (!a.isCurrent && b.isCurrent)
101
+ return 1;
102
+ if (isDefault(a.name) && !isDefault(b.name))
103
+ return -1;
104
+ if (!isDefault(a.name) && isDefault(b.name))
105
+ return 1;
106
+ return 0;
107
+ })
108
+ .map((b) => (_jsxs("div", { className: cn('flex items-center justify-between rounded px-2 py-1.5', b.isCurrent && 'bg-muted/60'), children: [_jsxs("span", { className: "inline-flex items-center gap-1.5 text-sm", children: [b.isCurrent ? (_jsx("span", { className: "size-2 shrink-0 rounded-full bg-green-500" })) : /^(main|master)$/.test(b.name) ? (_jsx("span", { className: "size-2 shrink-0 rounded-full bg-blue-500" })) : (_jsx("span", { className: "bg-foreground/10 size-2 shrink-0 rounded-full" })), _jsx("code", { className: "font-mono text-[13px]", children: b.name }), b.isCurrent ? (_jsx("span", { className: "text-foreground/40 text-[10px]", children: "current" })) : null] }), _jsx("span", { className: "text-foreground/40 text-xs", children: b.lastCommitDate })] }, b.name))) }) })) : null, syncError ? (_jsx(Section, { icon: AlertTriangle, title: "Issues", children: _jsx(Card, { className: "bg-destructive/5 border-transparent", children: _jsx("p", { className: "text-destructive text-sm", children: syncError }) }) })) : null] }));
37
109
  }
@@ -1 +1 @@
1
- {"version":3,"file":"use-drawer-sync.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/control-center-drawer/use-drawer-sync.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAKhD;;;;;;;;GAQG;AACH,wBAAgB,aAAa,CAC3B,MAAM,EAAE,OAAO,EACf,SAAS,EAAE,MAAM,GAAG,IAAI,EACxB,OAAO,EAAE,KAAK,CAAC,QAAQ,CAAC,KAAK,CAAC,cAAc,CAAC,UAAU,CAAC,CAAC,GACxD,IAAI,CAwCN"}
1
+ {"version":3,"file":"use-drawer-sync.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/control-center-drawer/use-drawer-sync.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAKhD;;;;;;;;GAQG;AACH,wBAAgB,aAAa,CAC3B,MAAM,EAAE,OAAO,EACf,SAAS,EAAE,MAAM,GAAG,IAAI,EACxB,OAAO,EAAE,KAAK,CAAC,QAAQ,CAAC,KAAK,CAAC,cAAc,CAAC,UAAU,CAAC,CAAC,GACxD,IAAI,CAkDN"}
@@ -16,14 +16,21 @@ const BACKGROUND_SYNC_INTERVAL_MS = 15_000;
16
16
  export function useDrawerSync(isOpen, featureId, setView) {
17
17
  const wasOpenRef = useRef(isOpen);
18
18
  const isFetchingRef = useRef(false);
19
+ // Incremented on every open transition — stale fetches from a previous
20
+ // open/close cycle are discarded so they can't overwrite fresh data.
21
+ const generationRef = useRef(0);
19
22
  const syncFromServer = useCallback(async () => {
20
23
  if (!featureId || isFetchingRef.current)
21
24
  return;
22
25
  isFetchingRef.current = true;
26
+ const gen = generationRef.current;
23
27
  try {
24
28
  const data = await getFeatureDrawerData(featureId);
25
29
  if (!data)
26
30
  return;
31
+ // Discard result if the drawer was closed and reopened while fetching
32
+ if (gen !== generationRef.current)
33
+ return;
27
34
  setView((prev) => mergeFeatureData(prev, data));
28
35
  }
29
36
  catch {
@@ -40,6 +47,10 @@ export function useDrawerSync(isOpen, featureId, setView) {
40
47
  useEffect(() => {
41
48
  if (isOpen && (!wasOpenRef.current || !hasFetchedOnMountRef.current)) {
42
49
  hasFetchedOnMountRef.current = true;
50
+ // New open transition — bump generation to invalidate any in-flight fetch
51
+ // from a previous cycle and reset the fetching guard.
52
+ generationRef.current += 1;
53
+ isFetchingRef.current = false;
43
54
  void syncFromServer();
44
55
  }
45
56
  wasOpenRef.current = isOpen;
@@ -1 +1 @@
1
- {"version":3,"file":"deployment-status-badge.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/deployment-status-badge/deployment-status-badge.tsx"],"names":[],"mappings":"AAIA,OAAO,EAAE,eAAe,EAAE,MAAM,sCAAsC,CAAC;AAIvE,MAAM,WAAW,0BAA0B;IACzC,MAAM,EAAE,eAAe,GAAG,IAAI,CAAC;IAC/B,GAAG,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACpB,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB;AAED,wBAAgB,qBAAqB,CAAC,EAAE,MAAM,EAAE,GAAG,EAAE,QAAQ,EAAE,EAAE,0BAA0B,kDAgF1F"}
1
+ {"version":3,"file":"deployment-status-badge.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/deployment-status-badge/deployment-status-badge.tsx"],"names":[],"mappings":"AAIA,OAAO,EAAE,eAAe,EAAE,MAAM,sCAAsC,CAAC;AAIvE,MAAM,WAAW,0BAA0B;IACzC,MAAM,EAAE,eAAe,GAAG,IAAI,CAAC;IAC/B,GAAG,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACpB,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB;AAKD,wBAAgB,qBAAqB,CAAC,EAAE,MAAM,EAAE,GAAG,EAAE,QAAQ,EAAE,EAAE,0BAA0B,kDAkG1F"}
@@ -3,22 +3,23 @@ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-run
3
3
  import { useState } from 'react';
4
4
  import { Loader2, ExternalLink, Terminal } from 'lucide-react';
5
5
  import { DeploymentState } from '../../../../../../packages/core/src/domain/generated/output.js';
6
- import { Badge } from '../../ui/badge.js';
6
+ import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '../../ui/tooltip.js';
7
7
  import { ServerLogViewer } from '../../common/server-log-viewer/index.js';
8
+ const tbBtn = 'text-muted-foreground hover:bg-foreground/8 hover:text-foreground inline-flex size-7 items-center justify-center rounded-[3px]';
8
9
  export function DeploymentStatusBadge({ status, url, targetId }) {
9
10
  const [logViewerOpen, setLogViewerOpen] = useState(false);
10
11
  const showLogButton = targetId && (status === DeploymentState.Booting || status === DeploymentState.Ready);
11
12
  switch (status) {
12
13
  case DeploymentState.Booting:
13
- return (_jsxs(_Fragment, { children: [_jsxs(Badge, { className: "border-transparent bg-blue-50 text-blue-700 hover:bg-blue-50", children: [_jsx(Loader2, { className: "me-1 h-3.5 w-3.5 animate-spin" }), "Starting...", showLogButton ? (_jsx("button", { type: "button", "aria-label": "View server logs", className: "ms-1.5 inline-flex items-center rounded-sm p-0.5 hover:bg-blue-100", onClick: (e) => {
14
- e.stopPropagation();
15
- setLogViewerOpen(true);
16
- }, children: _jsx(Terminal, { className: "h-3 w-3" }) })) : null] }), showLogButton ? (_jsx(ServerLogViewer, { open: logViewerOpen, onOpenChange: setLogViewerOpen, targetId: targetId })) : null] }));
14
+ return (_jsxs(_Fragment, { children: [_jsxs("div", { className: "flex items-center gap-1 pl-1", children: [_jsx(Loader2, { className: "size-3 animate-spin text-blue-500" }), _jsx("span", { className: "text-muted-foreground text-[11px]", children: "Starting..." }), showLogButton ? (_jsx(TooltipProvider, { delayDuration: 300, children: _jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { asChild: true, children: _jsx("button", { type: "button", "aria-label": "View server logs", className: tbBtn, onClick: (e) => {
15
+ e.stopPropagation();
16
+ setLogViewerOpen(true);
17
+ }, children: _jsx(Terminal, { className: "size-3.5" }) }) }), _jsx(TooltipContent, { side: "bottom", className: "text-xs", children: "Server logs" })] }) })) : null] }), showLogButton ? (_jsx(ServerLogViewer, { open: logViewerOpen, onOpenChange: setLogViewerOpen, targetId: targetId })) : null] }));
17
18
  case DeploymentState.Ready:
18
- return (_jsxs(_Fragment, { children: [_jsxs(Badge, { className: "border-transparent bg-green-50 text-green-700 hover:bg-green-50", children: [_jsx("span", { className: "me-1 inline-block h-2 w-2 rounded-full bg-green-500" }), url ? (_jsxs("a", { href: url, target: "_blank", rel: "noopener noreferrer", className: "inline-flex items-center gap-1 hover:underline", onClick: (e) => e.stopPropagation(), children: [url, _jsx(ExternalLink, { className: "h-3 w-3" })] })) : ('Ready'), showLogButton ? (_jsx("button", { type: "button", "aria-label": "View server logs", className: "ms-1.5 inline-flex items-center rounded-sm p-0.5 hover:bg-green-100", onClick: (e) => {
19
- e.stopPropagation();
20
- setLogViewerOpen(true);
21
- }, children: _jsx(Terminal, { className: "h-3 w-3" }) })) : null] }), showLogButton ? (_jsx(ServerLogViewer, { open: logViewerOpen, onOpenChange: setLogViewerOpen, targetId: targetId })) : null] }));
19
+ return (_jsxs(_Fragment, { children: [_jsxs("div", { className: "flex translate-y-px items-center gap-1 pl-1", children: [_jsx("span", { className: "inline-block size-1.5 rounded-full bg-green-500" }), url ? (_jsxs("a", { href: url, target: "_blank", rel: "noopener noreferrer", className: "inline-flex items-center gap-0.5 text-[11px] text-green-600 hover:text-green-500 dark:text-green-400 dark:hover:text-green-300", onClick: (e) => e.stopPropagation(), children: [url, _jsx(ExternalLink, { className: "size-2.5 shrink-0" })] })) : (_jsx("span", { className: "text-[11px] text-green-600 dark:text-green-400", children: "Ready" })), showLogButton ? (_jsx(TooltipProvider, { delayDuration: 300, children: _jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { asChild: true, children: _jsx("button", { type: "button", "aria-label": "View server logs", className: tbBtn, onClick: (e) => {
20
+ e.stopPropagation();
21
+ setLogViewerOpen(true);
22
+ }, children: _jsx(Terminal, { className: "size-3.5" }) }) }), _jsx(TooltipContent, { side: "bottom", className: "text-xs", children: "Server logs" })] }) })) : null] }), showLogButton ? (_jsx(ServerLogViewer, { open: logViewerOpen, onOpenChange: setLogViewerOpen, targetId: targetId })) : null] }));
22
23
  default:
23
24
  return null;
24
25
  }
@@ -14,6 +14,10 @@ export interface FeatureDrawerTabsProps {
14
14
  headerContent?: React.ReactNode;
15
15
  featureNode: FeatureNodeData;
16
16
  featureId: string;
17
+ /** Action handlers for the status chip in the title row. */
18
+ onRetry?: (featureId: string) => void;
19
+ onStop?: (featureId: string) => void;
20
+ onStart?: (featureId: string) => void;
17
21
  initialTab?: FeatureTabKey;
18
22
  /** Tab key from URL path segment (e.g. /feature/[id]/activity → 'activity'). */
19
23
  urlTab?: FeatureTabKey;
@@ -47,5 +51,5 @@ export interface FeatureDrawerTabsProps {
47
51
  /** When false, the Chat tab is hidden from the tab bar (FR-17). Defaults to true. */
48
52
  interactiveAgentEnabled?: boolean;
49
53
  }
50
- export declare function FeatureDrawerTabs({ featureName, headerContent, featureNode, featureId, initialTab, urlTab, prdData, prdSelections, onPrdSelect, onPrdApprove, onPrdReject, isPrdLoading, techData, onTechApprove, onTechReject, isTechLoading, productData, mergeData, onMergeApprove, onMergeReject, isMergeLoading, syncStatus, syncLoading, syncError, onRefreshSync, onRebaseOnMain, rebaseLoading, rebaseError, isRejecting, chatInput, onChatInputChange, sseEvents, interactiveAgentEnabled, }: FeatureDrawerTabsProps): import("react/jsx-runtime").JSX.Element;
54
+ export declare function FeatureDrawerTabs({ featureName, headerContent, featureNode, featureId, initialTab, urlTab, prdData, prdSelections, onPrdSelect, onPrdApprove, onPrdReject, isPrdLoading, techData, onTechApprove, onTechReject, isTechLoading, productData, mergeData, onMergeApprove, onMergeReject, isMergeLoading, syncStatus, syncLoading, syncError, onRefreshSync, onRebaseOnMain, rebaseLoading, rebaseError, isRejecting, chatInput, onChatInputChange, sseEvents, interactiveAgentEnabled, onRetry, onStop, onStart, }: FeatureDrawerTabsProps): import("react/jsx-runtime").JSX.Element;
51
55
  //# sourceMappingURL=feature-drawer-tabs.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"feature-drawer-tabs.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/feature-drawer-tabs/feature-drawer-tabs.tsx"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"feature-drawer-tabs.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/feature-drawer-tabs/feature-drawer-tabs.tsx"],"names":[],"mappings":"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"}
@@ -3,11 +3,14 @@ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
3
3
  import { useState, useEffect, useCallback, useRef, useMemo } from 'react';
4
4
  import { useTranslation } from 'react-i18next';
5
5
  import { usePathname } from 'next/navigation';
6
- import { Loader2, AlertCircle } from 'lucide-react';
6
+ import { Loader2, AlertCircle, LayoutDashboard, Activity, ScrollText, Map, FileCheck, Cpu, Package, GitMerge, MessageSquare, Play, Square, RotateCcw, Zap, Layers, } from 'lucide-react';
7
7
  import { Tabs, TabsList, TabsTrigger, TabsContent } from '../../ui/tabs.js';
8
- import { Separator } from '../../ui/separator.js';
8
+ import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '../../ui/tooltip.js';
9
9
  import { getFeaturePhaseTimings } from '../../../app/actions/get-feature-phase-timings.js';
10
10
  import { getFeaturePlan } from '../../../app/actions/get-feature-plan.js';
11
+ import { cn } from '../../../lib/utils.js';
12
+ import { featureNodeStateConfig } from '../../common/feature-node/index.js';
13
+ import { CometSpinner } from '../../ui/comet-spinner.js';
11
14
  import { PrdQuestionnaire } from '../../common/prd-questionnaire/index.js';
12
15
  import { TechDecisionsContent } from '../../common/tech-decisions-review/index.js';
13
16
  import { ProductDecisionsSummary } from '../../common/product-decisions-summary/index.js';
@@ -22,15 +25,15 @@ import { useFeatureLogs } from '../../../hooks/use-feature-logs.js';
22
25
  import { useTabDataFetch } from './use-tab-data-fetch.js';
23
26
  /** All possible tabs in display order. */
24
27
  const ALL_TABS = [
25
- { key: 'overview', label: 'Overview' },
26
- { key: 'activity', label: 'Activity' },
27
- { key: 'log', label: 'Log' },
28
- { key: 'plan', label: 'Plan' },
29
- { key: 'prd-review', label: 'PRD Review' },
30
- { key: 'tech-decisions', label: 'Tech Decisions' },
31
- { key: 'product-decisions', label: 'Product' },
32
- { key: 'merge-review', label: 'Merge Review' },
33
- { key: 'chat', label: 'Chat' },
28
+ { key: 'overview', label: 'Overview', icon: LayoutDashboard },
29
+ { key: 'activity', label: 'Activity', icon: Activity },
30
+ { key: 'log', label: 'Log', icon: ScrollText },
31
+ { key: 'plan', label: 'Plan', icon: Map },
32
+ { key: 'prd-review', label: 'PRD Review', icon: FileCheck },
33
+ { key: 'tech-decisions', label: 'Tech Decisions', icon: Cpu },
34
+ { key: 'product-decisions', label: 'Product', icon: Package },
35
+ { key: 'merge-review', label: 'Merge Review', icon: GitMerge },
36
+ { key: 'chat', label: 'Chat', icon: MessageSquare },
34
37
  ];
35
38
  /** Compute which tabs are visible based on feature lifecycle + state. */
36
39
  function computeVisibleTabs(node, interactiveAgentEnabled = true) {
@@ -75,7 +78,7 @@ const TAB_FETCHERS = {
75
78
  activity: fetchActivity,
76
79
  plan: fetchPlan,
77
80
  };
78
- export function FeatureDrawerTabs({ featureName, headerContent, featureNode, featureId, initialTab, urlTab, prdData, prdSelections, onPrdSelect, onPrdApprove, onPrdReject, isPrdLoading, techData, onTechApprove, onTechReject, isTechLoading, productData, mergeData, onMergeApprove, onMergeReject, isMergeLoading, syncStatus, syncLoading, syncError, onRefreshSync, onRebaseOnMain, rebaseLoading, rebaseError, isRejecting, chatInput, onChatInputChange, sseEvents, interactiveAgentEnabled = true, }) {
81
+ export function FeatureDrawerTabs({ featureName, headerContent, featureNode, featureId, initialTab, urlTab, prdData, prdSelections, onPrdSelect, onPrdApprove, onPrdReject, isPrdLoading, techData, onTechApprove, onTechReject, isTechLoading, productData, mergeData, onMergeApprove, onMergeReject, isMergeLoading, syncStatus, syncLoading, syncError, onRefreshSync, onRebaseOnMain, rebaseLoading, rebaseError, isRejecting, chatInput, onChatInputChange, sseEvents, interactiveAgentEnabled = true, onRetry, onStop, onStart, }) {
79
82
  const pathname = usePathname();
80
83
  const visibleTabs = useMemo(() => computeVisibleTabs(featureNode, interactiveAgentEnabled), [featureNode, interactiveAgentEnabled]);
81
84
  const visibleTabDefs = useMemo(() => ALL_TABS.filter((t) => visibleTabs.includes(t.key)).map((t) => t.key === 'merge-review' && featureNode.lifecycle === 'maintain'
@@ -252,7 +255,15 @@ export function FeatureDrawerTabs({ featureName, headerContent, featureNode, fea
252
255
  fetchTab(tab);
253
256
  }
254
257
  }, [fetchTab]);
255
- return (_jsx("div", { className: "flex min-h-0 flex-1 flex-col", children: _jsxs(Tabs, { value: activeTab, onValueChange: handleTabChange, className: "flex min-h-0 flex-1 flex-col", children: [_jsxs("div", { className: "shrink-0 px-4 pt-4 pb-3", "data-testid": "feature-drawer-header", children: [_jsxs("div", { className: "flex items-baseline gap-4 pe-6", children: [featureName ? (_jsx("h2", { className: "text-foreground min-w-0 shrink truncate text-base font-semibold tracking-tight", children: featureName })) : null, _jsx(TabsList, { className: "h-auto shrink-0 gap-0.5 rounded-none border-0 bg-transparent p-0", children: visibleTabDefs.map((tab) => (_jsx(TabsTrigger, { value: tab.key, className: "text-muted-foreground hover:text-foreground data-[state=active]:text-foreground data-[state=active]:border-primary h-auto rounded-none border-b-2 border-transparent bg-transparent px-2 py-0.5 text-[12px] font-medium shadow-none transition-colors data-[state=active]:bg-transparent data-[state=active]:shadow-none", children: tab.label }, tab.key))) })] }), headerContent] }), _jsx(Separator, {}), _jsx(TabsContent, { value: "overview", className: "mt-0 flex-1 overflow-y-auto", children: _jsx(OverviewTab, { data: featureNode, syncStatus: syncStatus, syncLoading: syncLoading, syncError: syncError, onRefreshSync: onRefreshSync, onRebaseOnMain: onRebaseOnMain, rebaseLoading: rebaseLoading, rebaseError: rebaseError }) }), _jsx(TabsContent, { value: "activity", className: "mt-0 flex-1 overflow-y-auto", children: _jsx(ActivityTab, { timings: tabs.activity.data?.timings ?? null, loading: tabs.activity.loading, error: tabs.activity.error, rejectionFeedback: tabs.activity.data?.rejectionFeedback }) }), _jsx(TabsContent, { value: "log", className: "mt-0 flex-1 overflow-hidden", children: _jsx(LogTab, { content: featureLogs.content, isConnected: featureLogs.isConnected, error: featureLogs.error }) }), _jsx(TabsContent, { value: "plan", className: "mt-0 flex-1 overflow-y-auto", children: _jsx(PlanTab, { plan: tabs.plan.data, loading: tabs.plan.loading, error: tabs.plan.error }) }), visibleTabs.includes('prd-review') ? (_jsx(TabsContent, { value: "prd-review", className: "mt-0 flex min-h-0 flex-1 flex-col", children: prdData ? (_jsx(PrdQuestionnaire, { data: prdData, selections: prdSelections ?? {}, onSelect: onPrdSelect ?? (() => undefined), onApprove: onPrdApprove ?? (() => undefined), onReject: onPrdReject, isProcessing: isPrdLoading, isRejecting: isRejecting, chatInput: chatInput, onChatInputChange: onChatInputChange })) : (_jsx("div", { className: "flex items-center justify-center p-8", children: _jsx(Loader2, { className: "text-muted-foreground h-6 w-6 animate-spin" }) })) })) : null, visibleTabs.includes('tech-decisions') ? (_jsx(TabsContent, { value: "tech-decisions", className: "mt-0 flex min-h-0 flex-1 flex-col", children: techData ? (_jsxs("div", { className: "flex min-h-0 flex-1 flex-col", children: [_jsx("div", { className: "flex-1 overflow-y-auto", children: _jsx(TechDecisionsContent, { data: techData }) }), _jsx(DrawerActionBarForTech, { onApprove: onTechApprove ?? (() => undefined), onReject: onTechReject, isProcessing: isTechLoading, isRejecting: isRejecting, chatInput: chatInput, onChatInputChange: onChatInputChange })] })) : (_jsx("div", { className: "flex items-center justify-center p-8", children: _jsx(Loader2, { className: "text-muted-foreground h-6 w-6 animate-spin" }) })) })) : null, visibleTabs.includes('product-decisions') ? (_jsx(TabsContent, { value: "product-decisions", className: "mt-0 flex-1 overflow-y-auto", children: productData === null ? (_jsx("div", { className: "flex items-center justify-center p-8", children: _jsx(Loader2, { className: "text-muted-foreground h-6 w-6 animate-spin" }) })) : productData ? (_jsx(ProductDecisionsSummary, { data: productData })) : (_jsx("p", { className: "text-muted-foreground p-4 text-center text-sm", children: "No product decisions available." })) })) : null, visibleTabs.includes('merge-review') ? (_jsx(TabsContent, { value: "merge-review", className: "mt-0 flex min-h-0 flex-1 flex-col", children: mergeData ? (_jsx(MergeReview, { data: mergeData, readOnly: featureNode.lifecycle === 'maintain', onApprove: onMergeApprove ?? (() => undefined), onReject: onMergeReject, isProcessing: isMergeLoading, isRejecting: isRejecting, chatInput: chatInput, onChatInputChange: onChatInputChange })) : (_jsx("div", { className: "flex items-center justify-center p-8", children: isMergeLoading ? (_jsx(Loader2, { className: "text-muted-foreground h-6 w-6 animate-spin" })) : (_jsxs("div", { className: "text-muted-foreground flex flex-col items-center gap-2 text-sm", children: [_jsx(AlertCircle, { className: "h-6 w-6" }), _jsx("span", { children: "Merge review data unavailable" })] })) })) })) : null, visibleTabs.includes('chat') ? (_jsx(TabsContent, { value: "chat", className: "mt-0 flex min-h-0 flex-1 flex-col overflow-hidden", children: _jsx(ChatTab, { featureId: featureId, worktreePath: featureNode.worktreePath }) })) : null] }) }));
258
+ return (_jsx("div", { className: "flex min-h-0 flex-1 flex-col", children: _jsxs(Tabs, { value: activeTab, onValueChange: handleTabChange, className: "flex min-h-0 flex-1 flex-col", children: [_jsx(TabsList, { className: "bg-muted/50 h-auto w-full shrink-0 justify-start gap-0 rounded-none border-b p-0", children: visibleTabDefs.map((tab) => {
259
+ const Icon = tab.icon;
260
+ return (_jsxs(TabsTrigger, { value: tab.key, className: "text-muted-foreground hover:bg-muted hover:text-foreground data-[state=active]:bg-background data-[state=active]:text-foreground data-[state=active]:border-t-primary [&:not([data-state=active])]:border-r-border relative h-auto rounded-none border-t-2 border-r border-t-transparent border-r-transparent bg-transparent px-3.5 py-2.5 text-[13px] font-normal shadow-none transition-none last:border-r-transparent data-[state=active]:shadow-none", children: [_jsx(Icon, { className: "mr-1.5 size-4" }), tab.label] }, tab.key));
261
+ }) }), _jsxs("div", { className: "bg-muted/40 shrink-0 border-b", children: [featureName ? (_jsxs("div", { className: "flex h-12 items-stretch gap-2 pr-0 pl-4", "data-testid": "feature-drawer-header", children: [_jsx("div", { className: "flex items-center", children: featureNode.fastMode ? (_jsx(Zap, { className: "size-4 shrink-0 text-amber-500" })) : (_jsx(Layers, { className: "text-muted-foreground/50 size-4 shrink-0" })) }), _jsx("h2", { className: "text-foreground flex min-w-0 items-center truncate text-base font-semibold tracking-tight", children: featureName }), featureNode.repositoryName ? (_jsxs("span", { className: "animate-in fade-in flex shrink-0 items-center gap-1.5 self-center duration-200", children: [_jsx("span", { className: "text-muted-foreground/30 text-sm", children: "/" }), featureNode.remoteUrl ? (_jsx("a", { href: featureNode.remoteUrl, target: "_blank", rel: "noopener noreferrer", className: "text-muted-foreground hover:text-foreground text-sm", children: featureNode.repositoryName })) : (_jsx("span", { className: "text-muted-foreground text-sm", children: featureNode.repositoryName }))] })) : (_jsxs("span", { className: "flex items-center gap-1.5 self-center", children: [_jsx("span", { className: "text-muted-foreground/30 text-sm", children: "/" }), _jsx("span", { className: "bg-muted h-4 w-16 animate-pulse rounded" })] })), _jsx(TooltipProvider, { delayDuration: 300, children: _jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { asChild: true, children: _jsxs("div", { className: "ml-auto flex shrink-0 cursor-default items-center self-stretch text-xs font-medium", children: [_jsxs("div", { className: cn('flex items-center gap-1.5 self-stretch px-3', featureNodeStateConfig[featureNode.state].labelClass), children: [featureNode.state === 'running' ? (_jsx(CometSpinner, { size: "sm", className: "shrink-0" })) : ((() => {
262
+ const I = featureNodeStateConfig[featureNode.state].icon;
263
+ return _jsx(I, { className: "size-3.5 shrink-0" });
264
+ })()), featureNodeStateConfig[featureNode.state].label] }), featureNode.state === 'pending' && onStart ? (_jsxs("button", { type: "button", onClick: () => onStart(featureNode.featureId), className: "text-muted-foreground flex items-center gap-1 self-stretch px-3 hover:bg-green-500/10 hover:text-green-600 dark:hover:text-green-400", "data-testid": "feature-drawer-start-button", children: [_jsx(Play, { className: "size-3.5" }), " Start"] })) : featureNode.state === 'error' && onRetry ? (_jsxs("button", { type: "button", onClick: () => onRetry(featureNode.featureId), className: "text-muted-foreground flex items-center gap-1 self-stretch px-3 hover:bg-red-500/10 hover:text-red-500 dark:hover:text-red-400", "data-testid": "feature-drawer-retry-button", children: [_jsx(RotateCcw, { className: "size-3.5" }), " Retry"] })) : featureNode.state === 'running' && onStop ? (_jsxs("button", { type: "button", onClick: () => onStop(featureNode.featureId), className: "text-muted-foreground flex items-center gap-1 self-stretch px-3 hover:bg-red-500/10 hover:text-red-500 dark:hover:text-red-400", "data-testid": "feature-drawer-stop-button", children: [_jsx(Square, { className: "size-3.5" }), " Stop"] })) : null] }) }), featureNode.errorMessage ? (_jsxs(TooltipContent, { side: "bottom", align: "end", sideOffset: 4, className: "z-[100] max-w-xs cursor-pointer text-xs leading-relaxed select-text", onClick: () => {
265
+ void navigator.clipboard.writeText(featureNode.errorMessage);
266
+ }, children: [featureNode.errorMessage, _jsx("span", { className: "text-muted-foreground ml-1 text-[10px] italic", children: "(click to copy)" })] })) : null] }) })] })) : null, headerContent] }), _jsx(TabsContent, { value: "overview", className: "mt-0 flex-1 overflow-y-auto", children: _jsx(OverviewTab, { data: featureNode, syncStatus: syncStatus, syncLoading: syncLoading, syncError: syncError, onRefreshSync: onRefreshSync, onRebaseOnMain: onRebaseOnMain, rebaseLoading: rebaseLoading, rebaseError: rebaseError }) }), _jsx(TabsContent, { value: "activity", className: "mt-0 flex-1 overflow-y-auto", children: _jsx(ActivityTab, { timings: tabs.activity.data?.timings ?? null, loading: tabs.activity.loading, error: tabs.activity.error, rejectionFeedback: tabs.activity.data?.rejectionFeedback }) }), _jsx(TabsContent, { value: "log", className: "mt-0 flex-1 overflow-hidden", children: _jsx(LogTab, { content: featureLogs.content, isConnected: featureLogs.isConnected, error: featureLogs.error }) }), _jsx(TabsContent, { value: "plan", className: "mt-0 flex-1 overflow-y-auto", children: _jsx(PlanTab, { plan: tabs.plan.data, loading: tabs.plan.loading, error: tabs.plan.error }) }), visibleTabs.includes('prd-review') ? (_jsx(TabsContent, { value: "prd-review", className: "mt-0 flex min-h-0 flex-1 flex-col", children: prdData ? (_jsx(PrdQuestionnaire, { data: prdData, selections: prdSelections ?? {}, onSelect: onPrdSelect ?? (() => undefined), onApprove: onPrdApprove ?? (() => undefined), onReject: onPrdReject, isProcessing: isPrdLoading, isRejecting: isRejecting, chatInput: chatInput, onChatInputChange: onChatInputChange })) : (_jsx("div", { className: "flex items-center justify-center p-8", children: _jsx(Loader2, { className: "text-muted-foreground h-6 w-6 animate-spin" }) })) })) : null, visibleTabs.includes('tech-decisions') ? (_jsx(TabsContent, { value: "tech-decisions", className: "mt-0 flex min-h-0 flex-1 flex-col", children: techData ? (_jsxs("div", { className: "flex min-h-0 flex-1 flex-col", children: [_jsx("div", { className: "flex-1 overflow-y-auto", children: _jsx(TechDecisionsContent, { data: techData }) }), _jsx(DrawerActionBarForTech, { onApprove: onTechApprove ?? (() => undefined), onReject: onTechReject, isProcessing: isTechLoading, isRejecting: isRejecting, chatInput: chatInput, onChatInputChange: onChatInputChange })] })) : (_jsx("div", { className: "flex items-center justify-center p-8", children: _jsx(Loader2, { className: "text-muted-foreground h-6 w-6 animate-spin" }) })) })) : null, visibleTabs.includes('product-decisions') ? (_jsx(TabsContent, { value: "product-decisions", className: "mt-0 flex-1 overflow-y-auto", children: productData === null ? (_jsx("div", { className: "flex items-center justify-center p-8", children: _jsx(Loader2, { className: "text-muted-foreground h-6 w-6 animate-spin" }) })) : productData ? (_jsx(ProductDecisionsSummary, { data: productData })) : (_jsx("p", { className: "text-muted-foreground p-4 text-center text-sm", children: "No product decisions available." })) })) : null, visibleTabs.includes('merge-review') ? (_jsx(TabsContent, { value: "merge-review", className: "mt-0 flex min-h-0 flex-1 flex-col", children: mergeData ? (_jsx(MergeReview, { data: mergeData, readOnly: featureNode.lifecycle === 'maintain', onApprove: onMergeApprove ?? (() => undefined), onReject: onMergeReject, isProcessing: isMergeLoading, isRejecting: isRejecting, chatInput: chatInput, onChatInputChange: onChatInputChange })) : (_jsx("div", { className: "flex items-center justify-center p-8", children: isMergeLoading ? (_jsx(Loader2, { className: "text-muted-foreground h-6 w-6 animate-spin" })) : (_jsxs("div", { className: "text-muted-foreground flex flex-col items-center gap-2 text-sm", children: [_jsx(AlertCircle, { className: "h-6 w-6" }), _jsx("span", { children: "Merge review data unavailable" })] })) })) })) : null, visibleTabs.includes('chat') ? (_jsx(TabsContent, { value: "chat", className: "mt-0 flex min-h-0 flex-1 flex-col overflow-hidden", children: _jsx(ChatTab, { featureId: featureId, worktreePath: featureNode.worktreePath }) })) : null] }) }));
256
267
  }
257
268
  // ── Private helper ──────────────────────────────────────────────────────
258
269
  function DrawerActionBarForTech({ onApprove, onReject, isProcessing, isRejecting, chatInput, onChatInputChange, }) {
@@ -1 +1 @@
1
- {"version":3,"file":"overview-tab.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/feature-drawer-tabs/overview-tab.tsx"],"names":[],"mappings":"AAyBA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,kCAAkC,CAAC;AAQxE,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,gCAAgC,CAAC;AAErE,MAAM,WAAW,gBAAgB;IAC/B,IAAI,EAAE,eAAe,CAAC;IACtB,UAAU,CAAC,EAAE,cAAc,GAAG,IAAI,CAAC;IACnC,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,aAAa,CAAC,EAAE,MAAM,IAAI,CAAC;IAC3B,cAAc,CAAC,EAAE,MAAM,IAAI,CAAC;IAC5B,aAAa,CAAC,EAAE,OAAO,CAAC;IACxB,WAAW,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;CAC7B;AAED,wBAAgB,WAAW,CAAC,EAC1B,IAAI,EACJ,UAAU,EACV,WAAW,EACX,SAAS,EACT,aAAa,EACb,cAAc,EACd,aAAa,EACb,WAAW,GACZ,EAAE,gBAAgB,2CA4ElB"}
1
+ {"version":3,"file":"overview-tab.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/common/feature-drawer-tabs/overview-tab.tsx"],"names":[],"mappings":"AA4BA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,kCAAkC,CAAC;AAOxE,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,gCAAgC,CAAC;AA2GrE,MAAM,WAAW,gBAAgB;IAC/B,IAAI,EAAE,eAAe,CAAC;IACtB,UAAU,CAAC,EAAE,cAAc,GAAG,IAAI,CAAC;IACnC,WAAW,CAAC,EAAE,OAAO,CAAC;IACtB,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAC1B,aAAa,CAAC,EAAE,MAAM,IAAI,CAAC;IAC3B,cAAc,CAAC,EAAE,MAAM,IAAI,CAAC;IAC5B,aAAa,CAAC,EAAE,OAAO,CAAC;IACxB,WAAW,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;CAC7B;AAED,wBAAgB,WAAW,CAAC,EAC1B,IAAI,EACJ,UAAU,EACV,WAAW,EACX,SAAS,EACT,aAAa,EACb,cAAc,EACd,aAAa,EACb,WAAW,GACZ,EAAE,gBAAgB,2CAuLlB"}