login.php 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081
  1. <?php
  2. require_once __DIR__ . '/auth.php';
  3. if (ADMIN_LOGGED_IN) {
  4. header('Location: index.php');
  5. exit;
  6. }
  7. $error = '';
  8. if ($_SERVER['REQUEST_METHOD'] === 'POST') {
  9. $user = trim($_POST['username'] ?? '');
  10. $pass = $_POST['password'] ?? '';
  11. if ($user === '' || $pass === '') {
  12. $error = 'Please enter username and password.';
  13. } else {
  14. $pdo = get_pdo();
  15. $stmt = $pdo->prepare("SELECT password_hash FROM admin_users WHERE username = ?");
  16. $stmt->execute([$user]);
  17. $row = $stmt->fetch();
  18. if ($row && password_verify($pass, $row['password_hash'])) {
  19. session_regenerate_id(true);
  20. $_SESSION['admin_logged_in'] = true;
  21. $_SESSION['admin_user'] = $user;
  22. header('Location: index.php');
  23. exit;
  24. } else {
  25. $error = 'Invalid username or password.';
  26. // Slow down brute-force attempts
  27. sleep(1);
  28. }
  29. }
  30. }
  31. ?>
  32. <!DOCTYPE html>
  33. <html lang="en">
  34. <head>
  35. <meta charset="UTF-8">
  36. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  37. <title>PackIt Admin – Login</title>
  38. <link rel="preconnect" href="https://fonts.googleapis.com">
  39. <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
  40. <link href="https://fonts.googleapis.com/css2?family=Bebas+Neue&family=DM+Sans:wght@300;400;500;600&display=swap" rel="stylesheet">
  41. <style>
  42. :root{--bg:#0f1117;--surface:#181c27;--border:#2a2f3f;--accent:#e8c547;--accent2:#4fd1c5;--text:#e8eaf0;--muted:#7a8099;--danger:#e05555;--radius:10px;}
  43. *,*::before,*::after{box-sizing:border-box;margin:0;padding:0;}
  44. body{background:var(--bg);color:var(--text);font-family:'DM Sans',sans-serif;min-height:100vh;display:flex;align-items:center;justify-content:center;}
  45. body::before{content:'';position:fixed;inset:0;background:radial-gradient(ellipse 60% 60% at 50% 40%, rgba(232,197,71,.05) 0%, transparent 70%);pointer-events:none;}
  46. .login-box{background:var(--surface);border:1px solid var(--border);border-radius:var(--radius);padding:2.5rem 2rem;width:100%;max-width:360px;position:relative;}
  47. .login-logo{font-family:'Bebas Neue',sans-serif;font-size:2.8rem;color:var(--accent);letter-spacing:.05em;margin-bottom:.25rem;}
  48. .login-sub{font-size:.78rem;color:var(--muted);text-transform:uppercase;letter-spacing:.1em;margin-bottom:2rem;}
  49. label{display:block;font-size:.75rem;font-weight:600;color:var(--muted);text-transform:uppercase;letter-spacing:.08em;margin-bottom:.3rem;}
  50. input{display:block;width:100%;background:var(--bg);border:1.5px solid var(--border);color:var(--text);padding:.6rem .85rem;border-radius:var(--radius);font-family:'DM Sans',sans-serif;font-size:.9rem;outline:none;margin-bottom:1rem;transition:border-color .15s;}
  51. input:focus{border-color:var(--accent2);}
  52. .btn-login{width:100%;background:var(--accent);color:#0f1117;border:none;padding:.75rem;border-radius:var(--radius);font-family:'DM Sans',sans-serif;font-size:.95rem;font-weight:600;cursor:pointer;margin-top:.5rem;transition:background .15s;}
  53. .btn-login:hover{background:#f0d260;}
  54. .error-msg{background:rgba(224,85,85,.1);border:1px solid rgba(224,85,85,.3);color:var(--danger);padding:.65rem .9rem;border-radius:var(--radius);font-size:.85rem;margin-bottom:1rem;}
  55. </style>
  56. </head>
  57. <body>
  58. <div class="login-box">
  59. <div class="login-logo">PackIt</div>
  60. <div class="login-sub">Admin Access</div>
  61. <?php if ($error): ?>
  62. <div class="error-msg"><?= htmlspecialchars($error) ?></div>
  63. <?php endif; ?>
  64. <form method="POST">
  65. <label for="username">Username</label>
  66. <input type="text" id="username" name="username"
  67. value="<?= htmlspecialchars($_POST['username'] ?? '') ?>"
  68. autocomplete="username" required>
  69. <label for="password">Password</label>
  70. <input type="password" id="password" name="password"
  71. autocomplete="current-password" required>
  72. <button class="btn-login" type="submit">Sign in</button>
  73. </form>
  74. </div>
  75. </body>
  76. </html>