| "use client"; |
|
|
| import { Button } from "@/components/ui/button"; |
| import { |
| Collapsible, |
| CollapsibleContent, |
| CollapsibleTrigger, |
| } from "@/components/ui/collapsible"; |
| import { ScrollArea } from "@/components/ui/scroll-area"; |
| import { cn } from "@/lib/utils"; |
| import { ChevronDownIcon, PaperclipIcon } from "lucide-react"; |
| import type { ComponentProps } from "react"; |
|
|
| export type QueueMessagePart = { |
| type: string; |
| text?: string; |
| url?: string; |
| filename?: string; |
| mediaType?: string; |
| }; |
|
|
| export type QueueMessage = { |
| id: string; |
| parts: QueueMessagePart[]; |
| }; |
|
|
| export type QueueTodo = { |
| id: string; |
| title: string; |
| description?: string; |
| status?: "pending" | "completed"; |
| }; |
|
|
| export type QueueItemProps = ComponentProps<"li">; |
|
|
| export const QueueItem = ({ className, ...props }: QueueItemProps) => ( |
| <li |
| className={cn( |
| "group flex flex-col gap-1 rounded-md px-3 py-1 text-sm transition-colors hover:bg-muted", |
| className |
| )} |
| {...props} |
| /> |
| ); |
|
|
| export type QueueItemIndicatorProps = ComponentProps<"span"> & { |
| completed?: boolean; |
| }; |
|
|
| export const QueueItemIndicator = ({ |
| completed = false, |
| className, |
| ...props |
| }: QueueItemIndicatorProps) => ( |
| <span |
| className={cn( |
| "mt-0.5 inline-block size-2.5 rounded-full border", |
| completed |
| ? "border-muted-foreground/20 bg-muted-foreground/10" |
| : "border-muted-foreground/50", |
| className |
| )} |
| {...props} |
| /> |
| ); |
|
|
| export type QueueItemContentProps = ComponentProps<"span"> & { |
| completed?: boolean; |
| }; |
|
|
| export const QueueItemContent = ({ |
| completed = false, |
| className, |
| ...props |
| }: QueueItemContentProps) => ( |
| <span |
| className={cn( |
| "line-clamp-1 grow break-words", |
| completed |
| ? "text-muted-foreground/50 line-through" |
| : "text-muted-foreground", |
| className |
| )} |
| {...props} |
| /> |
| ); |
|
|
| export type QueueItemDescriptionProps = ComponentProps<"div"> & { |
| completed?: boolean; |
| }; |
|
|
| export const QueueItemDescription = ({ |
| completed = false, |
| className, |
| ...props |
| }: QueueItemDescriptionProps) => ( |
| <div |
| className={cn( |
| "ml-6 text-xs", |
| completed |
| ? "text-muted-foreground/40 line-through" |
| : "text-muted-foreground", |
| className |
| )} |
| {...props} |
| /> |
| ); |
|
|
| export type QueueItemActionsProps = ComponentProps<"div">; |
|
|
| export const QueueItemActions = ({ |
| className, |
| ...props |
| }: QueueItemActionsProps) => ( |
| <div className={cn("flex gap-1", className)} {...props} /> |
| ); |
|
|
| export type QueueItemActionProps = Omit< |
| ComponentProps<typeof Button>, |
| "variant" | "size" |
| >; |
|
|
| export const QueueItemAction = ({ |
| className, |
| ...props |
| }: QueueItemActionProps) => ( |
| <Button |
| className={cn( |
| "size-auto rounded p-1 text-muted-foreground opacity-0 transition-opacity hover:bg-muted-foreground/10 hover:text-foreground group-hover:opacity-100", |
| className |
| )} |
| size="icon" |
| type="button" |
| variant="ghost" |
| {...props} |
| /> |
| ); |
|
|
| export type QueueItemAttachmentProps = ComponentProps<"div">; |
|
|
| export const QueueItemAttachment = ({ |
| className, |
| ...props |
| }: QueueItemAttachmentProps) => ( |
| <div className={cn("mt-1 flex flex-wrap gap-2", className)} {...props} /> |
| ); |
|
|
| export type QueueItemImageProps = ComponentProps<"img">; |
|
|
| export const QueueItemImage = ({ |
| className, |
| ...props |
| }: QueueItemImageProps) => ( |
| <img |
| alt="" |
| className={cn("h-8 w-8 rounded border object-cover", className)} |
| height={32} |
| width={32} |
| {...props} |
| /> |
| ); |
|
|
| export type QueueItemFileProps = ComponentProps<"span">; |
|
|
| export const QueueItemFile = ({ |
| children, |
| className, |
| ...props |
| }: QueueItemFileProps) => ( |
| <span |
| className={cn( |
| "flex items-center gap-1 rounded border bg-muted px-2 py-1 text-xs", |
| className |
| )} |
| {...props} |
| > |
| <PaperclipIcon size={12} /> |
| <span className="max-w-[100px] truncate">{children}</span> |
| </span> |
| ); |
|
|
| export type QueueListProps = ComponentProps<typeof ScrollArea>; |
|
|
| export const QueueList = ({ |
| children, |
| className, |
| ...props |
| }: QueueListProps) => ( |
| <ScrollArea className={cn("-mb-1 mt-2", className)} {...props}> |
| <div className="max-h-40 pr-4"> |
| <ul>{children}</ul> |
| </div> |
| </ScrollArea> |
| ); |
|
|
| |
| export type QueueSectionProps = ComponentProps<typeof Collapsible>; |
|
|
| export const QueueSection = ({ |
| className, |
| defaultOpen = true, |
| ...props |
| }: QueueSectionProps) => ( |
| <Collapsible className={cn(className)} defaultOpen={defaultOpen} {...props} /> |
| ); |
|
|
| |
| export type QueueSectionTriggerProps = ComponentProps<"button">; |
|
|
| export const QueueSectionTrigger = ({ |
| children, |
| className, |
| ...props |
| }: QueueSectionTriggerProps) => ( |
| <CollapsibleTrigger asChild> |
| <button |
| className={cn( |
| "group flex w-full items-center justify-between rounded-md bg-muted/40 px-3 py-2 text-left font-medium text-muted-foreground text-sm transition-colors hover:bg-muted", |
| className |
| )} |
| type="button" |
| {...props} |
| > |
| {children} |
| </button> |
| </CollapsibleTrigger> |
| ); |
|
|
| |
| export type QueueSectionLabelProps = ComponentProps<"span"> & { |
| count?: number; |
| label: string; |
| icon?: React.ReactNode; |
| }; |
|
|
| export const QueueSectionLabel = ({ |
| count, |
| label, |
| icon, |
| className, |
| ...props |
| }: QueueSectionLabelProps) => ( |
| <span className={cn("flex items-center gap-2", className)} {...props}> |
| <ChevronDownIcon className="group-data-[state=closed]:-rotate-90 size-4 transition-transform" /> |
| {icon} |
| <span> |
| {count} {label} |
| </span> |
| </span> |
| ); |
|
|
| |
| export type QueueSectionContentProps = ComponentProps< |
| typeof CollapsibleContent |
| >; |
|
|
| export const QueueSectionContent = ({ |
| className, |
| ...props |
| }: QueueSectionContentProps) => ( |
| <CollapsibleContent className={cn(className)} {...props} /> |
| ); |
|
|
| export type QueueProps = ComponentProps<"div">; |
|
|
| export const Queue = ({ className, ...props }: QueueProps) => ( |
| <div |
| className={cn( |
| "flex flex-col gap-2 rounded-xl border border-border bg-background px-3 pt-2 pb-2 shadow-xs", |
| className |
| )} |
| {...props} |
| /> |
| ); |
|
|