@open-mercato/ui 0.4.2-canary-c02407ff85

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 (319) hide show
  1. package/build.mjs +62 -0
  2. package/dist/backend/AppShell.js +902 -0
  3. package/dist/backend/AppShell.js.map +7 -0
  4. package/dist/backend/ConfirmDialog.js +17 -0
  5. package/dist/backend/ConfirmDialog.js.map +7 -0
  6. package/dist/backend/ContextHelp.js +31 -0
  7. package/dist/backend/ContextHelp.js.map +7 -0
  8. package/dist/backend/CrudForm.js +2028 -0
  9. package/dist/backend/CrudForm.js.map +7 -0
  10. package/dist/backend/DataTable.js +1363 -0
  11. package/dist/backend/DataTable.js.map +7 -0
  12. package/dist/backend/EmptyState.js +52 -0
  13. package/dist/backend/EmptyState.js.map +7 -0
  14. package/dist/backend/FilterBar.js +140 -0
  15. package/dist/backend/FilterBar.js.map +7 -0
  16. package/dist/backend/FilterOverlay.js +279 -0
  17. package/dist/backend/FilterOverlay.js.map +7 -0
  18. package/dist/backend/FlashMessages.js +66 -0
  19. package/dist/backend/FlashMessages.js.map +7 -0
  20. package/dist/backend/JsonBuilder.js +322 -0
  21. package/dist/backend/JsonBuilder.js.map +7 -0
  22. package/dist/backend/JsonDisplay.js +203 -0
  23. package/dist/backend/JsonDisplay.js.map +7 -0
  24. package/dist/backend/Page.js +27 -0
  25. package/dist/backend/Page.js.map +7 -0
  26. package/dist/backend/PerspectiveSidebar.js +282 -0
  27. package/dist/backend/PerspectiveSidebar.js.map +7 -0
  28. package/dist/backend/RowActions.js +148 -0
  29. package/dist/backend/RowActions.js.map +7 -0
  30. package/dist/backend/TruncatedCell.js +92 -0
  31. package/dist/backend/TruncatedCell.js.map +7 -0
  32. package/dist/backend/UserMenu.js +107 -0
  33. package/dist/backend/UserMenu.js.map +7 -0
  34. package/dist/backend/ValueIcons.js +34 -0
  35. package/dist/backend/ValueIcons.js.map +7 -0
  36. package/dist/backend/custom-fields/FieldDefinitionsEditor.js +1264 -0
  37. package/dist/backend/custom-fields/FieldDefinitionsEditor.js.map +7 -0
  38. package/dist/backend/custom-fields/FieldDefinitionsManager.js +332 -0
  39. package/dist/backend/custom-fields/FieldDefinitionsManager.js.map +7 -0
  40. package/dist/backend/dashboard/DashboardScreen.js +578 -0
  41. package/dist/backend/dashboard/DashboardScreen.js.map +7 -0
  42. package/dist/backend/dashboard/index.js +5 -0
  43. package/dist/backend/dashboard/index.js.map +7 -0
  44. package/dist/backend/dashboard/widgetRegistry.js +55 -0
  45. package/dist/backend/dashboard/widgetRegistry.js.map +7 -0
  46. package/dist/backend/detail/ActivitiesSection.js +962 -0
  47. package/dist/backend/detail/ActivitiesSection.js.map +7 -0
  48. package/dist/backend/detail/AddressEditor.js +413 -0
  49. package/dist/backend/detail/AddressEditor.js.map +7 -0
  50. package/dist/backend/detail/AddressTiles.js +437 -0
  51. package/dist/backend/detail/AddressTiles.js.map +7 -0
  52. package/dist/backend/detail/AddressesSection.js +264 -0
  53. package/dist/backend/detail/AddressesSection.js.map +7 -0
  54. package/dist/backend/detail/AttachmentDeleteDialog.js +41 -0
  55. package/dist/backend/detail/AttachmentDeleteDialog.js.map +7 -0
  56. package/dist/backend/detail/AttachmentMetadataDialog.js +517 -0
  57. package/dist/backend/detail/AttachmentMetadataDialog.js.map +7 -0
  58. package/dist/backend/detail/AttachmentsSection.js +367 -0
  59. package/dist/backend/detail/AttachmentsSection.js.map +7 -0
  60. package/dist/backend/detail/CustomDataSection.js +433 -0
  61. package/dist/backend/detail/CustomDataSection.js.map +7 -0
  62. package/dist/backend/detail/DetailFieldsSection.js +75 -0
  63. package/dist/backend/detail/DetailFieldsSection.js.map +7 -0
  64. package/dist/backend/detail/ErrorMessage.js +28 -0
  65. package/dist/backend/detail/ErrorMessage.js.map +7 -0
  66. package/dist/backend/detail/InlineEditors.js +681 -0
  67. package/dist/backend/detail/InlineEditors.js.map +7 -0
  68. package/dist/backend/detail/LoadingMessage.js +14 -0
  69. package/dist/backend/detail/LoadingMessage.js.map +7 -0
  70. package/dist/backend/detail/NotesSection.js +1032 -0
  71. package/dist/backend/detail/NotesSection.js.map +7 -0
  72. package/dist/backend/detail/TabEmptyState.js +25 -0
  73. package/dist/backend/detail/TabEmptyState.js.map +7 -0
  74. package/dist/backend/detail/TagsSection.js +254 -0
  75. package/dist/backend/detail/TagsSection.js.map +7 -0
  76. package/dist/backend/detail/addressFormat.js +77 -0
  77. package/dist/backend/detail/addressFormat.js.map +7 -0
  78. package/dist/backend/detail/index.js +34 -0
  79. package/dist/backend/detail/index.js.map +7 -0
  80. package/dist/backend/fields/registry.generated.js +8 -0
  81. package/dist/backend/fields/registry.generated.js.map +7 -0
  82. package/dist/backend/fields/registry.js +29 -0
  83. package/dist/backend/fields/registry.js.map +7 -0
  84. package/dist/backend/indexes/PartialIndexBanner.js +58 -0
  85. package/dist/backend/indexes/PartialIndexBanner.js.map +7 -0
  86. package/dist/backend/indexes/store.js +62 -0
  87. package/dist/backend/indexes/store.js.map +7 -0
  88. package/dist/backend/injection/InjectionSpot.js +179 -0
  89. package/dist/backend/injection/InjectionSpot.js.map +7 -0
  90. package/dist/backend/injection/PageInjectionBoundary.js +26 -0
  91. package/dist/backend/injection/PageInjectionBoundary.js.map +7 -0
  92. package/dist/backend/injection/helpers.js +26 -0
  93. package/dist/backend/injection/helpers.js.map +7 -0
  94. package/dist/backend/injection/widgetRegistry.js +55 -0
  95. package/dist/backend/injection/widgetRegistry.js.map +7 -0
  96. package/dist/backend/inputs/ComboboxInput.js +225 -0
  97. package/dist/backend/inputs/ComboboxInput.js.map +7 -0
  98. package/dist/backend/inputs/LookupSelect.js +191 -0
  99. package/dist/backend/inputs/LookupSelect.js.map +7 -0
  100. package/dist/backend/inputs/PhoneNumberField.js +100 -0
  101. package/dist/backend/inputs/PhoneNumberField.js.map +7 -0
  102. package/dist/backend/inputs/SwitchableMarkdownInput.js +92 -0
  103. package/dist/backend/inputs/SwitchableMarkdownInput.js.map +7 -0
  104. package/dist/backend/inputs/TagsInput.js +222 -0
  105. package/dist/backend/inputs/TagsInput.js.map +7 -0
  106. package/dist/backend/inputs/index.js +6 -0
  107. package/dist/backend/inputs/index.js.map +7 -0
  108. package/dist/backend/operations/LastOperationBanner.js +80 -0
  109. package/dist/backend/operations/LastOperationBanner.js.map +7 -0
  110. package/dist/backend/operations/store.js +183 -0
  111. package/dist/backend/operations/store.js.map +7 -0
  112. package/dist/backend/schedule/ScheduleAgenda.js +107 -0
  113. package/dist/backend/schedule/ScheduleAgenda.js.map +7 -0
  114. package/dist/backend/schedule/ScheduleGrid.js +107 -0
  115. package/dist/backend/schedule/ScheduleGrid.js.map +7 -0
  116. package/dist/backend/schedule/ScheduleToolbar.js +166 -0
  117. package/dist/backend/schedule/ScheduleToolbar.js.map +7 -0
  118. package/dist/backend/schedule/ScheduleView.js +165 -0
  119. package/dist/backend/schedule/ScheduleView.js.map +7 -0
  120. package/dist/backend/schedule/index.js +6 -0
  121. package/dist/backend/schedule/index.js.map +7 -0
  122. package/dist/backend/schedule/recurrence.js +83 -0
  123. package/dist/backend/schedule/recurrence.js.map +7 -0
  124. package/dist/backend/schedule/types.js +1 -0
  125. package/dist/backend/schedule/types.js.map +7 -0
  126. package/dist/backend/upgrades/UpgradeActionBanner.js +91 -0
  127. package/dist/backend/upgrades/UpgradeActionBanner.js.map +7 -0
  128. package/dist/backend/utils/api.js +127 -0
  129. package/dist/backend/utils/api.js.map +7 -0
  130. package/dist/backend/utils/apiCall.js +48 -0
  131. package/dist/backend/utils/apiCall.js.map +7 -0
  132. package/dist/backend/utils/crud.js +126 -0
  133. package/dist/backend/utils/crud.js.map +7 -0
  134. package/dist/backend/utils/customFieldColumns.js +56 -0
  135. package/dist/backend/utils/customFieldColumns.js.map +7 -0
  136. package/dist/backend/utils/customFieldDefs.js +143 -0
  137. package/dist/backend/utils/customFieldDefs.js.map +7 -0
  138. package/dist/backend/utils/customFieldFilters.js +126 -0
  139. package/dist/backend/utils/customFieldFilters.js.map +7 -0
  140. package/dist/backend/utils/customFieldForms.js +162 -0
  141. package/dist/backend/utils/customFieldForms.js.map +7 -0
  142. package/dist/backend/utils/customFieldValues.js +26 -0
  143. package/dist/backend/utils/customFieldValues.js.map +7 -0
  144. package/dist/backend/utils/flash.js +16 -0
  145. package/dist/backend/utils/flash.js.map +7 -0
  146. package/dist/backend/utils/nav.js +185 -0
  147. package/dist/backend/utils/nav.js.map +7 -0
  148. package/dist/backend/utils/serverErrors.js +230 -0
  149. package/dist/backend/utils/serverErrors.js.map +7 -0
  150. package/dist/frontend/AuthFooter.js +23 -0
  151. package/dist/frontend/AuthFooter.js.map +7 -0
  152. package/dist/frontend/LanguageSwitcher.js +57 -0
  153. package/dist/frontend/LanguageSwitcher.js.map +7 -0
  154. package/dist/frontend/Layout.js +14 -0
  155. package/dist/frontend/Layout.js.map +7 -0
  156. package/dist/index.js +32 -0
  157. package/dist/index.js.map +7 -0
  158. package/dist/primitives/DataLoader.js +67 -0
  159. package/dist/primitives/DataLoader.js.map +7 -0
  160. package/dist/primitives/ErrorNotice.js +20 -0
  161. package/dist/primitives/ErrorNotice.js.map +7 -0
  162. package/dist/primitives/alert.js +38 -0
  163. package/dist/primitives/alert.js.map +7 -0
  164. package/dist/primitives/badge.js +28 -0
  165. package/dist/primitives/badge.js.map +7 -0
  166. package/dist/primitives/button.js +44 -0
  167. package/dist/primitives/button.js.map +7 -0
  168. package/dist/primitives/card.js +91 -0
  169. package/dist/primitives/card.js.map +7 -0
  170. package/dist/primitives/checkbox.js +28 -0
  171. package/dist/primitives/checkbox.js.map +7 -0
  172. package/dist/primitives/dialog.js +90 -0
  173. package/dist/primitives/dialog.js.map +7 -0
  174. package/dist/primitives/input.js +22 -0
  175. package/dist/primitives/input.js.map +7 -0
  176. package/dist/primitives/label.js +21 -0
  177. package/dist/primitives/label.js.map +7 -0
  178. package/dist/primitives/separator.js +9 -0
  179. package/dist/primitives/separator.js.map +7 -0
  180. package/dist/primitives/spinner.js +24 -0
  181. package/dist/primitives/spinner.js.map +7 -0
  182. package/dist/primitives/switch.js +80 -0
  183. package/dist/primitives/switch.js.map +7 -0
  184. package/dist/primitives/table.js +29 -0
  185. package/dist/primitives/table.js.map +7 -0
  186. package/dist/primitives/tabs.js +87 -0
  187. package/dist/primitives/tabs.js.map +7 -0
  188. package/dist/primitives/textarea.js +21 -0
  189. package/dist/primitives/textarea.js.map +7 -0
  190. package/dist/primitives/tooltip.js +60 -0
  191. package/dist/primitives/tooltip.js.map +7 -0
  192. package/dist/theme/QueryProvider.js +44 -0
  193. package/dist/theme/QueryProvider.js.map +7 -0
  194. package/dist/theme/ThemeProvider.js +95 -0
  195. package/dist/theme/ThemeProvider.js.map +7 -0
  196. package/dist/theme/ThemeToggle.js +88 -0
  197. package/dist/theme/ThemeToggle.js.map +7 -0
  198. package/dist/theme/index.js +10 -0
  199. package/dist/theme/index.js.map +7 -0
  200. package/dist/types/react-big-calendar.d.js +1 -0
  201. package/dist/types/react-big-calendar.d.js.map +7 -0
  202. package/jest.config.cjs +23 -0
  203. package/jest.setup.ts +55 -0
  204. package/package.json +105 -0
  205. package/src/backend/AppShell.tsx +1096 -0
  206. package/src/backend/ConfirmDialog.tsx +19 -0
  207. package/src/backend/ContextHelp.tsx +38 -0
  208. package/src/backend/CrudForm.tsx +2503 -0
  209. package/src/backend/DataTable.tsx +1730 -0
  210. package/src/backend/EmptyState.tsx +65 -0
  211. package/src/backend/FilterBar.tsx +161 -0
  212. package/src/backend/FilterOverlay.tsx +328 -0
  213. package/src/backend/FlashMessages.tsx +82 -0
  214. package/src/backend/JsonBuilder.tsx +362 -0
  215. package/src/backend/JsonDisplay.tsx +254 -0
  216. package/src/backend/Page.tsx +30 -0
  217. package/src/backend/PerspectiveSidebar.tsx +337 -0
  218. package/src/backend/RowActions.tsx +151 -0
  219. package/src/backend/TruncatedCell.tsx +133 -0
  220. package/src/backend/UserMenu.tsx +118 -0
  221. package/src/backend/ValueIcons.tsx +48 -0
  222. package/src/backend/__tests__/AppShell.test.tsx +115 -0
  223. package/src/backend/__tests__/CrudForm.render.test.tsx +30 -0
  224. package/src/backend/__tests__/DataTable.render.test.tsx +48 -0
  225. package/src/backend/__tests__/custom-field-filters.test.ts +72 -0
  226. package/src/backend/__tests__/custom-field-forms.test.ts +54 -0
  227. package/src/backend/__tests__/serverErrors.test.ts +83 -0
  228. package/src/backend/custom-fields/FieldDefinitionsEditor.tsx +1292 -0
  229. package/src/backend/custom-fields/FieldDefinitionsManager.tsx +381 -0
  230. package/src/backend/dashboard/DashboardScreen.tsx +684 -0
  231. package/src/backend/dashboard/__tests__/DashboardScreen.test.tsx +112 -0
  232. package/src/backend/dashboard/index.ts +1 -0
  233. package/src/backend/dashboard/widgetRegistry.ts +68 -0
  234. package/src/backend/detail/ActivitiesSection.tsx +1284 -0
  235. package/src/backend/detail/AddressEditor.tsx +472 -0
  236. package/src/backend/detail/AddressTiles.tsx +587 -0
  237. package/src/backend/detail/AddressesSection.tsx +346 -0
  238. package/src/backend/detail/AttachmentDeleteDialog.tsx +56 -0
  239. package/src/backend/detail/AttachmentMetadataDialog.tsx +672 -0
  240. package/src/backend/detail/AttachmentsSection.tsx +414 -0
  241. package/src/backend/detail/CustomDataSection.tsx +530 -0
  242. package/src/backend/detail/DetailFieldsSection.tsx +147 -0
  243. package/src/backend/detail/ErrorMessage.tsx +32 -0
  244. package/src/backend/detail/InlineEditors.tsx +877 -0
  245. package/src/backend/detail/LoadingMessage.tsx +14 -0
  246. package/src/backend/detail/NotesSection.tsx +1275 -0
  247. package/src/backend/detail/TabEmptyState.tsx +48 -0
  248. package/src/backend/detail/TagsSection.tsx +314 -0
  249. package/src/backend/detail/addressFormat.tsx +121 -0
  250. package/src/backend/detail/index.ts +44 -0
  251. package/src/backend/fields/registry.generated.ts +8 -0
  252. package/src/backend/fields/registry.ts +38 -0
  253. package/src/backend/indexes/PartialIndexBanner.tsx +68 -0
  254. package/src/backend/indexes/store.ts +88 -0
  255. package/src/backend/injection/InjectionSpot.tsx +236 -0
  256. package/src/backend/injection/PageInjectionBoundary.tsx +31 -0
  257. package/src/backend/injection/helpers.ts +35 -0
  258. package/src/backend/injection/widgetRegistry.ts +68 -0
  259. package/src/backend/inputs/ComboboxInput.tsx +269 -0
  260. package/src/backend/inputs/LookupSelect.tsx +247 -0
  261. package/src/backend/inputs/PhoneNumberField.tsx +129 -0
  262. package/src/backend/inputs/SwitchableMarkdownInput.tsx +128 -0
  263. package/src/backend/inputs/TagsInput.tsx +259 -0
  264. package/src/backend/inputs/index.ts +5 -0
  265. package/src/backend/operations/LastOperationBanner.tsx +85 -0
  266. package/src/backend/operations/__tests__/LastOperationBanner.test.tsx +99 -0
  267. package/src/backend/operations/store.ts +230 -0
  268. package/src/backend/schedule/ScheduleAgenda.tsx +136 -0
  269. package/src/backend/schedule/ScheduleGrid.tsx +136 -0
  270. package/src/backend/schedule/ScheduleToolbar.tsx +178 -0
  271. package/src/backend/schedule/ScheduleView.tsx +198 -0
  272. package/src/backend/schedule/index.ts +5 -0
  273. package/src/backend/schedule/recurrence.ts +99 -0
  274. package/src/backend/schedule/types.ts +26 -0
  275. package/src/backend/upgrades/UpgradeActionBanner.tsx +128 -0
  276. package/src/backend/utils/__tests__/apiCall.test.ts +109 -0
  277. package/src/backend/utils/__tests__/crud.test.ts +87 -0
  278. package/src/backend/utils/__tests__/customFieldDefs.test.ts +25 -0
  279. package/src/backend/utils/__tests__/customFieldValues.test.ts +35 -0
  280. package/src/backend/utils/api.ts +149 -0
  281. package/src/backend/utils/apiCall.ts +96 -0
  282. package/src/backend/utils/crud.ts +174 -0
  283. package/src/backend/utils/customFieldColumns.ts +71 -0
  284. package/src/backend/utils/customFieldDefs.ts +245 -0
  285. package/src/backend/utils/customFieldFilters.ts +145 -0
  286. package/src/backend/utils/customFieldForms.ts +196 -0
  287. package/src/backend/utils/customFieldValues.ts +41 -0
  288. package/src/backend/utils/flash.ts +17 -0
  289. package/src/backend/utils/nav.ts +238 -0
  290. package/src/backend/utils/serverErrors.ts +302 -0
  291. package/src/frontend/AuthFooter.tsx +29 -0
  292. package/src/frontend/LanguageSwitcher.tsx +66 -0
  293. package/src/frontend/Layout.tsx +13 -0
  294. package/src/index.ts +32 -0
  295. package/src/primitives/DataLoader.tsx +92 -0
  296. package/src/primitives/ErrorNotice.tsx +26 -0
  297. package/src/primitives/alert.tsx +52 -0
  298. package/src/primitives/badge.tsx +31 -0
  299. package/src/primitives/button.tsx +47 -0
  300. package/src/primitives/card.tsx +92 -0
  301. package/src/primitives/checkbox.tsx +28 -0
  302. package/src/primitives/dialog.tsx +110 -0
  303. package/src/primitives/input.tsx +20 -0
  304. package/src/primitives/label.tsx +18 -0
  305. package/src/primitives/separator.tsx +7 -0
  306. package/src/primitives/spinner.tsx +27 -0
  307. package/src/primitives/switch.tsx +86 -0
  308. package/src/primitives/table.tsx +27 -0
  309. package/src/primitives/tabs.tsx +128 -0
  310. package/src/primitives/textarea.tsx +20 -0
  311. package/src/primitives/tooltip.tsx +85 -0
  312. package/src/theme/QueryProvider.tsx +46 -0
  313. package/src/theme/ThemeProvider.tsx +120 -0
  314. package/src/theme/ThemeToggle.tsx +88 -0
  315. package/src/theme/index.ts +3 -0
  316. package/src/types/react-big-calendar.d.ts +16 -0
  317. package/tsconfig.build.json +11 -0
  318. package/tsconfig.json +9 -0
  319. package/watch.mjs +6 -0
