@codemation/ui 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (41) hide show
  1. package/.turbo/turbo-build.log +19 -0
  2. package/.turbo/turbo-lint.log +4 -0
  3. package/.turbo/turbo-typecheck.log +4 -0
  4. package/CHANGELOG.md +25 -0
  5. package/LICENSE +37 -0
  6. package/dist/index.cjs +845 -0
  7. package/dist/index.cjs.map +1 -0
  8. package/dist/index.d.cts +417 -0
  9. package/dist/index.d.ts +417 -0
  10. package/dist/index.js +749 -0
  11. package/dist/index.js.map +1 -0
  12. package/package.json +71 -0
  13. package/src/components/StatusPill.tsx +33 -0
  14. package/src/components/composite/CodemationDialog.tsx +137 -0
  15. package/src/components/composite/JsonMonacoEditor.tsx +75 -0
  16. package/src/components/reui/tree/Tree.tsx +35 -0
  17. package/src/components/reui/tree/TreeContext.ts +21 -0
  18. package/src/components/reui/tree/TreeDragLine.tsx +28 -0
  19. package/src/components/reui/tree/TreeItem.tsx +51 -0
  20. package/src/components/reui/tree/TreeItemLabel.tsx +58 -0
  21. package/src/components/ui/badge.tsx +40 -0
  22. package/src/components/ui/button.tsx +64 -0
  23. package/src/components/ui/collapsible.tsx +26 -0
  24. package/src/components/ui/dialog.tsx +137 -0
  25. package/src/components/ui/dropdown-menu.tsx +239 -0
  26. package/src/components/ui/input.tsx +20 -0
  27. package/src/components/ui/label.tsx +18 -0
  28. package/src/components/ui/select.tsx +171 -0
  29. package/src/components/ui/switch.tsx +29 -0
  30. package/src/components/ui/tabs.tsx +76 -0
  31. package/src/components/ui/textarea.tsx +18 -0
  32. package/src/index.ts +76 -0
  33. package/src/lib/cn.ts +6 -0
  34. package/src/lucide-icons.d.ts +46 -0
  35. package/test/StatusPill.test.tsx +26 -0
  36. package/test/primitives.test.tsx +707 -0
  37. package/test/setup.ts +7 -0
  38. package/test/ui-components.test.tsx +208 -0
  39. package/tsconfig.json +11 -0
  40. package/tsdown.config.ts +10 -0
  41. package/vitest.ui.config.ts +40 -0
