create-ec-app 1.8.0 → 1.9.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 (103) hide show
  1. package/README.md +72 -17
  2. package/dist/cssScope.js +3 -5
  3. package/dist/cssScope.js.map +1 -1
  4. package/dist/index.d.ts +46 -1
  5. package/dist/index.d.ts.map +1 -1
  6. package/dist/index.js +129 -53
  7. package/dist/index.js.map +1 -1
  8. package/dist/libFunctions.d.ts +13 -6
  9. package/dist/libFunctions.d.ts.map +1 -1
  10. package/dist/libFunctions.js +24 -9
  11. package/dist/libFunctions.js.map +1 -1
  12. package/dist/pcf.d.ts.map +1 -1
  13. package/dist/pcf.js +4 -1
  14. package/dist/pcf.js.map +1 -1
  15. package/dist/portalContainers.js +7 -5
  16. package/dist/portalContainers.js.map +1 -1
  17. package/package.json +18 -11
  18. package/scripts/build-generated.mjs +59 -0
  19. package/scripts/refresh-shadcn-template.ts +406 -0
  20. package/scripts/smoke-scaffold.mjs +245 -0
  21. package/templates/base/eslint.config.js +1 -1
  22. package/templates/base/package-lock.json +380 -476
  23. package/templates/base/package.json +14 -19
  24. package/templates/pcf/base/package-lock.json +35 -53
  25. package/templates/targets/code-apps/AGENTS.md +1 -1
  26. package/templates/targets/code-apps/CLAUDE.md +1 -0
  27. package/templates/targets/code-apps/package.patch.json +1 -1
  28. package/templates/targets/power-pages/AGENTS.md +1 -1
  29. package/templates/targets/power-pages/CLAUDE.md +1 -0
  30. package/templates/targets/power-pages/README.md +22 -2
  31. package/templates/targets/power-pages/src/App.patch.tsx +3 -1
  32. package/templates/targets/power-pages/src/components/shared/AuthError.tsx +18 -0
  33. package/templates/targets/power-pages/src/context/AuthContext.tsx +0 -4
  34. package/templates/targets/swa/CLAUDE.md +1 -0
  35. package/templates/targets/webresource/AGENTS.md +5 -4
  36. package/templates/targets/webresource/CLAUDE.md +1 -0
  37. package/templates/targets/webresource/README.md +5 -5
  38. package/templates/ui/kendo/package.patch.json +2 -2
  39. package/templates/ui/shadcn-ui/SHADCN_TEMPLATE.md +20 -0
  40. package/templates/ui/shadcn-ui/package.patch.json +18 -9
  41. package/templates/ui/shadcn-ui/src/components/ui/accordion.tsx +79 -0
  42. package/templates/ui/shadcn-ui/src/components/ui/alert-dialog.tsx +199 -0
  43. package/templates/ui/shadcn-ui/src/components/ui/alert.tsx +76 -0
  44. package/templates/ui/shadcn-ui/src/components/ui/aspect-ratio.tsx +11 -0
  45. package/templates/ui/shadcn-ui/src/components/ui/attachment.tsx +206 -0
  46. package/templates/ui/shadcn-ui/src/components/ui/avatar.tsx +110 -0
  47. package/templates/ui/shadcn-ui/src/components/ui/badge.tsx +49 -0
  48. package/templates/ui/shadcn-ui/src/components/ui/breadcrumb.tsx +122 -0
  49. package/templates/ui/shadcn-ui/src/components/ui/bubble.tsx +125 -0
  50. package/templates/ui/shadcn-ui/src/components/ui/button-group.tsx +83 -0
  51. package/templates/ui/shadcn-ui/src/components/ui/button.tsx +67 -0
  52. package/templates/ui/shadcn-ui/src/components/ui/calendar.tsx +222 -0
  53. package/templates/ui/shadcn-ui/src/components/ui/card.tsx +103 -0
  54. package/templates/ui/shadcn-ui/src/components/ui/carousel.tsx +240 -0
  55. package/templates/ui/shadcn-ui/src/components/ui/chart.tsx +373 -0
  56. package/templates/ui/shadcn-ui/src/components/ui/checkbox.tsx +31 -0
  57. package/templates/ui/shadcn-ui/src/components/ui/collapsible.tsx +33 -0
  58. package/templates/ui/shadcn-ui/src/components/ui/combobox.tsx +299 -0
  59. package/templates/ui/shadcn-ui/src/components/ui/command.tsx +195 -0
  60. package/templates/ui/shadcn-ui/src/components/ui/context-menu.tsx +264 -0
  61. package/templates/ui/shadcn-ui/src/components/ui/dialog.tsx +170 -0
  62. package/templates/ui/shadcn-ui/src/components/ui/direction.tsx +22 -0
  63. package/templates/ui/shadcn-ui/src/components/ui/drawer.tsx +134 -0
  64. package/templates/ui/shadcn-ui/src/components/ui/dropdown-menu.tsx +272 -0
  65. package/templates/ui/shadcn-ui/src/components/ui/empty.tsx +104 -0
  66. package/templates/ui/shadcn-ui/src/components/ui/field.tsx +236 -0
  67. package/templates/ui/shadcn-ui/src/components/ui/hover-card.tsx +44 -0
  68. package/templates/ui/shadcn-ui/src/components/ui/input-group.tsx +156 -0
  69. package/templates/ui/shadcn-ui/src/components/ui/input-otp.tsx +87 -0
  70. package/templates/ui/shadcn-ui/src/components/ui/input.tsx +19 -0
  71. package/templates/ui/shadcn-ui/src/components/ui/item.tsx +196 -0
  72. package/templates/ui/shadcn-ui/src/components/ui/kbd.tsx +26 -0
  73. package/templates/ui/shadcn-ui/src/components/ui/label.tsx +22 -0
  74. package/templates/ui/shadcn-ui/src/components/ui/marker.tsx +69 -0
  75. package/templates/ui/shadcn-ui/src/components/ui/menubar.tsx +282 -0
  76. package/templates/ui/shadcn-ui/src/components/ui/message-scroller.tsx +129 -0
  77. package/templates/ui/shadcn-ui/src/components/ui/message.tsx +92 -0
  78. package/templates/ui/shadcn-ui/src/components/ui/native-select.tsx +61 -0
  79. package/templates/ui/shadcn-ui/src/components/ui/navigation-menu.tsx +164 -0
  80. package/templates/ui/shadcn-ui/src/components/ui/pagination.tsx +129 -0
  81. package/templates/ui/shadcn-ui/src/components/ui/popover.tsx +89 -0
  82. package/templates/ui/shadcn-ui/src/components/ui/progress.tsx +31 -0
  83. package/templates/ui/shadcn-ui/src/components/ui/radio-group.tsx +42 -0
  84. package/templates/ui/shadcn-ui/src/components/ui/resizable.tsx +50 -0
  85. package/templates/ui/shadcn-ui/src/components/ui/scroll-area.tsx +53 -0
  86. package/templates/ui/shadcn-ui/src/components/ui/select.tsx +194 -0
  87. package/templates/ui/shadcn-ui/src/components/ui/separator.tsx +26 -0
  88. package/templates/ui/shadcn-ui/src/components/ui/sheet.tsx +149 -0
  89. package/templates/ui/shadcn-ui/src/components/ui/sidebar.tsx +702 -0
  90. package/templates/ui/shadcn-ui/src/components/ui/skeleton.tsx +13 -0
  91. package/templates/ui/shadcn-ui/src/components/ui/slider.tsx +59 -0
  92. package/templates/ui/shadcn-ui/src/components/ui/sonner.tsx +47 -0
  93. package/templates/ui/shadcn-ui/src/components/ui/spinner.tsx +10 -0
  94. package/templates/ui/shadcn-ui/src/components/ui/switch.tsx +33 -0
  95. package/templates/ui/shadcn-ui/src/components/ui/table.tsx +114 -0
  96. package/templates/ui/shadcn-ui/src/components/ui/tabs.tsx +90 -0
  97. package/templates/ui/shadcn-ui/src/components/ui/textarea.tsx +18 -0
  98. package/templates/ui/shadcn-ui/src/components/ui/toggle-group.tsx +87 -0
  99. package/templates/ui/shadcn-ui/src/components/ui/toggle.tsx +45 -0
  100. package/templates/ui/shadcn-ui/src/components/ui/tooltip.tsx +59 -0
  101. package/templates/ui/shadcn-ui/src/index.patch.css +0 -118
  102. package/templates/ui/shadcn-ui/src/runtime/PortalContainer.ts +8 -0
  103. package/templates/base/biome.json +0 -54
