@nxtode/app-shell 0.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (31) hide show
  1. package/README.md +13 -0
  2. package/dist/components/brand/nxtode-logo.d.ts +8 -0
  3. package/dist/components/brand/nxtode-logo.d.ts.map +1 -0
  4. package/dist/components/brand/nxtode-logo.js +10 -0
  5. package/dist/components/brand/nxtode-logo.js.map +1 -0
  6. package/dist/components/layout/app-switcher-menu.d.ts +2 -0
  7. package/dist/components/layout/app-switcher-menu.d.ts.map +1 -0
  8. package/dist/components/layout/app-switcher-menu.js +95 -0
  9. package/dist/components/layout/app-switcher-menu.js.map +1 -0
  10. package/dist/components/layout/notification-menu.d.ts +6 -0
  11. package/dist/components/layout/notification-menu.d.ts.map +1 -0
  12. package/dist/components/layout/notification-menu.js +205 -0
  13. package/dist/components/layout/notification-menu.js.map +1 -0
  14. package/dist/components/layout/profile-menu.d.ts +16 -0
  15. package/dist/components/layout/profile-menu.d.ts.map +1 -0
  16. package/dist/components/layout/profile-menu.js +160 -0
  17. package/dist/components/layout/profile-menu.js.map +1 -0
  18. package/dist/components/layout/site-footer.d.ts +6 -0
  19. package/dist/components/layout/site-footer.d.ts.map +1 -0
  20. package/dist/components/layout/site-footer.js +8 -0
  21. package/dist/components/layout/site-footer.js.map +1 -0
  22. package/dist/config/app.d.ts +14 -0
  23. package/dist/config/app.d.ts.map +1 -0
  24. package/dist/config/app.js +14 -0
  25. package/dist/config/app.js.map +1 -0
  26. package/dist/index.d.ts +49 -0
  27. package/dist/index.d.ts.map +1 -0
  28. package/dist/index.js +348 -0
  29. package/dist/index.js.map +1 -0
  30. package/package.json +45 -0
  31. package/styles.css +8106 -0
