@srcroot/ui 0.0.55 → 0.0.56
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.
- package/README.md +151 -151
- package/dist/index.d.ts +0 -0
- package/dist/index.js +33 -17
- package/package.json +7 -2
- package/src/registry/analytics/google-analytics.tsx +36 -39
- package/src/registry/analytics/google-tag-manager.tsx +62 -65
- package/src/registry/analytics/meta-pixel.tsx +44 -47
- package/src/registry/analytics/microsoft-clarity.tsx +31 -34
- package/src/registry/analytics/tiktok-pixel.tsx +34 -37
- package/src/registry/lib/utils.ts +0 -0
- package/src/registry/themes/v3/blue.css +157 -157
- package/src/registry/themes/v3/glass.css +153 -153
- package/src/registry/themes/v3/gray.css +157 -157
- package/src/registry/themes/v3/green.css +157 -157
- package/src/registry/themes/v3/neutral.css +157 -157
- package/src/registry/themes/v3/orange.css +157 -157
- package/src/registry/themes/v3/rose.css +157 -157
- package/src/registry/themes/v3/slate.css +157 -157
- package/src/registry/themes/v3/stone.css +157 -157
- package/src/registry/themes/v3/violet.css +186 -186
- package/src/registry/themes/v3/zinc.css +157 -157
- package/src/registry/themes/v4/blue.css +184 -184
- package/src/registry/themes/v4/glass.css +180 -180
- package/src/registry/themes/v4/gray.css +184 -184
- package/src/registry/themes/v4/green.css +184 -184
- package/src/registry/themes/v4/neutral.css +184 -184
- package/src/registry/themes/v4/orange.css +184 -184
- package/src/registry/themes/v4/rose.css +184 -184
- package/src/registry/themes/v4/slate.css +184 -184
- package/src/registry/themes/v4/stone.css +184 -184
- package/src/registry/themes/v4/violet.css +184 -184
- package/src/registry/themes/v4/zinc.css +184 -184
- package/src/registry/ui/accordion.tsx +164 -165
- package/src/registry/ui/alert-dialog.tsx +213 -214
- package/src/registry/ui/alert.tsx +73 -76
- package/src/registry/ui/aspect-ratio.tsx +44 -47
- package/src/registry/ui/avatar.tsx +96 -97
- package/src/registry/ui/badge.tsx +52 -55
- package/src/registry/ui/breadcrumb.tsx +147 -150
- package/src/registry/ui/button-group.tsx +64 -67
- package/src/registry/ui/button.tsx +71 -72
- package/src/registry/ui/calendar.tsx +514 -515
- package/src/registry/ui/card.tsx +88 -91
- package/src/registry/ui/carousel.tsx +214 -214
- package/src/registry/ui/chart.tsx +373 -373
- package/src/registry/ui/chatbot.tsx +86 -13
- package/src/registry/ui/checkbox.tsx +93 -94
- package/src/registry/ui/collapsible.tsx +107 -108
- package/src/registry/ui/combobox.tsx +171 -171
- package/src/registry/ui/command.tsx +300 -300
- package/src/registry/ui/container.tsx +44 -47
- package/src/registry/ui/context-menu.tsx +221 -221
- package/src/registry/ui/date-picker.tsx +228 -228
- package/src/registry/ui/dialog.tsx +269 -270
- package/src/registry/ui/drawer.tsx +10 -4
- package/src/registry/ui/dropdown-menu.tsx +529 -530
- package/src/registry/ui/empty-state.tsx +0 -2
- package/src/registry/ui/file-upload.tsx +0 -0
- package/src/registry/ui/floating-dock.tsx +0 -0
- package/src/registry/ui/form-field.tsx +91 -94
- package/src/registry/ui/google-analytics.tsx +38 -0
- package/src/registry/ui/google-tag-manager.tsx +64 -0
- package/src/registry/ui/hover-card.tsx +223 -223
- package/src/registry/ui/image.tsx +144 -147
- package/src/registry/ui/input-group.tsx +82 -85
- package/src/registry/ui/input.tsx +125 -125
- package/src/registry/ui/kbd.tsx +60 -63
- package/src/registry/ui/label.tsx +36 -37
- package/src/registry/ui/loading-spinner.tsx +108 -111
- package/src/registry/ui/map.tsx +0 -0
- package/src/registry/ui/marquee.tsx +2 -0
- package/src/registry/ui/menubar.tsx +246 -246
- package/src/registry/ui/meta-pixel.tsx +46 -0
- package/src/registry/ui/microsoft-clarity.tsx +33 -0
- package/src/registry/ui/native-select.tsx +49 -52
- package/src/registry/ui/otp-input.tsx +152 -155
- package/src/registry/ui/pagination.tsx +149 -152
- package/src/registry/ui/patterns.tsx +28 -0
- package/src/registry/ui/popover.tsx +226 -227
- package/src/registry/ui/progress.tsx +51 -52
- package/src/registry/ui/radio.tsx +99 -102
- package/src/registry/ui/resizable.tsx +314 -314
- package/src/registry/ui/scroll-animation.tsx +45 -0
- package/src/registry/ui/scroll-area.tsx +121 -122
- package/src/registry/ui/scroll-to-top.tsx +0 -0
- package/src/registry/ui/search.tsx +147 -150
- package/src/registry/ui/select.tsx +292 -293
- package/src/registry/ui/separator.tsx +46 -47
- package/src/registry/ui/sheet.tsx +6 -3
- package/src/registry/ui/sidebar.tsx +628 -628
- package/src/registry/ui/skeleton.tsx +26 -29
- package/src/registry/ui/slider.tsx +196 -197
- package/src/registry/ui/slot.tsx +69 -72
- package/src/registry/ui/star-rating.tsx +131 -134
- package/src/registry/ui/switch.tsx +72 -73
- package/src/registry/ui/table-of-contents.tsx +96 -96
- package/src/registry/ui/table.tsx +138 -139
- package/src/registry/ui/tabs.tsx +124 -125
- package/src/registry/ui/text.tsx +61 -64
- package/src/registry/ui/textarea.tsx +41 -42
- package/src/registry/ui/theme-switcher.tsx +66 -66
- package/src/registry/ui/tiktok-pixel.tsx +36 -0
- package/src/registry/ui/toast.tsx +97 -98
- package/src/registry/ui/toggle-group.tsx +129 -129
- package/src/registry/ui/toggle.tsx +72 -72
- package/src/registry/ui/tooltip.tsx +143 -144
- package/src/registry/ui/whatsapp.tsx +0 -0
|
@@ -1,52 +1,49 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
*
|
|
10
|
-
*
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
"
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
{
|
|
27
|
-
>
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
>
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
export { NativeSelect }
|
|
52
|
-
|
|
1
|
+
import * as React from "react"
|
|
2
|
+
import { cn } from "@/lib/utils"
|
|
3
|
+
|
|
4
|
+
export interface NativeSelectProps extends React.SelectHTMLAttributes<HTMLSelectElement> { }
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* NativeSelect - Styled browser-native select element
|
|
8
|
+
*
|
|
9
|
+
* Uses the browser's native <select> for accessibility and mobile UX,
|
|
10
|
+
* with custom styling to match the design system.
|
|
11
|
+
*/
|
|
12
|
+
const NativeSelect = React.forwardRef<HTMLSelectElement, NativeSelectProps>(
|
|
13
|
+
({ className, children, ...props }, ref) => {
|
|
14
|
+
return (
|
|
15
|
+
<div className="relative">
|
|
16
|
+
<select
|
|
17
|
+
ref={ref}
|
|
18
|
+
className={cn(
|
|
19
|
+
"flex h-10 w-full appearance-none rounded-md border border-input bg-background px-3 py-2 pr-8 text-sm ring-offset-background",
|
|
20
|
+
"focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2",
|
|
21
|
+
"disabled:cursor-not-allowed disabled:opacity-50",
|
|
22
|
+
className
|
|
23
|
+
)}
|
|
24
|
+
{...props}
|
|
25
|
+
>
|
|
26
|
+
{children}
|
|
27
|
+
</select>
|
|
28
|
+
{/* Custom chevron icon */}
|
|
29
|
+
<div className="pointer-events-none absolute inset-y-0 right-0 flex items-center pr-2">
|
|
30
|
+
<svg
|
|
31
|
+
className="h-4 w-4 opacity-50"
|
|
32
|
+
xmlns="http://www.w3.org/2000/svg"
|
|
33
|
+
viewBox="0 0 24 24"
|
|
34
|
+
fill="none"
|
|
35
|
+
stroke="currentColor"
|
|
36
|
+
strokeWidth="2"
|
|
37
|
+
strokeLinecap="round"
|
|
38
|
+
strokeLinejoin="round"
|
|
39
|
+
>
|
|
40
|
+
<path d="m6 9 6 6 6-6" />
|
|
41
|
+
</svg>
|
|
42
|
+
</div>
|
|
43
|
+
</div>
|
|
44
|
+
)
|
|
45
|
+
}
|
|
46
|
+
)
|
|
47
|
+
NativeSelect.displayName = "NativeSelect"
|
|
48
|
+
|
|
49
|
+
export { NativeSelect }
|
|
@@ -1,155 +1,152 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
/**
|
|
8
|
-
|
|
9
|
-
/** Callback when
|
|
10
|
-
|
|
11
|
-
/**
|
|
12
|
-
|
|
13
|
-
/**
|
|
14
|
-
|
|
15
|
-
/**
|
|
16
|
-
|
|
17
|
-
/**
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
*
|
|
26
|
-
*
|
|
27
|
-
*
|
|
28
|
-
*
|
|
29
|
-
*
|
|
30
|
-
*
|
|
31
|
-
*
|
|
32
|
-
*
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
)
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
"
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
export { OtpInput }
|
|
155
|
-
|
|
1
|
+
import * as React from "react"
|
|
2
|
+
import { cn } from "@/lib/utils"
|
|
3
|
+
|
|
4
|
+
interface OtpInputProps {
|
|
5
|
+
/** Number of OTP digits */
|
|
6
|
+
length?: number
|
|
7
|
+
/** Callback when OTP is complete */
|
|
8
|
+
onComplete?: (otp: string) => void
|
|
9
|
+
/** Callback when value changes */
|
|
10
|
+
onChange?: (value: string) => void
|
|
11
|
+
/** Current value */
|
|
12
|
+
value?: string
|
|
13
|
+
/** Whether input is disabled */
|
|
14
|
+
disabled?: boolean
|
|
15
|
+
/** Auto focus first input */
|
|
16
|
+
autoFocus?: boolean
|
|
17
|
+
/** Input type (number or password for hidden) */
|
|
18
|
+
type?: "number" | "password"
|
|
19
|
+
className?: string
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* OTP Input component with auto-advance
|
|
24
|
+
*
|
|
25
|
+
* @example
|
|
26
|
+
* const [otp, setOtp] = useState("")
|
|
27
|
+
* <OtpInput
|
|
28
|
+
* length={6}
|
|
29
|
+
* value={otp}
|
|
30
|
+
* onChange={setOtp}
|
|
31
|
+
* onComplete={(code) => verifyOtp(code)}
|
|
32
|
+
* />
|
|
33
|
+
*/
|
|
34
|
+
const OtpInput = React.forwardRef<HTMLDivElement, OtpInputProps>(
|
|
35
|
+
(
|
|
36
|
+
{
|
|
37
|
+
length = 6,
|
|
38
|
+
onComplete,
|
|
39
|
+
onChange,
|
|
40
|
+
value = "",
|
|
41
|
+
disabled,
|
|
42
|
+
autoFocus,
|
|
43
|
+
type = "number",
|
|
44
|
+
className,
|
|
45
|
+
},
|
|
46
|
+
ref
|
|
47
|
+
) => {
|
|
48
|
+
const inputRefs = React.useRef<(HTMLInputElement | null)[]>([])
|
|
49
|
+
const [values, setValues] = React.useState<string[]>(
|
|
50
|
+
value.split("").concat(Array(length - value.length).fill(""))
|
|
51
|
+
)
|
|
52
|
+
|
|
53
|
+
React.useEffect(() => {
|
|
54
|
+
const newValues = value.split("").concat(Array(length - value.length).fill(""))
|
|
55
|
+
setValues(newValues.slice(0, length))
|
|
56
|
+
}, [value, length])
|
|
57
|
+
|
|
58
|
+
const focusInput = (index: number) => {
|
|
59
|
+
if (index >= 0 && index < length) {
|
|
60
|
+
inputRefs.current[index]?.focus()
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
const handleChange = (index: number, inputValue: string) => {
|
|
65
|
+
if (disabled) return
|
|
66
|
+
|
|
67
|
+
// Only allow single digit
|
|
68
|
+
const digit = inputValue.slice(-1)
|
|
69
|
+
if (type === "number" && digit && !/^\d$/.test(digit)) return
|
|
70
|
+
|
|
71
|
+
const newValues = [...values]
|
|
72
|
+
newValues[index] = digit
|
|
73
|
+
setValues(newValues)
|
|
74
|
+
|
|
75
|
+
const newOtp = newValues.join("")
|
|
76
|
+
onChange?.(newOtp)
|
|
77
|
+
|
|
78
|
+
// Auto-advance to next input
|
|
79
|
+
if (digit && index < length - 1) {
|
|
80
|
+
focusInput(index + 1)
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
// Check if complete
|
|
84
|
+
if (newOtp.length === length && !newOtp.includes("")) {
|
|
85
|
+
onComplete?.(newOtp)
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
const handleKeyDown = (index: number, e: React.KeyboardEvent<HTMLInputElement>) => {
|
|
90
|
+
if (e.key === "Backspace") {
|
|
91
|
+
if (!values[index] && index > 0) {
|
|
92
|
+
focusInput(index - 1)
|
|
93
|
+
}
|
|
94
|
+
} else if (e.key === "ArrowLeft") {
|
|
95
|
+
e.preventDefault()
|
|
96
|
+
focusInput(index - 1)
|
|
97
|
+
} else if (e.key === "ArrowRight") {
|
|
98
|
+
e.preventDefault()
|
|
99
|
+
focusInput(index + 1)
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
const handlePaste = (e: React.ClipboardEvent) => {
|
|
104
|
+
e.preventDefault()
|
|
105
|
+
const pastedData = e.clipboardData.getData("text").slice(0, length)
|
|
106
|
+
|
|
107
|
+
if (type === "number" && !/^\d+$/.test(pastedData)) return
|
|
108
|
+
|
|
109
|
+
const newValues = pastedData.split("").concat(Array(length - pastedData.length).fill(""))
|
|
110
|
+
setValues(newValues.slice(0, length))
|
|
111
|
+
|
|
112
|
+
const newOtp = newValues.slice(0, length).join("")
|
|
113
|
+
onChange?.(newOtp)
|
|
114
|
+
|
|
115
|
+
if (newOtp.length === length) {
|
|
116
|
+
onComplete?.(newOtp)
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
focusInput(Math.min(pastedData.length, length - 1))
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
return (
|
|
123
|
+
<div ref={ref} className={cn("flex gap-2", className)}>
|
|
124
|
+
{Array.from({ length }).map((_, index) => (
|
|
125
|
+
<input
|
|
126
|
+
key={index}
|
|
127
|
+
ref={(el) => { inputRefs.current[index] = el }}
|
|
128
|
+
type={type === "password" ? "password" : "text"}
|
|
129
|
+
inputMode="numeric"
|
|
130
|
+
maxLength={1}
|
|
131
|
+
value={values[index] || ""}
|
|
132
|
+
onChange={(e) => handleChange(index, e.target.value)}
|
|
133
|
+
onKeyDown={(e) => handleKeyDown(index, e)}
|
|
134
|
+
onPaste={handlePaste}
|
|
135
|
+
onFocus={(e) => e.target.select()}
|
|
136
|
+
disabled={disabled}
|
|
137
|
+
autoFocus={autoFocus && index === 0}
|
|
138
|
+
className={cn(
|
|
139
|
+
"h-12 w-12 rounded-md border border-input bg-transparent text-center text-lg font-semibold shadow-sm transition-colors",
|
|
140
|
+
"focus:outline-none focus:ring-2 focus:ring-ring focus:border-transparent",
|
|
141
|
+
"disabled:cursor-not-allowed disabled:opacity-50"
|
|
142
|
+
)}
|
|
143
|
+
aria-label={`Digit ${index + 1} of ${length}`}
|
|
144
|
+
/>
|
|
145
|
+
))}
|
|
146
|
+
</div>
|
|
147
|
+
)
|
|
148
|
+
}
|
|
149
|
+
)
|
|
150
|
+
OtpInput.displayName = "OtpInput"
|
|
151
|
+
|
|
152
|
+
export { OtpInput }
|