sheet.tsx 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  1. "use client"
  2. import * as React from "react"
  3. import * as SheetPrimitive from "@radix-ui/react-dialog"
  4. import { XIcon } from "lucide-react"
  5. import { cn } from "@/lib/utils"
  6. function Sheet({ ...props }: React.ComponentProps<typeof SheetPrimitive.Root>) {
  7. return <SheetPrimitive.Root data-slot="sheet" {...props} />
  8. }
  9. function SheetTrigger({
  10. ...props
  11. }: React.ComponentProps<typeof SheetPrimitive.Trigger>) {
  12. return <SheetPrimitive.Trigger data-slot="sheet-trigger" {...props} />
  13. }
  14. function SheetClose({
  15. ...props
  16. }: React.ComponentProps<typeof SheetPrimitive.Close>) {
  17. return <SheetPrimitive.Close data-slot="sheet-close" {...props} />
  18. }
  19. function SheetPortal({
  20. ...props
  21. }: React.ComponentProps<typeof SheetPrimitive.Portal>) {
  22. return <SheetPrimitive.Portal data-slot="sheet-portal" {...props} />
  23. }
  24. function SheetOverlay({
  25. className,
  26. ...props
  27. }: React.ComponentProps<typeof SheetPrimitive.Overlay>) {
  28. return (
  29. <SheetPrimitive.Overlay
  30. data-slot="sheet-overlay"
  31. className={cn(
  32. "data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 fixed inset-0 z-50 bg-black/50",
  33. className
  34. )}
  35. {...props}
  36. />
  37. )
  38. }
  39. function SheetContent({
  40. className,
  41. children,
  42. side = "right",
  43. ...props
  44. }: React.ComponentProps<typeof SheetPrimitive.Content> & {
  45. side?: "top" | "right" | "bottom" | "left"
  46. }) {
  47. return (
  48. <SheetPortal>
  49. <SheetOverlay />
  50. <SheetPrimitive.Content
  51. data-slot="sheet-content"
  52. className={cn(
  53. "bg-background data-[state=open]:animate-in data-[state=closed]:animate-out fixed z-50 flex flex-col gap-4 shadow-lg transition ease-in-out data-[state=closed]:duration-300 data-[state=open]:duration-500",
  54. side === "right" &&
  55. "data-[state=closed]:slide-out-to-right data-[state=open]:slide-in-from-right inset-y-0 right-0 h-full w-3/4 border-l sm:max-w-sm",
  56. side === "left" &&
  57. "data-[state=closed]:slide-out-to-left data-[state=open]:slide-in-from-left inset-y-0 left-0 h-full w-3/4 border-r sm:max-w-sm",
  58. side === "top" &&
  59. "data-[state=closed]:slide-out-to-top data-[state=open]:slide-in-from-top inset-x-0 top-0 h-auto border-b",
  60. side === "bottom" &&
  61. "data-[state=closed]:slide-out-to-bottom data-[state=open]:slide-in-from-bottom inset-x-0 bottom-0 h-auto border-t",
  62. className
  63. )}
  64. {...props}
  65. >
  66. {children}
  67. <SheetPrimitive.Close className="ring-offset-background focus:ring-ring data-[state=open]:bg-secondary absolute top-4 right-4 rounded-xs opacity-70 transition-opacity hover:opacity-100 focus:ring-2 focus:ring-offset-2 focus:outline-hidden disabled:pointer-events-none">
  68. <XIcon className="size-4" />
  69. <span className="sr-only">Close</span>
  70. </SheetPrimitive.Close>
  71. </SheetPrimitive.Content>
  72. </SheetPortal>
  73. )
  74. }
  75. function SheetHeader({ className, ...props }: React.ComponentProps<"div">) {
  76. return (
  77. <div
  78. data-slot="sheet-header"
  79. className={cn("flex flex-col gap-1.5 p-4", className)}
  80. {...props}
  81. />
  82. )
  83. }
  84. function SheetFooter({ className, ...props }: React.ComponentProps<"div">) {
  85. return (
  86. <div
  87. data-slot="sheet-footer"
  88. className={cn("mt-auto flex flex-col gap-2 p-4", className)}
  89. {...props}
  90. />
  91. )
  92. }
  93. function SheetTitle({
  94. className,
  95. ...props
  96. }: React.ComponentProps<typeof SheetPrimitive.Title>) {
  97. return (
  98. <SheetPrimitive.Title
  99. data-slot="sheet-title"
  100. className={cn("text-foreground font-semibold", className)}
  101. {...props}
  102. />
  103. )
  104. }
  105. function SheetDescription({
  106. className,
  107. ...props
  108. }: React.ComponentProps<typeof SheetPrimitive.Description>) {
  109. return (
  110. <SheetPrimitive.Description
  111. data-slot="sheet-description"
  112. className={cn("text-muted-foreground text-sm", className)}
  113. {...props}
  114. />
  115. )
  116. }
  117. export {
  118. Sheet,
  119. SheetTrigger,
  120. SheetClose,
  121. SheetContent,
  122. SheetHeader,
  123. SheetFooter,
  124. SheetTitle,
  125. SheetDescription,
  126. }