@stigmer/react 0.0.36

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 (190) hide show
  1. package/LICENSE +190 -0
  2. package/README.md +248 -0
  3. package/agent/components/AgentCard.d.ts +9 -0
  4. package/agent/components/AgentCard.d.ts.map +1 -0
  5. package/agent/components/AgentCard.js +26 -0
  6. package/agent/components/AgentCard.js.map +1 -0
  7. package/agent/components/AgentOverview.d.ts +7 -0
  8. package/agent/components/AgentOverview.d.ts.map +1 -0
  9. package/agent/components/AgentOverview.js +36 -0
  10. package/agent/components/AgentOverview.js.map +1 -0
  11. package/agent/components/AgentPicker.d.ts +17 -0
  12. package/agent/components/AgentPicker.d.ts.map +1 -0
  13. package/agent/components/AgentPicker.js +86 -0
  14. package/agent/components/AgentPicker.js.map +1 -0
  15. package/agent/hooks/useAgentSearch.d.ts +28 -0
  16. package/agent/hooks/useAgentSearch.d.ts.map +1 -0
  17. package/agent/hooks/useAgentSearch.js +63 -0
  18. package/agent/hooks/useAgentSearch.js.map +1 -0
  19. package/agent/index.d.ts +9 -0
  20. package/agent/index.d.ts.map +1 -0
  21. package/agent/index.js +7 -0
  22. package/agent/index.js.map +1 -0
  23. package/agent-execution/components/ApprovalControls.d.ts +10 -0
  24. package/agent-execution/components/ApprovalControls.d.ts.map +1 -0
  25. package/agent-execution/components/ApprovalControls.js +19 -0
  26. package/agent-execution/components/ApprovalControls.js.map +1 -0
  27. package/agent-execution/components/ExecutionStatus.d.ts +8 -0
  28. package/agent-execution/components/ExecutionStatus.d.ts.map +1 -0
  29. package/agent-execution/components/ExecutionStatus.js +14 -0
  30. package/agent-execution/components/ExecutionStatus.js.map +1 -0
  31. package/agent-execution/components/ExecutionStream.d.ts +16 -0
  32. package/agent-execution/components/ExecutionStream.d.ts.map +1 -0
  33. package/agent-execution/components/ExecutionStream.js +39 -0
  34. package/agent-execution/components/ExecutionStream.js.map +1 -0
  35. package/agent-execution/components/MessageEntry.d.ts +17 -0
  36. package/agent-execution/components/MessageEntry.d.ts.map +1 -0
  37. package/agent-execution/components/MessageEntry.js +36 -0
  38. package/agent-execution/components/MessageEntry.js.map +1 -0
  39. package/agent-execution/components/MessageInput.d.ts +10 -0
  40. package/agent-execution/components/MessageInput.d.ts.map +1 -0
  41. package/agent-execution/components/MessageInput.js +27 -0
  42. package/agent-execution/components/MessageInput.js.map +1 -0
  43. package/agent-execution/components/OutputBlock.d.ts +9 -0
  44. package/agent-execution/components/OutputBlock.d.ts.map +1 -0
  45. package/agent-execution/components/OutputBlock.js +15 -0
  46. package/agent-execution/components/OutputBlock.js.map +1 -0
  47. package/agent-execution/components/SubAgentCard.d.ts +11 -0
  48. package/agent-execution/components/SubAgentCard.d.ts.map +1 -0
  49. package/agent-execution/components/SubAgentCard.js +19 -0
  50. package/agent-execution/components/SubAgentCard.js.map +1 -0
  51. package/agent-execution/components/ToolCallCard.d.ts +11 -0
  52. package/agent-execution/components/ToolCallCard.d.ts.map +1 -0
  53. package/agent-execution/components/ToolCallCard.js +25 -0
  54. package/agent-execution/components/ToolCallCard.js.map +1 -0
  55. package/agent-execution/helpers.d.ts +35 -0
  56. package/agent-execution/helpers.d.ts.map +1 -0
  57. package/agent-execution/helpers.js +157 -0
  58. package/agent-execution/helpers.js.map +1 -0
  59. package/agent-execution/hooks/useAgentExecution.d.ts +21 -0
  60. package/agent-execution/hooks/useAgentExecution.d.ts.map +1 -0
  61. package/agent-execution/hooks/useAgentExecution.js +99 -0
  62. package/agent-execution/hooks/useAgentExecution.js.map +1 -0
  63. package/agent-execution/hooks/useApproval.d.ts +12 -0
  64. package/agent-execution/hooks/useApproval.d.ts.map +1 -0
  65. package/agent-execution/hooks/useApproval.js +32 -0
  66. package/agent-execution/hooks/useApproval.js.map +1 -0
  67. package/agent-execution/index.d.ts +14 -0
  68. package/agent-execution/index.d.ts.map +1 -0
  69. package/agent-execution/index.js +15 -0
  70. package/agent-execution/index.js.map +1 -0
  71. package/catalog/components/ResourceSearchCard.d.ts +23 -0
  72. package/catalog/components/ResourceSearchCard.d.ts.map +1 -0
  73. package/catalog/components/ResourceSearchCard.js +36 -0
  74. package/catalog/components/ResourceSearchCard.js.map +1 -0
  75. package/catalog/index.d.ts +4 -0
  76. package/catalog/index.d.ts.map +1 -0
  77. package/catalog/index.js +5 -0
  78. package/catalog/index.js.map +1 -0
  79. package/catalog/internal/time.d.ts +13 -0
  80. package/catalog/internal/time.d.ts.map +1 -0
  81. package/catalog/internal/time.js +41 -0
  82. package/catalog/internal/time.js.map +1 -0
  83. package/context.d.ts +12 -0
  84. package/context.d.ts.map +1 -0
  85. package/context.js +13 -0
  86. package/context.js.map +1 -0
  87. package/hooks.d.ts +19 -0
  88. package/hooks.d.ts.map +1 -0
  89. package/hooks.js +28 -0
  90. package/hooks.js.map +1 -0
  91. package/index.d.ts +4 -0
  92. package/index.d.ts.map +1 -0
  93. package/index.js +6 -0
  94. package/index.js.map +1 -0
  95. package/internal/badge.d.ts +8 -0
  96. package/internal/badge.d.ts.map +1 -0
  97. package/internal/badge.js +34 -0
  98. package/internal/badge.js.map +1 -0
  99. package/internal/button.d.ts +9 -0
  100. package/internal/button.d.ts.map +1 -0
  101. package/internal/button.js +36 -0
  102. package/internal/button.js.map +1 -0
  103. package/internal/collapsible.d.ts +6 -0
  104. package/internal/collapsible.d.ts.map +1 -0
  105. package/internal/collapsible.js +14 -0
  106. package/internal/collapsible.js.map +1 -0
  107. package/internal/section.d.ts +8 -0
  108. package/internal/section.d.ts.map +1 -0
  109. package/internal/section.js +6 -0
  110. package/internal/section.js.map +1 -0
  111. package/internal/textarea.d.ts +4 -0
  112. package/internal/textarea.d.ts.map +1 -0
  113. package/internal/textarea.js +9 -0
  114. package/internal/textarea.js.map +1 -0
  115. package/mcp-server/hooks/useMcpServerSearch.d.ts +25 -0
  116. package/mcp-server/hooks/useMcpServerSearch.d.ts.map +1 -0
  117. package/mcp-server/hooks/useMcpServerSearch.js +57 -0
  118. package/mcp-server/hooks/useMcpServerSearch.js.map +1 -0
  119. package/mcp-server/index.d.ts +3 -0
  120. package/mcp-server/index.d.ts.map +1 -0
  121. package/mcp-server/index.js +3 -0
  122. package/mcp-server/index.js.map +1 -0
  123. package/package.json +75 -0
  124. package/provider.d.ts +55 -0
  125. package/provider.d.ts.map +1 -0
  126. package/provider.js +34 -0
  127. package/provider.js.map +1 -0
  128. package/session/components/AgentSessionHistory.d.ts +8 -0
  129. package/session/components/AgentSessionHistory.d.ts.map +1 -0
  130. package/session/components/AgentSessionHistory.js +11 -0
  131. package/session/components/AgentSessionHistory.js.map +1 -0
  132. package/session/components/SessionCard.d.ts +8 -0
  133. package/session/components/SessionCard.d.ts.map +1 -0
  134. package/session/components/SessionCard.js +57 -0
  135. package/session/components/SessionCard.js.map +1 -0
  136. package/session/hooks/useAgentSessionList.d.ts +21 -0
  137. package/session/hooks/useAgentSessionList.d.ts.map +1 -0
  138. package/session/hooks/useAgentSessionList.js +90 -0
  139. package/session/hooks/useAgentSessionList.js.map +1 -0
  140. package/session/index.d.ts +7 -0
  141. package/session/index.d.ts.map +1 -0
  142. package/session/index.js +6 -0
  143. package/session/index.js.map +1 -0
  144. package/skill/hooks/useSkillSearch.d.ts +25 -0
  145. package/skill/hooks/useSkillSearch.d.ts.map +1 -0
  146. package/skill/hooks/useSkillSearch.js +57 -0
  147. package/skill/hooks/useSkillSearch.js.map +1 -0
  148. package/skill/index.d.ts +3 -0
  149. package/skill/index.d.ts.map +1 -0
  150. package/skill/index.js +3 -0
  151. package/skill/index.js.map +1 -0
  152. package/src/agent/components/AgentCard.tsx +125 -0
  153. package/src/agent/components/AgentOverview.tsx +209 -0
  154. package/src/agent/components/AgentPicker.tsx +255 -0
  155. package/src/agent/hooks/useAgentSearch.ts +94 -0
  156. package/src/agent/index.ts +17 -0
  157. package/src/agent-execution/components/ApprovalControls.tsx +99 -0
  158. package/src/agent-execution/components/ExecutionStatus.tsx +33 -0
  159. package/src/agent-execution/components/ExecutionStream.tsx +148 -0
  160. package/src/agent-execution/components/MessageEntry.tsx +125 -0
  161. package/src/agent-execution/components/MessageInput.tsx +70 -0
  162. package/src/agent-execution/components/OutputBlock.tsx +43 -0
  163. package/src/agent-execution/components/SubAgentCard.tsx +138 -0
  164. package/src/agent-execution/components/ToolCallCard.tsx +153 -0
  165. package/src/agent-execution/helpers.ts +193 -0
  166. package/src/agent-execution/hooks/useAgentExecution.ts +147 -0
  167. package/src/agent-execution/hooks/useApproval.ts +56 -0
  168. package/src/agent-execution/index.ts +46 -0
  169. package/src/catalog/components/ResourceSearchCard.tsx +137 -0
  170. package/src/catalog/index.ts +6 -0
  171. package/src/catalog/internal/time.ts +40 -0
  172. package/src/context.ts +15 -0
  173. package/src/hooks.ts +32 -0
  174. package/src/index.ts +6 -0
  175. package/src/internal/badge.tsx +52 -0
  176. package/src/internal/button.tsx +60 -0
  177. package/src/internal/collapsible.tsx +21 -0
  178. package/src/internal/section.tsx +18 -0
  179. package/src/internal/textarea.tsx +23 -0
  180. package/src/mcp-server/hooks/useMcpServerSearch.ts +79 -0
  181. package/src/mcp-server/index.ts +6 -0
  182. package/src/provider.tsx +73 -0
  183. package/src/session/components/AgentSessionHistory.tsx +109 -0
  184. package/src/session/components/SessionCard.tsx +113 -0
  185. package/src/session/hooks/useAgentSessionList.ts +117 -0
  186. package/src/session/index.ts +13 -0
  187. package/src/skill/hooks/useSkillSearch.ts +79 -0
  188. package/src/skill/index.ts +6 -0
  189. package/src/styles.css +72 -0
  190. package/styles.css +2 -0
