- Add demo dashboard with 7 widgets to seed script - Dashboard is_default=true — appears as home page on fresh seed - Add views/dashboards/dashboardWidgets to seed reset - Fix My tickets: now filters by first non-system user (not any owner) - Pass owner param in sidebar My tickets link - Update page.tsx view=my to respect owner URL param - Scrip engine already sorts by sort_order (verified) Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
1228 lines
52 KiB
TypeScript
1228 lines
52 KiB
TypeScript
"use client";
|
|
|
|
import { Suspense, useCallback, useEffect, useMemo, useState } from "react";
|
|
import { useRouter, useSearchParams } from "next/navigation";
|
|
import {
|
|
ArrowDownAZIcon,
|
|
CheckCircle2Icon,
|
|
ChevronRightIcon,
|
|
GaugeIcon,
|
|
LayoutListIcon,
|
|
PlusIcon,
|
|
RefreshCwIcon,
|
|
SaveIcon,
|
|
SearchIcon,
|
|
SlidersHorizontalIcon,
|
|
XIcon,
|
|
} from "lucide-react";
|
|
import { formatDistanceToNow } from "date-fns";
|
|
import { createTicket, getCustomFields, getLifecycles, getQueueCustomFields, getQueues, getTickets, getUsers, getViews, createView, deleteView, getDashboards } from "@/lib/api";
|
|
import type { CustomField, Lifecycle, Queue, QueueCustomField, SavedView, Ticket, User } from "@/lib/types";
|
|
import { Button } from "@/components/ui/button";
|
|
import {
|
|
Dialog,
|
|
DialogContent,
|
|
DialogDescription,
|
|
DialogFooter,
|
|
DialogHeader,
|
|
DialogTitle,
|
|
} from "@/components/ui/dialog";
|
|
import { cn, formatTicketId } from "@/lib/utils";
|
|
|
|
const STATUS_META: Record<string, { label: string; color: string; tone: string }> = {
|
|
new: { label: "New", color: "#64748b", tone: "bg-slate-500/10 text-slate-700 dark:text-slate-300" },
|
|
open: { label: "Open", color: "#2563eb", tone: "bg-blue-500/10 text-blue-700 dark:text-blue-300" },
|
|
in_progress: { label: "In progress", color: "#d97706", tone: "bg-amber-500/10 text-amber-700 dark:text-amber-300" },
|
|
resolved: { label: "Resolved", color: "#16a34a", tone: "bg-emerald-500/10 text-emerald-700 dark:text-emerald-300" },
|
|
closed: { label: "Closed", color: "#71717a", tone: "bg-zinc-500/10 text-zinc-700 dark:text-zinc-300" },
|
|
};
|
|
|
|
const VIEW_LABELS: Record<string, string> = {
|
|
my: "My tickets",
|
|
unassigned: "Unassigned",
|
|
recent: "Recently updated",
|
|
};
|
|
|
|
type Density = "comfortable" | "compact";
|
|
type SortKey = "updated" | "created" | "id";
|
|
|
|
interface Filter {
|
|
id: string;
|
|
field: string; // "status" | "queue" | "owner" | custom field key ("cf.<key>")
|
|
operator: string; // "is" | "is_not"
|
|
value: string;
|
|
label: string; // human-readable label for the chip
|
|
}
|
|
|
|
function buildFilterLabel(field: string, operator: string, valueLabel: string): string {
|
|
const fieldLabel = field.startsWith("cf.") ? field.slice(3) : field;
|
|
const op = operator === "is_not" ? "is not" : "is";
|
|
return `${fieldLabel} ${op} ${valueLabel}`;
|
|
}
|
|
|
|
function statusLabel(status: string) {
|
|
return STATUS_META[status]?.label ?? status.replaceAll("_", " ");
|
|
}
|
|
|
|
function queueName(queues: Queue[], queueId: string) {
|
|
return queues.find((queue) => queue.id === queueId)?.name ?? queueId.slice(0, 8);
|
|
}
|
|
|
|
function relativeTime(value: string) {
|
|
return formatDistanceToNow(new Date(value), { addSuffix: true });
|
|
}
|
|
|
|
function TicketStatusBadge({ status }: { status: string }) {
|
|
const meta = STATUS_META[status];
|
|
return (
|
|
<span
|
|
className={cn(
|
|
"inline-flex h-6 items-center gap-1.5 rounded px-2 text-xs font-semibold capitalize",
|
|
meta?.tone ?? "bg-muted text-muted-foreground"
|
|
)}
|
|
>
|
|
<span
|
|
className="h-1.5 w-1.5 rounded-full"
|
|
style={{ backgroundColor: meta?.color ?? "#71717a" }}
|
|
/>
|
|
{statusLabel(status)}
|
|
</span>
|
|
);
|
|
}
|
|
|
|
function StatPill({
|
|
label,
|
|
value,
|
|
}: {
|
|
label: string;
|
|
value: number;
|
|
}) {
|
|
return (
|
|
<span className="inline-flex h-6 items-center gap-1.5 rounded border border-border/80 bg-card/80 px-2 text-[11px] text-muted-foreground shadow-sm">
|
|
<span className="font-semibold tabular-nums text-foreground">{value}</span>
|
|
{label}
|
|
</span>
|
|
);
|
|
}
|
|
|
|
function SkeletonWorkbench() {
|
|
return (
|
|
<div className="flex h-full flex-col">
|
|
<div className="border-b border-border px-5 py-4">
|
|
<div className="h-7 w-48 animate-pulse rounded bg-muted" />
|
|
<div className="mt-3 h-9 w-full max-w-xl animate-pulse rounded bg-muted" />
|
|
</div>
|
|
<div className="grid flex-1 grid-cols-1 overflow-hidden xl:grid-cols-[minmax(0,1fr)_360px]">
|
|
<div className="overflow-hidden">
|
|
{Array.from({ length: 12 }).map((_, index) => (
|
|
<div key={index} className="flex items-center gap-4 border-b border-border px-5 py-4">
|
|
<div className="h-4 w-4 animate-pulse rounded bg-muted" />
|
|
<div className="min-w-0 flex-1 space-y-2">
|
|
<div className="h-4 w-2/5 animate-pulse rounded bg-muted" />
|
|
<div className="h-3 w-3/5 animate-pulse rounded bg-muted" />
|
|
</div>
|
|
<div className="h-6 w-20 animate-pulse rounded bg-muted" />
|
|
</div>
|
|
))}
|
|
</div>
|
|
<div className="hidden border-l border-border p-5 xl:block">
|
|
<div className="h-40 animate-pulse rounded-md bg-muted" />
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function TicketWorkbenchContent() {
|
|
const router = useRouter();
|
|
const searchParams = useSearchParams();
|
|
|
|
const [tickets, setTickets] = useState<Ticket[]>([]);
|
|
const [queues, setQueues] = useState<Queue[]>([]);
|
|
const [lifecycles, setLifecycles] = useState<Lifecycle[]>([]);
|
|
const [users, setUsers] = useState<User[]>([]);
|
|
const [customFields, setCustomFields] = useState<CustomField[]>([]);
|
|
const [clock, setClock] = useState(0);
|
|
const [loading, setLoading] = useState(true);
|
|
const [refreshing, setRefreshing] = useState(false);
|
|
const [error, setError] = useState<string | null>(null);
|
|
const [selectedId, setSelectedId] = useState<number | null>(null);
|
|
|
|
const [searchQuery, setSearchQuery] = useState("");
|
|
const [filters, setFilters] = useState<Filter[]>([]);
|
|
const [density, setDensity] = useState<Density>("comfortable");
|
|
const [sortKey, setSortKey] = useState<SortKey>("updated");
|
|
|
|
// Saved views
|
|
const [savedViewsList, setSavedViewsList] = useState<SavedView[]>([]);
|
|
const [viewIdFromUrl, setViewIdFromUrl] = useState<string | null>(null);
|
|
const [saveViewOpen, setSaveViewOpen] = useState(false);
|
|
const [saveViewName, setSaveViewName] = useState("");
|
|
const [addFilterOpen, setAddFilterOpen] = useState(false);
|
|
const [addFilterField, setAddFilterField] = useState<string | null>(null); // which field type is selected
|
|
const [addFilterOperator, setAddFilterOperator] = useState("is");
|
|
const [addFilterValue, setAddFilterValue] = useState("");
|
|
|
|
const [dialogOpen, setDialogOpen] = useState(false);
|
|
const [newSubject, setNewSubject] = useState("");
|
|
const [newQueueId, setNewQueueId] = useState("");
|
|
const [newDescription, setNewDescription] = useState("");
|
|
const [newQueueFields, setNewQueueFields] = useState<QueueCustomField[]>([]);
|
|
const [newCustomFieldValues, setNewCustomFieldValues] = useState<Record<string, string>>({});
|
|
const [newFieldsLoading, setNewFieldsLoading] = useState(false);
|
|
const [submitting, setSubmitting] = useState(false);
|
|
const [createError, setCreateError] = useState<string | null>(null);
|
|
|
|
const view = searchParams.get("view") ?? "all";
|
|
const routeQueue = searchParams.get("queue") ?? "";
|
|
|
|
const fetchData = useCallback(
|
|
async (background = false) => {
|
|
if (background) {
|
|
setRefreshing(true);
|
|
} else {
|
|
setLoading(true);
|
|
}
|
|
setError(null);
|
|
const fetchedAt = Date.now();
|
|
|
|
const activeQueue = routeQueue;
|
|
const apiStatus = filters.find((f) => f.field === "status")?.value;
|
|
const apiOwner = filters.find((f) => f.field === "owner")?.value;
|
|
const apiQueue = filters.find((f) => f.field === "queue")?.value;
|
|
const customFieldFilters: Record<string, string> = {};
|
|
for (const f of filters) {
|
|
if (f.field.startsWith("cf.")) {
|
|
customFieldFilters[f.field.slice(3)] = f.value;
|
|
}
|
|
}
|
|
|
|
const [ticketsRes, queuesRes, usersRes, fieldsRes, lifecycleRes] = await Promise.all([
|
|
getTickets({
|
|
q: searchQuery.trim() || undefined,
|
|
status: apiStatus || undefined,
|
|
queue_id: activeQueue || apiQueue || undefined,
|
|
owner_id: apiOwner || undefined,
|
|
custom_fields: Object.keys(customFieldFilters).length > 0 ? customFieldFilters : undefined,
|
|
}),
|
|
getQueues(),
|
|
getUsers(),
|
|
getCustomFields(),
|
|
getLifecycles(),
|
|
]);
|
|
|
|
if (ticketsRes.error) {
|
|
setError(ticketsRes.error);
|
|
setTickets([]);
|
|
} else {
|
|
setTickets(ticketsRes.data ?? []);
|
|
}
|
|
|
|
if (queuesRes.error) {
|
|
setError((current) => current ?? queuesRes.error);
|
|
} else {
|
|
const queueData = queuesRes.data ?? [];
|
|
setQueues(queueData);
|
|
if (!newQueueId && queueData.length > 0) {
|
|
setNewQueueId(queueData[0].id);
|
|
}
|
|
}
|
|
|
|
if (usersRes.error) {
|
|
setError((current) => current ?? usersRes.error);
|
|
} else {
|
|
setUsers(usersRes.data ?? []);
|
|
}
|
|
|
|
if (fieldsRes.error) {
|
|
setError((current) => current ?? fieldsRes.error);
|
|
} else {
|
|
setCustomFields(fieldsRes.data ?? []);
|
|
}
|
|
|
|
if (lifecycleRes.error) {
|
|
setError((current) => current ?? lifecycleRes.error);
|
|
} else {
|
|
setLifecycles(lifecycleRes.data ?? []);
|
|
}
|
|
|
|
setLoading(false);
|
|
setRefreshing(false);
|
|
setClock(fetchedAt);
|
|
},
|
|
[filters, newQueueId, routeQueue, searchQuery]
|
|
);
|
|
|
|
useEffect(() => {
|
|
void Promise.resolve().then(() => fetchData());
|
|
}, [fetchData]);
|
|
|
|
// Redirect to default dashboard if one exists and no params set
|
|
useEffect(() => {
|
|
if (searchParams.toString()) return;
|
|
getDashboards().then(({ data }) => {
|
|
const def = data?.find((d) => d.is_default);
|
|
if (def) router.replace(`/dashboards/${def.id}`);
|
|
});
|
|
}, [searchParams]);
|
|
|
|
useEffect(() => {
|
|
if (searchParams.get("new") === "true") {
|
|
queueMicrotask(() => setDialogOpen(true));
|
|
const url = new URL(window.location.href);
|
|
url.searchParams.delete("new");
|
|
window.history.replaceState({}, "", url.toString());
|
|
}
|
|
}, [searchParams]);
|
|
|
|
useEffect(() => {
|
|
if (!newQueueId) {
|
|
queueMicrotask(() => {
|
|
setNewQueueFields([]);
|
|
setNewCustomFieldValues({});
|
|
});
|
|
return;
|
|
}
|
|
|
|
let cancelled = false;
|
|
queueMicrotask(() => {
|
|
if (!cancelled) setNewFieldsLoading(true);
|
|
});
|
|
void getQueueCustomFields(newQueueId).then(({ data, error }) => {
|
|
if (cancelled) return;
|
|
setNewFieldsLoading(false);
|
|
if (error) {
|
|
setCreateError(error);
|
|
setNewQueueFields([]);
|
|
setNewCustomFieldValues({});
|
|
return;
|
|
}
|
|
const assignments = data ?? [];
|
|
setNewQueueFields(assignments);
|
|
setNewCustomFieldValues((current) => {
|
|
const next: Record<string, string> = {};
|
|
for (const assignment of assignments) {
|
|
next[assignment.custom_field_id] = current[assignment.custom_field_id] ?? "";
|
|
}
|
|
return next;
|
|
});
|
|
});
|
|
|
|
return () => {
|
|
cancelled = true;
|
|
};
|
|
}, [newQueueId]);
|
|
|
|
// Load saved views list
|
|
useEffect(() => {
|
|
getViews().then(({ data }) => {
|
|
if (data) setSavedViewsList(data);
|
|
});
|
|
}, [clock]);
|
|
|
|
// Load view from URL param
|
|
useEffect(() => {
|
|
const paramViewId = searchParams.get("view_id");
|
|
setViewIdFromUrl(paramViewId);
|
|
if (paramViewId) {
|
|
getViews().then(({ data }) => {
|
|
const view = data?.find((v) => v.id === paramViewId);
|
|
if (view?.filters && Array.isArray(view.filters)) {
|
|
setSearchQuery("");
|
|
setFilters(
|
|
(view.filters as { field: string; operator: string; value: string }[])
|
|
.filter((f) => f.field && f.value)
|
|
.map((f) => ({
|
|
id: crypto.randomUUID(),
|
|
field: f.field,
|
|
operator: f.operator || "is",
|
|
value: f.value,
|
|
label: buildFilterLabel(f.field, f.operator || "is", f.value),
|
|
}))
|
|
);
|
|
if (view.sort_key) setSortKey(view.sort_key as SortKey);
|
|
}
|
|
});
|
|
} else if (!paramViewId && viewIdFromUrl) {
|
|
// User navigated away from a view — clear filters
|
|
setFilters([]);
|
|
setSearchQuery("");
|
|
}
|
|
}, [searchParams]);
|
|
|
|
const statusOptions = useMemo(() => {
|
|
const queueFilterValue = filters.find((f) => f.field === "queue")?.value;
|
|
const selectedFilterQueueId = routeQueue || queueFilterValue || "";
|
|
const selectedFilterQueue = selectedFilterQueueId
|
|
? queues.find((queue) => queue.id === selectedFilterQueueId)
|
|
: null;
|
|
const selectedFilterLifecycle = selectedFilterQueue?.lifecycle_id
|
|
? lifecycles.find((lifecycle) => lifecycle.id === selectedFilterQueue.lifecycle_id)
|
|
: null;
|
|
const lifecycleStatuses = selectedFilterLifecycle
|
|
? [
|
|
...selectedFilterLifecycle.definition.statuses.initial,
|
|
...selectedFilterLifecycle.definition.statuses.active,
|
|
...selectedFilterLifecycle.definition.statuses.inactive,
|
|
]
|
|
: lifecycles.flatMap((lifecycle) => [
|
|
...lifecycle.definition.statuses.initial,
|
|
...lifecycle.definition.statuses.active,
|
|
...lifecycle.definition.statuses.inactive,
|
|
]);
|
|
|
|
return [
|
|
{ key: "all", label: "All" },
|
|
...Array.from(new Set([...lifecycleStatuses, ...tickets.map((ticket) => ticket.status)]))
|
|
.filter(Boolean)
|
|
.map((status) => ({ key: status, label: statusLabel(status) })),
|
|
];
|
|
}, [filters, lifecycles, queues, routeQueue, tickets]);
|
|
|
|
const inactiveStatuses = useMemo(
|
|
() => new Set(
|
|
lifecycles.length > 0
|
|
? lifecycles.flatMap((lifecycle) => lifecycle.definition.statuses.inactive)
|
|
: ["resolved", "closed"]
|
|
),
|
|
[lifecycles]
|
|
);
|
|
|
|
const metrics = useMemo(() => {
|
|
const now = clock || 0;
|
|
const day = 24 * 60 * 60 * 1000;
|
|
return {
|
|
total: tickets.length,
|
|
active: tickets.filter((ticket) => !inactiveStatuses.has(ticket.status)).length,
|
|
unassigned: tickets.filter((ticket) => !ticket.owner_id).length,
|
|
stale: tickets.filter((ticket) => now && now - new Date(ticket.updated_at).getTime() > 3 * day).length,
|
|
recent: tickets.filter((ticket) => now && now - new Date(ticket.updated_at).getTime() < day).length,
|
|
};
|
|
}, [clock, inactiveStatuses, tickets]);
|
|
|
|
const filteredTickets = useMemo(() => {
|
|
const query = searchQuery.trim().toLowerCase();
|
|
const now = clock || 0;
|
|
const queue = routeQueue;
|
|
const statusFilterValue = filters.find((f) => f.field === "status")?.value;
|
|
const queueFilterValue = filters.find((f) => f.field === "queue")?.value;
|
|
|
|
return tickets
|
|
.filter((ticket) => {
|
|
if (view === "my") {
|
|
const myOwner = searchParams.get("owner");
|
|
if (myOwner && ticket.owner_id !== myOwner) return false;
|
|
if (!myOwner && !ticket.owner_id) return false;
|
|
}
|
|
if (view === "unassigned" && ticket.owner_id) return false;
|
|
if (view === "recent") {
|
|
const week = 7 * 24 * 60 * 60 * 1000;
|
|
if (!now || now - new Date(ticket.updated_at).getTime() > week) return false;
|
|
}
|
|
if (statusFilterValue && ticket.status !== statusFilterValue) return false;
|
|
if (queueFilterValue && ticket.queue_id !== queueFilterValue) return false;
|
|
if (queue && ticket.queue_id !== queue) return false;
|
|
if (!query) return true;
|
|
return (
|
|
ticket.subject.toLowerCase().includes(query) ||
|
|
formatTicketId(ticket.id).toLowerCase().includes(query) ||
|
|
statusLabel(ticket.status).toLowerCase().includes(query) ||
|
|
queueName(queues, ticket.queue_id).toLowerCase().includes(query)
|
|
);
|
|
})
|
|
.sort((a, b) => {
|
|
if (sortKey === "id") return b.id - a.id;
|
|
const aDate = sortKey === "created" ? a.created_at : a.updated_at;
|
|
const bDate = sortKey === "created" ? b.created_at : b.updated_at;
|
|
return new Date(bDate).getTime() - new Date(aDate).getTime();
|
|
});
|
|
}, [clock, filters, queues, routeQueue, searchQuery, sortKey, tickets, view]);
|
|
|
|
const selectedTicket =
|
|
filteredTickets.find((ticket) => ticket.id === selectedId) ?? filteredTickets[0] ?? null;
|
|
const visibleTitle = routeQueue
|
|
? queueName(queues, routeQueue)
|
|
: VIEW_LABELS[view] ?? "All tickets";
|
|
const newTicketFields = newQueueFields
|
|
.map((assignment) => assignment.custom_field)
|
|
.filter((field): field is CustomField => Boolean(field));
|
|
const selectedNewQueue = queues.find((queue) => queue.id === newQueueId);
|
|
const selectedNewLifecycle = selectedNewQueue?.lifecycle_id
|
|
? lifecycles.find((lifecycle) => lifecycle.id === selectedNewQueue.lifecycle_id)
|
|
: null;
|
|
const newTicketInitialStatus = selectedNewLifecycle?.definition.statuses.initial[0] ?? "new";
|
|
const hasQueryFilters = searchQuery.trim() || filters.length > 0;
|
|
|
|
const clearQueryFilters = () => {
|
|
setSearchQuery("");
|
|
setFilters([]);
|
|
if (routeQueue) router.push("/");
|
|
};
|
|
|
|
const handleCreate = async () => {
|
|
if (!newSubject.trim() || !newQueueId) return;
|
|
for (const field of newTicketFields) {
|
|
const value = (newCustomFieldValues[field.id] ?? "").trim();
|
|
if (value && field.pattern) {
|
|
const regex = new RegExp(field.pattern);
|
|
if (!regex.test(value)) {
|
|
setCreateError(`${field.name}: value does not match the required pattern.`);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
setSubmitting(true);
|
|
setCreateError(null);
|
|
|
|
const { data, error } = await createTicket({
|
|
subject: newSubject.trim(),
|
|
queue_id: newQueueId,
|
|
description: newDescription.trim() || undefined,
|
|
custom_fields: Object.fromEntries(
|
|
Object.entries(newCustomFieldValues)
|
|
.map(([fieldId, value]) => [fieldId, value.trim()])
|
|
.filter(([, value]) => value)
|
|
),
|
|
});
|
|
|
|
setSubmitting(false);
|
|
|
|
if (error) {
|
|
setCreateError(error);
|
|
return;
|
|
}
|
|
|
|
setDialogOpen(false);
|
|
setNewSubject("");
|
|
setNewDescription("");
|
|
setNewCustomFieldValues({});
|
|
if (data) router.push(`/tickets/${data.ticket.id}`);
|
|
};
|
|
|
|
if (loading) return <SkeletonWorkbench />;
|
|
|
|
return (
|
|
<div className="flex h-full flex-col bg-background/80">
|
|
<header className="shrink-0 border-b border-border bg-card/82 backdrop-blur">
|
|
<div className="flex flex-col gap-3 px-5 py-4 lg:px-6">
|
|
<div className="flex flex-col gap-3 lg:flex-row lg:items-center lg:justify-between">
|
|
<div className="min-w-0">
|
|
<div className="flex items-center gap-2 text-[11px] font-semibold uppercase text-muted-foreground">
|
|
<GaugeIcon className="h-3.5 w-3.5" />
|
|
Work queue
|
|
</div>
|
|
<h1 className="mt-1 truncate text-2xl font-semibold text-foreground">
|
|
{visibleTitle}
|
|
</h1>
|
|
</div>
|
|
<div className="flex flex-wrap items-center gap-2">
|
|
<Button
|
|
variant="outline"
|
|
size="sm"
|
|
onClick={() => fetchData(true)}
|
|
disabled={refreshing}
|
|
className="h-8 border-border/80 bg-card/70"
|
|
>
|
|
<RefreshCwIcon className={cn("h-4 w-4", refreshing && "animate-spin")} />
|
|
Refresh
|
|
</Button>
|
|
{filters.length > 0 && (
|
|
<Button
|
|
size="sm"
|
|
variant="outline"
|
|
onClick={() => {
|
|
setSaveViewName("");
|
|
setSaveViewOpen(true);
|
|
}}
|
|
className="h-8 border-border/80 bg-card/70"
|
|
>
|
|
<SaveIcon className="h-4 w-4" />
|
|
Save view
|
|
</Button>
|
|
)}
|
|
<Button size="sm" onClick={() => setDialogOpen(true)} className="h-8 bg-primary shadow-sm">
|
|
<PlusIcon className="h-4 w-4" />
|
|
New ticket
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="flex flex-wrap items-center gap-2">
|
|
<StatPill label="total" value={metrics.total} />
|
|
<StatPill label="active" value={metrics.active} />
|
|
<StatPill label="unassigned" value={metrics.unassigned} />
|
|
<StatPill label="updated today" value={metrics.recent} />
|
|
<StatPill label="needs review" value={metrics.stale} />
|
|
</div>
|
|
|
|
<div className="flex flex-col gap-2">
|
|
{/* Row 1: search + sort/density */}
|
|
<div className="flex items-center gap-2">
|
|
<div className="relative flex-1">
|
|
<SearchIcon className="absolute left-3 top-1/2 h-4 w-4 -translate-y-1/2 text-muted-foreground" />
|
|
<input
|
|
value={searchQuery}
|
|
onChange={(event) => setSearchQuery(event.target.value)}
|
|
placeholder="Search subject, ticket ID, queue, or status"
|
|
className="h-9 w-full rounded-md border border-input bg-card/90 pl-9 pr-3 text-sm shadow-sm outline-none transition-colors placeholder:text-muted-foreground focus:border-ring"
|
|
/>
|
|
</div>
|
|
{hasQueryFilters && (
|
|
<Button
|
|
type="button"
|
|
variant="outline"
|
|
size="sm"
|
|
onClick={clearQueryFilters}
|
|
className="h-9 shrink-0 border-border/80 bg-card/70"
|
|
>
|
|
Clear
|
|
</Button>
|
|
)}
|
|
<button
|
|
type="button"
|
|
onClick={() => setSortKey(sortKey === "updated" ? "created" : sortKey === "created" ? "id" : "updated")}
|
|
className="inline-flex h-9 items-center gap-2 rounded-md border border-border bg-card/90 px-3 text-sm text-muted-foreground shadow-sm transition-colors hover:bg-accent hover:text-foreground shrink-0"
|
|
title="Change sort"
|
|
>
|
|
<ArrowDownAZIcon className="h-4 w-4" />
|
|
{sortKey === "updated" ? "Updated" : sortKey === "created" ? "Created" : "ID"}
|
|
</button>
|
|
<button
|
|
type="button"
|
|
onClick={() => setDensity(density === "comfortable" ? "compact" : "comfortable")}
|
|
className="inline-flex h-9 w-9 shrink-0 items-center justify-center rounded-md border border-border bg-card/90 text-muted-foreground shadow-sm transition-colors hover:bg-accent hover:text-foreground"
|
|
title="Toggle density"
|
|
>
|
|
{density === "comfortable" ? (
|
|
<LayoutListIcon className="h-4 w-4" />
|
|
) : (
|
|
<SlidersHorizontalIcon className="h-4 w-4" />
|
|
)}
|
|
</button>
|
|
</div>
|
|
|
|
{/* Row 2: status quick-filter pills */}
|
|
<div className="flex min-w-0 items-center gap-1 overflow-x-auto rounded-md border border-border bg-muted/55 p-1 shadow-sm">
|
|
{statusOptions.map((opt) => {
|
|
const isActive = opt.key === "all"
|
|
? !filters.some((f) => f.field === "status")
|
|
: filters.some((f) => f.field === "status" && f.value === opt.key);
|
|
return (
|
|
<button
|
|
key={opt.key}
|
|
type="button"
|
|
onClick={() => {
|
|
if (opt.key === "all") {
|
|
setFilters((prev) => prev.filter((f) => f.field !== "status"));
|
|
} else {
|
|
setFilters((prev) => {
|
|
const rest = prev.filter((f) => f.field !== "status");
|
|
if (isActive) return rest;
|
|
return [...rest, {
|
|
id: crypto.randomUUID(),
|
|
field: "status",
|
|
operator: "is",
|
|
value: opt.key,
|
|
label: buildFilterLabel("status", "is", opt.label),
|
|
}];
|
|
});
|
|
}
|
|
}}
|
|
className={cn(
|
|
"h-7 whitespace-nowrap rounded px-2.5 text-xs font-semibold transition-colors",
|
|
isActive
|
|
? "bg-card text-foreground shadow-sm"
|
|
: "text-muted-foreground hover:bg-accent hover:text-foreground"
|
|
)}
|
|
>
|
|
{opt.label}
|
|
</button>
|
|
);
|
|
})}
|
|
</div>
|
|
|
|
{/* Row 3: active filter chips + add filter */}
|
|
<div className="flex flex-wrap items-center gap-1.5">
|
|
{filters
|
|
.filter((f) => f.field !== "status")
|
|
.map((f) => (
|
|
<span
|
|
key={f.id}
|
|
className="inline-flex h-7 items-center gap-1.5 rounded border border-border bg-accent/60 px-2 text-xs font-medium text-foreground"
|
|
>
|
|
{f.label}
|
|
<button
|
|
type="button"
|
|
onClick={() => setFilters((prev) => prev.filter((x) => x.id !== f.id))}
|
|
className="ml-0.5 inline-flex h-4 w-4 items-center justify-center rounded-sm text-muted-foreground hover:text-foreground"
|
|
>
|
|
<XIcon className="h-3 w-3" />
|
|
</button>
|
|
</span>
|
|
))}
|
|
<div>
|
|
<button
|
|
type="button"
|
|
onClick={(event) => {
|
|
const rect = event.currentTarget.getBoundingClientRect();
|
|
const popover = document.getElementById("add-filter-popover");
|
|
if (popover) {
|
|
popover.style.left = `${rect.left}px`;
|
|
popover.style.top = `${rect.bottom + 4}px`;
|
|
}
|
|
setAddFilterOpen((prev) => !prev);
|
|
}}
|
|
className="inline-flex h-7 items-center gap-1 rounded border border-dashed border-border px-2 text-xs font-medium text-muted-foreground hover:border-ring hover:text-foreground transition-colors"
|
|
>
|
|
<PlusIcon className="h-3 w-3" />
|
|
Add filter
|
|
</button>
|
|
{addFilterOpen && (
|
|
<>
|
|
<div
|
|
className="fixed inset-0 z-40"
|
|
onClick={() => {
|
|
setAddFilterOpen(false);
|
|
setAddFilterField(null);
|
|
}}
|
|
/>
|
|
<div
|
|
id="add-filter-popover"
|
|
className="fixed z-50 w-52 rounded-md border border-border bg-popover p-1 shadow-lg"
|
|
>
|
|
{!addFilterField ? (
|
|
/* Step 1: choose field */
|
|
<>
|
|
<button
|
|
type="button"
|
|
className="w-full rounded px-2 py-1.5 text-left text-xs text-foreground hover:bg-accent"
|
|
onClick={() => {
|
|
if (!filters.find((f) => f.field === "queue")) {
|
|
setAddFilterField("queue");
|
|
setAddFilterOperator("is");
|
|
setAddFilterValue("");
|
|
} else {
|
|
setAddFilterOpen(false);
|
|
}
|
|
}}
|
|
>
|
|
Queue
|
|
</button>
|
|
<button
|
|
type="button"
|
|
className="w-full rounded px-2 py-1.5 text-left text-xs text-foreground hover:bg-accent"
|
|
onClick={() => {
|
|
if (!filters.find((f) => f.field === "owner")) {
|
|
setAddFilterField("owner");
|
|
setAddFilterOperator("is");
|
|
setAddFilterValue("");
|
|
} else {
|
|
setAddFilterOpen(false);
|
|
}
|
|
}}
|
|
>
|
|
Owner
|
|
</button>
|
|
{customFields.map((cf) => (
|
|
<button
|
|
key={`cf-field-${cf.id}`}
|
|
type="button"
|
|
className="w-full rounded px-2 py-1.5 text-left text-xs text-foreground hover:bg-accent"
|
|
onClick={() => {
|
|
setAddFilterField(`cf.${cf.key}`);
|
|
setAddFilterOperator("is");
|
|
setAddFilterValue("");
|
|
}}
|
|
>
|
|
{cf.name}
|
|
</button>
|
|
))}
|
|
</>
|
|
) : (
|
|
/* Step 2: operator + value */
|
|
<div className="space-y-2 p-1">
|
|
<div className="flex items-center gap-1 text-xs">
|
|
<button
|
|
type="button"
|
|
onClick={() => {
|
|
setAddFilterField(null);
|
|
}}
|
|
className="text-muted-foreground hover:text-foreground"
|
|
>
|
|
←
|
|
</button>
|
|
<span className="font-medium text-foreground">
|
|
{addFilterField.startsWith("cf.") ? addFilterField.slice(3) : addFilterField}
|
|
</span>
|
|
</div>
|
|
<select
|
|
value={addFilterOperator}
|
|
onChange={(e) => setAddFilterOperator(e.target.value)}
|
|
className="h-7 w-full rounded border border-input bg-card px-2 text-xs outline-none"
|
|
>
|
|
<option value="is">is</option>
|
|
<option value="is_not">is not</option>
|
|
</select>
|
|
{addFilterField === "queue" ? (
|
|
<select
|
|
value={addFilterValue}
|
|
onChange={(e) => setAddFilterValue(e.target.value)}
|
|
className="h-7 w-full rounded border border-input bg-card px-2 text-xs outline-none"
|
|
>
|
|
<option value="">Select queue...</option>
|
|
{queues.map((q) => (
|
|
<option key={q.id} value={q.id}>{q.name}</option>
|
|
))}
|
|
</select>
|
|
) : addFilterField === "owner" ? (
|
|
<select
|
|
value={addFilterValue}
|
|
onChange={(e) => setAddFilterValue(e.target.value)}
|
|
className="h-7 w-full rounded border border-input bg-card px-2 text-xs outline-none"
|
|
>
|
|
<option value="">Select owner...</option>
|
|
<option value="unassigned">Unassigned</option>
|
|
{users.map((u) => (
|
|
<option key={u.id} value={u.id}>{u.username}</option>
|
|
))}
|
|
</select>
|
|
) : (
|
|
<input
|
|
value={addFilterValue}
|
|
onChange={(e) => setAddFilterValue(e.target.value)}
|
|
placeholder="Value"
|
|
className="h-7 w-full rounded border border-input bg-card px-2 text-xs outline-none"
|
|
onKeyDown={(e) => {
|
|
if (e.key === "Enter" && addFilterValue.trim()) {
|
|
const field = addFilterField;
|
|
const value = addFilterValue;
|
|
let valueLabel = value;
|
|
setFilters((prev) => [...prev, {
|
|
id: crypto.randomUUID(),
|
|
field,
|
|
operator: addFilterOperator,
|
|
value,
|
|
label: buildFilterLabel(field, addFilterOperator, valueLabel),
|
|
}]);
|
|
setAddFilterField(null);
|
|
setAddFilterOpen(false);
|
|
}
|
|
}}
|
|
/>
|
|
)}
|
|
<div className="flex items-center justify-end gap-1 pt-1">
|
|
<button
|
|
type="button"
|
|
onClick={() => {
|
|
setAddFilterField(null);
|
|
setAddFilterOpen(false);
|
|
}}
|
|
className="rounded px-2 py-1 text-xs text-muted-foreground hover:text-foreground"
|
|
>
|
|
Cancel
|
|
</button>
|
|
<button
|
|
type="button"
|
|
disabled={!addFilterValue.trim()}
|
|
onClick={() => {
|
|
if (!addFilterValue.trim()) return;
|
|
const field = addFilterField;
|
|
const value = addFilterValue;
|
|
const operator = addFilterOperator;
|
|
let valueLabel = value;
|
|
if (field === "queue") {
|
|
valueLabel = queues.find((q) => q.id === value)?.name ?? value;
|
|
} else if (field === "owner") {
|
|
valueLabel = value === "unassigned" ? "Unassigned"
|
|
: users.find((u) => u.id === value)?.username ?? value;
|
|
}
|
|
setFilters((prev) => [...prev, {
|
|
id: crypto.randomUUID(),
|
|
field,
|
|
operator,
|
|
value,
|
|
label: buildFilterLabel(field, operator, valueLabel),
|
|
}]);
|
|
setAddFilterField(null);
|
|
setAddFilterOpen(false);
|
|
}}
|
|
className="rounded bg-primary px-2 py-1 text-xs font-medium text-primary-foreground hover:bg-primary/90 disabled:opacity-50"
|
|
>
|
|
Apply
|
|
</button>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</>
|
|
)}
|
|
</div>
|
|
</div>
|
|
|
|
</div>
|
|
</div>
|
|
</header>
|
|
|
|
{error && (
|
|
<div className="shrink-0 border-b border-destructive/20 bg-destructive/10 px-5 py-2 text-sm text-destructive">
|
|
{error}
|
|
</div>
|
|
)}
|
|
|
|
<div className="grid min-h-0 flex-1 grid-cols-1 xl:grid-cols-[minmax(0,1fr)_372px]">
|
|
<section className="min-w-0 overflow-auto bg-card/48">
|
|
<div className="min-w-[760px]">
|
|
<div className="sticky top-0 z-10 grid h-9 grid-cols-[112px_minmax(280px,1fr)_150px_132px_116px_40px] items-center border-b border-border bg-muted/80 px-5 text-[11px] font-semibold uppercase text-muted-foreground backdrop-blur">
|
|
<span>Ticket</span>
|
|
<span>Subject</span>
|
|
<span>Queue</span>
|
|
<span>Status</span>
|
|
<span>Updated</span>
|
|
<span />
|
|
</div>
|
|
|
|
{filteredTickets.length === 0 ? (
|
|
<div className="flex min-h-80 flex-col items-center justify-center px-5 text-center">
|
|
<div className="mb-3 flex h-11 w-11 items-center justify-center rounded-md border border-border bg-background">
|
|
<SearchIcon className="h-5 w-5 text-muted-foreground" />
|
|
</div>
|
|
<p className="text-sm font-medium text-foreground">No tickets in this view</p>
|
|
<p className="mt-1 max-w-sm text-sm text-muted-foreground">
|
|
Adjust the filters or create a new ticket for this queue.
|
|
</p>
|
|
<Button onClick={() => setDialogOpen(true)} className="mt-4 h-8" size="sm">
|
|
<PlusIcon className="h-4 w-4" />
|
|
New ticket
|
|
</Button>
|
|
</div>
|
|
) : (
|
|
filteredTickets.map((ticket) => {
|
|
const selected = ticket.id === selectedId;
|
|
return (
|
|
<button
|
|
key={ticket.id}
|
|
type="button"
|
|
onClick={() => setSelectedId(ticket.id)}
|
|
onDoubleClick={() => router.push(`/tickets/${ticket.id}`)}
|
|
className={cn(
|
|
"grid w-full grid-cols-[112px_minmax(280px,1fr)_150px_132px_116px_40px] items-center border-b border-border/80 px-5 text-left transition-colors",
|
|
density === "compact" ? "min-h-11" : "min-h-16",
|
|
selected
|
|
? "bg-accent/80 shadow-[inset_3px_0_0_var(--primary)]"
|
|
: "hover:bg-accent/45"
|
|
)}
|
|
>
|
|
<span className="font-mono text-xs font-semibold text-muted-foreground">
|
|
{formatTicketId(ticket.id)}
|
|
</span>
|
|
<span className="min-w-0 pr-4">
|
|
<span className="block truncate text-sm font-semibold text-foreground">
|
|
{ticket.subject}
|
|
</span>
|
|
{density === "comfortable" && (
|
|
<span className="mt-1 flex items-center gap-2 text-xs text-muted-foreground">
|
|
{ticket.owner_id ? "Assigned" : "Unassigned"}
|
|
<span className="h-1 w-1 rounded-full bg-border" />
|
|
Created {relativeTime(ticket.created_at)}
|
|
</span>
|
|
)}
|
|
</span>
|
|
<span className="truncate text-sm font-medium text-muted-foreground">
|
|
{queueName(queues, ticket.queue_id)}
|
|
</span>
|
|
<span>
|
|
<TicketStatusBadge status={ticket.status} />
|
|
</span>
|
|
<span className="text-xs text-muted-foreground">
|
|
{relativeTime(ticket.updated_at)}
|
|
</span>
|
|
<span className="flex justify-end text-muted-foreground">
|
|
<ChevronRightIcon className="h-4 w-4" />
|
|
</span>
|
|
</button>
|
|
);
|
|
})
|
|
)}
|
|
</div>
|
|
</section>
|
|
|
|
<aside className="hidden min-h-0 border-l border-border bg-card/76 backdrop-blur xl:flex xl:flex-col">
|
|
{selectedTicket ? (
|
|
<>
|
|
<div className="border-b border-border bg-card/82 p-5">
|
|
<div className="flex items-start justify-between gap-3">
|
|
<div className="min-w-0">
|
|
<p className="font-mono text-xs font-semibold text-muted-foreground">
|
|
{formatTicketId(selectedTicket.id)}
|
|
</p>
|
|
<h2 className="mt-2 text-lg font-semibold leading-snug text-foreground">
|
|
{selectedTicket.subject}
|
|
</h2>
|
|
</div>
|
|
<TicketStatusBadge status={selectedTicket.status} />
|
|
</div>
|
|
<Button
|
|
onClick={() => router.push(`/tickets/${selectedTicket.id}`)}
|
|
className="mt-4 h-8 w-full bg-primary"
|
|
size="sm"
|
|
>
|
|
Open ticket
|
|
<ChevronRightIcon className="h-4 w-4" />
|
|
</Button>
|
|
</div>
|
|
|
|
<div className="flex-1 overflow-auto p-5">
|
|
<dl className="grid overflow-hidden rounded-md border border-border bg-background/55 text-sm">
|
|
<div className="grid grid-cols-[104px_minmax(0,1fr)] border-b border-border px-3 py-2.5">
|
|
<dt className="text-xs font-semibold uppercase text-muted-foreground">Queue</dt>
|
|
<dd className="truncate text-foreground">{queueName(queues, selectedTicket.queue_id)}</dd>
|
|
</div>
|
|
<div className="grid grid-cols-[104px_minmax(0,1fr)] border-b border-border px-3 py-2.5">
|
|
<dt className="text-xs font-semibold uppercase text-muted-foreground">Owner</dt>
|
|
<dd className="truncate text-foreground">{selectedTicket.owner_id ?? "Unassigned"}</dd>
|
|
</div>
|
|
<div className="grid grid-cols-[104px_minmax(0,1fr)] border-b border-border px-3 py-2.5">
|
|
<dt className="text-xs font-semibold uppercase text-muted-foreground">Creator</dt>
|
|
<dd className="truncate text-foreground">{selectedTicket.creator_id}</dd>
|
|
</div>
|
|
<div className="grid grid-cols-[104px_minmax(0,1fr)] border-b border-border px-3 py-2.5">
|
|
<dt className="text-xs font-semibold uppercase text-muted-foreground">Created</dt>
|
|
<dd className="truncate text-foreground">{new Date(selectedTicket.created_at).toLocaleString()}</dd>
|
|
</div>
|
|
<div className="grid grid-cols-[104px_minmax(0,1fr)] px-3 py-2.5">
|
|
<dt className="text-xs font-semibold uppercase text-muted-foreground">Updated</dt>
|
|
<dd className="truncate text-foreground">{new Date(selectedTicket.updated_at).toLocaleString()}</dd>
|
|
</div>
|
|
</dl>
|
|
|
|
<div className="mt-5 rounded-md border border-border bg-accent/42 p-4">
|
|
<div className="flex items-center gap-2 text-sm font-medium text-foreground">
|
|
<CheckCircle2Icon className="h-4 w-4 text-emerald-600" />
|
|
Next action
|
|
</div>
|
|
<p className="mt-2 text-sm text-muted-foreground">
|
|
Open the ticket to reply, change status, and review its transaction history.
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</>
|
|
) : (
|
|
<div className="flex flex-1 items-center justify-center p-6 text-center text-sm text-muted-foreground">
|
|
Select a ticket to inspect it.
|
|
</div>
|
|
)}
|
|
</aside>
|
|
</div>
|
|
|
|
<Dialog open={dialogOpen} onOpenChange={setDialogOpen}>
|
|
<DialogContent className="sm:max-w-2xl" showCloseButton>
|
|
<DialogHeader>
|
|
<DialogTitle>New ticket</DialogTitle>
|
|
<DialogDescription>
|
|
Create a request, incident, task, or follow-up for the right queue.
|
|
</DialogDescription>
|
|
</DialogHeader>
|
|
|
|
<div className="grid gap-3">
|
|
<label className="grid gap-1.5 text-sm font-medium">
|
|
Subject
|
|
<input
|
|
value={newSubject}
|
|
onChange={(event) => setNewSubject(event.target.value)}
|
|
placeholder="Short summary"
|
|
className="h-9 rounded-md border border-input bg-background px-3 text-sm font-normal outline-none focus:border-ring"
|
|
autoFocus
|
|
/>
|
|
</label>
|
|
|
|
<label className="grid gap-1.5 text-sm font-medium">
|
|
Queue
|
|
<select
|
|
value={newQueueId}
|
|
onChange={(event) => {
|
|
setNewQueueId(event.target.value);
|
|
setCreateError(null);
|
|
}}
|
|
className="h-9 rounded-md border border-input bg-background px-3 text-sm font-normal outline-none focus:border-ring"
|
|
>
|
|
<option value="" disabled>
|
|
Select queue
|
|
</option>
|
|
{queues.map((queue) => (
|
|
<option key={queue.id} value={queue.id}>
|
|
{queue.name}
|
|
</option>
|
|
))}
|
|
</select>
|
|
<span className="text-xs font-normal text-muted-foreground">
|
|
Starts as <span className="font-mono text-foreground">{newTicketInitialStatus}</span>
|
|
{selectedNewLifecycle ? ` from ${selectedNewLifecycle.name}` : ""}
|
|
</span>
|
|
</label>
|
|
|
|
<label className="grid gap-1.5 text-sm font-medium">
|
|
Description
|
|
<textarea
|
|
value={newDescription}
|
|
onChange={(event) => setNewDescription(event.target.value)}
|
|
placeholder="Optional context"
|
|
rows={4}
|
|
className="resize-none rounded-md border border-input bg-background px-3 py-2 text-sm font-normal outline-none focus:border-ring"
|
|
/>
|
|
</label>
|
|
|
|
{(newFieldsLoading || newTicketFields.length > 0) && (
|
|
<div className="rounded-md border border-border bg-muted/25">
|
|
<div className="border-b border-border px-3 py-2">
|
|
<div className="text-sm font-semibold text-foreground">Queue fields</div>
|
|
<div className="mt-0.5 text-xs text-muted-foreground">
|
|
Metadata attached to this ticket from the selected queue.
|
|
</div>
|
|
</div>
|
|
<div className="grid gap-3 p-3">
|
|
{newFieldsLoading ? (
|
|
<div className="h-9 animate-pulse rounded-md bg-muted" />
|
|
) : (
|
|
newTicketFields.map((field) => {
|
|
const options = Array.isArray(field.values)
|
|
? field.values.map((value) => String(value))
|
|
: [];
|
|
return (
|
|
<label key={field.id} className="grid gap-1.5 text-sm font-medium">
|
|
<span className="flex min-w-0 items-center justify-between gap-3">
|
|
<span className="truncate">{field.name}</span>
|
|
<span className="shrink-0 font-mono text-[11px] font-normal text-muted-foreground">{field.key}</span>
|
|
</span>
|
|
{options.length > 0 ? (
|
|
<select
|
|
value={newCustomFieldValues[field.id] ?? ""}
|
|
onChange={(event) => setNewCustomFieldValues((current) => ({
|
|
...current,
|
|
[field.id]: event.target.value,
|
|
}))}
|
|
className="h-9 rounded-md border border-input bg-background px-3 text-sm font-normal outline-none focus:border-ring"
|
|
>
|
|
<option value="">No value</option>
|
|
{options.map((option) => (
|
|
<option key={option} value={option}>{option}</option>
|
|
))}
|
|
</select>
|
|
) : (
|
|
<input
|
|
value={newCustomFieldValues[field.id] ?? ""}
|
|
onChange={(event) => setNewCustomFieldValues((current) => ({
|
|
...current,
|
|
[field.id]: event.target.value,
|
|
}))}
|
|
placeholder={field.pattern ? `Pattern: ${field.pattern}` : "Optional value"}
|
|
className="h-9 rounded-md border border-input bg-background px-3 text-sm font-normal outline-none focus:border-ring"
|
|
/>
|
|
)}
|
|
{field.pattern && (
|
|
<span className="text-[11px] text-muted-foreground">
|
|
Must match: <code className="rounded bg-muted px-1 font-mono">{field.pattern}</code>
|
|
</span>
|
|
)}
|
|
</label>
|
|
);
|
|
})
|
|
)}
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
{createError && <p className="text-sm text-destructive">{createError}</p>}
|
|
|
|
<DialogFooter showCloseButton={false}>
|
|
<Button
|
|
onClick={handleCreate}
|
|
disabled={!newSubject.trim() || !newQueueId || submitting}
|
|
>
|
|
{submitting ? "Creating..." : "Create ticket"}
|
|
</Button>
|
|
</DialogFooter>
|
|
</DialogContent>
|
|
</Dialog>
|
|
|
|
<Dialog open={saveViewOpen} onOpenChange={setSaveViewOpen}>
|
|
<DialogContent>
|
|
<DialogHeader>
|
|
<DialogTitle>Save view</DialogTitle>
|
|
<DialogDescription>
|
|
Save the current filters as a named view. It will appear in your sidebar.
|
|
</DialogDescription>
|
|
</DialogHeader>
|
|
<input
|
|
value={saveViewName}
|
|
onChange={(event) => setSaveViewName(event.target.value)}
|
|
placeholder="View name"
|
|
className="h-9 w-full rounded-md border border-input bg-card px-3 text-sm outline-none focus:border-ring"
|
|
autoFocus
|
|
onKeyDown={async (event) => {
|
|
if (event.key === "Enter" && saveViewName.trim()) {
|
|
const storedFilters = filters.map((f) => ({
|
|
field: f.field,
|
|
operator: f.operator,
|
|
value: f.value,
|
|
}));
|
|
const { data, error } = await createView({
|
|
name: saveViewName.trim(),
|
|
filters: storedFilters,
|
|
sort_key: sortKey,
|
|
});
|
|
if (!error && data) {
|
|
setSavedViewsList((prev) => [...prev, data]);
|
|
setSaveViewOpen(false);
|
|
setSaveViewName("");
|
|
}
|
|
}
|
|
}}
|
|
/>
|
|
<DialogFooter>
|
|
<Button
|
|
variant="outline"
|
|
size="sm"
|
|
onClick={() => setSaveViewOpen(false)}
|
|
>
|
|
Cancel
|
|
</Button>
|
|
<Button
|
|
size="sm"
|
|
disabled={!saveViewName.trim()}
|
|
onClick={async () => {
|
|
if (!saveViewName.trim()) return;
|
|
const storedFilters = filters.map((f) => ({
|
|
field: f.field,
|
|
operator: f.operator,
|
|
value: f.value,
|
|
}));
|
|
const { data, error } = await createView({
|
|
name: saveViewName.trim(),
|
|
filters: storedFilters,
|
|
sort_key: sortKey,
|
|
});
|
|
if (!error && data) {
|
|
setSavedViewsList((prev) => [...prev, data]);
|
|
setSaveViewOpen(false);
|
|
setSaveViewName("");
|
|
}
|
|
}}
|
|
>
|
|
Save
|
|
</Button>
|
|
</DialogFooter>
|
|
</DialogContent>
|
|
</Dialog>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default function TicketWorkbenchPage() {
|
|
return (
|
|
<Suspense fallback={<SkeletonWorkbench />}>
|
|
<TicketWorkbenchContent />
|
|
</Suspense>
|
|
);
|
|
}
|