@replicated/portal-components 0.0.2 → 0.0.3
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/components/metadata/registry.json +83 -2
- package/components/metadata/registry.md +27 -2
- package/dist/actions/index.d.mts +566 -3
- package/dist/actions/index.d.ts +566 -3
- package/dist/actions/index.js +1853 -12
- package/dist/actions/index.js.map +1 -1
- package/dist/airgap-instances.d.mts +26 -0
- package/dist/airgap-instances.d.ts +26 -0
- package/dist/airgap-instances.js +354 -0
- package/dist/airgap-instances.js.map +1 -0
- package/dist/error-page.d.mts +14 -0
- package/dist/error-page.d.ts +14 -0
- package/dist/error-page.js +153 -0
- package/dist/error-page.js.map +1 -0
- package/dist/error.d.mts +15 -0
- package/dist/error.d.ts +15 -0
- package/dist/error.js +144 -0
- package/dist/error.js.map +1 -0
- package/dist/esm/actions/index.js +1816 -13
- package/dist/esm/actions/index.js.map +1 -1
- package/dist/esm/airgap-instances.js +352 -0
- package/dist/esm/airgap-instances.js.map +1 -0
- package/dist/esm/error-page.js +151 -0
- package/dist/esm/error-page.js.map +1 -0
- package/dist/esm/error.js +142 -0
- package/dist/esm/error.js.map +1 -0
- package/dist/esm/helm-install-wizard.js +1007 -0
- package/dist/esm/helm-install-wizard.js.map +1 -0
- package/dist/esm/index.js +2232 -155
- package/dist/esm/index.js.map +1 -1
- package/dist/esm/install-actions.js +746 -0
- package/dist/esm/install-actions.js.map +1 -0
- package/dist/esm/install-card.js +115 -0
- package/dist/esm/install-card.js.map +1 -0
- package/dist/esm/install-targets.js +48 -0
- package/dist/esm/install-targets.js.map +1 -0
- package/dist/esm/instance-card.js +197 -0
- package/dist/esm/instance-card.js.map +1 -0
- package/dist/esm/join-team.js +218 -0
- package/dist/esm/join-team.js.map +1 -0
- package/dist/esm/license-card.js +131 -0
- package/dist/esm/license-card.js.map +1 -0
- package/dist/esm/license-details.js +667 -0
- package/dist/esm/license-details.js.map +1 -0
- package/dist/esm/linux-install-wizard.js +1083 -0
- package/dist/esm/linux-install-wizard.js.map +1 -0
- package/dist/esm/login.js +261 -0
- package/dist/esm/login.js.map +1 -0
- package/dist/esm/online-instance-list.js +287 -0
- package/dist/esm/online-instance-list.js.map +1 -0
- package/dist/esm/pending-installations.js +235 -0
- package/dist/esm/pending-installations.js.map +1 -0
- package/dist/esm/release-history-panel.js +100 -0
- package/dist/esm/release-history-panel.js.map +1 -0
- package/dist/esm/release-notes-card.js +23 -0
- package/dist/esm/release-notes-card.js.map +1 -0
- package/dist/esm/security-card.js +700 -0
- package/dist/esm/security-card.js.map +1 -0
- package/dist/esm/support-bundle-collection-card.js +170 -0
- package/dist/esm/support-bundle-collection-card.js.map +1 -0
- package/dist/esm/support-bundles-card.js +306 -0
- package/dist/esm/support-bundles-card.js.map +1 -0
- package/dist/esm/support-card.js +305 -0
- package/dist/esm/support-card.js.map +1 -0
- package/dist/esm/team-selection.js +117 -0
- package/dist/esm/team-selection.js.map +1 -0
- package/dist/esm/team-settings-card.js +78 -0
- package/dist/esm/team-settings-card.js.map +1 -0
- package/dist/esm/team-settings.js +136 -0
- package/dist/esm/team-settings.js.map +1 -0
- package/dist/esm/top-nav-user-menu.js +173 -0
- package/dist/esm/top-nav-user-menu.js.map +1 -0
- package/dist/esm/top-nav.js +398 -0
- package/dist/esm/top-nav.js.map +1 -0
- package/dist/esm/update-layout.js +405 -0
- package/dist/esm/update-layout.js.map +1 -0
- package/dist/esm/updates-card.js +85 -0
- package/dist/esm/updates-card.js.map +1 -0
- package/dist/esm/upload-support-bundle-modal.js +143 -0
- package/dist/esm/upload-support-bundle-modal.js.map +1 -0
- package/dist/esm/user-settings-card.js +21 -0
- package/dist/esm/user-settings-card.js.map +1 -0
- package/dist/esm/user-settings.js +368 -0
- package/dist/esm/user-settings.js.map +1 -0
- package/dist/esm/utils/index.js +170 -0
- package/dist/esm/utils/index.js.map +1 -0
- package/dist/helm-install-wizard.d.mts +38 -0
- package/dist/helm-install-wizard.d.ts +38 -0
- package/dist/helm-install-wizard.js +1011 -0
- package/dist/helm-install-wizard.js.map +1 -0
- package/dist/index.d.mts +11 -27
- package/dist/index.d.ts +11 -27
- package/dist/index.js +2258 -154
- package/dist/index.js.map +1 -1
- package/dist/install-B19AaKF_.d.mts +233 -0
- package/dist/install-Bi1qJ8Bu.d.ts +233 -0
- package/dist/install-actions.d.mts +141 -0
- package/dist/install-actions.d.ts +141 -0
- package/dist/install-actions.js +765 -0
- package/dist/install-actions.js.map +1 -0
- package/dist/install-card.d.mts +15 -0
- package/dist/install-card.d.ts +15 -0
- package/dist/install-card.js +117 -0
- package/dist/install-card.js.map +1 -0
- package/dist/install-targets.d.mts +19 -0
- package/dist/install-targets.d.ts +19 -0
- package/dist/install-targets.js +50 -0
- package/dist/install-targets.js.map +1 -0
- package/dist/instance-card.d.mts +22 -0
- package/dist/instance-card.d.ts +22 -0
- package/dist/instance-card.js +199 -0
- package/dist/instance-card.js.map +1 -0
- package/dist/join-team.d.mts +30 -0
- package/dist/join-team.d.ts +30 -0
- package/dist/join-team.js +220 -0
- package/dist/join-team.js.map +1 -0
- package/dist/license-card.d.mts +15 -0
- package/dist/license-card.d.ts +15 -0
- package/dist/license-card.js +133 -0
- package/dist/license-card.js.map +1 -0
- package/dist/license-details.d.mts +10 -0
- package/dist/license-details.d.ts +10 -0
- package/dist/license-details.js +669 -0
- package/dist/license-details.js.map +1 -0
- package/dist/linux-install-wizard.d.mts +66 -0
- package/dist/linux-install-wizard.d.ts +66 -0
- package/dist/linux-install-wizard.js +1093 -0
- package/dist/linux-install-wizard.js.map +1 -0
- package/dist/login.d.mts +37 -0
- package/dist/login.d.ts +37 -0
- package/dist/login.js +263 -0
- package/dist/login.js.map +1 -0
- package/dist/online-instance-list.d.mts +22 -0
- package/dist/online-instance-list.d.ts +22 -0
- package/dist/online-instance-list.js +289 -0
- package/dist/online-instance-list.js.map +1 -0
- package/dist/pending-installations.d.mts +15 -0
- package/dist/pending-installations.d.ts +15 -0
- package/dist/pending-installations.js +237 -0
- package/dist/pending-installations.js.map +1 -0
- package/dist/release-history-panel.d.mts +22 -0
- package/dist/release-history-panel.d.ts +22 -0
- package/dist/release-history-panel.js +102 -0
- package/dist/release-history-panel.js.map +1 -0
- package/dist/release-notes-card.d.mts +13 -0
- package/dist/release-notes-card.d.ts +13 -0
- package/dist/release-notes-card.js +25 -0
- package/dist/release-notes-card.js.map +1 -0
- package/dist/security-card.d.mts +73 -0
- package/dist/security-card.d.ts +73 -0
- package/dist/security-card.js +702 -0
- package/dist/security-card.js.map +1 -0
- package/dist/styles.css +1877 -194
- package/dist/support-bundle-collection-card.d.mts +20 -0
- package/dist/support-bundle-collection-card.d.ts +20 -0
- package/dist/support-bundle-collection-card.js +172 -0
- package/dist/support-bundle-collection-card.js.map +1 -0
- package/dist/support-bundles-card.d.mts +19 -0
- package/dist/support-bundles-card.d.ts +19 -0
- package/dist/support-bundles-card.js +308 -0
- package/dist/support-bundles-card.js.map +1 -0
- package/dist/support-card.d.mts +8 -0
- package/dist/support-card.d.ts +8 -0
- package/dist/support-card.js +307 -0
- package/dist/support-card.js.map +1 -0
- package/dist/team-selection.d.mts +23 -0
- package/dist/team-selection.d.ts +23 -0
- package/dist/team-selection.js +119 -0
- package/dist/team-selection.js.map +1 -0
- package/dist/team-settings-card-Dq1d9b5c.d.mts +14 -0
- package/dist/team-settings-card-Dq1d9b5c.d.ts +14 -0
- package/dist/team-settings-card.d.mts +2 -0
- package/dist/team-settings-card.d.ts +2 -0
- package/dist/team-settings-card.js +80 -0
- package/dist/team-settings-card.js.map +1 -0
- package/dist/team-settings.d.mts +25 -0
- package/dist/team-settings.d.ts +25 -0
- package/dist/team-settings.js +138 -0
- package/dist/team-settings.js.map +1 -0
- package/dist/top-nav-0mb1K_H0.d.mts +32 -0
- package/dist/top-nav-0mb1K_H0.d.ts +32 -0
- package/dist/top-nav-user-menu.d.mts +18 -0
- package/dist/top-nav-user-menu.d.ts +18 -0
- package/dist/top-nav-user-menu.js +175 -0
- package/dist/top-nav-user-menu.js.map +1 -0
- package/dist/top-nav.d.mts +3 -0
- package/dist/top-nav.d.ts +3 -0
- package/dist/top-nav.js +400 -0
- package/dist/top-nav.js.map +1 -0
- package/dist/update-layout.d.mts +12 -0
- package/dist/update-layout.d.ts +12 -0
- package/dist/update-layout.js +407 -0
- package/dist/update-layout.js.map +1 -0
- package/dist/updates-card-BbubBrVR.d.mts +18 -0
- package/dist/updates-card-BbubBrVR.d.ts +18 -0
- package/dist/updates-card.d.mts +2 -0
- package/dist/updates-card.d.ts +2 -0
- package/dist/updates-card.js +87 -0
- package/dist/updates-card.js.map +1 -0
- package/dist/upload-support-bundle-modal.d.mts +19 -0
- package/dist/upload-support-bundle-modal.d.ts +19 -0
- package/dist/upload-support-bundle-modal.js +145 -0
- package/dist/upload-support-bundle-modal.js.map +1 -0
- package/dist/user-settings-card.d.mts +8 -0
- package/dist/user-settings-card.d.ts +8 -0
- package/dist/user-settings-card.js +23 -0
- package/dist/user-settings-card.js.map +1 -0
- package/dist/user-settings.d.mts +47 -0
- package/dist/user-settings.d.ts +47 -0
- package/dist/user-settings.js +370 -0
- package/dist/user-settings.js.map +1 -0
- package/dist/utils/index.d.mts +70 -0
- package/dist/utils/index.d.ts +70 -0
- package/dist/utils/index.js +177 -0
- package/dist/utils/index.js.map +1 -0
- package/package.json +163 -3
|
@@ -0,0 +1,370 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
'use strict';
|
|
3
|
+
|
|
4
|
+
var react = require('react');
|
|
5
|
+
var jsxRuntime = require('react/jsx-runtime');
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Enterprise Portal Components
|
|
9
|
+
* This file is generated by tsup. Do not edit manually.
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
var NOTIFICATION_DESCRIPTIONS = {
|
|
13
|
+
"new-version": "New version available"
|
|
14
|
+
};
|
|
15
|
+
var EmailNotificationsSection = ({
|
|
16
|
+
teamName,
|
|
17
|
+
notifications,
|
|
18
|
+
isLoading,
|
|
19
|
+
onToggle
|
|
20
|
+
}) => {
|
|
21
|
+
const allNotificationTypes = Object.keys(NOTIFICATION_DESCRIPTIONS);
|
|
22
|
+
const mergedNotifications = allNotificationTypes.map((type) => {
|
|
23
|
+
const existing = notifications.find((n) => n.type === type);
|
|
24
|
+
return {
|
|
25
|
+
type,
|
|
26
|
+
enabled: existing ? existing.enabled : type === "new-version"
|
|
27
|
+
// Default new-version to true
|
|
28
|
+
};
|
|
29
|
+
});
|
|
30
|
+
if (isLoading) {
|
|
31
|
+
return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "rounded-2xl border border-gray-200 bg-white p-6 shadow-sm", children: [
|
|
32
|
+
/* @__PURE__ */ jsxRuntime.jsxs("h2", { className: "text-lg font-semibold text-gray-900", children: [
|
|
33
|
+
"Email Notifications for ",
|
|
34
|
+
teamName
|
|
35
|
+
] }),
|
|
36
|
+
/* @__PURE__ */ jsxRuntime.jsx("p", { className: "mt-1 text-sm text-gray-500", children: "Choose which email notifications you want to receive for this team." }),
|
|
37
|
+
/* @__PURE__ */ jsxRuntime.jsx("div", { className: "mt-6 flex justify-center py-8", children: /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-sm text-gray-500", children: "Loading notifications..." }) })
|
|
38
|
+
] });
|
|
39
|
+
}
|
|
40
|
+
return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "rounded-2xl border border-gray-200 bg-white p-6 shadow-sm", children: [
|
|
41
|
+
/* @__PURE__ */ jsxRuntime.jsxs("h2", { className: "text-lg font-semibold text-gray-900", children: [
|
|
42
|
+
"Email Notifications for ",
|
|
43
|
+
teamName
|
|
44
|
+
] }),
|
|
45
|
+
/* @__PURE__ */ jsxRuntime.jsx("p", { className: "mt-1 text-sm text-gray-500", children: "Choose which email notifications you want to receive for this team." }),
|
|
46
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { className: "mt-6 overflow-hidden rounded-lg border border-gray-200", children: [
|
|
47
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center justify-between bg-gray-50 px-4 py-2", children: [
|
|
48
|
+
/* @__PURE__ */ jsxRuntime.jsx("span", { className: "text-xs font-medium uppercase text-gray-600", children: "Description" }),
|
|
49
|
+
/* @__PURE__ */ jsxRuntime.jsx("span", { className: "text-xs font-medium uppercase text-gray-600", children: "Enabled" })
|
|
50
|
+
] }),
|
|
51
|
+
mergedNotifications.map((notification) => /* @__PURE__ */ jsxRuntime.jsxs(
|
|
52
|
+
"div",
|
|
53
|
+
{
|
|
54
|
+
className: "flex items-center justify-between border-t border-gray-200 px-4 py-3",
|
|
55
|
+
children: [
|
|
56
|
+
/* @__PURE__ */ jsxRuntime.jsx("span", { className: "text-sm text-gray-900", children: NOTIFICATION_DESCRIPTIONS[notification.type] }),
|
|
57
|
+
/* @__PURE__ */ jsxRuntime.jsx(
|
|
58
|
+
"input",
|
|
59
|
+
{
|
|
60
|
+
type: "checkbox",
|
|
61
|
+
checked: notification.enabled,
|
|
62
|
+
onChange: () => onToggle?.(notification.type, !notification.enabled),
|
|
63
|
+
className: "portal-checkbox"
|
|
64
|
+
}
|
|
65
|
+
)
|
|
66
|
+
]
|
|
67
|
+
},
|
|
68
|
+
notification.type
|
|
69
|
+
))
|
|
70
|
+
] }),
|
|
71
|
+
/* @__PURE__ */ jsxRuntime.jsxs("p", { className: "mt-3 text-xs text-gray-500", children: [
|
|
72
|
+
"These notification preferences are specific to the ",
|
|
73
|
+
teamName,
|
|
74
|
+
" team. Switch teams to manage notifications for other teams."
|
|
75
|
+
] })
|
|
76
|
+
] });
|
|
77
|
+
};
|
|
78
|
+
var TeamsSection = ({
|
|
79
|
+
teams,
|
|
80
|
+
currentCustomer,
|
|
81
|
+
isLoading,
|
|
82
|
+
onTeamSwitch,
|
|
83
|
+
primaryColor
|
|
84
|
+
}) => {
|
|
85
|
+
const groupedTeams = react.useMemo(() => {
|
|
86
|
+
const groups = /* @__PURE__ */ new Map();
|
|
87
|
+
for (const team of teams) {
|
|
88
|
+
const existing = groups.get(team.appId);
|
|
89
|
+
if (existing) {
|
|
90
|
+
existing.teams.push(team);
|
|
91
|
+
} else {
|
|
92
|
+
groups.set(team.appId, {
|
|
93
|
+
appName: team.appName,
|
|
94
|
+
teams: [team]
|
|
95
|
+
});
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
return groups;
|
|
99
|
+
}, [teams]);
|
|
100
|
+
if (isLoading) {
|
|
101
|
+
return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "rounded-2xl border border-gray-200 bg-white p-6 shadow-sm", children: [
|
|
102
|
+
/* @__PURE__ */ jsxRuntime.jsx("h2", { className: "text-lg font-semibold text-gray-900", children: "Teams" }),
|
|
103
|
+
/* @__PURE__ */ jsxRuntime.jsx("p", { className: "mt-1 text-sm text-gray-500", children: "Teams associate your email address with any additional licenses your account has access to." }),
|
|
104
|
+
/* @__PURE__ */ jsxRuntime.jsx("div", { className: "mt-6", children: /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-sm text-gray-500", children: "Loading teams..." }) })
|
|
105
|
+
] });
|
|
106
|
+
}
|
|
107
|
+
return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "rounded-2xl border border-gray-200 bg-white p-6 shadow-sm", children: [
|
|
108
|
+
/* @__PURE__ */ jsxRuntime.jsx("h2", { className: "text-lg font-semibold text-gray-900", children: "Teams" }),
|
|
109
|
+
/* @__PURE__ */ jsxRuntime.jsx("p", { className: "mt-1 text-sm text-gray-500", children: "Teams associate your email address with any additional licenses your account has access to." }),
|
|
110
|
+
/* @__PURE__ */ jsxRuntime.jsx("div", { className: "mt-6 space-y-6", children: Array.from(groupedTeams.values()).map((appGroup) => /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "space-y-3", children: [
|
|
111
|
+
/* @__PURE__ */ jsxRuntime.jsx("h3", { className: "text-base font-medium text-gray-900", children: appGroup.appName }),
|
|
112
|
+
/* @__PURE__ */ jsxRuntime.jsx("div", { className: "space-y-2", children: appGroup.teams.map((team) => {
|
|
113
|
+
const isCurrentTeam = currentCustomer?.id === team.id;
|
|
114
|
+
return /* @__PURE__ */ jsxRuntime.jsxs(
|
|
115
|
+
"button",
|
|
116
|
+
{
|
|
117
|
+
type: "button",
|
|
118
|
+
onClick: () => !isCurrentTeam && onTeamSwitch?.(team),
|
|
119
|
+
disabled: isCurrentTeam,
|
|
120
|
+
className: `flex w-full items-center justify-between rounded-lg border p-4 text-left transition ${isCurrentTeam ? "cursor-default border-gray-300 bg-gray-50" : "cursor-pointer border-gray-200 hover:border-indigo-500"}`,
|
|
121
|
+
style: !isCurrentTeam && primaryColor ? { "--hover-border": primaryColor } : void 0,
|
|
122
|
+
children: [
|
|
123
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-center gap-3", children: [
|
|
124
|
+
/* @__PURE__ */ jsxRuntime.jsx(
|
|
125
|
+
"svg",
|
|
126
|
+
{
|
|
127
|
+
className: "h-4 w-4",
|
|
128
|
+
style: { color: primaryColor || "#6366f1" },
|
|
129
|
+
fill: "none",
|
|
130
|
+
viewBox: "0 0 24 24",
|
|
131
|
+
stroke: "currentColor",
|
|
132
|
+
strokeWidth: 2,
|
|
133
|
+
children: /* @__PURE__ */ jsxRuntime.jsx(
|
|
134
|
+
"path",
|
|
135
|
+
{
|
|
136
|
+
strokeLinecap: "round",
|
|
137
|
+
strokeLinejoin: "round",
|
|
138
|
+
d: "M17 20h5v-2a3 3 0 00-5.356-1.857M17 20H7m10 0v-2c0-.656-.126-1.283-.356-1.857M7 20H2v-2a3 3 0 015.356-1.857M7 20v-2c0-.656.126-1.283.356-1.857m0 0a5.002 5.002 0 019.288 0M15 7a3 3 0 11-6 0 3 3 0 016 0zm6 3a2 2 0 11-4 0 2 2 0 014 0zM7 10a2 2 0 11-4 0 2 2 0 014 0z"
|
|
139
|
+
}
|
|
140
|
+
)
|
|
141
|
+
}
|
|
142
|
+
),
|
|
143
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
|
|
144
|
+
/* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-sm font-medium text-gray-900", children: team.name }),
|
|
145
|
+
isCurrentTeam && /* @__PURE__ */ jsxRuntime.jsx("p", { className: "text-xs text-gray-500", children: "Current Team" })
|
|
146
|
+
] })
|
|
147
|
+
] }),
|
|
148
|
+
!isCurrentTeam && /* @__PURE__ */ jsxRuntime.jsx(
|
|
149
|
+
"svg",
|
|
150
|
+
{
|
|
151
|
+
className: "h-4 w-4",
|
|
152
|
+
style: { color: primaryColor || "#6366f1" },
|
|
153
|
+
fill: "none",
|
|
154
|
+
viewBox: "0 0 24 24",
|
|
155
|
+
stroke: "currentColor",
|
|
156
|
+
strokeWidth: 2,
|
|
157
|
+
children: /* @__PURE__ */ jsxRuntime.jsx(
|
|
158
|
+
"path",
|
|
159
|
+
{
|
|
160
|
+
strokeLinecap: "round",
|
|
161
|
+
strokeLinejoin: "round",
|
|
162
|
+
d: "M9 5l7 7-7 7"
|
|
163
|
+
}
|
|
164
|
+
)
|
|
165
|
+
}
|
|
166
|
+
)
|
|
167
|
+
]
|
|
168
|
+
},
|
|
169
|
+
team.id
|
|
170
|
+
);
|
|
171
|
+
}) })
|
|
172
|
+
] }, appGroup.appName)) })
|
|
173
|
+
] });
|
|
174
|
+
};
|
|
175
|
+
var ProfileSection = ({
|
|
176
|
+
user,
|
|
177
|
+
isLoading,
|
|
178
|
+
isUpdating,
|
|
179
|
+
error,
|
|
180
|
+
updateError,
|
|
181
|
+
onSave,
|
|
182
|
+
primaryColor
|
|
183
|
+
}) => {
|
|
184
|
+
const [isEditing, setIsEditing] = react.useState(false);
|
|
185
|
+
const [firstName, setFirstName] = react.useState(user?.firstName || "");
|
|
186
|
+
const [lastName, setLastName] = react.useState(user?.lastName || "");
|
|
187
|
+
const [inputError, setInputError] = react.useState(null);
|
|
188
|
+
const resetForm = () => {
|
|
189
|
+
setFirstName(user?.firstName || "");
|
|
190
|
+
setLastName(user?.lastName || "");
|
|
191
|
+
setInputError(null);
|
|
192
|
+
};
|
|
193
|
+
const handleSave = async () => {
|
|
194
|
+
if (!firstName && !lastName) {
|
|
195
|
+
setInputError("Please enter a first name or last name");
|
|
196
|
+
return;
|
|
197
|
+
}
|
|
198
|
+
if (firstName === user?.firstName && lastName === user?.lastName) {
|
|
199
|
+
setIsEditing(false);
|
|
200
|
+
return;
|
|
201
|
+
}
|
|
202
|
+
try {
|
|
203
|
+
await onSave?.({ firstName, lastName });
|
|
204
|
+
setIsEditing(false);
|
|
205
|
+
} catch (err) {
|
|
206
|
+
console.error("Error updating user information:", err);
|
|
207
|
+
}
|
|
208
|
+
};
|
|
209
|
+
const handleCancel = () => {
|
|
210
|
+
setIsEditing(false);
|
|
211
|
+
resetForm();
|
|
212
|
+
};
|
|
213
|
+
const handleEdit = () => {
|
|
214
|
+
resetForm();
|
|
215
|
+
setIsEditing(true);
|
|
216
|
+
};
|
|
217
|
+
const buttonStyle = primaryColor ? { backgroundColor: primaryColor, borderColor: primaryColor } : void 0;
|
|
218
|
+
return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "rounded-2xl border border-gray-200 bg-white p-6 shadow-sm", children: [
|
|
219
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex items-start justify-between gap-4", children: [
|
|
220
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
|
|
221
|
+
/* @__PURE__ */ jsxRuntime.jsx("h2", { className: "text-lg font-semibold text-gray-900", children: "Profile Information" }),
|
|
222
|
+
/* @__PURE__ */ jsxRuntime.jsx("p", { className: "mt-1 text-sm text-gray-500", children: "Your account information." }),
|
|
223
|
+
(error || updateError) && /* @__PURE__ */ jsxRuntime.jsx("p", { className: "mt-2 text-sm text-red-600", children: error?.message || updateError?.message || "Failed to update profile" }),
|
|
224
|
+
inputError && /* @__PURE__ */ jsxRuntime.jsx("p", { className: "mt-2 text-sm text-red-600", children: inputError })
|
|
225
|
+
] }),
|
|
226
|
+
isEditing ? /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex gap-2", children: [
|
|
227
|
+
/* @__PURE__ */ jsxRuntime.jsx(
|
|
228
|
+
"button",
|
|
229
|
+
{
|
|
230
|
+
type: "button",
|
|
231
|
+
onClick: handleCancel,
|
|
232
|
+
disabled: isUpdating,
|
|
233
|
+
className: "inline-flex items-center rounded-md border border-gray-300 bg-white px-3 py-1.5 text-sm font-medium text-gray-700 shadow-sm transition hover:bg-gray-50 disabled:opacity-50",
|
|
234
|
+
children: "Cancel"
|
|
235
|
+
}
|
|
236
|
+
),
|
|
237
|
+
/* @__PURE__ */ jsxRuntime.jsx(
|
|
238
|
+
"button",
|
|
239
|
+
{
|
|
240
|
+
type: "button",
|
|
241
|
+
onClick: handleSave,
|
|
242
|
+
disabled: isUpdating,
|
|
243
|
+
style: buttonStyle,
|
|
244
|
+
className: "inline-flex items-center rounded-md bg-indigo-500 px-3 py-1.5 text-sm font-medium text-white shadow-sm transition hover:bg-indigo-600 disabled:opacity-50",
|
|
245
|
+
children: isUpdating ? "Saving..." : "Save"
|
|
246
|
+
}
|
|
247
|
+
)
|
|
248
|
+
] }) : /* @__PURE__ */ jsxRuntime.jsx(
|
|
249
|
+
"button",
|
|
250
|
+
{
|
|
251
|
+
type: "button",
|
|
252
|
+
onClick: handleEdit,
|
|
253
|
+
style: buttonStyle,
|
|
254
|
+
className: "inline-flex items-center rounded-md bg-indigo-500 px-3 py-1.5 text-sm font-medium text-white shadow-sm transition hover:bg-indigo-600",
|
|
255
|
+
children: "Edit"
|
|
256
|
+
}
|
|
257
|
+
)
|
|
258
|
+
] }),
|
|
259
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { className: "mt-6 flex gap-8", children: [
|
|
260
|
+
isEditing ? /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex gap-4", children: [
|
|
261
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
|
|
262
|
+
/* @__PURE__ */ jsxRuntime.jsx("label", { className: "block text-sm font-medium text-gray-600", children: "First Name" }),
|
|
263
|
+
/* @__PURE__ */ jsxRuntime.jsx(
|
|
264
|
+
"input",
|
|
265
|
+
{
|
|
266
|
+
type: "text",
|
|
267
|
+
value: firstName,
|
|
268
|
+
onChange: (e) => {
|
|
269
|
+
setInputError(null);
|
|
270
|
+
setFirstName(e.target.value);
|
|
271
|
+
},
|
|
272
|
+
disabled: isUpdating,
|
|
273
|
+
placeholder: "First Name",
|
|
274
|
+
className: "portal-input mt-1 block w-48"
|
|
275
|
+
}
|
|
276
|
+
)
|
|
277
|
+
] }),
|
|
278
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
|
|
279
|
+
/* @__PURE__ */ jsxRuntime.jsx("label", { className: "block text-sm font-medium text-gray-600", children: "Last Name" }),
|
|
280
|
+
/* @__PURE__ */ jsxRuntime.jsx(
|
|
281
|
+
"input",
|
|
282
|
+
{
|
|
283
|
+
type: "text",
|
|
284
|
+
value: lastName,
|
|
285
|
+
onChange: (e) => {
|
|
286
|
+
setInputError(null);
|
|
287
|
+
setLastName(e.target.value);
|
|
288
|
+
},
|
|
289
|
+
disabled: isUpdating,
|
|
290
|
+
placeholder: "Last Name",
|
|
291
|
+
className: "portal-input mt-1 block w-48"
|
|
292
|
+
}
|
|
293
|
+
)
|
|
294
|
+
] })
|
|
295
|
+
] }) : /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex-1", children: [
|
|
296
|
+
/* @__PURE__ */ jsxRuntime.jsx("label", { className: "block text-sm font-medium text-gray-600", children: "Full Name (optional)" }),
|
|
297
|
+
/* @__PURE__ */ jsxRuntime.jsx("p", { className: "mt-1 text-sm text-gray-900", children: user?.firstName || user?.lastName ? `${user.firstName || ""} ${user.lastName || ""}`.trim() : "\u2014" })
|
|
298
|
+
] }),
|
|
299
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { className: "flex-1", children: [
|
|
300
|
+
/* @__PURE__ */ jsxRuntime.jsx("label", { className: "block text-sm font-medium text-gray-600", children: "Email" }),
|
|
301
|
+
/* @__PURE__ */ jsxRuntime.jsx("p", { className: "mt-1 text-sm text-gray-500", children: isLoading ? "Loading..." : user?.emailAddress || "" })
|
|
302
|
+
] })
|
|
303
|
+
] })
|
|
304
|
+
] });
|
|
305
|
+
};
|
|
306
|
+
var UserSettings = ({
|
|
307
|
+
user,
|
|
308
|
+
isUserLoading = false,
|
|
309
|
+
userError,
|
|
310
|
+
isUpdating = false,
|
|
311
|
+
updateError,
|
|
312
|
+
onUpdateUser,
|
|
313
|
+
teams = [],
|
|
314
|
+
currentCustomer,
|
|
315
|
+
isTeamsLoading = false,
|
|
316
|
+
onTeamSwitch,
|
|
317
|
+
notifications = [],
|
|
318
|
+
isNotificationsLoading = false,
|
|
319
|
+
onToggleNotification,
|
|
320
|
+
primaryColor
|
|
321
|
+
}) => {
|
|
322
|
+
if (userError) {
|
|
323
|
+
return /* @__PURE__ */ jsxRuntime.jsx("div", { className: "p-8", children: /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "text-red-600", children: [
|
|
324
|
+
"Failed to load user settings. ",
|
|
325
|
+
userError?.message,
|
|
326
|
+
" Please try again later."
|
|
327
|
+
] }) });
|
|
328
|
+
}
|
|
329
|
+
return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "w-full space-y-6", children: [
|
|
330
|
+
/* @__PURE__ */ jsxRuntime.jsx("header", { children: /* @__PURE__ */ jsxRuntime.jsx("h1", { className: "text-2xl font-semibold text-gray-900", children: "User Settings" }) }),
|
|
331
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { className: "space-y-6", children: [
|
|
332
|
+
/* @__PURE__ */ jsxRuntime.jsx(
|
|
333
|
+
ProfileSection,
|
|
334
|
+
{
|
|
335
|
+
user,
|
|
336
|
+
isLoading: isUserLoading,
|
|
337
|
+
isUpdating,
|
|
338
|
+
error: userError,
|
|
339
|
+
updateError,
|
|
340
|
+
onSave: onUpdateUser,
|
|
341
|
+
primaryColor
|
|
342
|
+
}
|
|
343
|
+
),
|
|
344
|
+
/* @__PURE__ */ jsxRuntime.jsx(
|
|
345
|
+
TeamsSection,
|
|
346
|
+
{
|
|
347
|
+
teams,
|
|
348
|
+
currentCustomer,
|
|
349
|
+
isLoading: isTeamsLoading,
|
|
350
|
+
onTeamSwitch,
|
|
351
|
+
primaryColor
|
|
352
|
+
}
|
|
353
|
+
),
|
|
354
|
+
currentCustomer && /* @__PURE__ */ jsxRuntime.jsx(
|
|
355
|
+
EmailNotificationsSection,
|
|
356
|
+
{
|
|
357
|
+
teamName: currentCustomer.name,
|
|
358
|
+
notifications,
|
|
359
|
+
isLoading: isNotificationsLoading,
|
|
360
|
+
onToggle: onToggleNotification
|
|
361
|
+
}
|
|
362
|
+
)
|
|
363
|
+
] })
|
|
364
|
+
] });
|
|
365
|
+
};
|
|
366
|
+
UserSettings.displayName = "UserSettings";
|
|
367
|
+
|
|
368
|
+
exports.UserSettings = UserSettings;
|
|
369
|
+
//# sourceMappingURL=user-settings.js.map
|
|
370
|
+
//# sourceMappingURL=user-settings.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/user-settings.tsx"],"names":["jsxs","jsx","useMemo","useState"],"mappings":"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","file":"user-settings.js","sourcesContent":["\"use client\";\n\nimport { useState, useMemo } from \"react\";\n\n// =============================================================================\n// Types\n// =============================================================================\n\nexport interface UserSettingsUser {\n emailAddress: string;\n firstName?: string;\n lastName?: string;\n}\n\nexport interface UserSettingsTeam {\n id: string;\n name: string;\n appId: string;\n appName: string;\n appSlug: string;\n}\n\nexport interface UserSettingsCustomer {\n id: string;\n name: string;\n}\n\nexport interface UserSettingsNotification {\n type: string;\n enabled: boolean;\n}\n\nexport interface UserSettingsProps {\n // User profile\n user?: UserSettingsUser;\n isUserLoading?: boolean;\n userError?: Error | null;\n \n // Profile editing\n isUpdating?: boolean;\n updateError?: Error | null;\n onUpdateUser?: (data: { firstName?: string; lastName?: string }) => Promise<void>;\n \n // Teams\n teams?: UserSettingsTeam[];\n currentCustomer?: UserSettingsCustomer;\n isTeamsLoading?: boolean;\n onTeamSwitch?: (team: UserSettingsTeam) => Promise<void>;\n \n // Notifications\n notifications?: UserSettingsNotification[];\n isNotificationsLoading?: boolean;\n onToggleNotification?: (notificationType: string, enabled: boolean) => Promise<void>;\n \n // Optional branding\n primaryColor?: string;\n}\n\n// =============================================================================\n// Notification descriptions\n// =============================================================================\n\nconst NOTIFICATION_DESCRIPTIONS: Record<string, string> = {\n \"new-version\": \"New version available\",\n};\n\n// =============================================================================\n// Email Notifications Section\n// =============================================================================\n\ninterface EmailNotificationsSectionProps {\n teamName: string;\n notifications: UserSettingsNotification[];\n isLoading: boolean;\n onToggle?: (type: string, enabled: boolean) => Promise<void>;\n}\n\nconst EmailNotificationsSection = ({\n teamName,\n notifications,\n isLoading,\n onToggle,\n}: EmailNotificationsSectionProps) => {\n // Merge API data with all notification types to ensure all options are always shown\n const allNotificationTypes = Object.keys(NOTIFICATION_DESCRIPTIONS);\n const mergedNotifications = allNotificationTypes.map((type) => {\n const existing = notifications.find((n) => n.type === type);\n return {\n type,\n enabled: existing ? existing.enabled : type === \"new-version\", // Default new-version to true\n };\n });\n\n if (isLoading) {\n return (\n <div className=\"rounded-2xl border border-gray-200 bg-white p-6 shadow-sm\">\n <h2 className=\"text-lg font-semibold text-gray-900\">\n Email Notifications for {teamName}\n </h2>\n <p className=\"mt-1 text-sm text-gray-500\">\n Choose which email notifications you want to receive for this team.\n </p>\n <div className=\"mt-6 flex justify-center py-8\">\n <p className=\"text-sm text-gray-500\">Loading notifications...</p>\n </div>\n </div>\n );\n }\n\n return (\n <div className=\"rounded-2xl border border-gray-200 bg-white p-6 shadow-sm\">\n <h2 className=\"text-lg font-semibold text-gray-900\">\n Email Notifications for {teamName}\n </h2>\n <p className=\"mt-1 text-sm text-gray-500\">\n Choose which email notifications you want to receive for this team.\n </p>\n\n <div className=\"mt-6 overflow-hidden rounded-lg border border-gray-200\">\n <div className=\"flex items-center justify-between bg-gray-50 px-4 py-2\">\n <span className=\"text-xs font-medium uppercase text-gray-600\">\n Description\n </span>\n <span className=\"text-xs font-medium uppercase text-gray-600\">\n Enabled\n </span>\n </div>\n {mergedNotifications.map((notification) => (\n <div\n key={notification.type}\n className=\"flex items-center justify-between border-t border-gray-200 px-4 py-3\"\n >\n <span className=\"text-sm text-gray-900\">\n {NOTIFICATION_DESCRIPTIONS[notification.type]}\n </span>\n <input\n type=\"checkbox\"\n checked={notification.enabled}\n onChange={() => onToggle?.(notification.type, !notification.enabled)}\n className=\"portal-checkbox\"\n />\n </div>\n ))}\n </div>\n <p className=\"mt-3 text-xs text-gray-500\">\n These notification preferences are specific to the {teamName} team.\n Switch teams to manage notifications for other teams.\n </p>\n </div>\n );\n};\n\n// =============================================================================\n// Teams Section\n// =============================================================================\n\ninterface TeamsSectionProps {\n teams: UserSettingsTeam[];\n currentCustomer?: UserSettingsCustomer;\n isLoading: boolean;\n onTeamSwitch?: (team: UserSettingsTeam) => Promise<void>;\n primaryColor?: string;\n}\n\nconst TeamsSection = ({\n teams,\n currentCustomer,\n isLoading,\n onTeamSwitch,\n primaryColor,\n}: TeamsSectionProps) => {\n // Group teams by app\n const groupedTeams = useMemo(() => {\n const groups = new Map<string, { appName: string; teams: UserSettingsTeam[] }>();\n \n for (const team of teams) {\n const existing = groups.get(team.appId);\n if (existing) {\n existing.teams.push(team);\n } else {\n groups.set(team.appId, {\n appName: team.appName,\n teams: [team],\n });\n }\n }\n \n return groups;\n }, [teams]);\n\n if (isLoading) {\n return (\n <div className=\"rounded-2xl border border-gray-200 bg-white p-6 shadow-sm\">\n <h2 className=\"text-lg font-semibold text-gray-900\">Teams</h2>\n <p className=\"mt-1 text-sm text-gray-500\">\n Teams associate your email address with any additional licenses your\n account has access to.\n </p>\n <div className=\"mt-6\">\n <p className=\"text-sm text-gray-500\">Loading teams...</p>\n </div>\n </div>\n );\n }\n\n return (\n <div className=\"rounded-2xl border border-gray-200 bg-white p-6 shadow-sm\">\n <h2 className=\"text-lg font-semibold text-gray-900\">Teams</h2>\n <p className=\"mt-1 text-sm text-gray-500\">\n Teams associate your email address with any additional licenses your\n account has access to.\n </p>\n\n <div className=\"mt-6 space-y-6\">\n {Array.from(groupedTeams.values()).map((appGroup) => (\n <div key={appGroup.appName} className=\"space-y-3\">\n <h3 className=\"text-base font-medium text-gray-900\">\n {appGroup.appName}\n </h3>\n <div className=\"space-y-2\">\n {appGroup.teams.map((team) => {\n const isCurrentTeam = currentCustomer?.id === team.id;\n return (\n <button\n key={team.id}\n type=\"button\"\n onClick={() => !isCurrentTeam && onTeamSwitch?.(team)}\n disabled={isCurrentTeam}\n className={`flex w-full items-center justify-between rounded-lg border p-4 text-left transition ${\n isCurrentTeam\n ? \"cursor-default border-gray-300 bg-gray-50\"\n : \"cursor-pointer border-gray-200 hover:border-indigo-500\"\n }`}\n style={\n !isCurrentTeam && primaryColor\n ? { \"--hover-border\": primaryColor } as React.CSSProperties\n : undefined\n }\n >\n <div className=\"flex items-center gap-3\">\n <svg\n className=\"h-4 w-4\"\n style={{ color: primaryColor || \"#6366f1\" }}\n fill=\"none\"\n viewBox=\"0 0 24 24\"\n stroke=\"currentColor\"\n strokeWidth={2}\n >\n <path\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n d=\"M17 20h5v-2a3 3 0 00-5.356-1.857M17 20H7m10 0v-2c0-.656-.126-1.283-.356-1.857M7 20H2v-2a3 3 0 015.356-1.857M7 20v-2c0-.656.126-1.283.356-1.857m0 0a5.002 5.002 0 019.288 0M15 7a3 3 0 11-6 0 3 3 0 016 0zm6 3a2 2 0 11-4 0 2 2 0 014 0zM7 10a2 2 0 11-4 0 2 2 0 014 0z\"\n />\n </svg>\n <div>\n <p className=\"text-sm font-medium text-gray-900\">\n {team.name}\n </p>\n {isCurrentTeam && (\n <p className=\"text-xs text-gray-500\">Current Team</p>\n )}\n </div>\n </div>\n {!isCurrentTeam && (\n <svg\n className=\"h-4 w-4\"\n style={{ color: primaryColor || \"#6366f1\" }}\n fill=\"none\"\n viewBox=\"0 0 24 24\"\n stroke=\"currentColor\"\n strokeWidth={2}\n >\n <path\n strokeLinecap=\"round\"\n strokeLinejoin=\"round\"\n d=\"M9 5l7 7-7 7\"\n />\n </svg>\n )}\n </button>\n );\n })}\n </div>\n </div>\n ))}\n </div>\n </div>\n );\n};\n\n// =============================================================================\n// Profile Section\n// =============================================================================\n\ninterface ProfileSectionProps {\n user?: UserSettingsUser;\n isLoading: boolean;\n isUpdating: boolean;\n error?: Error | null;\n updateError?: Error | null;\n onSave?: (data: { firstName?: string; lastName?: string }) => Promise<void>;\n primaryColor?: string;\n}\n\nconst ProfileSection = ({\n user,\n isLoading,\n isUpdating,\n error,\n updateError,\n onSave,\n primaryColor,\n}: ProfileSectionProps) => {\n const [isEditing, setIsEditing] = useState(false);\n const [firstName, setFirstName] = useState(user?.firstName || \"\");\n const [lastName, setLastName] = useState(user?.lastName || \"\");\n const [inputError, setInputError] = useState<string | null>(null);\n\n // Reset form when user changes\n const resetForm = () => {\n setFirstName(user?.firstName || \"\");\n setLastName(user?.lastName || \"\");\n setInputError(null);\n };\n\n const handleSave = async () => {\n if (!firstName && !lastName) {\n setInputError(\"Please enter a first name or last name\");\n return;\n }\n\n if (firstName === user?.firstName && lastName === user?.lastName) {\n setIsEditing(false);\n return;\n }\n\n try {\n await onSave?.({ firstName, lastName });\n setIsEditing(false);\n } catch (err) {\n console.error(\"Error updating user information:\", err);\n }\n };\n\n const handleCancel = () => {\n setIsEditing(false);\n resetForm();\n };\n\n const handleEdit = () => {\n resetForm();\n setIsEditing(true);\n };\n\n const buttonStyle = primaryColor\n ? { backgroundColor: primaryColor, borderColor: primaryColor }\n : undefined;\n\n return (\n <div className=\"rounded-2xl border border-gray-200 bg-white p-6 shadow-sm\">\n <div className=\"flex items-start justify-between gap-4\">\n <div>\n <h2 className=\"text-lg font-semibold text-gray-900\">\n Profile Information\n </h2>\n <p className=\"mt-1 text-sm text-gray-500\">Your account information.</p>\n {(error || updateError) && (\n <p className=\"mt-2 text-sm text-red-600\">\n {error?.message || updateError?.message || \"Failed to update profile\"}\n </p>\n )}\n {inputError && (\n <p className=\"mt-2 text-sm text-red-600\">{inputError}</p>\n )}\n </div>\n {isEditing ? (\n <div className=\"flex gap-2\">\n <button\n type=\"button\"\n onClick={handleCancel}\n disabled={isUpdating}\n className=\"inline-flex items-center rounded-md border border-gray-300 bg-white px-3 py-1.5 text-sm font-medium text-gray-700 shadow-sm transition hover:bg-gray-50 disabled:opacity-50\"\n >\n Cancel\n </button>\n <button\n type=\"button\"\n onClick={handleSave}\n disabled={isUpdating}\n style={buttonStyle}\n className=\"inline-flex items-center rounded-md bg-indigo-500 px-3 py-1.5 text-sm font-medium text-white shadow-sm transition hover:bg-indigo-600 disabled:opacity-50\"\n >\n {isUpdating ? \"Saving...\" : \"Save\"}\n </button>\n </div>\n ) : (\n <button\n type=\"button\"\n onClick={handleEdit}\n style={buttonStyle}\n className=\"inline-flex items-center rounded-md bg-indigo-500 px-3 py-1.5 text-sm font-medium text-white shadow-sm transition hover:bg-indigo-600\"\n >\n Edit\n </button>\n )}\n </div>\n\n <div className=\"mt-6 flex gap-8\">\n {isEditing ? (\n <div className=\"flex gap-4\">\n <div>\n <label className=\"block text-sm font-medium text-gray-600\">\n First Name\n </label>\n <input\n type=\"text\"\n value={firstName}\n onChange={(e) => {\n setInputError(null);\n setFirstName(e.target.value);\n }}\n disabled={isUpdating}\n placeholder=\"First Name\"\n className=\"portal-input mt-1 block w-48\"\n />\n </div>\n <div>\n <label className=\"block text-sm font-medium text-gray-600\">\n Last Name\n </label>\n <input\n type=\"text\"\n value={lastName}\n onChange={(e) => {\n setInputError(null);\n setLastName(e.target.value);\n }}\n disabled={isUpdating}\n placeholder=\"Last Name\"\n className=\"portal-input mt-1 block w-48\"\n />\n </div>\n </div>\n ) : (\n <div className=\"flex-1\">\n <label className=\"block text-sm font-medium text-gray-600\">\n Full Name (optional)\n </label>\n <p className=\"mt-1 text-sm text-gray-900\">\n {user?.firstName || user?.lastName\n ? `${user.firstName || \"\"} ${user.lastName || \"\"}`.trim()\n : \"—\"}\n </p>\n </div>\n )}\n <div className=\"flex-1\">\n <label className=\"block text-sm font-medium text-gray-600\">Email</label>\n <p className=\"mt-1 text-sm text-gray-500\">\n {isLoading ? \"Loading...\" : user?.emailAddress || \"\"}\n </p>\n </div>\n </div>\n </div>\n );\n};\n\n// =============================================================================\n// Main Component\n// =============================================================================\n\nexport const UserSettings = ({\n user,\n isUserLoading = false,\n userError,\n isUpdating = false,\n updateError,\n onUpdateUser,\n teams = [],\n currentCustomer,\n isTeamsLoading = false,\n onTeamSwitch,\n notifications = [],\n isNotificationsLoading = false,\n onToggleNotification,\n primaryColor,\n}: UserSettingsProps) => {\n if (userError) {\n return (\n <div className=\"p-8\">\n <div className=\"text-red-600\">\n Failed to load user settings. {userError?.message} Please try again later.\n </div>\n </div>\n );\n }\n\n return (\n <div className=\"w-full space-y-6\">\n <header>\n <h1 className=\"text-2xl font-semibold text-gray-900\">User Settings</h1>\n </header>\n\n <div className=\"space-y-6\">\n <ProfileSection\n user={user}\n isLoading={isUserLoading}\n isUpdating={isUpdating}\n error={userError}\n updateError={updateError}\n onSave={onUpdateUser}\n primaryColor={primaryColor}\n />\n\n <TeamsSection\n teams={teams}\n currentCustomer={currentCustomer}\n isLoading={isTeamsLoading}\n onTeamSwitch={onTeamSwitch}\n primaryColor={primaryColor}\n />\n\n {currentCustomer && (\n <EmailNotificationsSection\n teamName={currentCustomer.name}\n notifications={notifications}\n isLoading={isNotificationsLoading}\n onToggle={onToggleNotification}\n />\n )}\n </div>\n </div>\n );\n};\n\nUserSettings.displayName = \"UserSettings\";\n"]}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Centralized API client utility for handling authenticated requests
|
|
3
|
+
* with automatic 401 detection, cookie deletion, and redirect.
|
|
4
|
+
*/
|
|
5
|
+
interface ApiFetchOptions extends RequestInit {
|
|
6
|
+
token?: string;
|
|
7
|
+
}
|
|
8
|
+
declare class UnauthorizedError extends Error {
|
|
9
|
+
constructor(message?: string);
|
|
10
|
+
}
|
|
11
|
+
/**
|
|
12
|
+
* Helper to check if an error is a Next.js redirect error.
|
|
13
|
+
* These errors should NOT be caught as they control navigation flow.
|
|
14
|
+
*/
|
|
15
|
+
declare function isRedirectError(error: unknown): boolean;
|
|
16
|
+
/**
|
|
17
|
+
* Fetch wrapper that automatically handles error responses by:
|
|
18
|
+
* - 401: Redirecting to "/" with expired parameter
|
|
19
|
+
* - 502, 503, 504: Redirecting to "/error" with status code
|
|
20
|
+
*
|
|
21
|
+
* This function should be used for all authenticated API calls in server actions.
|
|
22
|
+
*
|
|
23
|
+
* IMPORTANT: This must be called from a Server Action context, not directly from
|
|
24
|
+
* a Server Component, because it modifies cookies.
|
|
25
|
+
*/
|
|
26
|
+
declare function authenticatedFetch(url: string, options?: ApiFetchOptions): Promise<Response>;
|
|
27
|
+
|
|
28
|
+
interface RawBranding {
|
|
29
|
+
brandingData?: string;
|
|
30
|
+
}
|
|
31
|
+
declare const decodeBranding: ({ brandingData }: RawBranding) => {
|
|
32
|
+
logo?: undefined;
|
|
33
|
+
title?: undefined;
|
|
34
|
+
customColor1?: undefined;
|
|
35
|
+
customColor2?: undefined;
|
|
36
|
+
} | {
|
|
37
|
+
logo: string | undefined;
|
|
38
|
+
title: string | undefined;
|
|
39
|
+
customColor1: string | undefined;
|
|
40
|
+
customColor2: string | undefined;
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
/**
|
|
44
|
+
* Session validation utilities for handling authentication tokens.
|
|
45
|
+
*/
|
|
46
|
+
/**
|
|
47
|
+
* Validates a session token by making a lightweight authenticated request.
|
|
48
|
+
* Returns true if valid, false if invalid (401).
|
|
49
|
+
*
|
|
50
|
+
* This should be called from Server Components (like the home page) to validate
|
|
51
|
+
* the session cookie before rendering authenticated content.
|
|
52
|
+
*/
|
|
53
|
+
declare function validateSession(token: string): Promise<boolean>;
|
|
54
|
+
/**
|
|
55
|
+
* Deletes the portal_session cookie.
|
|
56
|
+
* This MUST be called from a Server Action or Route Handler, not from Server Component render.
|
|
57
|
+
*
|
|
58
|
+
* Usage example in a Server Action:
|
|
59
|
+
* ```typescript
|
|
60
|
+
* "use server";
|
|
61
|
+
* import { deleteSessionCookie } from "@replicated/portal-components/utils";
|
|
62
|
+
* export async function logout() {
|
|
63
|
+
* await deleteSessionCookie();
|
|
64
|
+
* redirect("/");
|
|
65
|
+
* }
|
|
66
|
+
* ```
|
|
67
|
+
*/
|
|
68
|
+
declare function deleteSessionCookie(): Promise<void>;
|
|
69
|
+
|
|
70
|
+
export { UnauthorizedError, authenticatedFetch, decodeBranding, deleteSessionCookie, isRedirectError, validateSession };
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Centralized API client utility for handling authenticated requests
|
|
3
|
+
* with automatic 401 detection, cookie deletion, and redirect.
|
|
4
|
+
*/
|
|
5
|
+
interface ApiFetchOptions extends RequestInit {
|
|
6
|
+
token?: string;
|
|
7
|
+
}
|
|
8
|
+
declare class UnauthorizedError extends Error {
|
|
9
|
+
constructor(message?: string);
|
|
10
|
+
}
|
|
11
|
+
/**
|
|
12
|
+
* Helper to check if an error is a Next.js redirect error.
|
|
13
|
+
* These errors should NOT be caught as they control navigation flow.
|
|
14
|
+
*/
|
|
15
|
+
declare function isRedirectError(error: unknown): boolean;
|
|
16
|
+
/**
|
|
17
|
+
* Fetch wrapper that automatically handles error responses by:
|
|
18
|
+
* - 401: Redirecting to "/" with expired parameter
|
|
19
|
+
* - 502, 503, 504: Redirecting to "/error" with status code
|
|
20
|
+
*
|
|
21
|
+
* This function should be used for all authenticated API calls in server actions.
|
|
22
|
+
*
|
|
23
|
+
* IMPORTANT: This must be called from a Server Action context, not directly from
|
|
24
|
+
* a Server Component, because it modifies cookies.
|
|
25
|
+
*/
|
|
26
|
+
declare function authenticatedFetch(url: string, options?: ApiFetchOptions): Promise<Response>;
|
|
27
|
+
|
|
28
|
+
interface RawBranding {
|
|
29
|
+
brandingData?: string;
|
|
30
|
+
}
|
|
31
|
+
declare const decodeBranding: ({ brandingData }: RawBranding) => {
|
|
32
|
+
logo?: undefined;
|
|
33
|
+
title?: undefined;
|
|
34
|
+
customColor1?: undefined;
|
|
35
|
+
customColor2?: undefined;
|
|
36
|
+
} | {
|
|
37
|
+
logo: string | undefined;
|
|
38
|
+
title: string | undefined;
|
|
39
|
+
customColor1: string | undefined;
|
|
40
|
+
customColor2: string | undefined;
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
/**
|
|
44
|
+
* Session validation utilities for handling authentication tokens.
|
|
45
|
+
*/
|
|
46
|
+
/**
|
|
47
|
+
* Validates a session token by making a lightweight authenticated request.
|
|
48
|
+
* Returns true if valid, false if invalid (401).
|
|
49
|
+
*
|
|
50
|
+
* This should be called from Server Components (like the home page) to validate
|
|
51
|
+
* the session cookie before rendering authenticated content.
|
|
52
|
+
*/
|
|
53
|
+
declare function validateSession(token: string): Promise<boolean>;
|
|
54
|
+
/**
|
|
55
|
+
* Deletes the portal_session cookie.
|
|
56
|
+
* This MUST be called from a Server Action or Route Handler, not from Server Component render.
|
|
57
|
+
*
|
|
58
|
+
* Usage example in a Server Action:
|
|
59
|
+
* ```typescript
|
|
60
|
+
* "use server";
|
|
61
|
+
* import { deleteSessionCookie } from "@replicated/portal-components/utils";
|
|
62
|
+
* export async function logout() {
|
|
63
|
+
* await deleteSessionCookie();
|
|
64
|
+
* redirect("/");
|
|
65
|
+
* }
|
|
66
|
+
* ```
|
|
67
|
+
*/
|
|
68
|
+
declare function deleteSessionCookie(): Promise<void>;
|
|
69
|
+
|
|
70
|
+
export { UnauthorizedError, authenticatedFetch, decodeBranding, deleteSessionCookie, isRedirectError, validateSession };
|