@shepai/cli 1.152.0-pr476.5b101f5 → 1.152.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/packages/core/src/application/ports/output/repositories/interactive-session-repository.interface.d.ts +0 -5
- package/dist/packages/core/src/application/ports/output/repositories/interactive-session-repository.interface.d.ts.map +1 -1
- package/dist/packages/core/src/application/ports/output/services/interactive-session-service.interface.d.ts +0 -5
- package/dist/packages/core/src/application/ports/output/services/interactive-session-service.interface.d.ts.map +1 -1
- package/dist/packages/core/src/infrastructure/repositories/sqlite-interactive-session.repository.d.ts +0 -1
- package/dist/packages/core/src/infrastructure/repositories/sqlite-interactive-session.repository.d.ts.map +1 -1
- package/dist/packages/core/src/infrastructure/repositories/sqlite-interactive-session.repository.js +0 -15
- package/dist/packages/core/src/infrastructure/services/agents/common/executors/claude-code-interactive-executor.service.d.ts.map +1 -1
- package/dist/packages/core/src/infrastructure/services/agents/common/executors/claude-code-interactive-executor.service.js +0 -8
- package/dist/packages/core/src/infrastructure/services/interactive/interactive-session.service.d.ts +0 -1
- package/dist/packages/core/src/infrastructure/services/interactive/interactive-session.service.d.ts.map +1 -1
- package/dist/packages/core/src/infrastructure/services/interactive/interactive-session.service.js +5 -51
- package/dist/src/presentation/web/app/api/interactive/chat/turn-statuses/route.d.ts +5 -4
- package/dist/src/presentation/web/app/api/interactive/chat/turn-statuses/route.d.ts.map +1 -1
- package/dist/src/presentation/web/app/api/interactive/chat/turn-statuses/route.js +13 -5
- package/dist/src/presentation/web/components/common/floating-action-button/floating-action-button.d.ts +2 -8
- package/dist/src/presentation/web/components/common/floating-action-button/floating-action-button.d.ts.map +1 -1
- package/dist/src/presentation/web/components/common/floating-action-button/floating-action-button.js +7 -10
- package/dist/src/presentation/web/components/common/repository-node/repository-node.d.ts.map +1 -1
- package/dist/src/presentation/web/components/common/repository-node/repository-node.js +3 -2
- package/dist/src/presentation/web/components/features/chat/ChatSheet.d.ts.map +1 -1
- package/dist/src/presentation/web/components/features/chat/ChatSheet.js +30 -43
- package/dist/src/presentation/web/components/features/control-center/control-center-inner.d.ts.map +1 -1
- package/dist/src/presentation/web/components/features/control-center/control-center-inner.js +3 -66
- package/dist/src/presentation/web/components/features/features-canvas/canvas-toolbar.d.ts +6 -1
- package/dist/src/presentation/web/components/features/features-canvas/canvas-toolbar.d.ts.map +1 -1
- package/dist/src/presentation/web/components/features/features-canvas/canvas-toolbar.js +6 -5
- package/dist/src/presentation/web/components/features/features-canvas/features-canvas.js +1 -1
- package/dist/src/presentation/web/components/layouts/app-shell/app-shell.d.ts.map +1 -1
- package/dist/src/presentation/web/components/layouts/app-shell/app-shell.js +5 -9
- package/dist/src/presentation/web/hooks/turn-statuses-provider.d.ts +8 -4
- package/dist/src/presentation/web/hooks/turn-statuses-provider.d.ts.map +1 -1
- package/dist/src/presentation/web/hooks/turn-statuses-provider.js +4 -4
- package/dist/src/presentation/web/hooks/use-turn-statuses.d.ts +5 -4
- package/dist/src/presentation/web/hooks/use-turn-statuses.d.ts.map +1 -1
- package/dist/src/presentation/web/hooks/use-turn-statuses.js +12 -7
- package/dist/tsconfig.build.tsbuildinfo +1 -1
- package/package.json +1 -1
- package/web/.next/BUILD_ID +1 -1
- package/web/.next/build-manifest.json +2 -2
- package/web/.next/fallback-build-manifest.json +2 -2
- package/web/.next/prerender-manifest.json +3 -3
- package/web/.next/required-server-files.js +3 -3
- package/web/.next/required-server-files.json +3 -3
- package/web/.next/server/app/(dashboard)/@drawer/adopt/page/server-reference-manifest.json +28 -28
- package/web/.next/server/app/(dashboard)/@drawer/adopt/page.js.nft.json +1 -1
- package/web/.next/server/app/(dashboard)/@drawer/adopt/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/(dashboard)/@drawer/chat/page/server-reference-manifest.json +26 -26
- package/web/.next/server/app/(dashboard)/@drawer/chat/page.js.nft.json +1 -1
- package/web/.next/server/app/(dashboard)/@drawer/chat/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/(dashboard)/@drawer/create/page/server-reference-manifest.json +29 -29
- package/web/.next/server/app/(dashboard)/@drawer/create/page.js.nft.json +1 -1
- package/web/.next/server/app/(dashboard)/@drawer/create/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page/server-reference-manifest.json +36 -36
- package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page.js.nft.json +1 -1
- package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/[tab]/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page/server-reference-manifest.json +36 -36
- package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page.js.nft.json +1 -1
- package/web/.next/server/app/(dashboard)/@drawer/feature/[featureId]/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/[tab]/page/server-reference-manifest.json +26 -26
- package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/[tab]/page.js.nft.json +1 -1
- package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/[tab]/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page/server-reference-manifest.json +26 -26
- package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page.js.nft.json +1 -1
- package/web/.next/server/app/(dashboard)/@drawer/repository/[repositoryId]/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/(dashboard)/chat/page/server-reference-manifest.json +26 -26
- package/web/.next/server/app/(dashboard)/chat/page.js.nft.json +1 -1
- package/web/.next/server/app/(dashboard)/chat/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/(dashboard)/create/page/server-reference-manifest.json +29 -29
- package/web/.next/server/app/(dashboard)/create/page.js.nft.json +1 -1
- package/web/.next/server/app/(dashboard)/create/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page/server-reference-manifest.json +36 -36
- package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page.js.nft.json +1 -1
- package/web/.next/server/app/(dashboard)/feature/[featureId]/[tab]/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/(dashboard)/feature/[featureId]/page/server-reference-manifest.json +36 -36
- package/web/.next/server/app/(dashboard)/feature/[featureId]/page.js.nft.json +1 -1
- package/web/.next/server/app/(dashboard)/feature/[featureId]/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/(dashboard)/page/server-reference-manifest.json +26 -26
- package/web/.next/server/app/(dashboard)/page.js.nft.json +1 -1
- package/web/.next/server/app/(dashboard)/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/(dashboard)/repository/[repositoryId]/[tab]/page/server-reference-manifest.json +26 -26
- package/web/.next/server/app/(dashboard)/repository/[repositoryId]/[tab]/page.js.nft.json +1 -1
- package/web/.next/server/app/(dashboard)/repository/[repositoryId]/[tab]/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page/server-reference-manifest.json +26 -26
- package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page.js.nft.json +1 -1
- package/web/.next/server/app/(dashboard)/repository/[repositoryId]/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/_global-error.html +2 -2
- package/web/.next/server/app/_global-error.rsc +1 -1
- package/web/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +1 -1
- package/web/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/web/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/web/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/web/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/web/.next/server/app/_not-found/page/server-reference-manifest.json +5 -5
- package/web/.next/server/app/_not-found/page.js.nft.json +1 -1
- package/web/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/settings/page/server-reference-manifest.json +8 -8
- package/web/.next/server/app/settings/page.js.nft.json +1 -1
- package/web/.next/server/app/settings/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/skills/page/server-reference-manifest.json +10 -10
- package/web/.next/server/app/skills/page.js.nft.json +1 -1
- package/web/.next/server/app/skills/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/tools/page/server-reference-manifest.json +10 -10
- package/web/.next/server/app/tools/page.js.nft.json +1 -1
- package/web/.next/server/app/tools/page_client-reference-manifest.js +1 -1
- package/web/.next/server/app/version/page/server-reference-manifest.json +5 -5
- package/web/.next/server/app/version/page.js.nft.json +1 -1
- package/web/.next/server/app/version/page_client-reference-manifest.js +1 -1
- package/web/.next/server/chunks/[root-of-the-server]__a402b567._.js +1 -1
- package/web/.next/server/chunks/[root-of-the-server]__ab4951b1._.js +1 -1
- package/web/.next/server/chunks/[root-of-the-server]__ab4951b1._.js.map +1 -1
- package/web/.next/server/chunks/ssr/744ca_web_components_common_control-center-drawer_create-drawer-client_tsx_5e26fc0a._.js +1 -1
- package/web/.next/server/chunks/ssr/744ca_web_components_common_control-center-drawer_create-drawer-client_tsx_5e26fc0a._.js.map +1 -1
- package/web/.next/server/chunks/ssr/744ca_web_components_common_control-center-drawer_feature-drawer-client_tsx_e9755fc8._.js +2 -2
- package/web/.next/server/chunks/ssr/744ca_web_components_common_control-center-drawer_feature-drawer-client_tsx_e9755fc8._.js.map +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__17ed7ed1._.js +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__17ed7ed1._.js.map +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__28d0d265._.js +2 -2
- package/web/.next/server/chunks/ssr/[root-of-the-server]__357d99f9._.js +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__42bf1807._.js +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__42bf1807._.js.map +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__563f2f7a._.js +3 -0
- package/web/.next/server/chunks/ssr/[root-of-the-server]__563f2f7a._.js.map +1 -0
- package/web/.next/server/chunks/ssr/[root-of-the-server]__56b70465._.js +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__56b70465._.js.map +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__88f7e8e6._.js +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__88f7e8e6._.js.map +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__8b0aac03._.js +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__c30f1f82._.js +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__c30f1f82._.js.map +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__f80bfc75._.js +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__f80bfc75._.js.map +1 -1
- package/web/.next/server/chunks/ssr/_05c23ad9._.js +1 -1
- package/web/.next/server/chunks/ssr/_05c23ad9._.js.map +1 -1
- package/web/.next/server/chunks/ssr/_16eb4fec._.js +1 -1
- package/web/.next/server/chunks/ssr/_16eb4fec._.js.map +1 -1
- package/web/.next/server/chunks/ssr/_3a0b989f._.js +2 -2
- package/web/.next/server/chunks/ssr/_3a0b989f._.js.map +1 -1
- package/web/.next/server/chunks/ssr/_56b9d60f._.js +1 -1
- package/web/.next/server/chunks/ssr/_56b9d60f._.js.map +1 -1
- package/web/.next/server/chunks/ssr/_5f69c13f._.js +1 -1
- package/web/.next/server/chunks/ssr/_5f69c13f._.js.map +1 -1
- package/web/.next/server/chunks/ssr/{_fa0acc22._.js → _5fce01a7._.js} +2 -2
- package/web/.next/server/chunks/ssr/{_fa0acc22._.js.map → _5fce01a7._.js.map} +1 -1
- package/web/.next/server/chunks/ssr/_7c5b97c6._.js +1 -1
- package/web/.next/server/chunks/ssr/_7c5b97c6._.js.map +1 -1
- package/web/.next/server/chunks/ssr/_82c57f10._.js +1 -1
- package/web/.next/server/chunks/ssr/_82c57f10._.js.map +1 -1
- package/web/.next/server/chunks/ssr/_9495d50b._.js +1 -1
- package/web/.next/server/chunks/ssr/_9495d50b._.js.map +1 -1
- package/web/.next/server/chunks/ssr/_a0e3f7e4._.js +1 -1
- package/web/.next/server/chunks/ssr/_a0e3f7e4._.js.map +1 -1
- package/web/.next/server/chunks/ssr/_ac4a3873._.js +1 -1
- package/web/.next/server/chunks/ssr/_ac4a3873._.js.map +1 -1
- package/web/.next/server/chunks/ssr/_ca0aa7f0._.js +1 -1
- package/web/.next/server/chunks/ssr/_ca0aa7f0._.js.map +1 -1
- package/web/.next/server/chunks/ssr/_cb5a021e._.js +1 -1
- package/web/.next/server/chunks/ssr/_cb5a021e._.js.map +1 -1
- package/web/.next/server/chunks/ssr/{_245b6623._.js → _cf1c5b73._.js} +2 -2
- package/web/.next/server/chunks/ssr/{_245b6623._.js.map → _cf1c5b73._.js.map} +1 -1
- package/web/.next/server/chunks/ssr/_d86175ae._.js +1 -1
- package/web/.next/server/chunks/ssr/_d86175ae._.js.map +1 -1
- package/web/.next/server/chunks/ssr/_d8bedf13._.js +1 -1
- package/web/.next/server/chunks/ssr/_d8bedf13._.js.map +1 -1
- package/web/.next/server/chunks/ssr/_e9a73a63._.js +1 -1
- package/web/.next/server/chunks/ssr/_e9a73a63._.js.map +1 -1
- package/web/.next/server/chunks/ssr/_f78b4b9d._.js +3 -0
- package/web/.next/server/chunks/ssr/{_51af0bf4._.js.map → _f78b4b9d._.js.map} +1 -1
- package/web/.next/server/chunks/ssr/_fa7efce3._.js +2 -2
- package/web/.next/server/chunks/ssr/_fa7efce3._.js.map +1 -1
- package/web/.next/server/chunks/ssr/b1a17_presentation_web_components_features_settings_settings-page-client_tsx_6ed9d5f8._.js +1 -1
- package/web/.next/server/chunks/ssr/b1a17_presentation_web_components_features_settings_settings-page-client_tsx_6ed9d5f8._.js.map +1 -1
- package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_skills_page_actions_cdc632e3.js +1 -1
- package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_skills_page_actions_cdc632e3.js.map +1 -1
- package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_tools_page_actions_39ca0924.js +1 -1
- package/web/.next/server/chunks/ssr/src_presentation_web__next-internal_server_app_tools_page_actions_39ca0924.js.map +1 -1
- package/web/.next/server/chunks/ssr/src_presentation_web_app_actions_open-ide_ts_baaca5d5._.js +1 -1
- package/web/.next/server/chunks/ssr/src_presentation_web_components_357e3eb0._.js +1 -1
- package/web/.next/server/chunks/ssr/src_presentation_web_components_357e3eb0._.js.map +1 -1
- package/web/.next/server/chunks/ssr/src_presentation_web_components_features_control-center_7ac3562e._.js +1 -1
- package/web/.next/server/chunks/ssr/src_presentation_web_components_features_control-center_7ac3562e._.js.map +1 -1
- package/web/.next/server/pages/500.html +2 -2
- package/web/.next/server/server-reference-manifest.js +1 -1
- package/web/.next/server/server-reference-manifest.json +45 -45
- package/web/.next/static/chunks/{0df0e1499e672836.js → 1b4429259bbf3064.js} +2 -2
- package/web/.next/static/chunks/{2ea8e9030cd5ab2b.js → 277a7f64d4ec189d.js} +1 -1
- package/web/.next/static/chunks/4908997348bd55f8.js +1 -0
- package/web/.next/static/chunks/6bf775818e4ad42e.js +1 -0
- package/web/.next/static/chunks/76858a51f2fbe99a.js +7 -0
- package/web/.next/static/chunks/{fab1eb6312340c1b.js → 80c4c8b3a5c8e0b6.js} +2 -2
- package/web/.next/static/chunks/{0bb071bc5f7487be.js → a2257ffb1349f838.js} +1 -1
- package/web/.next/static/chunks/a919a9df4ab12a5c.css +1 -0
- package/web/.next/static/chunks/{29f375627d0eccca.js → bdd340ad42a34b41.js} +1 -1
- package/web/.next/static/chunks/ee20803fb301d59e.js +1 -0
- package/web/.next/static/chunks/{0aa0c51000c61a82.js → ef1c9f43aafff65f.js} +2 -2
- package/web/.next/static/chunks/{9033e51dfc21b48d.js → f33efe6a12242a8a.js} +1 -1
- package/web/.next/static/chunks/{8a83eca1de6a4ed0.js → f9a80d3854a8d453.js} +1 -1
- package/web/.next/server/chunks/ssr/[root-of-the-server]__f8dd4422._.js +0 -3
- package/web/.next/server/chunks/ssr/[root-of-the-server]__f8dd4422._.js.map +0 -1
- package/web/.next/server/chunks/ssr/_51af0bf4._.js +0 -3
- package/web/.next/server/chunks/ssr/src_presentation_web_324a47da._.js +0 -3
- package/web/.next/server/chunks/ssr/src_presentation_web_324a47da._.js.map +0 -1
- package/web/.next/static/chunks/19e9b142ad142695.js +0 -7
- package/web/.next/static/chunks/811d8a0007da9137.js +0 -1
- package/web/.next/static/chunks/a36ca50cbdfb3ffe.js +0 -1
- package/web/.next/static/chunks/be7d8a4fee3b9860.js +0 -1
- package/web/.next/static/chunks/c86594475dcb03d5.css +0 -1
- /package/web/.next/static/{vc6t9IqXHzQXsbrkf5_dt → lPbWG8a1RwBDDcvRngp58}/_buildManifest.js +0 -0
- /package/web/.next/static/{vc6t9IqXHzQXsbrkf5_dt → lPbWG8a1RwBDDcvRngp58}/_clientMiddlewareManifest.json +0 -0
- /package/web/.next/static/{vc6t9IqXHzQXsbrkf5_dt → lPbWG8a1RwBDDcvRngp58}/_ssgManifest.js +0 -0
package/dist/src/presentation/web/components/features/control-center/control-center-inner.js
CHANGED
|
@@ -1,20 +1,16 @@
|
|
|
1
1
|
'use client';
|
|
2
|
-
import { jsx as _jsx
|
|
2
|
+
import { jsx as _jsx } from "react/jsx-runtime";
|
|
3
3
|
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
|
4
4
|
import { useRouter, usePathname } from 'next/navigation';
|
|
5
|
-
import { Sparkles, FolderPlus, Github, GitBranch } from 'lucide-react';
|
|
6
5
|
import { useReactFlow } from '@xyflow/react';
|
|
7
6
|
import { FeaturesCanvas } from '../../features/features-canvas/index.js';
|
|
8
7
|
import { CanvasToolbar } from '../../features/features-canvas/canvas-toolbar.js';
|
|
9
|
-
import { FloatingActionButton, } from '../../common/floating-action-button/index.js';
|
|
10
8
|
import { useSidebarFeaturesContext, mapNodeStateToSidebarStatus, } from '../../../hooks/sidebar-features-context.js';
|
|
11
|
-
import { useFeatureFlags } from '../../../hooks/feature-flags-context.js';
|
|
12
9
|
import { useSelectedFeatureId } from '../../../hooks/use-selected-feature-id.js';
|
|
13
10
|
import { useSelectedRepository } from '../../../hooks/use-selected-repository.js';
|
|
14
11
|
import { useSoundAction } from '../../../hooks/use-sound-action.js';
|
|
15
12
|
import { useDrawerCloseGuard } from '../../../hooks/drawer-close-guard.js';
|
|
16
13
|
import { useViewportPersistence } from '../../../hooks/use-viewport-persistence.js';
|
|
17
|
-
import { useSidebar } from '../../ui/sidebar.js';
|
|
18
14
|
import { ControlCenterEmptyState } from './control-center-empty-state.js';
|
|
19
15
|
import { useControlCenterState } from './use-control-center-state.js';
|
|
20
16
|
const AUTO_FOCUS_OPTIONS = {
|
|
@@ -295,65 +291,6 @@ export function ControlCenterInner({ initialNodes, initialEdges }) {
|
|
|
295
291
|
const handlePickFolder = useCallback(() => {
|
|
296
292
|
window.dispatchEvent(new CustomEvent('shep:pick-folder'));
|
|
297
293
|
}, []);
|
|
298
|
-
const
|
|
299
|
-
|
|
300
|
-
const fabActions = useMemo(() => {
|
|
301
|
-
const actions = [
|
|
302
|
-
{
|
|
303
|
-
id: 'new-feature',
|
|
304
|
-
label: 'New Feature',
|
|
305
|
-
icon: _jsx(Sparkles, { className: "h-4 w-4" }),
|
|
306
|
-
onClick: () => {
|
|
307
|
-
clickSound.play();
|
|
308
|
-
guardedNavigate(() => router.push('/create'));
|
|
309
|
-
},
|
|
310
|
-
},
|
|
311
|
-
{
|
|
312
|
-
id: 'add-local-repo',
|
|
313
|
-
label: 'Local Folder',
|
|
314
|
-
icon: _jsx(FolderPlus, { className: "h-4 w-4" }),
|
|
315
|
-
onClick: handlePickFolder,
|
|
316
|
-
},
|
|
317
|
-
];
|
|
318
|
-
if (featureFlags.adoptBranch) {
|
|
319
|
-
actions.push({
|
|
320
|
-
id: 'adopt-branch',
|
|
321
|
-
label: 'Adopt Branch',
|
|
322
|
-
icon: _jsx(GitBranch, { className: "h-4 w-4" }),
|
|
323
|
-
onClick: () => {
|
|
324
|
-
guardedNavigate(() => router.push('/adopt'));
|
|
325
|
-
},
|
|
326
|
-
});
|
|
327
|
-
}
|
|
328
|
-
if (featureFlags.githubImport) {
|
|
329
|
-
actions.push({
|
|
330
|
-
id: 'add-github-repo',
|
|
331
|
-
label: 'From GitHub',
|
|
332
|
-
icon: _jsx(Github, { className: "h-4 w-4" }),
|
|
333
|
-
onClick: () => {
|
|
334
|
-
window.dispatchEvent(new CustomEvent('shep:open-github-import'));
|
|
335
|
-
},
|
|
336
|
-
});
|
|
337
|
-
}
|
|
338
|
-
return actions;
|
|
339
|
-
}, [
|
|
340
|
-
clickSound,
|
|
341
|
-
guardedNavigate,
|
|
342
|
-
router,
|
|
343
|
-
handlePickFolder,
|
|
344
|
-
featureFlags.adoptBranch,
|
|
345
|
-
featureFlags.githubImport,
|
|
346
|
-
]);
|
|
347
|
-
const canvasToolbar = (_jsx(CanvasToolbar, { showArchived: showArchived, onToggleArchived: () => setShowArchived(!showArchived), onResetViewport: resetViewport }));
|
|
348
|
-
return (_jsxs(_Fragment, { children: [_jsx(FeaturesCanvas, { nodes: showCanvas ? displayNodes : [], edges: showCanvas ? edges : [], selectedFeatureId: selectedFeatureId, selectedRepository: selectedRepository, defaultViewport: defaultViewport, onNodesChange: onNodesChange, onConnect: handleConnect, onAddFeature: handleAddFeature, onNodeClick: handleNodeClick, onPaneClick: handleClearDrawers, onMoveEnd: handleMoveEnd, toolbar: canvasToolbar, emptyState: _jsx(ControlCenterEmptyState, { onRepositorySelect: addRepoAndFocus }) }), showCanvas ? _jsx(CreateFab, { actions: fabActions }) : null] }));
|
|
349
|
-
}
|
|
350
|
-
/** (+) FAB that tracks sidebar width via CSS var + transition. */
|
|
351
|
-
function CreateFab({ actions }) {
|
|
352
|
-
const { state } = useSidebar();
|
|
353
|
-
// Sidebar expanded = var(--sidebar-width) = 16rem, collapsed = var(--sidebar-width-icon) = 3rem
|
|
354
|
-
// Position just outside the sidebar edge with 16px gap
|
|
355
|
-
const left = state === 'expanded'
|
|
356
|
-
? 'calc(var(--sidebar-width) + 32px)'
|
|
357
|
-
: 'calc(var(--sidebar-width-icon) + 32px)';
|
|
358
|
-
return (_jsx(FloatingActionButton, { actions: actions, className: "!fixed bottom-6", style: { left, transition: 'left 200ms ease-in-out' } }));
|
|
294
|
+
const canvasToolbar = (_jsx(CanvasToolbar, { showArchived: showArchived, onToggleArchived: () => setShowArchived(!showArchived), onAddFeature: handleAddFeature, onAddRepository: handlePickFolder, onResetViewport: resetViewport }));
|
|
295
|
+
return (_jsx(FeaturesCanvas, { nodes: showCanvas ? displayNodes : [], edges: showCanvas ? edges : [], selectedFeatureId: selectedFeatureId, selectedRepository: selectedRepository, defaultViewport: defaultViewport, onNodesChange: onNodesChange, onConnect: handleConnect, onAddFeature: handleAddFeature, onNodeClick: handleNodeClick, onPaneClick: handleClearDrawers, onMoveEnd: handleMoveEnd, toolbar: canvasToolbar, emptyState: _jsx(ControlCenterEmptyState, { onRepositorySelect: addRepoAndFocus }) }));
|
|
359
296
|
}
|
|
@@ -2,7 +2,12 @@ import type { Viewport } from '@xyflow/react';
|
|
|
2
2
|
export interface CanvasToolbarProps {
|
|
3
3
|
showArchived: boolean;
|
|
4
4
|
onToggleArchived: () => void;
|
|
5
|
+
onAddFeature: () => void;
|
|
6
|
+
onAddRepository: () => void;
|
|
7
|
+
addingRepo?: boolean;
|
|
5
8
|
onResetViewport?: () => Viewport;
|
|
9
|
+
/** Extra action buttons (adopt branch, github import, etc.) */
|
|
10
|
+
extraActions?: React.ReactNode;
|
|
6
11
|
}
|
|
7
|
-
export declare function CanvasToolbar({ showArchived, onToggleArchived, onResetViewport, }: CanvasToolbarProps): import("react/jsx-runtime").JSX.Element;
|
|
12
|
+
export declare function CanvasToolbar({ showArchived, onToggleArchived, onAddFeature, onAddRepository, addingRepo, onResetViewport, extraActions, }: CanvasToolbarProps): import("react/jsx-runtime").JSX.Element;
|
|
8
13
|
//# sourceMappingURL=canvas-toolbar.d.ts.map
|
package/dist/src/presentation/web/components/features/features-canvas/canvas-toolbar.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"canvas-toolbar.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/features/features-canvas/canvas-toolbar.tsx"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;
|
|
1
|
+
{"version":3,"file":"canvas-toolbar.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/features/features-canvas/canvas-toolbar.tsx"],"names":[],"mappings":"AAIA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAK9C,MAAM,WAAW,kBAAkB;IACjC,YAAY,EAAE,OAAO,CAAC;IACtB,gBAAgB,EAAE,MAAM,IAAI,CAAC;IAC7B,YAAY,EAAE,MAAM,IAAI,CAAC;IACzB,eAAe,EAAE,MAAM,IAAI,CAAC;IAC5B,UAAU,CAAC,EAAE,OAAO,CAAC;IACrB,eAAe,CAAC,EAAE,MAAM,QAAQ,CAAC;IACjC,+DAA+D;IAC/D,YAAY,CAAC,EAAE,KAAK,CAAC,SAAS,CAAC;CAChC;AAED,wBAAgB,aAAa,CAAC,EAC5B,YAAY,EACZ,gBAAgB,EAChB,YAAY,EACZ,eAAe,EACf,UAAU,EACV,eAAe,EACf,YAAY,GACb,EAAE,kBAAkB,2CAoEpB"}
|
|
@@ -2,9 +2,10 @@
|
|
|
2
2
|
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
3
3
|
import { useCallback } from 'react';
|
|
4
4
|
import { useReactFlow } from '@xyflow/react';
|
|
5
|
-
import { Eye, EyeOff, ZoomIn, ZoomOut, Maximize, RotateCcw } from 'lucide-react';
|
|
5
|
+
import { Plus, FolderPlus, Eye, EyeOff, ZoomIn, ZoomOut, Maximize, RotateCcw } from 'lucide-react';
|
|
6
6
|
import { cn } from '../../../lib/utils.js';
|
|
7
|
-
|
|
7
|
+
import { Separator } from '../../ui/separator.js';
|
|
8
|
+
export function CanvasToolbar({ showArchived, onToggleArchived, onAddFeature, onAddRepository, addingRepo, onResetViewport, extraActions, }) {
|
|
8
9
|
const { zoomIn, zoomOut, fitView, setViewport } = useReactFlow();
|
|
9
10
|
const handleZoomIn = useCallback(() => {
|
|
10
11
|
zoomIn({ duration: 200 });
|
|
@@ -21,9 +22,9 @@ export function CanvasToolbar({ showArchived, onToggleArchived, onResetViewport,
|
|
|
21
22
|
setViewport(viewport, { duration: 400 });
|
|
22
23
|
}
|
|
23
24
|
}, [onResetViewport, setViewport]);
|
|
24
|
-
return (_jsxs("div", { className: "bg-background flex items-center gap-1 rounded-xl border px-2 py-1.5 shadow-md dark:bg-neutral-900", children: [_jsx(ToolbarButton, { onClick:
|
|
25
|
+
return (_jsxs("div", { className: "bg-background flex items-center gap-1 rounded-xl border px-2 py-1.5 shadow-md dark:bg-neutral-900", children: [_jsx(ToolbarButton, { onClick: onAddFeature, title: "New Feature", label: "New", children: _jsx(Plus, { className: "h-4 w-4" }) }), _jsx(ToolbarButton, { onClick: onAddRepository, title: "Add Repository", label: "Repo", disabled: addingRepo, children: _jsx(FolderPlus, { className: "h-4 w-4" }) }), extraActions, _jsx(Separator, { orientation: "vertical", className: "mx-1 h-6" }), _jsx(ToolbarButton, { onClick: onToggleArchived, title: showArchived ? 'Hide archived' : 'Show archived', active: showArchived, children: showArchived ? _jsx(Eye, { className: "h-4 w-4" }) : _jsx(EyeOff, { className: "h-4 w-4" }) }), _jsx(Separator, { orientation: "vertical", className: "mx-1 h-6" }), _jsx(ToolbarButton, { onClick: handleZoomOut, title: "Zoom out", children: _jsx(ZoomOut, { className: "h-4 w-4" }) }), _jsx(ToolbarButton, { onClick: handleZoomIn, title: "Zoom in", children: _jsx(ZoomIn, { className: "h-4 w-4" }) }), _jsx(ToolbarButton, { onClick: handleFitView, title: "Fit view", children: _jsx(Maximize, { className: "h-4 w-4" }) }), onResetViewport ? (_jsx(ToolbarButton, { onClick: handleReset, title: "Reset view", children: _jsx(RotateCcw, { className: "h-4 w-4" }) })) : null] }));
|
|
25
26
|
}
|
|
26
27
|
// ── Internal button ──────────────────────────────────────────────────────
|
|
27
|
-
function ToolbarButton({ children, onClick, title, label, active, }) {
|
|
28
|
-
return (_jsxs("button", { type: "button", onClick: onClick, title: title, className: cn('inline-flex items-center gap-1.5 rounded-lg px-2 py-1.5 text-xs font-medium transition-colors', 'hover:bg-accent hover:text-accent-foreground', active ? 'text-primary bg-primary/10' : 'text-muted-foreground'), children: [children, label ? _jsx("span", { children: label }) : null] }));
|
|
28
|
+
function ToolbarButton({ children, onClick, title, label, active, disabled, }) {
|
|
29
|
+
return (_jsxs("button", { type: "button", onClick: onClick, title: title, disabled: disabled, className: cn('inline-flex items-center gap-1.5 rounded-lg px-2 py-1.5 text-xs font-medium transition-colors', 'hover:bg-accent hover:text-accent-foreground', 'disabled:pointer-events-none disabled:opacity-50', active ? 'text-primary bg-primary/10' : 'text-muted-foreground'), children: [children, label ? _jsx("span", { children: label }) : null] }));
|
|
29
30
|
}
|
|
@@ -69,5 +69,5 @@ export function FeaturesCanvas({ nodes, edges, selectedFeatureId, selectedReposi
|
|
|
69
69
|
}, [isEmpty]);
|
|
70
70
|
const fallbackEmptyState = isEmpty && !emptyState ? (_jsx(EmptyState, { title: "No features yet", description: "Get started by creating your first feature.", action: _jsxs(Button, { onClick: onAddFeature, children: [_jsx(Plus, { className: "mr-2 h-4 w-4" }), "New Feature"] }) })) : null;
|
|
71
71
|
const overlayContent = emptyState ?? fallbackEmptyState;
|
|
72
|
-
return (_jsxs("div", { "data-testid": isEmpty ? 'features-canvas-empty' : 'features-canvas', "data-no-drawer-close": true, className: "dark:bg-background pointer-events-auto relative h-full w-full bg-[#f6f7f8]", children: [_jsxs(ReactFlow, { nodes: enrichedNodes, edges: edges, nodeTypes: nodeTypes, edgeTypes: edgeTypes, onConnect: onConnect, onNodesChange: onNodesChange, onNodeClick: onNodeClick, onPaneClick: onPaneClick, onMoveStart: onCanvasDrag, onMoveEnd: onMoveEnd, defaultViewport: defaultViewport ?? FALLBACK_VIEWPORT, nodesDraggable: false, nodesConnectable: false, elementsSelectable: false, proOptions: { hideAttribution: true }, className: "[&_.react-flow__pane]:!cursor-default", children: [_jsx(Background, { variant: BackgroundVariant.Dots, gap: 24, size: 1, color: "#b8bcc4", className: "dark:[&_circle]:!fill-white/[0.1]" }), !isEmpty && toolbar ? (_jsx(Panel, { position: "
|
|
72
|
+
return (_jsxs("div", { "data-testid": isEmpty ? 'features-canvas-empty' : 'features-canvas', "data-no-drawer-close": true, className: "dark:bg-background pointer-events-auto relative h-full w-full bg-[#f6f7f8]", children: [_jsxs(ReactFlow, { nodes: enrichedNodes, edges: edges, nodeTypes: nodeTypes, edgeTypes: edgeTypes, onConnect: onConnect, onNodesChange: onNodesChange, onNodeClick: onNodeClick, onPaneClick: onPaneClick, onMoveStart: onCanvasDrag, onMoveEnd: onMoveEnd, defaultViewport: defaultViewport ?? FALLBACK_VIEWPORT, nodesDraggable: false, nodesConnectable: false, elementsSelectable: false, proOptions: { hideAttribution: true }, className: "[&_.react-flow__pane]:!cursor-default", children: [_jsx(Background, { variant: BackgroundVariant.Dots, gap: 24, size: 1, color: "#b8bcc4", className: "dark:[&_circle]:!fill-white/[0.1]" }), !isEmpty && toolbar ? (_jsx(Panel, { position: "bottom-center", className: "!mb-4", children: toolbar })) : null] }), showOverlay && overlayContent ? (_jsx("div", { className: cn('pointer-events-none absolute inset-0 z-10 flex items-center justify-center transition-opacity duration-300', overlayExiting ? 'opacity-0' : 'animate-in fade-in opacity-100 duration-200'), children: _jsx("div", { className: "pointer-events-auto h-full w-full", children: overlayContent }) })) : null] }));
|
|
73
73
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"app-shell.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/layouts/app-shell/app-shell.tsx"],"names":[],"mappings":"AAEA,OAAO,
|
|
1
|
+
{"version":3,"file":"app-shell.d.ts","sourceRoot":"","sources":["../../../../../../../src/presentation/web/components/layouts/app-shell/app-shell.tsx"],"names":[],"mappings":"AAEA,OAAO,EAA6C,KAAK,SAAS,EAAE,MAAM,OAAO,CAAC;AAmBlF,UAAU,aAAa;IACrB,QAAQ,EAAE,SAAS,CAAC;IACpB,6CAA6C;IAC7C,WAAW,CAAC,EAAE,OAAO,CAAC;CACvB;AA4GD,wBAAgB,QAAQ,CAAC,EAAE,QAAQ,EAAE,WAAW,EAAE,EAAE,aAAa,2CAYhE"}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
'use client';
|
|
2
2
|
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
3
|
-
import { useCallback, useEffect, useState } from 'react';
|
|
3
|
+
import { useCallback, useEffect, useMemo, useState } from 'react';
|
|
4
4
|
import { useRouter } from 'next/navigation';
|
|
5
5
|
import { SidebarProvider, SidebarInset } from '../../ui/sidebar.js';
|
|
6
6
|
import { AppSidebar } from '../../layouts/app-sidebar/index.js';
|
|
@@ -57,12 +57,6 @@ function AppShellInner({ children, sidebarOpen }) {
|
|
|
57
57
|
window.addEventListener('shep:pick-folder', handler);
|
|
58
58
|
return () => window.removeEventListener('shep:pick-folder', handler);
|
|
59
59
|
}, [handleAddRepository]);
|
|
60
|
-
// Listen for GitHub import event from (+) FAB
|
|
61
|
-
useEffect(() => {
|
|
62
|
-
const handler = () => setGithubDialogOpen(true);
|
|
63
|
-
window.addEventListener('shep:open-github-import', handler);
|
|
64
|
-
return () => window.removeEventListener('shep:open-github-import', handler);
|
|
65
|
-
}, []);
|
|
66
60
|
const handleReactPickerSelect = useCallback((path) => {
|
|
67
61
|
if (path) {
|
|
68
62
|
window.dispatchEvent(new CustomEvent('shep:add-repository', { detail: { path } }));
|
|
@@ -79,9 +73,11 @@ function AppShellInner({ children, sidebarOpen }) {
|
|
|
79
73
|
setShowReactPicker(false);
|
|
80
74
|
}, onSelect: handleReactPickerSelect })] }));
|
|
81
75
|
}
|
|
82
|
-
/** Wraps children with TurnStatusesProvider
|
|
76
|
+
/** Wraps children with TurnStatusesProvider, collecting scope IDs from sidebar features. */
|
|
83
77
|
function TurnStatusesBridge({ children }) {
|
|
84
|
-
|
|
78
|
+
const { features } = useSidebarFeaturesContext();
|
|
79
|
+
const scopeIds = useMemo(() => ['global', ...features.map((f) => f.featureId)], [features]);
|
|
80
|
+
return _jsx(TurnStatusesProvider, { scopeIds: scopeIds, children: children });
|
|
85
81
|
}
|
|
86
82
|
export function AppShell({ children, sidebarOpen }) {
|
|
87
83
|
return (_jsx(AgentEventsProvider, { children: _jsx(DrawerCloseGuardProvider, { children: _jsx(SidebarFeaturesProvider, { children: _jsx(TurnStatusesBridge, { children: _jsx(AppShellInner, { sidebarOpen: sidebarOpen, children: children }) }) }) }) }));
|
|
@@ -1,15 +1,19 @@
|
|
|
1
1
|
import { type ReactNode } from 'react';
|
|
2
2
|
import { type TurnStatus } from './use-turn-statuses.js';
|
|
3
|
+
interface TurnStatusesProviderProps {
|
|
4
|
+
/** All scope IDs to poll turn statuses for */
|
|
5
|
+
scopeIds: string[];
|
|
6
|
+
children: ReactNode;
|
|
7
|
+
}
|
|
3
8
|
/**
|
|
4
|
-
* Polls
|
|
9
|
+
* Polls turn statuses for all provided scope IDs in a single API call.
|
|
5
10
|
* Children use `useTurnStatus(scopeId)` to read individual statuses.
|
|
6
11
|
*/
|
|
7
|
-
export declare function TurnStatusesProvider({ children }:
|
|
8
|
-
children: ReactNode;
|
|
9
|
-
}): import("react/jsx-runtime").JSX.Element;
|
|
12
|
+
export declare function TurnStatusesProvider({ scopeIds, children }: TurnStatusesProviderProps): import("react/jsx-runtime").JSX.Element;
|
|
10
13
|
/**
|
|
11
14
|
* Get the turn status for a specific scope ID.
|
|
12
15
|
* Must be used within a TurnStatusesProvider.
|
|
13
16
|
*/
|
|
14
17
|
export declare function useTurnStatus(scopeId: string): TurnStatus;
|
|
18
|
+
export {};
|
|
15
19
|
//# sourceMappingURL=turn-statuses-provider.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"turn-statuses-provider.d.ts","sourceRoot":"","sources":["../../../../../src/presentation/web/hooks/turn-statuses-provider.tsx"],"names":[],"mappings":"AAEA,OAAO,EAAsC,KAAK,SAAS,EAAE,MAAM,OAAO,CAAC;AAC3E,OAAO,
|
|
1
|
+
{"version":3,"file":"turn-statuses-provider.d.ts","sourceRoot":"","sources":["../../../../../src/presentation/web/hooks/turn-statuses-provider.tsx"],"names":[],"mappings":"AAEA,OAAO,EAAsC,KAAK,SAAS,EAAE,MAAM,OAAO,CAAC;AAC3E,OAAO,EAAmB,KAAK,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAWvE,UAAU,yBAAyB;IACjC,8CAA8C;IAC9C,QAAQ,EAAE,MAAM,EAAE,CAAC;IACnB,QAAQ,EAAE,SAAS,CAAC;CACrB;AAED;;;GAGG;AACH,wBAAgB,oBAAoB,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAE,EAAE,yBAAyB,2CAWrF;AAED;;;GAGG;AACH,wBAAgB,aAAa,CAAC,OAAO,EAAE,MAAM,GAAG,UAAU,CAGzD"}
|
|
@@ -1,16 +1,16 @@
|
|
|
1
1
|
'use client';
|
|
2
2
|
import { jsx as _jsx } from "react/jsx-runtime";
|
|
3
3
|
import { createContext, useContext, useMemo } from 'react';
|
|
4
|
-
import {
|
|
4
|
+
import { useTurnStatuses } from './use-turn-statuses.js';
|
|
5
5
|
const TurnStatusesContext = createContext({
|
|
6
6
|
getStatus: () => 'idle',
|
|
7
7
|
});
|
|
8
8
|
/**
|
|
9
|
-
* Polls
|
|
9
|
+
* Polls turn statuses for all provided scope IDs in a single API call.
|
|
10
10
|
* Children use `useTurnStatus(scopeId)` to read individual statuses.
|
|
11
11
|
*/
|
|
12
|
-
export function TurnStatusesProvider({ children }) {
|
|
13
|
-
const statuses =
|
|
12
|
+
export function TurnStatusesProvider({ scopeIds, children }) {
|
|
13
|
+
const statuses = useTurnStatuses(scopeIds);
|
|
14
14
|
const value = useMemo(() => ({
|
|
15
15
|
getStatus: (scopeId) => statuses[scopeId] ?? 'idle',
|
|
16
16
|
}), [statuses]);
|
|
@@ -1,10 +1,11 @@
|
|
|
1
1
|
export type TurnStatus = 'idle' | 'processing' | 'unread';
|
|
2
2
|
/**
|
|
3
|
-
* Polls
|
|
4
|
-
*
|
|
5
|
-
*
|
|
3
|
+
* Polls turn statuses for multiple feature/scope IDs.
|
|
4
|
+
* Returns a map of featureId → TurnStatus for dot indicator rendering.
|
|
5
|
+
*
|
|
6
|
+
* Polls every 3 seconds to keep indicators responsive.
|
|
6
7
|
*/
|
|
7
|
-
export declare function
|
|
8
|
+
export declare function useTurnStatuses(featureIds: string[]): Record<string, TurnStatus>;
|
|
8
9
|
/**
|
|
9
10
|
* Marks a feature's chat as read (clears 'unread' → 'idle').
|
|
10
11
|
*/
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"use-turn-statuses.d.ts","sourceRoot":"","sources":["../../../../../src/presentation/web/hooks/use-turn-statuses.ts"],"names":[],"mappings":"AAIA,MAAM,MAAM,UAAU,GAAG,MAAM,GAAG,YAAY,GAAG,QAAQ,CAAC;AAE1D
|
|
1
|
+
{"version":3,"file":"use-turn-statuses.d.ts","sourceRoot":"","sources":["../../../../../src/presentation/web/hooks/use-turn-statuses.ts"],"names":[],"mappings":"AAIA,MAAM,MAAM,UAAU,GAAG,MAAM,GAAG,YAAY,GAAG,QAAQ,CAAC;AAE1D;;;;;GAKG;AACH,wBAAgB,eAAe,CAAC,UAAU,EAAE,MAAM,EAAE,GAAG,MAAM,CAAC,MAAM,EAAE,UAAU,CAAC,CAgBhF;AAED;;GAEG;AACH,wBAAsB,YAAY,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAEnE"}
|
|
@@ -1,20 +1,25 @@
|
|
|
1
1
|
'use client';
|
|
2
2
|
import { useQuery } from '@tanstack/react-query';
|
|
3
3
|
/**
|
|
4
|
-
* Polls
|
|
5
|
-
*
|
|
6
|
-
*
|
|
4
|
+
* Polls turn statuses for multiple feature/scope IDs.
|
|
5
|
+
* Returns a map of featureId → TurnStatus for dot indicator rendering.
|
|
6
|
+
*
|
|
7
|
+
* Polls every 3 seconds to keep indicators responsive.
|
|
7
8
|
*/
|
|
8
|
-
export function
|
|
9
|
+
export function useTurnStatuses(featureIds) {
|
|
10
|
+
const sortedIds = [...featureIds].sort().join(',');
|
|
9
11
|
const { data } = useQuery({
|
|
10
|
-
queryKey: ['turn-statuses'],
|
|
12
|
+
queryKey: ['turn-statuses', sortedIds],
|
|
11
13
|
queryFn: async () => {
|
|
12
|
-
|
|
14
|
+
if (!sortedIds)
|
|
15
|
+
return {};
|
|
16
|
+
const res = await fetch(`/api/interactive/chat/turn-statuses?featureIds=${sortedIds}`);
|
|
13
17
|
if (!res.ok)
|
|
14
18
|
return {};
|
|
15
19
|
return res.json();
|
|
16
20
|
},
|
|
17
|
-
refetchInterval:
|
|
21
|
+
refetchInterval: 3_000,
|
|
22
|
+
enabled: featureIds.length > 0,
|
|
18
23
|
});
|
|
19
24
|
return data ?? {};
|
|
20
25
|
}
|