premium-ds 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 (257) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +113 -0
  3. package/dist/alert.d.ts +31 -0
  4. package/dist/alert.js +6 -0
  5. package/dist/alert.js.map +1 -0
  6. package/dist/avatar-group.d.ts +13 -0
  7. package/dist/avatar-group.js +3 -0
  8. package/dist/avatar-group.js.map +1 -0
  9. package/dist/avatar.d.ts +25 -0
  10. package/dist/avatar.js +3 -0
  11. package/dist/avatar.js.map +1 -0
  12. package/dist/badge.d.ts +23 -0
  13. package/dist/badge.js +3 -0
  14. package/dist/badge.js.map +1 -0
  15. package/dist/button.d.ts +20 -0
  16. package/dist/button.js +3 -0
  17. package/dist/button.js.map +1 -0
  18. package/dist/checkbox.d.ts +25 -0
  19. package/dist/checkbox.js +3 -0
  20. package/dist/checkbox.js.map +1 -0
  21. package/dist/chunk-2OWHZ4JT.js +36 -0
  22. package/dist/chunk-2OWHZ4JT.js.map +1 -0
  23. package/dist/chunk-34SIXSYL.js +64 -0
  24. package/dist/chunk-34SIXSYL.js.map +1 -0
  25. package/dist/chunk-37O2ZXD6.js +55 -0
  26. package/dist/chunk-37O2ZXD6.js.map +1 -0
  27. package/dist/chunk-4AZL76UJ.js +89 -0
  28. package/dist/chunk-4AZL76UJ.js.map +1 -0
  29. package/dist/chunk-4HSCN5TZ.js +86 -0
  30. package/dist/chunk-4HSCN5TZ.js.map +1 -0
  31. package/dist/chunk-5DDOOT33.js +258 -0
  32. package/dist/chunk-5DDOOT33.js.map +1 -0
  33. package/dist/chunk-5FVHWIMY.js +117 -0
  34. package/dist/chunk-5FVHWIMY.js.map +1 -0
  35. package/dist/chunk-5K6KRJGX.js +147 -0
  36. package/dist/chunk-5K6KRJGX.js.map +1 -0
  37. package/dist/chunk-5PQMQBQC.js +74 -0
  38. package/dist/chunk-5PQMQBQC.js.map +1 -0
  39. package/dist/chunk-7OCTVQ7C.js +95 -0
  40. package/dist/chunk-7OCTVQ7C.js.map +1 -0
  41. package/dist/chunk-7OPMOET7.js +39 -0
  42. package/dist/chunk-7OPMOET7.js.map +1 -0
  43. package/dist/chunk-BXXS7YRC.js +270 -0
  44. package/dist/chunk-BXXS7YRC.js.map +1 -0
  45. package/dist/chunk-CV2Q4YXX.js +272 -0
  46. package/dist/chunk-CV2Q4YXX.js.map +1 -0
  47. package/dist/chunk-EIMMDWIW.js +282 -0
  48. package/dist/chunk-EIMMDWIW.js.map +1 -0
  49. package/dist/chunk-EZ2CWTBE.js +230 -0
  50. package/dist/chunk-EZ2CWTBE.js.map +1 -0
  51. package/dist/chunk-FGHDG3Y4.js +89 -0
  52. package/dist/chunk-FGHDG3Y4.js.map +1 -0
  53. package/dist/chunk-FPP2XLKX.js +127 -0
  54. package/dist/chunk-FPP2XLKX.js.map +1 -0
  55. package/dist/chunk-G6OY35DI.js +295 -0
  56. package/dist/chunk-G6OY35DI.js.map +1 -0
  57. package/dist/chunk-H6KWJNOE.js +65 -0
  58. package/dist/chunk-H6KWJNOE.js.map +1 -0
  59. package/dist/chunk-HGILYGY3.js +45 -0
  60. package/dist/chunk-HGILYGY3.js.map +1 -0
  61. package/dist/chunk-I3BCB4Z5.js +88 -0
  62. package/dist/chunk-I3BCB4Z5.js.map +1 -0
  63. package/dist/chunk-KBWNUUWM.js +582 -0
  64. package/dist/chunk-KBWNUUWM.js.map +1 -0
  65. package/dist/chunk-KN7JFAZ6.js +113 -0
  66. package/dist/chunk-KN7JFAZ6.js.map +1 -0
  67. package/dist/chunk-MEF7PI6U.js +16 -0
  68. package/dist/chunk-MEF7PI6U.js.map +1 -0
  69. package/dist/chunk-NKGMQL6I.js +310 -0
  70. package/dist/chunk-NKGMQL6I.js.map +1 -0
  71. package/dist/chunk-NMFQRGLL.js +127 -0
  72. package/dist/chunk-NMFQRGLL.js.map +1 -0
  73. package/dist/chunk-OUBWD6CX.js +433 -0
  74. package/dist/chunk-OUBWD6CX.js.map +1 -0
  75. package/dist/chunk-PFNXVBLU.js +96 -0
  76. package/dist/chunk-PFNXVBLU.js.map +1 -0
  77. package/dist/chunk-PUPZ4HME.js +165 -0
  78. package/dist/chunk-PUPZ4HME.js.map +1 -0
  79. package/dist/chunk-QFS52OK5.js +690 -0
  80. package/dist/chunk-QFS52OK5.js.map +1 -0
  81. package/dist/chunk-QNC6O3PG.js +45 -0
  82. package/dist/chunk-QNC6O3PG.js.map +1 -0
  83. package/dist/chunk-QUHOXWBK.js +82 -0
  84. package/dist/chunk-QUHOXWBK.js.map +1 -0
  85. package/dist/chunk-UIQGSTBJ.js +106 -0
  86. package/dist/chunk-UIQGSTBJ.js.map +1 -0
  87. package/dist/chunk-UJQKVP6V.js +193 -0
  88. package/dist/chunk-UJQKVP6V.js.map +1 -0
  89. package/dist/chunk-VVPGEAC6.js +11 -0
  90. package/dist/chunk-VVPGEAC6.js.map +1 -0
  91. package/dist/chunk-XA3T5KWA.js +58 -0
  92. package/dist/chunk-XA3T5KWA.js.map +1 -0
  93. package/dist/chunk-YSHJHSJM.js +19 -0
  94. package/dist/chunk-YSHJHSJM.js.map +1 -0
  95. package/dist/chunk-YVHOAVSM.js +182 -0
  96. package/dist/chunk-YVHOAVSM.js.map +1 -0
  97. package/dist/collapse.d.ts +16 -0
  98. package/dist/collapse.js +3 -0
  99. package/dist/collapse.js.map +1 -0
  100. package/dist/count-badge.d.ts +11 -0
  101. package/dist/count-badge.js +4 -0
  102. package/dist/count-badge.js.map +1 -0
  103. package/dist/date-field.d.ts +39 -0
  104. package/dist/date-field.js +8 -0
  105. package/dist/date-field.js.map +1 -0
  106. package/dist/date-range-field.d.ts +30 -0
  107. package/dist/date-range-field.js +8 -0
  108. package/dist/date-range-field.js.map +1 -0
  109. package/dist/datetime-field.d.ts +28 -0
  110. package/dist/datetime-field.js +10 -0
  111. package/dist/datetime-field.js.map +1 -0
  112. package/dist/dialog.d.ts +26 -0
  113. package/dist/dialog.js +7 -0
  114. package/dist/dialog.js.map +1 -0
  115. package/dist/index.d.ts +35 -0
  116. package/dist/index.js +40 -0
  117. package/dist/index.js.map +1 -0
  118. package/dist/motion-tokens.d.ts +29 -0
  119. package/dist/motion-tokens.js +3 -0
  120. package/dist/motion-tokens.js.map +1 -0
  121. package/dist/multi-select.d.ts +25 -0
  122. package/dist/multi-select.js +7 -0
  123. package/dist/multi-select.js.map +1 -0
  124. package/dist/number-field.d.ts +24 -0
  125. package/dist/number-field.js +4 -0
  126. package/dist/number-field.js.map +1 -0
  127. package/dist/otp-field.d.ts +20 -0
  128. package/dist/otp-field.js +3 -0
  129. package/dist/otp-field.js.map +1 -0
  130. package/dist/overlay.d.ts +31 -0
  131. package/dist/overlay.js +4 -0
  132. package/dist/overlay.js.map +1 -0
  133. package/dist/pagination.d.ts +24 -0
  134. package/dist/pagination.js +5 -0
  135. package/dist/pagination.js.map +1 -0
  136. package/dist/radio-group.d.ts +46 -0
  137. package/dist/radio-group.js +6 -0
  138. package/dist/radio-group.js.map +1 -0
  139. package/dist/select-core-SAyS-8w0.d.ts +16 -0
  140. package/dist/select.d.ts +27 -0
  141. package/dist/select.js +7 -0
  142. package/dist/select.js.map +1 -0
  143. package/dist/status-badge.d.ts +17 -0
  144. package/dist/status-badge.js +5 -0
  145. package/dist/status-badge.js.map +1 -0
  146. package/dist/table.d.ts +65 -0
  147. package/dist/table.js +5 -0
  148. package/dist/table.js.map +1 -0
  149. package/dist/tabs.d.ts +44 -0
  150. package/dist/tabs.js +5 -0
  151. package/dist/tabs.js.map +1 -0
  152. package/dist/tag.d.ts +28 -0
  153. package/dist/tag.js +5 -0
  154. package/dist/tag.js.map +1 -0
  155. package/dist/text-field.d.ts +30 -0
  156. package/dist/text-field.js +6 -0
  157. package/dist/text-field.js.map +1 -0
  158. package/dist/textarea.d.ts +33 -0
  159. package/dist/textarea.js +5 -0
  160. package/dist/textarea.js.map +1 -0
  161. package/dist/time-field.d.ts +27 -0
  162. package/dist/time-field.js +6 -0
  163. package/dist/time-field.js.map +1 -0
  164. package/dist/toast-store.d.ts +75 -0
  165. package/dist/toast-store.js +3 -0
  166. package/dist/toast-store.js.map +1 -0
  167. package/dist/toast.d.ts +3 -0
  168. package/dist/toast.js +6 -0
  169. package/dist/toast.js.map +1 -0
  170. package/dist/toggle-tag.d.ts +24 -0
  171. package/dist/toggle-tag.js +4 -0
  172. package/dist/toggle-tag.js.map +1 -0
  173. package/dist/toggle.d.ts +21 -0
  174. package/dist/toggle.js +3 -0
  175. package/dist/toggle.js.map +1 -0
  176. package/dist/tooltip.d.ts +27 -0
  177. package/dist/tooltip.js +4 -0
  178. package/dist/tooltip.js.map +1 -0
  179. package/llms.txt +165 -0
  180. package/package.json +205 -0
  181. package/src/components/alert/Alert.tsx +118 -0
  182. package/src/components/alert/alert.css +136 -0
  183. package/src/components/avatar/Avatar.tsx +128 -0
  184. package/src/components/avatar/AvatarGroup.tsx +50 -0
  185. package/src/components/avatar/avatar.css +200 -0
  186. package/src/components/badge/Badge.tsx +66 -0
  187. package/src/components/badge/CountBadge.tsx +46 -0
  188. package/src/components/badge/StatusBadge.tsx +132 -0
  189. package/src/components/badge/badge.css +243 -0
  190. package/src/components/button/Button.tsx +68 -0
  191. package/src/components/button/button.css +222 -0
  192. package/src/components/checkbox/Checkbox.tsx +90 -0
  193. package/src/components/checkbox/checkbox.css +179 -0
  194. package/src/components/date-picker/DateField.tsx +362 -0
  195. package/src/components/date-picker/DateRangeField.tsx +533 -0
  196. package/src/components/date-picker/DateTimeField.tsx +177 -0
  197. package/src/components/date-picker/TimeField.tsx +100 -0
  198. package/src/components/date-picker/date-picker.css +591 -0
  199. package/src/components/date-picker/date-utils.ts +55 -0
  200. package/src/components/date-picker/field-shell.tsx +78 -0
  201. package/src/components/date-picker/glide-pill.tsx +81 -0
  202. package/src/components/date-picker/time-core.tsx +305 -0
  203. package/src/components/dialog/Dialog.tsx +181 -0
  204. package/src/components/dialog/dialog.css +170 -0
  205. package/src/components/glass/glass.css +100 -0
  206. package/src/components/icon/Icon.tsx +76 -0
  207. package/src/components/icon/IconSlot.tsx +11 -0
  208. package/src/components/icon/icon.css +33 -0
  209. package/src/components/input/NumberField.tsx +117 -0
  210. package/src/components/input/OtpField.tsx +118 -0
  211. package/src/components/input/TextField.tsx +123 -0
  212. package/src/components/input/input.css +335 -0
  213. package/src/components/motion/Collapse.tsx +33 -0
  214. package/src/components/motion/collapse.css +41 -0
  215. package/src/components/overlay/Overlay.tsx +239 -0
  216. package/src/components/overlay/overlay-core.tsx +565 -0
  217. package/src/components/overlay/overlay.css +119 -0
  218. package/src/components/overlay/sheet-drag.tsx +146 -0
  219. package/src/components/pagination/Pagination.tsx +140 -0
  220. package/src/components/pagination/pagination.css +48 -0
  221. package/src/components/radio-group/RadioGroup.tsx +182 -0
  222. package/src/components/radio-group/radio-group.css +277 -0
  223. package/src/components/select/MultiSelect.tsx +251 -0
  224. package/src/components/select/Select.tsx +235 -0
  225. package/src/components/select/select-core.tsx +417 -0
  226. package/src/components/select/select.css +386 -0
  227. package/src/components/table/Table.tsx +433 -0
  228. package/src/components/table/table.css +348 -0
  229. package/src/components/tabs/Tabs.tsx +371 -0
  230. package/src/components/tabs/tabs.css +228 -0
  231. package/src/components/tag/Tag.tsx +145 -0
  232. package/src/components/tag/ToggleTag.tsx +125 -0
  233. package/src/components/tag/tag.css +248 -0
  234. package/src/components/textarea/Textarea.tsx +197 -0
  235. package/src/components/textarea/textarea.css +219 -0
  236. package/src/components/toast/Toast.tsx +349 -0
  237. package/src/components/toast/toast-store.ts +266 -0
  238. package/src/components/toast/toast.css +233 -0
  239. package/src/components/toggle/Toggle.tsx +94 -0
  240. package/src/components/toggle/toggle.css +152 -0
  241. package/src/components/tooltip/Tooltip.tsx +365 -0
  242. package/src/components/tooltip/tooltip.css +86 -0
  243. package/src/index.ts +42 -0
  244. package/src/styles.css +39 -0
  245. package/src/tokens/avatar.css +20 -0
  246. package/src/tokens/color.css +56 -0
  247. package/src/tokens/elevation.css +20 -0
  248. package/src/tokens/fonts.css +3 -0
  249. package/src/tokens/glass.css +21 -0
  250. package/src/tokens/icons.css +7 -0
  251. package/src/tokens/layers.css +6 -0
  252. package/src/tokens/motion-tokens.ts +72 -0
  253. package/src/tokens/motion.css +49 -0
  254. package/src/tokens/radius.css +11 -0
  255. package/src/tokens/semantic.css +75 -0
  256. package/src/tokens/spacing.css +26 -0
  257. package/src/tokens/typography.css +54 -0
