Twan07's picture
Upload 15 files
9ff1fd4 verified
raw
history blame
9.78 kB
import { render } from 'solid-js/web';
import { createSignal, Show, onMount } from 'solid-js';
const IconEye = () => <svg width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M1 12s4-8 11-8 11 8 11 8-4 8-11 8-11-8-11-8z"></path><circle cx="12" cy="12" r="3"></circle></svg>;
const IconEyeOff = () => <svg width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M17.94 17.94A10.07 10.07 0 0 1 12 20c-7 0-11-8-11-8a18.45 18.45 0 0 1 5.06-5.94M9.9 4.24A9.12 9.12 0 0 1 12 4c7 0 11 8 11 8a18.5 18.5 0 0 1-2.16 3.19m-6.72-1.07a3 3 0 1 1-4.24-4.24"></path><line x1="1" y1="1" x2="23" y2="23"></line></svg>;
function App() {
const [step, setStep] = createSignal(1);
const [loading, setLoading] = createSignal(false);
const [status, setStatus] = createSignal('');
const [identifier, setIdentifier] = createSignal('');
const [otpCode, setOtpCode] = createSignal('');
const [newPass, setNewPass] = createSignal('');
const [showPass, setShowPass] = createSignal(false);
async function handleSubmit(e) {
e.preventDefault();
setLoading(true);
setStatus('');
let body = {};
let action = '';
let nextStep = 0;
if (step() === 1) {
if (!identifier()) {
setStatus('Please enter your username or email.');
setLoading(false);
return;
}
action = 'SendOtp';
body = { identifier: identifier(), action };
nextStep = 2;
} else if (step() === 2) {
if (!otpCode() || !newPass()) {
setStatus('Please fill in both the OTP and your new password.');
setLoading(false);
return;
}
action = 'ResetPassword';
body = { identifier: identifier(), action, otpCode: otpCode(), newPass: newPass() };
}
try {
const res = await fetch('/private/server/exocore/web/forgotpass', {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify(body),
});
const data = await res.json();
if (!res.ok) throw new Error(data.error || 'Request failed.');
setStatus(data.data.message || 'Success.');
if (nextStep) {
setStep(nextStep);
} else {
setTimeout(() => {
window.location.href = '/private/server/exocore/web/public/login';
}, 2000);
}
} catch (err) {
setStatus(err.message);
} finally {
setLoading(false);
}
}
return (
<div class="otp-page-wrapper">
<style>{`
:root {
--bg-primary: #111217; --bg-secondary: #1a1b23; --text-primary: #e0e0e0;
--text-secondary: #8a8f98; --accent-primary: #00aaff; --accent-secondary: #0088cc;
--border-color: rgba(255, 255, 255, 0.1); --shadow-color: rgba(0, 0, 0, 0.5);
--radius-main: 16px; --radius-inner: 12px;
--font-body: 'Roboto', sans-serif;
--success-color: #2ecc71; --error-color: #e74c3c;
}
@import url('https://fonts.googleapis.com/css2?family=Roboto:wght@400;500;700&display=swap');
body { background-color: var(--bg-primary); font-family: var(--font-body); margin: 0; }
.otp-page-wrapper { display: flex; justify-content: center; align-items: center; min-height: 100vh; padding: 1rem; box-sizing: border-box; }
.otp-card { background: var(--bg-secondary); width: 100%; max-width: 420px; padding: 2.5rem; border-radius: var(--radius-main); border: 1px solid var(--border-color); box-shadow: 0 15px 40px var(--shadow-color); animation: fadeIn 0.5s ease-out; text-align: center; }
@keyframes fadeIn { from { opacity: 0; transform: translateY(10px); } to { opacity: 1; transform: translateY(0); } }
.otp-header { margin-bottom: 0.75rem; color: var(--text-primary); font-size: 1.75rem; font-weight: 700; }
.otp-subtext { color: var(--text-secondary); margin-bottom: 2rem; line-height: 1.5; font-size: 0.95rem; }
.form-group { margin-bottom: 1.25rem; text-align: left; }
.form-label { display: block; margin-bottom: 0.5rem; color: var(--text-secondary); font-weight: 500; }
.form-input { width: 100%; padding: 0.8rem 1rem; border: 1px solid var(--border-color); border-radius: var(--radius-inner); font-family: var(--font-body); font-size: 1rem; background-color: var(--bg-primary); color: var(--text-primary); box-sizing: border-box; transition: border-color 0.2s, box-shadow 0.2s; }
.form-input:focus { outline:0; border-color: var(--accent-primary); box-shadow: 0 0 0 3px rgba(0, 170, 255, 0.2); }
.otp-input { text-align: center; letter-spacing: 0.5em; font-size: 1.5rem !important; }
.otp-input::placeholder { letter-spacing: normal; }
.input-wrapper { position: relative; display: flex; align-items: center; }
.password-toggle { position: absolute; right: 0.5rem; background: none; border: none; color: var(--text-secondary); cursor: pointer; display: flex; align-items: center; padding: 0.5rem; border-radius: 50%; }
.password-toggle:hover { color: var(--accent-primary); }
.btn { width: 100%; padding: 0.9rem 1.5rem; border: none; border-radius: var(--radius-inner); background: linear-gradient(to right, var(--accent-primary), var(--accent-secondary)); color: #fff; font-size: 1.1rem; font-weight: 700; cursor: pointer; transition: all 0.2s ease; }
.btn:disabled { opacity: 0.6; cursor: not-allowed; }
.status-message { text-align: center; margin-top: 1.5rem; padding: 0.8rem; border-radius: var(--radius-inner); font-weight: 500; }
.status-success { background-color: rgba(46, 204, 113, 0.15); color: var(--success-color); border: 1px solid var(--success-color); }
.status-error { background-color: rgba(231, 76, 60, 0.15); color: var(--error-color); border: 1px solid var(--error-color); }
.login-link-wrapper { text-align: center; margin-top: 1.5rem; color: var(--text-secondary); font-size: 0.9rem; }
.login-link { color: var(--accent-primary); text-decoration: none; font-weight: 500; }
`}</style>
<div class="otp-card">
<form onSubmit={handleSubmit}>
<h1 class="otp-header">Forgot Password</h1>
<Show when={step() === 1}
fallback={
<>
<p class="otp-subtext">A code was sent. Enter it below with your new password.</p>
<div class="form-group">
<label class="form-label" for="otpCode">Verification Code</label>
<input id="otpCode" class="form-input otp-input" type="text" value={otpCode()} onInput={e => setOtpCode(e.currentTarget.value.replace(/\s/g, ''))} placeholder="------" maxlength="6"/>
</div>
<div class="form-group">
<label class="form-label" for="newPass">New Password</label>
<div class="input-wrapper">
<input id="newPass" class="form-input" type={showPass() ? 'text' : 'password'} value={newPass()} onInput={e => setNewPass(e.currentTarget.value.replace(/\s/g, ''))} />
<button type="button" class="password-toggle" onClick={() => setShowPass(!showPass())}>
<Show when={showPass()} fallback={<IconEye />}><IconEyeOff /></Show>
</button>
</div>
</div>
<button class="btn btn-primary" type="submit" disabled={loading() || !otpCode() || !newPass()}>{loading() ? 'Resetting...' : 'Reset Password'}</button>
</>
}
>
<p class="otp-subtext">Enter your username or email to receive a verification code.</p>
<div class="form-group">
<label class="form-label" for="identifier">Username or Email</label>
<input id="identifier" class="form-input" type="text" value={identifier()} onInput={e => setIdentifier(e.currentTarget.value.replace(/\s/g, ''))} />
</div>
<button class="btn btn-primary" type="submit" disabled={loading()}>{loading() ? 'Sending...' : 'Send Code'}</button>
</Show>
</form>
<Show when={status()}>
<div class={`status-message ${status().toLowerCase().includes('success') ? 'status-success' : 'status-error'}`}>{status()}</div>
</Show>
<div class="login-link-wrapper">
Remembered your password? <a href="/private/server/exocore/web/public/login" class="login-link">Login here</a>
</div>
</div>
</div>
);
}
render(() => <App />, document.getElementById('app'));