@nex-ai/nex 0.1.21 → 0.1.23
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/README.md +11 -2
- package/dist/agent/adoption.d.ts +23 -0
- package/dist/agent/adoption.js +87 -0
- package/dist/agent/adoption.js.map +1 -0
- package/dist/agent/gossip.d.ts +17 -0
- package/dist/agent/gossip.js +48 -0
- package/dist/agent/gossip.js.map +1 -0
- package/dist/agent/loop.d.ts +59 -0
- package/dist/agent/loop.js +389 -0
- package/dist/agent/loop.js.map +1 -0
- package/dist/agent/queues.d.ts +16 -0
- package/dist/agent/queues.js +44 -0
- package/dist/agent/queues.js.map +1 -0
- package/dist/agent/session-store.d.ts +21 -0
- package/dist/agent/session-store.js +96 -0
- package/dist/agent/session-store.js.map +1 -0
- package/dist/agent/templates.d.ts +5 -0
- package/dist/agent/templates.js +48 -0
- package/dist/agent/templates.js.map +1 -0
- package/dist/agent/tools.d.ts +25 -0
- package/dist/agent/tools.js +238 -0
- package/dist/agent/tools.js.map +1 -0
- package/dist/agent/types.d.ts +59 -0
- package/dist/agent/types.js +5 -0
- package/dist/agent/types.js.map +1 -0
- package/dist/calendar/scheduler.d.ts +32 -0
- package/dist/calendar/scheduler.js +178 -0
- package/dist/calendar/scheduler.js.map +1 -0
- package/dist/calendar/store.d.ts +15 -0
- package/dist/calendar/store.js +50 -0
- package/dist/calendar/store.js.map +1 -0
- package/dist/calendar/types.d.ts +17 -0
- package/dist/calendar/types.js +5 -0
- package/dist/calendar/types.js.map +1 -0
- package/dist/chat/channel.d.ts +20 -0
- package/dist/chat/channel.js +84 -0
- package/dist/chat/channel.js.map +1 -0
- package/dist/chat/message-store.d.ts +18 -0
- package/dist/chat/message-store.js +82 -0
- package/dist/chat/message-store.js.map +1 -0
- package/dist/chat/router.d.ts +17 -0
- package/dist/chat/router.js +65 -0
- package/dist/chat/router.js.map +1 -0
- package/dist/chat/suggested-responses.d.ts +6 -0
- package/dist/chat/suggested-responses.js +99 -0
- package/dist/chat/suggested-responses.js.map +1 -0
- package/dist/chat/types.d.ts +28 -0
- package/dist/chat/types.js +5 -0
- package/dist/chat/types.js.map +1 -0
- package/dist/cli.js +3 -1
- package/dist/cli.js.map +1 -1
- package/dist/commands/dispatch.d.ts +39 -0
- package/dist/commands/dispatch.js +1293 -0
- package/dist/commands/dispatch.js.map +1 -0
- package/dist/commands/init.d.ts +49 -0
- package/dist/commands/init.js +315 -0
- package/dist/commands/init.js.map +1 -0
- package/dist/commands/parse-input.d.ts +5 -0
- package/dist/commands/parse-input.js +37 -0
- package/dist/commands/parse-input.js.map +1 -0
- package/dist/index.d.ts +6 -23
- package/dist/index.js +118 -50
- package/dist/index.js.map +1 -1
- package/dist/lib/client.js +1 -1
- package/dist/lib/client.js.map +1 -1
- package/dist/lib/errors.js +1 -1
- package/dist/lib/errors.js.map +1 -1
- package/dist/lib/graph-html.d.ts +1 -0
- package/dist/lib/graph-html.js +16 -3
- package/dist/lib/graph-html.js.map +1 -1
- package/dist/lib/integrations.d.ts +23 -0
- package/dist/lib/integrations.js +25 -0
- package/dist/lib/integrations.js.map +1 -0
- package/dist/lib/nex-mcp-config.d.ts +14 -0
- package/dist/lib/nex-mcp-config.js +26 -0
- package/dist/lib/nex-mcp-config.js.map +1 -0
- package/dist/orchestration/budget.d.ts +32 -0
- package/dist/orchestration/budget.js +72 -0
- package/dist/orchestration/budget.js.map +1 -0
- package/dist/orchestration/executor.d.ts +59 -0
- package/dist/orchestration/executor.js +197 -0
- package/dist/orchestration/executor.js.map +1 -0
- package/dist/orchestration/router.d.ts +30 -0
- package/dist/orchestration/router.js +98 -0
- package/dist/orchestration/router.js.map +1 -0
- package/dist/orchestration/templates.d.ts +11 -0
- package/dist/orchestration/templates.js +99 -0
- package/dist/orchestration/templates.js.map +1 -0
- package/dist/orchestration/types.d.ts +58 -0
- package/dist/orchestration/types.js +7 -0
- package/dist/orchestration/types.js.map +1 -0
- package/dist/plugin/shared.js +1 -1
- package/dist/plugin/shared.js.map +1 -1
- package/dist/tui/agent-colors.d.ts +19 -0
- package/dist/tui/agent-colors.js +34 -0
- package/dist/tui/agent-colors.js.map +1 -0
- package/dist/tui/app.d.ts +14 -0
- package/dist/tui/app.js +267 -0
- package/dist/tui/app.js.map +1 -0
- package/dist/tui/channel-colors.d.ts +16 -0
- package/dist/tui/channel-colors.js +43 -0
- package/dist/tui/channel-colors.js.map +1 -0
- package/dist/tui/components/agent-card.d.ts +11 -0
- package/dist/tui/components/agent-card.js +40 -0
- package/dist/tui/components/agent-card.js.map +1 -0
- package/dist/tui/components/banner.d.ts +44 -0
- package/dist/tui/components/banner.js +130 -0
- package/dist/tui/components/banner.js.map +1 -0
- package/dist/tui/components/channel-colors.d.ts +21 -0
- package/dist/tui/components/channel-colors.js +53 -0
- package/dist/tui/components/channel-colors.js.map +1 -0
- package/dist/tui/components/channel-message.d.ts +53 -0
- package/dist/tui/components/channel-message.js +66 -0
- package/dist/tui/components/channel-message.js.map +1 -0
- package/dist/tui/components/chat-input.d.ts +11 -0
- package/dist/tui/components/chat-input.js +19 -0
- package/dist/tui/components/chat-input.js.map +1 -0
- package/dist/tui/components/data-table.d.ts +11 -0
- package/dist/tui/components/data-table.js +47 -0
- package/dist/tui/components/data-table.js.map +1 -0
- package/dist/tui/components/error-box.d.ts +11 -0
- package/dist/tui/components/error-box.js +59 -0
- package/dist/tui/components/error-box.js.map +1 -0
- package/dist/tui/components/help-screen.d.ts +3 -0
- package/dist/tui/components/help-screen.js +63 -0
- package/dist/tui/components/help-screen.js.map +1 -0
- package/dist/tui/components/index.d.ts +17 -0
- package/dist/tui/components/index.js +10 -0
- package/dist/tui/components/index.js.map +1 -0
- package/dist/tui/components/inline-confirm.d.ts +14 -0
- package/dist/tui/components/inline-confirm.js +13 -0
- package/dist/tui/components/inline-confirm.js.map +1 -0
- package/dist/tui/components/inline-select.d.ts +21 -0
- package/dist/tui/components/inline-select.js +20 -0
- package/dist/tui/components/inline-select.js.map +1 -0
- package/dist/tui/components/markdown.d.ts +6 -0
- package/dist/tui/components/markdown.js +179 -0
- package/dist/tui/components/markdown.js.map +1 -0
- package/dist/tui/components/mention-autocomplete.d.ts +77 -0
- package/dist/tui/components/mention-autocomplete.js +235 -0
- package/dist/tui/components/mention-autocomplete.js.map +1 -0
- package/dist/tui/components/message-list.d.ts +18 -0
- package/dist/tui/components/message-list.js +29 -0
- package/dist/tui/components/message-list.js.map +1 -0
- package/dist/tui/components/picker.d.ts +16 -0
- package/dist/tui/components/picker.js +32 -0
- package/dist/tui/components/picker.js.map +1 -0
- package/dist/tui/components/progress-steps.d.ts +13 -0
- package/dist/tui/components/progress-steps.js +21 -0
- package/dist/tui/components/progress-steps.js.map +1 -0
- package/dist/tui/components/slack/compose.d.ts +27 -0
- package/dist/tui/components/slack/compose.js +123 -0
- package/dist/tui/components/slack/compose.js.map +1 -0
- package/dist/tui/components/slack/layout.d.ts +36 -0
- package/dist/tui/components/slack/layout.js +96 -0
- package/dist/tui/components/slack/layout.js.map +1 -0
- package/dist/tui/components/slack/messages.d.ts +72 -0
- package/dist/tui/components/slack/messages.js +177 -0
- package/dist/tui/components/slack/messages.js.map +1 -0
- package/dist/tui/components/slack/quick-switcher.d.ts +33 -0
- package/dist/tui/components/slack/quick-switcher.js +98 -0
- package/dist/tui/components/slack/quick-switcher.js.map +1 -0
- package/dist/tui/components/slack/sidebar-types.d.ts +65 -0
- package/dist/tui/components/slack/sidebar-types.js +7 -0
- package/dist/tui/components/slack/sidebar-types.js.map +1 -0
- package/dist/tui/components/slack/sidebar.d.ts +46 -0
- package/dist/tui/components/slack/sidebar.js +52 -0
- package/dist/tui/components/slack/sidebar.js.map +1 -0
- package/dist/tui/components/slack/thread-panel.d.ts +45 -0
- package/dist/tui/components/slack/thread-panel.js +47 -0
- package/dist/tui/components/slack/thread-panel.js.map +1 -0
- package/dist/tui/components/slash-autocomplete.d.ts +81 -0
- package/dist/tui/components/slash-autocomplete.js +218 -0
- package/dist/tui/components/slash-autocomplete.js.map +1 -0
- package/dist/tui/components/spinner.d.ts +17 -0
- package/dist/tui/components/spinner.js +30 -0
- package/dist/tui/components/spinner.js.map +1 -0
- package/dist/tui/components/status-bar.d.ts +23 -0
- package/dist/tui/components/status-bar.js +55 -0
- package/dist/tui/components/status-bar.js.map +1 -0
- package/dist/tui/components/success-box.d.ts +7 -0
- package/dist/tui/components/success-box.js +10 -0
- package/dist/tui/components/success-box.js.map +1 -0
- package/dist/tui/components/tool-indicator.d.ts +12 -0
- package/dist/tui/components/tool-indicator.js +18 -0
- package/dist/tui/components/tool-indicator.js.map +1 -0
- package/dist/tui/components/viewport.d.ts +9 -0
- package/dist/tui/components/viewport.js +24 -0
- package/dist/tui/components/viewport.js.map +1 -0
- package/dist/tui/generative/bindings.d.ts +27 -0
- package/dist/tui/generative/bindings.js +152 -0
- package/dist/tui/generative/bindings.js.map +1 -0
- package/dist/tui/generative/registry.d.ts +19 -0
- package/dist/tui/generative/registry.js +31 -0
- package/dist/tui/generative/registry.js.map +1 -0
- package/dist/tui/generative/renderer.d.ts +24 -0
- package/dist/tui/generative/renderer.js +160 -0
- package/dist/tui/generative/renderer.js.map +1 -0
- package/dist/tui/generative/types.d.ts +68 -0
- package/dist/tui/generative/types.js +7 -0
- package/dist/tui/generative/types.js.map +1 -0
- package/dist/tui/hooks/use-cancellable.d.ts +28 -0
- package/dist/tui/hooks/use-cancellable.js +51 -0
- package/dist/tui/hooks/use-cancellable.js.map +1 -0
- package/dist/tui/hooks/use-streaming.d.ts +44 -0
- package/dist/tui/hooks/use-streaming.js +105 -0
- package/dist/tui/hooks/use-streaming.js.map +1 -0
- package/dist/tui/index.d.ts +4 -0
- package/dist/tui/index.js +7 -0
- package/dist/tui/index.js.map +1 -0
- package/dist/tui/keybindings.d.ts +25 -0
- package/dist/tui/keybindings.js +277 -0
- package/dist/tui/keybindings.js.map +1 -0
- package/dist/tui/register-views.d.ts +8 -0
- package/dist/tui/register-views.js +116 -0
- package/dist/tui/register-views.js.map +1 -0
- package/dist/tui/router.d.ts +29 -0
- package/dist/tui/router.js +87 -0
- package/dist/tui/router.js.map +1 -0
- package/dist/tui/services/agent-service.d.ts +58 -0
- package/dist/tui/services/agent-service.js +197 -0
- package/dist/tui/services/agent-service.js.map +1 -0
- package/dist/tui/services/calendar-service.d.ts +31 -0
- package/dist/tui/services/calendar-service.js +133 -0
- package/dist/tui/services/calendar-service.js.map +1 -0
- package/dist/tui/services/chat-service.d.ts +28 -0
- package/dist/tui/services/chat-service.js +91 -0
- package/dist/tui/services/chat-service.js.map +1 -0
- package/dist/tui/services/orchestration-service.d.ts +42 -0
- package/dist/tui/services/orchestration-service.js +153 -0
- package/dist/tui/services/orchestration-service.js.map +1 -0
- package/dist/tui/slash-commands.d.ts +106 -0
- package/dist/tui/slash-commands.js +1421 -0
- package/dist/tui/slash-commands.js.map +1 -0
- package/dist/tui/store.d.ts +96 -0
- package/dist/tui/store.js +120 -0
- package/dist/tui/store.js.map +1 -0
- package/dist/tui/theme.d.ts +40 -0
- package/dist/tui/theme.js +39 -0
- package/dist/tui/theme.js.map +1 -0
- package/dist/tui/tui-context.d.ts +22 -0
- package/dist/tui/tui-context.js +17 -0
- package/dist/tui/tui-context.js.map +1 -0
- package/dist/tui/views/agent-list.d.ts +8 -0
- package/dist/tui/views/agent-list.js +11 -0
- package/dist/tui/views/agent-list.js.map +1 -0
- package/dist/tui/views/ask-chat.d.ts +9 -0
- package/dist/tui/views/ask-chat.js +40 -0
- package/dist/tui/views/ask-chat.js.map +1 -0
- package/dist/tui/views/calendar.d.ts +15 -0
- package/dist/tui/views/calendar.js +29 -0
- package/dist/tui/views/calendar.js.map +1 -0
- package/dist/tui/views/chat.d.ts +18 -0
- package/dist/tui/views/chat.js +28 -0
- package/dist/tui/views/chat.js.map +1 -0
- package/dist/tui/views/generative.d.ts +14 -0
- package/dist/tui/views/generative.js +37 -0
- package/dist/tui/views/generative.js.map +1 -0
- package/dist/tui/views/help.d.ts +6 -0
- package/dist/tui/views/help.js +9 -0
- package/dist/tui/views/help.js.map +1 -0
- package/dist/tui/views/home-screen.d.ts +67 -0
- package/dist/tui/views/home-screen.js +192 -0
- package/dist/tui/views/home-screen.js.map +1 -0
- package/dist/tui/views/home.d.ts +33 -0
- package/dist/tui/views/home.js +60 -0
- package/dist/tui/views/home.js.map +1 -0
- package/dist/tui/views/index.d.ts +20 -0
- package/dist/tui/views/index.js +11 -0
- package/dist/tui/views/index.js.map +1 -0
- package/dist/tui/views/insights.d.ts +19 -0
- package/dist/tui/views/insights.js +46 -0
- package/dist/tui/views/insights.js.map +1 -0
- package/dist/tui/views/orchestration.d.ts +18 -0
- package/dist/tui/views/orchestration.js +73 -0
- package/dist/tui/views/orchestration.js.map +1 -0
- package/dist/tui/views/record-detail.d.ts +10 -0
- package/dist/tui/views/record-detail.js +22 -0
- package/dist/tui/views/record-detail.js.map +1 -0
- package/dist/tui/views/record-list.d.ts +15 -0
- package/dist/tui/views/record-list.js +22 -0
- package/dist/tui/views/record-list.js.map +1 -0
- package/dist/tui/views/slack-channel-header.d.ts +15 -0
- package/dist/tui/views/slack-channel-header.js +16 -0
- package/dist/tui/views/slack-channel-header.js.map +1 -0
- package/dist/tui/views/slack-home.d.ts +21 -0
- package/dist/tui/views/slack-home.js +572 -0
- package/dist/tui/views/slack-home.js.map +1 -0
- package/dist/tui/views/task-board.d.ts +21 -0
- package/dist/tui/views/task-board.js +33 -0
- package/dist/tui/views/task-board.js.map +1 -0
- package/dist/tui/views/timeline.d.ts +17 -0
- package/dist/tui/views/timeline.js +24 -0
- package/dist/tui/views/timeline.js.map +1 -0
- package/package.json +8 -3
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
import { Box, Text } from "ink";
|
|
3
|
+
import { useSpinner } from "./spinner.js";
|
|
4
|
+
// ── Step icon component ─────────────────────────────────────────────
|
|
5
|
+
function StepIcon({ status }) {
|
|
6
|
+
const frame = useSpinner();
|
|
7
|
+
if (status === "done") {
|
|
8
|
+
return _jsx(Text, { color: "green", children: "✓" });
|
|
9
|
+
}
|
|
10
|
+
if (status === "active") {
|
|
11
|
+
return _jsx(Text, { color: "cyan", children: frame });
|
|
12
|
+
}
|
|
13
|
+
// pending
|
|
14
|
+
return _jsx(Text, { dimColor: true, children: "○" });
|
|
15
|
+
}
|
|
16
|
+
// ── Component ───────────────────────────────────────────────────────
|
|
17
|
+
export function ProgressSteps({ steps, direction = "column", }) {
|
|
18
|
+
return (_jsx(Box, { flexDirection: direction, gap: direction === "row" ? 2 : 0, children: steps.map((step, i) => (_jsxs(Box, { gap: 1, children: [_jsx(StepIcon, { status: step.status }), _jsx(Text, { color: step.status === "active" ? "white" : undefined, dimColor: step.status === "pending", bold: step.status === "active", children: step.label })] }, i))) }));
|
|
19
|
+
}
|
|
20
|
+
export default ProgressSteps;
|
|
21
|
+
//# sourceMappingURL=progress-steps.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"progress-steps.js","sourceRoot":"","sources":["../../../src/tui/components/progress-steps.tsx"],"names":[],"mappings":";AACA,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,MAAM,KAAK,CAAC;AAChC,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AAiB1C,uEAAuE;AAEvE,SAAS,QAAQ,CAAC,EAAE,MAAM,EAA0B;IAClD,MAAM,KAAK,GAAG,UAAU,EAAE,CAAC;IAE3B,IAAI,MAAM,KAAK,MAAM,EAAE,CAAC;QACtB,OAAO,KAAC,IAAI,IAAC,KAAK,EAAC,OAAO,YAAE,GAAG,GAAQ,CAAC;IAC1C,CAAC;IACD,IAAI,MAAM,KAAK,QAAQ,EAAE,CAAC;QACxB,OAAO,KAAC,IAAI,IAAC,KAAK,EAAC,MAAM,YAAE,KAAK,GAAQ,CAAC;IAC3C,CAAC;IACD,UAAU;IACV,OAAO,KAAC,IAAI,IAAC,QAAQ,kBAAE,GAAG,GAAQ,CAAC;AACrC,CAAC;AAED,uEAAuE;AAEvE,MAAM,UAAU,aAAa,CAAC,EAC5B,KAAK,EACL,SAAS,GAAG,QAAQ,GACD;IACnB,OAAO,CACL,KAAC,GAAG,IAAC,aAAa,EAAE,SAAS,EAAE,GAAG,EAAE,SAAS,KAAK,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,YAC5D,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,CAAC,EAAE,EAAE,CAAC,CACtB,MAAC,GAAG,IAAS,GAAG,EAAE,CAAC,aACjB,KAAC,QAAQ,IAAC,MAAM,EAAE,IAAI,CAAC,MAAM,GAAI,EACjC,KAAC,IAAI,IACH,KAAK,EAAE,IAAI,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS,EACrD,QAAQ,EAAE,IAAI,CAAC,MAAM,KAAK,SAAS,EACnC,IAAI,EAAE,IAAI,CAAC,MAAM,KAAK,QAAQ,YAE7B,IAAI,CAAC,KAAK,GACN,KARC,CAAC,CASL,CACP,CAAC,GACE,CACP,CAAC;AACJ,CAAC;AAED,eAAe,aAAa,CAAC"}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Slack-style compose area component.
|
|
3
|
+
*
|
|
4
|
+
* Message input at the bottom of the main panel with slash command
|
|
5
|
+
* and @mention autocomplete integration. Uses TextInput with a
|
|
6
|
+
* submitKey remount trick for clearing after send.
|
|
7
|
+
*/
|
|
8
|
+
import React from "react";
|
|
9
|
+
import type { SlashCommandEntry } from "../slash-autocomplete.js";
|
|
10
|
+
import type { AgentEntry } from "../mention-autocomplete.js";
|
|
11
|
+
export interface ComposeAreaProps {
|
|
12
|
+
channelName: string;
|
|
13
|
+
channelType: "channel" | "dm" | "group-dm";
|
|
14
|
+
/** DM recipient name for placeholder */
|
|
15
|
+
recipientName?: string;
|
|
16
|
+
focused: boolean;
|
|
17
|
+
isThread?: boolean;
|
|
18
|
+
onSubmit: (value: string) => void;
|
|
19
|
+
onSlashCommand?: (command: string, args: string) => void;
|
|
20
|
+
slashCommands: SlashCommandEntry[];
|
|
21
|
+
agents: AgentEntry[];
|
|
22
|
+
}
|
|
23
|
+
export declare function HintBar({ visible, }: {
|
|
24
|
+
visible: boolean;
|
|
25
|
+
}): React.JSX.Element | null;
|
|
26
|
+
export declare function ComposeArea({ channelName, channelType, recipientName, focused, isThread, onSubmit, onSlashCommand, slashCommands, agents, }: ComposeAreaProps): React.JSX.Element;
|
|
27
|
+
export default ComposeArea;
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
/**
|
|
3
|
+
* Slack-style compose area component.
|
|
4
|
+
*
|
|
5
|
+
* Message input at the bottom of the main panel with slash command
|
|
6
|
+
* and @mention autocomplete integration. Uses TextInput with a
|
|
7
|
+
* submitKey remount trick for clearing after send.
|
|
8
|
+
*/
|
|
9
|
+
import { useState, useEffect, useCallback, useRef } from "react";
|
|
10
|
+
import { Box, Text } from "ink";
|
|
11
|
+
import { TextInput } from "@inkjs/ui";
|
|
12
|
+
import { SlashAutocomplete, useSlashAutocomplete, } from "../slash-autocomplete.js";
|
|
13
|
+
import { MentionAutocomplete, useMentionAutocomplete, } from "../mention-autocomplete.js";
|
|
14
|
+
// ── Helpers ─────────────────────────────────────────────────────────
|
|
15
|
+
function getPlaceholder(channelName, channelType, recipientName, isThread) {
|
|
16
|
+
if (isThread)
|
|
17
|
+
return "Reply in thread...";
|
|
18
|
+
if (channelType === "dm" && recipientName) {
|
|
19
|
+
return `Message ${recipientName}`;
|
|
20
|
+
}
|
|
21
|
+
if (channelType === "group-dm") {
|
|
22
|
+
return `Message ${channelName}`;
|
|
23
|
+
}
|
|
24
|
+
return `Message #${channelName}`;
|
|
25
|
+
}
|
|
26
|
+
// ── HintBar ─────────────────────────────────────────────────────────
|
|
27
|
+
export function HintBar({ visible, }) {
|
|
28
|
+
if (!visible)
|
|
29
|
+
return null;
|
|
30
|
+
return (_jsx(Box, { paddingX: 1, children: _jsx(Text, { color: "gray", children: "@ mention \u00B7 / command \u00B7 Enter send" }) }));
|
|
31
|
+
}
|
|
32
|
+
// ── ComposeArea ─────────────────────────────────────────────────────
|
|
33
|
+
export function ComposeArea({ channelName, channelType, recipientName, focused, isThread, onSubmit, onSlashCommand, slashCommands, agents, }) {
|
|
34
|
+
const [value, setValue] = useState("");
|
|
35
|
+
const [submitKey, setSubmitKey] = useState(0);
|
|
36
|
+
// Autocomplete hooks
|
|
37
|
+
const slash = useSlashAutocomplete(slashCommands);
|
|
38
|
+
const mention = useMentionAutocomplete(agents, value);
|
|
39
|
+
const placeholder = getPlaceholder(channelName, channelType, recipientName, isThread);
|
|
40
|
+
// Refs so all callbacks can read current state/actions without being recreated
|
|
41
|
+
const slashRef = useRef(slash);
|
|
42
|
+
slashRef.current = slash;
|
|
43
|
+
const mentionRef = useRef(mention);
|
|
44
|
+
mentionRef.current = mention;
|
|
45
|
+
const setValueRef = useRef(setValue);
|
|
46
|
+
setValueRef.current = setValue;
|
|
47
|
+
// Stable ([] deps) — reads actions through refs so TextInput never gets a new onChange
|
|
48
|
+
const handleChange = useCallback((newValue) => {
|
|
49
|
+
setValue(newValue);
|
|
50
|
+
slashRef.current.actions.update(newValue);
|
|
51
|
+
mentionRef.current.actions.update(newValue);
|
|
52
|
+
}, []);
|
|
53
|
+
// Handle submit — Enter accepts first autocomplete item if overlay is visible
|
|
54
|
+
const handleSubmit = useCallback((submitted) => {
|
|
55
|
+
// If slash autocomplete is visible, Enter accepts the selected (or first) match
|
|
56
|
+
if (slashRef.current.state.visible) {
|
|
57
|
+
const result = slashRef.current.actions.onAccept();
|
|
58
|
+
if (result) {
|
|
59
|
+
setValueRef.current(result.text);
|
|
60
|
+
return;
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
// If mention autocomplete is visible, Enter accepts the selected (or first) match
|
|
64
|
+
if (mentionRef.current.state.visible) {
|
|
65
|
+
const result = mentionRef.current.actions.onAccept();
|
|
66
|
+
if (result) {
|
|
67
|
+
setValueRef.current(result.text);
|
|
68
|
+
return;
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
const trimmed = submitted.trim();
|
|
72
|
+
if (!trimmed)
|
|
73
|
+
return;
|
|
74
|
+
// Check for slash commands
|
|
75
|
+
if (trimmed.startsWith("/")) {
|
|
76
|
+
const spaceIdx = trimmed.indexOf(" ");
|
|
77
|
+
const command = spaceIdx > 0 ? trimmed.slice(1, spaceIdx) : trimmed.slice(1);
|
|
78
|
+
const args = spaceIdx > 0 ? trimmed.slice(spaceIdx + 1).trim() : "";
|
|
79
|
+
if (onSlashCommand) {
|
|
80
|
+
onSlashCommand(command, args);
|
|
81
|
+
}
|
|
82
|
+
else {
|
|
83
|
+
onSubmit(trimmed);
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
else {
|
|
87
|
+
onSubmit(trimmed);
|
|
88
|
+
}
|
|
89
|
+
// Clear input via remount trick
|
|
90
|
+
setValue("");
|
|
91
|
+
setSubmitKey((k) => k + 1);
|
|
92
|
+
}, [onSubmit, onSlashCommand]);
|
|
93
|
+
useEffect(() => {
|
|
94
|
+
const g = globalThis;
|
|
95
|
+
g.__nexHomeTabComplete = (direction) => {
|
|
96
|
+
const s = slashRef.current;
|
|
97
|
+
const m = mentionRef.current;
|
|
98
|
+
if (s.state.visible) {
|
|
99
|
+
const result = direction > 0 ? s.actions.onTab() : s.actions.onShiftTab();
|
|
100
|
+
if (result)
|
|
101
|
+
setValueRef.current(result.text);
|
|
102
|
+
return true;
|
|
103
|
+
}
|
|
104
|
+
if (m.state.visible) {
|
|
105
|
+
const result = direction > 0 ? m.actions.onTab() : m.actions.onShiftTab();
|
|
106
|
+
if (result)
|
|
107
|
+
setValueRef.current(result.text);
|
|
108
|
+
return true;
|
|
109
|
+
}
|
|
110
|
+
return false;
|
|
111
|
+
};
|
|
112
|
+
return () => { delete g.__nexHomeTabComplete; };
|
|
113
|
+
}, []); // register once, refs keep it current
|
|
114
|
+
return (_jsxs(Box, { flexDirection: "column", children: [_jsx(SlashAutocomplete, { state: slash.state, maxVisible: 6 }), _jsx(MentionAutocomplete, { state: mention.state, maxVisible: 6 }), _jsx(HintBar, { visible: focused }), _jsxs(Box, { borderStyle: "single", borderColor: focused ? "cyan" : "gray", flexDirection: "column", paddingX: 1, children: [_jsxs(Box, { justifyContent: "space-between", children: [focused
|
|
115
|
+
? _jsx(Text, { color: "black", backgroundColor: "cyan", bold: true, children: " COMPOSE " })
|
|
116
|
+
: _jsx(Text, { color: "gray", children: isThread
|
|
117
|
+
? "Reply"
|
|
118
|
+
: channelType === "channel"
|
|
119
|
+
? `Message #${channelName}`
|
|
120
|
+
: `Message ${channelName}` }), focused && _jsx(Text, { color: "gray", children: "@ \u00B7 / \u00B7 Enter=send" })] }), _jsx(Box, { children: focused ? (_jsx(TextInput, { placeholder: placeholder, onChange: handleChange, onSubmit: handleSubmit }, submitKey)) : (_jsx(Text, { dimColor: true, children: placeholder })) }), !focused && (_jsx(Box, { justifyContent: "flex-end", children: _jsx(Text, { dimColor: true, children: "Tab=focus" }) }))] })] }));
|
|
121
|
+
}
|
|
122
|
+
export default ComposeArea;
|
|
123
|
+
//# sourceMappingURL=compose.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"compose.js","sourceRoot":"","sources":["../../../../src/tui/components/slack/compose.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SlackLayout — root 3-panel responsive layout.
|
|
3
|
+
*
|
|
4
|
+
* Arranges sidebar + main panel + optional thread panel side by side.
|
|
5
|
+
* Responsive breakpoints:
|
|
6
|
+
* < 60 cols: main only (no sidebar)
|
|
7
|
+
* 60–79 cols: narrow sidebar (20) + main
|
|
8
|
+
* 80–119 cols: sidebar (24) + main; thread replaces main if open
|
|
9
|
+
* ≥ 120 cols: sidebar (28) + main + thread (45)
|
|
10
|
+
*/
|
|
11
|
+
import React from "react";
|
|
12
|
+
import type { ReactNode } from "react";
|
|
13
|
+
export interface LayoutMetrics {
|
|
14
|
+
sidebarWidth: number;
|
|
15
|
+
showSidebar: boolean;
|
|
16
|
+
mainWidth: number;
|
|
17
|
+
threadWidth: number;
|
|
18
|
+
showThread: boolean;
|
|
19
|
+
/** When true, thread replaces main (narrow terminal + thread open) */
|
|
20
|
+
threadReplacesMain: boolean;
|
|
21
|
+
}
|
|
22
|
+
export declare function computeLayout(cols: number, threadOpen: boolean): LayoutMetrics;
|
|
23
|
+
export type FocusSection = "sidebar" | "messages" | "compose" | "thread";
|
|
24
|
+
export interface SlackLayoutProps {
|
|
25
|
+
cols: number;
|
|
26
|
+
rows: number;
|
|
27
|
+
threadOpen: boolean;
|
|
28
|
+
focusSection: FocusSection;
|
|
29
|
+
sidebar: ReactNode;
|
|
30
|
+
main: ReactNode;
|
|
31
|
+
thread?: ReactNode;
|
|
32
|
+
/** Overlay content (quick switcher) — rendered on top */
|
|
33
|
+
overlay?: ReactNode;
|
|
34
|
+
}
|
|
35
|
+
export declare function SlackLayout({ cols, rows, threadOpen, focusSection, sidebar, main, thread, overlay, }: SlackLayoutProps): React.JSX.Element;
|
|
36
|
+
export default SlackLayout;
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
import { Box, Text } from "ink";
|
|
3
|
+
export function computeLayout(cols, threadOpen) {
|
|
4
|
+
const SIDEBAR_WIDE = 28;
|
|
5
|
+
const SIDEBAR_NORMAL = 24;
|
|
6
|
+
const SIDEBAR_NARROW = 20;
|
|
7
|
+
const THREAD_WIDTH = 45;
|
|
8
|
+
const BORDER = 1; // right border on sidebar
|
|
9
|
+
if (cols < 60) {
|
|
10
|
+
return {
|
|
11
|
+
sidebarWidth: 0,
|
|
12
|
+
showSidebar: false,
|
|
13
|
+
mainWidth: cols,
|
|
14
|
+
threadWidth: 0,
|
|
15
|
+
showThread: false,
|
|
16
|
+
threadReplacesMain: false,
|
|
17
|
+
};
|
|
18
|
+
}
|
|
19
|
+
if (cols < 80) {
|
|
20
|
+
if (threadOpen) {
|
|
21
|
+
// Thread replaces main at narrow widths
|
|
22
|
+
return {
|
|
23
|
+
sidebarWidth: SIDEBAR_NARROW,
|
|
24
|
+
showSidebar: true,
|
|
25
|
+
mainWidth: 0,
|
|
26
|
+
threadWidth: cols - SIDEBAR_NARROW - BORDER,
|
|
27
|
+
showThread: true,
|
|
28
|
+
threadReplacesMain: true,
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
return {
|
|
32
|
+
sidebarWidth: SIDEBAR_NARROW,
|
|
33
|
+
showSidebar: true,
|
|
34
|
+
mainWidth: cols - SIDEBAR_NARROW - BORDER,
|
|
35
|
+
threadWidth: 0,
|
|
36
|
+
showThread: false,
|
|
37
|
+
threadReplacesMain: false,
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
if (cols < 120) {
|
|
41
|
+
if (threadOpen) {
|
|
42
|
+
// Thread replaces main at medium widths
|
|
43
|
+
return {
|
|
44
|
+
sidebarWidth: SIDEBAR_NORMAL,
|
|
45
|
+
showSidebar: true,
|
|
46
|
+
mainWidth: 0,
|
|
47
|
+
threadWidth: cols - SIDEBAR_NORMAL - BORDER,
|
|
48
|
+
showThread: true,
|
|
49
|
+
threadReplacesMain: true,
|
|
50
|
+
};
|
|
51
|
+
}
|
|
52
|
+
return {
|
|
53
|
+
sidebarWidth: SIDEBAR_NORMAL,
|
|
54
|
+
showSidebar: true,
|
|
55
|
+
mainWidth: cols - SIDEBAR_NORMAL - BORDER,
|
|
56
|
+
threadWidth: 0,
|
|
57
|
+
showThread: false,
|
|
58
|
+
threadReplacesMain: false,
|
|
59
|
+
};
|
|
60
|
+
}
|
|
61
|
+
// Wide: full 3-panel layout
|
|
62
|
+
if (threadOpen) {
|
|
63
|
+
const mainW = cols - SIDEBAR_WIDE - THREAD_WIDTH - BORDER - 1;
|
|
64
|
+
return {
|
|
65
|
+
sidebarWidth: SIDEBAR_WIDE,
|
|
66
|
+
showSidebar: true,
|
|
67
|
+
mainWidth: Math.max(mainW, 30),
|
|
68
|
+
threadWidth: THREAD_WIDTH,
|
|
69
|
+
showThread: true,
|
|
70
|
+
threadReplacesMain: false,
|
|
71
|
+
};
|
|
72
|
+
}
|
|
73
|
+
return {
|
|
74
|
+
sidebarWidth: SIDEBAR_WIDE,
|
|
75
|
+
showSidebar: true,
|
|
76
|
+
mainWidth: cols - SIDEBAR_WIDE - BORDER,
|
|
77
|
+
threadWidth: 0,
|
|
78
|
+
showThread: false,
|
|
79
|
+
threadReplacesMain: false,
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
/**
|
|
83
|
+
* FocusBadge — small colored label showing which section has focus.
|
|
84
|
+
* Renders at the top of each panel, only visible when that panel is focused.
|
|
85
|
+
*/
|
|
86
|
+
function FocusBadge({ label, visible }) {
|
|
87
|
+
if (!visible)
|
|
88
|
+
return null;
|
|
89
|
+
return (_jsx(Box, { children: _jsx(Text, { color: "black", backgroundColor: "cyan", bold: true, children: ` ${label} ` }) }));
|
|
90
|
+
}
|
|
91
|
+
export function SlackLayout({ cols, rows, threadOpen, focusSection, sidebar, main, thread, overlay, }) {
|
|
92
|
+
const layout = computeLayout(cols, threadOpen);
|
|
93
|
+
return (_jsxs(Box, { flexDirection: "column", width: cols, height: rows, children: [_jsxs(Box, { flexDirection: "row", flexGrow: 1, children: [layout.showSidebar && (_jsxs(Box, { width: layout.sidebarWidth, flexDirection: "column", flexShrink: 0, borderStyle: "single", borderLeft: false, borderTop: false, borderBottom: false, borderRight: true, borderColor: focusSection === "sidebar" ? "cyan" : "gray", children: [_jsx(FocusBadge, { label: "SIDEBAR", visible: focusSection === "sidebar" }), sidebar] })), layout.mainWidth > 0 && !layout.threadReplacesMain && (_jsx(Box, { width: layout.mainWidth, flexDirection: "column", flexGrow: 1, children: main })), layout.showThread && thread && (_jsxs(Box, { width: layout.threadWidth, flexDirection: "column", flexShrink: 0, borderStyle: "single", borderRight: false, borderTop: false, borderBottom: false, borderLeft: true, borderColor: focusSection === "thread" ? "cyan" : "gray", children: [_jsx(FocusBadge, { label: "THREAD", visible: focusSection === "thread" }), thread] }))] }), overlay] }));
|
|
94
|
+
}
|
|
95
|
+
export default SlackLayout;
|
|
96
|
+
//# sourceMappingURL=layout.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"layout.js","sourceRoot":"","sources":["../../../../src/tui/components/slack/layout.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Slack-style message rendering components.
|
|
3
|
+
*
|
|
4
|
+
* Includes message grouping logic, date separators, unread markers,
|
|
5
|
+
* thread indicators, system messages, and the scrollable message list.
|
|
6
|
+
*/
|
|
7
|
+
import React from "react";
|
|
8
|
+
export interface ChatMessageInput {
|
|
9
|
+
id: string;
|
|
10
|
+
sender: string;
|
|
11
|
+
senderType: "agent" | "human" | "system";
|
|
12
|
+
content: string;
|
|
13
|
+
timestamp: number;
|
|
14
|
+
threadReplyCount?: number;
|
|
15
|
+
threadParticipants?: string[];
|
|
16
|
+
threadLastReply?: number;
|
|
17
|
+
reactions?: ReactionData[];
|
|
18
|
+
edited?: boolean;
|
|
19
|
+
isError?: boolean;
|
|
20
|
+
}
|
|
21
|
+
export interface ReactionData {
|
|
22
|
+
emoji: string;
|
|
23
|
+
count: number;
|
|
24
|
+
reacted: boolean;
|
|
25
|
+
}
|
|
26
|
+
export interface GroupedMessage {
|
|
27
|
+
id: string;
|
|
28
|
+
sender: string;
|
|
29
|
+
senderType: "agent" | "human" | "system";
|
|
30
|
+
initials: string;
|
|
31
|
+
content: string;
|
|
32
|
+
timestamp: number;
|
|
33
|
+
isFirstInGroup: boolean;
|
|
34
|
+
threadReplyCount?: number;
|
|
35
|
+
threadParticipants?: string[];
|
|
36
|
+
threadLastReply?: number;
|
|
37
|
+
reactions?: ReactionData[];
|
|
38
|
+
edited?: boolean;
|
|
39
|
+
isSystem?: boolean;
|
|
40
|
+
dateSeparator?: string;
|
|
41
|
+
isUnreadMarker?: boolean;
|
|
42
|
+
isError?: boolean;
|
|
43
|
+
}
|
|
44
|
+
export declare function groupMessages(messages: ChatMessageInput[], unreadAfterTimestamp?: number): GroupedMessage[];
|
|
45
|
+
export interface DateSeparatorProps {
|
|
46
|
+
label: string;
|
|
47
|
+
width?: number;
|
|
48
|
+
}
|
|
49
|
+
export declare function DateSeparator({ label, width, }: DateSeparatorProps): React.JSX.Element;
|
|
50
|
+
export interface UnreadSeparatorProps {
|
|
51
|
+
width?: number;
|
|
52
|
+
}
|
|
53
|
+
export declare function UnreadSeparator({ width, }: UnreadSeparatorProps): React.JSX.Element;
|
|
54
|
+
export interface SystemMessageProps {
|
|
55
|
+
content: string;
|
|
56
|
+
timestamp: number;
|
|
57
|
+
}
|
|
58
|
+
export declare function SystemMessage({ content, timestamp, }: SystemMessageProps): React.JSX.Element;
|
|
59
|
+
export interface ThreadIndicatorProps {
|
|
60
|
+
replyCount: number;
|
|
61
|
+
lastReplyTimestamp: number;
|
|
62
|
+
onClick?: () => void;
|
|
63
|
+
}
|
|
64
|
+
export declare function ThreadIndicator({ replyCount, lastReplyTimestamp, onClick, }: ThreadIndicatorProps): React.JSX.Element;
|
|
65
|
+
export interface SlackMessageListProps {
|
|
66
|
+
messages: ChatMessageInput[];
|
|
67
|
+
unreadAfterTimestamp?: number;
|
|
68
|
+
onThreadOpen?: (messageId: string) => void;
|
|
69
|
+
width?: number;
|
|
70
|
+
}
|
|
71
|
+
export declare function SlackMessageList({ messages, unreadAfterTimestamp, onThreadOpen, width, }: SlackMessageListProps): React.JSX.Element;
|
|
72
|
+
export default SlackMessageList;
|
|
@@ -0,0 +1,177 @@
|
|
|
1
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
2
|
+
/**
|
|
3
|
+
* Slack-style message rendering components.
|
|
4
|
+
*
|
|
5
|
+
* Includes message grouping logic, date separators, unread markers,
|
|
6
|
+
* thread indicators, system messages, and the scrollable message list.
|
|
7
|
+
*/
|
|
8
|
+
import { useMemo } from "react";
|
|
9
|
+
import { Box, Text } from "ink";
|
|
10
|
+
import { getAgentColor } from "../../agent-colors.js";
|
|
11
|
+
import { Markdown } from "../markdown.js";
|
|
12
|
+
// ── Time Formatting ─────────────────────────────────────────────────
|
|
13
|
+
function formatTime(timestamp) {
|
|
14
|
+
const d = new Date(timestamp);
|
|
15
|
+
const hours = d.getHours();
|
|
16
|
+
const minutes = d.getMinutes().toString().padStart(2, "0");
|
|
17
|
+
const ampm = hours >= 12 ? "PM" : "AM";
|
|
18
|
+
const h = hours % 12 || 12;
|
|
19
|
+
return `${h}:${minutes} ${ampm}`;
|
|
20
|
+
}
|
|
21
|
+
function formatTimeWithDate(timestamp) {
|
|
22
|
+
const d = new Date(timestamp);
|
|
23
|
+
const now = new Date();
|
|
24
|
+
const isToday = d.getFullYear() === now.getFullYear() &&
|
|
25
|
+
d.getMonth() === now.getMonth() &&
|
|
26
|
+
d.getDate() === now.getDate();
|
|
27
|
+
if (isToday)
|
|
28
|
+
return formatTime(timestamp);
|
|
29
|
+
const month = d.toLocaleDateString("en-US", { month: "short" });
|
|
30
|
+
const day = d.getDate();
|
|
31
|
+
return `${month} ${day}, ${formatTime(timestamp)}`;
|
|
32
|
+
}
|
|
33
|
+
function getDateLabel(timestamp) {
|
|
34
|
+
const d = new Date(timestamp);
|
|
35
|
+
const now = new Date();
|
|
36
|
+
const isToday = d.getFullYear() === now.getFullYear() &&
|
|
37
|
+
d.getMonth() === now.getMonth() &&
|
|
38
|
+
d.getDate() === now.getDate();
|
|
39
|
+
if (isToday)
|
|
40
|
+
return "Today";
|
|
41
|
+
const yesterday = new Date(now);
|
|
42
|
+
yesterday.setDate(yesterday.getDate() - 1);
|
|
43
|
+
const isYesterday = d.getFullYear() === yesterday.getFullYear() &&
|
|
44
|
+
d.getMonth() === yesterday.getMonth() &&
|
|
45
|
+
d.getDate() === yesterday.getDate();
|
|
46
|
+
if (isYesterday)
|
|
47
|
+
return "Yesterday";
|
|
48
|
+
return d.toLocaleDateString("en-US", {
|
|
49
|
+
weekday: "long",
|
|
50
|
+
month: "long",
|
|
51
|
+
day: "numeric",
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
// ── groupMessages ───────────────────────────────────────────────────
|
|
55
|
+
const GROUP_WINDOW_MS = 5 * 60 * 1000; // 5 minutes
|
|
56
|
+
export function groupMessages(messages, unreadAfterTimestamp) {
|
|
57
|
+
const result = [];
|
|
58
|
+
let lastSender = "";
|
|
59
|
+
let lastTimestamp = 0;
|
|
60
|
+
let lastDateKey = "";
|
|
61
|
+
let unreadMarkerInserted = false;
|
|
62
|
+
for (const msg of messages) {
|
|
63
|
+
const msgDate = new Date(msg.timestamp);
|
|
64
|
+
const dateKey = `${msgDate.getFullYear()}-${msgDate.getMonth()}-${msgDate.getDate()}`;
|
|
65
|
+
// Date separator
|
|
66
|
+
let dateSeparator;
|
|
67
|
+
if (dateKey !== lastDateKey) {
|
|
68
|
+
dateSeparator = getDateLabel(msg.timestamp);
|
|
69
|
+
lastDateKey = dateKey;
|
|
70
|
+
}
|
|
71
|
+
// Unread marker
|
|
72
|
+
let isUnreadMarker = false;
|
|
73
|
+
if (!unreadMarkerInserted &&
|
|
74
|
+
unreadAfterTimestamp !== undefined &&
|
|
75
|
+
msg.timestamp > unreadAfterTimestamp) {
|
|
76
|
+
isUnreadMarker = true;
|
|
77
|
+
unreadMarkerInserted = true;
|
|
78
|
+
}
|
|
79
|
+
// Grouping: same sender within 5 min = continuation
|
|
80
|
+
const isFirstInGroup = msg.sender !== lastSender ||
|
|
81
|
+
msg.timestamp - lastTimestamp > GROUP_WINDOW_MS ||
|
|
82
|
+
dateSeparator !== undefined ||
|
|
83
|
+
msg.senderType === "system";
|
|
84
|
+
// Compute initials
|
|
85
|
+
const initials = msg.sender
|
|
86
|
+
.split(/[\s-]+/)
|
|
87
|
+
.slice(0, 2)
|
|
88
|
+
.map((w) => w[0]?.toUpperCase() ?? "")
|
|
89
|
+
.join("");
|
|
90
|
+
result.push({
|
|
91
|
+
id: msg.id,
|
|
92
|
+
sender: msg.sender,
|
|
93
|
+
senderType: msg.senderType,
|
|
94
|
+
initials,
|
|
95
|
+
content: msg.content,
|
|
96
|
+
timestamp: msg.timestamp,
|
|
97
|
+
isFirstInGroup,
|
|
98
|
+
isSystem: msg.senderType === "system",
|
|
99
|
+
dateSeparator,
|
|
100
|
+
isUnreadMarker,
|
|
101
|
+
threadReplyCount: msg.threadReplyCount,
|
|
102
|
+
threadParticipants: msg.threadParticipants,
|
|
103
|
+
threadLastReply: msg.threadLastReply,
|
|
104
|
+
reactions: msg.reactions,
|
|
105
|
+
edited: msg.edited,
|
|
106
|
+
isError: msg.isError,
|
|
107
|
+
});
|
|
108
|
+
lastSender = msg.sender;
|
|
109
|
+
lastTimestamp = msg.timestamp;
|
|
110
|
+
}
|
|
111
|
+
return result;
|
|
112
|
+
}
|
|
113
|
+
export function DateSeparator({ label, width = 60, }) {
|
|
114
|
+
const labelWithPad = ` ${label} `;
|
|
115
|
+
const remaining = Math.max(0, width - labelWithPad.length);
|
|
116
|
+
const left = Math.floor(remaining / 2);
|
|
117
|
+
const right = remaining - left;
|
|
118
|
+
const line = "─".repeat(left) + labelWithPad + "─".repeat(right);
|
|
119
|
+
return (_jsx(Box, { justifyContent: "center", marginY: 0, children: _jsx(Text, { color: "gray", children: line }) }));
|
|
120
|
+
}
|
|
121
|
+
export function UnreadSeparator({ width = 60, }) {
|
|
122
|
+
const labelWithPad = " New ";
|
|
123
|
+
const remaining = Math.max(0, width - labelWithPad.length);
|
|
124
|
+
const left = Math.floor(remaining / 2);
|
|
125
|
+
const right = remaining - left;
|
|
126
|
+
const line = "─".repeat(left) + labelWithPad + "─".repeat(right);
|
|
127
|
+
return (_jsx(Box, { justifyContent: "center", marginY: 0, children: _jsx(Text, { color: "red", bold: true, children: line }) }));
|
|
128
|
+
}
|
|
129
|
+
export function SystemMessage({ content, timestamp, }) {
|
|
130
|
+
return (_jsx(Box, { justifyContent: "center", paddingX: 4, children: _jsxs(Text, { color: "gray", dimColor: true, italic: true, children: ["✦ ", content] }) }));
|
|
131
|
+
}
|
|
132
|
+
export function ThreadIndicator({ replyCount, lastReplyTimestamp, onClick, }) {
|
|
133
|
+
const replyText = replyCount === 1 ? "1 reply" : `${replyCount} replies`;
|
|
134
|
+
const lastReply = `Last reply ${formatTimeWithDate(lastReplyTimestamp)}`;
|
|
135
|
+
return (_jsx(Box, { paddingLeft: 6, children: _jsxs(Text, { color: "cyan", children: ["↳ ", replyText, " ", lastReply] }) }));
|
|
136
|
+
}
|
|
137
|
+
function FirstMessage({ message, onThreadOpen, }) {
|
|
138
|
+
const isHuman = message.senderType === "human";
|
|
139
|
+
const avatarChar = isHuman ? ">" : message.initials.charAt(0) || "?";
|
|
140
|
+
const nameColor = isHuman ? "white" : getAgentColor(message.sender);
|
|
141
|
+
const time = formatTimeWithDate(message.timestamp);
|
|
142
|
+
return (_jsxs(Box, { flexDirection: "column", children: [_jsxs(Box, { children: [_jsxs(Text, { color: nameColor, bold: true, children: ["[", avatarChar, "] "] }), _jsx(Text, { color: nameColor, bold: true, children: message.sender }), _jsxs(Text, { color: "gray", dimColor: true, children: [" ", time] })] }), _jsx(Box, { paddingLeft: 4, children: _jsx(Markdown, { content: message.content }) }), message.edited && (_jsx(Box, { paddingLeft: 4, children: _jsx(Text, { color: "gray", dimColor: true, children: "(edited)" }) })), message.threadReplyCount != null &&
|
|
143
|
+
message.threadReplyCount > 0 &&
|
|
144
|
+
message.threadLastReply != null && (_jsx(ThreadIndicator, { replyCount: message.threadReplyCount, lastReplyTimestamp: message.threadLastReply, onClick: onThreadOpen ? () => onThreadOpen(message.id) : undefined }))] }));
|
|
145
|
+
}
|
|
146
|
+
function ContinuationMessage({ message, onThreadOpen, }) {
|
|
147
|
+
return (_jsxs(Box, { flexDirection: "column", children: [_jsx(Box, { paddingLeft: 4, children: _jsx(Markdown, { content: message.content }) }), message.edited && (_jsx(Box, { paddingLeft: 4, children: _jsx(Text, { color: "gray", dimColor: true, children: "(edited)" }) })), message.threadReplyCount != null &&
|
|
148
|
+
message.threadReplyCount > 0 &&
|
|
149
|
+
message.threadLastReply != null && (_jsx(ThreadIndicator, { replyCount: message.threadReplyCount, lastReplyTimestamp: message.threadLastReply, onClick: onThreadOpen ? () => onThreadOpen(message.id) : undefined }))] }));
|
|
150
|
+
}
|
|
151
|
+
export function SlackMessageList({ messages, unreadAfterTimestamp, onThreadOpen, width = 60, }) {
|
|
152
|
+
const grouped = useMemo(() => groupMessages(messages, unreadAfterTimestamp), [messages, unreadAfterTimestamp]);
|
|
153
|
+
return (_jsx(Box, { flexDirection: "column", flexGrow: 1, children: grouped.map((msg) => {
|
|
154
|
+
const elements = [];
|
|
155
|
+
// Date separator before this message
|
|
156
|
+
if (msg.dateSeparator) {
|
|
157
|
+
elements.push(_jsx(DateSeparator, { label: msg.dateSeparator, width: width }, `date-${msg.id}`));
|
|
158
|
+
}
|
|
159
|
+
// Unread marker before this message
|
|
160
|
+
if (msg.isUnreadMarker) {
|
|
161
|
+
elements.push(_jsx(UnreadSeparator, { width: width }, `unread-${msg.id}`));
|
|
162
|
+
}
|
|
163
|
+
// The message itself
|
|
164
|
+
if (msg.isSystem) {
|
|
165
|
+
elements.push(_jsx(SystemMessage, { content: msg.content, timestamp: msg.timestamp }, msg.id));
|
|
166
|
+
}
|
|
167
|
+
else if (msg.isFirstInGroup) {
|
|
168
|
+
elements.push(_jsx(FirstMessage, { message: msg, onThreadOpen: onThreadOpen }, msg.id));
|
|
169
|
+
}
|
|
170
|
+
else {
|
|
171
|
+
elements.push(_jsx(ContinuationMessage, { message: msg, onThreadOpen: onThreadOpen }, msg.id));
|
|
172
|
+
}
|
|
173
|
+
return _jsx(Box, { flexDirection: "column", children: elements }, `row-${msg.id}`);
|
|
174
|
+
}) }));
|
|
175
|
+
}
|
|
176
|
+
export default SlackMessageList;
|
|
177
|
+
//# sourceMappingURL=messages.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"messages.js","sourceRoot":"","sources":["../../../../src/tui/components/slack/messages.tsx"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Quick Switcher (Ctrl+K) modal overlay.
|
|
3
|
+
*
|
|
4
|
+
* Provides fuzzy search across channels and DMs with frecency-based
|
|
5
|
+
* ranking. Renders as a centered overlay capturing all keyboard input.
|
|
6
|
+
*/
|
|
7
|
+
import React from "react";
|
|
8
|
+
export interface QuickSwitcherItem {
|
|
9
|
+
id: string;
|
|
10
|
+
name: string;
|
|
11
|
+
type: "channel" | "dm" | "group-dm";
|
|
12
|
+
online?: boolean;
|
|
13
|
+
unread: number;
|
|
14
|
+
/** Frecency score for ranking (higher = more recent/frequent). */
|
|
15
|
+
score: number;
|
|
16
|
+
}
|
|
17
|
+
export interface QuickSwitcherProps {
|
|
18
|
+
open: boolean;
|
|
19
|
+
items: QuickSwitcherItem[];
|
|
20
|
+
onSelect: (id: string) => void;
|
|
21
|
+
onClose: () => void;
|
|
22
|
+
}
|
|
23
|
+
/**
|
|
24
|
+
* Case-insensitive substring match on name.
|
|
25
|
+
* Matches partial words and hyphen/underscore-separated tokens.
|
|
26
|
+
*/
|
|
27
|
+
export declare function fuzzyMatch(query: string, name: string): boolean;
|
|
28
|
+
/**
|
|
29
|
+
* Filter and sort items by fuzzy match + frecency score.
|
|
30
|
+
*/
|
|
31
|
+
export declare function filterItems(query: string, items: QuickSwitcherItem[]): QuickSwitcherItem[];
|
|
32
|
+
export declare function QuickSwitcher({ open, items, onSelect, onClose, }: QuickSwitcherProps): React.JSX.Element | null;
|
|
33
|
+
export default QuickSwitcher;
|