@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.
- package/LICENSE +190 -0
- package/README.md +248 -0
- package/agent/components/AgentCard.d.ts +9 -0
- package/agent/components/AgentCard.d.ts.map +1 -0
- package/agent/components/AgentCard.js +26 -0
- package/agent/components/AgentCard.js.map +1 -0
- package/agent/components/AgentOverview.d.ts +7 -0
- package/agent/components/AgentOverview.d.ts.map +1 -0
- package/agent/components/AgentOverview.js +36 -0
- package/agent/components/AgentOverview.js.map +1 -0
- package/agent/components/AgentPicker.d.ts +17 -0
- package/agent/components/AgentPicker.d.ts.map +1 -0
- package/agent/components/AgentPicker.js +86 -0
- package/agent/components/AgentPicker.js.map +1 -0
- package/agent/hooks/useAgentSearch.d.ts +28 -0
- package/agent/hooks/useAgentSearch.d.ts.map +1 -0
- package/agent/hooks/useAgentSearch.js +63 -0
- package/agent/hooks/useAgentSearch.js.map +1 -0
- package/agent/index.d.ts +9 -0
- package/agent/index.d.ts.map +1 -0
- package/agent/index.js +7 -0
- package/agent/index.js.map +1 -0
- package/agent-execution/components/ApprovalControls.d.ts +10 -0
- package/agent-execution/components/ApprovalControls.d.ts.map +1 -0
- package/agent-execution/components/ApprovalControls.js +19 -0
- package/agent-execution/components/ApprovalControls.js.map +1 -0
- package/agent-execution/components/ExecutionStatus.d.ts +8 -0
- package/agent-execution/components/ExecutionStatus.d.ts.map +1 -0
- package/agent-execution/components/ExecutionStatus.js +14 -0
- package/agent-execution/components/ExecutionStatus.js.map +1 -0
- package/agent-execution/components/ExecutionStream.d.ts +16 -0
- package/agent-execution/components/ExecutionStream.d.ts.map +1 -0
- package/agent-execution/components/ExecutionStream.js +39 -0
- package/agent-execution/components/ExecutionStream.js.map +1 -0
- package/agent-execution/components/MessageEntry.d.ts +17 -0
- package/agent-execution/components/MessageEntry.d.ts.map +1 -0
- package/agent-execution/components/MessageEntry.js +36 -0
- package/agent-execution/components/MessageEntry.js.map +1 -0
- package/agent-execution/components/MessageInput.d.ts +10 -0
- package/agent-execution/components/MessageInput.d.ts.map +1 -0
- package/agent-execution/components/MessageInput.js +27 -0
- package/agent-execution/components/MessageInput.js.map +1 -0
- package/agent-execution/components/OutputBlock.d.ts +9 -0
- package/agent-execution/components/OutputBlock.d.ts.map +1 -0
- package/agent-execution/components/OutputBlock.js +15 -0
- package/agent-execution/components/OutputBlock.js.map +1 -0
- package/agent-execution/components/SubAgentCard.d.ts +11 -0
- package/agent-execution/components/SubAgentCard.d.ts.map +1 -0
- package/agent-execution/components/SubAgentCard.js +19 -0
- package/agent-execution/components/SubAgentCard.js.map +1 -0
- package/agent-execution/components/ToolCallCard.d.ts +11 -0
- package/agent-execution/components/ToolCallCard.d.ts.map +1 -0
- package/agent-execution/components/ToolCallCard.js +25 -0
- package/agent-execution/components/ToolCallCard.js.map +1 -0
- package/agent-execution/helpers.d.ts +35 -0
- package/agent-execution/helpers.d.ts.map +1 -0
- package/agent-execution/helpers.js +157 -0
- package/agent-execution/helpers.js.map +1 -0
- package/agent-execution/hooks/useAgentExecution.d.ts +21 -0
- package/agent-execution/hooks/useAgentExecution.d.ts.map +1 -0
- package/agent-execution/hooks/useAgentExecution.js +99 -0
- package/agent-execution/hooks/useAgentExecution.js.map +1 -0
- package/agent-execution/hooks/useApproval.d.ts +12 -0
- package/agent-execution/hooks/useApproval.d.ts.map +1 -0
- package/agent-execution/hooks/useApproval.js +32 -0
- package/agent-execution/hooks/useApproval.js.map +1 -0
- package/agent-execution/index.d.ts +14 -0
- package/agent-execution/index.d.ts.map +1 -0
- package/agent-execution/index.js +15 -0
- package/agent-execution/index.js.map +1 -0
- package/catalog/components/ResourceSearchCard.d.ts +23 -0
- package/catalog/components/ResourceSearchCard.d.ts.map +1 -0
- package/catalog/components/ResourceSearchCard.js +36 -0
- package/catalog/components/ResourceSearchCard.js.map +1 -0
- package/catalog/index.d.ts +4 -0
- package/catalog/index.d.ts.map +1 -0
- package/catalog/index.js +5 -0
- package/catalog/index.js.map +1 -0
- package/catalog/internal/time.d.ts +13 -0
- package/catalog/internal/time.d.ts.map +1 -0
- package/catalog/internal/time.js +41 -0
- package/catalog/internal/time.js.map +1 -0
- package/context.d.ts +12 -0
- package/context.d.ts.map +1 -0
- package/context.js +13 -0
- package/context.js.map +1 -0
- package/hooks.d.ts +19 -0
- package/hooks.d.ts.map +1 -0
- package/hooks.js +28 -0
- package/hooks.js.map +1 -0
- package/index.d.ts +4 -0
- package/index.d.ts.map +1 -0
- package/index.js +6 -0
- package/index.js.map +1 -0
- package/internal/badge.d.ts +8 -0
- package/internal/badge.d.ts.map +1 -0
- package/internal/badge.js +34 -0
- package/internal/badge.js.map +1 -0
- package/internal/button.d.ts +9 -0
- package/internal/button.d.ts.map +1 -0
- package/internal/button.js +36 -0
- package/internal/button.js.map +1 -0
- package/internal/collapsible.d.ts +6 -0
- package/internal/collapsible.d.ts.map +1 -0
- package/internal/collapsible.js +14 -0
- package/internal/collapsible.js.map +1 -0
- package/internal/section.d.ts +8 -0
- package/internal/section.d.ts.map +1 -0
- package/internal/section.js +6 -0
- package/internal/section.js.map +1 -0
- package/internal/textarea.d.ts +4 -0
- package/internal/textarea.d.ts.map +1 -0
- package/internal/textarea.js +9 -0
- package/internal/textarea.js.map +1 -0
- package/mcp-server/hooks/useMcpServerSearch.d.ts +25 -0
- package/mcp-server/hooks/useMcpServerSearch.d.ts.map +1 -0
- package/mcp-server/hooks/useMcpServerSearch.js +57 -0
- package/mcp-server/hooks/useMcpServerSearch.js.map +1 -0
- package/mcp-server/index.d.ts +3 -0
- package/mcp-server/index.d.ts.map +1 -0
- package/mcp-server/index.js +3 -0
- package/mcp-server/index.js.map +1 -0
- package/package.json +75 -0
- package/provider.d.ts +55 -0
- package/provider.d.ts.map +1 -0
- package/provider.js +34 -0
- package/provider.js.map +1 -0
- package/session/components/AgentSessionHistory.d.ts +8 -0
- package/session/components/AgentSessionHistory.d.ts.map +1 -0
- package/session/components/AgentSessionHistory.js +11 -0
- package/session/components/AgentSessionHistory.js.map +1 -0
- package/session/components/SessionCard.d.ts +8 -0
- package/session/components/SessionCard.d.ts.map +1 -0
- package/session/components/SessionCard.js +57 -0
- package/session/components/SessionCard.js.map +1 -0
- package/session/hooks/useAgentSessionList.d.ts +21 -0
- package/session/hooks/useAgentSessionList.d.ts.map +1 -0
- package/session/hooks/useAgentSessionList.js +90 -0
- package/session/hooks/useAgentSessionList.js.map +1 -0
- package/session/index.d.ts +7 -0
- package/session/index.d.ts.map +1 -0
- package/session/index.js +6 -0
- package/session/index.js.map +1 -0
- package/skill/hooks/useSkillSearch.d.ts +25 -0
- package/skill/hooks/useSkillSearch.d.ts.map +1 -0
- package/skill/hooks/useSkillSearch.js +57 -0
- package/skill/hooks/useSkillSearch.js.map +1 -0
- package/skill/index.d.ts +3 -0
- package/skill/index.d.ts.map +1 -0
- package/skill/index.js +3 -0
- package/skill/index.js.map +1 -0
- package/src/agent/components/AgentCard.tsx +125 -0
- package/src/agent/components/AgentOverview.tsx +209 -0
- package/src/agent/components/AgentPicker.tsx +255 -0
- package/src/agent/hooks/useAgentSearch.ts +94 -0
- package/src/agent/index.ts +17 -0
- package/src/agent-execution/components/ApprovalControls.tsx +99 -0
- package/src/agent-execution/components/ExecutionStatus.tsx +33 -0
- package/src/agent-execution/components/ExecutionStream.tsx +148 -0
- package/src/agent-execution/components/MessageEntry.tsx +125 -0
- package/src/agent-execution/components/MessageInput.tsx +70 -0
- package/src/agent-execution/components/OutputBlock.tsx +43 -0
- package/src/agent-execution/components/SubAgentCard.tsx +138 -0
- package/src/agent-execution/components/ToolCallCard.tsx +153 -0
- package/src/agent-execution/helpers.ts +193 -0
- package/src/agent-execution/hooks/useAgentExecution.ts +147 -0
- package/src/agent-execution/hooks/useApproval.ts +56 -0
- package/src/agent-execution/index.ts +46 -0
- package/src/catalog/components/ResourceSearchCard.tsx +137 -0
- package/src/catalog/index.ts +6 -0
- package/src/catalog/internal/time.ts +40 -0
- package/src/context.ts +15 -0
- package/src/hooks.ts +32 -0
- package/src/index.ts +6 -0
- package/src/internal/badge.tsx +52 -0
- package/src/internal/button.tsx +60 -0
- package/src/internal/collapsible.tsx +21 -0
- package/src/internal/section.tsx +18 -0
- package/src/internal/textarea.tsx +23 -0
- package/src/mcp-server/hooks/useMcpServerSearch.ts +79 -0
- package/src/mcp-server/index.ts +6 -0
- package/src/provider.tsx +73 -0
- package/src/session/components/AgentSessionHistory.tsx +109 -0
- package/src/session/components/SessionCard.tsx +113 -0
- package/src/session/hooks/useAgentSessionList.ts +117 -0
- package/src/session/index.ts +13 -0
- package/src/skill/hooks/useSkillSearch.ts +79 -0
- package/src/skill/index.ts +6 -0
- package/src/styles.css +72 -0
- 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
|
+
}
|
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}}
|