325 lines
17 KiB
PHP
325 lines
17 KiB
PHP
<?php
|
|
require_once 'auth.php';
|
|
|
|
// Check if user is logged in
|
|
if (!is_logged_in()) {
|
|
header('Location: login.php');
|
|
exit;
|
|
}
|
|
|
|
require_once 'db/config.php';
|
|
|
|
// Fetch data for analytics
|
|
$pdo = db();
|
|
|
|
// Candidate stats
|
|
$stmt = $pdo->query('SELECT status, COUNT(*) as count FROM candidates GROUP BY status');
|
|
$candidate_stats = $stmt->fetchAll(PDO::FETCH_ASSOC);
|
|
$total_candidates = array_sum(array_column($candidate_stats, 'count'));
|
|
|
|
// Task stats
|
|
$stmt = $pdo->query('SELECT status, COUNT(*) as count FROM tasks GROUP BY status');
|
|
$task_stats = $stmt->fetchAll(PDO::FETCH_ASSOC);
|
|
$total_tasks = array_sum(array_column($task_stats, 'count'));
|
|
$completed_tasks = 0;
|
|
foreach ($task_stats as $stat) {
|
|
if ($stat['status'] === 'Completed') {
|
|
$completed_tasks = $stat['count'];
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Candidates per day
|
|
$stmt = $pdo->query("SELECT DATE(created_at) as date, COUNT(*) as count FROM candidates GROUP BY DATE(created_at) ORDER BY DATE(created_at) DESC LIMIT 7");
|
|
$candidates_per_day = $stmt->fetchAll(PDO::FETCH_ASSOC);
|
|
|
|
// Tasks created per day
|
|
$stmt = $pdo->query("SELECT DATE(created_at) as date, COUNT(*) as count FROM tasks GROUP BY DATE(created_at) ORDER BY DATE(created_at) DESC LIMIT 7");
|
|
$tasks_created_per_day = $stmt->fetchAll(PDO::FETCH_ASSOC);
|
|
|
|
// Tasks completed per day
|
|
$stmt = $pdo->query("SELECT DATE(updated_at) as date, COUNT(*) as count FROM tasks WHERE status = 'Done' GROUP BY DATE(updated_at) ORDER BY DATE(updated_at) DESC LIMIT 7");
|
|
$tasks_completed_per_day = $stmt->fetchAll(PDO::FETCH_ASSOC);
|
|
|
|
|
|
// Fetch candidates for table
|
|
$page = isset($_GET['page']) ? (int)$_GET['page'] : 1;
|
|
$limit = 5;
|
|
$offset = ($page - 1) * $limit;
|
|
$stmt = $pdo->prepare("SELECT * FROM candidates LIMIT :limit OFFSET :offset");
|
|
$stmt->bindParam(':limit', $limit, PDO::PARAM_INT);
|
|
$stmt->bindParam(':offset', $offset, PDO::PARAM_INT);
|
|
$stmt->execute();
|
|
$candidates = $stmt->fetchAll(PDO::FETCH_ASSOC);
|
|
$stmt = $pdo->query("SELECT COUNT(*) FROM candidates");
|
|
$total_candidates_records = $stmt->fetchColumn();
|
|
$total_candidate_pages = ceil($total_candidates_records / $limit);
|
|
|
|
// Fetch tasks for table
|
|
$stmt = $pdo->prepare("SELECT * FROM tasks LIMIT :limit OFFSET :offset");
|
|
$stmt->bindParam(':limit', $limit, PDO::PARAM_INT);
|
|
$stmt->bindParam(':offset', $offset, PDO::PARAM_INT);
|
|
$stmt->execute();
|
|
$tasks = $stmt->fetchAll(PDO::FETCH_ASSOC);
|
|
$stmt = $pdo->query("SELECT COUNT(*) FROM tasks");
|
|
$total_tasks_records = $stmt->fetchColumn();
|
|
$total_task_pages = ceil($total_tasks_records / $limit);
|
|
|
|
?>
|
|
<!DOCTYPE html>
|
|
<html lang="en">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
<title>Analytics Dashboard</title>
|
|
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/css/bootstrap.min.css" rel="stylesheet">
|
|
<link rel="stylesheet" href="assets/css/custom.css?v=<?php echo time(); ?>">
|
|
</head>
|
|
<body>
|
|
<header class="header d-flex justify-content-between align-items-center">
|
|
<div class="logo">FinMox<span class="dot">.</span></div>
|
|
<nav class="d-flex align-items-center">
|
|
<a href="index.php" class="btn btn-outline-primary me-2">Home</a>
|
|
<a href="workflows.php" class="btn btn-outline-primary me-3">Workflows</a>
|
|
<div class="dropdown">
|
|
<button class="btn btn-outline-secondary dropdown-toggle" type="button" id="userDropdown" data-bs-toggle="dropdown" aria-expanded="false">
|
|
<?php echo htmlspecialchars($_SESSION['username']); ?>
|
|
</button>
|
|
<ul class="dropdown-menu dropdown-menu-end" aria-labelledby="userDropdown">
|
|
<li><a class="dropdown-item" href="roles.php">Manage Roles</a></li>
|
|
<li><hr class="dropdown-divider"></li>
|
|
<li><a class="dropdown-item" href="logout.php">Logout</a></li>
|
|
</ul>
|
|
</div>
|
|
</nav>
|
|
</header>
|
|
|
|
<main class="container-fluid mt-4">
|
|
<h2 class="mb-4">Analytics Dashboard</h2>
|
|
|
|
<!-- Nav tabs -->
|
|
<ul class="nav nav-tabs" id="myTab" role="tablist">
|
|
<li class="nav-item" role="presentation">
|
|
<button class="nav-link active" id="overview-tab" data-bs-toggle="tab" data-bs-target="#overview" type="button" role="tab" aria-controls="overview" aria-selected="true">Overview</button>
|
|
</li>
|
|
<?php if (hasPermission('view_candidates')) { ?>
|
|
<li class="nav-item" role="presentation">
|
|
<button class="nav-link" id="candidates-tab" data-bs-toggle="tab" data-bs-target="#candidates" type="button" role="tab" aria-controls="candidates" aria-selected="false">Candidates</button>
|
|
</li>
|
|
<?php } ?>
|
|
<?php if (hasPermission('view_tasks')) { ?>
|
|
<li class="nav-item" role="presentation">
|
|
<button class="nav-link" id="tasks-tab" data-bs-toggle="tab" data-bs-target="#tasks" type="button" role="tab" aria-controls="tasks" aria-selected="false">Tasks</button>
|
|
</li>
|
|
<?php } ?>
|
|
</ul>
|
|
|
|
<!-- Tab content -->
|
|
<div class="tab-content" id="myTabContent">
|
|
<!-- Overview Tab -->
|
|
<div class="tab-pane fade show active" id="overview" role="tabpanel" aria-labelledby="overview-tab">
|
|
<!-- Key Metrics -->
|
|
<div class="row mb-4 mt-4">
|
|
<?php if (hasPermission('view_candidates')) { ?>
|
|
<div class="col-md-4">
|
|
<div class="card text-center shadow-sm">
|
|
<div class="card-body">
|
|
<h5 class="card-title">Total Candidates</h5>
|
|
<p class="card-text fs-4" id="total-candidates"><?php echo $total_candidates; ?></p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<?php } ?>
|
|
<?php if (hasPermission('view_tasks')) { ?>
|
|
<div class="col-md-4">
|
|
<div class="card text-center shadow-sm">
|
|
<div class="card-body">
|
|
<h5 class="card-title">Total Tasks</h5>
|
|
<p class="card-text fs-4" id="total-tasks"><?php echo $total_tasks; ?></p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="col-md-4">
|
|
<div class="card text-center shadow-sm">
|
|
<div class="card-body">
|
|
<h5 class="card-title">Completed Tasks</h5>
|
|
<p class="card-text fs-4" id="completed-tasks"><?php echo $completed_tasks; ?></p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<?php } ?>
|
|
</div>
|
|
|
|
<!-- Summaries -->
|
|
<div class="row">
|
|
<?php if (hasPermission('view_candidates')) { ?>
|
|
<div class="col-md-6 mb-4">
|
|
<div class="card shadow-sm">
|
|
<div class="card-body">
|
|
<h5 class="card-title">Candidates by Status</h5>
|
|
<p class="card-text">
|
|
<?php foreach ($candidate_stats as $stat): ?>
|
|
<span class="badge bg-light text-dark me-2"><?php echo htmlspecialchars($stat['status']); ?>: <?php echo $stat['count']; ?></span>
|
|
<?php endforeach; ?>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<?php } ?>
|
|
<?php if (hasPermission('view_tasks')) { ?>
|
|
<div class="col-md-6 mb-4">
|
|
<div class="card shadow-sm">
|
|
<div class="card-body">
|
|
<h5 class="card-title">Tasks by Status</h5>
|
|
<p class="card-text">
|
|
<?php foreach ($task_stats as $stat): ?>
|
|
<span class="badge bg-light text-dark me-2"><?php echo htmlspecialchars($stat['status']); ?>: <?php echo $stat['count']; ?></span>
|
|
<?php endforeach; ?>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<?php } ?>
|
|
<?php if (hasPermission('view_candidates')) { ?>
|
|
<div class="col-md-6 mb-4">
|
|
<div class="card shadow-sm">
|
|
<div class="card-body">
|
|
<h5 class="card-title">Recent Candidate Activity</h5>
|
|
<p class="card-text">
|
|
<?php foreach ($candidates_per_day as $day): ?>
|
|
<span class="badge bg-light text-dark me-2"><?php echo date("M j", strtotime($day['date'])); ?>: <?php echo $day['count']; ?></span>
|
|
<?php endforeach; ?>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<?php } ?>
|
|
<?php if (hasPermission('view_tasks')) { ?>
|
|
<div class="col-md-6 mb-4">
|
|
<div class="card shadow-sm">
|
|
<div class="card-body">
|
|
<h5 class="card-title">Recent Task Activity</h5>
|
|
<p class="card-text">
|
|
<strong>Created:</strong>
|
|
<?php foreach ($tasks_created_per_day as $day): ?>
|
|
<span class="badge bg-light text-dark me-2"><?php echo date("M j", strtotime($day['date'])); ?>: <?php echo $day['count']; ?></span>
|
|
<?php endforeach; ?>
|
|
</p>
|
|
<p class="card-text">
|
|
<strong>Completed:</strong>
|
|
<?php foreach ($tasks_completed_per_day as $day): ?>
|
|
<span class="badge bg-light text-dark me-2"><?php echo date("M j", strtotime($day['date'])); ?>: <?php echo $day['count']; ?></span>
|
|
<?php endforeach; ?>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<?php } ?>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Candidates Tab -->
|
|
<?php if (hasPermission('view_candidates')) { ?>
|
|
<div class="tab-pane fade" id="candidates" role="tabpanel" aria-labelledby="candidates-tab">
|
|
<div class="row mt-4">
|
|
<div class="col-md-12">
|
|
<div class="card shadow-sm">
|
|
<div class="card-body">
|
|
<div class="d-flex justify-content-between align-items-center mb-3">
|
|
<h5 class="card-title">Recent Candidates</h5>
|
|
<a href="edit_candidate.php" class="btn btn-primary">Add Candidate</a>
|
|
</div>
|
|
<div class="table-responsive">
|
|
<table class="table table-striped">
|
|
<thead>
|
|
<tr>
|
|
<th>Name</th>
|
|
<th>Email</th>
|
|
<th>Status</th>
|
|
<th>Actions</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<?php foreach ($candidates as $candidate) { ?>
|
|
<tr>
|
|
<td><?php echo htmlspecialchars($candidate['name']); ?></td>
|
|
<td><?php echo htmlspecialchars($candidate['email']); ?></td>
|
|
<td><span class="badge bg-secondary"><?php echo htmlspecialchars($candidate['status']); ?></span></td>
|
|
<td>
|
|
<a href="edit_candidate.php?id=<?php echo $candidate['id']; ?>" class="btn btn-sm btn-outline-primary">Edit</a>
|
|
<a href="delete_candidate.php?id=<?php echo $candidate['id']; ?>" class="btn btn-sm btn-outline-danger" onclick="return confirm('Are you sure?')">Delete</a>
|
|
</td>
|
|
</tr>
|
|
<?php } ?>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
<nav>
|
|
<ul class="pagination">
|
|
<?php for ($i = 1; $i <= $total_candidate_pages; $i++) { ?>
|
|
<li class="page-item <?php if ($i == $page) echo 'active'; ?>"><a class="page-link" href="?page=<?php echo $i; ?>"><?php echo $i; ?></a></li>
|
|
<?php } ?>
|
|
</ul>
|
|
</nav>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<?php } ?>
|
|
|
|
<!-- Tasks Tab -->
|
|
<?php if (hasPermission('view_tasks')) { ?>
|
|
<div class="tab-pane fade" id="tasks" role="tabpanel" aria-labelledby="tasks-tab">
|
|
<div class="row mt-4">
|
|
<div class="col-md-12">
|
|
<div class="card shadow-sm">
|
|
<div class="card-body">
|
|
<div class="d-flex justify-content-between align-items-center mb-3">
|
|
<h5 class="card-title">Recent Tasks</h5>
|
|
<a href="edit_task.php" class="btn btn-primary">Add Task</a>
|
|
</div>
|
|
<div class="table-responsive">
|
|
<table class="table table-striped">
|
|
<thead>
|
|
<tr>
|
|
<th>Title</th>
|
|
<th>Status</th>
|
|
<th>Assigned To</th>
|
|
<th>Actions</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
<?php foreach ($tasks as $task) { ?>
|
|
<tr>
|
|
<td><?php echo htmlspecialchars($task['title']); ?></td>
|
|
<td><span class="badge bg-info"><?php echo htmlspecialchars($task['status']); ?></span></td>
|
|
<td><?php echo htmlspecialchars($task['assigned_to'] ?? 'N/A'); ?></td>
|
|
<td>
|
|
<a href="edit_task.php?id=<?php echo $task['id']; ?>" class="btn btn-sm btn-outline-primary">Edit</a>
|
|
<a href="delete_task.php?id=<?php echo $task['id']; ?>" class="btn btn-sm btn-outline-danger" onclick="return confirm('Are you sure?')">Delete</a>
|
|
</td>
|
|
</tr>
|
|
<?php } ?>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
<nav>
|
|
<ul class="pagination">
|
|
<?php for ($i = 1; $i <= $total_task_pages; $i++) { ?>
|
|
<li class="page-item <?php if ($i == $page) echo 'active'; ?>"><a class="page-link" href="?page=<?php echo $i; ?>"><?php echo $i; ?></a></li>
|
|
<?php } ?>
|
|
</ul>
|
|
</nav>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<?php } ?>
|
|
</div>
|
|
</main>
|
|
|
|
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.2/dist/js/bootstrap.bundle.min.js"></script>
|
|
</body>
|
|
</html>
|