banhaten 0.1.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 (201) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +361 -0
  3. package/banhaten.config.example.json +13 -0
  4. package/package.json +59 -0
  5. package/registry/assets/activity-feed-avatar.png +0 -0
  6. package/registry/assets/avatars/avatar-01.jpg +0 -0
  7. package/registry/assets/avatars/avatar-02.jpg +0 -0
  8. package/registry/assets/avatars/avatar-03.jpg +0 -0
  9. package/registry/assets/avatars/avatar-04.jpg +0 -0
  10. package/registry/assets/avatars/avatar-05.jpg +0 -0
  11. package/registry/assets/avatars/avatar-06.jpg +0 -0
  12. package/registry/assets/avatars/avatar-07.jpg +0 -0
  13. package/registry/assets/avatars/avatar-08.jpg +0 -0
  14. package/registry/assets/avatars/avatar-09.jpg +0 -0
  15. package/registry/assets/avatars/avatar-10.jpg +0 -0
  16. package/registry/assets/avatars/avatar-11.jpg +0 -0
  17. package/registry/assets/avatars/avatar-12.jpg +0 -0
  18. package/registry/assets/avatars/avatar-13.jpg +0 -0
  19. package/registry/assets/avatars/avatar-14.jpg +0 -0
  20. package/registry/assets/avatars/avatar-15.jpg +0 -0
  21. package/registry/assets/avatars/avatar-16.jpg +0 -0
  22. package/registry/assets/avatars/avatar-17.jpg +0 -0
  23. package/registry/assets/avatars/avatar-18.jpg +0 -0
  24. package/registry/assets/avatars/avatar-19.jpg +0 -0
  25. package/registry/assets/avatars/avatar-20.jpg +0 -0
  26. package/registry/assets/avatars/avatar-21.jpg +0 -0
  27. package/registry/assets/avatars/avatar-22.jpg +0 -0
  28. package/registry/assets/avatars/avatar-23.jpg +0 -0
  29. package/registry/assets/avatars/avatar-24.jpg +0 -0
  30. package/registry/assets/avatars/avatar-25.jpg +0 -0
  31. package/registry/assets/avatars/avatar-26.jpg +0 -0
  32. package/registry/assets/avatars/avatar-27.jpg +0 -0
  33. package/registry/assets/avatars/avatar-28.jpg +0 -0
  34. package/registry/assets/avatars/avatar-29.jpg +0 -0
  35. package/registry/assets/avatars/avatar-30.jpg +0 -0
  36. package/registry/assets/avatars/avatar-31.jpg +0 -0
  37. package/registry/assets/avatars/avatar-32.jpg +0 -0
  38. package/registry/assets/avatars/avatar-33.jpg +0 -0
  39. package/registry/assets/avatars/avatar-34.jpg +0 -0
  40. package/registry/assets/avatars/avatar-35.jpg +0 -0
  41. package/registry/assets/image-assets.json +744 -0
  42. package/registry/assets/images/art-01.jpg +0 -0
  43. package/registry/assets/images/art-02.jpg +0 -0
  44. package/registry/assets/images/art-03.jpg +0 -0
  45. package/registry/assets/images/art-04.jpg +0 -0
  46. package/registry/assets/images/art-05.jpg +0 -0
  47. package/registry/assets/images/art-06.jpg +0 -0
  48. package/registry/assets/images/art-07.jpg +0 -0
  49. package/registry/assets/images/art-08.jpg +0 -0
  50. package/registry/assets/images/art-09.jpg +0 -0
  51. package/registry/assets/images/art-10.jpg +0 -0
  52. package/registry/assets/images/art-11.jpg +0 -0
  53. package/registry/assets/images/art-12.jpg +0 -0
  54. package/registry/assets/images/art-13.jpg +0 -0
  55. package/registry/assets/images/art-14.jpg +0 -0
  56. package/registry/assets/images/art-15.jpg +0 -0
  57. package/registry/assets/images/art-16.jpg +0 -0
  58. package/registry/assets/images/art-17.jpg +0 -0
  59. package/registry/assets/images/art-18.jpg +0 -0
  60. package/registry/assets/images/art-19.jpg +0 -0
  61. package/registry/assets/images/art-20.jpg +0 -0
  62. package/registry/assets/images/art-21.jpg +0 -0
  63. package/registry/assets/images/art-22.jpg +0 -0
  64. package/registry/assets/images/art-23.jpg +0 -0
  65. package/registry/assets/images/art-24.jpg +0 -0
  66. package/registry/assets/images/art-25.jpg +0 -0
  67. package/registry/assets/images/art-26.jpg +0 -0
  68. package/registry/assets/images/art-27.jpg +0 -0
  69. package/registry/assets/images/nature-01.jpg +0 -0
  70. package/registry/assets/images/nature-02.jpg +0 -0
  71. package/registry/assets/images/nature-03.jpg +0 -0
  72. package/registry/assets/images/nature-04.jpg +0 -0
  73. package/registry/assets/images/nature-05.jpg +0 -0
  74. package/registry/assets/images/nature-06.jpg +0 -0
  75. package/registry/assets/images/nature-07.jpg +0 -0
  76. package/registry/assets/images/nature-08.jpg +0 -0
  77. package/registry/assets/images/nature-09.jpg +0 -0
  78. package/registry/assets/images/nature-10.jpg +0 -0
  79. package/registry/assets/images/nature-11.jpg +0 -0
  80. package/registry/assets/images/nature-12.jpg +0 -0
  81. package/registry/assets/images/nature-13.jpg +0 -0
  82. package/registry/assets/images/nature-14.jpg +0 -0
  83. package/registry/assets/images/nature-15.jpg +0 -0
  84. package/registry/assets/images/nature-16.jpg +0 -0
  85. package/registry/assets/images/nature-17.jpg +0 -0
  86. package/registry/assets/images/nature-18.jpg +0 -0
  87. package/registry/assets/images/nature-19.jpg +0 -0
  88. package/registry/assets/images/nature-20.jpg +0 -0
  89. package/registry/components/accordion.tsx +119 -0
  90. package/registry/components/alert.tsx +282 -0
  91. package/registry/components/attribute.tsx +452 -0
  92. package/registry/components/avatar.tsx +142 -0
  93. package/registry/components/badge.tsx +567 -0
  94. package/registry/components/button-group.tsx +246 -0
  95. package/registry/components/button.tsx +102 -0
  96. package/registry/components/card.tsx +613 -0
  97. package/registry/components/checkbox.tsx +244 -0
  98. package/registry/components/date-picker.tsx +1143 -0
  99. package/registry/components/divider.tsx +82 -0
  100. package/registry/components/expanded/ActivityFeed.tsx +226 -0
  101. package/registry/components/expanded/Banner.tsx +145 -0
  102. package/registry/components/expanded/BannerBoard.tsx +225 -0
  103. package/registry/components/expanded/Breadcrumbs.tsx +156 -0
  104. package/registry/components/expanded/CatalogComponentsShowcase.tsx +211 -0
  105. package/registry/components/expanded/CatalogDivider.tsx +48 -0
  106. package/registry/components/expanded/CatalogTag.tsx +92 -0
  107. package/registry/components/expanded/CommandBar.tsx +406 -0
  108. package/registry/components/expanded/FileUpload.tsx +231 -0
  109. package/registry/components/expanded/IconExplorer.tsx +612 -0
  110. package/registry/components/expanded/OnboardingStepListItem.tsx +67 -0
  111. package/registry/components/expanded/PageHeader.tsx +184 -0
  112. package/registry/components/expanded/Slideout.tsx +514 -0
  113. package/registry/components/expanded/Steps.tsx +266 -0
  114. package/registry/components/expanded/Table.tsx +1014 -0
  115. package/registry/components/expanded/Tabs.tsx +86 -0
  116. package/registry/components/expanded/Timeline.tsx +235 -0
  117. package/registry/components/expanded/TimelineShowcase.tsx +158 -0
  118. package/registry/components/expanded/activityFeed.css +292 -0
  119. package/registry/components/expanded/banner.css +312 -0
  120. package/registry/components/expanded/breadcrumbs.css +140 -0
  121. package/registry/components/expanded/catalogComponentsShowcase.css +87 -0
  122. package/registry/components/expanded/commandBar.css +473 -0
  123. package/registry/components/expanded/divider.css +75 -0
  124. package/registry/components/expanded/fileUpload.css +228 -0
  125. package/registry/components/expanded/iconExplorer.css +764 -0
  126. package/registry/components/expanded/iconPacks.ts +866 -0
  127. package/registry/components/expanded/onboardingStepListItem.css +126 -0
  128. package/registry/components/expanded/pageHeader.css +287 -0
  129. package/registry/components/expanded/slideout.css +955 -0
  130. package/registry/components/expanded/steps.css +329 -0
  131. package/registry/components/expanded/table.css +607 -0
  132. package/registry/components/expanded/tabs.css +197 -0
  133. package/registry/components/expanded/tag.css +148 -0
  134. package/registry/components/expanded/timeline.css +282 -0
  135. package/registry/components/input-content.ts +106 -0
  136. package/registry/components/input.tsx +866 -0
  137. package/registry/components/menu.tsx +758 -0
  138. package/registry/components/modal.tsx +799 -0
  139. package/registry/components/pagination.tsx +543 -0
  140. package/registry/components/progress-slider.tsx +216 -0
  141. package/registry/components/progress.tsx +367 -0
  142. package/registry/components/radio-card.tsx +654 -0
  143. package/registry/components/radio-group.tsx +570 -0
  144. package/registry/components/select-content.tsx +313 -0
  145. package/registry/components/select.tsx +871 -0
  146. package/registry/components/slider.tsx +380 -0
  147. package/registry/components/social-button.tsx +360 -0
  148. package/registry/components/spinner.tsx +31 -0
  149. package/registry/components/tag.tsx +423 -0
  150. package/registry/components/textarea.tsx +625 -0
  151. package/registry/components/toggle.tsx +272 -0
  152. package/registry/components/toolbar.tsx +467 -0
  153. package/registry/components/tooltip.tsx +427 -0
  154. package/registry/examples/accordion-demo.tsx +34 -0
  155. package/registry/examples/alert-demo.tsx +14 -0
  156. package/registry/examples/attribute-demo.tsx +65 -0
  157. package/registry/examples/avatar-demo.tsx +74 -0
  158. package/registry/examples/badge-demo.tsx +53 -0
  159. package/registry/examples/button-demo.tsx +83 -0
  160. package/registry/examples/button-group-demo.tsx +42 -0
  161. package/registry/examples/card-demo.tsx +48 -0
  162. package/registry/examples/checkbox-demo.tsx +67 -0
  163. package/registry/examples/date-picker-demo.tsx +74 -0
  164. package/registry/examples/divider-demo.tsx +17 -0
  165. package/registry/examples/expanded/activity-feed-demo.tsx +22 -0
  166. package/registry/examples/expanded/banner-demo.tsx +23 -0
  167. package/registry/examples/expanded/catalog-components-demo.tsx +5 -0
  168. package/registry/examples/expanded/command-bar-demo.tsx +10 -0
  169. package/registry/examples/expanded/icons-demo.tsx +5 -0
  170. package/registry/examples/expanded/onboarding-step-demo.tsx +11 -0
  171. package/registry/examples/expanded/page-header-demo.tsx +19 -0
  172. package/registry/examples/expanded/slideout-demo.tsx +15 -0
  173. package/registry/examples/expanded/steps-demo.tsx +18 -0
  174. package/registry/examples/expanded/tabs-demo.tsx +13 -0
  175. package/registry/examples/expanded/timeline-demo.tsx +18 -0
  176. package/registry/examples/input-demo.tsx +87 -0
  177. package/registry/examples/menu-demo.tsx +109 -0
  178. package/registry/examples/modal-demo.tsx +16 -0
  179. package/registry/examples/pagination-demo.tsx +17 -0
  180. package/registry/examples/progress-demo.tsx +37 -0
  181. package/registry/examples/progress-slider-demo.tsx +29 -0
  182. package/registry/examples/radio-card-demo.tsx +51 -0
  183. package/registry/examples/radio-group-demo.tsx +62 -0
  184. package/registry/examples/select-demo.tsx +73 -0
  185. package/registry/examples/slider-demo.tsx +31 -0
  186. package/registry/examples/social-button-demo.tsx +51 -0
  187. package/registry/examples/tag-demo.tsx +29 -0
  188. package/registry/examples/textarea-demo.tsx +79 -0
  189. package/registry/examples/toggle-demo.tsx +59 -0
  190. package/registry/examples/toolbar-demo.tsx +80 -0
  191. package/registry/examples/tooltip-demo.tsx +115 -0
  192. package/registry/hooks/use-direction.ts +27 -0
  193. package/registry/index.json +1213 -0
  194. package/registry/styles/globals.css +4600 -0
  195. package/registry/utils/cn.ts +6 -0
  196. package/src/cli/index.js +826 -0
  197. package/tokens/Color mode.zip +0 -0
  198. package/tokens/Numbers.zip +0 -0
  199. package/tokens/Radius.zip +0 -0
  200. package/tokens/Theme.zip +0 -0
  201. package/tokens/banhaten.tokens.json +5525 -0