@@ -0,0 +1,109 @@
1
+ "use client";
2
+
3
+ import { History, Loader2, AlertCircle, RotateCw } from "lucide-react";
4
+ import { cn } from "@stigmer/theme";
5
+ import { SessionCard } from "./SessionCard";
6
+ import { useAgentSessionList } from "../hooks/useAgentSessionList";
7
+
8
+ export interface AgentSessionHistoryProps {
9
+ agentId: string;
10
+ onSessionSelect?: (sessionId: string) => void;
11
+ pageSize?: number;
12
+ className?: string;
13
+ }
14
+
15
+ export function AgentSessionHistory({
16
+ agentId,
17
+ onSessionSelect,
18
+ pageSize = 5,
19
+ className,
20
+ }: AgentSessionHistoryProps) {
21
+ const {
22
+ sessions,
23
+ isLoading,
24
+ error,
25
+ hasMore,
26
+ isLoadingMore,
27
+ loadMore,
28
+ retry,
29
+ } = useAgentSessionList(agentId, { pageSize });
30
+
31
+ return (
32
+ <section className={cn("stgm-agent-session-history", className)}>
33
+ <h3 className="text-muted-foreground mb-3 text-sm font-semibold tracking-wider uppercase">
34
+ Recent Sessions
35
+ </h3>
36
+
37
+ {error && (
38
+ <div className="bg-destructive/10 text-destructive flex items-center gap-2 rounded-lg px-3 py-2 text-sm">
39
+ <AlertCircle className="size-4 shrink-0" />
40
+ <span className="min-w-0 flex-1 truncate">
41
+ {error.message || "Something went wrong"}
42
+ </span>
43
+ <button
44
+ type="button"
45
+ onClick={retry}
46
+ className="text-destructive hover:text-destructive/80 inline-flex shrink-0 items-center gap-1 text-xs font-medium transition-colors"
47
+ >
48
+ <RotateCw className="size-3" />
49
+ Retry
50
+ </button>
51
+ </div>
52
+ )}
53
+
54
+ {isLoading && sessions.length === 0 && !error && (
55
+ <div className="space-y-3">
56
+ {Array.from({ length: 3 }).map((_, i) => (
57
+ <div
58
+ key={i}
59
+ className="bg-muted/50 h-[52px] animate-pulse rounded-xl"
60
+ />
61
+ ))}
62
+ </div>
63
+ )}
64
+
65
+ {!isLoading && sessions.length === 0 && !error && (
66
+ <div className="border-border flex flex-col items-center justify-center rounded-lg border border-dashed py-12 text-center">
67
+ <History className="text-muted-foreground/40 mb-3 size-10" />
68
+ <p className="text-sm font-medium">No sessions yet</p>
69
+ <p className="text-muted-foreground mt-1 text-xs">
70
+ Run this agent to create your first session.
71
+ </p>
72
+ </div>
73
+ )}
74
+
75
+ {sessions.length > 0 && (
76
+ <div className="space-y-2">
77
+ {sessions.map((session) => (
78
+ <SessionCard
79
+ key={session.metadata?.id}
80
+ session={session}
81
+ onNavigate={onSessionSelect}
82
+ />
83
+ ))}
84
+ </div>
85
+ )}
86
+
87
+ {hasMore && (
88
+ <div className="flex justify-center pt-3">
89
+ <button
90
+ type="button"
91
+ onClick={loadMore}
92
+ disabled={isLoadingMore}
93
+ className={cn(
94
+ "border-border inline-flex h-8 items-center gap-1.5 rounded-lg border px-3 text-sm font-medium",
95
+ "bg-background hover:bg-accent hover:text-accent-foreground transition-colors",
96
+ "disabled:pointer-events-none disabled:opacity-50",
97
+ "focus-visible:ring-ring focus-visible:ring-2 focus-visible:outline-none",
98
+ )}
99
+ >
100
+ {isLoadingMore && (
101
+ <Loader2 className="size-3.5 animate-spin" />
102
+ )}
103
+ Load more
104
+ </button>
105
+ </div>
106
+ )}
107
+ </section>
108
+ );
109
+ }
@@ -0,0 +1,113 @@
1
+ "use client";
2
+
3
+ import type { Session } from "@stigmer/protos/ai/stigmer/agentic/session/v1/api_pb";
4
+ import type { Timestamp } from "@bufbuild/protobuf/wkt";
5
+ import { MessageSquare, ChevronRight } from "lucide-react";
6
+ import { cn } from "@stigmer/theme";
7
+
8
+ export interface SessionCardProps {
9
+ session: Session;
10
+ onNavigate?: (sessionId: string) => void;
11
+ className?: string;
12
+ }
13
+
14
+ export function SessionCard({ session, onNavigate, className }: SessionCardProps) {
15
+ const id = session.metadata?.id ?? "";
16
+ const displayName =
17
+ session.spec?.subject || session.metadata?.name || "Untitled session";
18
+
19
+ const createdAt = session.status?.audit?.specAudit?.createdAt;
20
+ const updatedAt = session.status?.audit?.specAudit?.updatedAt;
21
+
22
+ const interactive = !!onNavigate;
23
+
24
+ return (
25
+ <div
26
+ role={interactive ? "button" : undefined}
27
+ tabIndex={interactive ? 0 : undefined}
28
+ onClick={interactive ? () => onNavigate(id) : undefined}
29
+ onKeyDown={
30
+ interactive
31
+ ? (e) => {
32
+ if (e.key === "Enter" || e.key === " ") {
33
+ e.preventDefault();
34
+ onNavigate(id);
35
+ }
36
+ }
37
+ : undefined
38
+ }
39
+ className={cn(
40
+ "stgm-session-card",
41
+ "bg-card text-card-foreground ring-foreground/10 flex flex-col gap-3 overflow-hidden rounded-xl py-3 text-sm ring-1",
42
+ interactive && "cursor-pointer hover:bg-accent/50 transition-colors",
43
+ "focus-visible:ring-ring focus-visible:ring-2 focus-visible:outline-none",
44
+ className,
45
+ )}
46
+ >
47
+ <div className="grid auto-rows-min grid-cols-[1fr_auto] items-start gap-1 px-3">
48
+ <div className="flex items-center gap-2 text-sm font-medium leading-snug">
49
+ <MessageSquare className="text-muted-foreground size-4 shrink-0" />
50
+ <span className="truncate">{displayName}</span>
51
+ </div>
52
+
53
+ {interactive && (
54
+ <div className="col-start-2 row-span-2 row-start-1 self-start justify-self-end">
55
+ <ChevronRight className="text-muted-foreground size-4" />
56
+ </div>
57
+ )}
58
+
59
+ <div className="text-muted-foreground col-start-1 text-sm">
60
+ <span className="flex items-center gap-3">
61
+ {createdAt && (
62
+ <time dateTime={toISOString(createdAt)}>
63
+ {formatRelativeTime(createdAt)}
64
+ </time>
65
+ )}
66
+ {updatedAt && createdAt && (
67
+ <span className="text-muted-foreground/60">
68
+ updated {formatRelativeTime(updatedAt)}
69
+ </span>
70
+ )}
71
+ </span>
72
+ </div>
73
+ </div>
74
+ </div>
75
+ );
76
+ }
77
+
78
+ function toDate(ts: Timestamp | undefined): Date | null {
79
+ if (!ts) return null;
80
+ const seconds = Number(ts.seconds);
81
+ if (!seconds && seconds !== 0) return null;
82
+ return new Date(seconds * 1000 + Math.floor(ts.nanos / 1_000_000));
83
+ }
84
+
85
+ function toISOString(ts: Timestamp | undefined): string {
86
+ return toDate(ts)?.toISOString() ?? "";
87
+ }
88
+
89
+ function formatRelativeTime(ts: Timestamp | undefined): string {
90
+ const date = toDate(ts);
91
+ if (!date) return "";
92
+
93
+ const now = Date.now();
94
+ const diffMs = now - date.getTime();
95
+
96
+ if (diffMs < 0) return "just now";
97
+ if (diffMs < 60_000) return "just now";
98
+
99
+ const minutes = Math.floor(diffMs / 60_000);
100
+ if (minutes < 60) return `${minutes}m ago`;
101
+
102
+ const hours = Math.floor(minutes / 60);
103
+ if (hours < 24) return `${hours}h ago`;
104
+
105
+ const days = Math.floor(hours / 24);
106
+ if (days === 1) return "yesterday";
107
+ if (days < 7) return `${days}d ago`;
108
+
109
+ return date.toLocaleDateString(undefined, {
110
+ month: "short",
111
+ day: "numeric",
112
+ });
113
+ }
@@ -0,0 +1,117 @@
1
+ "use client";
2
+
3
+ import { useCallback, useEffect, useRef, useState } from "react";
4
+ import type { Session } from "@stigmer/protos/ai/stigmer/agentic/session/v1/api_pb";
5
+ import { create } from "@bufbuild/protobuf";
6
+ import { ListSessionsByAgentRequestSchema } from "@stigmer/protos/ai/stigmer/agentic/session/v1/io_pb";
7
+ import { useStigmer } from "../../hooks";
8
+
9
+ const DEFAULT_PAGE_SIZE = 5;
10
+
11
+ export interface UseAgentSessionListOptions {
12
+ pageSize?: number;
13
+ }
14
+
15
+ export interface UseAgentSessionListReturn {
16
+ sessions: Session[];
17
+ isLoading: boolean;
18
+ error: Error | null;
19
+ hasMore: boolean;
20
+ isLoadingMore: boolean;
21
+ loadMore: () => void;
22
+ retry: () => void;
23
+ }
24
+
25
+ /**
26
+ * Fetches sessions scoped to a specific agent with "load more" pagination.
27
+ *
28
+ * Returns sessions sorted by creation time descending (most recent first).
29
+ * Automatically resets when `agentId` changes.
30
+ */
31
+ export function useAgentSessionList(
32
+ agentId: string,
33
+ options?: UseAgentSessionListOptions,
34
+ ): UseAgentSessionListReturn {
35
+ const pageSize = options?.pageSize ?? DEFAULT_PAGE_SIZE;
36
+ const stigmer = useStigmer();
37
+
38
+ const [sessions, setSessions] = useState<Session[]>([]);
39
+ const [isLoading, setIsLoading] = useState(false);
40
+ const [isLoadingMore, setIsLoadingMore] = useState(false);
41
+ const [error, setError] = useState<Error | null>(null);
42
+ const [currentPage, setCurrentPage] = useState(0);
43
+ const [totalPages, setTotalPages] = useState(0);
44
+
45
+ const hasMore = currentPage + 1 < totalPages;
46
+ const latestAgentId = useRef(agentId);
47
+
48
+ const fetchPage = useCallback(
49
+ async (page: number, append: boolean) => {
50
+ if (!agentId) return;
51
+
52
+ const isInitial = !append;
53
+ if (isInitial) setIsLoading(true);
54
+ else setIsLoadingMore(true);
55
+ setError(null);
56
+
57
+ try {
58
+ const result = await stigmer.session.listByAgent(
59
+ create(ListSessionsByAgentRequestSchema, {
60
+ agentId,
61
+ pageSize,
62
+ pageToken: page > 0 ? String(page) : "",
63
+ }),
64
+ );
65
+
66
+ if (latestAgentId.current !== agentId) return;
67
+
68
+ setTotalPages(result.totalPages);
69
+ setCurrentPage(page);
70
+
71
+ if (append) {
72
+ setSessions((prev) => [...prev, ...result.entries]);
73
+ } else {
74
+ setSessions(result.entries);
75
+ }
76
+ } catch (err) {
77
+ if (latestAgentId.current !== agentId) return;
78
+ setError(err instanceof Error ? err : new Error("Failed to load sessions"));
79
+ } finally {
80
+ if (latestAgentId.current !== agentId) return;
81
+ setIsLoading(false);
82
+ setIsLoadingMore(false);
83
+ }
84
+ },
85
+ [agentId, pageSize, stigmer],
86
+ );
87
+
88
+ useEffect(() => {
89
+ latestAgentId.current = agentId;
90
+ setSessions([]);
91
+ setCurrentPage(0);
92
+ setTotalPages(0);
93
+ fetchPage(0, false);
94
+ }, [agentId, fetchPage]);
95
+
96
+ const loadMore = useCallback(() => {
97
+ if (!hasMore || isLoadingMore) return;
98
+ fetchPage(currentPage + 1, true);
99
+ }, [hasMore, isLoadingMore, currentPage, fetchPage]);
100
+
101
+ const retry = useCallback(() => {
102
+ setSessions([]);
103
+ setCurrentPage(0);
104
+ setTotalPages(0);
105
+ fetchPage(0, false);
106
+ }, [fetchPage]);
107
+
108
+ return {
109
+ sessions,
110
+ isLoading,
111
+ error,
112
+ hasMore,
113
+ isLoadingMore,
114
+ loadMore,
115
+ retry,
116
+ };
117
+ }
@@ -0,0 +1,13 @@
1
+ // Hooks
2
+ export { useAgentSessionList } from "./hooks/useAgentSessionList";
3
+ export type {
4
+ UseAgentSessionListOptions,
5
+ UseAgentSessionListReturn,
6
+ } from "./hooks/useAgentSessionList";
7
+
8
+ // Components
9
+ export { SessionCard } from "./components/SessionCard";
10
+ export type { SessionCardProps } from "./components/SessionCard";
11
+
12
+ export { AgentSessionHistory } from "./components/AgentSessionHistory";
13
+ export type { AgentSessionHistoryProps } from "./components/AgentSessionHistory";
@@ -0,0 +1,79 @@
1
+ "use client";
2
+
3
+ import { useEffect, useRef, useState } from "react";
4
+ import { useStigmer } from "../../hooks";
5
+ import type { SearchResult } from "@stigmer/protos/ai/stigmer/search/v1/io_pb";
6
+
7
+ const DEBOUNCE_MS = 300;
8
+ const DEFAULT_PAGE_SIZE = 20;
9
+
10
+ export interface UseSkillSearchOptions {
11
+ org: string;
12
+ debounceMs?: number;
13
+ pageSize?: number;
14
+ }
15
+
16
+ export interface UseSkillSearchReturn {
17
+ query: string;
18
+ setQuery: (query: string) => void;
19
+ results: SearchResult[];
20
+ isLoading: boolean;
21
+ error: string | null;
22
+ }
23
+
24
+ /**
25
+ * Debounced skill search scoped to a specific organization.
26
+ *
27
+ * On mount fetches all accessible skills (empty query). Typing fires at most
28
+ * one request per debounce interval. Stale responses from out-of-order
29
+ * requests are discarded.
30
+ *
31
+ * Returns full {@link SearchResult} objects suitable for use with
32
+ * `ResourceSearchCard`.
33
+ */
34
+ export function useSkillSearch(options: UseSkillSearchOptions): UseSkillSearchReturn {
35
+ const { org, debounceMs = DEBOUNCE_MS, pageSize = DEFAULT_PAGE_SIZE } = options;
36
+ const stigmer = useStigmer();
37
+
38
+ const [query, setQuery] = useState("");
39
+ const [results, setResults] = useState<SearchResult[]>([]);
40
+ const [isLoading, setIsLoading] = useState(false);
41
+ const [error, setError] = useState<string | null>(null);
42
+
43
+ const latestRequestId = useRef(0);
44
+ const debouncedQuery = useDebouncedValue(query, debounceMs);
45
+
46
+ useEffect(() => {
47
+ const requestId = ++latestRequestId.current;
48
+ setIsLoading(true);
49
+ setError(null);
50
+
51
+ stigmer.skill
52
+ .list({ query: debouncedQuery, org, page: { num: 1, size: pageSize } })
53
+ .then((response) => {
54
+ if (requestId !== latestRequestId.current) return;
55
+ setResults(response.entries);
56
+ })
57
+ .catch((err: unknown) => {
58
+ if (requestId !== latestRequestId.current) return;
59
+ setError(err instanceof Error ? err.message : "Search failed");
60
+ })
61
+ .finally(() => {
62
+ if (requestId !== latestRequestId.current) return;
63
+ setIsLoading(false);
64
+ });
65
+ }, [debouncedQuery, org, pageSize, stigmer]);
66
+
67
+ return { query, setQuery, results, isLoading, error };
68
+ }
69
+
70
+ function useDebouncedValue<T>(value: T, delayMs: number): T {
71
+ const [debounced, setDebounced] = useState(value);
72
+
73
+ useEffect(() => {
74
+ const timer = setTimeout(() => setDebounced(value), delayMs);
75
+ return () => clearTimeout(timer);
76
+ }, [value, delayMs]);
77
+
78
+ return debounced;
79
+ }
@@ -0,0 +1,6 @@
1
+ // Hooks
2
+ export { useSkillSearch } from "./hooks/useSkillSearch";
3
+ export type {
4
+ UseSkillSearchOptions,
5
+ UseSkillSearchReturn,
6
+ } from "./hooks/useSkillSearch";
package/src/styles.css ADDED
@@ -0,0 +1,72 @@
1
+ @layer stgm;
2
+
3
+ @import "tailwindcss/theme.css" layer(stgm);
4
+ @import "tailwindcss/utilities.css" layer(stgm);
5
+ @import "@stigmer/theme/tokens.css" layer(stgm);
6
+
7
+ @custom-variant dark (&:is(.dark *));
8
+
9
+ @theme inline {
10
+ --radius-sm: calc(var(--stgm-radius) - 4px);
11
+ --radius-md: calc(var(--stgm-radius) - 2px);
12
+ --radius-lg: var(--stgm-radius);
13
+ --radius-xl: calc(var(--stgm-radius) + 4px);
14
+ --color-background: var(--stgm-background);
15
+ --color-foreground: var(--stgm-foreground);
16
+ --color-card: var(--stgm-card);
17
+ --color-card-foreground: var(--stgm-card-foreground);
18
+ --color-popover: var(--stgm-popover);
19
+ --color-popover-foreground: var(--stgm-popover-foreground);
20
+ --color-primary: var(--stgm-primary);
21
+ --color-primary-foreground: var(--stgm-primary-foreground);
22
+ --color-secondary: var(--stgm-secondary);
23
+ --color-secondary-foreground: var(--stgm-secondary-foreground);
24
+ --color-muted: var(--stgm-muted);
25
+ --color-muted-foreground: var(--stgm-muted-foreground);
26
+ --color-accent: var(--stgm-accent);
27
+ --color-accent-foreground: var(--stgm-accent-foreground);
28
+ --color-destructive: var(--stgm-destructive);
29
+ --color-destructive-foreground: var(--stgm-destructive-foreground);
30
+ --color-success: var(--stgm-success);
31
+ --color-success-foreground: var(--stgm-success-foreground);
32
+ --color-warning: var(--stgm-warning);
33
+ --color-warning-foreground: var(--stgm-warning-foreground);
34
+ --color-info: var(--stgm-info);
35
+ --color-info-foreground: var(--stgm-info-foreground);
36
+ --color-border: var(--stgm-border);
37
+ --color-input: var(--stgm-input);
38
+ --color-ring: var(--stgm-ring);
39
+ --color-chart-1: var(--stgm-chart-1);
40
+ --color-chart-2: var(--stgm-chart-2);
41
+ --color-chart-3: var(--stgm-chart-3);
42
+ --color-chart-4: var(--stgm-chart-4);
43
+ --color-chart-5: var(--stgm-chart-5);
44
+ --font-sans: var(--stgm-font-sans);
45
+ --font-mono: var(--stgm-font-mono);
46
+ --shadow-sm: var(--stgm-shadow-sm);
47
+ --shadow-md: var(--stgm-shadow-md);
48
+ --shadow-lg: var(--stgm-shadow-lg);
49
+ --default-transition-duration: var(--stgm-transition-duration);
50
+ --default-transition-timing-function: var(--stgm-transition-timing);
51
+ }
52
+
53
+ @utility z-popover {
54
+ z-index: var(--stgm-z-popover);
55
+ }
56
+
57
+ @layer stgm {
58
+ .stgm {
59
+ line-height: 1.5;
60
+ -webkit-font-smoothing: antialiased;
61
+ -moz-osx-font-smoothing: grayscale;
62
+ }
63
+
64
+ .stgm *,
65
+ .stgm *::before,
66
+ .stgm *::after {
67
+ box-sizing: border-box;
68
+ border-width: 0;
69
+ border-style: solid;
70
+ border-color: var(--stgm-border, currentColor);
71
+ }
72
+ }
package/styles.css ADDED
@@ -0,0 +1,2 @@
1
+ /*! tailwindcss v4.2.1 | MIT License | https://tailwindcss.com */
2
+ @layer properties{@supports (((-webkit-hyphens:none)) and (not (margin-trim:inline))) or ((-moz-orient:inline) and (not (color:rgb(from red r g b)))){*,:before,:after,::backdrop{--tw-translate-x:0;--tw-translate-y:0;--tw-translate-z:0;--tw-space-y-reverse:0;--tw-border-style:solid;--tw-leading:initial;--tw-font-weight:initial;--tw-tracking:initial;--tw-shadow:0 0 #0000;--tw-shadow-color:initial;--tw-shadow-alpha:100%;--tw-inset-shadow:0 0 #0000;--tw-inset-shadow-color:initial;--tw-inset-shadow-alpha:100%;--tw-ring-color:initial;--tw-ring-shadow:0 0 #0000;--tw-inset-ring-color:initial;--tw-inset-ring-shadow:0 0 #0000;--tw-ring-inset:initial;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-offset-shadow:0 0 #0000;--tw-outline-style:solid;--tw-content:""}}}@layer stgm{:root,:host{--spacing:.25rem;--text-xs:.75rem;--text-xs--line-height:calc(1 / .75);--text-sm:.875rem;--text-sm--line-height:calc(1.25 / .875);--text-base:1rem;--text-base--line-height:calc(1.5 / 1);--text-xl:1.25rem;--text-xl--line-height:calc(1.75 / 1.25);--font-weight-medium:500;--font-weight-semibold:600;--tracking-wider:.05em;--leading-snug:1.375;--leading-relaxed:1.625;--radius-md:calc(var(--stgm-radius) - 2px);--radius-4xl:2rem;--animate-spin:spin 1s linear infinite;--animate-pulse:pulse 2s cubic-bezier(.4, 0, .6, 1) infinite}.pointer-events-none{pointer-events:none}.absolute{position:absolute}.relative{position:relative}.start{inset-inline-start:var(--spacing)}.top-1\/2{top:50%}.right-3{right:calc(var(--spacing) * 3)}.right-4{right:calc(var(--spacing) * 4)}.bottom-4{bottom:calc(var(--spacing) * 4)}.left-3{left:calc(var(--spacing) * 3)}.z-10{z-index:10}.z-popover{z-index:var(--stgm-z-popover)}.col-start-1{grid-column-start:1}.col-start-2{grid-column-start:2}.row-span-2{grid-row:span 2/span 2}.row-start-1{grid-row-start:1}.container{width:100%}@media (min-width:40rem){.container{max-width:40rem}}@media (min-width:48rem){.container{max-width:48rem}}@media (min-width:64rem){.container{max-width:64rem}}@media (min-width:80rem){.container{max-width:80rem}}@media (min-width:96rem){.container{max-width:96rem}}.mt-0\.5{margin-top:calc(var(--spacing) * .5)}.mt-1{margin-top:calc(var(--spacing) * 1)}.mt-1\.5{margin-top:calc(var(--spacing) * 1.5)}.mt-2{margin-top:calc(var(--spacing) * 2)}.mt-3{margin-top:calc(var(--spacing) * 3)}.mr-0\.5{margin-right:calc(var(--spacing) * .5)}.mb-1{margin-bottom:calc(var(--spacing) * 1)}.mb-3{margin-bottom:calc(var(--spacing) * 3)}.ml-0\.5{margin-left:calc(var(--spacing) * .5)}.ml-auto{margin-left:auto}.line-clamp-1{-webkit-line-clamp:1;-webkit-box-orient:vertical;display:-webkit-box;overflow:hidden}.line-clamp-2{-webkit-line-clamp:2;-webkit-box-orient:vertical;display:-webkit-box;overflow:hidden}.flex{display:flex}.grid{display:grid}.inline-block{display:inline-block}.inline-flex{display:inline-flex}.field-sizing-content{field-sizing:content}.size-2\.5{width:calc(var(--spacing) * 2.5);height:calc(var(--spacing) * 2.5)}.size-3{width:calc(var(--spacing) * 3);height:calc(var(--spacing) * 3)}.size-3\.5{width:calc(var(--spacing) * 3.5);height:calc(var(--spacing) * 3.5)}.size-4{width:calc(var(--spacing) * 4);height:calc(var(--spacing) * 4)}.size-5{width:calc(var(--spacing) * 5);height:calc(var(--spacing) * 5)}.size-6{width:calc(var(--spacing) * 6);height:calc(var(--spacing) * 6)}.size-7{width:calc(var(--spacing) * 7);height:calc(var(--spacing) * 7)}.size-8{width:calc(var(--spacing) * 8);height:calc(var(--spacing) * 8)}.size-9{width:calc(var(--spacing) * 9);height:calc(var(--spacing) * 9)}.size-10{width:calc(var(--spacing) * 10);height:calc(var(--spacing) * 10)}.h-4{height:calc(var(--spacing) * 4)}.h-5{height:calc(var(--spacing) * 5)}.h-6{height:calc(var(--spacing) * 6)}.h-7{height:calc(var(--spacing) * 7)}.h-8{height:calc(var(--spacing) * 8)}.h-9{height:calc(var(--spacing) * 9)}.h-\[52px\]{height:52px}.max-h-64{max-height:calc(var(--spacing) * 64)}.min-h-10{min-height:calc(var(--spacing) * 10)}.min-h-16{min-height:calc(var(--spacing) * 16)}.w-0\.5{width:calc(var(--spacing) * .5)}.w-fit{width:fit-content}.w-full{width:100%}.max-w-none{max-width:none}.max-w-prose{max-width:65ch}.min-w-0{min-width:calc(var(--spacing) * 0)}.flex-1{flex:1}.shrink-0{flex-shrink:0}.-translate-y-1\/2{--tw-translate-y:calc(calc(1 / 2 * 100%) * -1);translate:var(--tw-translate-x) var(--tw-translate-y)}.rotate-90{rotate:90deg}.rotate-180{rotate:180deg}.animate-pulse{animation:var(--animate-pulse)}.animate-spin{animation:var(--animate-spin)}.cursor-pointer{cursor:pointer}.resize-none{resize:none}.auto-rows-min{grid-auto-rows:min-content}.grid-cols-\[1fr_auto\]{grid-template-columns:1fr auto}.flex-col{flex-direction:column}.flex-wrap{flex-wrap:wrap}.items-center{align-items:center}.items-end{align-items:flex-end}.items-start{align-items:flex-start}.justify-between{justify-content:space-between}.justify-center{justify-content:center}.gap-1{gap:calc(var(--spacing) * 1)}.gap-1\.5{gap:calc(var(--spacing) * 1.5)}.gap-2{gap:calc(var(--spacing) * 2)}.gap-3{gap:calc(var(--spacing) * 3)}:where(.space-y-1>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 1) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 1) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-2>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 2) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 2) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-3>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 3) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 3) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-4>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 4) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 4) * calc(1 - var(--tw-space-y-reverse)))}:where(.space-y-8>:not(:last-child)){--tw-space-y-reverse:0;margin-block-start:calc(calc(var(--spacing) * 8) * var(--tw-space-y-reverse));margin-block-end:calc(calc(var(--spacing) * 8) * calc(1 - var(--tw-space-y-reverse)))}.self-center{align-self:center}.self-start{align-self:flex-start}.justify-self-end{justify-self:flex-end}.truncate{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.overflow-hidden{overflow:hidden}.overflow-x-auto{overflow-x:auto}.overflow-y-auto{overflow-y:auto}.rounded{border-radius:.25rem}.rounded-4xl{border-radius:var(--radius-4xl)}.rounded-\[min\(var\(--radius-md\)\,10px\)\]{border-radius:min(var(--radius-md), 10px)}.rounded-\[min\(var\(--radius-md\)\,12px\)\]{border-radius:min(var(--radius-md), 12px)}.rounded-full{border-radius:3.40282e38px}.rounded-lg{border-radius:var(--stgm-radius)}.rounded-md{border-radius:calc(var(--stgm-radius) - 2px)}.rounded-xl{border-radius:calc(var(--stgm-radius) + 4px)}.rounded-t-lg{border-top-left-radius:var(--stgm-radius);border-top-right-radius:var(--stgm-radius)}.border{border-style:var(--tw-border-style);border-width:1px}.border-2{border-style:var(--tw-border-style);border-width:2px}.border-t{border-top-style:var(--tw-border-style);border-top-width:1px}.border-b{border-bottom-style:var(--tw-border-style);border-bottom-width:1px}.border-dashed{--tw-border-style:dashed;border-style:dashed}.border-border{border-color:var(--stgm-border)}.border-destructive\/30{border-color:var(--stgm-destructive)}@supports (color:color-mix(in lab, red, red)){.border-destructive\/30{border-color:color-mix(in oklab, var(--stgm-destructive) 30%, transparent)}}.border-input{border-color:var(--stgm-input)}.border-primary\/30{border-color:var(--stgm-primary)}@supports (color:color-mix(in lab, red, red)){.border-primary\/30{border-color:color-mix(in oklab, var(--stgm-primary) 30%, transparent)}}.border-primary\/40{border-color:var(--stgm-primary)}@supports (color:color-mix(in lab, red, red)){.border-primary\/40{border-color:color-mix(in oklab, var(--stgm-primary) 40%, transparent)}}.border-transparent{border-color:#0000}.bg-accent{background-color:var(--stgm-accent)}.bg-background{background-color:var(--stgm-background)}.bg-card{background-color:var(--stgm-card)}.bg-destructive\/5{background-color:var(--stgm-destructive)}@supports (color:color-mix(in lab, red, red)){.bg-destructive\/5{background-color:color-mix(in oklab, var(--stgm-destructive) 5%, transparent)}}.bg-destructive\/10{background-color:var(--stgm-destructive)}@supports (color:color-mix(in lab, red, red)){.bg-destructive\/10{background-color:color-mix(in oklab, var(--stgm-destructive) 10%, transparent)}}.bg-foreground{background-color:var(--stgm-foreground)}.bg-muted,.bg-muted\/50{background-color:var(--stgm-muted)}@supports (color:color-mix(in lab, red, red)){.bg-muted\/50{background-color:color-mix(in oklab, var(--stgm-muted) 50%, transparent)}}.bg-popover{background-color:var(--stgm-popover)}.bg-primary,.bg-primary\/5{background-color:var(--stgm-primary)}@supports (color:color-mix(in lab, red, red)){.bg-primary\/5{background-color:color-mix(in oklab, var(--stgm-primary) 5%, transparent)}}.bg-primary\/10{background-color:var(--stgm-primary)}@supports (color:color-mix(in lab, red, red)){.bg-primary\/10{background-color:color-mix(in oklab, var(--stgm-primary) 10%, transparent)}}.bg-secondary{background-color:var(--stgm-secondary)}.bg-transparent{background-color:#0000}.bg-clip-padding{background-clip:padding-box}.object-cover{object-fit:cover}.p-2{padding:calc(var(--spacing) * 2)}.p-3{padding:calc(var(--spacing) * 3)}.p-4{padding:calc(var(--spacing) * 4)}.px-2{padding-inline:calc(var(--spacing) * 2)}.px-2\.5{padding-inline:calc(var(--spacing) * 2.5)}.px-3{padding-inline:calc(var(--spacing) * 3)}.px-4{padding-inline:calc(var(--spacing) * 4)}.py-0\.5{padding-block:calc(var(--spacing) * .5)}.py-2{padding-block:calc(var(--spacing) * 2)}.py-2\.5{padding-block:calc(var(--spacing) * 2.5)}.py-3{padding-block:calc(var(--spacing) * 3)}.py-12{padding-block:calc(var(--spacing) * 12)}.pt-3{padding-top:calc(var(--spacing) * 3)}.pr-9{padding-right:calc(var(--spacing) * 9)}.pl-9{padding-left:calc(var(--spacing) * 9)}.text-center{text-align:center}.text-left{text-align:left}.align-text-bottom{vertical-align:text-bottom}.font-mono{font-family:var(--stgm-font-mono)}.text-base{font-size:var(--text-base);line-height:var(--tw-leading,var(--text-base--line-height))}.text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}.text-xl{font-size:var(--text-xl);line-height:var(--tw-leading,var(--text-xl--line-height))}.text-xs{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.text-\[0\.8rem\]{font-size:.8rem}.text-\[10px\]{font-size:10px}.text-\[11px\]{font-size:11px}.leading-relaxed{--tw-leading:var(--leading-relaxed);line-height:var(--leading-relaxed)}.leading-snug{--tw-leading:var(--leading-snug);line-height:var(--leading-snug)}.font-medium{--tw-font-weight:var(--font-weight-medium);font-weight:var(--font-weight-medium)}.font-semibold{--tw-font-weight:var(--font-weight-semibold);font-weight:var(--font-weight-semibold)}.tracking-wider{--tw-tracking:var(--tracking-wider);letter-spacing:var(--tracking-wider)}.break-words{overflow-wrap:break-word}.whitespace-nowrap{white-space:nowrap}.whitespace-pre-wrap{white-space:pre-wrap}.text-accent-foreground{color:var(--stgm-accent-foreground)}.text-card-foreground{color:var(--stgm-card-foreground)}.text-destructive{color:var(--stgm-destructive)}.text-foreground{color:var(--stgm-foreground)}.text-muted-foreground,.text-muted-foreground\/40{color:var(--stgm-muted-foreground)}@supports (color:color-mix(in lab, red, red)){.text-muted-foreground\/40{color:color-mix(in oklab, var(--stgm-muted-foreground) 40%, transparent)}}.text-muted-foreground\/60{color:var(--stgm-muted-foreground)}@supports (color:color-mix(in lab, red, red)){.text-muted-foreground\/60{color:color-mix(in oklab, var(--stgm-muted-foreground) 60%, transparent)}}.text-muted-foreground\/70{color:var(--stgm-muted-foreground)}@supports (color:color-mix(in lab, red, red)){.text-muted-foreground\/70{color:color-mix(in oklab, var(--stgm-muted-foreground) 70%, transparent)}}.text-primary{color:var(--stgm-primary)}.text-primary-foreground{color:var(--stgm-primary-foreground)}.text-secondary-foreground{color:var(--stgm-secondary-foreground)}.uppercase{text-transform:uppercase}.underline-offset-4{text-underline-offset:4px}.shadow-md{--tw-shadow:var(--stgm-shadow-md);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.ring-1{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.ring-foreground\/10{--tw-ring-color:var(--stgm-foreground)}@supports (color:color-mix(in lab, red, red)){.ring-foreground\/10{--tw-ring-color:color-mix(in oklab, var(--stgm-foreground) 10%, transparent)}}.ring-primary\/20{--tw-ring-color:var(--stgm-primary)}@supports (color:color-mix(in lab, red, red)){.ring-primary\/20{--tw-ring-color:color-mix(in oklab, var(--stgm-primary) 20%, transparent)}}.outline{outline-style:var(--tw-outline-style);outline-width:1px}.transition-all{transition-property:all;transition-timing-function:var(--tw-ease,var(--stgm-transition-timing));transition-duration:var(--tw-duration,var(--stgm-transition-duration))}.transition-colors{transition-property:color,background-color,border-color,outline-color,text-decoration-color,fill,stroke,--tw-gradient-from,--tw-gradient-via,--tw-gradient-to;transition-timing-function:var(--tw-ease,var(--stgm-transition-timing));transition-duration:var(--tw-duration,var(--stgm-transition-duration))}.transition-transform{transition-property:transform,translate,scale,rotate;transition-timing-function:var(--tw-ease,var(--stgm-transition-timing));transition-duration:var(--tw-duration,var(--stgm-transition-duration))}.outline-none{--tw-outline-style:none;outline-style:none}.select-none{-webkit-user-select:none;user-select:none}.placeholder\:text-muted-foreground::placeholder{color:var(--stgm-muted-foreground)}@media (hover:hover){.hover\:bg-accent:hover,.hover\:bg-accent\/50:hover{background-color:var(--stgm-accent)}@supports (color:color-mix(in lab, red, red)){.hover\:bg-accent\/50:hover{background-color:color-mix(in oklab, var(--stgm-accent) 50%, transparent)}}.hover\:bg-destructive\/20:hover{background-color:var(--stgm-destructive)}@supports (color:color-mix(in lab, red, red)){.hover\:bg-destructive\/20:hover{background-color:color-mix(in oklab, var(--stgm-destructive) 20%, transparent)}}.hover\:bg-muted:hover,.hover\:bg-muted\/50:hover{background-color:var(--stgm-muted)}@supports (color:color-mix(in lab, red, red)){.hover\:bg-muted\/50:hover{background-color:color-mix(in oklab, var(--stgm-muted) 50%, transparent)}}.hover\:bg-secondary\/80:hover{background-color:var(--stgm-secondary)}@supports (color:color-mix(in lab, red, red)){.hover\:bg-secondary\/80:hover{background-color:color-mix(in oklab, var(--stgm-secondary) 80%, transparent)}}.hover\:text-accent-foreground:hover{color:var(--stgm-accent-foreground)}.hover\:text-destructive\/80:hover{color:var(--stgm-destructive)}@supports (color:color-mix(in lab, red, red)){.hover\:text-destructive\/80:hover{color:color-mix(in oklab, var(--stgm-destructive) 80%, transparent)}}.hover\:text-foreground:hover{color:var(--stgm-foreground)}.hover\:text-muted-foreground:hover{color:var(--stgm-muted-foreground)}.hover\:underline:hover{text-decoration-line:underline}}.focus-visible\:border-destructive\/40:focus-visible{border-color:var(--stgm-destructive)}@supports (color:color-mix(in lab, red, red)){.focus-visible\:border-destructive\/40:focus-visible{border-color:color-mix(in oklab, var(--stgm-destructive) 40%, transparent)}}.focus-visible\:border-ring:focus-visible{border-color:var(--stgm-ring)}.focus-visible\:ring-2:focus-visible{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.focus-visible\:ring-3:focus-visible,.focus-visible\:ring-\[3px\]:focus-visible{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(3px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.focus-visible\:ring-destructive\/20:focus-visible{--tw-ring-color:var(--stgm-destructive)}@supports (color:color-mix(in lab, red, red)){.focus-visible\:ring-destructive\/20:focus-visible{--tw-ring-color:color-mix(in oklab, var(--stgm-destructive) 20%, transparent)}}.focus-visible\:ring-ring:focus-visible,.focus-visible\:ring-ring\/50:focus-visible{--tw-ring-color:var(--stgm-ring)}@supports (color:color-mix(in lab, red, red)){.focus-visible\:ring-ring\/50:focus-visible{--tw-ring-color:color-mix(in oklab, var(--stgm-ring) 50%, transparent)}}.focus-visible\:outline-none:focus-visible{--tw-outline-style:none;outline-style:none}.active\:translate-y-px:active{--tw-translate-y:1px;translate:var(--tw-translate-x) var(--tw-translate-y)}.disabled\:pointer-events-none:disabled{pointer-events:none}.disabled\:cursor-not-allowed:disabled{cursor:not-allowed}.disabled\:bg-input\/50:disabled{background-color:var(--stgm-input)}@supports (color:color-mix(in lab, red, red)){.disabled\:bg-input\/50:disabled{background-color:color-mix(in oklab, var(--stgm-input) 50%, transparent)}}.disabled\:opacity-50:disabled{opacity:.5}:where([data-slot=button-group]) .in-data-\[slot\=button-group\]\:rounded-lg{border-radius:var(--stgm-radius)}.has-data-\[icon\=inline-end\]\:pr-1\.5:has([data-icon=inline-end]){padding-right:calc(var(--spacing) * 1.5)}.has-data-\[icon\=inline-end\]\:pr-2:has([data-icon=inline-end]){padding-right:calc(var(--spacing) * 2)}.has-data-\[icon\=inline-end\]\:pr-3:has([data-icon=inline-end]){padding-right:calc(var(--spacing) * 3)}.has-data-\[icon\=inline-start\]\:pl-1\.5:has([data-icon=inline-start]){padding-left:calc(var(--spacing) * 1.5)}.has-data-\[icon\=inline-start\]\:pl-2:has([data-icon=inline-start]){padding-left:calc(var(--spacing) * 2)}.has-data-\[icon\=inline-start\]\:pl-3:has([data-icon=inline-start]){padding-left:calc(var(--spacing) * 3)}.aria-expanded\:bg-muted[aria-expanded=true]{background-color:var(--stgm-muted)}.aria-expanded\:bg-secondary[aria-expanded=true]{background-color:var(--stgm-secondary)}.aria-expanded\:text-foreground[aria-expanded=true]{color:var(--stgm-foreground)}.aria-expanded\:text-secondary-foreground[aria-expanded=true]{color:var(--stgm-secondary-foreground)}.aria-invalid\:border-destructive[aria-invalid=true]{border-color:var(--stgm-destructive)}.aria-invalid\:ring-3[aria-invalid=true]{--tw-ring-shadow:var(--tw-ring-inset,) 0 0 0 calc(3px + var(--tw-ring-offset-width)) var(--tw-ring-color,currentcolor);box-shadow:var(--tw-inset-shadow), var(--tw-inset-ring-shadow), var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow)}.aria-invalid\:ring-destructive\/20[aria-invalid=true]{--tw-ring-color:var(--stgm-destructive)}@supports (color:color-mix(in lab, red, red)){.aria-invalid\:ring-destructive\/20[aria-invalid=true]{--tw-ring-color:color-mix(in oklab, var(--stgm-destructive) 20%, transparent)}}@media (min-width:48rem){.md\:text-sm{font-size:var(--text-sm);line-height:var(--tw-leading,var(--text-sm--line-height))}}.dark\:border-input:is(.dark *){border-color:var(--stgm-input)}.dark\:bg-destructive\/20:is(.dark *){background-color:var(--stgm-destructive)}@supports (color:color-mix(in lab, red, red)){.dark\:bg-destructive\/20:is(.dark *){background-color:color-mix(in oklab, var(--stgm-destructive) 20%, transparent)}}.dark\:bg-input\/30:is(.dark *){background-color:var(--stgm-input)}@supports (color:color-mix(in lab, red, red)){.dark\:bg-input\/30:is(.dark *){background-color:color-mix(in oklab, var(--stgm-input) 30%, transparent)}}@media (hover:hover){.dark\:hover\:bg-destructive\/30:is(.dark *):hover{background-color:var(--stgm-destructive)}@supports (color:color-mix(in lab, red, red)){.dark\:hover\:bg-destructive\/30:is(.dark *):hover{background-color:color-mix(in oklab, var(--stgm-destructive) 30%, transparent)}}.dark\:hover\:bg-input\/50:is(.dark *):hover{background-color:var(--stgm-input)}@supports (color:color-mix(in lab, red, red)){.dark\:hover\:bg-input\/50:is(.dark *):hover{background-color:color-mix(in oklab, var(--stgm-input) 50%, transparent)}}.dark\:hover\:bg-muted\/50:is(.dark *):hover{background-color:var(--stgm-muted)}@supports (color:color-mix(in lab, red, red)){.dark\:hover\:bg-muted\/50:is(.dark *):hover{background-color:color-mix(in oklab, var(--stgm-muted) 50%, transparent)}}}.dark\:focus-visible\:ring-destructive\/40:is(.dark *):focus-visible{--tw-ring-color:var(--stgm-destructive)}@supports (color:color-mix(in lab, red, red)){.dark\:focus-visible\:ring-destructive\/40:is(.dark *):focus-visible{--tw-ring-color:color-mix(in oklab, var(--stgm-destructive) 40%, transparent)}}.dark\:disabled\:bg-input\/80:is(.dark *):disabled{background-color:var(--stgm-input)}@supports (color:color-mix(in lab, red, red)){.dark\:disabled\:bg-input\/80:is(.dark *):disabled{background-color:color-mix(in oklab, var(--stgm-input) 80%, transparent)}}.dark\:aria-invalid\:border-destructive\/50:is(.dark *)[aria-invalid=true]{border-color:var(--stgm-destructive)}@supports (color:color-mix(in lab, red, red)){.dark\:aria-invalid\:border-destructive\/50:is(.dark *)[aria-invalid=true]{border-color:color-mix(in oklab, var(--stgm-destructive) 50%, transparent)}}.dark\:aria-invalid\:ring-destructive\/40:is(.dark *)[aria-invalid=true]{--tw-ring-color:var(--stgm-destructive)}@supports (color:color-mix(in lab, red, red)){.dark\:aria-invalid\:ring-destructive\/40:is(.dark *)[aria-invalid=true]{--tw-ring-color:color-mix(in oklab, var(--stgm-destructive) 40%, transparent)}}.\[\&_code\]\:rounded code{border-radius:.25rem}.\[\&_code\]\:bg-muted code{background-color:var(--stgm-muted)}.\[\&_code\]\:px-1\.5 code{padding-inline:calc(var(--spacing) * 1.5)}.\[\&_code\]\:py-0\.5 code{padding-block:calc(var(--spacing) * .5)}.\[\&_code\]\:text-xs code{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.\[\&_code\]\:before\:content-none code:before,.\[\&_code\]\:after\:content-none code:after{content:var(--tw-content);--tw-content:none;content:none}.\[\&_pre\]\:overflow-x-auto pre{overflow-x:auto}.\[\&_pre\]\:rounded-lg pre{border-radius:var(--stgm-radius)}.\[\&_pre\]\:bg-muted pre{background-color:var(--stgm-muted)}.\[\&_pre\]\:p-3 pre{padding:calc(var(--spacing) * 3)}.\[\&_pre\]\:text-xs pre{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.\[\&_svg\]\:pointer-events-none svg{pointer-events:none}.\[\&_svg\]\:shrink-0 svg{flex-shrink:0}.\[\&_svg\:not\(\[class\*\=\'size-\'\]\)\]\:size-3 svg:not([class*=size-]){width:calc(var(--spacing) * 3);height:calc(var(--spacing) * 3)}.\[\&_svg\:not\(\[class\*\=\'size-\'\]\)\]\:size-3\.5 svg:not([class*=size-]){width:calc(var(--spacing) * 3.5);height:calc(var(--spacing) * 3.5)}.\[\&_svg\:not\(\[class\*\=\'size-\'\]\)\]\:size-4 svg:not([class*=size-]){width:calc(var(--spacing) * 4);height:calc(var(--spacing) * 4)}.\[\&_table\]\:text-xs table{font-size:var(--text-xs);line-height:var(--tw-leading,var(--text-xs--line-height))}.\[\&_td\]\:px-2 td{padding-inline:calc(var(--spacing) * 2)}.\[\&_td\]\:py-1 td{padding-block:calc(var(--spacing) * 1)}.\[\&_th\]\:px-2 th{padding-inline:calc(var(--spacing) * 2)}.\[\&_th\]\:py-1 th{padding-block:calc(var(--spacing) * 1)}@media (hover:hover){.\[a\]\:hover\:bg-destructive\/20:is(a):hover{background-color:var(--stgm-destructive)}@supports (color:color-mix(in lab, red, red)){.\[a\]\:hover\:bg-destructive\/20:is(a):hover{background-color:color-mix(in oklab, var(--stgm-destructive) 20%, transparent)}}.\[a\]\:hover\:bg-muted:is(a):hover{background-color:var(--stgm-muted)}.\[a\]\:hover\:bg-primary\/80:is(a):hover{background-color:var(--stgm-primary)}@supports (color:color-mix(in lab, red, red)){.\[a\]\:hover\:bg-primary\/80:is(a):hover{background-color:color-mix(in oklab, var(--stgm-primary) 80%, transparent)}}.\[a\]\:hover\:bg-secondary\/80:is(a):hover{background-color:var(--stgm-secondary)}@supports (color:color-mix(in lab, red, red)){.\[a\]\:hover\:bg-secondary\/80:is(a):hover{background-color:color-mix(in oklab, var(--stgm-secondary) 80%, transparent)}}.\[a\]\:hover\:text-muted-foreground:is(a):hover{color:var(--stgm-muted-foreground)}}.\[\&\>svg\]\:pointer-events-none>svg{pointer-events:none}.\[\&\>svg\]\:size-3\!>svg{width:calc(var(--spacing) * 3)!important;height:calc(var(--spacing) * 3)!important}:root{--stgm-font-sans:var(--font-geist-sans,system-ui, sans-serif);--stgm-font-mono:var(--font-geist-mono,ui-monospace, monospace);--stgm-radius:.625rem;--stgm-background:oklch(100% 0 0);--stgm-foreground:oklch(14.5% 0 0);--stgm-card:oklch(100% 0 0);--stgm-card-foreground:oklch(14.5% 0 0);--stgm-popover:oklch(100% 0 0);--stgm-popover-foreground:oklch(14.5% 0 0);--stgm-primary:oklch(55% .12 190);--stgm-primary-foreground:oklch(98.5% 0 0);--stgm-secondary:oklch(97% 0 0);--stgm-secondary-foreground:oklch(20.5% 0 0);--stgm-muted:oklch(97% 0 0);--stgm-muted-foreground:oklch(55.6% 0 0);--stgm-accent:oklch(97% 0 0);--stgm-accent-foreground:oklch(20.5% 0 0);--stgm-destructive:oklch(57.7% .245 27.325);--stgm-destructive-foreground:oklch(98.5% 0 0);--stgm-success:oklch(55% .15 150);--stgm-success-foreground:oklch(98.5% 0 0);--stgm-warning:oklch(75% .18 80);--stgm-warning-foreground:oklch(14.5% 0 0);--stgm-info:oklch(55% .15 250);--stgm-info-foreground:oklch(98.5% 0 0);--stgm-border:oklch(92.2% 0 0);--stgm-input:oklch(92.2% 0 0);--stgm-ring:oklch(55% .1 190);--stgm-chart-1:oklch(64.6% .222 41.116);--stgm-chart-2:oklch(60% .118 184.704);--stgm-chart-3:oklch(39.8% .07 227.392);--stgm-chart-4:oklch(82.8% .189 84.429);--stgm-chart-5:oklch(76.9% .188 70.08);--stgm-shadow-sm:0 1px 3px 0 #0000001a, 0 1px 2px -1px #0000001a;--stgm-shadow-md:0 4px 6px -1px #0000001a, 0 2px 4px -2px #0000001a;--stgm-shadow-lg:0 10px 15px -3px #0000001a, 0 4px 6px -4px #0000001a;--stgm-transition-duration:.15s;--stgm-transition-timing:cubic-bezier(.4, 0, .2, 1);--stgm-z-popover:50;--stgm-sidebar:oklch(98.5% 0 0);--stgm-sidebar-foreground:oklch(14.5% 0 0);--stgm-sidebar-primary:oklch(55% .12 190);--stgm-sidebar-primary-foreground:oklch(98.5% 0 0);--stgm-sidebar-accent:oklch(97% 0 0);--stgm-sidebar-accent-foreground:oklch(20.5% 0 0);--stgm-sidebar-border:oklch(92.2% 0 0);--stgm-sidebar-ring:oklch(55% .1 190)}.dark{--stgm-background:oklch(14.5% 0 0);--stgm-foreground:oklch(98.5% 0 0);--stgm-card:oklch(20.5% 0 0);--stgm-card-foreground:oklch(98.5% 0 0);--stgm-popover:oklch(26.9% 0 0);--stgm-popover-foreground:oklch(98.5% 0 0);--stgm-primary:oklch(72% .12 190);--stgm-primary-foreground:oklch(14.5% 0 0);--stgm-secondary:oklch(26.9% 0 0);--stgm-secondary-foreground:oklch(98.5% 0 0);--stgm-muted:oklch(26.9% 0 0);--stgm-muted-foreground:oklch(70.8% 0 0);--stgm-accent:oklch(37.1% 0 0);--stgm-accent-foreground:oklch(98.5% 0 0);--stgm-destructive:oklch(70.4% .191 22.216);--stgm-destructive-foreground:oklch(98.5% 0 0);--stgm-success:oklch(70% .15 150);--stgm-success-foreground:oklch(98.5% 0 0);--stgm-warning:oklch(78% .16 80);--stgm-warning-foreground:oklch(14.5% 0 0);--stgm-info:oklch(70% .15 250);--stgm-info-foreground:oklch(98.5% 0 0);--stgm-border:oklch(100% 0 0/.1);--stgm-input:oklch(100% 0 0/.15);--stgm-ring:oklch(55% .1 190);--stgm-chart-1:oklch(48.8% .243 264.376);--stgm-chart-2:oklch(69.6% .17 162.48);--stgm-chart-3:oklch(76.9% .188 70.08);--stgm-chart-4:oklch(62.7% .265 303.9);--stgm-chart-5:oklch(64.5% .246 16.439);--stgm-shadow-sm:0 1px 3px 0 #00000040, 0 1px 2px -1px #00000040;--stgm-shadow-md:0 4px 6px -1px #0000004d, 0 2px 4px -2px #00000040;--stgm-shadow-lg:0 10px 15px -3px #00000059, 0 4px 6px -4px #0000004d;--stgm-sidebar:oklch(20.5% 0 0);--stgm-sidebar-foreground:oklch(98.5% 0 0);--stgm-sidebar-primary:oklch(72% .12 190);--stgm-sidebar-primary-foreground:oklch(14.5% 0 0);--stgm-sidebar-accent:oklch(26.9% 0 0);--stgm-sidebar-accent-foreground:oklch(98.5% 0 0);--stgm-sidebar-border:oklch(100% 0 0/.1);--stgm-sidebar-ring:oklch(50% .08 190)}.stgm{-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale;line-height:1.5}.stgm *,.stgm :before,.stgm :after{box-sizing:border-box;border-style:solid;border-width:0;border-color:var(--stgm-border,currentColor)}}@property --tw-translate-x{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-y{syntax:"*";inherits:false;initial-value:0}@property --tw-translate-z{syntax:"*";inherits:false;initial-value:0}@property --tw-space-y-reverse{syntax:"*";inherits:false;initial-value:0}@property --tw-border-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-leading{syntax:"*";inherits:false}@property --tw-font-weight{syntax:"*";inherits:false}@property --tw-tracking{syntax:"*";inherits:false}@property --tw-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-shadow-color{syntax:"*";inherits:false}@property --tw-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-inset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-shadow-color{syntax:"*";inherits:false}@property --tw-inset-shadow-alpha{syntax:"<percentage>";inherits:false;initial-value:100%}@property --tw-ring-color{syntax:"*";inherits:false}@property --tw-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-inset-ring-color{syntax:"*";inherits:false}@property --tw-inset-ring-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-ring-inset{syntax:"*";inherits:false}@property --tw-ring-offset-width{syntax:"<length>";inherits:false;initial-value:0}@property --tw-ring-offset-color{syntax:"*";inherits:false;initial-value:#fff}@property --tw-ring-offset-shadow{syntax:"*";inherits:false;initial-value:0 0 #0000}@property --tw-outline-style{syntax:"*";inherits:false;initial-value:solid}@property --tw-content{syntax:"*";inherits:false;initial-value:""}@keyframes spin{to{transform:rotate(360deg)}}@keyframes pulse{50%{opacity:.5}}