forgot-password.php 4.2 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. <?php
  2. require_once __DIR__ . '/../config/database.php';
  3. require_once __DIR__ . '/../lib/auth.php';
  4. require_once __DIR__ . '/../lib/csrf.php';
  5. if (isLoggedIn()) {
  6. header('Location: /dashboard/dashboard.php');
  7. exit;
  8. }
  9. $sent = false;
  10. $error = '';
  11. if ($_SERVER['REQUEST_METHOD'] === 'POST') {
  12. if (!verifyCsrfToken($_POST['csrf_token'] ?? '')) {
  13. $error = 'Invalid request. Please try again.';
  14. } else {
  15. $email = trim($_POST['email'] ?? '');
  16. if ($email === '' || !filter_var($email, FILTER_VALIDATE_EMAIL)) {
  17. $error = 'Please enter a valid email address.';
  18. } else {
  19. // createPasswordResetToken returns null if email not found,
  20. // but we show the same success message to avoid email enumeration.
  21. $token = createPasswordResetToken($email);
  22. if ($token !== null) {
  23. // TODO: send email with reset link.
  24. // Reset link: /login/reset-password.php?token={$token}
  25. // Until SMTP is configured, the token is logged for development.
  26. error_log("Password reset token for {$email}: {$token}");
  27. }
  28. $sent = true; // Always show success to prevent email enumeration
  29. }
  30. }
  31. }
  32. $pageTitle = 'Forgot Password';
  33. include __DIR__ . '/_head.php';
  34. ?>
  35. <div class="container">
  36. <div class="row justify-content-center">
  37. <div class="col-xl-10 col-lg-12 col-md-9">
  38. <div class="card o-hidden border-0 shadow-lg my-5">
  39. <div class="card-body p-0">
  40. <div class="row">
  41. <div class="col-lg-6 d-none d-lg-block bg-login-image"></div>
  42. <div class="col-lg-6">
  43. <div class="p-5">
  44. <div class="text-center mb-4">
  45. <h1 class="h4 text-gray-900">Forgot Your Password?</h1>
  46. <p class="text-muted small">Enter your email and we'll send you a reset link.</p>
  47. </div>
  48. <?php if ($sent): ?>
  49. <div class="alert alert-success" role="alert">
  50. If that email is registered, a reset link has been sent. Please check your inbox.
  51. </div>
  52. <?php else: ?>
  53. <?php if ($error !== ''): ?>
  54. <div class="alert alert-danger"><?= htmlspecialchars($error, ENT_QUOTES, 'UTF-8') ?></div>
  55. <?php endif; ?>
  56. <form method="POST" action="/login/forgot-password.php" novalidate>
  57. <input type="hidden" name="csrf_token" value="<?= generateCsrfToken() ?>">
  58. <div class="mb-3">
  59. <input type="email" name="email" class="form-control form-control-user"
  60. placeholder="Email Address" required autofocus
  61. value="<?= htmlspecialchars($_POST['email'] ?? '', ENT_QUOTES, 'UTF-8') ?>">
  62. </div>
  63. <button type="submit" class="btn btn-success btn-user w-100 mb-3">
  64. Send Reset Link
  65. </button>
  66. </form>
  67. <?php endif; ?>
  68. <hr>
  69. <div class="text-center">
  70. <a class="small" href="/login/register.php">Create an Account!</a>
  71. </div>
  72. <div class="text-center">
  73. <a class="small" href="/login/login.php">Already have an account? Login!</a>
  74. </div>
  75. </div>
  76. </div>
  77. </div>
  78. </div>
  79. </div>
  80. </div>
  81. </div>
  82. </div>
  83. <?php include __DIR__ . '/_foot.php'; ?>