338 lines
16 KiB
PHP
338 lines
16 KiB
PHP
<?php
|
|
require_once __DIR__ . '/../Logic/Backend/Authentication_Handler.php';
|
|
require_once __DIR__ . '/../db/config.php';
|
|
|
|
$Auth = new Authentication_Handler();
|
|
$Auth->Check_Auth();
|
|
|
|
if ($_SESSION['Access_Level'] < 1) {
|
|
header('Location: Voting_Screen.php');
|
|
exit;
|
|
}
|
|
|
|
$db = db();
|
|
|
|
// Handle New Election Creation
|
|
if ($_SERVER['REQUEST_METHOD'] === 'POST' && isset($_POST['create_election'])) {
|
|
$year = $_POST['election_year'] ?? date('Y');
|
|
|
|
// Default configuration for a new election
|
|
$defaultParties = json_encode(['Independent']);
|
|
$defaultPositions = json_encode([
|
|
['name' => 'President', 'type' => 'Uniform'],
|
|
['name' => 'Vice President', 'type' => 'Uniform'],
|
|
['name' => 'Secretary', 'type' => 'Uniform'],
|
|
['name' => 'Treasurer', 'type' => 'Uniform']
|
|
]);
|
|
|
|
$startDate = date('Y-m-d H:i:s');
|
|
$endDate = date('Y-m-d H:i:s', strtotime('+7 days'));
|
|
|
|
try {
|
|
$stmt = $db->prepare("INSERT INTO Election_History (Year, Parties, Positions, Status, Start_Date, End_Date, Total_Voters) VALUES (?, ?, ?, ?, ?, ?, ?)");
|
|
$stmt->execute([$year, $defaultParties, $defaultPositions, 'Preparing', $startDate, $endDate, 0]);
|
|
|
|
$Auth->Log_Action($_SESSION['User_ID'], $_SESSION['User_Role'], 'Create New Election', "Created a new election for School Year $year");
|
|
|
|
header("Location: Election_Dashboard.php?success=New election created successfully");
|
|
exit;
|
|
} catch (Exception $e) {
|
|
header("Location: Election_Dashboard.php?error=Error creating election: " . $e->getMessage());
|
|
exit;
|
|
}
|
|
}
|
|
|
|
// Handle Status Change
|
|
if ($_SERVER['REQUEST_METHOD'] === 'POST' && isset($_POST['update_status'])) {
|
|
$newStatus = $_POST['new_status'];
|
|
$electionId = $_POST['election_id'];
|
|
|
|
// Fetch current status
|
|
$currentElectionStmt = $db->prepare("SELECT * FROM Election_History WHERE Election_ID = ?");
|
|
$currentElectionStmt->execute([$electionId]);
|
|
$electionData = $currentElectionStmt->fetch();
|
|
|
|
if ($newStatus === 'Finished') {
|
|
// Save current details to history record
|
|
$totalVoters = $db->query("SELECT COUNT(*) FROM Voters")->fetchColumn();
|
|
$votedCount = $db->query("SELECT COUNT(*) FROM Voters WHERE Has_Voted = 1")->fetchColumn();
|
|
|
|
// Fetch Candidates and potentially calculate results if there was a voting table
|
|
$candidates = $db->query("SELECT Name, Position, Party FROM Candidates")->fetchAll(PDO::FETCH_ASSOC);
|
|
|
|
$stmt = $db->prepare("UPDATE Election_History SET Status = ?, Total_Voters = ?, Results = ? WHERE Election_ID = ?");
|
|
$stmt->execute([$newStatus, $totalVoters, json_encode($candidates), $electionId]);
|
|
} else {
|
|
$stmt = $db->prepare("UPDATE Election_History SET Status = ? WHERE Election_ID = ?");
|
|
$stmt->execute([$newStatus, $electionId]);
|
|
}
|
|
|
|
// Log to Audit Trail
|
|
$year = $electionData['Year'] ?? 'Unknown';
|
|
$oldStatus = $electionData['Status'] ?? 'Unknown';
|
|
$Auth->Log_Action($_SESSION['User_ID'], $_SESSION['User_Role'], 'Update Election Status', "Changed SY $year Election status from $oldStatus to $newStatus");
|
|
|
|
header("Location: Election_Dashboard.php");
|
|
exit;
|
|
}
|
|
|
|
// Fetch Stats
|
|
$totalVoters = $db->query("SELECT COUNT(*) FROM Voters")->fetchColumn();
|
|
$totalCandidates = $db->query("SELECT COUNT(*) FROM Candidates")->fetchColumn();
|
|
$totalVotes = $db->query("SELECT COUNT(*) FROM Voters WHERE Has_Voted = 1")->fetchColumn();
|
|
|
|
// Fetch Active/Preparing/Ongoing Elections
|
|
$activeElections = $db->query("SELECT * FROM Election_History WHERE Status IN ('Preparing', 'Ongoing', 'Active') ORDER BY Start_Date DESC")->fetchAll();
|
|
|
|
?>
|
|
<!DOCTYPE html>
|
|
<html lang="en">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
<title>Election Dashboard | Online School Election System</title>
|
|
<link rel="stylesheet" href="../Design/Style.css?v=<?php echo time(); ?>">
|
|
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css">
|
|
<style>
|
|
.Status_Badge {
|
|
padding: 4px 12px;
|
|
border-radius: 20px;
|
|
font-size: 0.75rem;
|
|
font-weight: 700;
|
|
text-transform: uppercase;
|
|
}
|
|
.Status_Active, .Status_Ongoing { background: #DCFCE7; color: #166534; }
|
|
.Status_Preparing { background: #FEF3C7; color: #92400E; }
|
|
.Status_Completed, .Status_Finished { background: #F1F5F9; color: #475569; }
|
|
|
|
/* Modal Styles */
|
|
.Modal {
|
|
display: none;
|
|
position: fixed;
|
|
top: 0;
|
|
left: 0;
|
|
width: 100%;
|
|
height: 100%;
|
|
background: rgba(0,0,0,0.5);
|
|
z-index: 1000;
|
|
justify-content: center;
|
|
align-items: center;
|
|
backdrop-filter: blur(4px);
|
|
}
|
|
.Modal_Content {
|
|
background: white;
|
|
padding: 32px;
|
|
border-radius: 20px;
|
|
width: 90%;
|
|
max-width: 400px;
|
|
position: relative;
|
|
box-shadow: 0 20px 25px -5px rgba(0,0,0,0.1);
|
|
}
|
|
.Modal_Header {
|
|
display: flex;
|
|
justify-content: space-between;
|
|
align-items: center;
|
|
margin-bottom: 24px;
|
|
}
|
|
.Modal_Title {
|
|
font-size: 1.25rem;
|
|
font-weight: 700;
|
|
}
|
|
.Modal_Close {
|
|
cursor: pointer;
|
|
font-size: 1.25rem;
|
|
color: var(--Text_Secondary);
|
|
}
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<div class="Dashboard_Container">
|
|
<!-- Sidebar -->
|
|
<aside class="Sidebar">
|
|
<div class="Sidebar_Header">
|
|
<div class="Logo_Text">
|
|
<div class="Logo_Title" style="font-size: 1.1rem; color: var(--Primary_Color);">Click to Vote</div>
|
|
<div class="Logo_Subtitle">Administrator Portal</div>
|
|
</div>
|
|
</div>
|
|
<nav class="Sidebar_Nav">
|
|
<a href="Election_Dashboard.php" class="Nav_Item Active">
|
|
<i class="fas fa-th-large"></i> Election Dashboard
|
|
</a>
|
|
<a href="Election_History.php" class="Nav_Item">
|
|
<i class="fas fa-vote-yea"></i> Election History
|
|
</a>
|
|
<a href="Voters.php" class="Nav_Item">
|
|
<i class="fas fa-users"></i> Voter Management
|
|
</a>
|
|
<a href="Candidates.php" class="Nav_Item">
|
|
<i class="fas fa-user-tie"></i> Candidate Management
|
|
</a>
|
|
<a href="Officers.php" class="Nav_Item">
|
|
<i class="fas fa-user-shield"></i> Officers Management
|
|
</a>
|
|
<a href="Audit_Trail.php" class="Nav_Item">
|
|
<i class="fas fa-file-alt"></i> Reports & Audit
|
|
</a>
|
|
<a href="Credits.php" class="Nav_Item">
|
|
<i class="fas fa-info-circle"></i> Credits
|
|
</a>
|
|
</nav>
|
|
<div style="padding: 24px; border-top: 1px solid var(--Border_Color);">
|
|
<a href="Logout.php" class="Nav_Item" style="padding: 0; color: var(--Error_Color);">
|
|
<i class="fas fa-sign-out-alt"></i> Logout
|
|
</a>
|
|
</div>
|
|
</aside>
|
|
|
|
<!-- Main Content -->
|
|
<main class="Main_Content">
|
|
<!-- Top Bar -->
|
|
<header class="Top_Bar">
|
|
<div class="Search_Box">
|
|
<i class="fas fa-search" style="color: var(--Secondary_Color);"></i>
|
|
<input type="text" placeholder="Search for voters, candidates, or records...">
|
|
</div>
|
|
<div class="Top_Bar_Actions">
|
|
<div style="display: flex; align-items: center; gap: 12px;">
|
|
<div style="text-align: right;">
|
|
<div style="font-weight: 700; font-size: 0.9rem;"><?php echo htmlspecialchars($_SESSION['User_Name']); ?></div>
|
|
<div style="font-size: 0.75rem; color: var(--Text_Secondary);"><?php echo htmlspecialchars($_SESSION['User_Role']); ?></div>
|
|
</div>
|
|
<div style="width: 40px; height: 40px; background: var(--Primary_Light); color: var(--Primary_Color); border-radius: 50%; display: flex; justify-content: center; align-items: center; font-weight: 700;">
|
|
<?php echo strtoupper(substr($_SESSION['User_Name'] ?? 'A', 0, 1)); ?>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</header>
|
|
|
|
<!-- Content Body -->
|
|
<div class="Content_Body">
|
|
<?php if (isset($_GET['success'])): ?>
|
|
<div class="Badge Badge_Success" style="width: 100%; padding: 15px; margin-bottom: 20px; text-align: center;">
|
|
<i class="fas fa-check-circle"></i> <?php echo htmlspecialchars($_GET['success']); ?>
|
|
</div>
|
|
<?php endif; ?>
|
|
<?php if (isset($_GET['error'])): ?>
|
|
<div class="Badge" style="width: 100%; padding: 15px; margin-bottom: 20px; text-align: center; background: #FEE2E2; color: #EF4444;">
|
|
<i class="fas fa-exclamation-circle"></i> <?php echo htmlspecialchars($_GET['error']); ?>
|
|
</div>
|
|
<?php endif; ?>
|
|
|
|
<div class="Page_Title_Section">
|
|
<h1 class="Page_Title">Election Dashboard</h1>
|
|
<p class="Page_Subtitle">Welcome back! Here's what's happening with the current elections.</p>
|
|
</div>
|
|
|
|
<!-- Stat Cards -->
|
|
<div class="Stat_Cards_Grid">
|
|
<div class="Stat_Card">
|
|
<div class="Stat_Label">Total Voters</div>
|
|
<div class="Stat_Value"><?php echo number_format($totalVoters); ?></div>
|
|
<div class="Stat_Trend Trend_Up"><i class="fas fa-users"></i> Registered Students</div>
|
|
</div>
|
|
<div class="Stat_Card">
|
|
<div class="Stat_Label">Total Candidates</div>
|
|
<div class="Stat_Value"><?php echo number_format($totalCandidates); ?></div>
|
|
<div class="Stat_Trend" style="color: var(--Primary_Color);"><i class="fas fa-user-tie"></i> Running for Office</div>
|
|
</div>
|
|
<div class="Stat_Card">
|
|
<div class="Stat_Label">Total Votes Cast</div>
|
|
<div class="Stat_Value"><?php echo number_format($totalVotes); ?></div>
|
|
<div class="Stat_Trend Trend_Up"><i class="fas fa-check-circle"></i> Verified Ballots</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Dashboard Grid -->
|
|
<div class="Dashboard_Grid">
|
|
<!-- Main Content: Active Elections -->
|
|
<div class="Card">
|
|
<div class="Card_Header">
|
|
<h2 class="Card_Title">Active & Upcoming Elections</h2>
|
|
<button onclick="openNewElectionModal()" class="Button_Primary" style="width: auto; padding: 8px 16px; font-size: 0.85rem;">
|
|
<i class="fas fa-plus"></i> New Election
|
|
</button>
|
|
</div>
|
|
<div class="Table_Wrapper">
|
|
<table>
|
|
<thead>
|
|
<tr>
|
|
<th>Election Title</th>
|
|
<th>Period</th>
|
|
<th>Status</th>
|
|
<th>Quick Actions</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<?php if (empty($activeElections)): ?>
|
|
<tr>
|
|
<td colspan="4" style="text-align: center; padding: 40px; color: var(--Text_Secondary);">
|
|
No active elections found. <a href="javascript:void(0)" onclick="openNewElectionModal()" style="color: var(--Primary_Color); font-weight: 600;">Create one now.</a>
|
|
</td>
|
|
</tr>
|
|
<?php else: ?>
|
|
<?php foreach ($activeElections as $election): ?>
|
|
<tr>
|
|
<td style="font-weight: 600;">School Year <?php echo $election['Year']; ?> Election</td>
|
|
<td><?php echo date('M d', strtotime($election['Start_Date'])); ?> - <?php echo date('M d, Y', strtotime($election['End_Date'])); ?></td>
|
|
<td>
|
|
<span class="Status_Badge Status_<?php echo $election['Status']; ?>">
|
|
<?php echo $election['Status']; ?>
|
|
</span>
|
|
</td>
|
|
<td>
|
|
<form method="POST" style="display: inline-flex; gap: 8px; align-items: center;">
|
|
<input type="hidden" name="election_id" value="<?php echo $election['Election_ID']; ?>">
|
|
<select name="new_status" class="Select" style="font-size: 0.75rem; padding: 4px 8px;">
|
|
<option value="Preparing" <?php echo $election['Status'] == 'Preparing' ? 'selected' : ''; ?>>Preparing</option>
|
|
<option value="Ongoing" <?php echo $election['Status'] == 'Ongoing' ? 'selected' : ''; ?>>Ongoing</option>
|
|
<option value="Finished" <?php echo $election['Status'] == 'Finished' ? 'selected' : ''; ?>>Finished</option>
|
|
</select>
|
|
<button type="submit" name="update_status" class="Button_Primary" style="width: auto; padding: 6px 10px; margin-top: 0; font-size: 0.7rem;">Update</button>
|
|
</form>
|
|
</td>
|
|
</tr>
|
|
<?php endforeach; ?>
|
|
<?php endif; ?>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</main>
|
|
</div>
|
|
|
|
<!-- New Election Modal -->
|
|
<div id="NewElectionModal" class="Modal">
|
|
<div class="Modal_Content">
|
|
<div class="Modal_Header">
|
|
<h2 class="Modal_Title">Create New Election</h2>
|
|
<span class="Modal_Close" onclick="closeNewElectionModal()">×</span>
|
|
</div>
|
|
<form method="POST">
|
|
<div class="Form_Group">
|
|
<label class="Label">School Year</label>
|
|
<input type="number" name="election_year" class="Input" value="<?php echo date('Y'); ?>" min="2000" max="2100" required>
|
|
<p class="Text_Muted" style="font-size: 0.75rem; margin-top: 5px;">Example: 2026 for SY 2026-2027</p>
|
|
</div>
|
|
<button type="submit" name="create_election" class="Button_Primary" style="width: 100%; margin-top: 24px;">Initialize Election</button>
|
|
</form>
|
|
</div>
|
|
</div>
|
|
|
|
<script>
|
|
function openNewElectionModal() {
|
|
document.getElementById('NewElectionModal').style.display = 'flex';
|
|
}
|
|
function closeNewElectionModal() {
|
|
document.getElementById('NewElectionModal').style.display = 'none';
|
|
}
|
|
window.onclick = function(event) {
|
|
if (event.target == document.getElementById('NewElectionModal')) {
|
|
closeNewElectionModal();
|
|
}
|
|
}
|
|
</script>
|
|
</body>
|
|
</html>
|