@@ -0,0 +1,107 @@
1
+ "use client";
2
+ import { jsx, jsxs } from "react/jsx-runtime";
3
+ import * as React from "react";
4
+ import { User, LogOut } from "lucide-react";
5
+ import { useT } from "@open-mercato/shared/lib/i18n/context";
6
+ function UserMenu({ email }) {
7
+ const t = useT();
8
+ const [open, setOpen] = React.useState(false);
9
+ const buttonRef = React.useRef(null);
10
+ const menuRef = React.useRef(null);
11
+ const logoutButtonRef = React.useRef(null);
12
+ const toggle = () => setOpen((v) => !v);
13
+ const onMouseEnter = () => setOpen(true);
14
+ const onMouseLeave = () => setOpen(false);
15
+ React.useEffect(() => {
16
+ if (!open) return;
17
+ function handleClick(event) {
18
+ if (menuRef.current && !menuRef.current.contains(event.target) && buttonRef.current && !buttonRef.current.contains(event.target)) {
19
+ setOpen(false);
20
+ }
21
+ }
22
+ document.addEventListener("mousedown", handleClick);
23
+ return () => document.removeEventListener("mousedown", handleClick);
24
+ }, [open]);
25
+ React.useEffect(() => {
26
+ if (!open) return;
27
+ function handleKeyDown(event) {
28
+ if (event.key === "Escape") {
29
+ setOpen(false);
30
+ buttonRef.current?.focus();
31
+ } else if (event.key === "ArrowDown" || event.key === "Tab") {
32
+ event.preventDefault();
33
+ logoutButtonRef.current?.focus();
34
+ } else if (event.key === "ArrowUp") {
35
+ event.preventDefault();
36
+ logoutButtonRef.current?.focus();
37
+ }
38
+ }
39
+ document.addEventListener("keydown", handleKeyDown);
40
+ return () => document.removeEventListener("keydown", handleKeyDown);
41
+ }, [open]);
42
+ React.useEffect(() => {
43
+ if (open) {
44
+ setTimeout(() => {
45
+ logoutButtonRef.current?.focus();
46
+ }, 0);
47
+ }
48
+ }, [open]);
49
+ return /* @__PURE__ */ jsxs("div", { className: "relative", onMouseEnter, onMouseLeave, children: [
50
+ /* @__PURE__ */ jsx(
51
+ "button",
52
+ {
53
+ ref: buttonRef,
54
+ className: "text-sm px-2 py-1 rounded hover:bg-accent inline-flex items-center gap-2",
55
+ onClick: () => setOpen(true),
56
+ "aria-expanded": open,
57
+ "aria-haspopup": "menu",
58
+ "aria-controls": "user-menu-dropdown",
59
+ id: "user-menu-button",
60
+ type: "button",
61
+ title: email || t("ui.userMenu.userFallback", "User"),
62
+ children: /* @__PURE__ */ jsx(User, { className: "size-4" })
63
+ }
64
+ ),
65
+ open && /* @__PURE__ */ jsxs(
66
+ "div",
67
+ {
68
+ ref: menuRef,
69
+ id: "user-menu-dropdown",
70
+ className: "absolute right-0 top-full mt-0 w-56 rounded-md border bg-background p-1 shadow z-50",
71
+ role: "menu",
72
+ "aria-labelledby": "user-menu-button",
73
+ tabIndex: -1,
74
+ children: [
75
+ email && /* @__PURE__ */ jsxs("div", { className: "px-2 py-2 text-xs text-muted-foreground border-b mb-1", children: [
76
+ /* @__PURE__ */ jsx("div", { className: "font-medium", children: t("ui.userMenu.loggedInAs", "Logged in as:") }),
77
+ /* @__PURE__ */ jsx("div", { className: "truncate", children: email })
78
+ ] }),
79
+ /* @__PURE__ */ jsx("form", { action: "/api/auth/logout", method: "POST", children: /* @__PURE__ */ jsxs(
80
+ "button",
81
+ {
82
+ ref: logoutButtonRef,
83
+ className: "w-full text-left text-sm px-2 py-1 rounded hover:bg-accent inline-flex items-center gap-2 outline-none focus:outline-none focus-visible:outline-none ring-0 focus:ring-0 focus-visible:ring-0",
84
+ type: "submit",
85
+ role: "menuitem",
86
+ tabIndex: 0,
87
+ onKeyDown: (e) => {
88
+ if (e.key === "Escape") {
89
+ setOpen(false);
90
+ buttonRef.current?.focus();
91
+ }
92
+ },
93
+ children: [
94
+ /* @__PURE__ */ jsx(LogOut, { className: "size-4" }),
95
+ /* @__PURE__ */ jsx("span", { children: t("ui.userMenu.logout", "Logout") })
96
+ ]
97
+ }
98
+ ) })
99
+ ]
100
+ }
101
+ )
102
+ ] });
103
+ }
104
+ export {
105
+ UserMenu
106
+ };
107
+ //# sourceMappingURL=UserMenu.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/backend/UserMenu.tsx"],
4
+ "sourcesContent": ["\"use client\"\nimport * as React from 'react'\nimport { User, LogOut } from 'lucide-react'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\n\nexport function UserMenu({ email }: { email?: string }) {\n const t = useT()\n const [open, setOpen] = React.useState(false)\n const buttonRef = React.useRef<HTMLButtonElement>(null)\n const menuRef = React.useRef<HTMLDivElement>(null)\n const logoutButtonRef = React.useRef<HTMLButtonElement>(null)\n\n // Toggle menu open/close\n const toggle = () => setOpen((v) => !v)\n\n // Open on hover, close when mouse leaves the menu area\n const onMouseEnter = () => setOpen(true)\n const onMouseLeave = () => setOpen(false)\n\n // Close menu when clicking outside\n React.useEffect(() => {\n if (!open) return\n function handleClick(event: MouseEvent) {\n if (\n menuRef.current &&\n !menuRef.current.contains(event.target as Node) &&\n buttonRef.current &&\n !buttonRef.current.contains(event.target as Node)\n ) {\n setOpen(false)\n }\n }\n document.addEventListener('mousedown', handleClick)\n return () => document.removeEventListener('mousedown', handleClick)\n }, [open])\n\n // Keyboard navigation\n React.useEffect(() => {\n if (!open) return\n function handleKeyDown(event: KeyboardEvent) {\n if (event.key === 'Escape') {\n setOpen(false)\n buttonRef.current?.focus()\n } else if (event.key === 'ArrowDown' || event.key === 'Tab') {\n event.preventDefault()\n logoutButtonRef.current?.focus()\n } else if (event.key === 'ArrowUp') {\n event.preventDefault()\n logoutButtonRef.current?.focus()\n }\n }\n document.addEventListener('keydown', handleKeyDown)\n return () => document.removeEventListener('keydown', handleKeyDown)\n }, [open])\n\n // Focus the first menu item when menu opens\n React.useEffect(() => {\n if (open) {\n setTimeout(() => {\n logoutButtonRef.current?.focus()\n }, 0)\n }\n }, [open])\n\n return (\n <div className=\"relative\" onMouseEnter={onMouseEnter} onMouseLeave={onMouseLeave}>\n <button\n ref={buttonRef}\n className=\"text-sm px-2 py-1 rounded hover:bg-accent inline-flex items-center gap-2\"\n onClick={() => setOpen(true)}\n aria-expanded={open}\n aria-haspopup=\"menu\"\n aria-controls=\"user-menu-dropdown\"\n id=\"user-menu-button\"\n type=\"button\"\n title={email || t('ui.userMenu.userFallback', 'User')}\n >\n <User className=\"size-4\" />\n </button>\n {open && (\n <div\n ref={menuRef}\n id=\"user-menu-dropdown\"\n className=\"absolute right-0 top-full mt-0 w-56 rounded-md border bg-background p-1 shadow z-50\"\n role=\"menu\"\n aria-labelledby=\"user-menu-button\"\n tabIndex={-1}\n >\n {email && (\n <div className=\"px-2 py-2 text-xs text-muted-foreground border-b mb-1\">\n <div className=\"font-medium\">{t('ui.userMenu.loggedInAs', 'Logged in as:')}</div>\n <div className=\"truncate\">{email}</div>\n </div>\n )}\n <form action=\"/api/auth/logout\" method=\"POST\">\n <button\n ref={logoutButtonRef}\n className=\"w-full text-left text-sm px-2 py-1 rounded hover:bg-accent inline-flex items-center gap-2 outline-none focus:outline-none focus-visible:outline-none ring-0 focus:ring-0 focus-visible:ring-0\"\n type=\"submit\"\n role=\"menuitem\"\n tabIndex={0}\n onKeyDown={(e) => {\n if (e.key === 'Escape') {\n setOpen(false)\n buttonRef.current?.focus()\n }\n }}\n >\n <LogOut className=\"size-4\" />\n <span>{t('ui.userMenu.logout', 'Logout')}</span>\n </button>\n </form>\n </div>\n )}\n </div>\n )\n}\n\n"],
5
+ "mappings": "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",
6
+ "names": []
7
+ }
@@ -0,0 +1,34 @@
1
+ "use client";
2
+ import { jsx, jsxs } from "react/jsx-runtime";
3
+ import { Check, X, AlertTriangle, Minus, Circle } from "lucide-react";
4
+ import { useT } from "@open-mercato/shared/lib/i18n/context";
5
+ function BooleanIcon({ value, trueLabel, falseLabel, className }) {
6
+ const v = !!value;
7
+ return /* @__PURE__ */ jsxs("span", { className: `inline-flex items-center gap-1 ${className ?? ""}`, children: [
8
+ v ? /* @__PURE__ */ jsx(Check, { className: "size-4 text-emerald-600" }) : /* @__PURE__ */ jsx(X, { className: "size-4 text-muted-foreground" }),
9
+ v && trueLabel ? /* @__PURE__ */ jsx("span", { className: "text-xs", children: trueLabel }) : null,
10
+ !v && falseLabel ? /* @__PURE__ */ jsx("span", { className: "text-xs", children: falseLabel }) : null
11
+ ] });
12
+ }
13
+ function EnumBadge({ value, map, fallback }) {
14
+ if (!value) return /* @__PURE__ */ jsx("span", { className: "text-muted-foreground text-xs", children: fallback ?? "\u2014" });
15
+ const cfg = map[value] || { label: String(value) };
16
+ return /* @__PURE__ */ jsxs("span", { className: `inline-flex items-center gap-1 px-2 py-0.5 rounded-full text-xs border ${cfg.className ?? ""}`, children: [
17
+ cfg.icon,
18
+ /* @__PURE__ */ jsx("span", { children: cfg.label })
19
+ ] });
20
+ }
21
+ function useSeverityPreset() {
22
+ const t = useT();
23
+ return {
24
+ low: { label: t("ui.badges.severity.low", "Low"), className: "border-amber-200 text-amber-700 bg-amber-50", icon: /* @__PURE__ */ jsx(Circle, { className: "size-3" }) },
25
+ medium: { label: t("ui.badges.severity.medium", "Medium"), className: "border-yellow-200 text-yellow-800 bg-yellow-50", icon: /* @__PURE__ */ jsx(Minus, { className: "size-3" }) },
26
+ high: { label: t("ui.badges.severity.high", "High"), className: "border-red-200 text-red-700 bg-red-50", icon: /* @__PURE__ */ jsx(AlertTriangle, { className: "size-3" }) }
27
+ };
28
+ }
29
+ export {
30
+ BooleanIcon,
31
+ EnumBadge,
32
+ useSeverityPreset
33
+ };
34
+ //# sourceMappingURL=ValueIcons.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/backend/ValueIcons.tsx"],
4
+ "sourcesContent": ["\"use client\"\nimport * as React from 'react'\nimport { Check, X, AlertTriangle, Minus, Circle } from 'lucide-react'\nimport { useT } from '@open-mercato/shared/lib/i18n/context'\n\nexport function BooleanIcon({ value, trueLabel, falseLabel, className }: {\n value?: boolean | null\n trueLabel?: string\n falseLabel?: string\n className?: string\n}) {\n const v = !!value\n return (\n <span className={`inline-flex items-center gap-1 ${className ?? ''}`}>\n {v ? (\n <Check className=\"size-4 text-emerald-600\" />\n ) : (\n <X className=\"size-4 text-muted-foreground\" />\n )}\n {v && trueLabel ? <span className=\"text-xs\">{trueLabel}</span> : null}\n {!v && falseLabel ? <span className=\"text-xs\">{falseLabel}</span> : null}\n </span>\n )\n}\n\nexport type EnumBadgeMap = Record<string, { label: string; className?: string; icon?: React.ReactNode }>\n\nexport function EnumBadge({ value, map, fallback }: { value?: string | null; map: EnumBadgeMap; fallback?: string }) {\n if (!value) return <span className=\"text-muted-foreground text-xs\">{fallback ?? '\u2014'}</span>\n const cfg = map[value] || { label: String(value) }\n return (\n <span className={`inline-flex items-center gap-1 px-2 py-0.5 rounded-full text-xs border ${cfg.className ?? ''}`}>\n {cfg.icon}\n <span>{cfg.label}</span>\n </span>\n )\n}\n\n// Presets for common enums (optional helper)\nexport function useSeverityPreset(): EnumBadgeMap {\n const t = useT()\n return {\n low: { label: t('ui.badges.severity.low', 'Low'), className: 'border-amber-200 text-amber-700 bg-amber-50', icon: <Circle className=\"size-3\" /> },\n medium: { label: t('ui.badges.severity.medium', 'Medium'), className: 'border-yellow-200 text-yellow-800 bg-yellow-50', icon: <Minus className=\"size-3\" /> },\n high: { label: t('ui.badges.severity.high', 'High'), className: 'border-red-200 text-red-700 bg-red-50', icon: <AlertTriangle className=\"size-3\" /> },\n }\n}\n\n"],
5
+ "mappings": ";AAaI,SAEI,KAFJ;AAXJ,SAAS,OAAO,GAAG,eAAe,OAAO,cAAc;AACvD,SAAS,YAAY;AAEd,SAAS,YAAY,EAAE,OAAO,WAAW,YAAY,UAAU,GAKnE;AACD,QAAM,IAAI,CAAC,CAAC;AACZ,SACE,qBAAC,UAAK,WAAW,kCAAkC,aAAa,EAAE,IAC/D;AAAA,QACC,oBAAC,SAAM,WAAU,2BAA0B,IAE3C,oBAAC,KAAE,WAAU,gCAA+B;AAAA,IAE7C,KAAK,YAAY,oBAAC,UAAK,WAAU,WAAW,qBAAU,IAAU;AAAA,IAChE,CAAC,KAAK,aAAa,oBAAC,UAAK,WAAU,WAAW,sBAAW,IAAU;AAAA,KACtE;AAEJ;AAIO,SAAS,UAAU,EAAE,OAAO,KAAK,SAAS,GAAoE;AACnH,MAAI,CAAC,MAAO,QAAO,oBAAC,UAAK,WAAU,iCAAiC,sBAAY,UAAI;AACpF,QAAM,MAAM,IAAI,KAAK,KAAK,EAAE,OAAO,OAAO,KAAK,EAAE;AACjD,SACE,qBAAC,UAAK,WAAW,0EAA0E,IAAI,aAAa,EAAE,IAC3G;AAAA,QAAI;AAAA,IACL,oBAAC,UAAM,cAAI,OAAM;AAAA,KACnB;AAEJ;AAGO,SAAS,oBAAkC;AAChD,QAAM,IAAI,KAAK;AACf,SAAO;AAAA,IACL,KAAK,EAAE,OAAO,EAAE,0BAA0B,KAAK,GAAG,WAAW,+CAA+C,MAAM,oBAAC,UAAO,WAAU,UAAS,EAAG;AAAA,IAChJ,QAAQ,EAAE,OAAO,EAAE,6BAA6B,QAAQ,GAAG,WAAW,kDAAkD,MAAM,oBAAC,SAAM,WAAU,UAAS,EAAG;AAAA,IAC3J,MAAM,EAAE,OAAO,EAAE,2BAA2B,MAAM,GAAG,WAAW,yCAAyC,MAAM,oBAAC,iBAAc,WAAU,UAAS,EAAG;AAAA,EACtJ;AACF;",
6
+ "names": []
7
+ }