web-mojo 2.2.57 → 2.2.59
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/dist/admin.cjs.js +1 -1
- package/dist/admin.cjs.js.map +1 -1
- package/dist/admin.es.js +1 -10105
- package/dist/admin.es.js.map +1 -1
- package/dist/auth.cjs.js +1 -1
- package/dist/auth.es.js +1 -588
- package/dist/auth.es.js.map +1 -1
- package/dist/charts.cjs.js +1 -1
- package/dist/charts.es.js +1 -571
- package/dist/charts.es.js.map +1 -1
- package/dist/chunks/ChatView-D4A9rIX3.js +2 -0
- package/dist/chunks/ChatView-D4A9rIX3.js.map +1 -0
- package/dist/chunks/ChatView-nxaq8aIo.js +2 -0
- package/dist/chunks/ChatView-nxaq8aIo.js.map +1 -0
- package/dist/chunks/Collection-1sPoIFvQ.js +2 -0
- package/dist/chunks/{Collection-DaiL0uGl.js.map → Collection-1sPoIFvQ.js.map} +1 -1
- package/dist/chunks/{Collection-CxbNKOas.js → Collection-DSBRXpwK.js} +2 -2
- package/dist/chunks/{Collection-CxbNKOas.js.map → Collection-DSBRXpwK.js.map} +1 -1
- package/dist/chunks/{ContextMenu-ClwHEbbD.js → ContextMenu-BWy7WqF4.js} +2 -2
- package/dist/chunks/{ContextMenu-ClwHEbbD.js.map → ContextMenu-BWy7WqF4.js.map} +1 -1
- package/dist/chunks/ContextMenu-BvniQz-N.js +3 -0
- package/dist/chunks/{ContextMenu-sgvgSACY.js.map → ContextMenu-BvniQz-N.js.map} +1 -1
- package/dist/chunks/DataView--nUWtq6r.js +2 -0
- package/dist/chunks/{DataView-Dzo0jbs2.js.map → DataView--nUWtq6r.js.map} +1 -1
- package/dist/chunks/{DataView-1xh3GFeC.js → DataView-CK3Z0TJH.js} +2 -2
- package/dist/chunks/{DataView-1xh3GFeC.js.map → DataView-CK3Z0TJH.js.map} +1 -1
- package/dist/chunks/Dialog-BcgSR01Z.js +2 -0
- package/dist/chunks/{Dialog-DOGDalUq.js.map → Dialog-BcgSR01Z.js.map} +1 -1
- package/dist/chunks/{Dialog-CQlTDhZS.js → Dialog-DwCTFV6O.js} +2 -2
- package/dist/chunks/{Dialog-CQlTDhZS.js.map → Dialog-DwCTFV6O.js.map} +1 -1
- package/dist/chunks/FormPlugins-DvQ-G5J5.js +2 -0
- package/dist/chunks/{FormPlugins-DY6e88YT.js.map → FormPlugins-DvQ-G5J5.js.map} +1 -1
- package/dist/chunks/{FormView-DaKA4Sys.js → FormView-CRmEReTC.js} +3 -3
- package/dist/chunks/{FormView-DaKA4Sys.js.map → FormView-CRmEReTC.js.map} +1 -1
- package/dist/chunks/FormView-OLA7t-yv.js +3 -0
- package/dist/chunks/{FormView-Dz3mYasQ.js.map → FormView-OLA7t-yv.js.map} +1 -1
- package/dist/chunks/ListView-6JQ6tRXs.js +2 -0
- package/dist/chunks/{ListView-X5w5jf51.js.map → ListView-6JQ6tRXs.js.map} +1 -1
- package/dist/chunks/{ListView-CDzKIpd8.js → ListView-DVStKiMi.js} +2 -2
- package/dist/chunks/{ListView-CDzKIpd8.js.map → ListView-DVStKiMi.js.map} +1 -1
- package/dist/chunks/{MetricsCountryMapView-Dx2cw7ya.js → MetricsCountryMapView-CnAEbUw_.js} +2 -2
- package/dist/chunks/{MetricsCountryMapView-Dx2cw7ya.js.map → MetricsCountryMapView-CnAEbUw_.js.map} +1 -1
- package/dist/chunks/MetricsCountryMapView-J067qrrt.js +2 -0
- package/dist/chunks/{MetricsCountryMapView-B2xz6zUw.js.map → MetricsCountryMapView-J067qrrt.js.map} +1 -1
- package/dist/chunks/{MetricsMiniChartWidget-CBuso0OE.js → MetricsMiniChartWidget-BeD1slGs.js} +2 -2
- package/dist/chunks/{MetricsMiniChartWidget-CBuso0OE.js.map → MetricsMiniChartWidget-BeD1slGs.js.map} +1 -1
- package/dist/chunks/MetricsMiniChartWidget-x2gFjHOU.js +2 -0
- package/dist/chunks/{MetricsMiniChartWidget-DvKd7Qrk.js.map → MetricsMiniChartWidget-x2gFjHOU.js.map} +1 -1
- package/dist/chunks/PDFViewer-CsyKn-gh.js +2 -0
- package/dist/chunks/{PDFViewer-EJ9cOfPF.js.map → PDFViewer-CsyKn-gh.js.map} +1 -1
- package/dist/chunks/{PDFViewer-ofMGdSaj.js → PDFViewer-DSa4BZCm.js} +2 -2
- package/dist/chunks/{PDFViewer-ofMGdSaj.js.map → PDFViewer-DSa4BZCm.js.map} +1 -1
- package/dist/chunks/Rest-DHbszkuP.js +2 -0
- package/dist/chunks/Rest-DHbszkuP.js.map +1 -0
- package/dist/chunks/Rest-Ds9e8tN8.js +2 -0
- package/dist/chunks/Rest-Ds9e8tN8.js.map +1 -0
- package/dist/chunks/TokenManager-D6SjKgPZ.js +2 -0
- package/dist/chunks/{TokenManager-DoN9e6q6.js.map → TokenManager-D6SjKgPZ.js.map} +1 -1
- package/dist/chunks/{TokenManager-Gqvj7SDX.js → TokenManager-REbha1Le.js} +2 -2
- package/dist/chunks/{TokenManager-Gqvj7SDX.js.map → TokenManager-REbha1Le.js.map} +1 -1
- package/dist/chunks/WebApp-CULZpO_0.js +2 -0
- package/dist/chunks/{WebApp-6qvqmOts.js.map → WebApp-CULZpO_0.js.map} +1 -1
- package/dist/chunks/{WebApp-_dgpwtFw.js → WebApp-DovLtA60.js} +2 -2
- package/dist/chunks/{WebApp-_dgpwtFw.js.map → WebApp-DovLtA60.js.map} +1 -1
- package/dist/chunks/WebSocketClient-B-wc3mez.js +2 -0
- package/dist/chunks/{WebSocketClient-DG2olXpH.js.map → WebSocketClient-B-wc3mez.js.map} +1 -1
- package/dist/chunks/{WebSocketClient-MFkFlSue.js → WebSocketClient-BdZ9QYll.js} +2 -2
- package/dist/chunks/{WebSocketClient-MFkFlSue.js.map → WebSocketClient-BdZ9QYll.js.map} +1 -1
- package/dist/chunks/version-C3dnl1bg.js +2 -0
- package/dist/chunks/version-C3dnl1bg.js.map +1 -0
- package/dist/chunks/{version-BVADfTA5.js → version-ioN546cp.js} +2 -2
- package/dist/chunks/{version-BVADfTA5.js.map → version-ioN546cp.js.map} +1 -1
- package/dist/css/web-mojo.css +1 -1
- package/dist/docit.cjs.js +1 -1
- package/dist/docit.es.js +1 -957
- package/dist/docit.es.js.map +1 -1
- package/dist/index.cjs.js +1 -1
- package/dist/index.es.js +1 -3252
- package/dist/index.es.js.map +1 -1
- package/dist/lightbox.cjs.js +1 -1
- package/dist/lightbox.es.js +1 -3737
- package/dist/lightbox.es.js.map +1 -1
- package/dist/loader.umd.js +2 -2
- package/dist/map.cjs.js +1 -1
- package/dist/map.es.js +1 -1032
- package/dist/map.es.js.map +1 -1
- package/dist/mojo-auth.es.js +338 -0
- package/dist/mojo-auth.umd.js +1 -0
- package/dist/timeline.cjs.js +1 -1
- package/dist/timeline.es.js +1 -224
- package/dist/timeline.es.js.map +1 -1
- package/dist/web-mojo.lite.iife.js +14 -3
- package/dist/web-mojo.lite.iife.js.map +1 -1
- package/dist/web-mojo.lite.iife.min.js +6 -6
- package/dist/web-mojo.lite.iife.min.js.map +1 -1
- package/package.json +2 -2
- package/dist/chunks/ChatView-9k6xBWXk.js +0 -7632
- package/dist/chunks/ChatView-9k6xBWXk.js.map +0 -1
- package/dist/chunks/ChatView-CdtuCDYm.js +0 -2
- package/dist/chunks/ChatView-CdtuCDYm.js.map +0 -1
- package/dist/chunks/Collection-DaiL0uGl.js +0 -1014
- package/dist/chunks/ContextMenu-sgvgSACY.js +0 -1535
- package/dist/chunks/DataView-Dzo0jbs2.js +0 -862
- package/dist/chunks/Dialog-DOGDalUq.js +0 -1579
- package/dist/chunks/FormPlugins-DY6e88YT.js +0 -124
- package/dist/chunks/FormView-Dz3mYasQ.js +0 -8636
- package/dist/chunks/ListView-X5w5jf51.js +0 -495
- package/dist/chunks/MetricsCountryMapView-B2xz6zUw.js +0 -1054
- package/dist/chunks/MetricsMiniChartWidget-DvKd7Qrk.js +0 -3283
- package/dist/chunks/PDFViewer-EJ9cOfPF.js +0 -946
- package/dist/chunks/Rest-CgSjfMaU.js +0 -2
- package/dist/chunks/Rest-CgSjfMaU.js.map +0 -1
- package/dist/chunks/Rest-W-sPfGh9.js +0 -4375
- package/dist/chunks/Rest-W-sPfGh9.js.map +0 -1
- package/dist/chunks/TokenManager-DoN9e6q6.js +0 -1423
- package/dist/chunks/WebApp-6qvqmOts.js +0 -1386
- package/dist/chunks/WebSocketClient-DG2olXpH.js +0 -209
- package/dist/chunks/version-OyPGnx30.js +0 -38
- package/dist/chunks/version-OyPGnx30.js.map +0 -1
package/dist/auth.es.js
CHANGED
|
@@ -1,589 +1,2 @@
|
|
|
1
|
-
import { B, a, V, b, c, d } from "./chunks/version-OyPGnx30.js";
|
|
2
|
-
function createAuthClient({
|
|
3
|
-
baseURL,
|
|
4
|
-
fetchImpl = typeof fetch !== "undefined" ? fetch.bind(window) : null,
|
|
5
|
-
storage = typeof localStorage !== "undefined" ? localStorage : null,
|
|
6
|
-
endpoints = {}
|
|
7
|
-
} = {}) {
|
|
8
|
-
if (!baseURL) {
|
|
9
|
-
throw new Error("createAuthClient: baseURL is required");
|
|
10
|
-
}
|
|
11
|
-
if (!fetchImpl) {
|
|
12
|
-
throw new Error("createAuthClient: fetch implementation is not available in this environment");
|
|
13
|
-
}
|
|
14
|
-
if (!storage) {
|
|
15
|
-
throw new Error("createAuthClient: storage (localStorage) is not available in this environment");
|
|
16
|
-
}
|
|
17
|
-
const KEYS = {
|
|
18
|
-
access: "access_token",
|
|
19
|
-
refresh: "refresh_token",
|
|
20
|
-
user: "user"
|
|
21
|
-
};
|
|
22
|
-
const EP = {
|
|
23
|
-
login: "/login",
|
|
24
|
-
forgot: "/auth/forgot",
|
|
25
|
-
resetCode: "/auth/password/reset/code",
|
|
26
|
-
resetToken: "/auth/password/reset/token",
|
|
27
|
-
...endpoints
|
|
28
|
-
};
|
|
29
|
-
async function post(path, body) {
|
|
30
|
-
const res = await fetchImpl(`${baseURL}${path}`, {
|
|
31
|
-
method: "POST",
|
|
32
|
-
headers: { "Content-Type": "application/json" },
|
|
33
|
-
body: JSON.stringify(body || {})
|
|
34
|
-
});
|
|
35
|
-
let json = {};
|
|
36
|
-
try {
|
|
37
|
-
json = await res.json();
|
|
38
|
-
} catch (_) {
|
|
39
|
-
}
|
|
40
|
-
if (!res.ok) {
|
|
41
|
-
throw json || { message: `Request failed with status ${res.status}` };
|
|
42
|
-
}
|
|
43
|
-
return json;
|
|
44
|
-
}
|
|
45
|
-
function parseResponse(r) {
|
|
46
|
-
return r && r.data && r.data.data || r && r.data || r;
|
|
47
|
-
}
|
|
48
|
-
function saveAuthData(resp) {
|
|
49
|
-
const d2 = parseResponse(resp);
|
|
50
|
-
if (!d2 || !d2.access_token) {
|
|
51
|
-
throw new Error("No access_token in response.");
|
|
52
|
-
}
|
|
53
|
-
storage.setItem(KEYS.access, d2.access_token);
|
|
54
|
-
if (d2.refresh_token) storage.setItem(KEYS.refresh, d2.refresh_token);
|
|
55
|
-
if (d2.user) storage.setItem(KEYS.user, JSON.stringify(d2.user));
|
|
56
|
-
}
|
|
57
|
-
function getErrorMessage(err) {
|
|
58
|
-
return err?.message || err?.error || Array.isArray(err?.errors) && err.errors[0]?.message || "An error occurred. Please try again.";
|
|
59
|
-
}
|
|
60
|
-
return {
|
|
61
|
-
async login(username, password) {
|
|
62
|
-
const resp = await post(EP.login, { username, password });
|
|
63
|
-
saveAuthData(resp);
|
|
64
|
-
return parseResponse(resp);
|
|
65
|
-
},
|
|
66
|
-
async forgot({ email, method }) {
|
|
67
|
-
return post(EP.forgot, { email, method });
|
|
68
|
-
},
|
|
69
|
-
async resetWithCode({ email, code, newPassword }) {
|
|
70
|
-
const resp = await post(EP.resetCode, { email, code, new_password: newPassword });
|
|
71
|
-
saveAuthData(resp);
|
|
72
|
-
return parseResponse(resp);
|
|
73
|
-
},
|
|
74
|
-
async resetWithToken({ token, newPassword }) {
|
|
75
|
-
const resp = await post(EP.resetToken, { token, new_password: newPassword });
|
|
76
|
-
saveAuthData(resp);
|
|
77
|
-
return parseResponse(resp);
|
|
78
|
-
},
|
|
79
|
-
logout() {
|
|
80
|
-
storage.removeItem(KEYS.access);
|
|
81
|
-
storage.removeItem(KEYS.refresh);
|
|
82
|
-
storage.removeItem(KEYS.user);
|
|
83
|
-
},
|
|
84
|
-
isAuthenticated() {
|
|
85
|
-
return !!storage.getItem(KEYS.access);
|
|
86
|
-
},
|
|
87
|
-
getToken() {
|
|
88
|
-
return storage.getItem(KEYS.access);
|
|
89
|
-
},
|
|
90
|
-
getUser() {
|
|
91
|
-
const raw = storage.getItem(KEYS.user);
|
|
92
|
-
try {
|
|
93
|
-
return raw ? JSON.parse(raw) : null;
|
|
94
|
-
} catch {
|
|
95
|
-
return null;
|
|
96
|
-
}
|
|
97
|
-
},
|
|
98
|
-
getAuthHeader() {
|
|
99
|
-
const t = storage.getItem(KEYS.access);
|
|
100
|
-
return t ? `Bearer ${t}` : null;
|
|
101
|
-
},
|
|
102
|
-
getErrorMessage,
|
|
103
|
-
parseResponse
|
|
104
|
-
};
|
|
105
|
-
}
|
|
106
|
-
function mountAuth(container, options = {}) {
|
|
107
|
-
if (!container || !(container instanceof Element)) {
|
|
108
|
-
throw new Error("mountAuth: container must be a DOM Element");
|
|
109
|
-
}
|
|
110
|
-
const {
|
|
111
|
-
baseURL,
|
|
112
|
-
onSuccessRedirect,
|
|
113
|
-
allowRedirectOrigins,
|
|
114
|
-
branding = {},
|
|
115
|
-
theme,
|
|
116
|
-
endpoints,
|
|
117
|
-
providers,
|
|
118
|
-
texts = {}
|
|
119
|
-
} = options;
|
|
120
|
-
if (!baseURL) {
|
|
121
|
-
throw new Error("mountAuth: baseURL is required");
|
|
122
|
-
}
|
|
123
|
-
const urlParams = new URLSearchParams(window.location.search);
|
|
124
|
-
const redirectParam = urlParams.get("redirect") || urlParams.get("next") || urlParams.get("returnTo");
|
|
125
|
-
const redirectTarget = String(onSuccessRedirect || redirectParam || "/");
|
|
126
|
-
function isAllowedRedirect(url) {
|
|
127
|
-
if (!allowRedirectOrigins || allowRedirectOrigins.length === 0) {
|
|
128
|
-
return true;
|
|
129
|
-
}
|
|
130
|
-
try {
|
|
131
|
-
const target = new URL(url, window.location.origin);
|
|
132
|
-
return allowRedirectOrigins.includes(target.origin);
|
|
133
|
-
} catch {
|
|
134
|
-
return false;
|
|
135
|
-
}
|
|
136
|
-
}
|
|
137
|
-
function performRedirect() {
|
|
138
|
-
if (!isAllowedRedirect(redirectTarget)) {
|
|
139
|
-
window.location.href = "/";
|
|
140
|
-
return;
|
|
141
|
-
}
|
|
142
|
-
window.location.href = redirectTarget.startsWith("http") ? redirectTarget : new URL(redirectTarget, window.location.origin).href;
|
|
143
|
-
}
|
|
144
|
-
const auth = createAuthClient({ baseURL, endpoints });
|
|
145
|
-
const B2 = {
|
|
146
|
-
title: branding.title || "Sign In",
|
|
147
|
-
subtitle: branding.subtitle || "Sign in to your account",
|
|
148
|
-
logoUrl: branding.logoUrl || ""
|
|
149
|
-
};
|
|
150
|
-
const T = {
|
|
151
|
-
emailOrUsername: texts.emailOrUsername || "Email or Username",
|
|
152
|
-
password: texts.password || "Password",
|
|
153
|
-
signIn: texts.signIn || "Sign In",
|
|
154
|
-
forgotPassword: texts.forgotPassword || "Forgot password?",
|
|
155
|
-
resetYourPassword: texts.resetYourPassword || "Reset Your Password",
|
|
156
|
-
emailAddress: texts.emailAddress || "Email Address",
|
|
157
|
-
resetMethod: texts.resetMethod || "Reset Method",
|
|
158
|
-
emailCode: texts.emailCode || "Email me a code",
|
|
159
|
-
emailLink: texts.emailLink || "Email me a magic link",
|
|
160
|
-
sendReset: texts.sendReset || "Send Reset",
|
|
161
|
-
back: texts.back || "Back",
|
|
162
|
-
enterResetCode: texts.enterResetCode || "Enter Reset Code",
|
|
163
|
-
weSentCodeTo: texts.weSentCodeTo || "We sent a code to",
|
|
164
|
-
resetCode: texts.resetCode || "Reset Code",
|
|
165
|
-
newPassword: texts.newPassword || "New Password",
|
|
166
|
-
confirmPassword: texts.confirmPassword || "Confirm Password",
|
|
167
|
-
resetPassword: texts.resetPassword || "Reset Password",
|
|
168
|
-
setYourNewPassword: texts.setYourNewPassword || "Set Your New Password",
|
|
169
|
-
setPassword: texts.setPassword || "Set Password",
|
|
170
|
-
invalidCredentials: texts.invalidCredentials || "Invalid credentials.",
|
|
171
|
-
successRedirecting: texts.successRedirecting || "Success! Redirecting...",
|
|
172
|
-
pleaseFillAllFields: texts.pleaseFillAllFields || "Please fill in all fields.",
|
|
173
|
-
passwordsDoNotMatch: texts.passwordsDoNotMatch || "Passwords do not match."
|
|
174
|
-
};
|
|
175
|
-
const HTML = `
|
|
176
|
-
<div class="auth-container">
|
|
177
|
-
<div class="auth-card">
|
|
178
|
-
<div class="auth-header">
|
|
179
|
-
${B2.logoUrl ? `<img src="${B2.logoUrl}" alt="${B2.title}" style="max-height:60px;margin-bottom:10px" />` : ""}
|
|
180
|
-
<h1 class="auth-title">${B2.title}</h1>
|
|
181
|
-
<p class="auth-subtitle">${B2.subtitle}</p>
|
|
182
|
-
</div>
|
|
183
|
-
|
|
184
|
-
<div id="status-message" class="alert" role="status" style="display:none;"></div>
|
|
185
|
-
|
|
186
|
-
<!-- Sign In View -->
|
|
187
|
-
<div id="view-signin" class="auth-view">
|
|
188
|
-
<form id="form-signin" novalidate>
|
|
189
|
-
<div class="mb-3">
|
|
190
|
-
<label for="signin-username" class="form-label">${T.emailOrUsername}</label>
|
|
191
|
-
<input type="text" class="form-control" id="signin-username" placeholder="${T.emailOrUsername}" autocomplete="username" required />
|
|
192
|
-
</div>
|
|
193
|
-
<div class="mb-3">
|
|
194
|
-
<label for="signin-password" class="form-label">${T.password}</label>
|
|
195
|
-
<input type="password" class="form-control" id="signin-password" placeholder="${T.password}" autocomplete="current-password" required />
|
|
196
|
-
</div>
|
|
197
|
-
<button type="submit" class="btn btn-primary w-100 mb-3" id="btn-signin">
|
|
198
|
-
<span class="btn-text">${T.signIn}</span>
|
|
199
|
-
<span class="btn-spinner spinner-border spinner-border-sm" style="display:none;"></span>
|
|
200
|
-
</button>
|
|
201
|
-
<div class="text-center">
|
|
202
|
-
<a href="#" id="link-forgot" class="text-decoration-none">${T.forgotPassword}</a>
|
|
203
|
-
</div>
|
|
204
|
-
|
|
205
|
-
${providers && (providers.google || providers.passkey) ? `
|
|
206
|
-
<div class="position-relative my-3">
|
|
207
|
-
<hr class="text-muted" />
|
|
208
|
-
<span class="position-absolute top-50 start-50 translate-middle bg-white px-3 text-muted small">OR</span>
|
|
209
|
-
</div>
|
|
210
|
-
<div class="d-grid gap-2">
|
|
211
|
-
${providers.google ? `<button type="button" class="btn btn-outline-primary" id="btn-google"><i class="bi bi-google me-2"></i>Continue with Google</button>` : ""}
|
|
212
|
-
${providers.passkey ? `<button type="button" class="btn btn-outline-secondary" id="btn-passkey"><i class="bi bi-fingerprint me-2"></i>Sign in with Passkey</button>` : ""}
|
|
213
|
-
</div>
|
|
214
|
-
` : ""}
|
|
215
|
-
</form>
|
|
216
|
-
</div>
|
|
217
|
-
|
|
218
|
-
<!-- Forgot Password View -->
|
|
219
|
-
<div id="view-forgot" class="auth-view" style="display:none;">
|
|
220
|
-
<button type="button" class="btn btn-link p-0 mb-3" id="btn-back-signin">
|
|
221
|
-
<span aria-hidden="true">←</span> ${T.back}
|
|
222
|
-
</button>
|
|
223
|
-
<h2 class="h5 mb-3">${T.resetYourPassword}</h2>
|
|
224
|
-
<form id="form-forgot" novalidate>
|
|
225
|
-
<div class="mb-3">
|
|
226
|
-
<label for="forgot-email" class="form-label">${T.emailAddress}</label>
|
|
227
|
-
<input type="email" class="form-control" id="forgot-email" placeholder="${T.emailAddress}" autocomplete="email" required />
|
|
228
|
-
</div>
|
|
229
|
-
<div class="mb-3">
|
|
230
|
-
<label class="form-label">${T.resetMethod}</label>
|
|
231
|
-
<div class="form-check">
|
|
232
|
-
<input class="form-check-input" type="radio" name="reset-method" id="method-code" value="code" checked />
|
|
233
|
-
<label class="form-check-label" for="method-code">${T.emailCode}</label>
|
|
234
|
-
</div>
|
|
235
|
-
<div class="form-check">
|
|
236
|
-
<input class="form-check-input" type="radio" name="reset-method" id="method-link" value="link" />
|
|
237
|
-
<label class="form-check-label" for="method-link">${T.emailLink}</label>
|
|
238
|
-
</div>
|
|
239
|
-
</div>
|
|
240
|
-
<button type="submit" class="btn btn-primary w-100" id="btn-forgot">
|
|
241
|
-
<span class="btn-text">${T.sendReset}</span>
|
|
242
|
-
<span class="btn-spinner spinner-border spinner-border-sm" style="display:none;"></span>
|
|
243
|
-
</button>
|
|
244
|
-
</form>
|
|
245
|
-
</div>
|
|
246
|
-
|
|
247
|
-
<!-- Reset with Code View -->
|
|
248
|
-
<div id="view-reset-code" class="auth-view" style="display:none;">
|
|
249
|
-
<button type="button" class="btn btn-link p-0 mb-3" id="btn-back-forgot">
|
|
250
|
-
<span aria-hidden="true">←</span> ${T.back}
|
|
251
|
-
</button>
|
|
252
|
-
<h2 class="h5 mb-3">${T.enterResetCode}</h2>
|
|
253
|
-
<p class="text-muted small mb-3">${T.weSentCodeTo} <strong id="reset-email-display"></strong></p>
|
|
254
|
-
<form id="form-reset-code" novalidate>
|
|
255
|
-
<div class="mb-3">
|
|
256
|
-
<label for="reset-code" class="form-label">${T.resetCode}</label>
|
|
257
|
-
<input type="text" class="form-control" id="reset-code" placeholder="${T.resetCode}" required />
|
|
258
|
-
</div>
|
|
259
|
-
<div class="mb-3">
|
|
260
|
-
<label for="reset-password" class="form-label">${T.newPassword}</label>
|
|
261
|
-
<input type="password" class="form-control" id="reset-password" placeholder="${T.newPassword}" autocomplete="new-password" required />
|
|
262
|
-
</div>
|
|
263
|
-
<div class="mb-3">
|
|
264
|
-
<label for="reset-password-confirm" class="form-label">${T.confirmPassword}</label>
|
|
265
|
-
<input type="password" class="form-control" id="reset-password-confirm" placeholder="${T.confirmPassword}" autocomplete="new-password" required />
|
|
266
|
-
</div>
|
|
267
|
-
<button type="submit" class="btn btn-primary w-100" id="btn-reset-code">
|
|
268
|
-
<span class="btn-text">${T.resetPassword}</span>
|
|
269
|
-
<span class="btn-spinner spinner-border spinner-border-sm" style="display:none;"></span>
|
|
270
|
-
</button>
|
|
271
|
-
</form>
|
|
272
|
-
</div>
|
|
273
|
-
|
|
274
|
-
<!-- Set Password via Magic Link View -->
|
|
275
|
-
<div id="view-set-password" class="auth-view" style="display:none;">
|
|
276
|
-
<h2 class="h5 mb-3">${T.setYourNewPassword}</h2>
|
|
277
|
-
<form id="form-set-password" novalidate>
|
|
278
|
-
<div class="mb-3">
|
|
279
|
-
<label for="set-password" class="form-label">${T.newPassword}</label>
|
|
280
|
-
<input type="password" class="form-control" id="set-password" placeholder="${T.newPassword}" autocomplete="new-password" required />
|
|
281
|
-
</div>
|
|
282
|
-
<div class="mb-3">
|
|
283
|
-
<label for="set-password-confirm" class="form-label">${T.confirmPassword}</label>
|
|
284
|
-
<input type="password" class="form-control" id="set-password-confirm" placeholder="${T.confirmPassword}" autocomplete="new-password" required />
|
|
285
|
-
</div>
|
|
286
|
-
<button type="submit" class="btn btn-primary w-100" id="btn-set-password">
|
|
287
|
-
<span class="btn-text">${T.setPassword}</span>
|
|
288
|
-
<span class="btn-spinner spinner-border spinner-border-sm" style="display:none;"></span>
|
|
289
|
-
</button>
|
|
290
|
-
</form>
|
|
291
|
-
</div>
|
|
292
|
-
</div>
|
|
293
|
-
</div>
|
|
294
|
-
`;
|
|
295
|
-
container.innerHTML = HTML;
|
|
296
|
-
if (theme) {
|
|
297
|
-
container.classList.add(String(theme));
|
|
298
|
-
}
|
|
299
|
-
const els = {
|
|
300
|
-
views: {
|
|
301
|
-
signin: container.querySelector("#view-signin"),
|
|
302
|
-
forgot: container.querySelector("#view-forgot"),
|
|
303
|
-
resetCode: container.querySelector("#view-reset-code"),
|
|
304
|
-
setPassword: container.querySelector("#view-set-password")
|
|
305
|
-
},
|
|
306
|
-
forms: {
|
|
307
|
-
signin: container.querySelector("#form-signin"),
|
|
308
|
-
forgot: container.querySelector("#form-forgot"),
|
|
309
|
-
resetCode: container.querySelector("#form-reset-code"),
|
|
310
|
-
setPassword: container.querySelector("#form-set-password")
|
|
311
|
-
},
|
|
312
|
-
buttons: {
|
|
313
|
-
signin: container.querySelector("#btn-signin"),
|
|
314
|
-
forgot: container.querySelector("#btn-forgot"),
|
|
315
|
-
resetCode: container.querySelector("#btn-reset-code"),
|
|
316
|
-
setPassword: container.querySelector("#btn-set-password"),
|
|
317
|
-
backSignin: container.querySelector("#btn-back-signin"),
|
|
318
|
-
backForgot: container.querySelector("#btn-back-forgot"),
|
|
319
|
-
google: container.querySelector("#btn-google"),
|
|
320
|
-
passkey: container.querySelector("#btn-passkey")
|
|
321
|
-
},
|
|
322
|
-
inputs: {
|
|
323
|
-
signinUsername: container.querySelector("#signin-username"),
|
|
324
|
-
signinPassword: container.querySelector("#signin-password"),
|
|
325
|
-
forgotEmail: container.querySelector("#forgot-email"),
|
|
326
|
-
resetCode: container.querySelector("#reset-code"),
|
|
327
|
-
resetPassword: container.querySelector("#reset-password"),
|
|
328
|
-
resetPasswordConfirm: container.querySelector("#reset-password-confirm"),
|
|
329
|
-
setPassword: container.querySelector("#set-password"),
|
|
330
|
-
setPasswordConfirm: container.querySelector("#set-password-confirm")
|
|
331
|
-
},
|
|
332
|
-
radios: {
|
|
333
|
-
resetMethodCode: container.querySelector("#method-code"),
|
|
334
|
-
resetMethodLink: container.querySelector("#method-link")
|
|
335
|
-
},
|
|
336
|
-
labels: {
|
|
337
|
-
resetEmailDisplay: container.querySelector("#reset-email-display")
|
|
338
|
-
},
|
|
339
|
-
links: {
|
|
340
|
-
forgot: container.querySelector("#link-forgot")
|
|
341
|
-
},
|
|
342
|
-
message: container.querySelector("#status-message")
|
|
343
|
-
};
|
|
344
|
-
function showView(name) {
|
|
345
|
-
Object.entries(els.views).forEach(([key, el]) => {
|
|
346
|
-
if (el) el.style.display = key === name ? "block" : "none";
|
|
347
|
-
});
|
|
348
|
-
setTimeout(() => {
|
|
349
|
-
const view = els.views[name];
|
|
350
|
-
const heading = view?.querySelector("h1, h2, .auth-title, .h5");
|
|
351
|
-
if (heading) {
|
|
352
|
-
heading.setAttribute("tabindex", "-1");
|
|
353
|
-
heading.focus?.();
|
|
354
|
-
} else {
|
|
355
|
-
const firstInput = view?.querySelector("input, button");
|
|
356
|
-
firstInput?.focus?.();
|
|
357
|
-
}
|
|
358
|
-
}, 60);
|
|
359
|
-
}
|
|
360
|
-
function showMessage(message, type = "info") {
|
|
361
|
-
const el = els.message;
|
|
362
|
-
if (!el) return;
|
|
363
|
-
el.textContent = message;
|
|
364
|
-
el.className = `alert alert-${type}`;
|
|
365
|
-
el.style.display = "block";
|
|
366
|
-
el.setAttribute("role", type === "danger" ? "alert" : "status");
|
|
367
|
-
}
|
|
368
|
-
function hideMessage() {
|
|
369
|
-
const el = els.message;
|
|
370
|
-
if (!el) return;
|
|
371
|
-
el.style.display = "none";
|
|
372
|
-
}
|
|
373
|
-
function setButtonLoading(button, loading) {
|
|
374
|
-
if (!button) return;
|
|
375
|
-
const textSpan = button.querySelector(".btn-text");
|
|
376
|
-
const spinner = button.querySelector(".btn-spinner");
|
|
377
|
-
button.disabled = !!loading;
|
|
378
|
-
if (textSpan) textSpan.style.display = loading ? "none" : "inline";
|
|
379
|
-
if (spinner) spinner.style.display = loading ? "inline-block" : "none";
|
|
380
|
-
}
|
|
381
|
-
function getResetMethod() {
|
|
382
|
-
if (els.radios.resetMethodCode?.checked) return "code";
|
|
383
|
-
if (els.radios.resetMethodLink?.checked) return "link";
|
|
384
|
-
return "code";
|
|
385
|
-
}
|
|
386
|
-
async function handleSignin(e) {
|
|
387
|
-
e?.preventDefault?.();
|
|
388
|
-
hideMessage();
|
|
389
|
-
const username = els.inputs.signinUsername?.value?.trim();
|
|
390
|
-
const password = els.inputs.signinPassword?.value;
|
|
391
|
-
if (!username || !password) {
|
|
392
|
-
showMessage("Please enter both username and password.", "danger");
|
|
393
|
-
return;
|
|
394
|
-
}
|
|
395
|
-
setButtonLoading(els.buttons.signin, true);
|
|
396
|
-
try {
|
|
397
|
-
await auth.login(username, password);
|
|
398
|
-
showMessage(`${T.successRedirecting}`, "success");
|
|
399
|
-
setTimeout(performRedirect, 350);
|
|
400
|
-
} catch (err) {
|
|
401
|
-
showMessage(auth.getErrorMessage(err) || T.invalidCredentials, "danger");
|
|
402
|
-
setButtonLoading(els.buttons.signin, false);
|
|
403
|
-
}
|
|
404
|
-
}
|
|
405
|
-
async function handleForgot(e) {
|
|
406
|
-
e?.preventDefault?.();
|
|
407
|
-
hideMessage();
|
|
408
|
-
const email = els.inputs.forgotEmail?.value?.trim();
|
|
409
|
-
const method = getResetMethod();
|
|
410
|
-
if (!email) {
|
|
411
|
-
showMessage("Please enter your email address.", "danger");
|
|
412
|
-
return;
|
|
413
|
-
}
|
|
414
|
-
setButtonLoading(els.buttons.forgot, true);
|
|
415
|
-
try {
|
|
416
|
-
await auth.forgot({ email, method });
|
|
417
|
-
if (method === "code") {
|
|
418
|
-
sessionStorage.setItem("reset_email", email);
|
|
419
|
-
sessionStorage.setItem("reset_method", method);
|
|
420
|
-
if (els.labels.resetEmailDisplay) els.labels.resetEmailDisplay.textContent = email;
|
|
421
|
-
showView("resetCode");
|
|
422
|
-
showMessage("Reset code sent! Check your email.", "success");
|
|
423
|
-
} else {
|
|
424
|
-
showMessage("Magic link sent! Check your email and click the link.", "success");
|
|
425
|
-
}
|
|
426
|
-
} catch (err) {
|
|
427
|
-
showMessage(auth.getErrorMessage(err) || "Something went wrong. Please try again.", "danger");
|
|
428
|
-
} finally {
|
|
429
|
-
setButtonLoading(els.buttons.forgot, false);
|
|
430
|
-
}
|
|
431
|
-
}
|
|
432
|
-
async function handleResetCode(e) {
|
|
433
|
-
e?.preventDefault?.();
|
|
434
|
-
hideMessage();
|
|
435
|
-
const code = els.inputs.resetCode?.value?.trim();
|
|
436
|
-
const newPassword = els.inputs.resetPassword?.value;
|
|
437
|
-
const confirmPassword = els.inputs.resetPasswordConfirm?.value;
|
|
438
|
-
const email = sessionStorage.getItem("reset_email");
|
|
439
|
-
if (!email) {
|
|
440
|
-
showMessage("Session expired. Please restart the password reset process.", "danger");
|
|
441
|
-
showView("forgot");
|
|
442
|
-
return;
|
|
443
|
-
}
|
|
444
|
-
if (!code || !newPassword) {
|
|
445
|
-
showMessage(T.pleaseFillAllFields, "danger");
|
|
446
|
-
return;
|
|
447
|
-
}
|
|
448
|
-
if (newPassword !== confirmPassword) {
|
|
449
|
-
showMessage(T.passwordsDoNotMatch, "danger");
|
|
450
|
-
return;
|
|
451
|
-
}
|
|
452
|
-
setButtonLoading(els.buttons.resetCode, true);
|
|
453
|
-
try {
|
|
454
|
-
await auth.resetWithCode({ email, code, newPassword });
|
|
455
|
-
sessionStorage.removeItem("reset_email");
|
|
456
|
-
sessionStorage.removeItem("reset_method");
|
|
457
|
-
showMessage(T.successRedirecting, "success");
|
|
458
|
-
setTimeout(performRedirect, 350);
|
|
459
|
-
} catch (err) {
|
|
460
|
-
showMessage(auth.getErrorMessage(err) || "Invalid code or code expired.", "danger");
|
|
461
|
-
setButtonLoading(els.buttons.resetCode, false);
|
|
462
|
-
}
|
|
463
|
-
}
|
|
464
|
-
async function handleSetPassword(e) {
|
|
465
|
-
e?.preventDefault?.();
|
|
466
|
-
hideMessage();
|
|
467
|
-
const newPassword = els.inputs.setPassword?.value;
|
|
468
|
-
const confirmPassword = els.inputs.setPasswordConfirm?.value;
|
|
469
|
-
const token = sessionStorage.getItem("login_token");
|
|
470
|
-
if (!token) {
|
|
471
|
-
showMessage("Invalid or expired link. Please request a new one.", "danger");
|
|
472
|
-
showView("forgot");
|
|
473
|
-
return;
|
|
474
|
-
}
|
|
475
|
-
if (!newPassword) {
|
|
476
|
-
showMessage("Please enter a new password.", "danger");
|
|
477
|
-
return;
|
|
478
|
-
}
|
|
479
|
-
if (newPassword !== confirmPassword) {
|
|
480
|
-
showMessage(T.passwordsDoNotMatch, "danger");
|
|
481
|
-
return;
|
|
482
|
-
}
|
|
483
|
-
setButtonLoading(els.buttons.setPassword, true);
|
|
484
|
-
try {
|
|
485
|
-
await auth.resetWithToken({ token, newPassword });
|
|
486
|
-
sessionStorage.removeItem("login_token");
|
|
487
|
-
showMessage(T.successRedirecting, "success");
|
|
488
|
-
setTimeout(performRedirect, 350);
|
|
489
|
-
} catch (err) {
|
|
490
|
-
showMessage(auth.getErrorMessage(err) || "Invalid or expired link.", "danger");
|
|
491
|
-
setButtonLoading(els.buttons.setPassword, false);
|
|
492
|
-
}
|
|
493
|
-
}
|
|
494
|
-
function goToForgot(e) {
|
|
495
|
-
e?.preventDefault?.();
|
|
496
|
-
hideMessage();
|
|
497
|
-
showView("forgot");
|
|
498
|
-
}
|
|
499
|
-
function backToSignin() {
|
|
500
|
-
hideMessage();
|
|
501
|
-
showView("signin");
|
|
502
|
-
}
|
|
503
|
-
function backToForgot() {
|
|
504
|
-
hideMessage();
|
|
505
|
-
showView("forgot");
|
|
506
|
-
}
|
|
507
|
-
function bindProviders() {
|
|
508
|
-
if (providers?.google && els.buttons.google) {
|
|
509
|
-
els.buttons.google.addEventListener("click", (e) => {
|
|
510
|
-
e?.preventDefault?.();
|
|
511
|
-
providers.google.onClick?.({ container, auth, redirect: performRedirect, showMessage });
|
|
512
|
-
});
|
|
513
|
-
}
|
|
514
|
-
if (providers?.passkey && els.buttons.passkey) {
|
|
515
|
-
els.buttons.passkey.addEventListener("click", (e) => {
|
|
516
|
-
e?.preventDefault?.();
|
|
517
|
-
providers.passkey.onClick?.({ container, auth, redirect: performRedirect, showMessage });
|
|
518
|
-
});
|
|
519
|
-
}
|
|
520
|
-
}
|
|
521
|
-
els.forms.signin?.addEventListener("submit", handleSignin);
|
|
522
|
-
els.forms.forgot?.addEventListener("submit", handleForgot);
|
|
523
|
-
els.forms.resetCode?.addEventListener("submit", handleResetCode);
|
|
524
|
-
els.forms.setPassword?.addEventListener("submit", handleSetPassword);
|
|
525
|
-
els.links.forgot?.addEventListener("click", goToForgot);
|
|
526
|
-
els.buttons.backSignin?.addEventListener("click", backToSignin);
|
|
527
|
-
els.buttons.backForgot?.addEventListener("click", backToForgot);
|
|
528
|
-
bindProviders();
|
|
529
|
-
(function init() {
|
|
530
|
-
const params = new URLSearchParams(window.location.search);
|
|
531
|
-
const loginToken = params.get("login_token");
|
|
532
|
-
if (loginToken) {
|
|
533
|
-
sessionStorage.setItem("login_token", loginToken);
|
|
534
|
-
params.delete("login_token");
|
|
535
|
-
const newUrl = params.toString() ? `${window.location.pathname}?${params.toString()}` : window.location.pathname;
|
|
536
|
-
window.history.replaceState({}, "", newUrl);
|
|
537
|
-
showView("setPassword");
|
|
538
|
-
showMessage("Please set your new password.", "info");
|
|
539
|
-
return;
|
|
540
|
-
}
|
|
541
|
-
const resetEmail = sessionStorage.getItem("reset_email");
|
|
542
|
-
if (resetEmail) {
|
|
543
|
-
if (els.labels.resetEmailDisplay) els.labels.resetEmailDisplay.textContent = resetEmail;
|
|
544
|
-
showView("resetCode");
|
|
545
|
-
return;
|
|
546
|
-
}
|
|
547
|
-
showView("signin");
|
|
548
|
-
})();
|
|
549
|
-
return {
|
|
550
|
-
destroy() {
|
|
551
|
-
els.forms.signin?.removeEventListener("submit", handleSignin);
|
|
552
|
-
els.forms.forgot?.removeEventListener("submit", handleForgot);
|
|
553
|
-
els.forms.resetCode?.removeEventListener("submit", handleResetCode);
|
|
554
|
-
els.forms.setPassword?.removeEventListener("submit", handleSetPassword);
|
|
555
|
-
els.links.forgot?.removeEventListener("click", goToForgot);
|
|
556
|
-
els.buttons.backSignin?.removeEventListener("click", backToSignin);
|
|
557
|
-
els.buttons.backForgot?.removeEventListener("click", backToForgot);
|
|
558
|
-
if (providers?.google && els.buttons.google) {
|
|
559
|
-
els.buttons.google.replaceWith(els.buttons.google.cloneNode(true));
|
|
560
|
-
}
|
|
561
|
-
if (providers?.passkey && els.buttons.passkey) {
|
|
562
|
-
els.buttons.passkey.replaceWith(els.buttons.passkey.cloneNode(true));
|
|
563
|
-
}
|
|
564
|
-
container.innerHTML = "";
|
|
565
|
-
}
|
|
566
|
-
};
|
|
567
|
-
}
|
|
568
|
-
const index = {
|
|
569
|
-
mountAuth,
|
|
570
|
-
createAuthClient
|
|
571
|
-
};
|
|
572
|
-
const SimpleAuth = /* @__PURE__ */ Object.freeze(/* @__PURE__ */ Object.defineProperty({
|
|
573
|
-
__proto__: null,
|
|
574
|
-
createAuthClient,
|
|
575
|
-
default: index,
|
|
576
|
-
mountAuth
|
|
577
|
-
}, Symbol.toStringTag, { value: "Module" }));
|
|
578
|
-
export {
|
|
579
|
-
B as BUILD_TIME,
|
|
580
|
-
a as VERSION,
|
|
581
|
-
V as VERSION_INFO,
|
|
582
|
-
b as VERSION_MAJOR,
|
|
583
|
-
c as VERSION_MINOR,
|
|
584
|
-
d as VERSION_REVISION,
|
|
585
|
-
createAuthClient,
|
|
586
|
-
SimpleAuth as default,
|
|
587
|
-
mountAuth
|
|
588
|
-
};
|
|
1
|
+
import{B as e,V as s,a as t,b as n,c as o,d as r}from"./chunks/version-C3dnl1bg.js";function a({baseURL:e,fetchImpl:s=("undefined"!=typeof fetch?fetch.bind(window):null),storage:t=("undefined"!=typeof localStorage?localStorage:null),endpoints:n={}}={}){if(!e)throw new Error("createAuthClient: baseURL is required");if(!s)throw new Error("createAuthClient: fetch implementation is not available in this environment");if(!t)throw new Error("createAuthClient: storage (localStorage) is not available in this environment");const o="access_token",r="refresh_token",a="user",i={login:"/login",forgot:"/auth/forgot",resetCode:"/auth/password/reset/code",resetToken:"/auth/password/reset/token",...n};async function l(t,n){const o=await s(`${e}${t}`,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify(n||{})});let r={};try{r=await o.json()}catch(a){}if(!o.ok)throw r||{message:`Request failed with status ${o.status}`};return r}function d(e){return e&&e.data&&e.data.data||e&&e.data||e}function c(e){const s=d(e);if(!s||!s.access_token)throw new Error("No access_token in response.");t.setItem(o,s.access_token),s.refresh_token&&t.setItem(r,s.refresh_token),s.user&&t.setItem(a,JSON.stringify(s.user))}return{async login(e,s){const t=await l(i.login,{username:e,password:s});return c(t),d(t)},forgot:async({email:e,method:s})=>l(i.forgot,{email:e,method:s}),async resetWithCode({email:e,code:s,newPassword:t}){const n=await l(i.resetCode,{email:e,code:s,new_password:t});return c(n),d(n)},async resetWithToken({token:e,newPassword:s}){const t=await l(i.resetToken,{token:e,new_password:s});return c(t),d(t)},logout(){t.removeItem(o),t.removeItem(r),t.removeItem(a)},isAuthenticated:()=>!!t.getItem(o),getToken:()=>t.getItem(o),getUser(){const e=t.getItem(a);try{return e?JSON.parse(e):null}catch{return null}},getAuthHeader(){const e=t.getItem(o);return e?`Bearer ${e}`:null},getErrorMessage:function(e){return e?.message||e?.error||Array.isArray(e?.errors)&&e.errors[0]?.message||"An error occurred. Please try again."},parseResponse:d}}function i(e,s={}){if(!(e&&e instanceof Element))throw new Error("mountAuth: container must be a DOM Element");const{baseURL:t,onSuccessRedirect:n,allowRedirectOrigins:o,branding:r={},theme:i,endpoints:l,providers:d,texts:c={}}=s;if(!t)throw new Error("mountAuth: baseURL is required");const u=new URLSearchParams(window.location.search),m=u.get("redirect")||u.get("next")||u.get("returnTo"),p=String(n||m||"/");function g(){!function(e){if(!o||0===o.length)return!0;try{const s=new URL(e,window.location.origin);return o.includes(s.origin)}catch{return!1}}(p)?window.location.href="/":window.location.href=p.startsWith("http")?p:new URL(p,window.location.origin).href}const w=a({baseURL:t,endpoints:l}),b=r.title||"Sign In",f=r.subtitle||"Sign in to your account",h=r.logoUrl||"",y={emailOrUsername:c.emailOrUsername||"Email or Username",password:c.password||"Password",signIn:c.signIn||"Sign In",forgotPassword:c.forgotPassword||"Forgot password?",resetYourPassword:c.resetYourPassword||"Reset Your Password",emailAddress:c.emailAddress||"Email Address",resetMethod:c.resetMethod||"Reset Method",emailCode:c.emailCode||"Email me a code",emailLink:c.emailLink||"Email me a magic link",sendReset:c.sendReset||"Send Reset",back:c.back||"Back",enterResetCode:c.enterResetCode||"Enter Reset Code",weSentCodeTo:c.weSentCodeTo||"We sent a code to",resetCode:c.resetCode||"Reset Code",newPassword:c.newPassword||"New Password",confirmPassword:c.confirmPassword||"Confirm Password",resetPassword:c.resetPassword||"Reset Password",setYourNewPassword:c.setYourNewPassword||"Set Your New Password",setPassword:c.setPassword||"Set Password",invalidCredentials:c.invalidCredentials||"Invalid credentials.",successRedirecting:c.successRedirecting||"Success! Redirecting...",pleaseFillAllFields:c.pleaseFillAllFields||"Please fill in all fields.",passwordsDoNotMatch:c.passwordsDoNotMatch||"Passwords do not match."},v=`\n <div class="auth-container">\n <div class="auth-card">\n <div class="auth-header">\n ${h?`<img src="${h}" alt="${b}" style="max-height:60px;margin-bottom:10px" />`:""}\n <h1 class="auth-title">${b}</h1>\n <p class="auth-subtitle">${f}</p>\n </div>\n\n <div id="status-message" class="alert" role="status" style="display:none;"></div>\n\n \x3c!-- Sign In View --\x3e\n <div id="view-signin" class="auth-view">\n <form id="form-signin" novalidate>\n <div class="mb-3">\n <label for="signin-username" class="form-label">${y.emailOrUsername}</label>\n <input type="text" class="form-control" id="signin-username" placeholder="${y.emailOrUsername}" autocomplete="username" required />\n </div>\n <div class="mb-3">\n <label for="signin-password" class="form-label">${y.password}</label>\n <input type="password" class="form-control" id="signin-password" placeholder="${y.password}" autocomplete="current-password" required />\n </div>\n <button type="submit" class="btn btn-primary w-100 mb-3" id="btn-signin">\n <span class="btn-text">${y.signIn}</span>\n <span class="btn-spinner spinner-border spinner-border-sm" style="display:none;"></span>\n </button>\n <div class="text-center">\n <a href="#" id="link-forgot" class="text-decoration-none">${y.forgotPassword}</a>\n </div>\n\n ${d&&(d.google||d.passkey)?`\n <div class="position-relative my-3">\n <hr class="text-muted" />\n <span class="position-absolute top-50 start-50 translate-middle bg-white px-3 text-muted small">OR</span>\n </div>\n <div class="d-grid gap-2">\n ${d.google?'<button type="button" class="btn btn-outline-primary" id="btn-google"><i class="bi bi-google me-2"></i>Continue with Google</button>':""}\n ${d.passkey?'<button type="button" class="btn btn-outline-secondary" id="btn-passkey"><i class="bi bi-fingerprint me-2"></i>Sign in with Passkey</button>':""}\n </div>\n `:""}\n </form>\n </div>\n\n \x3c!-- Forgot Password View --\x3e\n <div id="view-forgot" class="auth-view" style="display:none;">\n <button type="button" class="btn btn-link p-0 mb-3" id="btn-back-signin">\n <span aria-hidden="true">←</span> ${y.back}\n </button>\n <h2 class="h5 mb-3">${y.resetYourPassword}</h2>\n <form id="form-forgot" novalidate>\n <div class="mb-3">\n <label for="forgot-email" class="form-label">${y.emailAddress}</label>\n <input type="email" class="form-control" id="forgot-email" placeholder="${y.emailAddress}" autocomplete="email" required />\n </div>\n <div class="mb-3">\n <label class="form-label">${y.resetMethod}</label>\n <div class="form-check">\n <input class="form-check-input" type="radio" name="reset-method" id="method-code" value="code" checked />\n <label class="form-check-label" for="method-code">${y.emailCode}</label>\n </div>\n <div class="form-check">\n <input class="form-check-input" type="radio" name="reset-method" id="method-link" value="link" />\n <label class="form-check-label" for="method-link">${y.emailLink}</label>\n </div>\n </div>\n <button type="submit" class="btn btn-primary w-100" id="btn-forgot">\n <span class="btn-text">${y.sendReset}</span>\n <span class="btn-spinner spinner-border spinner-border-sm" style="display:none;"></span>\n </button>\n </form>\n </div>\n\n \x3c!-- Reset with Code View --\x3e\n <div id="view-reset-code" class="auth-view" style="display:none;">\n <button type="button" class="btn btn-link p-0 mb-3" id="btn-back-forgot">\n <span aria-hidden="true">←</span> ${y.back}\n </button>\n <h2 class="h5 mb-3">${y.enterResetCode}</h2>\n <p class="text-muted small mb-3">${y.weSentCodeTo} <strong id="reset-email-display"></strong></p>\n <form id="form-reset-code" novalidate>\n <div class="mb-3">\n <label for="reset-code" class="form-label">${y.resetCode}</label>\n <input type="text" class="form-control" id="reset-code" placeholder="${y.resetCode}" required />\n </div>\n <div class="mb-3">\n <label for="reset-password" class="form-label">${y.newPassword}</label>\n <input type="password" class="form-control" id="reset-password" placeholder="${y.newPassword}" autocomplete="new-password" required />\n </div>\n <div class="mb-3">\n <label for="reset-password-confirm" class="form-label">${y.confirmPassword}</label>\n <input type="password" class="form-control" id="reset-password-confirm" placeholder="${y.confirmPassword}" autocomplete="new-password" required />\n </div>\n <button type="submit" class="btn btn-primary w-100" id="btn-reset-code">\n <span class="btn-text">${y.resetPassword}</span>\n <span class="btn-spinner spinner-border spinner-border-sm" style="display:none;"></span>\n </button>\n </form>\n </div>\n\n \x3c!-- Set Password via Magic Link View --\x3e\n <div id="view-set-password" class="auth-view" style="display:none;">\n <h2 class="h5 mb-3">${y.setYourNewPassword}</h2>\n <form id="form-set-password" novalidate>\n <div class="mb-3">\n <label for="set-password" class="form-label">${y.newPassword}</label>\n <input type="password" class="form-control" id="set-password" placeholder="${y.newPassword}" autocomplete="new-password" required />\n </div>\n <div class="mb-3">\n <label for="set-password-confirm" class="form-label">${y.confirmPassword}</label>\n <input type="password" class="form-control" id="set-password-confirm" placeholder="${y.confirmPassword}" autocomplete="new-password" required />\n </div>\n <button type="submit" class="btn btn-primary w-100" id="btn-set-password">\n <span class="btn-text">${y.setPassword}</span>\n <span class="btn-spinner spinner-border spinner-border-sm" style="display:none;"></span>\n </button>\n </form>\n </div>\n </div>\n </div>\n `;e.innerHTML=v,i&&e.classList.add(String(i));const k={views:{signin:e.querySelector("#view-signin"),forgot:e.querySelector("#view-forgot"),resetCode:e.querySelector("#view-reset-code"),setPassword:e.querySelector("#view-set-password")},forms:{signin:e.querySelector("#form-signin"),forgot:e.querySelector("#form-forgot"),resetCode:e.querySelector("#form-reset-code"),setPassword:e.querySelector("#form-set-password")},buttons:{signin:e.querySelector("#btn-signin"),forgot:e.querySelector("#btn-forgot"),resetCode:e.querySelector("#btn-reset-code"),setPassword:e.querySelector("#btn-set-password"),backSignin:e.querySelector("#btn-back-signin"),backForgot:e.querySelector("#btn-back-forgot"),google:e.querySelector("#btn-google"),passkey:e.querySelector("#btn-passkey")},inputs:{signinUsername:e.querySelector("#signin-username"),signinPassword:e.querySelector("#signin-password"),forgotEmail:e.querySelector("#forgot-email"),resetCode:e.querySelector("#reset-code"),resetPassword:e.querySelector("#reset-password"),resetPasswordConfirm:e.querySelector("#reset-password-confirm"),setPassword:e.querySelector("#set-password"),setPasswordConfirm:e.querySelector("#set-password-confirm")},radios:{resetMethodCode:e.querySelector("#method-code"),resetMethodLink:e.querySelector("#method-link")},labels:{resetEmailDisplay:e.querySelector("#reset-email-display")},links:{forgot:e.querySelector("#link-forgot")},message:e.querySelector("#status-message")};function S(e){Object.entries(k.views).forEach(([s,t])=>{t&&(t.style.display=s===e?"block":"none")}),setTimeout(()=>{const s=k.views[e],t=s?.querySelector("h1, h2, .auth-title, .h5");if(t)t.setAttribute("tabindex","-1"),t.focus?.();else{const e=s?.querySelector("input, button");e?.focus?.()}},60)}function P(e,s="info"){const t=k.message;t&&(t.textContent=e,t.className=`alert alert-${s}`,t.style.display="block",t.setAttribute("role","danger"===s?"alert":"status"))}function C(){const e=k.message;e&&(e.style.display="none")}function E(e,s){if(!e)return;const t=e.querySelector(".btn-text"),n=e.querySelector(".btn-spinner");e.disabled=!!s,t&&(t.style.display=s?"none":"inline"),n&&(n.style.display=s?"inline-block":"none")}async function $(e){e?.preventDefault?.(),C();const s=k.inputs.signinUsername?.value?.trim(),t=k.inputs.signinPassword?.value;if(s&&t){E(k.buttons.signin,!0);try{await w.login(s,t),P(`${y.successRedirecting}`,"success"),setTimeout(g,350)}catch(n){P(w.getErrorMessage(n)||y.invalidCredentials,"danger"),E(k.buttons.signin,!1)}}else P("Please enter both username and password.","danger")}async function q(e){e?.preventDefault?.(),C();const s=k.inputs.forgotEmail?.value?.trim(),t=k.radios.resetMethodCode?.checked?"code":k.radios.resetMethodLink?.checked?"link":"code";if(s){E(k.buttons.forgot,!0);try{await w.forgot({email:s,method:t}),"code"===t?(sessionStorage.setItem("reset_email",s),sessionStorage.setItem("reset_method",t),k.labels.resetEmailDisplay&&(k.labels.resetEmailDisplay.textContent=s),S("resetCode"),P("Reset code sent! Check your email.","success")):P("Magic link sent! Check your email and click the link.","success")}catch(n){P(w.getErrorMessage(n)||"Something went wrong. Please try again.","danger")}finally{E(k.buttons.forgot,!1)}}else P("Please enter your email address.","danger")}async function R(e){e?.preventDefault?.(),C();const s=k.inputs.resetCode?.value?.trim(),t=k.inputs.resetPassword?.value,n=k.inputs.resetPasswordConfirm?.value,o=sessionStorage.getItem("reset_email");if(!o)return P("Session expired. Please restart the password reset process.","danger"),void S("forgot");if(s&&t)if(t===n){E(k.buttons.resetCode,!0);try{await w.resetWithCode({email:o,code:s,newPassword:t}),sessionStorage.removeItem("reset_email"),sessionStorage.removeItem("reset_method"),P(y.successRedirecting,"success"),setTimeout(g,350)}catch(r){P(w.getErrorMessage(r)||"Invalid code or code expired.","danger"),E(k.buttons.resetCode,!1)}}else P(y.passwordsDoNotMatch,"danger");else P(y.pleaseFillAllFields,"danger")}async function I(e){e?.preventDefault?.(),C();const s=k.inputs.setPassword?.value,t=k.inputs.setPasswordConfirm?.value,n=sessionStorage.getItem("login_token");if(!n)return P("Invalid or expired link. Please request a new one.","danger"),void S("forgot");if(s)if(s===t){E(k.buttons.setPassword,!0);try{await w.resetWithToken({token:n,newPassword:s}),sessionStorage.removeItem("login_token"),P(y.successRedirecting,"success"),setTimeout(g,350)}catch(o){P(w.getErrorMessage(o)||"Invalid or expired link.","danger"),E(k.buttons.setPassword,!1)}}else P(y.passwordsDoNotMatch,"danger");else P("Please enter a new password.","danger")}function x(e){e?.preventDefault?.(),C(),S("forgot")}function L(){C(),S("signin")}function _(){C(),S("forgot")}return k.forms.signin?.addEventListener("submit",$),k.forms.forgot?.addEventListener("submit",q),k.forms.resetCode?.addEventListener("submit",R),k.forms.setPassword?.addEventListener("submit",I),k.links.forgot?.addEventListener("click",x),k.buttons.backSignin?.addEventListener("click",L),k.buttons.backForgot?.addEventListener("click",_),d?.google&&k.buttons.google&&k.buttons.google.addEventListener("click",s=>{s?.preventDefault?.(),d.google.onClick?.({container:e,auth:w,redirect:g,showMessage:P})}),d?.passkey&&k.buttons.passkey&&k.buttons.passkey.addEventListener("click",s=>{s?.preventDefault?.(),d.passkey.onClick?.({container:e,auth:w,redirect:g,showMessage:P})}),function(){const e=new URLSearchParams(window.location.search),s=e.get("login_token");if(s){sessionStorage.setItem("login_token",s),e.delete("login_token");const t=e.toString()?`${window.location.pathname}?${e.toString()}`:window.location.pathname;return window.history.replaceState({},"",t),S("setPassword"),void P("Please set your new password.","info")}const t=sessionStorage.getItem("reset_email");if(t)return k.labels.resetEmailDisplay&&(k.labels.resetEmailDisplay.textContent=t),void S("resetCode");S("signin")}(),{destroy(){k.forms.signin?.removeEventListener("submit",$),k.forms.forgot?.removeEventListener("submit",q),k.forms.resetCode?.removeEventListener("submit",R),k.forms.setPassword?.removeEventListener("submit",I),k.links.forgot?.removeEventListener("click",x),k.buttons.backSignin?.removeEventListener("click",L),k.buttons.backForgot?.removeEventListener("click",_),d?.google&&k.buttons.google&&k.buttons.google.replaceWith(k.buttons.google.cloneNode(!0)),d?.passkey&&k.buttons.passkey&&k.buttons.passkey.replaceWith(k.buttons.passkey.cloneNode(!0)),e.innerHTML=""}}}const l={mountAuth:i,createAuthClient:a},d=/* @__PURE__ */Object.freeze(/* @__PURE__ */Object.defineProperty({__proto__:null,createAuthClient:a,default:l,mountAuth:i},Symbol.toStringTag,{value:"Module"}));export{e as BUILD_TIME,s as VERSION,t as VERSION_INFO,n as VERSION_MAJOR,o as VERSION_MINOR,r as VERSION_REVISION,a as createAuthClient,d as default,i as mountAuth};
|
|
589
2
|
//# sourceMappingURL=auth.es.js.map
|