@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.
- package/build.mjs +62 -0
- package/dist/backend/AppShell.js +902 -0
- package/dist/backend/AppShell.js.map +7 -0
- package/dist/backend/ConfirmDialog.js +17 -0
- package/dist/backend/ConfirmDialog.js.map +7 -0
- package/dist/backend/ContextHelp.js +31 -0
- package/dist/backend/ContextHelp.js.map +7 -0
- package/dist/backend/CrudForm.js +2028 -0
- package/dist/backend/CrudForm.js.map +7 -0
- package/dist/backend/DataTable.js +1363 -0
- package/dist/backend/DataTable.js.map +7 -0
- package/dist/backend/EmptyState.js +52 -0
- package/dist/backend/EmptyState.js.map +7 -0
- package/dist/backend/FilterBar.js +140 -0
- package/dist/backend/FilterBar.js.map +7 -0
- package/dist/backend/FilterOverlay.js +279 -0
- package/dist/backend/FilterOverlay.js.map +7 -0
- package/dist/backend/FlashMessages.js +66 -0
- package/dist/backend/FlashMessages.js.map +7 -0
- package/dist/backend/JsonBuilder.js +322 -0
- package/dist/backend/JsonBuilder.js.map +7 -0
- package/dist/backend/JsonDisplay.js +203 -0
- package/dist/backend/JsonDisplay.js.map +7 -0
- package/dist/backend/Page.js +27 -0
- package/dist/backend/Page.js.map +7 -0
- package/dist/backend/PerspectiveSidebar.js +282 -0
- package/dist/backend/PerspectiveSidebar.js.map +7 -0
- package/dist/backend/RowActions.js +148 -0
- package/dist/backend/RowActions.js.map +7 -0
- package/dist/backend/TruncatedCell.js +92 -0
- package/dist/backend/TruncatedCell.js.map +7 -0
- package/dist/backend/UserMenu.js +107 -0
- package/dist/backend/UserMenu.js.map +7 -0
- package/dist/backend/ValueIcons.js +34 -0
- package/dist/backend/ValueIcons.js.map +7 -0
- package/dist/backend/custom-fields/FieldDefinitionsEditor.js +1264 -0
- package/dist/backend/custom-fields/FieldDefinitionsEditor.js.map +7 -0
- package/dist/backend/custom-fields/FieldDefinitionsManager.js +332 -0
- package/dist/backend/custom-fields/FieldDefinitionsManager.js.map +7 -0
- package/dist/backend/dashboard/DashboardScreen.js +578 -0
- package/dist/backend/dashboard/DashboardScreen.js.map +7 -0
- package/dist/backend/dashboard/index.js +5 -0
- package/dist/backend/dashboard/index.js.map +7 -0
- package/dist/backend/dashboard/widgetRegistry.js +55 -0
- package/dist/backend/dashboard/widgetRegistry.js.map +7 -0
- package/dist/backend/detail/ActivitiesSection.js +962 -0
- package/dist/backend/detail/ActivitiesSection.js.map +7 -0
- package/dist/backend/detail/AddressEditor.js +413 -0
- package/dist/backend/detail/AddressEditor.js.map +7 -0
- package/dist/backend/detail/AddressTiles.js +437 -0
- package/dist/backend/detail/AddressTiles.js.map +7 -0
- package/dist/backend/detail/AddressesSection.js +264 -0
- package/dist/backend/detail/AddressesSection.js.map +7 -0
- package/dist/backend/detail/AttachmentDeleteDialog.js +41 -0
- package/dist/backend/detail/AttachmentDeleteDialog.js.map +7 -0
- package/dist/backend/detail/AttachmentMetadataDialog.js +517 -0
- package/dist/backend/detail/AttachmentMetadataDialog.js.map +7 -0
- package/dist/backend/detail/AttachmentsSection.js +367 -0
- package/dist/backend/detail/AttachmentsSection.js.map +7 -0
- package/dist/backend/detail/CustomDataSection.js +433 -0
- package/dist/backend/detail/CustomDataSection.js.map +7 -0
- package/dist/backend/detail/DetailFieldsSection.js +75 -0
- package/dist/backend/detail/DetailFieldsSection.js.map +7 -0
- package/dist/backend/detail/ErrorMessage.js +28 -0
- package/dist/backend/detail/ErrorMessage.js.map +7 -0
- package/dist/backend/detail/InlineEditors.js +681 -0
- package/dist/backend/detail/InlineEditors.js.map +7 -0
- package/dist/backend/detail/LoadingMessage.js +14 -0
- package/dist/backend/detail/LoadingMessage.js.map +7 -0
- package/dist/backend/detail/NotesSection.js +1032 -0
- package/dist/backend/detail/NotesSection.js.map +7 -0
- package/dist/backend/detail/TabEmptyState.js +25 -0
- package/dist/backend/detail/TabEmptyState.js.map +7 -0
- package/dist/backend/detail/TagsSection.js +254 -0
- package/dist/backend/detail/TagsSection.js.map +7 -0
- package/dist/backend/detail/addressFormat.js +77 -0
- package/dist/backend/detail/addressFormat.js.map +7 -0
- package/dist/backend/detail/index.js +34 -0
- package/dist/backend/detail/index.js.map +7 -0
- package/dist/backend/fields/registry.generated.js +8 -0
- package/dist/backend/fields/registry.generated.js.map +7 -0
- package/dist/backend/fields/registry.js +29 -0
- package/dist/backend/fields/registry.js.map +7 -0
- package/dist/backend/indexes/PartialIndexBanner.js +58 -0
- package/dist/backend/indexes/PartialIndexBanner.js.map +7 -0
- package/dist/backend/indexes/store.js +62 -0
- package/dist/backend/indexes/store.js.map +7 -0
- package/dist/backend/injection/InjectionSpot.js +179 -0
- package/dist/backend/injection/InjectionSpot.js.map +7 -0
- package/dist/backend/injection/PageInjectionBoundary.js +26 -0
- package/dist/backend/injection/PageInjectionBoundary.js.map +7 -0
- package/dist/backend/injection/helpers.js +26 -0
- package/dist/backend/injection/helpers.js.map +7 -0
- package/dist/backend/injection/widgetRegistry.js +55 -0
- package/dist/backend/injection/widgetRegistry.js.map +7 -0
- package/dist/backend/inputs/ComboboxInput.js +225 -0
- package/dist/backend/inputs/ComboboxInput.js.map +7 -0
- package/dist/backend/inputs/LookupSelect.js +191 -0
- package/dist/backend/inputs/LookupSelect.js.map +7 -0
- package/dist/backend/inputs/PhoneNumberField.js +100 -0
- package/dist/backend/inputs/PhoneNumberField.js.map +7 -0
- package/dist/backend/inputs/SwitchableMarkdownInput.js +92 -0
- package/dist/backend/inputs/SwitchableMarkdownInput.js.map +7 -0
- package/dist/backend/inputs/TagsInput.js +222 -0
- package/dist/backend/inputs/TagsInput.js.map +7 -0
- package/dist/backend/inputs/index.js +6 -0
- package/dist/backend/inputs/index.js.map +7 -0
- package/dist/backend/operations/LastOperationBanner.js +80 -0
- package/dist/backend/operations/LastOperationBanner.js.map +7 -0
- package/dist/backend/operations/store.js +183 -0
- package/dist/backend/operations/store.js.map +7 -0
- package/dist/backend/schedule/ScheduleAgenda.js +107 -0
- package/dist/backend/schedule/ScheduleAgenda.js.map +7 -0
- package/dist/backend/schedule/ScheduleGrid.js +107 -0
- package/dist/backend/schedule/ScheduleGrid.js.map +7 -0
- package/dist/backend/schedule/ScheduleToolbar.js +166 -0
- package/dist/backend/schedule/ScheduleToolbar.js.map +7 -0
- package/dist/backend/schedule/ScheduleView.js +165 -0
- package/dist/backend/schedule/ScheduleView.js.map +7 -0
- package/dist/backend/schedule/index.js +6 -0
- package/dist/backend/schedule/index.js.map +7 -0
- package/dist/backend/schedule/recurrence.js +83 -0
- package/dist/backend/schedule/recurrence.js.map +7 -0
- package/dist/backend/schedule/types.js +1 -0
- package/dist/backend/schedule/types.js.map +7 -0
- package/dist/backend/upgrades/UpgradeActionBanner.js +91 -0
- package/dist/backend/upgrades/UpgradeActionBanner.js.map +7 -0
- package/dist/backend/utils/api.js +127 -0
- package/dist/backend/utils/api.js.map +7 -0
- package/dist/backend/utils/apiCall.js +48 -0
- package/dist/backend/utils/apiCall.js.map +7 -0
- package/dist/backend/utils/crud.js +126 -0
- package/dist/backend/utils/crud.js.map +7 -0
- package/dist/backend/utils/customFieldColumns.js +56 -0
- package/dist/backend/utils/customFieldColumns.js.map +7 -0
- package/dist/backend/utils/customFieldDefs.js +143 -0
- package/dist/backend/utils/customFieldDefs.js.map +7 -0
- package/dist/backend/utils/customFieldFilters.js +126 -0
- package/dist/backend/utils/customFieldFilters.js.map +7 -0
- package/dist/backend/utils/customFieldForms.js +162 -0
- package/dist/backend/utils/customFieldForms.js.map +7 -0
- package/dist/backend/utils/customFieldValues.js +26 -0
- package/dist/backend/utils/customFieldValues.js.map +7 -0
- package/dist/backend/utils/flash.js +16 -0
- package/dist/backend/utils/flash.js.map +7 -0
- package/dist/backend/utils/nav.js +185 -0
- package/dist/backend/utils/nav.js.map +7 -0
- package/dist/backend/utils/serverErrors.js +230 -0
- package/dist/backend/utils/serverErrors.js.map +7 -0
- package/dist/frontend/AuthFooter.js +23 -0
- package/dist/frontend/AuthFooter.js.map +7 -0
- package/dist/frontend/LanguageSwitcher.js +57 -0
- package/dist/frontend/LanguageSwitcher.js.map +7 -0
- package/dist/frontend/Layout.js +14 -0
- package/dist/frontend/Layout.js.map +7 -0
- package/dist/index.js +32 -0
- package/dist/index.js.map +7 -0
- package/dist/primitives/DataLoader.js +67 -0
- package/dist/primitives/DataLoader.js.map +7 -0
- package/dist/primitives/ErrorNotice.js +20 -0
- package/dist/primitives/ErrorNotice.js.map +7 -0
- package/dist/primitives/alert.js +38 -0
- package/dist/primitives/alert.js.map +7 -0
- package/dist/primitives/badge.js +28 -0
- package/dist/primitives/badge.js.map +7 -0
- package/dist/primitives/button.js +44 -0
- package/dist/primitives/button.js.map +7 -0
- package/dist/primitives/card.js +91 -0
- package/dist/primitives/card.js.map +7 -0
- package/dist/primitives/checkbox.js +28 -0
- package/dist/primitives/checkbox.js.map +7 -0
- package/dist/primitives/dialog.js +90 -0
- package/dist/primitives/dialog.js.map +7 -0
- package/dist/primitives/input.js +22 -0
- package/dist/primitives/input.js.map +7 -0
- package/dist/primitives/label.js +21 -0
- package/dist/primitives/label.js.map +7 -0
- package/dist/primitives/separator.js +9 -0
- package/dist/primitives/separator.js.map +7 -0
- package/dist/primitives/spinner.js +24 -0
- package/dist/primitives/spinner.js.map +7 -0
- package/dist/primitives/switch.js +80 -0
- package/dist/primitives/switch.js.map +7 -0
- package/dist/primitives/table.js +29 -0
- package/dist/primitives/table.js.map +7 -0
- package/dist/primitives/tabs.js +87 -0
- package/dist/primitives/tabs.js.map +7 -0
- package/dist/primitives/textarea.js +21 -0
- package/dist/primitives/textarea.js.map +7 -0
- package/dist/primitives/tooltip.js +60 -0
- package/dist/primitives/tooltip.js.map +7 -0
- package/dist/theme/QueryProvider.js +44 -0
- package/dist/theme/QueryProvider.js.map +7 -0
- package/dist/theme/ThemeProvider.js +95 -0
- package/dist/theme/ThemeProvider.js.map +7 -0
- package/dist/theme/ThemeToggle.js +88 -0
- package/dist/theme/ThemeToggle.js.map +7 -0
- package/dist/theme/index.js +10 -0
- package/dist/theme/index.js.map +7 -0
- package/dist/types/react-big-calendar.d.js +1 -0
- package/dist/types/react-big-calendar.d.js.map +7 -0
- package/jest.config.cjs +23 -0
- package/jest.setup.ts +55 -0
- package/package.json +105 -0
- package/src/backend/AppShell.tsx +1096 -0
- package/src/backend/ConfirmDialog.tsx +19 -0
- package/src/backend/ContextHelp.tsx +38 -0
- package/src/backend/CrudForm.tsx +2503 -0
- package/src/backend/DataTable.tsx +1730 -0
- package/src/backend/EmptyState.tsx +65 -0
- package/src/backend/FilterBar.tsx +161 -0
- package/src/backend/FilterOverlay.tsx +328 -0
- package/src/backend/FlashMessages.tsx +82 -0
- package/src/backend/JsonBuilder.tsx +362 -0
- package/src/backend/JsonDisplay.tsx +254 -0
- package/src/backend/Page.tsx +30 -0
- package/src/backend/PerspectiveSidebar.tsx +337 -0
- package/src/backend/RowActions.tsx +151 -0
- package/src/backend/TruncatedCell.tsx +133 -0
- package/src/backend/UserMenu.tsx +118 -0
- package/src/backend/ValueIcons.tsx +48 -0
- package/src/backend/__tests__/AppShell.test.tsx +115 -0
- package/src/backend/__tests__/CrudForm.render.test.tsx +30 -0
- package/src/backend/__tests__/DataTable.render.test.tsx +48 -0
- package/src/backend/__tests__/custom-field-filters.test.ts +72 -0
- package/src/backend/__tests__/custom-field-forms.test.ts +54 -0
- package/src/backend/__tests__/serverErrors.test.ts +83 -0
- package/src/backend/custom-fields/FieldDefinitionsEditor.tsx +1292 -0
- package/src/backend/custom-fields/FieldDefinitionsManager.tsx +381 -0
- package/src/backend/dashboard/DashboardScreen.tsx +684 -0
- package/src/backend/dashboard/__tests__/DashboardScreen.test.tsx +112 -0
- package/src/backend/dashboard/index.ts +1 -0
- package/src/backend/dashboard/widgetRegistry.ts +68 -0
- package/src/backend/detail/ActivitiesSection.tsx +1284 -0
- package/src/backend/detail/AddressEditor.tsx +472 -0
- package/src/backend/detail/AddressTiles.tsx +587 -0
- package/src/backend/detail/AddressesSection.tsx +346 -0
- package/src/backend/detail/AttachmentDeleteDialog.tsx +56 -0
- package/src/backend/detail/AttachmentMetadataDialog.tsx +672 -0
- package/src/backend/detail/AttachmentsSection.tsx +414 -0
- package/src/backend/detail/CustomDataSection.tsx +530 -0
- package/src/backend/detail/DetailFieldsSection.tsx +147 -0
- package/src/backend/detail/ErrorMessage.tsx +32 -0
- package/src/backend/detail/InlineEditors.tsx +877 -0
- package/src/backend/detail/LoadingMessage.tsx +14 -0
- package/src/backend/detail/NotesSection.tsx +1275 -0
- package/src/backend/detail/TabEmptyState.tsx +48 -0
- package/src/backend/detail/TagsSection.tsx +314 -0
- package/src/backend/detail/addressFormat.tsx +121 -0
- package/src/backend/detail/index.ts +44 -0
- package/src/backend/fields/registry.generated.ts +8 -0
- package/src/backend/fields/registry.ts +38 -0
- package/src/backend/indexes/PartialIndexBanner.tsx +68 -0
- package/src/backend/indexes/store.ts +88 -0
- package/src/backend/injection/InjectionSpot.tsx +236 -0
- package/src/backend/injection/PageInjectionBoundary.tsx +31 -0
- package/src/backend/injection/helpers.ts +35 -0
- package/src/backend/injection/widgetRegistry.ts +68 -0
- package/src/backend/inputs/ComboboxInput.tsx +269 -0
- package/src/backend/inputs/LookupSelect.tsx +247 -0
- package/src/backend/inputs/PhoneNumberField.tsx +129 -0
- package/src/backend/inputs/SwitchableMarkdownInput.tsx +128 -0
- package/src/backend/inputs/TagsInput.tsx +259 -0
- package/src/backend/inputs/index.ts +5 -0
- package/src/backend/operations/LastOperationBanner.tsx +85 -0
- package/src/backend/operations/__tests__/LastOperationBanner.test.tsx +99 -0
- package/src/backend/operations/store.ts +230 -0
- package/src/backend/schedule/ScheduleAgenda.tsx +136 -0
- package/src/backend/schedule/ScheduleGrid.tsx +136 -0
- package/src/backend/schedule/ScheduleToolbar.tsx +178 -0
- package/src/backend/schedule/ScheduleView.tsx +198 -0
- package/src/backend/schedule/index.ts +5 -0
- package/src/backend/schedule/recurrence.ts +99 -0
- package/src/backend/schedule/types.ts +26 -0
- package/src/backend/upgrades/UpgradeActionBanner.tsx +128 -0
- package/src/backend/utils/__tests__/apiCall.test.ts +109 -0
- package/src/backend/utils/__tests__/crud.test.ts +87 -0
- package/src/backend/utils/__tests__/customFieldDefs.test.ts +25 -0
- package/src/backend/utils/__tests__/customFieldValues.test.ts +35 -0
- package/src/backend/utils/api.ts +149 -0
- package/src/backend/utils/apiCall.ts +96 -0
- package/src/backend/utils/crud.ts +174 -0
- package/src/backend/utils/customFieldColumns.ts +71 -0
- package/src/backend/utils/customFieldDefs.ts +245 -0
- package/src/backend/utils/customFieldFilters.ts +145 -0
- package/src/backend/utils/customFieldForms.ts +196 -0
- package/src/backend/utils/customFieldValues.ts +41 -0
- package/src/backend/utils/flash.ts +17 -0
- package/src/backend/utils/nav.ts +238 -0
- package/src/backend/utils/serverErrors.ts +302 -0
- package/src/frontend/AuthFooter.tsx +29 -0
- package/src/frontend/LanguageSwitcher.tsx +66 -0
- package/src/frontend/Layout.tsx +13 -0
- package/src/index.ts +32 -0
- package/src/primitives/DataLoader.tsx +92 -0
- package/src/primitives/ErrorNotice.tsx +26 -0
- package/src/primitives/alert.tsx +52 -0
- package/src/primitives/badge.tsx +31 -0
- package/src/primitives/button.tsx +47 -0
- package/src/primitives/card.tsx +92 -0
- package/src/primitives/checkbox.tsx +28 -0
- package/src/primitives/dialog.tsx +110 -0
- package/src/primitives/input.tsx +20 -0
- package/src/primitives/label.tsx +18 -0
- package/src/primitives/separator.tsx +7 -0
- package/src/primitives/spinner.tsx +27 -0
- package/src/primitives/switch.tsx +86 -0
- package/src/primitives/table.tsx +27 -0
- package/src/primitives/tabs.tsx +128 -0
- package/src/primitives/textarea.tsx +20 -0
- package/src/primitives/tooltip.tsx +85 -0
- package/src/theme/QueryProvider.tsx +46 -0
- package/src/theme/ThemeProvider.tsx +120 -0
- package/src/theme/ThemeToggle.tsx +88 -0
- package/src/theme/index.ts +3 -0
- package/src/types/react-big-calendar.d.ts +16 -0
- package/tsconfig.build.json +11 -0
- package/tsconfig.json +9 -0
- 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
|
+
}
|