|
|
|
|
| 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} |
| /> |
| ); |
|
|