@godxjp/ui 5.0.2 → 6.0.1

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 (242) hide show
  1. package/README.md +101 -142
  2. package/dist/app/index.d.ts +140 -0
  3. package/dist/app/index.js +38 -0
  4. package/dist/app.prop-IobwLwaM.d.ts +90 -0
  5. package/dist/checkbox-NkFkqsQ8.d.ts +13 -0
  6. package/dist/chunk-2XW7J3EI.js +226 -0
  7. package/dist/chunk-3F2AKYRD.js +416 -0
  8. package/dist/chunk-3KPEZ5CF.js +37 -0
  9. package/dist/chunk-3UGU5TYP.js +89 -0
  10. package/dist/chunk-6RA3KSVK.js +588 -0
  11. package/dist/chunk-7NZFVD24.js +122 -0
  12. package/dist/chunk-7PWBC4BY.js +25 -0
  13. package/dist/chunk-7S7MYFXE.js +61 -0
  14. package/dist/chunk-7WRZG2IG.js +71 -0
  15. package/dist/chunk-B775Y6BE.js +1 -0
  16. package/dist/chunk-BHV2FUOA.js +111 -0
  17. package/dist/chunk-BI3HERR7.js +70 -0
  18. package/dist/chunk-BPSKQUL2.js +68 -0
  19. package/dist/chunk-CDWPQ5RP.js +187 -0
  20. package/dist/chunk-CP2LET6N.js +244 -0
  21. package/dist/chunk-CQBADMFG.js +20 -0
  22. package/dist/chunk-CRERCLIZ.js +33 -0
  23. package/dist/chunk-DU6ZYZRP.js +238 -0
  24. package/dist/chunk-DY5C44UP.js +55 -0
  25. package/dist/chunk-E4HJNQ62.js +117 -0
  26. package/dist/chunk-E76QIYSY.js +93 -0
  27. package/dist/chunk-F7PG4OEV.js +37 -0
  28. package/dist/chunk-GDSVW62T.js +171 -0
  29. package/dist/chunk-H2FHJOLU.js +178 -0
  30. package/dist/chunk-HJEBRCXL.js +55 -0
  31. package/dist/chunk-ICM6XBST.js +16 -0
  32. package/dist/chunk-IK7I3ABN.js +88 -0
  33. package/dist/chunk-L6J44O74.js +144 -0
  34. package/dist/chunk-LDSLS6HE.js +1 -0
  35. package/dist/chunk-LVNUHUEZ.js +191 -0
  36. package/dist/chunk-M64MVRLS.js +92 -0
  37. package/dist/chunk-NGQW3KEM.js +402 -0
  38. package/dist/chunk-NZ4FOC5N.js +559 -0
  39. package/dist/chunk-PIIRNAXA.js +26 -0
  40. package/dist/chunk-S66TJXJU.js +33 -0
  41. package/dist/chunk-SMLKNECP.js +133 -0
  42. package/dist/chunk-TAHBM3F2.js +66 -0
  43. package/dist/chunk-TO33OY4L.js +150 -0
  44. package/dist/chunk-TO7URV7U.js +51 -0
  45. package/dist/chunk-TOO5AEKL.js +81 -0
  46. package/dist/chunk-U3GHAOIJ.js +299 -0
  47. package/dist/chunk-U7N2A7A3.js +9 -0
  48. package/dist/chunk-UX634MYF.js +123 -0
  49. package/dist/chunk-V6UWJKZF.js +28 -0
  50. package/dist/chunk-VOHTRR5X.js +28 -0
  51. package/dist/chunk-WRFKVUPW.js +332 -0
  52. package/dist/chunk-WXW43RK5.js +24 -0
  53. package/dist/chunk-XG7XDYIM.js +60 -0
  54. package/dist/chunk-YFCQKO3B.js +842 -0
  55. package/dist/chunk-ZDWXGWLY.js +73 -0
  56. package/dist/chunk-ZLK5SPT6.js +11 -0
  57. package/dist/chunk-ZS6DTAM2.js +31 -0
  58. package/dist/chunk-ZT5UEUBO.js +1 -0
  59. package/dist/components/admin/index.d.ts +80 -0
  60. package/dist/components/admin/index.js +38 -0
  61. package/dist/components/data-display/badge.d.ts +12 -0
  62. package/dist/components/data-display/badge.js +3 -0
  63. package/dist/components/data-display/card.d.ts +87 -0
  64. package/dist/components/data-display/card.js +2 -0
  65. package/dist/components/data-display/index.d.ts +72 -0
  66. package/dist/components/data-display/index.js +89 -0
  67. package/dist/components/data-display/popover.d.ts +13 -0
  68. package/dist/components/data-display/popover.js +2 -0
  69. package/dist/components/data-display/scroll-area.d.ts +7 -0
  70. package/dist/components/data-display/scroll-area.js +2 -0
  71. package/dist/components/data-display/table.d.ts +10 -0
  72. package/dist/components/data-display/table.js +3 -0
  73. package/dist/components/data-entry/autocomplete.d.ts +16 -0
  74. package/dist/components/data-entry/autocomplete.js +8 -0
  75. package/dist/components/data-entry/calendar.d.ts +16 -0
  76. package/dist/components/data-entry/calendar.js +4 -0
  77. package/dist/components/data-entry/cascader.d.ts +32 -0
  78. package/dist/components/data-entry/cascader.js +13 -0
  79. package/dist/components/data-entry/checkbox.d.ts +13 -0
  80. package/dist/components/data-entry/checkbox.js +4 -0
  81. package/dist/components/data-entry/color-picker.d.ts +16 -0
  82. package/dist/components/data-entry/color-picker.js +6 -0
  83. package/dist/components/data-entry/command.d.ts +69 -0
  84. package/dist/components/data-entry/command.js +3 -0
  85. package/dist/components/data-entry/date-picker.d.ts +16 -0
  86. package/dist/components/data-entry/date-picker.js +8 -0
  87. package/dist/components/data-entry/date-range-picker.d.ts +16 -0
  88. package/dist/components/data-entry/date-range-picker.js +8 -0
  89. package/dist/components/data-entry/index.d.ts +53 -0
  90. package/dist/components/data-entry/index.js +152 -0
  91. package/dist/components/data-entry/input.d.ts +6 -0
  92. package/dist/components/data-entry/input.js +2 -0
  93. package/dist/components/data-entry/label.d.ts +8 -0
  94. package/dist/components/data-entry/label.js +2 -0
  95. package/dist/components/data-entry/radio.d.ts +24 -0
  96. package/dist/components/data-entry/radio.js +4 -0
  97. package/dist/components/data-entry/select.d.ts +18 -0
  98. package/dist/components/data-entry/select.js +3 -0
  99. package/dist/components/data-entry/slider.d.ts +16 -0
  100. package/dist/components/data-entry/slider.js +2 -0
  101. package/dist/components/data-entry/switch.d.ts +17 -0
  102. package/dist/components/data-entry/switch.js +2 -0
  103. package/dist/components/data-entry/textarea.d.ts +6 -0
  104. package/dist/components/data-entry/textarea.js +3 -0
  105. package/dist/components/data-entry/time-picker.d.ts +16 -0
  106. package/dist/components/data-entry/time-picker.js +7 -0
  107. package/dist/components/data-entry/transfer.d.ts +17 -0
  108. package/dist/components/data-entry/transfer.js +11 -0
  109. package/dist/components/data-entry/tree-select.d.ts +26 -0
  110. package/dist/components/data-entry/tree-select.js +13 -0
  111. package/dist/components/data-entry/upload.d.ts +42 -0
  112. package/dist/components/data-entry/upload.js +11 -0
  113. package/dist/components/feedback/alert.d.ts +60 -0
  114. package/dist/components/feedback/alert.js +7 -0
  115. package/dist/components/feedback/dialog.d.ts +64 -0
  116. package/dist/components/feedback/dialog.js +7 -0
  117. package/dist/components/feedback/index.d.ts +16 -0
  118. package/dist/components/feedback/index.js +14 -0
  119. package/dist/components/feedback/sheet.d.ts +24 -0
  120. package/dist/components/feedback/sheet.js +2 -0
  121. package/dist/components/feedback/sonner.d.ts +6 -0
  122. package/dist/components/feedback/sonner.js +1 -0
  123. package/dist/components/general/button.d.ts +20 -0
  124. package/dist/components/general/button.js +2 -0
  125. package/dist/components/general/index.d.ts +6 -0
  126. package/dist/components/general/index.js +2 -0
  127. package/dist/components/layout/index.d.ts +71 -0
  128. package/dist/components/layout/index.js +5 -0
  129. package/dist/components/navigation/dropdown-menu.d.ts +28 -0
  130. package/dist/components/navigation/dropdown-menu.js +2 -0
  131. package/dist/components/navigation/index.d.ts +32 -0
  132. package/dist/components/navigation/index.js +12 -0
  133. package/dist/components/navigation/pagination.d.ts +11 -0
  134. package/dist/components/navigation/pagination.js +7 -0
  135. package/dist/components/navigation/steps.d.ts +12 -0
  136. package/dist/components/navigation/steps.js +3 -0
  137. package/dist/components/navigation/tabs-items.d.ts +12 -0
  138. package/dist/components/navigation/tabs-items.js +3 -0
  139. package/dist/components/navigation/tabs.d.ts +12 -0
  140. package/dist/components/navigation/tabs.js +2 -0
  141. package/dist/components/query/index.d.ts +43 -0
  142. package/dist/components/query/index.js +8 -0
  143. package/dist/components/ui/index.d.ts +57 -0
  144. package/dist/components/ui/index.js +37 -0
  145. package/dist/content.prop-D1Dd3TAc.d.ts +31 -0
  146. package/dist/data-display.prop-DNTAzmDy.d.ts +58 -0
  147. package/dist/data-entry.prop-BEGA1lTq.d.ts +323 -0
  148. package/dist/data-table-D1u_rKLK.d.ts +83 -0
  149. package/dist/data.prop-BVvfKC_g.d.ts +41 -0
  150. package/dist/feedback.prop-BmxUlpAW.d.ts +64 -0
  151. package/dist/filter-bar-B7OGFO9S.d.ts +10 -0
  152. package/dist/form/index.d.ts +21 -0
  153. package/dist/form/index.js +4 -0
  154. package/dist/form.prop-BHgpuFFm.d.ts +41 -0
  155. package/dist/format-date-ByyZoqI5.d.ts +51 -0
  156. package/dist/general.prop-D7brMPNL.d.ts +16 -0
  157. package/dist/i18n/index.d.ts +217 -0
  158. package/dist/i18n/index.js +2 -0
  159. package/dist/index.d.ts +46 -26
  160. package/dist/index.js +39 -13062
  161. package/dist/inline-C5u6ptJV.d.ts +10 -0
  162. package/dist/interaction.prop-Cdn7wOtq.d.ts +25 -0
  163. package/dist/layout.prop-4TCNvyQZ.d.ts +20 -0
  164. package/dist/layout.prop-C795F0qg.d.ts +112 -0
  165. package/dist/lib/datetime/index.d.ts +31 -0
  166. package/dist/lib/datetime/index.js +1 -0
  167. package/dist/lib/utils.d.ts +5 -0
  168. package/dist/lib/utils.js +1 -0
  169. package/dist/navigation.prop-DpZqcXey.d.ts +78 -0
  170. package/dist/navigation.prop-DxBiClEH.d.ts +20 -0
  171. package/dist/props/components/index.d.ts +27 -0
  172. package/dist/props/components/index.js +1 -0
  173. package/dist/props/index.d.ts +28 -0
  174. package/dist/props/index.js +3 -0
  175. package/dist/props/registry.d.ts +649 -0
  176. package/dist/props/registry.js +1 -0
  177. package/dist/props/vocabulary/index.d.ts +7 -0
  178. package/dist/props/vocabulary/index.js +1 -0
  179. package/dist/query.prop-hIPrk2zI.d.ts +71 -0
  180. package/dist/search-input-uP01rY1L.d.ts +22 -0
  181. package/dist/shared.prop-BNRJc9K0.d.ts +45 -0
  182. package/dist/styles/alert-layout.css +191 -0
  183. package/dist/styles/badge-layout.css +22 -0
  184. package/dist/styles/card-layout.css +373 -0
  185. package/dist/styles/control.css +504 -0
  186. package/dist/styles/data-display-layout.css +246 -0
  187. package/dist/styles/density.css +43 -0
  188. package/dist/styles/dialog-layout.css +84 -0
  189. package/dist/styles/index.css +105 -0
  190. package/dist/styles/layout.css +479 -0
  191. package/dist/styles/shell-layout.css +604 -0
  192. package/dist/styles/table-layout.css +109 -0
  193. package/dist/theme/example.service.css +37 -0
  194. package/dist/tokens/base.css +13 -0
  195. package/dist/tokens/foundation.css +151 -0
  196. package/dist/tokens/primitives/badge.css +13 -0
  197. package/dist/tokens/primitives/card.css +29 -0
  198. package/dist/tokens/primitives/control.css +55 -0
  199. package/dist/tokens/primitives/feedback.css +17 -0
  200. package/dist/tokens/primitives/layout.css +20 -0
  201. package/dist/tokens/primitives/navigation.css +13 -0
  202. package/dist/tokens/primitives/table.css +10 -0
  203. package/dist/types-mvzYGrma.d.ts +37 -0
  204. package/dist/use-toast-Dol5bdY3.d.ts +34 -0
  205. package/package.json +279 -117
  206. package/scripts/ui-audit.mjs +179 -0
  207. package/BRAND.md +0 -296
  208. package/CHANGELOG.md +0 -668
  209. package/config/eslint.js +0 -54
  210. package/config/prettier.cjs +0 -20
  211. package/config/tsconfig.base.json +0 -22
  212. package/config/vitest.base.ts +0 -26
  213. package/dist/MiniMonth-YAmPGEpC.d.ts +0 -143
  214. package/dist/Table.types-BbsxoIYE.d.ts +0 -352
  215. package/dist/color-DO0qqUAb.d.ts +0 -38
  216. package/dist/components/composites.d.ts +0 -963
  217. package/dist/components/composites.js +0 -7343
  218. package/dist/components/composites.js.map +0 -1
  219. package/dist/components/primitives.d.ts +0 -2744
  220. package/dist/components/primitives.js +0 -7356
  221. package/dist/components/primitives.js.map +0 -1
  222. package/dist/components/shell.d.ts +0 -182
  223. package/dist/components/shell.js +0 -774
  224. package/dist/components/shell.js.map +0 -1
  225. package/dist/hooks.d.ts +0 -100
  226. package/dist/hooks.js +0 -558
  227. package/dist/hooks.js.map +0 -1
  228. package/dist/i18n.d.ts +0 -61
  229. package/dist/i18n.js +0 -860
  230. package/dist/i18n.js.map +0 -1
  231. package/dist/index.js.map +0 -1
  232. package/dist/padding-DY0JV5Ja.d.ts +0 -16
  233. package/dist/preferences.d.ts +0 -132
  234. package/dist/preferences.js +0 -262
  235. package/dist/preferences.js.map +0 -1
  236. package/dist/props.d.ts +0 -86
  237. package/dist/props.js +0 -16
  238. package/dist/props.js.map +0 -1
  239. package/dist/size-CQwNvOWd.d.ts +0 -19
  240. package/dist/types-LTj-2bl-.d.ts +0 -30
  241. package/dist/useTableViews-D5NIAJ7h.d.ts +0 -154
  242. package/src/tokens/tailwind.css +0 -158