@@ -0,0 +1,58 @@
1
+ "use client";
2
+
3
+ import ChevronDownIcon from "lucide-react/dist/esm/icons/chevron-down";
4
+ import MinusIcon from "lucide-react/dist/esm/icons/minus";
5
+ import PlusIcon from "lucide-react/dist/esm/icons/plus";
6
+ import { Slot } from "radix-ui";
7
+ import { Fragment, useContext, type HTMLAttributes } from "react";
8
+
9
+ import { cn } from "../../../lib/cn";
10
+
11
+ import { TreeContext } from "./TreeContext";
12
+
13
+ type TreeItemLabelProps = HTMLAttributes<HTMLSpanElement> & {
14
+ item?: {
15
+ isFolder: () => boolean;
16
+ isExpanded: () => boolean;
17
+ getItemName: () => string;
18
+ };
19
+ asChild?: boolean;
20
+ };
21
+
22
+ export function TreeItemLabel(args: Readonly<TreeItemLabelProps>) {
23
+ const { item: propItem, children, className, asChild = false, ...props } = args;
24
+ const { currentItem, toggleIconType } = useContext(TreeContext);
25
+ const item = propItem ?? currentItem;
26
+
27
+ if (!item) {
28
+ return null;
29
+ }
30
+
31
+ const Component = asChild ? Slot.Root : "span";
32
+
33
+ return (
34
+ <Component
35
+ data-slot="tree-item-label"
36
+ className={cn(
37
+ "in-focus-visible:ring-ring/50 bg-background hover:bg-accent in-data-[selected=true]:bg-accent in-data-[selected=true]:text-accent-foreground in-data-[drag-target=true]:bg-accent flex items-center gap-1 transition-colors not-in-data-[folder=true]:ps-7 in-focus-visible:ring-[3px] in-data-[search-match=true]:bg-blue-50! [&_svg]:pointer-events-none [&_svg]:shrink-0",
38
+ "rounded-sm py-1.5 px-2 text-sm",
39
+ className,
40
+ )}
41
+ {...props}
42
+ >
43
+ <Fragment>
44
+ {item.isFolder() &&
45
+ (toggleIconType === "plus-minus" ? (
46
+ item.isExpanded() ? (
47
+ <MinusIcon className="text-muted-foreground size-3.5" stroke="currentColor" strokeWidth="1" />
48
+ ) : (
49
+ <PlusIcon className="text-muted-foreground size-3.5" stroke="currentColor" strokeWidth="1" />
50
+ )
51
+ ) : (
52
+ <ChevronDownIcon className="text-muted-foreground size-4 in-aria-[expanded=false]:-rotate-90" />
53
+ ))}
54
+ {children ?? item.getItemName()}
55
+ </Fragment>
56
+ </Component>
57
+ );
58
+ }
@@ -0,0 +1,40 @@
1
+ import * as React from "react";
2
+ import { cva, type VariantProps } from "class-variance-authority";
3
+ import { Slot } from "radix-ui";
4
+
5
+ import { cn } from "../../lib/cn";
6
+
7
+ const badgeVariants = cva(
8
+ "group/badge inline-flex h-5 w-fit shrink-0 items-center justify-center gap-1 overflow-hidden rounded-full border border-transparent px-2 py-0.5 text-xs font-medium whitespace-nowrap shadow-none transition-all focus-visible:border-ring focus-visible:ring-[3px] focus-visible:ring-ring/50 has-data-[icon=inline-end]:pr-1.5 has-data-[icon=inline-start]:pl-1.5 aria-invalid:border-destructive aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40 [&>svg]:pointer-events-none [&>svg]:size-3!",
9
+ {
10
+ variants: {
11
+ variant: {
12
+ default: "bg-primary text-primary-foreground [a]:hover:bg-primary/80",
13
+ secondary: "bg-secondary text-secondary-foreground [a]:hover:bg-secondary/80",
14
+ destructive:
15
+ "bg-destructive/10 text-destructive focus-visible:ring-destructive/20 dark:bg-destructive/20 dark:focus-visible:ring-destructive/40 [a]:hover:bg-destructive/20",
16
+ outline: "border-border text-foreground [a]:hover:bg-muted [a]:hover:text-muted-foreground",
17
+ ghost: "hover:bg-muted hover:text-muted-foreground dark:hover:bg-muted/50",
18
+ link: "text-primary underline-offset-4 hover:underline",
19
+ },
20
+ },
21
+ defaultVariants: {
22
+ variant: "default",
23
+ },
24
+ },
25
+ );
26
+
27
+ function Badge({
28
+ className,
29
+ variant = "default",
30
+ asChild = false,
31
+ ...props
32
+ }: React.ComponentProps<"span"> & VariantProps<typeof badgeVariants> & { asChild?: boolean }) {
33
+ const Comp = asChild ? Slot.Root : "span";
34
+
35
+ return (
36
+ <Comp data-slot="badge" data-variant={variant} className={cn(badgeVariants({ variant }), className)} {...props} />
37
+ );
38
+ }
39
+
40
+ export { Badge, badgeVariants };
@@ -0,0 +1,64 @@
1
+ import * as React from "react";
2
+ import { cva, type VariantProps } from "class-variance-authority";
3
+ import { Slot } from "radix-ui";
4
+
5
+ import { cn } from "../../lib/cn";
6
+
7
+ const buttonVariants = cva(
8
+ "group/button inline-flex shrink-0 items-center justify-center rounded-md border border-transparent bg-clip-padding text-sm font-medium whitespace-nowrap shadow-sm transition-all outline-none select-none focus-visible:border-ring focus-visible:ring-3 focus-visible:ring-ring/50 active:translate-y-px disabled:pointer-events-none disabled:opacity-50 aria-invalid:border-destructive aria-invalid:ring-3 aria-invalid:ring-destructive/20 dark:aria-invalid:border-destructive/50 dark:aria-invalid:ring-destructive/40 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
9
+ {
10
+ variants: {
11
+ variant: {
12
+ default: "bg-primary text-primary-foreground [a]:hover:bg-primary/80",
13
+ outline:
14
+ "shadow-none border-border bg-background hover:bg-muted hover:text-foreground aria-expanded:bg-muted aria-expanded:text-foreground dark:border-input dark:bg-input/30 dark:hover:bg-input/50",
15
+ secondary:
16
+ "bg-secondary text-secondary-foreground hover:bg-secondary/80 aria-expanded:bg-secondary aria-expanded:text-secondary-foreground",
17
+ ghost:
18
+ "shadow-none hover:bg-muted hover:text-foreground aria-expanded:bg-muted aria-expanded:text-foreground dark:hover:bg-muted/50",
19
+ destructive:
20
+ "shadow-none bg-destructive/10 text-destructive hover:bg-destructive/20 focus-visible:border-destructive/40 focus-visible:ring-destructive/20 dark:bg-destructive/20 dark:hover:bg-destructive/30 dark:focus-visible:ring-destructive/40",
21
+ link: "shadow-none text-primary underline-offset-4 hover:underline",
22
+ },
23
+ size: {
24
+ default: "h-8 gap-1.5 px-2.5 has-data-[icon=inline-end]:pr-2 has-data-[icon=inline-start]:pl-2",
25
+ xs: "h-6 gap-1 rounded-sm px-2 text-xs in-data-[slot=button-group]:rounded-md has-data-[icon=inline-end]:pr-1.5 has-data-[icon=inline-start]:pl-1.5 [&_svg:not([class*='size-'])]:size-3",
26
+ sm: "h-7 gap-1 rounded-sm px-2.5 text-[0.8rem] in-data-[slot=button-group]:rounded-md has-data-[icon=inline-end]:pr-1.5 has-data-[icon=inline-start]:pl-1.5 [&_svg:not([class*='size-'])]:size-3.5",
27
+ lg: "h-9 gap-1.5 px-2.5 has-data-[icon=inline-end]:pr-3 has-data-[icon=inline-start]:pl-3",
28
+ icon: "size-8",
29
+ "icon-xs": "size-6 rounded-sm in-data-[slot=button-group]:rounded-md [&_svg:not([class*='size-'])]:size-3",
30
+ "icon-sm": "size-7 rounded-sm in-data-[slot=button-group]:rounded-md",
31
+ "icon-lg": "size-9",
32
+ },
33
+ },
34
+ defaultVariants: {
35
+ variant: "default",
36
+ size: "default",
37
+ },
38
+ },
39
+ );
40
+
41
+ function Button({
42
+ className,
43
+ variant = "default",
44
+ size = "default",
45
+ asChild = false,
46
+ ...props
47
+ }: React.ComponentProps<"button"> &
48
+ VariantProps<typeof buttonVariants> & {
49
+ asChild?: boolean;
50
+ }) {
51
+ const Comp = asChild ? Slot.Root : "button";
52
+
53
+ return (
54
+ <Comp
55
+ data-slot="button"
56
+ data-variant={variant}
57
+ data-size={size}
58
+ className={cn(buttonVariants({ variant, size, className }))}
59
+ {...props}
60
+ />
61
+ );
62
+ }
63
+
64
+ export { Button, buttonVariants };
@@ -0,0 +1,26 @@
1
+ "use client";
2
+
3
+ import * as React from "react";
4
+ import { Collapsible as CollapsiblePrimitive } from "radix-ui";
5
+
6
+ import { cn } from "../../lib/cn";
7
+
8
+ function Collapsible(props: React.ComponentProps<typeof CollapsiblePrimitive.Root>) {
9
+ return <CollapsiblePrimitive.Root data-slot="collapsible" {...props} />;
10
+ }
11
+
12
+ function CollapsibleTrigger(props: React.ComponentProps<typeof CollapsiblePrimitive.Trigger>) {
13
+ return <CollapsiblePrimitive.Trigger data-slot="collapsible-trigger" {...props} />;
14
+ }
15
+
16
+ function CollapsibleContent({ className, ...props }: React.ComponentProps<typeof CollapsiblePrimitive.Content>) {
17
+ return (
18
+ <CollapsiblePrimitive.Content
19
+ data-slot="collapsible-content"
20
+ className={cn("overflow-hidden", className)}
21
+ {...props}
22
+ />
23
+ );
24
+ }
25
+
26
+ export { Collapsible, CollapsibleContent, CollapsibleTrigger };
@@ -0,0 +1,137 @@
1
+ "use client";
2
+
3
+ import * as React from "react";
4
+ import { Dialog as DialogPrimitive } from "radix-ui";
5
+
6
+ import { cn } from "../../lib/cn";
7
+ import { Button } from "./button";
8
+ import XIcon from "lucide-react/dist/esm/icons/x";
9
+
10
+ function Dialog({ ...props }: React.ComponentProps<typeof DialogPrimitive.Root>) {
11
+ return <DialogPrimitive.Root data-slot="dialog" {...props} />;
12
+ }
13
+
14
+ function DialogTrigger({ ...props }: React.ComponentProps<typeof DialogPrimitive.Trigger>) {
15
+ return <DialogPrimitive.Trigger data-slot="dialog-trigger" {...props} />;
16
+ }
17
+
18
+ function DialogPortal({ ...props }: React.ComponentProps<typeof DialogPrimitive.Portal>) {
19
+ return <DialogPrimitive.Portal data-slot="dialog-portal" {...props} />;
20
+ }
21
+
22
+ function DialogClose({ ...props }: React.ComponentProps<typeof DialogPrimitive.Close>) {
23
+ return <DialogPrimitive.Close data-slot="dialog-close" {...props} />;
24
+ }
25
+
26
+ function DialogOverlay({ className, ...props }: React.ComponentProps<typeof DialogPrimitive.Overlay>) {
27
+ return (
28
+ <DialogPrimitive.Overlay
29
+ data-slot="dialog-overlay"
30
+ className={cn(
31
+ "fixed inset-0 isolate z-50 bg-black/10 duration-100 supports-backdrop-filter:backdrop-blur-xs data-open:animate-in data-open:fade-in-0 data-closed:animate-out data-closed:fade-out-0",
32
+ className,
33
+ )}
34
+ {...props}
35
+ />
36
+ );
37
+ }
38
+
39
+ function DialogContent({
40
+ className,
41
+ children,
42
+ showCloseButton = true,
43
+ ...props
44
+ }: React.ComponentProps<typeof DialogPrimitive.Content> & {
45
+ showCloseButton?: boolean;
46
+ }) {
47
+ return (
48
+ <DialogPortal>
49
+ <DialogOverlay />
50
+ <DialogPrimitive.Content
51
+ data-slot="dialog-content"
52
+ className={cn(
53
+ "fixed top-1/2 left-1/2 z-50 grid w-full max-w-[calc(100%-2rem)] -translate-x-1/2 -translate-y-1/2 gap-4 rounded-xl bg-background p-4 text-sm ring-1 ring-foreground/10 duration-100 outline-none sm:max-w-sm data-open:animate-in data-open:fade-in-0 data-open:zoom-in-95 data-closed:animate-out data-closed:fade-out-0 data-closed:zoom-out-95",
54
+ className,
55
+ )}
56
+ {...props}
57
+ >
58
+ {children}
59
+ {showCloseButton && (
60
+ <DialogPrimitive.Close data-slot="dialog-close" asChild>
61
+ <Button variant="ghost" className="absolute top-2 right-2" size="icon-sm">
62
+ <XIcon />
63
+ <span className="sr-only">Close</span>
64
+ </Button>
65
+ </DialogPrimitive.Close>
66
+ )}
67
+ </DialogPrimitive.Content>
68
+ </DialogPortal>
69
+ );
70
+ }
71
+
72
+ function DialogHeader({ className, ...props }: React.ComponentProps<"div">) {
73
+ return <div data-slot="dialog-header" className={cn("flex flex-col gap-2", className)} {...props} />;
74
+ }
75
+
76
+ function DialogFooter({
77
+ className,
78
+ showCloseButton = false,
79
+ children,
80
+ ...props
81
+ }: React.ComponentProps<"div"> & {
82
+ showCloseButton?: boolean;
83
+ }) {
84
+ return (
85
+ <div
86
+ data-slot="dialog-footer"
87
+ className={cn(
88
+ "-mx-4 -mb-4 flex flex-col-reverse gap-2 rounded-b-xl border-t bg-muted/50 p-4 sm:flex-row sm:justify-end",
89
+ className,
90
+ )}
91
+ {...props}
92
+ >
93
+ {children}
94
+ {showCloseButton && (
95
+ <DialogPrimitive.Close asChild>
96
+ <Button variant="outline">Close</Button>
97
+ </DialogPrimitive.Close>
98
+ )}
99
+ </div>
100
+ );
101
+ }
102
+
103
+ function DialogTitle({ className, ...props }: React.ComponentProps<typeof DialogPrimitive.Title>) {
104
+ return (
105
+ <DialogPrimitive.Title
106
+ data-slot="dialog-title"
107
+ className={cn("text-base leading-none font-medium", className)}
108
+ {...props}
109
+ />
110
+ );
111
+ }
112
+
113
+ function DialogDescription({ className, ...props }: React.ComponentProps<typeof DialogPrimitive.Description>) {
114
+ return (
115
+ <DialogPrimitive.Description
116
+ data-slot="dialog-description"
117
+ className={cn(
118
+ "text-sm text-muted-foreground *:[a]:underline *:[a]:underline-offset-3 *:[a]:hover:text-foreground",
119
+ className,
120
+ )}
121
+ {...props}
122
+ />
123
+ );
124
+ }
125
+
126
+ export {
127
+ Dialog,
128
+ DialogClose,
129
+ DialogContent,
130
+ DialogDescription,
131
+ DialogFooter,
132
+ DialogHeader,
133
+ DialogOverlay,
134
+ DialogPortal,
135
+ DialogTitle,
136
+ DialogTrigger,
137
+ };
@@ -0,0 +1,239 @@
1
+ "use client";
2
+
3
+ import * as React from "react";
4
+ import { DropdownMenu as DropdownMenuPrimitive } from "radix-ui";
5
+
6
+ import { cn } from "../../lib/cn";
7
+ import CheckIcon from "lucide-react/dist/esm/icons/check";
8
+ import ChevronRightIcon from "lucide-react/dist/esm/icons/chevron-right";
9
+
10
+ function DropdownMenu({ ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.Root>) {
11
+ return <DropdownMenuPrimitive.Root data-slot="dropdown-menu" {...props} />;
12
+ }
13
+
14
+ function DropdownMenuPortal({ ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.Portal>) {
15
+ return <DropdownMenuPrimitive.Portal data-slot="dropdown-menu-portal" {...props} />;
16
+ }
17
+
18
+ function DropdownMenuTrigger({ ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.Trigger>) {
19
+ return <DropdownMenuPrimitive.Trigger data-slot="dropdown-menu-trigger" {...props} />;
20
+ }
21
+
22
+ function DropdownMenuContent({
23
+ className,
24
+ align = "start",
25
+ sideOffset = 4,
26
+ ...props
27
+ }: React.ComponentProps<typeof DropdownMenuPrimitive.Content>) {
28
+ return (
29
+ <DropdownMenuPrimitive.Portal>
30
+ <DropdownMenuPrimitive.Content
31
+ data-slot="dropdown-menu-content"
32
+ sideOffset={sideOffset}
33
+ align={align}
34
+ className={cn(
35
+ "z-50 max-h-(--radix-dropdown-menu-content-available-height) w-(--radix-dropdown-menu-trigger-width) min-w-32 origin-(--radix-dropdown-menu-content-transform-origin) overflow-x-hidden overflow-y-auto rounded-lg bg-popover p-1 text-popover-foreground shadow-md ring-1 ring-foreground/10 duration-100 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 data-[state=closed]:overflow-hidden data-open:animate-in data-open:fade-in-0 data-open:zoom-in-95 data-closed:animate-out data-closed:fade-out-0 data-closed:zoom-out-95",
36
+ className,
37
+ )}
38
+ {...props}
39
+ />
40
+ </DropdownMenuPrimitive.Portal>
41
+ );
42
+ }
43
+
44
+ function DropdownMenuGroup({ ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.Group>) {
45
+ return <DropdownMenuPrimitive.Group data-slot="dropdown-menu-group" {...props} />;
46
+ }
47
+
48
+ function DropdownMenuItem({
49
+ className,
50
+ inset,
51
+ variant = "default",
52
+ ...props
53
+ }: React.ComponentProps<typeof DropdownMenuPrimitive.Item> & {
54
+ inset?: boolean;
55
+ variant?: "default" | "destructive";
56
+ }) {
57
+ return (
58
+ <DropdownMenuPrimitive.Item
59
+ data-slot="dropdown-menu-item"
60
+ data-inset={inset}
61
+ data-variant={variant}
62
+ className={cn(
63
+ "group/dropdown-menu-item relative flex cursor-default items-center gap-1.5 rounded-md px-1.5 py-1 text-sm outline-hidden select-none focus:bg-accent focus:text-accent-foreground not-data-[variant=destructive]:focus:**:text-accent-foreground data-inset:pl-7 data-[variant=destructive]:text-destructive data-[variant=destructive]:focus:bg-destructive/10 data-[variant=destructive]:focus:text-destructive dark:data-[variant=destructive]:focus:bg-destructive/20 data-disabled:pointer-events-none data-disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4 data-[variant=destructive]:*:[svg]:text-destructive",
64
+ className,
65
+ )}
66
+ {...props}
67
+ />
68
+ );
69
+ }
70
+
71
+ function DropdownMenuCheckboxItem({
72
+ className,
73
+ children,
74
+ checked,
75
+ inset,
76
+ ...props
77
+ }: React.ComponentProps<typeof DropdownMenuPrimitive.CheckboxItem> & {
78
+ inset?: boolean;
79
+ }) {
80
+ return (
81
+ <DropdownMenuPrimitive.CheckboxItem
82
+ data-slot="dropdown-menu-checkbox-item"
83
+ data-inset={inset}
84
+ className={cn(
85
+ "relative flex cursor-default items-center gap-1.5 rounded-md py-1 pr-8 pl-1.5 text-sm outline-hidden select-none focus:bg-accent focus:text-accent-foreground focus:**:text-accent-foreground data-inset:pl-7 data-disabled:pointer-events-none data-disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
86
+ className,
87
+ )}
88
+ checked={checked}
89
+ {...props}
90
+ >
91
+ <span
92
+ className="pointer-events-none absolute right-2 flex items-center justify-center"
93
+ data-slot="dropdown-menu-checkbox-item-indicator"
94
+ >
95
+ <DropdownMenuPrimitive.ItemIndicator>
96
+ <CheckIcon />
97
+ </DropdownMenuPrimitive.ItemIndicator>
98
+ </span>
99
+ {children}
100
+ </DropdownMenuPrimitive.CheckboxItem>
101
+ );
102
+ }
103
+
104
+ function DropdownMenuRadioGroup({ ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.RadioGroup>) {
105
+ return <DropdownMenuPrimitive.RadioGroup data-slot="dropdown-menu-radio-group" {...props} />;
106
+ }
107
+
108
+ function DropdownMenuRadioItem({
109
+ className,
110
+ children,
111
+ inset,
112
+ ...props
113
+ }: React.ComponentProps<typeof DropdownMenuPrimitive.RadioItem> & {
114
+ inset?: boolean;
115
+ }) {
116
+ return (
117
+ <DropdownMenuPrimitive.RadioItem
118
+ data-slot="dropdown-menu-radio-item"
119
+ data-inset={inset}
120
+ className={cn(
121
+ "relative flex cursor-default items-center gap-1.5 rounded-md py-1 pr-8 pl-1.5 text-sm outline-hidden select-none focus:bg-accent focus:text-accent-foreground focus:**:text-accent-foreground data-inset:pl-7 data-disabled:pointer-events-none data-disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
122
+ className,
123
+ )}
124
+ {...props}
125
+ >
126
+ <span
127
+ className="pointer-events-none absolute right-2 flex items-center justify-center"
128
+ data-slot="dropdown-menu-radio-item-indicator"
129
+ >
130
+ <DropdownMenuPrimitive.ItemIndicator>
131
+ <CheckIcon />
132
+ </DropdownMenuPrimitive.ItemIndicator>
133
+ </span>
134
+ {children}
135
+ </DropdownMenuPrimitive.RadioItem>
136
+ );
137
+ }
138
+
139
+ function DropdownMenuLabel({
140
+ className,
141
+ inset,
142
+ ...props
143
+ }: React.ComponentProps<typeof DropdownMenuPrimitive.Label> & {
144
+ inset?: boolean;
145
+ }) {
146
+ return (
147
+ <DropdownMenuPrimitive.Label
148
+ data-slot="dropdown-menu-label"
149
+ data-inset={inset}
150
+ className={cn("px-1.5 py-1 text-xs font-medium text-muted-foreground data-inset:pl-7", className)}
151
+ {...props}
152
+ />
153
+ );
154
+ }
155
+
156
+ function DropdownMenuSeparator({ className, ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.Separator>) {
157
+ return (
158
+ <DropdownMenuPrimitive.Separator
159
+ data-slot="dropdown-menu-separator"
160
+ className={cn("-mx-1 my-1 h-px bg-border", className)}
161
+ {...props}
162
+ />
163
+ );
164
+ }
165
+
166
+ function DropdownMenuShortcut({ className, ...props }: React.ComponentProps<"span">) {
167
+ return (
168
+ <span
169
+ data-slot="dropdown-menu-shortcut"
170
+ className={cn(
171
+ "ml-auto text-xs tracking-widest text-muted-foreground group-focus/dropdown-menu-item:text-accent-foreground",
172
+ className,
173
+ )}
174
+ {...props}
175
+ />
176
+ );
177
+ }
178
+
179
+ function DropdownMenuSub({ ...props }: React.ComponentProps<typeof DropdownMenuPrimitive.Sub>) {
180
+ return <DropdownMenuPrimitive.Sub data-slot="dropdown-menu-sub" {...props} />;
181
+ }
182
+
183
+ function DropdownMenuSubTrigger({
184
+ className,
185
+ inset,
186
+ children,
187
+ ...props
188
+ }: React.ComponentProps<typeof DropdownMenuPrimitive.SubTrigger> & {
189
+ inset?: boolean;
190
+ }) {
191
+ return (
192
+ <DropdownMenuPrimitive.SubTrigger
193
+ data-slot="dropdown-menu-sub-trigger"
194
+ data-inset={inset}
195
+ className={cn(
196
+ "flex cursor-default items-center gap-1.5 rounded-md px-1.5 py-1 text-sm outline-hidden select-none focus:bg-accent focus:text-accent-foreground not-data-[variant=destructive]:focus:**:text-accent-foreground data-inset:pl-7 data-open:bg-accent data-open:text-accent-foreground [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
197
+ className,
198
+ )}
199
+ {...props}
200
+ >
201
+ {children}
202
+ <ChevronRightIcon className="ml-auto" />
203
+ </DropdownMenuPrimitive.SubTrigger>
204
+ );
205
+ }
206
+
207
+ function DropdownMenuSubContent({
208
+ className,
209
+ ...props
210
+ }: React.ComponentProps<typeof DropdownMenuPrimitive.SubContent>) {
211
+ return (
212
+ <DropdownMenuPrimitive.SubContent
213
+ data-slot="dropdown-menu-sub-content"
214
+ className={cn(
215
+ "z-50 min-w-[96px] origin-(--radix-dropdown-menu-content-transform-origin) overflow-hidden rounded-lg bg-popover p-1 text-popover-foreground shadow-lg ring-1 ring-foreground/10 duration-100 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 data-open:animate-in data-open:fade-in-0 data-open:zoom-in-95 data-closed:animate-out data-closed:fade-out-0 data-closed:zoom-out-95",
216
+ className,
217
+ )}
218
+ {...props}
219
+ />
220
+ );
221
+ }
222
+
223
+ export {
224
+ DropdownMenu,
225
+ DropdownMenuPortal,
226
+ DropdownMenuTrigger,
227
+ DropdownMenuContent,
228
+ DropdownMenuGroup,
229
+ DropdownMenuLabel,
230
+ DropdownMenuItem,
231
+ DropdownMenuCheckboxItem,
232
+ DropdownMenuRadioGroup,
233
+ DropdownMenuRadioItem,
234
+ DropdownMenuSeparator,
235
+ DropdownMenuShortcut,
236
+ DropdownMenuSub,
237
+ DropdownMenuSubTrigger,
238
+ DropdownMenuSubContent,
239
+ };
@@ -0,0 +1,20 @@
1
+ import * as React from "react";
2
+ import { cn } from "../../lib/cn";
3
+
4
+ function Input({ className, type, ...props }: React.ComponentProps<"input">) {
5
+ return (
6
+ <input
7
+ type={type}
8
+ data-slot="input"
9
+ className={cn(
10
+ "border-input file:text-foreground placeholder:text-muted-foreground selection:bg-primary selection:text-primary-foreground flex h-9 w-full min-w-0 rounded-md border bg-transparent px-3 py-1 text-base shadow-xs transition-[color,box-shadow] outline-none file:inline-flex file:h-7 file:border-0 file:bg-transparent file:text-sm file:font-medium disabled:pointer-events-none disabled:cursor-not-allowed disabled:opacity-50 md:text-sm",
11
+ "focus-visible:border-ring focus-visible:ring-[3px] focus-visible:ring-ring/50",
12
+ "aria-invalid:border-destructive aria-invalid:ring-[3px] aria-invalid:ring-destructive/20 dark:aria-invalid:ring-destructive/40",
13
+ className,
14
+ )}
15
+ {...props}
16
+ />
17
+ );
18
+ }
19
+
20
+ export { Input };
@@ -0,0 +1,18 @@
1
+ import * as React from "react";
2
+ import { Label as LabelPrimitive } from "radix-ui";
3
+ import { cn } from "../../lib/cn";
4
+
5
+ function Label({ className, ...props }: React.ComponentProps<typeof LabelPrimitive.Root>) {
6
+ return (
7
+ <LabelPrimitive.Root
8
+ data-slot="label"
9
+ className={cn(
10
+ "text-sm leading-none font-medium select-none group-data-[disabled=true]/form-item:pointer-events-none group-data-[disabled=true]/form-item:opacity-50 peer-disabled:cursor-not-allowed peer-disabled:opacity-50",
11
+ className,
12
+ )}
13
+ {...props}
14
+ />
15
+ );
16
+ }
17
+
18
+ export { Label };