sheet.tsx 4.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788
  1. import * as SheetPrimitive from '@radix-ui/react-dialog';
  2. import { XIcon } from 'lucide-react';
  3. import * as React from 'react';
  4. import { cn } from '@/common/helpers/cn';
  5. function Sheet({ ...props }: React.ComponentProps<typeof SheetPrimitive.Root>) {
  6. return <SheetPrimitive.Root data-slot="sheet" {...props} />;
  7. }
  8. function SheetTrigger({ ...props }: React.ComponentProps<typeof SheetPrimitive.Trigger>) {
  9. return <SheetPrimitive.Trigger data-slot="sheet-trigger" {...props} />;
  10. }
  11. function SheetClose({ ...props }: React.ComponentProps<typeof SheetPrimitive.Close>) {
  12. return <SheetPrimitive.Close data-slot="sheet-close" {...props} />;
  13. }
  14. function SheetPortal({ ...props }: React.ComponentProps<typeof SheetPrimitive.Portal>) {
  15. return <SheetPrimitive.Portal data-slot="sheet-portal" {...props} />;
  16. }
  17. function SheetOverlay({ className, ...props }: React.ComponentProps<typeof SheetPrimitive.Overlay>) {
  18. return (
  19. <SheetPrimitive.Overlay
  20. data-slot="sheet-overlay"
  21. className={cn(
  22. 'fixed inset-0 z-50 bg-black/80 data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:animate-in data-[state=open]:fade-in-0',
  23. className,
  24. )}
  25. {...props}
  26. />
  27. );
  28. }
  29. function SheetContent({
  30. className,
  31. children,
  32. side = 'right',
  33. ...props
  34. }: React.ComponentProps<typeof SheetPrimitive.Content> & {
  35. side?: 'top' | 'right' | 'bottom' | 'left';
  36. }) {
  37. return (
  38. <SheetPortal>
  39. <SheetOverlay />
  40. <SheetPrimitive.Content
  41. data-slot="sheet-content"
  42. className={cn(
  43. 'fixed z-50 flex flex-col gap-4 bg-background shadow-lg transition ease-in-out data-[state=closed]:duration-300 data-[state=closed]:animate-out data-[state=open]:duration-500 data-[state=open]:animate-in',
  44. side === 'right' &&
  45. 'inset-y-0 right-0 h-full w-3/4 border-l data-[state=closed]:slide-out-to-right data-[state=open]:slide-in-from-right sm:max-w-sm',
  46. side === 'left' &&
  47. 'inset-y-0 left-0 h-full w-3/4 border-r data-[state=closed]:slide-out-to-left data-[state=open]:slide-in-from-left sm:max-w-sm',
  48. side === 'top' && 'inset-x-0 top-0 h-auto border-b data-[state=closed]:slide-out-to-top data-[state=open]:slide-in-from-top',
  49. side === 'bottom' &&
  50. 'inset-x-0 bottom-0 h-auto border-t data-[state=closed]:slide-out-to-bottom data-[state=open]:slide-in-from-bottom',
  51. className,
  52. )}
  53. {...props}
  54. >
  55. {children}
  56. <SheetPrimitive.Close className="absolute top-4 right-4 rounded-xs opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:ring-2 focus:ring-ring focus:ring-offset-2 focus:outline-hidden disabled:pointer-events-none data-[state=open]:bg-secondary">
  57. <XIcon className="size-4" />
  58. <span className="sr-only">Close</span>
  59. </SheetPrimitive.Close>
  60. </SheetPrimitive.Content>
  61. </SheetPortal>
  62. );
  63. }
  64. function SheetHeader({ className, ...props }: React.ComponentProps<'div'>) {
  65. return <div data-slot="sheet-header" className={cn('flex flex-col gap-1.5 p-4', className)} {...props} />;
  66. }
  67. function SheetFooter({ className, ...props }: React.ComponentProps<'div'>) {
  68. return <div data-slot="sheet-footer" className={cn('mt-auto flex flex-col gap-2 p-4', className)} {...props} />;
  69. }
  70. function SheetTitle({ className, ...props }: React.ComponentProps<typeof SheetPrimitive.Title>) {
  71. return <SheetPrimitive.Title data-slot="sheet-title" className={cn('font-semibold text-foreground', className)} {...props} />;
  72. }
  73. function SheetDescription({ className, ...props }: React.ComponentProps<typeof SheetPrimitive.Description>) {
  74. return <SheetPrimitive.Description data-slot="sheet-description" className={cn('text-sm text-muted-foreground', className)} {...props} />;
  75. }
  76. export { Sheet, SheetClose, SheetContent, SheetDescription, SheetFooter, SheetHeader, SheetTitle, SheetTrigger };