package/README.md ADDED
@@ -0,0 +1,13 @@
1
+ # @nxtode/app-shell
2
+
3
+ Shared NXTode application shell for top navigation, left sidebar, breadcrumbs, footer, scroll progress, and scroll-to-top behavior.
4
+
5
+ Install:
6
+
7
+ npm install @nxtode/app-shell
8
+
9
+ Basic usage:
10
+
11
+ import { AppShell } from "@nxtode/app-shell";
12
+
13
+ The shell is configured per app using appName, appDescription, accountHref, user, workspace, and navigation props.
@@ -0,0 +1,8 @@
1
+ type NxtodeLogoProps = {
2
+ variant?: "icon" | "wordmark";
3
+ className?: string;
4
+ priority?: boolean;
5
+ };
6
+ export declare function NxtodeLogo({ variant, className, priority, }: NxtodeLogoProps): import("react/jsx-runtime").JSX.Element;
7
+ export {};
8
+ //# sourceMappingURL=nxtode-logo.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"nxtode-logo.d.ts","sourceRoot":"","sources":["../../../src/components/brand/nxtode-logo.tsx"],"names":[],"mappings":"AAIA,KAAK,eAAe,GAAG;IACrB,OAAO,CAAC,EAAE,MAAM,GAAG,UAAU,CAAC;IAC9B,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,QAAQ,CAAC,EAAE,OAAO,CAAC;CACpB,CAAC;AAEF,wBAAgB,UAAU,CAAC,EACzB,OAAoB,EACpB,SAAc,EACd,QAAgB,GACjB,EAAE,eAAe,2CAuCjB"}
@@ -0,0 +1,10 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import Image from "next/image";
3
+ import { appConfig } from "../../config/app";
4
+ export function NxtodeLogo({ variant = "wordmark", className = "", priority = false, }) {
5
+ if (variant === "icon") {
6
+ return (_jsx(Image, { src: appConfig.logo.icon, alt: `${appConfig.name} icon`, width: 40, height: 40, priority: priority, className: `object-contain ${className}` }));
7
+ }
8
+ return (_jsxs("span", { className: `relative inline-block h-8 w-36 shrink-0 ${className}`, "aria-label": appConfig.name, children: [_jsx(Image, { src: appConfig.logo.wordmarkDark, alt: appConfig.name, fill: true, priority: priority, sizes: "144px", className: "nxtode-logo-dark object-contain object-left" }), _jsx(Image, { src: appConfig.logo.wordmarkLight, alt: "", fill: true, priority: priority, sizes: "144px", className: "nxtode-logo-light object-contain object-left", "aria-hidden": "true" })] }));
9
+ }
10
+ //# sourceMappingURL=nxtode-logo.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"nxtode-logo.js","sourceRoot":"","sources":["../../../src/components/brand/nxtode-logo.tsx"],"names":[],"mappings":";AAAA,OAAO,KAAK,MAAM,YAAY,CAAC;AAE/B,OAAO,EAAE,SAAS,EAAE,MAAM,kBAAkB,CAAC;AAQ7C,MAAM,UAAU,UAAU,CAAC,EACzB,OAAO,GAAG,UAAU,EACpB,SAAS,GAAG,EAAE,EACd,QAAQ,GAAG,KAAK,GACA;IAChB,IAAI,OAAO,KAAK,MAAM,EAAE,CAAC;QACvB,OAAO,CACL,KAAC,KAAK,IACJ,GAAG,EAAE,SAAS,CAAC,IAAI,CAAC,IAAI,EACxB,GAAG,EAAE,GAAG,SAAS,CAAC,IAAI,OAAO,EAC7B,KAAK,EAAE,EAAE,EACT,MAAM,EAAE,EAAE,EACV,QAAQ,EAAE,QAAQ,EAClB,SAAS,EAAE,kBAAkB,SAAS,EAAE,GACxC,CACH,CAAC;IACJ,CAAC;IAED,OAAO,CACL,gBACE,SAAS,EAAE,2CAA2C,SAAS,EAAE,gBACrD,SAAS,CAAC,IAAI,aAE1B,KAAC,KAAK,IACJ,GAAG,EAAE,SAAS,CAAC,IAAI,CAAC,YAAY,EAChC,GAAG,EAAE,SAAS,CAAC,IAAI,EACnB,IAAI,QACJ,QAAQ,EAAE,QAAQ,EAClB,KAAK,EAAC,OAAO,EACb,SAAS,EAAC,6CAA6C,GACvD,EAEF,KAAC,KAAK,IACJ,GAAG,EAAE,SAAS,CAAC,IAAI,CAAC,aAAa,EACjC,GAAG,EAAC,EAAE,EACN,IAAI,QACJ,QAAQ,EAAE,QAAQ,EAClB,KAAK,EAAC,OAAO,EACb,SAAS,EAAC,8CAA8C,iBAC5C,MAAM,GAClB,IACG,CACR,CAAC;AACJ,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare function AppSwitcherMenu(): import("react/jsx-runtime").JSX.Element;
2
+ //# sourceMappingURL=app-switcher-menu.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"app-switcher-menu.d.ts","sourceRoot":"","sources":["../../../src/components/layout/app-switcher-menu.tsx"],"names":[],"mappings":"AA+FA,wBAAgB,eAAe,4CA2F9B"}
@@ -0,0 +1,95 @@
1
+ "use client";
2
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
3
+ import Link from "next/link";
4
+ import { useEffect, useRef, useState } from "react";
5
+ const apps = [
6
+ {
7
+ name: "Account",
8
+ description: "Identity and profile",
9
+ href: "/dashboard",
10
+ initials: "A",
11
+ tone: "blue",
12
+ },
13
+ {
14
+ name: "Security",
15
+ description: "MFA and sessions",
16
+ href: "/settings?tab=security",
17
+ initials: "S",
18
+ tone: "emerald",
19
+ },
20
+ {
21
+ name: "Notifications",
22
+ description: "Alerts and history",
23
+ href: "/settings?tab=notifications",
24
+ initials: "N",
25
+ tone: "amber",
26
+ },
27
+ {
28
+ name: "Data",
29
+ description: "Export and privacy",
30
+ href: "/settings?tab=data",
31
+ initials: "D",
32
+ tone: "violet",
33
+ },
34
+ {
35
+ name: "App Settings",
36
+ description: "Product settings",
37
+ href: "/app-settings",
38
+ initials: "AS",
39
+ tone: "rose",
40
+ },
41
+ {
42
+ name: "NXTode Audit",
43
+ description: "HubSpot usage audits",
44
+ href: "https://audit.nxtode.com",
45
+ initials: "NA",
46
+ tone: "emerald",
47
+ },
48
+ {
49
+ name: "NXTClone",
50
+ description: "Sample app",
51
+ href: "/dashboard",
52
+ initials: "NC",
53
+ tone: "zinc",
54
+ },
55
+ ];
56
+ const toneClasses = {
57
+ blue: "border-blue-500/30 bg-blue-500/15 text-blue-200",
58
+ emerald: "border-emerald-500/30 bg-emerald-500/15 text-emerald-200",
59
+ amber: "border-amber-500/30 bg-amber-500/15 text-amber-200",
60
+ violet: "border-violet-500/30 bg-violet-500/15 text-violet-200",
61
+ rose: "border-rose-500/30 bg-rose-500/15 text-rose-200",
62
+ zinc: "border-zinc-700 bg-zinc-900 text-zinc-100",
63
+ };
64
+ function GridIcon() {
65
+ return (_jsxs("svg", { viewBox: "0 0 24 24", fill: "currentColor", className: "h-5 w-5", "aria-hidden": "true", children: [_jsx("circle", { cx: "5", cy: "5", r: "1.6" }), _jsx("circle", { cx: "12", cy: "5", r: "1.6" }), _jsx("circle", { cx: "19", cy: "5", r: "1.6" }), _jsx("circle", { cx: "5", cy: "12", r: "1.6" }), _jsx("circle", { cx: "12", cy: "12", r: "1.6" }), _jsx("circle", { cx: "19", cy: "12", r: "1.6" }), _jsx("circle", { cx: "5", cy: "19", r: "1.6" }), _jsx("circle", { cx: "12", cy: "19", r: "1.6" }), _jsx("circle", { cx: "19", cy: "19", r: "1.6" })] }));
66
+ }
67
+ export function AppSwitcherMenu() {
68
+ const menuRef = useRef(null);
69
+ const [isOpen, setIsOpen] = useState(false);
70
+ useEffect(() => {
71
+ if (!isOpen) {
72
+ return;
73
+ }
74
+ function handlePointerDown(event) {
75
+ if (menuRef.current &&
76
+ event.target instanceof Node &&
77
+ !menuRef.current.contains(event.target)) {
78
+ setIsOpen(false);
79
+ }
80
+ }
81
+ function handleKeyDown(event) {
82
+ if (event.key === "Escape") {
83
+ setIsOpen(false);
84
+ }
85
+ }
86
+ document.addEventListener("pointerdown", handlePointerDown);
87
+ document.addEventListener("keydown", handleKeyDown);
88
+ return () => {
89
+ document.removeEventListener("pointerdown", handlePointerDown);
90
+ document.removeEventListener("keydown", handleKeyDown);
91
+ };
92
+ }, [isOpen]);
93
+ return (_jsxs("div", { ref: menuRef, className: "relative", children: [_jsx("button", { type: "button", onClick: () => setIsOpen((current) => !current), className: "flex h-10 w-10 items-center justify-center border border-zinc-800 text-zinc-400 transition hover:bg-zinc-900 hover:text-zinc-100", "aria-label": "Open app switcher", "aria-haspopup": "menu", "aria-expanded": isOpen, children: _jsx(GridIcon, {}) }), isOpen ? (_jsxs("div", { className: "absolute right-0 top-full z-50 mt-2 w-[21rem] rounded-2xl border border-zinc-800 bg-zinc-950 p-3 shadow-2xl shadow-black/40", children: [_jsxs("div", { className: "px-2 pb-3 pt-1", children: [_jsx("p", { className: "text-sm font-medium text-zinc-100", children: "NXTode apps" }), _jsx("p", { className: "mt-0.5 text-xs text-zinc-500", children: "Open an app or workspace." })] }), _jsx("div", { className: "grid grid-cols-3 gap-2", children: apps.map((app) => (_jsxs(Link, { href: app.href, onClick: () => setIsOpen(false), className: "group flex min-h-28 flex-col items-center justify-start gap-2 px-2 py-3 text-center text-sm text-zinc-300 transition hover:bg-zinc-900 hover:text-white", title: app.description, children: [_jsx("span", { className: `flex h-12 w-12 items-center justify-center rounded-2xl border text-sm font-semibold shadow-sm ${toneClasses[app.tone]}`, children: app.initials }), _jsx("span", { className: "line-clamp-2 text-xs font-medium leading-4", children: app.name })] }, app.name))) }), _jsx("div", { className: "mt-3 border-t border-zinc-800 px-2 pt-3", children: _jsx(Link, { href: "/app-settings", onClick: () => setIsOpen(false), className: "block text-center text-xs font-medium text-zinc-400 transition hover:text-white", children: "View all apps" }) })] })) : null] }));
94
+ }
95
+ //# sourceMappingURL=app-switcher-menu.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"app-switcher-menu.js","sourceRoot":"","sources":["../../../src/components/layout/app-switcher-menu.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,6 @@
1
+ type NotificationMenuProps = {
2
+ unreadCount?: number;
3
+ };
4
+ export declare function NotificationMenu({ unreadCount: initialUnreadCount, }: NotificationMenuProps): import("react/jsx-runtime").JSX.Element;
5
+ export {};
6
+ //# sourceMappingURL=notification-menu.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"notification-menu.d.ts","sourceRoot":"","sources":["../../../src/components/layout/notification-menu.tsx"],"names":[],"mappings":"AAeA,KAAK,qBAAqB,GAAG;IAC3B,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB,CAAC;AAkGF,wBAAgB,gBAAgB,CAAC,EAC/B,WAAW,EAAE,kBAAsB,GACpC,EAAE,qBAAqB,2CAiWvB"}
@@ -0,0 +1,205 @@
1
+ "use client";
2
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
3
+ import Link from "next/link";
4
+ import { useCallback, useEffect, useMemo, useRef, useState } from "react";
5
+ const NOTIFICATIONS_UPDATED_EVENT = "nxtode:notifications-updated";
6
+ const NOTIFICATIONS_POLL_INTERVAL_MS = 30_000;
7
+ const DROPDOWN_FETCH_LIMIT = 30;
8
+ const DROPDOWN_VISIBLE_LIMIT = 5;
9
+ function getNotificationDetailHref(notificationId) {
10
+ return `/settings?tab=notifications&notificationId=${encodeURIComponent(notificationId)}`;
11
+ }
12
+ function BellIcon({ className = "h-5 w-5" }) {
13
+ return (_jsxs("svg", { viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.8", strokeLinecap: "round", strokeLinejoin: "round", className: className, "aria-hidden": "true", children: [_jsx("path", { d: "M18 8a6 6 0 1 0-12 0c0 7-3 7-3 9h18c0-2-3-2-3-9" }), _jsx("path", { d: "M10 21h4" })] }));
14
+ }
15
+ function CheckIcon({ className = "h-4 w-4" }) {
16
+ return (_jsx("svg", { viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.9", strokeLinecap: "round", strokeLinejoin: "round", className: className, "aria-hidden": "true", children: _jsx("path", { d: "m5 12 4 4L19 6" }) }));
17
+ }
18
+ function formatTime(value) {
19
+ return new Intl.DateTimeFormat("en", {
20
+ hour: "numeric",
21
+ minute: "2-digit",
22
+ }).format(new Date(value));
23
+ }
24
+ function getDayLabel(value) {
25
+ const date = new Date(value);
26
+ const today = new Date();
27
+ const yesterday = new Date();
28
+ yesterday.setDate(today.getDate() - 1);
29
+ const dateKey = date.toDateString();
30
+ if (dateKey === today.toDateString()) {
31
+ return "Today";
32
+ }
33
+ if (dateKey === yesterday.toDateString()) {
34
+ return "Yesterday";
35
+ }
36
+ return new Intl.DateTimeFormat("en", {
37
+ month: "short",
38
+ day: "numeric",
39
+ }).format(date);
40
+ }
41
+ function groupNotifications(notifications) {
42
+ return notifications.reduce((groups, notification) => {
43
+ const key = getDayLabel(notification.createdAt);
44
+ return {
45
+ ...groups,
46
+ [key]: [...(groups[key] ?? []), notification],
47
+ };
48
+ }, {});
49
+ }
50
+ export function NotificationMenu({ unreadCount: initialUnreadCount = 0, }) {
51
+ const menuRef = useRef(null);
52
+ const [isOpen, setIsOpen] = useState(false);
53
+ const [activeFilter, setActiveFilter] = useState("unread");
54
+ const [unreadCount, setUnreadCount] = useState(initialUnreadCount);
55
+ const [notifications, setNotifications] = useState([]);
56
+ const [isLoading, setIsLoading] = useState(false);
57
+ const [error, setError] = useState("");
58
+ const badgeLabel = unreadCount > 99 ? "99+" : String(unreadCount);
59
+ const readCount = notifications.filter((item) => item.readAt).length;
60
+ const filteredNotifications = useMemo(() => {
61
+ if (activeFilter === "unread") {
62
+ return notifications.filter((item) => !item.readAt);
63
+ }
64
+ if (activeFilter === "read") {
65
+ return notifications.filter((item) => item.readAt);
66
+ }
67
+ return notifications;
68
+ }, [activeFilter, notifications]);
69
+ const visibleNotifications = useMemo(() => filteredNotifications.slice(0, DROPDOWN_VISIBLE_LIMIT), [filteredNotifications]);
70
+ const hasMoreFilteredNotifications = filteredNotifications.length > visibleNotifications.length;
71
+ const groupedNotifications = useMemo(() => groupNotifications(visibleNotifications), [visibleNotifications]);
72
+ const loadNotifications = useCallback(async () => {
73
+ setError("");
74
+ try {
75
+ const response = await fetch(`/api/notifications?limit=${DROPDOWN_FETCH_LIMIT}`, {
76
+ method: "GET",
77
+ credentials: "same-origin",
78
+ cache: "no-store",
79
+ });
80
+ if (!response.ok) {
81
+ return;
82
+ }
83
+ const data = await response.json();
84
+ setUnreadCount(data.unreadCount ?? 0);
85
+ setNotifications(data.notifications ?? []);
86
+ }
87
+ catch {
88
+ setError("Could not load notifications.");
89
+ }
90
+ }, []);
91
+ useEffect(() => {
92
+ let isActive = true;
93
+ async function refreshUnreadCount() {
94
+ if (!isActive) {
95
+ return;
96
+ }
97
+ await loadNotifications();
98
+ }
99
+ void refreshUnreadCount();
100
+ const interval = window.setInterval(() => {
101
+ void refreshUnreadCount();
102
+ }, NOTIFICATIONS_POLL_INTERVAL_MS);
103
+ function handleWindowFocus() {
104
+ void refreshUnreadCount();
105
+ }
106
+ function handleVisibilityChange() {
107
+ if (document.visibilityState === "visible") {
108
+ void refreshUnreadCount();
109
+ }
110
+ }
111
+ function handleNotificationsUpdated() {
112
+ void refreshUnreadCount();
113
+ }
114
+ window.addEventListener("focus", handleWindowFocus);
115
+ document.addEventListener("visibilitychange", handleVisibilityChange);
116
+ window.addEventListener(NOTIFICATIONS_UPDATED_EVENT, handleNotificationsUpdated);
117
+ return () => {
118
+ isActive = false;
119
+ window.clearInterval(interval);
120
+ window.removeEventListener("focus", handleWindowFocus);
121
+ document.removeEventListener("visibilitychange", handleVisibilityChange);
122
+ window.removeEventListener(NOTIFICATIONS_UPDATED_EVENT, handleNotificationsUpdated);
123
+ };
124
+ }, [loadNotifications]);
125
+ useEffect(() => {
126
+ if (!isOpen) {
127
+ return;
128
+ }
129
+ function handlePointerDown(event) {
130
+ if (menuRef.current &&
131
+ event.target instanceof Node &&
132
+ !menuRef.current.contains(event.target)) {
133
+ setIsOpen(false);
134
+ }
135
+ }
136
+ function handleKeyDown(event) {
137
+ if (event.key === "Escape") {
138
+ setIsOpen(false);
139
+ }
140
+ }
141
+ document.addEventListener("pointerdown", handlePointerDown);
142
+ document.addEventListener("keydown", handleKeyDown);
143
+ return () => {
144
+ document.removeEventListener("pointerdown", handlePointerDown);
145
+ document.removeEventListener("keydown", handleKeyDown);
146
+ };
147
+ }, [isOpen]);
148
+ async function openMenu() {
149
+ setIsOpen((current) => !current);
150
+ if (!isOpen) {
151
+ setIsLoading(true);
152
+ await loadNotifications();
153
+ setIsLoading(false);
154
+ }
155
+ }
156
+ async function markAllRead() {
157
+ try {
158
+ await fetch("/api/notifications/read", {
159
+ method: "POST",
160
+ headers: {
161
+ "Content-Type": "application/json",
162
+ },
163
+ credentials: "same-origin",
164
+ body: JSON.stringify({
165
+ all: true,
166
+ }),
167
+ });
168
+ window.dispatchEvent(new Event(NOTIFICATIONS_UPDATED_EVENT));
169
+ await loadNotifications();
170
+ }
171
+ catch {
172
+ setError("Could not mark notifications as read.");
173
+ }
174
+ }
175
+ async function markOneRead(notificationId) {
176
+ try {
177
+ await fetch("/api/notifications/read", {
178
+ method: "POST",
179
+ headers: {
180
+ "Content-Type": "application/json",
181
+ },
182
+ credentials: "same-origin",
183
+ body: JSON.stringify({
184
+ notificationId,
185
+ }),
186
+ });
187
+ window.dispatchEvent(new Event(NOTIFICATIONS_UPDATED_EVENT));
188
+ await loadNotifications();
189
+ }
190
+ catch {
191
+ setError("Could not update notification.");
192
+ }
193
+ }
194
+ return (_jsxs("div", { ref: menuRef, className: "relative", children: [_jsxs("button", { type: "button", onClick: openMenu, className: "notification-trigger-button relative hidden h-10 w-10 items-center justify-center text-zinc-400 transition hover:text-zinc-100 sm:inline-flex", "aria-label": "Open notifications", "aria-haspopup": "menu", "aria-expanded": isOpen, children: [_jsx(BellIcon, {}), unreadCount > 0 ? (_jsx("span", { className: "notification-trigger-badge", children: badgeLabel })) : null] }), isOpen ? (_jsx("div", { className: "notification-popover", children: _jsxs("div", { className: "notification-popover-card", children: [_jsxs("div", { className: "notification-popover-header", children: [_jsxs("div", { children: [_jsx("h3", { className: "notification-popover-title", children: "Notifications" }), _jsx("p", { className: "notification-popover-subtitle", children: "Security and account updates" })] }), unreadCount > 0 ? (_jsxs("button", { type: "button", onClick: markAllRead, className: "notification-ghost-button", children: [_jsx(CheckIcon, {}), "Mark all"] })) : null] }), _jsxs("div", { className: "notification-tabs", role: "tablist", children: [_jsxs("button", { type: "button", onClick: () => setActiveFilter("unread"), className: `notification-tab ${activeFilter === "unread" ? "is-active" : ""}`, children: ["Unread", _jsx("span", { children: unreadCount })] }), _jsxs("button", { type: "button", onClick: () => setActiveFilter("read"), className: `notification-tab ${activeFilter === "read" ? "is-active" : ""}`, children: ["Read", _jsx("span", { children: readCount })] }), _jsxs("button", { type: "button", onClick: () => setActiveFilter("all"), className: `notification-tab ${activeFilter === "all" ? "is-active" : ""}`, children: ["All", _jsx("span", { children: notifications.length })] })] }), error ? _jsx("div", { className: "notification-error", children: error }) : null, _jsxs("div", { className: "notification-popover-list", children: [isLoading ? (_jsx("div", { className: "notification-empty-state", children: "Loading notifications..." })) : filteredNotifications.length === 0 ? (_jsxs("div", { className: "notification-empty-state", children: [_jsx("p", { children: "No notifications" }), _jsx("span", { children: activeFilter === "unread"
195
+ ? "You are caught up."
196
+ : "Security alerts and account updates will appear here." })] })) : (_jsx("div", { className: "notification-group-stack", children: Object.entries(groupedNotifications).map(([groupLabel, groupItems]) => (_jsxs("section", { className: "notification-group", children: [_jsxs("div", { className: "notification-group-header", children: [_jsx("span", { children: groupLabel }), _jsx("span", { children: groupItems.length })] }), _jsx("div", { className: "notification-card-stack", children: groupItems.map((notification) => {
197
+ const href = getNotificationDetailHref(notification.id);
198
+ const isUnread = !notification.readAt;
199
+ return (_jsxs(Link, { href: href, onClick: () => {
200
+ void markOneRead(notification.id);
201
+ setIsOpen(false);
202
+ }, className: `notification-card ${isUnread ? "is-unread" : ""}`, children: [_jsx("span", { className: "notification-status-dot" }), _jsxs("div", { className: "min-w-0 flex-1", children: [_jsxs("div", { className: "notification-card-title-row", children: [_jsx("p", { children: notification.title }), _jsx("span", { children: formatTime(notification.createdAt) })] }), _jsx("p", { className: "notification-card-message", children: notification.message })] })] }, notification.id));
203
+ }) })] }, groupLabel))) })), !isLoading && hasMoreFilteredNotifications ? (_jsxs("p", { className: "notification-popover-limit-note", children: ["Showing latest ", visibleNotifications.length, " of", " ", filteredNotifications.length, ". View all for full history."] })) : null] }), _jsx("div", { className: "notification-popover-footer", children: _jsx(Link, { href: "/settings?tab=notifications", onClick: () => setIsOpen(false), children: "View all" }) })] }) })) : null] }));
204
+ }
205
+ //# sourceMappingURL=notification-menu.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"notification-menu.js","sourceRoot":"","sources":["../../../src/components/layout/notification-menu.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,16 @@
1
+ type ProfileMenuUser = {
2
+ firstName: string;
3
+ lastName: string;
4
+ email: string;
5
+ imageUrl?: string | null;
6
+ originalImageUrl?: string | null;
7
+ imagePositionX?: number;
8
+ imagePositionY?: number;
9
+ imageScale?: number;
10
+ };
11
+ type ProfileMenuProps = {
12
+ user: ProfileMenuUser;
13
+ };
14
+ export declare function ProfileMenu({ user }: ProfileMenuProps): import("react/jsx-runtime").JSX.Element;
15
+ export {};
16
+ //# sourceMappingURL=profile-menu.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"profile-menu.d.ts","sourceRoot":"","sources":["../../../src/components/layout/profile-menu.tsx"],"names":[],"mappings":"AAMA,KAAK,eAAe,GAAG;IACrB,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,MAAM,CAAC;IACjB,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACzB,gBAAgB,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IACjC,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB,CAAC;AAEF,KAAK,gBAAgB,GAAG;IACtB,IAAI,EAAE,eAAe,CAAC;CACvB,CAAC;AAgMF,wBAAgB,WAAW,CAAC,EAAE,IAAI,EAAE,EAAE,gBAAgB,2CA+GrD"}
@@ -0,0 +1,160 @@
1
+ "use client";
2
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
3
+ import Link from "next/link";
4
+ import { useRouter } from "next/navigation";
5
+ import { useEffect, useRef, useState } from "react";
6
+ const CSRF_COOKIE_NAME = "nxtclone_csrf";
7
+ const CSRF_HEADER_NAME = "x-csrf-token";
8
+ function getCookieValue(name) {
9
+ const cookies = document.cookie.split(";").map((cookie) => cookie.trim());
10
+ for (const cookie of cookies) {
11
+ if (cookie.startsWith(`${name}=`)) {
12
+ return decodeURIComponent(cookie.slice(name.length + 1));
13
+ }
14
+ }
15
+ return "";
16
+ }
17
+ async function getCsrfToken() {
18
+ const existingToken = getCookieValue(CSRF_COOKIE_NAME);
19
+ if (existingToken) {
20
+ return existingToken;
21
+ }
22
+ const response = await fetch("/api/auth/csrf", {
23
+ method: "GET",
24
+ credentials: "same-origin",
25
+ cache: "no-store",
26
+ });
27
+ if (!response.ok) {
28
+ throw new Error("Could not prepare security token.");
29
+ }
30
+ const data = await response.json();
31
+ return data.csrfToken;
32
+ }
33
+ function isUnsafeMethod(method) {
34
+ return !["GET", "HEAD", "OPTIONS"].includes(method.toUpperCase());
35
+ }
36
+ async function csrfFetch(input, init = {}) {
37
+ const method = init.method ?? "GET";
38
+ const headers = new Headers(init.headers);
39
+ if (isUnsafeMethod(method)) {
40
+ const csrfToken = await getCsrfToken();
41
+ headers.set(CSRF_HEADER_NAME, csrfToken);
42
+ }
43
+ return fetch(input, {
44
+ ...init,
45
+ credentials: init.credentials ?? "same-origin",
46
+ headers,
47
+ });
48
+ }
49
+ const accountLinks = [
50
+ {
51
+ label: "Account settings",
52
+ href: "/settings?tab=account",
53
+ icon: UserIcon,
54
+ },
55
+ {
56
+ label: "Security",
57
+ href: "/settings?tab=security",
58
+ icon: ShieldIcon,
59
+ },
60
+ {
61
+ label: "App settings",
62
+ href: "/app-settings",
63
+ icon: SettingsIcon,
64
+ },
65
+ {
66
+ label: "Sessions",
67
+ href: "/settings?tab=sessions",
68
+ icon: MonitorIcon,
69
+ },
70
+ {
71
+ label: "Notifications",
72
+ href: "/settings?tab=notifications",
73
+ icon: BellIcon,
74
+ },
75
+ {
76
+ label: "Data & Privacy",
77
+ href: "/settings?tab=data",
78
+ icon: DatabaseIcon,
79
+ },
80
+ ];
81
+ function getInitials(user) {
82
+ const first = user.firstName?.[0] ?? "";
83
+ const last = user.lastName?.[0] ?? "";
84
+ return `${first}${last}`.toUpperCase() || "U";
85
+ }
86
+ function Avatar({ user, className }) {
87
+ const initials = getInitials(user);
88
+ return (_jsx("span", { className: `relative flex shrink-0 items-center justify-center overflow-hidden rounded-full border border-zinc-800 bg-zinc-900 font-semibold text-zinc-100 ${className}`, children: user.imageUrl ? (
89
+ // eslint-disable-next-line @next/next/no-img-element
90
+ _jsx("img", { src: user.imageUrl, alt: "", className: "h-full w-full object-cover" })) : (initials) }));
91
+ }
92
+ function UserIcon({ className = "h-4 w-4" }) {
93
+ return (_jsxs("svg", { viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.8", strokeLinecap: "round", strokeLinejoin: "round", className: className, "aria-hidden": "true", children: [_jsx("path", { d: "M20 21a8 8 0 0 0-16 0" }), _jsx("path", { d: "M12 13a5 5 0 1 0 0-10 5 5 0 0 0 0 10Z" })] }));
94
+ }
95
+ function ShieldIcon({ className = "h-4 w-4" }) {
96
+ return (_jsxs("svg", { viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.8", strokeLinecap: "round", strokeLinejoin: "round", className: className, "aria-hidden": "true", children: [_jsx("path", { d: "M12 22s8-4 8-10V5l-8-3-8 3v7c0 6 8 10 8 10Z" }), _jsx("path", { d: "m9.5 12 1.7 1.7L15 10" })] }));
97
+ }
98
+ function MonitorIcon({ className = "h-4 w-4" }) {
99
+ return (_jsxs("svg", { viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.8", strokeLinecap: "round", strokeLinejoin: "round", className: className, "aria-hidden": "true", children: [_jsx("rect", { x: "3", y: "4", width: "18", height: "12", rx: "2" }), _jsx("path", { d: "M8 20h8" }), _jsx("path", { d: "M12 16v4" })] }));
100
+ }
101
+ function BellIcon({ className = "h-4 w-4" }) {
102
+ return (_jsxs("svg", { viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.8", strokeLinecap: "round", strokeLinejoin: "round", className: className, "aria-hidden": "true", children: [_jsx("path", { d: "M18 8a6 6 0 1 0-12 0c0 7-3 7-3 9h18c0-2-3-2-3-9" }), _jsx("path", { d: "M10 21h4" })] }));
103
+ }
104
+ function DatabaseIcon({ className = "h-4 w-4" }) {
105
+ return (_jsxs("svg", { viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.8", strokeLinecap: "round", strokeLinejoin: "round", className: className, "aria-hidden": "true", children: [_jsx("ellipse", { cx: "12", cy: "5", rx: "8", ry: "3" }), _jsx("path", { d: "M4 5v6c0 1.66 3.58 3 8 3s8-1.34 8-3V5" }), _jsx("path", { d: "M4 11v6c0 1.66 3.58 3 8 3s8-1.34 8-3v-6" })] }));
106
+ }
107
+ function SettingsIcon({ className = "h-4 w-4" }) {
108
+ return (_jsxs("svg", { viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.8", strokeLinecap: "round", strokeLinejoin: "round", className: className, "aria-hidden": "true", children: [_jsx("path", { d: "M12 15.5A3.5 3.5 0 1 0 12 8a3.5 3.5 0 0 0 0 7.5Z" }), _jsx("path", { d: "M19.4 15a1.8 1.8 0 0 0 .36 2l.04.04a2.1 2.1 0 0 1-2.97 2.97l-.04-.04a1.8 1.8 0 0 0-2-.36 1.8 1.8 0 0 0-1.09 1.65V21a2.1 2.1 0 0 1-4.2 0v-.06a1.8 1.8 0 0 0-1.09-1.65 1.8 1.8 0 0 0-2 .36l-.04.04a2.1 2.1 0 0 1-2.97-2.97l.04-.04a1.8 1.8 0 0 0 .36-2 1.8 1.8 0 0 0-1.65-1.09H2a2.1 2.1 0 0 1 0-4.2h.06a1.8 1.8 0 0 0 1.65-1.09 1.8 1.8 0 0 0-.36-2l-.04-.04a2.1 2.1 0 0 1 2.97-2.97l.04.04a1.8 1.8 0 0 0 2 .36 1.8 1.8 0 0 0 1.09-1.65V2a2.1 2.1 0 0 1 4.2 0v.06a1.8 1.8 0 0 0 1.09 1.65 1.8 1.8 0 0 0 2-.36l.04-.04a2.1 2.1 0 0 1 2.97 2.97l-.04.04a1.8 1.8 0 0 0-.36 2 1.8 1.8 0 0 0 1.65 1.09H22a2.1 2.1 0 0 1 0 4.2h-.06A1.8 1.8 0 0 0 19.4 15Z" })] }));
109
+ }
110
+ function LogoutIcon({ className = "h-4 w-4" }) {
111
+ return (_jsxs("svg", { viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "1.8", strokeLinecap: "round", strokeLinejoin: "round", className: className, "aria-hidden": "true", children: [_jsx("path", { d: "M10 17 15 12l-5-5" }), _jsx("path", { d: "M15 12H3" }), _jsx("path", { d: "M21 3v18" })] }));
112
+ }
113
+ export function ProfileMenu({ user }) {
114
+ const router = useRouter();
115
+ const menuRef = useRef(null);
116
+ const [isOpen, setIsOpen] = useState(false);
117
+ const [isLoggingOut, setIsLoggingOut] = useState(false);
118
+ useEffect(() => {
119
+ if (!isOpen) {
120
+ return;
121
+ }
122
+ function handlePointerDown(event) {
123
+ if (menuRef.current &&
124
+ event.target instanceof Node &&
125
+ !menuRef.current.contains(event.target)) {
126
+ setIsOpen(false);
127
+ }
128
+ }
129
+ function handleKeyDown(event) {
130
+ if (event.key === "Escape") {
131
+ setIsOpen(false);
132
+ }
133
+ }
134
+ document.addEventListener("pointerdown", handlePointerDown);
135
+ document.addEventListener("keydown", handleKeyDown);
136
+ return () => {
137
+ document.removeEventListener("pointerdown", handlePointerDown);
138
+ document.removeEventListener("keydown", handleKeyDown);
139
+ };
140
+ }, [isOpen]);
141
+ async function handleLogout() {
142
+ setIsLoggingOut(true);
143
+ try {
144
+ await csrfFetch("/api/auth/logout", {
145
+ method: "POST",
146
+ credentials: "same-origin",
147
+ });
148
+ router.push("/login");
149
+ router.refresh();
150
+ }
151
+ finally {
152
+ setIsLoggingOut(false);
153
+ }
154
+ }
155
+ return (_jsxs("div", { ref: menuRef, className: "relative", children: [_jsx("button", { type: "button", onClick: () => setIsOpen((current) => !current), className: "app-profile-menu-button flex h-10 w-10 items-center justify-center rounded-full transition hover:ring-2 hover:ring-zinc-700", "aria-label": "Open account menu", "aria-haspopup": "menu", "aria-expanded": isOpen, children: _jsx(Avatar, { user: user, className: "h-10 w-10 text-xs" }) }), isOpen ? (_jsxs("div", { className: "absolute right-0 top-full z-50 mt-2 w-72 rounded-2xl border border-zinc-800 bg-zinc-950 p-2 shadow-2xl shadow-black/40", children: [_jsxs("div", { className: "flex items-center gap-3 border-b border-zinc-800 px-3 py-3", children: [_jsx(Avatar, { user: user, className: "h-11 w-11 text-sm" }), _jsxs("div", { className: "min-w-0", children: [_jsxs("p", { className: "truncate text-sm font-medium text-zinc-100", children: [user.firstName, " ", user.lastName] }), _jsx("p", { className: "truncate text-xs text-zinc-500", children: user.email })] })] }), _jsx("nav", { className: "py-2", "aria-label": "Account menu", children: accountLinks.map((item) => {
156
+ const Icon = item.icon;
157
+ return (_jsxs(Link, { href: item.href, onClick: () => setIsOpen(false), className: "flex h-10 items-center gap-3 rounded-xl px-3 text-sm text-zinc-300 transition hover:bg-zinc-900 hover:text-white", children: [_jsx(Icon, { className: "h-4 w-4 text-zinc-500" }), _jsx("span", { className: "truncate", children: item.label })] }, item.href));
158
+ }) }), _jsx("div", { className: "border-t border-zinc-800 pt-2", children: _jsxs("button", { type: "button", onClick: handleLogout, disabled: isLoggingOut, className: "flex h-10 w-full items-center gap-3 rounded-xl px-3 text-left text-sm text-red-300 transition hover:bg-red-950/30 hover:text-red-200 disabled:cursor-not-allowed disabled:opacity-60", children: [_jsx(LogoutIcon, { className: "h-4 w-4" }), isLoggingOut ? "Logging out..." : "Log out"] }) })] })) : null] }));
159
+ }
160
+ //# sourceMappingURL=profile-menu.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"profile-menu.js","sourceRoot":"","sources":["../../../src/components/layout/profile-menu.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,6 @@
1
+ type SiteFooterProps = {
2
+ compact?: boolean;
3
+ };
4
+ export declare function SiteFooter({ compact }: SiteFooterProps): import("react/jsx-runtime").JSX.Element;
5
+ export {};
6
+ //# sourceMappingURL=site-footer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"site-footer.d.ts","sourceRoot":"","sources":["../../../src/components/layout/site-footer.tsx"],"names":[],"mappings":"AAMA,KAAK,eAAe,GAAG;IACrB,OAAO,CAAC,EAAE,OAAO,CAAC;CACnB,CAAC;AAEF,wBAAgB,UAAU,CAAC,EAAE,OAAe,EAAE,EAAE,eAAe,2CAsD9D"}