@@ -0,0 +1,191 @@
1
+ import { AlertQueryError } from './chunk-E4HJNQ62.js';
2
+ import { Button } from './chunk-HJEBRCXL.js';
3
+ import { useTranslation } from './chunk-3F2AKYRD.js';
4
+ import * as React from 'react';
5
+ import { jsx, Fragment, jsxs } from 'react/jsx-runtime';
6
+ import { RefreshCw } from 'lucide-react';
7
+ import { Link } from 'react-router-dom';
8
+ import { useQueryClient } from '@tanstack/react-query';
9
+
10
+ function defaultIsEmpty(data) {
11
+ if (!data) return true;
12
+ if (Array.isArray(data)) return data.length === 0;
13
+ if (typeof data === "object" && data !== null) {
14
+ const obj = data;
15
+ if (Array.isArray(obj.items)) return obj.items.length === 0;
16
+ if (typeof obj.length === "number") return obj.length === 0;
17
+ }
18
+ return false;
19
+ }
20
+ function DataState({
21
+ query,
22
+ skeleton,
23
+ empty,
24
+ isEmpty = defaultIsEmpty,
25
+ errorRenderer,
26
+ showRetry = true,
27
+ onRetry,
28
+ children
29
+ }) {
30
+ const retry = React.useCallback(() => {
31
+ if (onRetry) {
32
+ void onRetry();
33
+ return;
34
+ }
35
+ void query.refetch();
36
+ }, [onRetry, query]);
37
+ if (query.isPending) return /* @__PURE__ */ jsx(Fragment, { children: skeleton });
38
+ if (query.isError) {
39
+ if (query.isFetching) return /* @__PURE__ */ jsx(Fragment, { children: skeleton });
40
+ if (errorRenderer) return /* @__PURE__ */ jsx(Fragment, { children: errorRenderer(query.error, retry) });
41
+ return /* @__PURE__ */ jsx(AlertQueryError, { error: query.error, onRetry: showRetry ? retry : void 0 });
42
+ }
43
+ const data = query.data;
44
+ if (data === void 0) return /* @__PURE__ */ jsx(Fragment, { children: skeleton });
45
+ if (empty && (data === null || isEmpty(data))) return /* @__PURE__ */ jsx(Fragment, { children: empty });
46
+ return /* @__PURE__ */ jsx(Fragment, { children: children(data) });
47
+ }
48
+ function MutationFeedback({
49
+ mutation,
50
+ onRetry,
51
+ showRetry = true,
52
+ pending,
53
+ className
54
+ }) {
55
+ if (mutation.isPending && pending) return /* @__PURE__ */ jsx(Fragment, { children: pending });
56
+ if (!mutation.isError || mutation.error == null) return null;
57
+ return /* @__PURE__ */ jsx(
58
+ AlertQueryError,
59
+ {
60
+ className,
61
+ error: mutation.error,
62
+ onRetry: showRetry ? onRetry : void 0
63
+ }
64
+ );
65
+ }
66
+ function QueryRefetchButton({
67
+ query,
68
+ label = "Refresh",
69
+ children,
70
+ variant = "outline",
71
+ size = "sm",
72
+ className,
73
+ ...props
74
+ }) {
75
+ const text = children ?? label;
76
+ return /* @__PURE__ */ jsxs(
77
+ Button,
78
+ {
79
+ type: "button",
80
+ variant,
81
+ size,
82
+ className,
83
+ onClick: () => void query.refetch(),
84
+ disabled: query.isFetching,
85
+ ...props,
86
+ children: [
87
+ /* @__PURE__ */ jsx(
88
+ RefreshCw,
89
+ {
90
+ className: "ui-query-refetch-icon",
91
+ "data-fetching": query.isFetching,
92
+ "aria-hidden": "true"
93
+ }
94
+ ),
95
+ text
96
+ ]
97
+ }
98
+ );
99
+ }
100
+ function flattenItemPages(data) {
101
+ if (!data) return [];
102
+ return data.pages.flatMap((page) => page.items);
103
+ }
104
+ function defaultIsEmptyFlat(flat) {
105
+ if (Array.isArray(flat)) return flat.length === 0;
106
+ return !flat;
107
+ }
108
+ function InfiniteQueryState({
109
+ query,
110
+ skeleton,
111
+ empty,
112
+ flatten,
113
+ isEmpty = defaultIsEmptyFlat,
114
+ errorRenderer,
115
+ showRetry = true,
116
+ onRetry,
117
+ loadingMore,
118
+ loadMore,
119
+ showLoadMore = true,
120
+ children
121
+ }) {
122
+ const { t } = useTranslation();
123
+ const retry = React.useCallback(() => {
124
+ if (onRetry) {
125
+ void onRetry();
126
+ return;
127
+ }
128
+ void query.refetch();
129
+ }, [onRetry, query]);
130
+ if (query.isPending) return /* @__PURE__ */ jsx(Fragment, { children: skeleton });
131
+ if (query.isError) {
132
+ if (query.isFetching && !query.isFetchingNextPage) return /* @__PURE__ */ jsx(Fragment, { children: skeleton });
133
+ if (errorRenderer) return /* @__PURE__ */ jsx(Fragment, { children: errorRenderer(query.error, retry) });
134
+ return /* @__PURE__ */ jsx(AlertQueryError, { error: query.error, onRetry: showRetry ? retry : void 0 });
135
+ }
136
+ const data = query.data;
137
+ if (!data) return /* @__PURE__ */ jsx(Fragment, { children: skeleton });
138
+ const flat = flatten(data);
139
+ if (empty && isEmpty(flat)) return /* @__PURE__ */ jsx(Fragment, { children: empty });
140
+ const footer = showLoadMore && query.hasNextPage ? loadMore ?? /* @__PURE__ */ jsx("div", { className: "flex justify-center pt-4", children: /* @__PURE__ */ jsx(
141
+ Button,
142
+ {
143
+ type: "button",
144
+ variant: "outline",
145
+ size: "sm",
146
+ disabled: query.isFetchingNextPage,
147
+ onClick: () => void query.fetchNextPage(),
148
+ children: query.isFetchingNextPage ? t("common.working") : t("query.loadMore")
149
+ }
150
+ ) }) : null;
151
+ return /* @__PURE__ */ jsxs(Fragment, { children: [
152
+ children(flat, {
153
+ fetchNextPage: () => void query.fetchNextPage(),
154
+ hasNextPage: !!query.hasNextPage,
155
+ isFetchingNextPage: query.isFetchingNextPage
156
+ }),
157
+ query.isFetchingNextPage && (loadingMore ?? /* @__PURE__ */ jsx("p", { className: "text-muted-foreground pt-2 text-center text-xs", children: t("common.working") })),
158
+ footer
159
+ ] });
160
+ }
161
+ function PrefetchLink({
162
+ queryKey,
163
+ queryFn,
164
+ prefetchOn = "both",
165
+ staleTime = 3e4,
166
+ onMouseEnter,
167
+ onFocus,
168
+ ...linkProps
169
+ }) {
170
+ const queryClient = useQueryClient();
171
+ const prefetch = React.useCallback(() => {
172
+ if (prefetchOn === "none") return;
173
+ void queryClient.prefetchQuery({ queryKey, queryFn, staleTime });
174
+ }, [prefetchOn, queryClient, queryFn, queryKey, staleTime]);
175
+ return /* @__PURE__ */ jsx(
176
+ Link,
177
+ {
178
+ ...linkProps,
179
+ onMouseEnter: (event) => {
180
+ if (prefetchOn === "hover" || prefetchOn === "both") prefetch();
181
+ onMouseEnter?.(event);
182
+ },
183
+ onFocus: (event) => {
184
+ if (prefetchOn === "focus" || prefetchOn === "both") prefetch();
185
+ onFocus?.(event);
186
+ }
187
+ }
188
+ );
189
+ }
190
+
191
+ export { DataState, InfiniteQueryState, MutationFeedback, PrefetchLink, QueryRefetchButton, flattenItemPages };
@@ -0,0 +1,92 @@
1
+ import { tableCellPaddingClass, tableRowHeightClass } from './chunk-ICM6XBST.js';
2
+ import { cn } from './chunk-U7N2A7A3.js';
3
+ import { jsx, jsxs } from 'react/jsx-runtime';
4
+ import { toast as toast$1 } from 'sonner';
5
+
6
+ function SkeletonBlock({ className, ...props }) {
7
+ return /* @__PURE__ */ jsx(
8
+ "div",
9
+ {
10
+ "aria-busy": "true",
11
+ "aria-live": "polite",
12
+ className: cn("ui-skeleton-block", className),
13
+ ...props
14
+ }
15
+ );
16
+ }
17
+ function SkeletonRows({ rows = 6, columns = 4, className }) {
18
+ return /* @__PURE__ */ jsx("div", { className: cn("ui-skeleton-rows", className), "aria-busy": "true", children: Array.from({ length: rows }).map((_, i) => /* @__PURE__ */ jsx("div", { className: "ui-skeleton-row", children: Array.from({ length: columns }).map((_2, j) => /* @__PURE__ */ jsx(
19
+ SkeletonBlock,
20
+ {
21
+ className: cn("h-4", j === 0 ? "w-1/4" : j === columns - 1 ? "w-1/6" : "flex-1")
22
+ },
23
+ j
24
+ )) }, i)) });
25
+ }
26
+ function SkeletonTable({ rows = 8, columns = 5 }) {
27
+ return /* @__PURE__ */ jsxs("div", { className: "ui-skeleton-table", "aria-busy": "true", children: [
28
+ /* @__PURE__ */ jsx("div", { className: cn("ui-skeleton-table-head", tableCellPaddingClass, tableRowHeightClass), children: Array.from({ length: columns }).map((_, j) => /* @__PURE__ */ jsx(SkeletonBlock, { className: cn("h-3", j === 0 ? "w-1/5" : "flex-1") }, j)) }),
29
+ /* @__PURE__ */ jsx("div", { className: "ui-skeleton-table-body", children: Array.from({ length: rows }).map((_, i) => /* @__PURE__ */ jsx(
30
+ "div",
31
+ {
32
+ className: cn("ui-skeleton-table-row", tableCellPaddingClass, tableRowHeightClass),
33
+ children: Array.from({ length: columns }).map((_2, j) => /* @__PURE__ */ jsx(SkeletonBlock, { className: cn("h-4", j === 0 ? "w-1/5" : "flex-1") }, j))
34
+ },
35
+ i
36
+ )) })
37
+ ] });
38
+ }
39
+ function SkeletonDetail() {
40
+ return /* @__PURE__ */ jsxs("div", { className: "ui-skeleton-detail ui-skeleton-detail-stack", "aria-busy": "true", children: [
41
+ /* @__PURE__ */ jsx(SkeletonBlock, { className: "h-7 w-1/3" }),
42
+ /* @__PURE__ */ jsx(SkeletonBlock, { className: "h-4 w-1/2" }),
43
+ /* @__PURE__ */ jsx("div", { className: "ui-skeleton-detail-box ui-skeleton-detail-stack", children: Array.from({ length: 6 }).map((_, i) => /* @__PURE__ */ jsxs("div", { className: "ui-skeleton-detail-stack", children: [
44
+ /* @__PURE__ */ jsx(SkeletonBlock, { className: "h-3 w-24" }),
45
+ /* @__PURE__ */ jsx(SkeletonBlock, { className: "h-4 w-full max-w-md" })
46
+ ] }, i)) })
47
+ ] });
48
+ }
49
+ function SkeletonCard() {
50
+ return /* @__PURE__ */ jsxs("div", { className: "ui-skeleton-card", "aria-busy": "true", children: [
51
+ /* @__PURE__ */ jsx(SkeletonBlock, { className: "h-3 w-24" }),
52
+ /* @__PURE__ */ jsx(SkeletonBlock, { className: "h-[length:var(--control-height)] w-32" }),
53
+ /* @__PURE__ */ jsx(SkeletonBlock, { className: "h-3 w-20" })
54
+ ] });
55
+ }
56
+ function nodeText(value) {
57
+ if (value == null) return "";
58
+ if (typeof value === "string") return value;
59
+ if (typeof value === "number" || typeof value === "boolean") return String(value);
60
+ return "";
61
+ }
62
+ function legacyToast(options) {
63
+ const { title, description, variant, ...rest } = options;
64
+ const titleText = nodeText(title);
65
+ const descText = nodeText(description);
66
+ const message = titleText || descText;
67
+ const desc = titleText && descText ? descText : void 0;
68
+ const sonnerOptions = { ...rest, description: desc };
69
+ switch (variant) {
70
+ case "destructive":
71
+ return toast$1.error(message, sonnerOptions);
72
+ case "success":
73
+ return toast$1.success(message, sonnerOptions);
74
+ default:
75
+ return toast$1(message, sonnerOptions);
76
+ }
77
+ }
78
+ var toast = Object.assign((messageOrOptions) => {
79
+ if (typeof messageOrOptions === "string") {
80
+ return toast$1(messageOrOptions);
81
+ }
82
+ return legacyToast(messageOrOptions);
83
+ }, toast$1);
84
+ function useToast() {
85
+ return {
86
+ toast: (options) => legacyToast(options),
87
+ dismiss: toast$1.dismiss,
88
+ toasts: []
89
+ };
90
+ }
91
+
92
+ export { SkeletonCard, SkeletonDetail, SkeletonRows, SkeletonTable, toast, useToast };
@@ -0,0 +1,402 @@
1
+ import { Table, TableHeader, TableRow, TableHead, TableBody, TableCell } from './chunk-ZS6DTAM2.js';
2
+ import { tableCellPaddingClass, tableRowHeightClass, controlIconSmClass, toneNeutralClass, toneInfoClass, toneDestructiveClass, toneWarningClass, toneSuccessClass } from './chunk-ICM6XBST.js';
3
+ import { Inline, densityClass } from './chunk-S66TJXJU.js';
4
+ import { Button } from './chunk-HJEBRCXL.js';
5
+ import { useTranslation } from './chunk-3F2AKYRD.js';
6
+ import { cn } from './chunk-U7N2A7A3.js';
7
+ import { jsx, jsxs } from 'react/jsx-runtime';
8
+ import { Layers, Layers2, ArrowUp, ArrowDown, ChevronsUpDown, MoreHorizontal, XCircle, Pause, CheckCircle2, Circle, Trash2, AlertCircle, Clock, Play } from 'lucide-react';
9
+ import * as React from 'react';
10
+
11
+ function EmptyState({ icon: Icon, title, description, action, className }) {
12
+ return /* @__PURE__ */ jsxs("div", { "data-slot": "empty-state", className: cn("ui-empty-state", className), children: [
13
+ Icon && /* @__PURE__ */ jsx("div", { className: "ui-empty-state-icon", children: /* @__PURE__ */ jsx(Icon, { className: "text-muted-foreground size-6", "aria-hidden": "true" }) }),
14
+ /* @__PURE__ */ jsx("h3", { className: "ui-empty-state-title", children: title }),
15
+ description && /* @__PURE__ */ jsx("p", { className: "ui-empty-state-description", children: description }),
16
+ action && /* @__PURE__ */ jsx("div", { children: action })
17
+ ] });
18
+ }
19
+ var STATUS_MAP = {
20
+ active: { tone: "success", icon: CheckCircle2 },
21
+ completed: { tone: "success", icon: CheckCircle2 },
22
+ delivered: { tone: "success", icon: CheckCircle2 },
23
+ done: { tone: "success", icon: CheckCircle2 },
24
+ permanent: { tone: "success", icon: CheckCircle2 },
25
+ succeeded: { tone: "success", icon: CheckCircle2 },
26
+ draft: { tone: "neutral", icon: Circle },
27
+ pending: { tone: "warning", icon: Clock },
28
+ scheduled: { tone: "info", icon: Clock },
29
+ sending: { tone: "info", icon: Play },
30
+ temporary: { tone: "warning", icon: Clock },
31
+ bounced: { tone: "destructive", icon: AlertCircle },
32
+ cancelled: { tone: "neutral", icon: Pause },
33
+ deleted: { tone: "destructive", icon: Trash2 },
34
+ failed: { tone: "destructive", icon: XCircle },
35
+ private: { tone: "neutral", icon: Circle },
36
+ internal: { tone: "info", icon: Circle },
37
+ public: { tone: "info", icon: Circle },
38
+ ASSIGNMENT_STATUS_ACTIVE: { tone: "success", icon: CheckCircle2 },
39
+ ASSIGNMENT_STATUS_SUSPENDED: { tone: "warning", icon: Pause },
40
+ ASSIGNMENT_STATUS_TERMINATED: { tone: "destructive", icon: XCircle }
41
+ };
42
+ var TONE_CLASSES = {
43
+ success: toneSuccessClass,
44
+ warning: toneWarningClass,
45
+ destructive: toneDestructiveClass,
46
+ info: toneInfoClass,
47
+ neutral: toneNeutralClass
48
+ };
49
+ function StatusBadge({ status, className, label: labelOverride }) {
50
+ const { t } = useTranslation();
51
+ const def = STATUS_MAP[status] ?? { tone: "neutral", icon: Circle };
52
+ const Icon = def.icon;
53
+ const resolvedLabel = labelOverride ?? (status in STATUS_MAP ? t(`status.${status}`) : status);
54
+ return /* @__PURE__ */ jsxs("span", { "data-slot": "status-badge", className: cn(TONE_CLASSES[def.tone], className), children: [
55
+ /* @__PURE__ */ jsx(Icon, { "data-slot": "status-badge-icon", "aria-hidden": "true" }),
56
+ /* @__PURE__ */ jsx("span", { children: resolvedLabel })
57
+ ] });
58
+ }
59
+ function KeyValueGrid({ columns = 2, className, children }) {
60
+ const colsClass = columns === 1 ? "grid-cols-1" : columns === 3 ? "grid-cols-1 sm:grid-cols-2 lg:grid-cols-3" : "grid-cols-1 sm:grid-cols-2";
61
+ return /* @__PURE__ */ jsx("dl", { className: cn("grid gap-x-6 gap-y-3", colsClass, className), children });
62
+ }
63
+ KeyValueGrid.Item = function KeyValueGridItem({
64
+ label,
65
+ mono,
66
+ span,
67
+ className,
68
+ children
69
+ }) {
70
+ const spanClass = span === 2 ? "sm:col-span-2" : span === 3 ? "sm:col-span-2 lg:col-span-3" : "";
71
+ return /* @__PURE__ */ jsxs("div", { className: cn(spanClass, className), children: [
72
+ /* @__PURE__ */ jsx("dt", { className: "text-muted-foreground text-xs", children: label }),
73
+ /* @__PURE__ */ jsx("dd", { className: cn("text-sm break-all", mono && "font-mono"), children })
74
+ ] });
75
+ };
76
+ var DataTableContext = React.createContext(null);
77
+ function useDataTableContext() {
78
+ const ctx = React.useContext(DataTableContext);
79
+ if (!ctx) throw new Error("DataTable subcomponents must be used inside <DataTable>");
80
+ return ctx;
81
+ }
82
+ function useOptionalDataTableContext() {
83
+ return React.useContext(DataTableContext);
84
+ }
85
+ var noopGetRowId = (row) => {
86
+ const id = row.id;
87
+ if (typeof id === "string") return id;
88
+ if (typeof id === "number") return String(id);
89
+ return "";
90
+ };
91
+ function DataTable({
92
+ data,
93
+ columns,
94
+ getRowId = noopGetRowId,
95
+ selectable = false,
96
+ selected: controlledSelected,
97
+ onSelectChange,
98
+ onRowClick,
99
+ density: controlledDensity,
100
+ onDensityChange,
101
+ sort,
102
+ onSortChange,
103
+ className,
104
+ children
105
+ }) {
106
+ const [internalDensity, setInternalDensity] = React.useState("compact");
107
+ const density = controlledDensity ?? internalDensity;
108
+ const setDensity = (d) => {
109
+ setInternalDensity(d);
110
+ onDensityChange?.(d);
111
+ };
112
+ const [internalSelected, setInternalSelected] = React.useState(/* @__PURE__ */ new Set());
113
+ const selected = controlledSelected ?? internalSelected;
114
+ const setSelected = (next) => {
115
+ setInternalSelected(next);
116
+ onSelectChange?.(next);
117
+ };
118
+ const toggleSelect = (id) => {
119
+ const next = new Set(selected);
120
+ if (next.has(id)) next.delete(id);
121
+ else next.add(id);
122
+ setSelected(next);
123
+ };
124
+ const allSelected = data.length > 0 && data.every((r) => selected.has(getRowId(r)));
125
+ const someSelected = !allSelected && data.some((r) => selected.has(getRowId(r)));
126
+ const toggleSelectAll = () => {
127
+ if (allSelected) setSelected(/* @__PURE__ */ new Set());
128
+ else setSelected(new Set(data.map(getRowId)));
129
+ };
130
+ const ctx = {
131
+ data,
132
+ columns,
133
+ density,
134
+ setDensity,
135
+ selected,
136
+ toggleSelect,
137
+ toggleSelectAll,
138
+ allSelected,
139
+ someSelected,
140
+ selectable,
141
+ getRowId,
142
+ onRowClick,
143
+ sort,
144
+ onSortChange
145
+ };
146
+ const hasContent = React.Children.toArray(children).some(
147
+ (c) => React.isValidElement(c) && c.type.displayName === "DataTable.Content"
148
+ );
149
+ return /* @__PURE__ */ jsx(DataTableContext.Provider, { value: ctx, children: /* @__PURE__ */ jsxs(
150
+ "div",
151
+ {
152
+ className: cn(
153
+ "ui-data-table-root",
154
+ densityClass[density === "compact" ? "compact" : "comfortable"],
155
+ className
156
+ ),
157
+ children: [
158
+ children,
159
+ !hasContent && /* @__PURE__ */ jsx(DataTable.Content, {})
160
+ ]
161
+ }
162
+ ) });
163
+ }
164
+ DataTable.Toolbar = function DataTableToolbar({
165
+ children,
166
+ className
167
+ }) {
168
+ return /* @__PURE__ */ jsx("div", { className: cn("ui-data-table-toolbar", className), children });
169
+ };
170
+ DataTable.Toolbar.displayName = "DataTable.Toolbar";
171
+ DataTable.SelectAll = function DataTableSelectAll() {
172
+ const { allSelected, someSelected, toggleSelectAll, selectable } = useDataTableContext();
173
+ const { t } = useTranslation();
174
+ if (!selectable) return null;
175
+ return /* @__PURE__ */ jsx(
176
+ "input",
177
+ {
178
+ type: "checkbox",
179
+ checked: allSelected,
180
+ ref: (el) => {
181
+ if (el) el.indeterminate = someSelected;
182
+ },
183
+ onChange: toggleSelectAll,
184
+ "aria-label": t("dataTable.selectAll")
185
+ }
186
+ );
187
+ };
188
+ DataTable.SelectAll.displayName = "DataTable.SelectAll";
189
+ DataTable.BulkActions = function DataTableBulkActions({
190
+ count,
191
+ children,
192
+ className
193
+ }) {
194
+ const ctx = useOptionalDataTableContext();
195
+ const { t } = useTranslation();
196
+ const c = count ?? ctx?.selected.size ?? 0;
197
+ if (c === 0) return null;
198
+ return /* @__PURE__ */ jsxs(
199
+ "div",
200
+ {
201
+ role: "region",
202
+ "aria-label": t("dataTable.bulkActions"),
203
+ className: cn("ui-data-table-bulk", className),
204
+ children: [
205
+ /* @__PURE__ */ jsx("span", { className: "text-muted-foreground", children: /* @__PURE__ */ jsx("strong", { className: "text-foreground", children: t("common.selectedCount", { count: c }) }) }),
206
+ /* @__PURE__ */ jsx("div", { className: "ui-data-table-bulk-actions", children })
207
+ ]
208
+ }
209
+ );
210
+ };
211
+ DataTable.BulkActions.displayName = "DataTable.BulkActions";
212
+ DataTable.DensityToggle = function DataTableDensityToggle() {
213
+ const { density, setDensity } = useDataTableContext();
214
+ const { t } = useTranslation();
215
+ const next = density === "compact" ? "comfortable" : "compact";
216
+ const Icon = density === "compact" ? Layers : Layers2;
217
+ const nextLabel = next === "compact" ? t("dataTable.densityCompact") : t("dataTable.densityComfortable");
218
+ return /* @__PURE__ */ jsx(
219
+ Button,
220
+ {
221
+ variant: "ghost",
222
+ size: "sm",
223
+ onClick: () => {
224
+ setDensity(next);
225
+ },
226
+ "aria-label": t("dataTable.densitySwitch", { density: nextLabel }),
227
+ children: /* @__PURE__ */ jsxs(Inline, { gap: "xs", children: [
228
+ /* @__PURE__ */ jsx(Icon, { className: "size-4", "aria-hidden": "true" }),
229
+ density === "compact" ? t("dataTable.densityCompact") : t("dataTable.densityComfortable")
230
+ ] })
231
+ }
232
+ );
233
+ };
234
+ DataTable.DensityToggle.displayName = "DataTable.DensityToggle";
235
+ DataTable.Content = function DataTableContent() {
236
+ const {
237
+ data,
238
+ columns,
239
+ density: _density,
240
+ selectable,
241
+ selected,
242
+ toggleSelect,
243
+ getRowId,
244
+ onRowClick,
245
+ sort,
246
+ onSortChange
247
+ } = useDataTableContext();
248
+ const { t } = useTranslation();
249
+ const rowPadding = tableRowHeightClass;
250
+ const cellPadding = tableCellPaddingClass;
251
+ const onHeaderClick = (col) => {
252
+ if (!col.sortable || !onSortChange) return;
253
+ if (sort?.key !== col.key) {
254
+ onSortChange({ key: col.key, direction: "asc" });
255
+ } else if (sort.direction === "asc") {
256
+ onSortChange({ key: col.key, direction: "desc" });
257
+ } else {
258
+ onSortChange(void 0);
259
+ }
260
+ };
261
+ return /* @__PURE__ */ jsx("div", { className: "ui-data-table-scroll", children: /* @__PURE__ */ jsx("div", { className: "ui-data-table-surface min-w-[640px] sm:min-w-0", children: /* @__PURE__ */ jsxs(Table, { children: [
262
+ /* @__PURE__ */ jsx(TableHeader, { className: "bg-secondary sticky top-0 z-10", children: /* @__PURE__ */ jsxs(TableRow, { children: [
263
+ selectable && /* @__PURE__ */ jsx(TableHead, { className: "w-10", children: /* @__PURE__ */ jsx(DataTable.SelectAll, {}) }),
264
+ columns.map((col) => /* @__PURE__ */ jsx(
265
+ TableHead,
266
+ {
267
+ className: cn(
268
+ col.width,
269
+ col.align === "right" && "text-right",
270
+ col.align === "center" && "text-center",
271
+ col.sortable && onSortChange && "cursor-pointer select-none"
272
+ ),
273
+ onClick: () => {
274
+ onHeaderClick(col);
275
+ },
276
+ children: /* @__PURE__ */ jsxs("span", { className: "ui-data-table-sort-label", children: [
277
+ col.header,
278
+ col.sortable && onSortChange && (sort?.key === col.key ? sort.direction === "asc" ? /* @__PURE__ */ jsx(ArrowUp, { className: "size-3", "aria-hidden": "true" }) : /* @__PURE__ */ jsx(ArrowDown, { className: "size-3", "aria-hidden": "true" }) : /* @__PURE__ */ jsx(
279
+ ChevronsUpDown,
280
+ {
281
+ className: "text-muted-foreground size-3",
282
+ "aria-hidden": "true"
283
+ }
284
+ ))
285
+ ] })
286
+ },
287
+ col.key
288
+ ))
289
+ ] }) }),
290
+ /* @__PURE__ */ jsx(TableBody, { children: data.map((row) => {
291
+ const id = getRowId(row);
292
+ const isSelected = selected.has(id);
293
+ return /* @__PURE__ */ jsxs(
294
+ TableRow,
295
+ {
296
+ "data-state": isSelected ? "selected" : void 0,
297
+ onClick: (e) => {
298
+ const target = e.target;
299
+ if (target.closest("button, a, input, select, textarea, [role=menuitem]"))
300
+ return;
301
+ onRowClick?.(row);
302
+ },
303
+ className: cn(
304
+ rowPadding,
305
+ onRowClick && "hover:bg-muted/50 cursor-pointer",
306
+ isSelected && "bg-muted/30"
307
+ ),
308
+ children: [
309
+ selectable && /* @__PURE__ */ jsx(TableCell, { className: cellPadding, children: /* @__PURE__ */ jsx(
310
+ "input",
311
+ {
312
+ type: "checkbox",
313
+ checked: isSelected,
314
+ onChange: () => {
315
+ toggleSelect(id);
316
+ },
317
+ "aria-label": t("dataTable.selectRow", { id }),
318
+ onClick: (e) => {
319
+ e.stopPropagation();
320
+ }
321
+ }
322
+ ) }),
323
+ columns.map((col) => /* @__PURE__ */ jsx(
324
+ TableCell,
325
+ {
326
+ className: cn(
327
+ cellPadding,
328
+ col.width,
329
+ col.align === "right" && "text-right",
330
+ col.align === "center" && "text-center"
331
+ ),
332
+ children: col.render ? col.render(row) : (() => {
333
+ const v = row[col.key];
334
+ if (v == null) return "\u2014";
335
+ if (typeof v === "string" || typeof v === "number") return String(v);
336
+ return "\u2014";
337
+ })()
338
+ },
339
+ col.key
340
+ ))
341
+ ]
342
+ },
343
+ id
344
+ );
345
+ }) })
346
+ ] }) }) });
347
+ };
348
+ DataTable.Content.displayName = "DataTable.Content";
349
+ DataTable.Pagination = function DataTablePagination({
350
+ cursor,
351
+ hasMore,
352
+ onChange,
353
+ className
354
+ }) {
355
+ const { t } = useTranslation();
356
+ return /* @__PURE__ */ jsxs("div", { className: cn("ui-data-table-pagination", className), children: [
357
+ /* @__PURE__ */ jsx(
358
+ Button,
359
+ {
360
+ variant: "outline",
361
+ size: "sm",
362
+ disabled: !cursor,
363
+ onClick: () => {
364
+ onChange(void 0);
365
+ },
366
+ children: t("common.first")
367
+ }
368
+ ),
369
+ /* @__PURE__ */ jsx(
370
+ Button,
371
+ {
372
+ variant: "outline",
373
+ size: "sm",
374
+ disabled: !hasMore,
375
+ onClick: () => {
376
+ onChange(cursor);
377
+ },
378
+ children: t("common.next")
379
+ }
380
+ )
381
+ ] });
382
+ };
383
+ DataTable.Pagination.displayName = "DataTable.Pagination";
384
+ DataTable.RowActions = function DataTableRowActions({ ariaLabel, children }) {
385
+ const { t } = useTranslation();
386
+ return /* @__PURE__ */ jsxs(
387
+ Button,
388
+ {
389
+ variant: "ghost",
390
+ size: "icon",
391
+ "aria-label": ariaLabel ?? t("dataTable.rowActions"),
392
+ className: controlIconSmClass,
393
+ children: [
394
+ /* @__PURE__ */ jsx(MoreHorizontal, { className: "size-4", "aria-hidden": "true" }),
395
+ children
396
+ ]
397
+ }
398
+ );
399
+ };
400
+ DataTable.RowActions.displayName = "DataTable.RowActions";
401
+
402
+ export { DataTable, EmptyState, KeyValueGrid, StatusBadge };