@@ -0,0 +1,380 @@
1
+ "use client"
2
+
3
+ import * as React from "react"
4
+ import * as SliderPrimitive from "@radix-ui/react-slider"
5
+ import { cva, type VariantProps } from "class-variance-authority"
6
+
7
+ import { cn } from "@/lib/utils"
8
+
9
+ type SliderTone = "brand" | "neutral" | "success" | "warning" | "danger"
10
+ type SliderSize = "sm" | "default" | "lg"
11
+ type SliderVariant = "default" | "pill"
12
+ type SliderIndicator =
13
+ | "none"
14
+ | "lower-text"
15
+ | "upper-text"
16
+ | "lower-tooltip"
17
+ | "upper-tooltip"
18
+ type SliderThumbLabel =
19
+ | React.ReactNode
20
+ | ((value: number, index: number, values: number[]) => React.ReactNode)
21
+
22
+ const sliderVariants = cva(
23
+ [
24
+ "group/slider relative flex touch-none select-none items-center",
25
+ "data-[disabled]:pointer-events-none data-[disabled]:opacity-[var(--bh-opacity-60)]",
26
+ "data-[orientation=horizontal]:w-[var(--bh-slider-width)] data-[orientation=horizontal]:max-w-full",
27
+ "data-[orientation=vertical]:h-[var(--bh-slider-vertical-height)] data-[orientation=vertical]:w-fit data-[orientation=vertical]:flex-col",
28
+ "[--bh-slider-track-size-current:var(--bh-slider-track-size-md)]",
29
+ "[--bh-slider-thumb-size-current:var(--bh-slider-thumb-size-md)]",
30
+ "[--bh-slider-thumb-ring-width-current:var(--bh-slider-thumb-ring-width-md)]",
31
+ "[--bh-slider-track-bg:var(--bh-bg-neutral-soft)]",
32
+ "[--bh-slider-range-bg:var(--bh-interactive-indicator-default)]",
33
+ "[--bh-slider-thumb-bg:var(--bh-slider-range-bg)]",
34
+ "[--bh-slider-thumb-ring-bg:var(--bh-bg-always-white)]",
35
+ "[--bh-slider-thumb-pill-bg:var(--bh-bg-brand-subtle)]",
36
+ "[--bh-slider-thumb-pill-border:var(--bh-border-brand-subtle)]",
37
+ "[--bh-slider-thumb-grip-bg:var(--bh-border-brand-strong)]",
38
+ ],
39
+ {
40
+ variants: {
41
+ size: {
42
+ sm: [
43
+ "[--bh-slider-track-size-current:var(--bh-slider-track-size-sm)]",
44
+ "[--bh-slider-thumb-size-current:var(--bh-slider-thumb-size-sm)]",
45
+ "[--bh-slider-thumb-ring-width-current:var(--bh-slider-thumb-ring-width-sm)]",
46
+ ],
47
+ default: [
48
+ "[--bh-slider-track-size-current:var(--bh-slider-track-size-md)]",
49
+ "[--bh-slider-thumb-size-current:var(--bh-slider-thumb-size-md)]",
50
+ "[--bh-slider-thumb-ring-width-current:var(--bh-slider-thumb-ring-width-md)]",
51
+ ],
52
+ lg: [
53
+ "[--bh-slider-track-size-current:var(--bh-slider-track-size-lg)]",
54
+ "[--bh-slider-thumb-size-current:var(--bh-slider-thumb-size-lg)]",
55
+ "[--bh-slider-thumb-ring-width-current:var(--bh-slider-thumb-ring-width-lg)]",
56
+ ],
57
+ },
58
+ tone: {
59
+ brand: [
60
+ "[--bh-slider-range-bg:var(--bh-interactive-indicator-default)]",
61
+ "[--bh-slider-thumb-pill-bg:var(--bh-bg-brand-subtle)]",
62
+ "[--bh-slider-thumb-pill-border:var(--bh-border-brand-subtle)]",
63
+ "[--bh-slider-thumb-grip-bg:var(--bh-border-brand-strong)]",
64
+ "[--bh-slider-pill-track-bg:var(--bh-bg-brand-soft)]",
65
+ ],
66
+ neutral: [
67
+ "[--bh-slider-range-bg:var(--bh-bg-neutral-bold)]",
68
+ "[--bh-slider-thumb-pill-bg:var(--bh-bg-neutral-subtle)]",
69
+ "[--bh-slider-thumb-pill-border:var(--bh-border-subtle)]",
70
+ "[--bh-slider-thumb-grip-bg:var(--bh-border-strong)]",
71
+ "[--bh-slider-pill-track-bg:var(--bh-bg-neutral-soft)]",
72
+ ],
73
+ success: [
74
+ "[--bh-slider-range-bg:var(--bh-bg-success-default)]",
75
+ "[--bh-slider-thumb-pill-bg:var(--bh-bg-success-subtle)]",
76
+ "[--bh-slider-thumb-pill-border:var(--bh-border-success-subtle)]",
77
+ "[--bh-slider-thumb-grip-bg:var(--bh-border-success-strong)]",
78
+ "[--bh-slider-pill-track-bg:var(--bh-bg-success-soft)]",
79
+ ],
80
+ warning: [
81
+ "[--bh-slider-range-bg:var(--bh-bg-warning-default)]",
82
+ "[--bh-slider-thumb-pill-bg:var(--bh-bg-warning-subtle)]",
83
+ "[--bh-slider-thumb-pill-border:var(--bh-border-warning-subtle)]",
84
+ "[--bh-slider-thumb-grip-bg:var(--bh-border-warning-strong)]",
85
+ "[--bh-slider-pill-track-bg:var(--bh-bg-warning-soft)]",
86
+ ],
87
+ danger: [
88
+ "[--bh-slider-range-bg:var(--bh-bg-danger-default)]",
89
+ "[--bh-slider-thumb-pill-bg:var(--bh-bg-danger-subtle)]",
90
+ "[--bh-slider-thumb-pill-border:var(--bh-border-danger-subtle)]",
91
+ "[--bh-slider-thumb-grip-bg:var(--bh-border-danger-strong)]",
92
+ "[--bh-slider-pill-track-bg:var(--bh-bg-danger-soft)]",
93
+ ],
94
+ },
95
+ variant: {
96
+ default: "",
97
+ pill: [
98
+ "[--bh-slider-track-size-current:var(--bh-slider-pill-height)]",
99
+ "[--bh-slider-thumb-size-current:var(--bh-slider-pill-height)]",
100
+ "[--bh-slider-track-bg:var(--bh-slider-pill-track-bg)]",
101
+ ],
102
+ },
103
+ },
104
+ defaultVariants: {
105
+ size: "default",
106
+ tone: "brand",
107
+ variant: "default",
108
+ },
109
+ }
110
+ )
111
+
112
+ type SliderProps = Omit<
113
+ React.ComponentPropsWithoutRef<typeof SliderPrimitive.Root>,
114
+ "children"
115
+ > &
116
+ VariantProps<typeof sliderVariants> & {
117
+ ariaLabels?: string[]
118
+ ariaValueText?:
119
+ | string
120
+ | string[]
121
+ | ((value: number, index: number, values: number[]) => string)
122
+ indicator?: SliderIndicator
123
+ showThumbGrip?: boolean
124
+ showTicks?: boolean
125
+ thumbLabel?: SliderThumbLabel
126
+ thumbLabels?: React.ReactNode[]
127
+ tickCount?: number
128
+ }
129
+
130
+ function Slider({
131
+ ariaLabels,
132
+ ariaValueText,
133
+ className,
134
+ defaultValue,
135
+ indicator,
136
+ max = 100,
137
+ min = 0,
138
+ onValueChange,
139
+ orientation = "horizontal",
140
+ showThumbGrip,
141
+ showTicks = false,
142
+ size,
143
+ thumbLabel,
144
+ thumbLabels,
145
+ tickCount = 8,
146
+ tone,
147
+ value,
148
+ variant,
149
+ ...props
150
+ }: SliderProps) {
151
+ const isControlled = value !== undefined
152
+ const initialValueRef = React.useRef(
153
+ normalizeSliderValue(value ?? defaultValue ?? [min])
154
+ )
155
+ const [uncontrolledValue, setUncontrolledValue] = React.useState(
156
+ initialValueRef.current
157
+ )
158
+ const values = normalizeSliderValue(isControlled ? value : uncontrolledValue)
159
+ const resolvedVariant = variant ?? "default"
160
+ const hasExplicitThumbLabel =
161
+ thumbLabel !== undefined || thumbLabels !== undefined
162
+ const resolvedIndicator =
163
+ indicator ?? (hasExplicitThumbLabel && resolvedVariant !== "pill" ? "lower-text" : "none")
164
+ const shouldShowGrip = showThumbGrip ?? resolvedVariant === "pill"
165
+
166
+ return (
167
+ <SliderPrimitive.Root
168
+ data-indicator={resolvedIndicator}
169
+ data-slot="slider"
170
+ data-variant={resolvedVariant}
171
+ defaultValue={isControlled ? undefined : initialValueRef.current}
172
+ max={max}
173
+ min={min}
174
+ onValueChange={(nextValue) => {
175
+ if (!isControlled) {
176
+ setUncontrolledValue(nextValue)
177
+ }
178
+ onValueChange?.(nextValue)
179
+ }}
180
+ orientation={orientation}
181
+ value={value}
182
+ className={cn(sliderVariants({ size, tone, variant }), className)}
183
+ {...props}
184
+ >
185
+ <SliderPrimitive.Track
186
+ data-slot="slider-track"
187
+ onPointerDown={
188
+ resolvedVariant === "pill" ? (event) => event.stopPropagation() : undefined
189
+ }
190
+ className="relative grow overflow-hidden rounded-[var(--bh-slider-track-radius)] bg-[var(--bh-slider-track-bg)] data-[orientation=horizontal]:h-[var(--bh-slider-track-size-current)] data-[orientation=horizontal]:w-full data-[orientation=vertical]:h-full data-[orientation=vertical]:w-[var(--bh-slider-track-size-current)] group-data-[variant=pill]/slider:rounded-[var(--bh-slider-pill-radius)]"
191
+ >
192
+ {showTicks ? (
193
+ <span
194
+ aria-hidden="true"
195
+ data-orientation={orientation}
196
+ data-slot="slider-ticks"
197
+ className="pointer-events-none absolute inset-0 z-[var(--bh-z-base)] flex items-center justify-evenly gap-[var(--bh-slider-tick-gap)] px-[var(--bh-space-6xl-32)] data-[orientation=vertical]:flex-col data-[orientation=vertical]:px-0 data-[orientation=vertical]:py-[var(--bh-space-xl-12)]"
198
+ >
199
+ {Array.from({ length: tickCount }).map((_, index) => (
200
+ <span
201
+ key={index}
202
+ className="block size-[var(--bh-slider-tick-size)] rounded-[var(--bh-radius-full)] bg-[var(--bh-slider-tick-bg)]"
203
+ />
204
+ ))}
205
+ </span>
206
+ ) : null}
207
+ <SliderPrimitive.Range
208
+ data-slot="slider-range"
209
+ className="absolute z-[var(--bh-z-raised)] bg-[var(--bh-slider-range-bg)] data-[orientation=horizontal]:h-full data-[orientation=vertical]:w-full group-data-[variant=pill]/slider:bg-transparent"
210
+ />
211
+ </SliderPrimitive.Track>
212
+
213
+ {values.map((thumbValue, index) => {
214
+ const label = getThumbLabel({
215
+ index,
216
+ thumbLabel,
217
+ thumbLabels,
218
+ value: thumbValue,
219
+ values,
220
+ })
221
+ const indicatorLabel = hasRenderableContent(label)
222
+ ? label
223
+ : formatSliderValue(thumbValue)
224
+ const showIndicator =
225
+ resolvedVariant !== "pill" && resolvedIndicator !== "none"
226
+ const showInlineLabel =
227
+ resolvedVariant === "pill" && hasRenderableContent(label)
228
+
229
+ return (
230
+ <SliderPrimitive.Thumb
231
+ aria-label={getThumbAriaLabel({
232
+ ariaLabels,
233
+ index,
234
+ values,
235
+ fallback: props["aria-label"],
236
+ })}
237
+ aria-valuetext={getThumbAriaValueText({
238
+ ariaValueText,
239
+ index,
240
+ value: thumbValue,
241
+ values,
242
+ })}
243
+ data-has-indicator={showIndicator ? "true" : undefined}
244
+ data-has-label={showInlineLabel ? "true" : undefined}
245
+ data-slot="slider-thumb"
246
+ key={index}
247
+ className="relative z-[var(--bh-z-popover)] flex size-[var(--bh-slider-thumb-size-current)] shrink-0 items-center justify-center rounded-[var(--bh-slider-thumb-radius)] border-[var(--bh-slider-thumb-ring-width-current)] border-[var(--bh-slider-thumb-ring-bg)] bg-[var(--bh-slider-thumb-bg)] text-[var(--bh-content-default)] shadow-[var(--shadow-slider-thumb)] outline-none transition-[background-color,border-color,box-shadow] focus-visible:shadow-[var(--shadow-slider-thumb-focus)] group-data-[variant=pill]/slider:h-[var(--bh-slider-pill-height)] group-data-[variant=pill]/slider:w-[var(--bh-slider-pill-thumb-width)] group-data-[variant=pill]/slider:justify-end group-data-[variant=pill]/slider:rounded-[var(--bh-slider-pill-radius)] group-data-[variant=pill]/slider:border-[length:var(--bh-slider-pill-thumb-border-width)] group-data-[variant=pill]/slider:border-[var(--bh-slider-thumb-pill-border)] group-data-[variant=pill]/slider:bg-[var(--bh-slider-thumb-pill-bg)] group-data-[variant=pill]/slider:px-[var(--bh-space-lg-10)] group-data-[disabled]/slider:border-[var(--bh-bg-default)] group-data-[disabled]/slider:bg-[var(--bh-content-disabled)] group-data-[disabled]/slider:text-[var(--bh-content-disabled)] group-data-[variant=pill]/slider:group-data-[disabled]/slider:border-[var(--bh-border-disabled)] group-data-[variant=pill]/slider:group-data-[disabled]/slider:bg-[var(--bh-bg-disabled)]"
248
+ >
249
+ {showIndicator ? (
250
+ <SliderThumbIndicator
251
+ indicator={resolvedIndicator as Exclude<SliderIndicator, "none">}
252
+ >
253
+ {indicatorLabel}
254
+ </SliderThumbIndicator>
255
+ ) : null}
256
+ {showInlineLabel ? (
257
+ <span
258
+ data-slot="slider-thumb-label"
259
+ className="pointer-events-none me-auto min-w-0 truncate pe-[var(--bh-space-sm-6)] text-[length:var(--bh-text-body-sm-medium-font-size)] leading-[var(--bh-text-body-sm-medium-line-height)] font-[var(--bh-text-body-sm-medium-font-weight)] tracking-[var(--bh-text-body-sm-medium-letter-spacing)]"
260
+ >
261
+ {label}
262
+ </span>
263
+ ) : null}
264
+ {shouldShowGrip ? (
265
+ <span
266
+ aria-hidden="true"
267
+ data-slot="slider-thumb-grip"
268
+ className="pointer-events-none block h-[var(--bh-space-6xl-32)] w-[var(--bh-slider-thumb-grip-width)] shrink-0 rounded-[var(--bh-radius-full)] bg-[var(--bh-slider-thumb-grip-bg)]"
269
+ />
270
+ ) : null}
271
+ </SliderPrimitive.Thumb>
272
+ )
273
+ })}
274
+ </SliderPrimitive.Root>
275
+ )
276
+ }
277
+
278
+ function SliderThumbIndicator({
279
+ children,
280
+ indicator,
281
+ }: {
282
+ children: React.ReactNode
283
+ indicator: Exclude<SliderIndicator, "none">
284
+ }) {
285
+ const isTooltip = indicator.endsWith("tooltip")
286
+ const isUpper = indicator.startsWith("upper")
287
+
288
+ return (
289
+ <span
290
+ data-indicator={indicator}
291
+ data-slot="slider-thumb-indicator"
292
+ dir="auto"
293
+ className={cn(
294
+ "pointer-events-none absolute left-1/2 z-[var(--bh-z-overlay)] -translate-x-1/2 whitespace-nowrap text-center",
295
+ isUpper
296
+ ? "bottom-[calc(100%+var(--bh-slider-indicator-gap))]"
297
+ : "top-[calc(100%+var(--bh-slider-indicator-gap))]",
298
+ isTooltip
299
+ ? "rounded-[var(--bh-radius-md-6)] border border-[var(--bh-border-subtle)] bg-[var(--bh-bg-raised)] px-[var(--bh-space-md-8)] py-[var(--bh-space-sm-6)] text-[length:var(--bh-text-body-2xs-medium-font-size)] leading-[var(--bh-text-body-2xs-medium-line-height)] font-[var(--bh-text-body-2xs-medium-font-weight)] tracking-[var(--bh-text-body-2xs-medium-letter-spacing)] text-[var(--bh-content-default)] shadow-[var(--shadow-slider-tooltip)]"
300
+ : "text-[length:var(--bh-text-body-2xs-regular-font-size)] leading-[var(--bh-text-body-2xs-regular-line-height)] font-[var(--bh-text-body-2xs-regular-font-weight)] tracking-[var(--bh-text-body-2xs-regular-letter-spacing)] text-[var(--bh-content-subtle)]"
301
+ )}
302
+ >
303
+ {children}
304
+ </span>
305
+ )
306
+ }
307
+
308
+ function normalizeSliderValue(value: number[] | readonly number[] | undefined) {
309
+ return Array.isArray(value) && value.length > 0 ? [...value] : [0]
310
+ }
311
+
312
+ function formatSliderValue(value: number) {
313
+ return `${Number.isInteger(value) ? value : value.toFixed(1)}%`
314
+ }
315
+
316
+ function hasRenderableContent(value: React.ReactNode) {
317
+ return value !== undefined && value !== null && value !== false && value !== ""
318
+ }
319
+
320
+ function getThumbLabel({
321
+ index,
322
+ thumbLabel,
323
+ thumbLabels,
324
+ value,
325
+ values,
326
+ }: {
327
+ index: number
328
+ thumbLabel?: SliderThumbLabel
329
+ thumbLabels?: React.ReactNode[]
330
+ value: number
331
+ values: number[]
332
+ }) {
333
+ if (thumbLabels?.[index] !== undefined) return thumbLabels[index]
334
+ if (typeof thumbLabel === "function") return thumbLabel(value, index, values)
335
+ return thumbLabel
336
+ }
337
+
338
+ function getThumbAriaLabel({
339
+ ariaLabels,
340
+ fallback,
341
+ index,
342
+ values,
343
+ }: {
344
+ ariaLabels?: string[]
345
+ fallback?: string
346
+ index: number
347
+ values: number[]
348
+ }) {
349
+ if (ariaLabels?.[index]) return ariaLabels[index]
350
+ if (fallback && values.length > 1) return `${fallback} ${index + 1}`
351
+ return fallback
352
+ }
353
+
354
+ function getThumbAriaValueText({
355
+ ariaValueText,
356
+ index,
357
+ value,
358
+ values,
359
+ }: {
360
+ ariaValueText?: SliderProps["ariaValueText"]
361
+ index: number
362
+ value: number
363
+ values: number[]
364
+ }) {
365
+ if (typeof ariaValueText === "function") {
366
+ return ariaValueText(value, index, values)
367
+ }
368
+ if (Array.isArray(ariaValueText)) return ariaValueText[index]
369
+ return ariaValueText
370
+ }
371
+
372
+ export { Slider, sliderVariants }
373
+ export type {
374
+ SliderIndicator,
375
+ SliderProps,
376
+ SliderSize,
377
+ SliderThumbLabel,
378
+ SliderTone,
379
+ SliderVariant,
380
+ }