@@ -0,0 +1,86 @@
1
+ 'use client';import { UIMotion } from './chunk-37O2ZXD6.js';
2
+ import { Badge } from './chunk-2OWHZ4JT.js';
3
+ import * as React from 'react';
4
+ import { jsx, jsxs } from 'react/jsx-runtime';
5
+
6
+ var SM = UIMotion;
7
+ var POST_STATUS = {
8
+ draft: { tone: "neutral", label: "Draft" },
9
+ scheduled: { tone: "info", label: "Scheduled" },
10
+ processing: { tone: "warning", label: "Processing", live: true },
11
+ published: { tone: "success", label: "Published" },
12
+ failed: { tone: "danger", label: "Failed" }
13
+ };
14
+ var TERMINAL = { published: true, failed: true };
15
+ function StatusBadge({ status, morph = false, ...rest }) {
16
+ if (!morph) {
17
+ const s = POST_STATUS[status] || POST_STATUS.draft;
18
+ return /* @__PURE__ */ jsx(Badge, { tone: s.tone, dot: true, live: !!s.live, ...rest, children: s.label });
19
+ }
20
+ return /* @__PURE__ */ jsx(StatusMorph, { status, ...rest });
21
+ }
22
+ function StatusMorph({ status, className = "", ...rest }) {
23
+ const s = POST_STATUS[status] || POST_STATUS.draft;
24
+ const prev = React.useRef(status);
25
+ const keyRef = React.useRef(1);
26
+ const labelRef = React.useRef(null);
27
+ const ghostRef = React.useRef(null);
28
+ const [words, setWords] = React.useState([{ key: 0, label: s.label, cls: "" }]);
29
+ const [boxW, setBoxW] = React.useState();
30
+ React.useLayoutEffect(() => {
31
+ if (ghostRef.current) setBoxW(ghostRef.current.offsetWidth);
32
+ }, []);
33
+ React.useEffect(() => {
34
+ if (prev.current === status) return;
35
+ prev.current = status;
36
+ const next = POST_STATUS[status] || POST_STATUS.draft;
37
+ const nk = keyRef.current++;
38
+ const nextW = ghostRef.current?.offsetWidth;
39
+ setWords((ws) => ws.concat({ key: nk, label: next.label, cls: "badge__word--in" }));
40
+ const raf = requestAnimationFrame(
41
+ () => requestAnimationFrame(() => {
42
+ setWords(
43
+ (ws) => ws.map((w) => w.key === nk ? { ...w, cls: "" } : { ...w, cls: "badge__word--out" })
44
+ );
45
+ if (nextW) setBoxW(nextW);
46
+ })
47
+ );
48
+ const drop = setTimeout(
49
+ () => setWords((ws) => ws.filter((w) => !w.cls.includes("--out"))),
50
+ SM.dur.base * 1e3 + 80
51
+ );
52
+ let glintT;
53
+ if (TERMINAL[status] && labelRef.current) {
54
+ const chip = labelRef.current.closest(".badge");
55
+ if (chip) {
56
+ chip.classList.remove("glass-glint");
57
+ void chip.offsetWidth;
58
+ chip.classList.add("glass-glint");
59
+ glintT = setTimeout(() => chip.classList.remove("glass-glint"), SM.dur.slow * 1e3 + 80);
60
+ }
61
+ }
62
+ return () => {
63
+ cancelAnimationFrame(raf);
64
+ clearTimeout(drop);
65
+ clearTimeout(glintT);
66
+ };
67
+ }, [status]);
68
+ return /* @__PURE__ */ jsx(
69
+ Badge,
70
+ {
71
+ tone: s.tone,
72
+ dot: true,
73
+ live: !!s.live,
74
+ className: ["badge--morph", className].filter(Boolean).join(" "),
75
+ ...rest,
76
+ children: /* @__PURE__ */ jsxs("span", { className: "badge__morph", ref: labelRef, style: boxW ? { width: boxW } : void 0, children: [
77
+ /* @__PURE__ */ jsx("span", { className: "badge__ghost", ref: ghostRef, "aria-hidden": "true", children: s.label }),
78
+ words.map((w) => /* @__PURE__ */ jsx("span", { className: ["badge__word", w.cls].filter(Boolean).join(" "), children: w.label }, w.key))
79
+ ] })
80
+ }
81
+ );
82
+ }
83
+
84
+ export { POST_STATUS, StatusBadge };
85
+ //# sourceMappingURL=chunk-4HSCN5TZ.js.map
86
+ //# sourceMappingURL=chunk-4HSCN5TZ.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/components/badge/StatusBadge.tsx"],"names":[],"mappings":"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","file":"chunk-4HSCN5TZ.js","sourcesContent":["'use client';\n\n// StatusBadge - a status as a badge; canonical tone+label mapping; morph tweens width in place.\n\nimport * as React from 'react';\nimport { Badge, type BadgeProps } from './Badge';\nimport { UIMotion } from '../../tokens/motion-tokens';\n\nconst SM = UIMotion;\n\nexport type PostStatus = 'draft' | 'scheduled' | 'processing' | 'published' | 'failed';\n\nexport const POST_STATUS: Record<\n PostStatus,\n { tone: BadgeProps['tone']; label: string; live?: boolean }\n> = {\n draft: { tone: 'neutral', label: 'Draft' },\n scheduled: { tone: 'info', label: 'Scheduled' },\n processing: { tone: 'warning', label: 'Processing', live: true },\n published: { tone: 'success', label: 'Published' },\n failed: { tone: 'danger', label: 'Failed' },\n};\n\nconst TERMINAL: Partial<Record<PostStatus, boolean>> = { published: true, failed: true };\n\nexport interface StatusBadgeProps extends Omit<BadgeProps, 'children' | 'tone'> {\n status: PostStatus;\n /** Morph in place as `status` changes instead of swapping. */\n morph?: boolean;\n}\n\nexport function StatusBadge({ status, morph = false, ...rest }: StatusBadgeProps) {\n if (!morph) {\n const s = POST_STATUS[status] || POST_STATUS.draft;\n return (\n <Badge tone={s.tone} dot live={!!s.live} {...rest}>\n {s.label}\n </Badge>\n );\n }\n return <StatusMorph status={status} {...rest} />;\n}\n\ninterface StatusMorphProps extends Omit<BadgeProps, 'children' | 'tone'> {\n status: PostStatus;\n}\n\ninterface Word {\n key: number;\n label: string;\n cls: string;\n}\n\nfunction StatusMorph({ status, className = '', ...rest }: StatusMorphProps) {\n const s = POST_STATUS[status] || POST_STATUS.draft;\n\n const prev = React.useRef(status);\n const keyRef = React.useRef(1);\n const labelRef = React.useRef<HTMLSpanElement>(null);\n const ghostRef = React.useRef<HTMLSpanElement>(null);\n const [words, setWords] = React.useState<Word[]>([{ key: 0, label: s.label, cls: '' }]);\n const [boxW, setBoxW] = React.useState<number>();\n\n // initial width only; on change the width eases in the rAF below, synced to the word roll\n React.useLayoutEffect(() => {\n if (ghostRef.current) setBoxW(ghostRef.current.offsetWidth);\n }, []);\n\n React.useEffect(() => {\n if (prev.current === status) return;\n prev.current = status;\n const next = POST_STATUS[status] || POST_STATUS.draft;\n const nk = keyRef.current++;\n // ghost already holds the new label\n const nextW = ghostRef.current?.offsetWidth;\n\n // add the new word primed below (invisible); the current word stays settled\n setWords((ws) => ws.concat({ key: nk, label: next.label, cls: 'badge__word--in' }));\n // next frame: roll the new word in, the old out, ease the box - one tween; clip hides the wider word\n const raf = requestAnimationFrame(() =>\n requestAnimationFrame(() => {\n setWords((ws) =>\n ws.map((w) => (w.key === nk ? { ...w, cls: '' } : { ...w, cls: 'badge__word--out' })),\n );\n if (nextW) setBoxW(nextW);\n }),\n );\n // old word done rolling out - drop it; width already settled in the rAF, so this only prunes the DOM\n const drop = setTimeout(\n () => setWords((ws) => ws.filter((w) => !w.cls.includes('--out'))),\n SM.dur.base * 1000 + 80,\n );\n // glint when a post lands on a terminal state\n let glintT: ReturnType<typeof setTimeout> | undefined;\n if (TERMINAL[status] && labelRef.current) {\n const chip = labelRef.current.closest('.badge') as HTMLElement | null;\n if (chip) {\n chip.classList.remove('glass-glint');\n // restart the one-shot\n void chip.offsetWidth;\n chip.classList.add('glass-glint');\n glintT = setTimeout(() => chip.classList.remove('glass-glint'), SM.dur.slow * 1000 + 80);\n }\n }\n return () => {\n cancelAnimationFrame(raf);\n clearTimeout(drop);\n clearTimeout(glintT);\n };\n }, [status]);\n\n return (\n <Badge\n tone={s.tone}\n dot\n live={!!s.live}\n className={['badge--morph', className].filter(Boolean).join(' ')}\n {...rest}\n >\n <span className=\"badge__morph\" ref={labelRef} style={boxW ? { width: boxW } : undefined}>\n <span className=\"badge__ghost\" ref={ghostRef} aria-hidden=\"true\">\n {s.label}\n </span>\n {words.map((w) => (\n <span key={w.key} className={['badge__word', w.cls].filter(Boolean).join(' ')}>\n {w.label}\n </span>\n ))}\n </span>\n </Badge>\n );\n}\n"]}
@@ -0,0 +1,258 @@
1
+ 'use client';import * as React from 'react';
2
+ import { jsxs, jsx } from 'react/jsx-runtime';
3
+
4
+ // src/components/date-picker/time-core.tsx
5
+ var { useState, useRef, useEffect } = React;
6
+ var tsgPad = (n) => String(n).padStart(2, "0");
7
+ var tsgDisp12 = (h24) => (h24 + 11) % 12 + 1;
8
+ var tsgToH24 = (d12, mer) => d12 % 12 + (mer === "PM" ? 12 : 0);
9
+ function tsgFeed(pend, d, hi) {
10
+ if (pend != null) {
11
+ const n = pend * 10 + d;
12
+ if (n <= hi) return { val: n, done: true };
13
+ }
14
+ return { val: d, done: d * 10 > hi };
15
+ }
16
+ function TimeSegments({
17
+ value = null,
18
+ onCommit,
19
+ format = "24h",
20
+ minuteStep = 5,
21
+ min = null,
22
+ max = null,
23
+ disabled = false,
24
+ ariaLabel = "Time",
25
+ className = ""
26
+ }) {
27
+ const is12 = format === "12h";
28
+ const seed = value ? value.split(":").map(Number) : [null, null];
29
+ const [h, setHState] = useState(seed[0]);
30
+ const [m, setMState] = useState(seed[1]);
31
+ const [mer, setMer] = useState(seed[0] != null && seed[0] >= 12 ? "PM" : "AM");
32
+ const [pend, setPendState] = useState(null);
33
+ const hLive = useRef(seed[0]);
34
+ const mLive = useRef(seed[1]);
35
+ const pendLive = useRef(null);
36
+ const setH = (v) => {
37
+ hLive.current = v;
38
+ setHState(v);
39
+ };
40
+ const setM = (v) => {
41
+ mLive.current = v;
42
+ setMState(v);
43
+ };
44
+ const setPend = (v) => {
45
+ pendLive.current = v;
46
+ setPendState(v);
47
+ };
48
+ const hRef = useRef(null);
49
+ const mRef = useRef(null);
50
+ const merRef = useRef(null);
51
+ const lastRef = useRef(value || null);
52
+ useEffect(() => {
53
+ if ((value || null) === lastRef.current) return;
54
+ lastRef.current = value || null;
55
+ const p = value ? value.split(":").map(Number) : [null, null];
56
+ setH(p[0]);
57
+ setM(p[1]);
58
+ if (p[0] != null) setMer(p[0] >= 12 ? "PM" : "AM");
59
+ setPend(null);
60
+ }, [value]);
61
+ function tryCommit(nh, nm) {
62
+ if (nh == null || nm == null) return;
63
+ let t = tsgPad(nh) + ":" + tsgPad(nm);
64
+ if (min && t < min) t = min;
65
+ if (max && t > max) t = max;
66
+ const p = t.split(":").map(Number);
67
+ if (p[0] !== nh) {
68
+ setH(p[0]);
69
+ setMer(p[0] >= 12 ? "PM" : "AM");
70
+ setPend(null);
71
+ }
72
+ if (p[1] !== nm) setM(p[1]);
73
+ if (t === lastRef.current) return;
74
+ lastRef.current = t;
75
+ if (onCommit) onCommit(t);
76
+ }
77
+ const focusSeg = (r) => {
78
+ if (r.current) r.current.focus();
79
+ };
80
+ function onHKey(e) {
81
+ const k = e.key;
82
+ if (/^[0-9]$/.test(k)) {
83
+ const pd = pend && pend.seg === "h" ? pend.d : null;
84
+ const r = tsgFeed(pd, +k, is12 ? 12 : 23);
85
+ const nh = is12 ? tsgToH24(r.val === 0 ? 12 : r.val, mer) : r.val;
86
+ setH(nh);
87
+ if (!is12) setMer(nh >= 12 ? "PM" : "AM");
88
+ setPend(r.done ? null : { seg: "h", d: r.val });
89
+ if (r.done) {
90
+ focusSeg(mRef);
91
+ tryCommit(nh, mLive.current);
92
+ }
93
+ } else if (k === "ArrowUp" || k === "ArrowDown") {
94
+ const dir = k === "ArrowUp" ? 1 : -1;
95
+ const nh = h == null ? (/* @__PURE__ */ new Date()).getHours() : (h + dir + 24) % 24;
96
+ setH(nh);
97
+ setMer(nh >= 12 ? "PM" : "AM");
98
+ setPend(null);
99
+ tryCommit(nh, mLive.current);
100
+ } else if (k === "Backspace" || k === "Delete") {
101
+ setH(null);
102
+ setPend(null);
103
+ } else if (k === "ArrowRight" || k === ":" || k === ";") {
104
+ focusSeg(mRef);
105
+ } else if (k === "Tab") {
106
+ return;
107
+ } else if (k.length === 1 && !e.metaKey && !e.ctrlKey) ; else return;
108
+ e.preventDefault();
109
+ }
110
+ function onMKey(e) {
111
+ const k = e.key;
112
+ if (/^[0-9]$/.test(k)) {
113
+ const pd = pend && pend.seg === "m" ? pend.d : null;
114
+ const r = tsgFeed(pd, +k, 59);
115
+ setM(r.val);
116
+ setPend(r.done ? null : { seg: "m", d: r.val });
117
+ if (r.done) {
118
+ if (is12) focusSeg(merRef);
119
+ tryCommit(hLive.current, r.val);
120
+ }
121
+ } else if (k === "ArrowUp" || k === "ArrowDown") {
122
+ const dir = k === "ArrowUp" ? 1 : -1;
123
+ let nm;
124
+ if (m == null) {
125
+ nm = Math.round((/* @__PURE__ */ new Date()).getMinutes() / minuteStep) * minuteStep % 60;
126
+ } else {
127
+ const next = dir > 0 ? (Math.floor(m / minuteStep) + 1) * minuteStep : (Math.ceil(m / minuteStep) - 1) * minuteStep;
128
+ nm = (next % 60 + 60) % 60;
129
+ }
130
+ setM(nm);
131
+ setPend(null);
132
+ tryCommit(hLive.current, nm);
133
+ } else if (k === "Backspace" || k === "Delete") {
134
+ if (m == null && !(pend && pend.seg === "m")) {
135
+ focusSeg(hRef);
136
+ } else {
137
+ setM(null);
138
+ setPend(null);
139
+ }
140
+ } else if (k === "ArrowLeft") {
141
+ focusSeg(hRef);
142
+ } else if (k === "ArrowRight" && is12) {
143
+ focusSeg(merRef);
144
+ } else if (k === "Tab") {
145
+ return;
146
+ } else if (k.length === 1 && !e.metaKey && !e.ctrlKey) ; else return;
147
+ e.preventDefault();
148
+ }
149
+ function setMeridiem(next) {
150
+ setMer(next);
151
+ if (h != null) {
152
+ const nh = tsgToH24(tsgDisp12(h), next);
153
+ setH(nh);
154
+ tryCommit(nh, mLive.current);
155
+ }
156
+ }
157
+ function onMerKey(e) {
158
+ const k = e.key;
159
+ if (k === "a" || k === "A") setMeridiem("AM");
160
+ else if (k === "p" || k === "P") setMeridiem("PM");
161
+ else if (k === "ArrowUp" || k === "ArrowDown") setMeridiem(mer === "AM" ? "PM" : "AM");
162
+ else if (k === "ArrowLeft") focusSeg(mRef);
163
+ else if (k === "Tab") {
164
+ return;
165
+ } else if (k.length === 1 && !e.metaKey && !e.ctrlKey) ; else return;
166
+ e.preventDefault();
167
+ }
168
+ function onPaste(e) {
169
+ if (disabled) return;
170
+ const txt = (e.clipboardData.getData("text") || "").trim();
171
+ const mt = txt.match(/^(\d{1,2})[:h.\s]?(\d{2})?\s*([ap])\.?m?\.?$/i) || txt.match(/^(\d{1,2})[:h.\s]?(\d{2})?$/);
172
+ if (!mt) return;
173
+ e.preventDefault();
174
+ let hh = +mt[1];
175
+ const mm = mt[2] != null ? +mt[2] : 0;
176
+ if (mt[3]) hh = tsgToH24(hh, /^p/i.test(mt[3]) ? "PM" : "AM");
177
+ if (hh > 23 || mm > 59) return;
178
+ setH(hh);
179
+ setM(mm);
180
+ setMer(hh >= 12 ? "PM" : "AM");
181
+ setPend(null);
182
+ tryCommit(hh, mm);
183
+ }
184
+ const onSegBlur = (seg) => () => {
185
+ if (pendLive.current && pendLive.current.seg === seg) {
186
+ setPend(null);
187
+ tryCommit(hLive.current, mLive.current);
188
+ }
189
+ };
190
+ const hText = pend && pend.seg === "h" ? tsgPad(pend.d) : h == null ? "--" : tsgPad(is12 ? tsgDisp12(h) : h);
191
+ const mText = pend && pend.seg === "m" ? tsgPad(pend.d) : m == null ? "--" : tsgPad(m);
192
+ const segCls = (empty) => "tsg__seg" + (empty ? " is-empty" : "");
193
+ const tab = disabled ? -1 : 0;
194
+ return /* @__PURE__ */ jsxs(
195
+ "div",
196
+ {
197
+ className: ("tsg" + (disabled ? " is-disabled" : "") + " " + className).trim(),
198
+ role: "group",
199
+ "aria-label": ariaLabel,
200
+ "aria-disabled": disabled || void 0,
201
+ onPaste,
202
+ children: [
203
+ /* @__PURE__ */ jsx(
204
+ "span",
205
+ {
206
+ ref: hRef,
207
+ className: segCls(h == null && !(pend && pend.seg === "h")),
208
+ role: "spinbutton",
209
+ tabIndex: tab,
210
+ "aria-label": "Hours",
211
+ "aria-valuemin": is12 ? 1 : 0,
212
+ "aria-valuemax": is12 ? 12 : 23,
213
+ "aria-valuenow": h == null ? void 0 : is12 ? tsgDisp12(h) : h,
214
+ "aria-valuetext": h == null ? "Empty" : hText,
215
+ onKeyDown: disabled ? void 0 : onHKey,
216
+ onBlur: onSegBlur("h"),
217
+ children: hText
218
+ }
219
+ ),
220
+ /* @__PURE__ */ jsx("span", { className: "tsg__sep", "aria-hidden": "true", children: ":" }),
221
+ /* @__PURE__ */ jsx(
222
+ "span",
223
+ {
224
+ ref: mRef,
225
+ className: segCls(m == null && !(pend && pend.seg === "m")),
226
+ role: "spinbutton",
227
+ tabIndex: tab,
228
+ "aria-label": "Minutes",
229
+ "aria-valuemin": 0,
230
+ "aria-valuemax": 59,
231
+ "aria-valuenow": m == null ? void 0 : m,
232
+ "aria-valuetext": m == null ? "Empty" : mText,
233
+ onKeyDown: disabled ? void 0 : onMKey,
234
+ onBlur: onSegBlur("m"),
235
+ children: mText
236
+ }
237
+ ),
238
+ is12 ? /* @__PURE__ */ jsx(
239
+ "span",
240
+ {
241
+ ref: merRef,
242
+ className: "tsg__seg tsg__seg--mer",
243
+ role: "spinbutton",
244
+ tabIndex: tab,
245
+ "aria-label": "AM or PM",
246
+ "aria-valuetext": mer,
247
+ onKeyDown: disabled ? void 0 : onMerKey,
248
+ children: mer
249
+ }
250
+ ) : null
251
+ ]
252
+ }
253
+ );
254
+ }
255
+
256
+ export { TimeSegments };
257
+ //# sourceMappingURL=chunk-5DDOOT33.js.map
258
+ //# sourceMappingURL=chunk-5DDOOT33.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/components/date-picker/time-core.tsx"],"names":[],"mappings":"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","file":"chunk-5DDOOT33.js","sourcesContent":["'use client';\n\n/* TimeSegments - the segmented HH:MM machine: every keystroke interpreted (never inserted); commit is live and clamped (saturate, never error). */\n\nimport * as React from 'react';\n\nconst { useState, useRef, useEffect } = React;\n\ntype Meridiem = 'AM' | 'PM';\ntype Segment = 'h' | 'm';\ninterface Pending {\n seg: Segment;\n d: number;\n}\n\nexport interface TimeSegmentsProps {\n value?: string | null;\n onCommit?: (value: string) => void;\n format?: '24h' | '12h';\n minuteStep?: number;\n min?: string | null;\n max?: string | null;\n disabled?: boolean;\n ariaLabel?: string;\n className?: string;\n}\n\nconst tsgPad = (n: number): string => String(n).padStart(2, '0');\nconst tsgDisp12 = (h24: number): number => ((h24 + 11) % 12) + 1;\nconst tsgToH24 = (d12: number, mer: Meridiem): number => (d12 % 12) + (mer === 'PM' ? 12 : 0);\n\n/* feed one digit into a bounded two-digit segment; done = saturated (can't take another). */\nfunction tsgFeed(pend: number | null, d: number, hi: number): { val: number; done: boolean } {\n if (pend != null) {\n const n = pend * 10 + d;\n if (n <= hi) return { val: n, done: true };\n }\n return { val: d, done: d * 10 > hi };\n}\n\nexport function TimeSegments({\n value = null,\n onCommit,\n format = '24h',\n minuteStep = 5,\n min = null,\n max = null,\n disabled = false,\n ariaLabel = 'Time',\n className = '',\n}: TimeSegmentsProps) {\n const is12 = format === '12h';\n const seed = value ? value.split(':').map(Number) : [null, null];\n\n const [h, setHState] = useState<number | null>(seed[0]);\n const [m, setMState] = useState<number | null>(seed[1]);\n const [mer, setMer] = useState<Meridiem>(seed[0] != null && seed[0] >= 12 ? 'PM' : 'AM');\n const [pend, setPendState] = useState<Pending | null>(null);\n\n /* live mirrors - blur fires synchronously before re-render, so blur-commit reads refs, not stale closure state. */\n const hLive = useRef<number | null>(seed[0]);\n const mLive = useRef<number | null>(seed[1]);\n const pendLive = useRef<Pending | null>(null);\n const setH = (v: number | null) => {\n hLive.current = v;\n setHState(v);\n };\n const setM = (v: number | null) => {\n mLive.current = v;\n setMState(v);\n };\n const setPend = (v: Pending | null) => {\n pendLive.current = v;\n setPendState(v);\n };\n\n const hRef = useRef<HTMLSpanElement>(null);\n const mRef = useRef<HTMLSpanElement>(null);\n const merRef = useRef<HTMLSpanElement>(null);\n const lastRef = useRef<string | null>(value || null);\n\n /* external value changes re-seed; our own commits (matched via lastRef) are left alone, never clobbering in-flight typing. */\n useEffect(() => {\n if ((value || null) === lastRef.current) return;\n lastRef.current = value || null;\n const p = value ? value.split(':').map(Number) : [null, null];\n setH(p[0]);\n setM(p[1]);\n if (p[0] != null) setMer(p[0] >= 12 ? 'PM' : 'AM');\n setPend(null);\n }, [value]);\n\n function tryCommit(nh: number | null, nm: number | null) {\n if (nh == null || nm == null) return;\n let t = tsgPad(nh) + ':' + tsgPad(nm);\n if (min && t < min) t = min;\n if (max && t > max) t = max;\n const p = t.split(':').map(Number);\n if (p[0] !== nh) {\n setH(p[0]);\n setMer(p[0] >= 12 ? 'PM' : 'AM');\n setPend(null);\n }\n if (p[1] !== nm) setM(p[1]);\n if (t === lastRef.current) return;\n lastRef.current = t;\n if (onCommit) onCommit(t);\n }\n\n const focusSeg = (r: React.RefObject<HTMLSpanElement | null>) => {\n if (r.current) r.current.focus();\n };\n\n function onHKey(e: React.KeyboardEvent<HTMLSpanElement>) {\n const k = e.key;\n if (/^[0-9]$/.test(k)) {\n const pd = pend && pend.seg === 'h' ? pend.d : null;\n const r = tsgFeed(pd, +k, is12 ? 12 : 23);\n const nh = is12 ? tsgToH24(r.val === 0 ? 12 : r.val, mer) : r.val;\n setH(nh);\n if (!is12) setMer(nh >= 12 ? 'PM' : 'AM');\n setPend(r.done ? null : { seg: 'h', d: r.val });\n if (r.done) {\n focusSeg(mRef);\n tryCommit(nh, mLive.current);\n }\n } else if (k === 'ArrowUp' || k === 'ArrowDown') {\n const dir = k === 'ArrowUp' ? 1 : -1;\n const nh = h == null ? new Date().getHours() : (h + dir + 24) % 24;\n setH(nh);\n setMer(nh >= 12 ? 'PM' : 'AM');\n setPend(null);\n tryCommit(nh, mLive.current);\n } else if (k === 'Backspace' || k === 'Delete') {\n setH(null);\n setPend(null);\n } else if (k === 'ArrowRight' || k === ':' || k === ';') {\n focusSeg(mRef);\n } else if (k === 'Tab') {\n return;\n } else if (k.length === 1 && !e.metaKey && !e.ctrlKey) {\n /* swallow */\n } else return;\n e.preventDefault();\n }\n\n function onMKey(e: React.KeyboardEvent<HTMLSpanElement>) {\n const k = e.key;\n if (/^[0-9]$/.test(k)) {\n const pd = pend && pend.seg === 'm' ? pend.d : null;\n const r = tsgFeed(pd, +k, 59);\n setM(r.val);\n setPend(r.done ? null : { seg: 'm', d: r.val });\n if (r.done) {\n if (is12) focusSeg(merRef);\n tryCommit(hLive.current, r.val);\n }\n } else if (k === 'ArrowUp' || k === 'ArrowDown') {\n const dir = k === 'ArrowUp' ? 1 : -1;\n let nm;\n if (m == null) {\n nm = (Math.round(new Date().getMinutes() / minuteStep) * minuteStep) % 60;\n } else {\n /* step to the NEXT grid point, so 07 + ↑(step 5) lands on 10 */\n const next =\n dir > 0\n ? (Math.floor(m / minuteStep) + 1) * minuteStep\n : (Math.ceil(m / minuteStep) - 1) * minuteStep;\n nm = ((next % 60) + 60) % 60;\n }\n setM(nm);\n setPend(null);\n tryCommit(hLive.current, nm);\n } else if (k === 'Backspace' || k === 'Delete') {\n if (m == null && !(pend && pend.seg === 'm')) {\n focusSeg(hRef);\n } else {\n setM(null);\n setPend(null);\n }\n } else if (k === 'ArrowLeft') {\n focusSeg(hRef);\n } else if (k === 'ArrowRight' && is12) {\n focusSeg(merRef);\n } else if (k === 'Tab') {\n return;\n } else if (k.length === 1 && !e.metaKey && !e.ctrlKey) {\n /* swallow */\n } else return;\n e.preventDefault();\n }\n\n function setMeridiem(next: Meridiem) {\n setMer(next);\n if (h != null) {\n const nh = tsgToH24(tsgDisp12(h), next);\n setH(nh);\n tryCommit(nh, mLive.current);\n }\n }\n function onMerKey(e: React.KeyboardEvent<HTMLSpanElement>) {\n const k = e.key;\n if (k === 'a' || k === 'A') setMeridiem('AM');\n else if (k === 'p' || k === 'P') setMeridiem('PM');\n else if (k === 'ArrowUp' || k === 'ArrowDown') setMeridiem(mer === 'AM' ? 'PM' : 'AM');\n else if (k === 'ArrowLeft') focusSeg(mRef);\n else if (k === 'Tab') {\n return;\n } else if (k.length === 1 && !e.metaKey && !e.ctrlKey) {\n /* swallow */\n } else return;\n e.preventDefault();\n }\n\n function onPaste(e: React.ClipboardEvent<HTMLDivElement>) {\n if (disabled) return;\n const txt = (e.clipboardData.getData('text') || '').trim();\n const mt =\n txt.match(/^(\\d{1,2})[:h.\\s]?(\\d{2})?\\s*([ap])\\.?m?\\.?$/i) ||\n txt.match(/^(\\d{1,2})[:h.\\s]?(\\d{2})?$/);\n if (!mt) return;\n e.preventDefault();\n let hh = +mt[1];\n const mm = mt[2] != null ? +mt[2] : 0;\n if (mt[3]) hh = tsgToH24(hh, /^p/i.test(mt[3]) ? 'PM' : 'AM');\n if (hh > 23 || mm > 59) return;\n setH(hh);\n setM(mm);\n setMer(hh >= 12 ? 'PM' : 'AM');\n setPend(null);\n tryCommit(hh, mm);\n }\n\n const onSegBlur = (seg: Segment) => () => {\n if (pendLive.current && pendLive.current.seg === seg) {\n setPend(null);\n tryCommit(hLive.current, mLive.current);\n }\n };\n\n /* pending digit overrides canonical display, so typing '0' in 12h shows '00', not a premature '12'. */\n const hText =\n pend && pend.seg === 'h' ? tsgPad(pend.d) : h == null ? '--' : tsgPad(is12 ? tsgDisp12(h) : h);\n const mText = pend && pend.seg === 'm' ? tsgPad(pend.d) : m == null ? '--' : tsgPad(m);\n\n const segCls = (empty: boolean): string => 'tsg__seg' + (empty ? ' is-empty' : '');\n const tab = disabled ? -1 : 0;\n\n return (\n <div\n className={('tsg' + (disabled ? ' is-disabled' : '') + ' ' + className).trim()}\n role=\"group\"\n aria-label={ariaLabel}\n aria-disabled={disabled || undefined}\n onPaste={onPaste}\n >\n <span\n ref={hRef}\n className={segCls(h == null && !(pend && pend.seg === 'h'))}\n role=\"spinbutton\"\n tabIndex={tab}\n aria-label=\"Hours\"\n aria-valuemin={is12 ? 1 : 0}\n aria-valuemax={is12 ? 12 : 23}\n aria-valuenow={h == null ? undefined : is12 ? tsgDisp12(h) : h}\n aria-valuetext={h == null ? 'Empty' : hText}\n onKeyDown={disabled ? undefined : onHKey}\n onBlur={onSegBlur('h')}\n >\n {hText}\n </span>\n <span className=\"tsg__sep\" aria-hidden=\"true\">\n :\n </span>\n <span\n ref={mRef}\n className={segCls(m == null && !(pend && pend.seg === 'm'))}\n role=\"spinbutton\"\n tabIndex={tab}\n aria-label=\"Minutes\"\n aria-valuemin={0}\n aria-valuemax={59}\n aria-valuenow={m == null ? undefined : m}\n aria-valuetext={m == null ? 'Empty' : mText}\n onKeyDown={disabled ? undefined : onMKey}\n onBlur={onSegBlur('m')}\n >\n {mText}\n </span>\n {is12 ? (\n <span\n ref={merRef}\n className=\"tsg__seg tsg__seg--mer\"\n role=\"spinbutton\"\n tabIndex={tab}\n aria-label=\"AM or PM\"\n aria-valuetext={mer}\n onKeyDown={disabled ? undefined : onMerKey}\n >\n {mer}\n </span>\n ) : null}\n </div>\n );\n}\n"]}
@@ -0,0 +1,117 @@
1
+ 'use client';import { Icon } from './chunk-KBWNUUWM.js';
2
+ import { UIMotion } from './chunk-37O2ZXD6.js';
3
+ import * as React from 'react';
4
+ import { animate } from 'motion/react';
5
+ import { jsxs, jsx } from 'react/jsx-runtime';
6
+
7
+ var { useEffect, useRef } = React;
8
+ var pgnTravelPx = null;
9
+ function pgnTravel() {
10
+ if (pgnTravelPx == null) {
11
+ const cs = getComputedStyle(document.documentElement);
12
+ const v = cs.getPropertyValue("--space-2").trim();
13
+ const n = parseFloat(v);
14
+ pgnTravelPx = v.endsWith("rem") ? n * parseFloat(cs.fontSize) : n;
15
+ }
16
+ return pgnTravelPx;
17
+ }
18
+ function pgnFormat(n) {
19
+ return n.toLocaleString("en-US").replace(/,/g, "\u2009");
20
+ }
21
+ function Pagination({
22
+ label = "Pagination",
23
+ range,
24
+ total = null,
25
+ hasPrev = false,
26
+ hasNext = false,
27
+ onPrev,
28
+ onNext,
29
+ loading = false,
30
+ className = ""
31
+ }) {
32
+ const rangeRef = useRef(null);
33
+ const prevBtnRef = useRef(null);
34
+ const nextBtnRef = useRef(null);
35
+ const shownRef = useRef(range);
36
+ const lastDirRef = useRef(0);
37
+ const from = range[0];
38
+ const to = range[1];
39
+ useEffect(() => {
40
+ const pf = shownRef.current[0], pt = shownRef.current[1];
41
+ if (pf === from && pt === to) return;
42
+ const dir = from > pf ? 1 : -1;
43
+ shownRef.current = range;
44
+ animate(rangeRef.current, { x: [dir * pgnTravel(), 0], opacity: [0, 1] }, UIMotion.t.enter);
45
+ }, [from, to]);
46
+ useEffect(() => {
47
+ if (loading || document.activeElement !== document.body) return;
48
+ const d = lastDirRef.current;
49
+ if (d === 1 && !hasNext && hasPrev && prevBtnRef.current) prevBtnRef.current.focus();
50
+ if (d === -1 && !hasPrev && hasNext && nextBtnRef.current) nextBtnRef.current.focus();
51
+ }, [loading, hasPrev, hasNext]);
52
+ const prevBusy = loading && lastDirRef.current === -1;
53
+ const nextBusy = loading && lastDirRef.current === 1;
54
+ return /* @__PURE__ */ jsxs(
55
+ "nav",
56
+ {
57
+ className: ("pgn " + className).trim(),
58
+ "aria-label": label,
59
+ "aria-busy": loading || void 0,
60
+ children: [
61
+ /* @__PURE__ */ jsx("span", { className: "pgn__readout", "aria-live": "polite", children: /* @__PURE__ */ jsxs("span", { className: "pgn__range", ref: rangeRef, children: [
62
+ /* @__PURE__ */ jsxs("b", { children: [
63
+ pgnFormat(from),
64
+ "-",
65
+ pgnFormat(to)
66
+ ] }),
67
+ total != null ? /* @__PURE__ */ jsxs("span", { className: "pgn__total", children: [
68
+ " of ",
69
+ pgnFormat(total)
70
+ ] }) : null
71
+ ] }) }),
72
+ /* @__PURE__ */ jsxs("div", { className: "pgn__nav", children: [
73
+ /* @__PURE__ */ jsxs(
74
+ "button",
75
+ {
76
+ type: "button",
77
+ ref: prevBtnRef,
78
+ className: "btn btn--ghost btn--sm pgn__btn pgn__btn--prev" + (prevBusy ? " is-loading" : ""),
79
+ disabled: !hasPrev || loading,
80
+ "aria-label": "Previous page",
81
+ onClick: () => {
82
+ lastDirRef.current = -1;
83
+ if (onPrev) onPrev();
84
+ },
85
+ children: [
86
+ /* @__PURE__ */ jsx(Icon, { name: "caret-left", size: "sm" }),
87
+ prevBusy ? /* @__PURE__ */ jsx("span", { className: "btn__spinner", "aria-hidden": "true" }) : null
88
+ ]
89
+ }
90
+ ),
91
+ /* @__PURE__ */ jsxs(
92
+ "button",
93
+ {
94
+ type: "button",
95
+ ref: nextBtnRef,
96
+ className: "btn btn--ghost btn--sm pgn__btn pgn__btn--next" + (nextBusy ? " is-loading" : ""),
97
+ disabled: !hasNext || loading,
98
+ "aria-label": "Next page",
99
+ onClick: () => {
100
+ lastDirRef.current = 1;
101
+ if (onNext) onNext();
102
+ },
103
+ children: [
104
+ /* @__PURE__ */ jsx(Icon, { name: "caret-right", size: "sm" }),
105
+ nextBusy ? /* @__PURE__ */ jsx("span", { className: "btn__spinner", "aria-hidden": "true" }) : null
106
+ ]
107
+ }
108
+ )
109
+ ] })
110
+ ]
111
+ }
112
+ );
113
+ }
114
+
115
+ export { Pagination };
116
+ //# sourceMappingURL=chunk-5FVHWIMY.js.map
117
+ //# sourceMappingURL=chunk-5FVHWIMY.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/components/pagination/Pagination.tsx"],"names":[],"mappings":"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","file":"chunk-5FVHWIMY.js","sourcesContent":["'use client';\n\n/* Pagination - pure cursor strip: mono range readout + prev/next, no page numbers. */\n\nimport * as React from 'react';\nimport { animate } from 'motion/react';\nimport { UIMotion } from '../../tokens/motion-tokens';\nimport { Icon } from '../icon/Icon';\n\nexport interface PaginationProps {\n /** Accessible name for the nav landmark - name the list (\"Posts\"), not \"pagination\". @default 'Pagination' */\n label?: string;\n /** Items currently shown, 1-based inclusive: `[from, to]` (e.g. `[26, 50]`). */\n range: [number, number];\n /** Total item count - render `of N` only when the API reports one; omit for endless lists. */\n total?: number | null;\n /** A previous cursor exists. @default false */\n hasPrev?: boolean;\n /** A next cursor exists. @default false */\n hasNext?: boolean;\n /** Fired when the previous arrow is pressed. */\n onPrev?: () => void;\n /** Fired when the next arrow is pressed. */\n onNext?: () => void;\n /** A page fetch is in flight: both arrows go inert, the clicked arrow carries the spinner. @default false */\n loading?: boolean;\n className?: string;\n}\n\nconst { useEffect, useRef } = React;\n\n/* --space-2 in real px - tokens resolve to \"0.5rem\"; bare parseFloat gives 0.5, so convert rem-px */\nlet pgnTravelPx: number | null = null;\nfunction pgnTravel() {\n if (pgnTravelPx == null) {\n const cs = getComputedStyle(document.documentElement);\n const v = cs.getPropertyValue('--space-2').trim();\n const n = parseFloat(v);\n pgnTravelPx = v.endsWith('rem') ? n * parseFloat(cs.fontSize) : n;\n }\n return pgnTravelPx;\n}\n\n/* thin-space thousands grouping (\"48 210\") - section E mono-numeral convention */\nfunction pgnFormat(n: number) {\n return n.toLocaleString('en-US').replace(/,/g, '\\u2009');\n}\n\nexport function Pagination({\n label = 'Pagination',\n range,\n total = null,\n hasPrev = false,\n hasNext = false,\n onPrev,\n onNext,\n loading = false,\n className = '',\n}: PaginationProps) {\n const rangeRef = useRef<HTMLSpanElement>(null);\n const prevBtnRef = useRef<HTMLButtonElement>(null);\n const nextBtnRef = useRef<HTMLButtonElement>(null);\n const shownRef = useRef<[number, number]>(range); // last range rendered (skip first mount)\n const lastDirRef = useRef(0); // -1 prev - +1 next - last arrow fired\n\n const from = range[0];\n const to = range[1];\n\n /* direction-aware entrance: new numbers arrive from the side you traveled toward (one persistent node, imperative keyframes) */\n useEffect(() => {\n const pf = shownRef.current[0],\n pt = shownRef.current[1];\n if (pf === from && pt === to) return;\n const dir = from > pf ? 1 : -1;\n shownRef.current = range;\n animate(rangeRef.current, { x: [dir * pgnTravel(), 0], opacity: [0, 1] }, UIMotion.t.enter);\n }, [from, to]);\n\n /* edge disables the focused arrow and focus falls to <body> - hand it to the surviving direction; activeElement guard avoids stealing focus */\n useEffect(() => {\n if (loading || document.activeElement !== document.body) return;\n const d = lastDirRef.current;\n if (d === 1 && !hasNext && hasPrev && prevBtnRef.current) prevBtnRef.current.focus();\n if (d === -1 && !hasPrev && hasNext && nextBtnRef.current) nextBtnRef.current.focus();\n }, [loading, hasPrev, hasNext]);\n\n const prevBusy = loading && lastDirRef.current === -1;\n const nextBusy = loading && lastDirRef.current === 1;\n\n return (\n <nav\n className={('pgn ' + className).trim()}\n aria-label={label}\n aria-busy={loading || undefined}\n >\n <span className=\"pgn__readout\" aria-live=\"polite\">\n <span className=\"pgn__range\" ref={rangeRef}>\n <b>\n {pgnFormat(from)}-{pgnFormat(to)}\n </b>\n {total != null ? <span className=\"pgn__total\"> of {pgnFormat(total)}</span> : null}\n </span>\n </span>\n <div className=\"pgn__nav\">\n <button\n type=\"button\"\n ref={prevBtnRef}\n className={\n 'btn btn--ghost btn--sm pgn__btn pgn__btn--prev' + (prevBusy ? ' is-loading' : '')\n }\n disabled={!hasPrev || loading}\n aria-label=\"Previous page\"\n onClick={() => {\n lastDirRef.current = -1;\n if (onPrev) onPrev();\n }}\n >\n <Icon name=\"caret-left\" size=\"sm\" />\n {prevBusy ? <span className=\"btn__spinner\" aria-hidden=\"true\"></span> : null}\n </button>\n <button\n type=\"button\"\n ref={nextBtnRef}\n className={\n 'btn btn--ghost btn--sm pgn__btn pgn__btn--next' + (nextBusy ? ' is-loading' : '')\n }\n disabled={!hasNext || loading}\n aria-label=\"Next page\"\n onClick={() => {\n lastDirRef.current = 1;\n if (onNext) onNext();\n }}\n >\n <Icon name=\"caret-right\" size=\"sm\" />\n {nextBusy ? <span className=\"btn__spinner\" aria-hidden=\"true\"></span> : null}\n </button>\n </div>\n </nav>\n );\n}\n"]}