@@ -0,0 +1,103 @@
1
+ import * as React from "react"
2
+
3
+ import { cn } from "@/lib/utils"
4
+
5
+ function Card({
6
+ className,
7
+ size = "default",
8
+ ...props
9
+ }: React.ComponentProps<"div"> & { size?: "default" | "sm" }) {
10
+ return (
11
+ <div
12
+ data-slot="card"
13
+ data-size={size}
14
+ className={cn(
15
+ "group/card flex flex-col gap-(--card-spacing) overflow-hidden rounded-xl bg-card py-(--card-spacing) text-sm text-card-foreground ring-1 ring-foreground/10 [--card-spacing:--spacing(4)] has-data-[slot=card-footer]:pb-0 has-[>img:first-child]:pt-0 data-[size=sm]:[--card-spacing:--spacing(3)] data-[size=sm]:has-data-[slot=card-footer]:pb-0 *:[img:first-child]:rounded-t-xl *:[img:last-child]:rounded-b-xl",
16
+ className
17
+ )}
18
+ {...props}
19
+ />
20
+ )
21
+ }
22
+
23
+ function CardHeader({ className, ...props }: React.ComponentProps<"div">) {
24
+ return (
25
+ <div
26
+ data-slot="card-header"
27
+ className={cn(
28
+ "group/card-header @container/card-header grid auto-rows-min items-start gap-1 rounded-t-xl px-(--card-spacing) has-data-[slot=card-action]:grid-cols-[1fr_auto] has-data-[slot=card-description]:grid-rows-[auto_auto] [.border-b]:pb-(--card-spacing)",
29
+ className
30
+ )}
31
+ {...props}
32
+ />
33
+ )
34
+ }
35
+
36
+ function CardTitle({ className, ...props }: React.ComponentProps<"div">) {
37
+ return (
38
+ <div
39
+ data-slot="card-title"
40
+ className={cn(
41
+ "text-base leading-snug font-medium group-data-[size=sm]/card:text-sm",
42
+ className
43
+ )}
44
+ {...props}
45
+ />
46
+ )
47
+ }
48
+
49
+ function CardDescription({ className, ...props }: React.ComponentProps<"div">) {
50
+ return (
51
+ <div
52
+ data-slot="card-description"
53
+ className={cn("text-sm text-muted-foreground", className)}
54
+ {...props}
55
+ />
56
+ )
57
+ }
58
+
59
+ function CardAction({ className, ...props }: React.ComponentProps<"div">) {
60
+ return (
61
+ <div
62
+ data-slot="card-action"
63
+ className={cn(
64
+ "col-start-2 row-span-2 row-start-1 self-start justify-self-end",
65
+ className
66
+ )}
67
+ {...props}
68
+ />
69
+ )
70
+ }
71
+
72
+ function CardContent({ className, ...props }: React.ComponentProps<"div">) {
73
+ return (
74
+ <div
75
+ data-slot="card-content"
76
+ className={cn("px-(--card-spacing)", className)}
77
+ {...props}
78
+ />
79
+ )
80
+ }
81
+
82
+ function CardFooter({ className, ...props }: React.ComponentProps<"div">) {
83
+ return (
84
+ <div
85
+ data-slot="card-footer"
86
+ className={cn(
87
+ "flex items-center rounded-b-xl border-t bg-muted/50 p-(--card-spacing)",
88
+ className
89
+ )}
90
+ {...props}
91
+ />
92
+ )
93
+ }
94
+
95
+ export {
96
+ Card,
97
+ CardHeader,
98
+ CardFooter,
99
+ CardTitle,
100
+ CardAction,
101
+ CardDescription,
102
+ CardContent,
103
+ }
@@ -0,0 +1,240 @@
1
+ import * as React from "react"
2
+ import useEmblaCarousel, {
3
+ type UseEmblaCarouselType,
4
+ } from "embla-carousel-react"
5
+
6
+ import { cn } from "@/lib/utils"
7
+ import { Button } from "@/components/ui/button"
8
+ import { ChevronLeftIcon, ChevronRightIcon } from "lucide-react"
9
+
10
+ type CarouselApi = UseEmblaCarouselType[1]
11
+ type UseCarouselParameters = Parameters<typeof useEmblaCarousel>
12
+ type CarouselOptions = UseCarouselParameters[0]
13
+ type CarouselPlugin = UseCarouselParameters[1]
14
+
15
+ type CarouselProps = {
16
+ opts?: CarouselOptions
17
+ plugins?: CarouselPlugin
18
+ orientation?: "horizontal" | "vertical"
19
+ setApi?: (api: CarouselApi) => void
20
+ }
21
+
22
+ type CarouselContextProps = {
23
+ carouselRef: ReturnType<typeof useEmblaCarousel>[0]
24
+ api: ReturnType<typeof useEmblaCarousel>[1]
25
+ scrollPrev: () => void
26
+ scrollNext: () => void
27
+ canScrollPrev: boolean
28
+ canScrollNext: boolean
29
+ } & CarouselProps
30
+
31
+ const CarouselContext = React.createContext<CarouselContextProps | null>(null)
32
+
33
+ function useCarousel() {
34
+ const context = React.useContext(CarouselContext)
35
+
36
+ if (!context) {
37
+ throw new Error("useCarousel must be used within a <Carousel />")
38
+ }
39
+
40
+ return context
41
+ }
42
+
43
+ function Carousel({
44
+ orientation = "horizontal",
45
+ opts,
46
+ setApi,
47
+ plugins,
48
+ className,
49
+ children,
50
+ ...props
51
+ }: React.ComponentProps<"div"> & CarouselProps) {
52
+ const [carouselRef, api] = useEmblaCarousel(
53
+ {
54
+ ...opts,
55
+ axis: orientation === "horizontal" ? "x" : "y",
56
+ },
57
+ plugins
58
+ )
59
+ const [canScrollPrev, setCanScrollPrev] = React.useState(false)
60
+ const [canScrollNext, setCanScrollNext] = React.useState(false)
61
+
62
+ const onSelect = React.useCallback((api: CarouselApi) => {
63
+ if (!api) return
64
+ setCanScrollPrev(api.canScrollPrev())
65
+ setCanScrollNext(api.canScrollNext())
66
+ }, [])
67
+
68
+ const scrollPrev = React.useCallback(() => {
69
+ api?.scrollPrev()
70
+ }, [api])
71
+
72
+ const scrollNext = React.useCallback(() => {
73
+ api?.scrollNext()
74
+ }, [api])
75
+
76
+ const handleKeyDown = React.useCallback(
77
+ (event: React.KeyboardEvent<HTMLDivElement>) => {
78
+ if (event.key === "ArrowLeft") {
79
+ event.preventDefault()
80
+ scrollPrev()
81
+ } else if (event.key === "ArrowRight") {
82
+ event.preventDefault()
83
+ scrollNext()
84
+ }
85
+ },
86
+ [scrollPrev, scrollNext]
87
+ )
88
+
89
+ React.useEffect(() => {
90
+ if (!api || !setApi) return
91
+ setApi(api)
92
+ }, [api, setApi])
93
+
94
+ React.useEffect(() => {
95
+ if (!api) return
96
+ onSelect(api)
97
+ api.on("reInit", onSelect)
98
+ api.on("select", onSelect)
99
+
100
+ return () => {
101
+ api?.off("select", onSelect)
102
+ }
103
+ }, [api, onSelect])
104
+
105
+ return (
106
+ <CarouselContext.Provider
107
+ value={{
108
+ carouselRef,
109
+ api: api,
110
+ opts,
111
+ orientation:
112
+ orientation || (opts?.axis === "y" ? "vertical" : "horizontal"),
113
+ scrollPrev,
114
+ scrollNext,
115
+ canScrollPrev,
116
+ canScrollNext,
117
+ }}
118
+ >
119
+ <div
120
+ onKeyDownCapture={handleKeyDown}
121
+ className={cn("relative", className)}
122
+ role="region"
123
+ aria-roledescription="carousel"
124
+ data-slot="carousel"
125
+ {...props}
126
+ >
127
+ {children}
128
+ </div>
129
+ </CarouselContext.Provider>
130
+ )
131
+ }
132
+
133
+ function CarouselContent({ className, ...props }: React.ComponentProps<"div">) {
134
+ const { carouselRef, orientation } = useCarousel()
135
+
136
+ return (
137
+ <div
138
+ ref={carouselRef}
139
+ className="overflow-hidden"
140
+ data-slot="carousel-content"
141
+ >
142
+ <div
143
+ className={cn(
144
+ "flex",
145
+ orientation === "horizontal" ? "-ml-4" : "-mt-4 flex-col",
146
+ className
147
+ )}
148
+ {...props}
149
+ />
150
+ </div>
151
+ )
152
+ }
153
+
154
+ function CarouselItem({ className, ...props }: React.ComponentProps<"div">) {
155
+ const { orientation } = useCarousel()
156
+
157
+ return (
158
+ <div
159
+ role="group"
160
+ aria-roledescription="slide"
161
+ data-slot="carousel-item"
162
+ className={cn(
163
+ "min-w-0 shrink-0 grow-0 basis-full",
164
+ orientation === "horizontal" ? "pl-4" : "pt-4",
165
+ className
166
+ )}
167
+ {...props}
168
+ />
169
+ )
170
+ }
171
+
172
+ function CarouselPrevious({
173
+ className,
174
+ variant = "outline",
175
+ size = "icon-sm",
176
+ ...props
177
+ }: React.ComponentProps<typeof Button>) {
178
+ const { orientation, scrollPrev, canScrollPrev } = useCarousel()
179
+
180
+ return (
181
+ <Button
182
+ data-slot="carousel-previous"
183
+ variant={variant}
184
+ size={size}
185
+ className={cn(
186
+ "absolute touch-manipulation rounded-full",
187
+ orientation === "horizontal"
188
+ ? "inset-y-0 -left-12 my-auto"
189
+ : "-top-12 left-1/2 -translate-x-1/2 rotate-90",
190
+ className
191
+ )}
192
+ disabled={!canScrollPrev}
193
+ onClick={scrollPrev}
194
+ {...props}
195
+ >
196
+ <ChevronLeftIcon />
197
+ <span className="sr-only">Previous slide</span>
198
+ </Button>
199
+ )
200
+ }
201
+
202
+ function CarouselNext({
203
+ className,
204
+ variant = "outline",
205
+ size = "icon-sm",
206
+ ...props
207
+ }: React.ComponentProps<typeof Button>) {
208
+ const { orientation, scrollNext, canScrollNext } = useCarousel()
209
+
210
+ return (
211
+ <Button
212
+ data-slot="carousel-next"
213
+ variant={variant}
214
+ size={size}
215
+ className={cn(
216
+ "absolute touch-manipulation rounded-full",
217
+ orientation === "horizontal"
218
+ ? "inset-y-0 -right-12 my-auto"
219
+ : "-bottom-12 left-1/2 -translate-x-1/2 rotate-90",
220
+ className
221
+ )}
222
+ disabled={!canScrollNext}
223
+ onClick={scrollNext}
224
+ {...props}
225
+ >
226
+ <ChevronRightIcon />
227
+ <span className="sr-only">Next slide</span>
228
+ </Button>
229
+ )
230
+ }
231
+
232
+ export {
233
+ type CarouselApi,
234
+ Carousel,
235
+ CarouselContent,
236
+ CarouselItem,
237
+ CarouselPrevious,
238
+ CarouselNext,
239
+ useCarousel,
240
+ }
@@ -0,0 +1,373 @@
1
+ "use client"
2
+
3
+ import * as React from "react"
4
+ import * as RechartsPrimitive from "recharts"
5
+ import type { TooltipValueType } from "recharts"
6
+
7
+ import { cn } from "@/lib/utils"
8
+
9
+ // Format: { THEME_NAME: CSS_SELECTOR }
10
+ const THEMES = { light: "", dark: ".dark" } as const
11
+
12
+ const INITIAL_DIMENSION = { width: 320, height: 200 } as const
13
+ type TooltipNameType = number | string
14
+
15
+ export type ChartConfig = Record<
16
+ string,
17
+ {
18
+ label?: React.ReactNode
19
+ icon?: React.ComponentType
20
+ } & (
21
+ | { color?: string; theme?: never }
22
+ | { color?: never; theme: Record<keyof typeof THEMES, string> }
23
+ )
24
+ >
25
+
26
+ type ChartContextProps = {
27
+ config: ChartConfig
28
+ }
29
+
30
+ const ChartContext = React.createContext<ChartContextProps | null>(null)
31
+
32
+ function useChart() {
33
+ const context = React.useContext(ChartContext)
34
+
35
+ if (!context) {
36
+ throw new Error("useChart must be used within a <ChartContainer />")
37
+ }
38
+
39
+ return context
40
+ }
41
+
42
+ function ChartContainer({
43
+ id,
44
+ className,
45
+ children,
46
+ config,
47
+ initialDimension = INITIAL_DIMENSION,
48
+ ...props
49
+ }: React.ComponentProps<"div"> & {
50
+ config: ChartConfig
51
+ children: React.ComponentProps<
52
+ typeof RechartsPrimitive.ResponsiveContainer
53
+ >["children"]
54
+ initialDimension?: {
55
+ width: number
56
+ height: number
57
+ }
58
+ }) {
59
+ const uniqueId = React.useId()
60
+ const chartId = `chart-${id ?? uniqueId.replace(/:/g, "")}`
61
+
62
+ return (
63
+ <ChartContext.Provider value={{ config }}>
64
+ <div
65
+ data-slot="chart"
66
+ data-chart={chartId}
67
+ className={cn(
68
+ "flex aspect-video justify-center text-xs [&_.recharts-cartesian-axis-tick_text]:fill-muted-foreground [&_.recharts-cartesian-grid_line[stroke='#ccc']]:stroke-border/50 [&_.recharts-curve.recharts-tooltip-cursor]:stroke-border [&_.recharts-dot[stroke='#fff']]:stroke-transparent [&_.recharts-layer]:outline-hidden [&_.recharts-polar-grid_[stroke='#ccc']]:stroke-border [&_.recharts-radial-bar-background-sector]:fill-muted [&_.recharts-rectangle.recharts-tooltip-cursor]:fill-muted [&_.recharts-reference-line_[stroke='#ccc']]:stroke-border [&_.recharts-sector]:outline-hidden [&_.recharts-sector[stroke='#fff']]:stroke-transparent [&_.recharts-surface]:outline-hidden",
69
+ className
70
+ )}
71
+ {...props}
72
+ >
73
+ <ChartStyle id={chartId} config={config} />
74
+ <RechartsPrimitive.ResponsiveContainer
75
+ initialDimension={initialDimension}
76
+ >
77
+ {children}
78
+ </RechartsPrimitive.ResponsiveContainer>
79
+ </div>
80
+ </ChartContext.Provider>
81
+ )
82
+ }
83
+
84
+ const ChartStyle = ({ id, config }: { id: string; config: ChartConfig }) => {
85
+ const colorConfig = Object.entries(config).filter(
86
+ ([, config]) => config.theme ?? config.color
87
+ )
88
+
89
+ if (!colorConfig.length) {
90
+ return null
91
+ }
92
+
93
+ return (
94
+ <style
95
+ dangerouslySetInnerHTML={{
96
+ __html: Object.entries(THEMES)
97
+ .map(
98
+ ([theme, prefix]) => `
99
+ ${prefix} [data-chart=${id}] {
100
+ ${colorConfig
101
+ .map(([key, itemConfig]) => {
102
+ const color =
103
+ itemConfig.theme?.[theme as keyof typeof itemConfig.theme] ??
104
+ itemConfig.color
105
+ return color ? ` --color-${key}: ${color};` : null
106
+ })
107
+ .join("\n")}
108
+ }
109
+ `
110
+ )
111
+ .join("\n"),
112
+ }}
113
+ />
114
+ )
115
+ }
116
+
117
+ const ChartTooltip = RechartsPrimitive.Tooltip
118
+
119
+ function ChartTooltipContent({
120
+ active,
121
+ payload,
122
+ className,
123
+ indicator = "dot",
124
+ hideLabel = false,
125
+ hideIndicator = false,
126
+ label,
127
+ labelFormatter,
128
+ labelClassName,
129
+ formatter,
130
+ color,
131
+ nameKey,
132
+ labelKey,
133
+ }: React.ComponentProps<typeof RechartsPrimitive.Tooltip> &
134
+ React.ComponentProps<"div"> & {
135
+ hideLabel?: boolean
136
+ hideIndicator?: boolean
137
+ indicator?: "line" | "dot" | "dashed"
138
+ nameKey?: string
139
+ labelKey?: string
140
+ } & Omit<
141
+ RechartsPrimitive.DefaultTooltipContentProps<
142
+ TooltipValueType,
143
+ TooltipNameType
144
+ >,
145
+ "accessibilityLayer"
146
+ >) {
147
+ const { config } = useChart()
148
+
149
+ const tooltipLabel = React.useMemo(() => {
150
+ if (hideLabel || !payload?.length) {
151
+ return null
152
+ }
153
+
154
+ const [item] = payload
155
+ const key = `${labelKey ?? item?.dataKey ?? item?.name ?? "value"}`
156
+ const itemConfig = getPayloadConfigFromPayload(config, item, key)
157
+ const value =
158
+ !labelKey && typeof label === "string"
159
+ ? (config[label]?.label ?? label)
160
+ : itemConfig?.label
161
+
162
+ if (labelFormatter) {
163
+ return (
164
+ <div className={cn("font-medium", labelClassName)}>
165
+ {labelFormatter(value, payload)}
166
+ </div>
167
+ )
168
+ }
169
+
170
+ if (!value) {
171
+ return null
172
+ }
173
+
174
+ return <div className={cn("font-medium", labelClassName)}>{value}</div>
175
+ }, [
176
+ label,
177
+ labelFormatter,
178
+ payload,
179
+ hideLabel,
180
+ labelClassName,
181
+ config,
182
+ labelKey,
183
+ ])
184
+
185
+ if (!active || !payload?.length) {
186
+ return null
187
+ }
188
+
189
+ const nestLabel = payload.length === 1 && indicator !== "dot"
190
+
191
+ return (
192
+ <div
193
+ className={cn(
194
+ "grid min-w-32 items-start gap-1.5 rounded-lg border border-border/50 bg-background px-2.5 py-1.5 text-xs shadow-xl",
195
+ className
196
+ )}
197
+ >
198
+ {!nestLabel ? tooltipLabel : null}
199
+ <div className="grid gap-1.5">
200
+ {payload
201
+ .filter((item) => item.type !== "none")
202
+ .map((item, index) => {
203
+ const key = `${nameKey ?? item.name ?? item.dataKey ?? "value"}`
204
+ const itemConfig = getPayloadConfigFromPayload(config, item, key)
205
+ const indicatorColor = color ?? item.payload?.fill ?? item.color
206
+
207
+ return (
208
+ <div
209
+ key={index}
210
+ className={cn(
211
+ "flex w-full flex-wrap items-stretch gap-2 [&>svg]:h-2.5 [&>svg]:w-2.5 [&>svg]:text-muted-foreground",
212
+ indicator === "dot" && "items-center"
213
+ )}
214
+ >
215
+ {formatter && item?.value !== undefined && item.name ? (
216
+ formatter(item.value, item.name, item, index, item.payload)
217
+ ) : (
218
+ <>
219
+ {itemConfig?.icon ? (
220
+ <itemConfig.icon />
221
+ ) : (
222
+ !hideIndicator && (
223
+ <div
224
+ className={cn(
225
+ "shrink-0 rounded-[2px] border-(--color-border) bg-(--color-bg)",
226
+ {
227
+ "h-2.5 w-2.5": indicator === "dot",
228
+ "w-1": indicator === "line",
229
+ "w-0 border-[1.5px] border-dashed bg-transparent":
230
+ indicator === "dashed",
231
+ "my-0.5": nestLabel && indicator === "dashed",
232
+ }
233
+ )}
234
+ style={
235
+ {
236
+ "--color-bg": indicatorColor,
237
+ "--color-border": indicatorColor,
238
+ } as React.CSSProperties
239
+ }
240
+ />
241
+ )
242
+ )}
243
+ <div
244
+ className={cn(
245
+ "flex flex-1 justify-between leading-none",
246
+ nestLabel ? "items-end" : "items-center"
247
+ )}
248
+ >
249
+ <div className="grid gap-1.5">
250
+ {nestLabel ? tooltipLabel : null}
251
+ <span className="text-muted-foreground">
252
+ {itemConfig?.label ?? item.name}
253
+ </span>
254
+ </div>
255
+ {item.value != null && (
256
+ <span className="font-mono font-medium text-foreground tabular-nums">
257
+ {typeof item.value === "number"
258
+ ? item.value.toLocaleString()
259
+ : String(item.value)}
260
+ </span>
261
+ )}
262
+ </div>
263
+ </>
264
+ )}
265
+ </div>
266
+ )
267
+ })}
268
+ </div>
269
+ </div>
270
+ )
271
+ }
272
+
273
+ const ChartLegend = RechartsPrimitive.Legend
274
+
275
+ function ChartLegendContent({
276
+ className,
277
+ hideIcon = false,
278
+ payload,
279
+ verticalAlign = "bottom",
280
+ nameKey,
281
+ }: React.ComponentProps<"div"> & {
282
+ hideIcon?: boolean
283
+ nameKey?: string
284
+ } & RechartsPrimitive.DefaultLegendContentProps) {
285
+ const { config } = useChart()
286
+
287
+ if (!payload?.length) {
288
+ return null
289
+ }
290
+
291
+ return (
292
+ <div
293
+ className={cn(
294
+ "flex items-center justify-center gap-4",
295
+ verticalAlign === "top" ? "pb-3" : "pt-3",
296
+ className
297
+ )}
298
+ >
299
+ {payload
300
+ .filter((item) => item.type !== "none")
301
+ .map((item, index) => {
302
+ const key = `${nameKey ?? item.dataKey ?? "value"}`
303
+ const itemConfig = getPayloadConfigFromPayload(config, item, key)
304
+
305
+ return (
306
+ <div
307
+ key={index}
308
+ className={cn(
309
+ "flex items-center gap-1.5 [&>svg]:h-3 [&>svg]:w-3 [&>svg]:text-muted-foreground"
310
+ )}
311
+ >
312
+ {itemConfig?.icon && !hideIcon ? (
313
+ <itemConfig.icon />
314
+ ) : (
315
+ <div
316
+ className="h-2 w-2 shrink-0 rounded-[2px]"
317
+ style={{
318
+ backgroundColor: item.color,
319
+ }}
320
+ />
321
+ )}
322
+ {itemConfig?.label}
323
+ </div>
324
+ )
325
+ })}
326
+ </div>
327
+ )
328
+ }
329
+
330
+ function getPayloadConfigFromPayload(
331
+ config: ChartConfig,
332
+ payload: unknown,
333
+ key: string
334
+ ) {
335
+ if (typeof payload !== "object" || payload === null) {
336
+ return undefined
337
+ }
338
+
339
+ const payloadPayload =
340
+ "payload" in payload &&
341
+ typeof payload.payload === "object" &&
342
+ payload.payload !== null
343
+ ? payload.payload
344
+ : undefined
345
+
346
+ let configLabelKey: string = key
347
+
348
+ if (
349
+ key in payload &&
350
+ typeof payload[key as keyof typeof payload] === "string"
351
+ ) {
352
+ configLabelKey = payload[key as keyof typeof payload] as string
353
+ } else if (
354
+ payloadPayload &&
355
+ key in payloadPayload &&
356
+ typeof payloadPayload[key as keyof typeof payloadPayload] === "string"
357
+ ) {
358
+ configLabelKey = payloadPayload[
359
+ key as keyof typeof payloadPayload
360
+ ] as string
361
+ }
362
+
363
+ return configLabelKey in config ? config[configLabelKey] : config[key]
364
+ }
365
+
366
+ export {
367
+ ChartContainer,
368
+ ChartTooltip,
369
+ ChartTooltipContent,
370
+ ChartLegend,
371
+ ChartLegendContent,
372
+ ChartStyle,
373
+ }