This commit is contained in:
Borgal
2025-11-16 21:13:04 +01:00
parent aeb2d87cf5
commit 1b9ba22bb5
11 changed files with 1104 additions and 234 deletions

424
kasse.php
View File

@@ -1,55 +1,265 @@
<?php
// PHP-Logik für die Datenabfrage
// KEIN LEERZEICHEN ODER ZEILENUMBRUCH VOR DIESEM <?php!
include('inc/check_login.php');
require_once('inc/db.php');
// --- 1. Jährliche Strafen-Statistik ---
// Passen Sie hier das Datum der Abschlussveranstaltung an.
$last_reset_date = '2024-01-01';
$is_admin = ($_SESSION['role'] === 'admin');
$penalties_data = [];
$total_penalties = 0;
$total_due = 0;
// Passen Sie hier die Höhe der Strafe in Euro an.
// --- JAHRESABSCHLUSS ---
if ($is_admin && $_SERVER['REQUEST_METHOD'] === 'POST' && isset($_POST['action']) && $_POST['action'] === 'close_year') {
$today = date('Y-m-d');
$user_id = $_SESSION['user_id'] ?? 0;
$last_closing_date = '2020-01-01';
$result_last = mysqli_query($conn, "SELECT MAX(closing_date) AS last_date FROM penalty_closings");
if ($result_last) {
$row = mysqli_fetch_assoc($result_last);
if ($row && !is_null($row['last_date'])) {
$last_closing_date = $row['last_date'];
}
}
$check = mysqli_prepare($conn, "SELECT 1 FROM penalty_closings WHERE closing_date = ?");
mysqli_stmt_bind_param($check, "s", $today);
mysqli_stmt_execute($check);
$exists = mysqli_stmt_get_result($check)->num_rows > 0;
mysqli_stmt_close($check);
if (!$exists) {
$insert = mysqli_prepare($conn, "INSERT INTO penalty_closings (closing_date, closed_by) VALUES (?, ?)");
mysqli_stmt_bind_param($insert, "si", $today, $user_id);
mysqli_stmt_execute($insert);
mysqli_stmt_close($insert);
$penalty_summary = [];
$stmt_email = mysqli_prepare($conn, "
SELECT u.username, COUNT(*) AS penalty_count
FROM meeting_teilnehmer mt
JOIN meetings m ON mt.meeting_id = m.id
JOIN users u ON mt.user_id = u.id
WHERE m.meeting_date >= ? AND m.meeting_date <= ?
AND mt.attended = 1 AND mt.wore_color = 0 AND m.is_completed = 1
GROUP BY u.username
");
mysqli_stmt_bind_param($stmt_email, "ss", $last_closing_date, $today);
mysqli_stmt_execute($stmt_email);
$result_email = mysqli_stmt_get_result($stmt_email);
while ($row = mysqli_fetch_assoc($result_email)) {
$penalty_summary[] = $row;
}
mysqli_stmt_close($stmt_email);
$all_users = [];
$stmt_users = mysqli_prepare($conn, "SELECT id, username, email FROM users WHERE email IS NOT NULL AND email != ''");
mysqli_stmt_execute($stmt_users);
$result_users = mysqli_stmt_get_result($stmt_users);
while ($row = mysqli_fetch_assoc($result_users)) {
$all_users[] = $row;
}
mysqli_stmt_close($stmt_users);
if (!empty($all_users) && file_exists(__DIR__ . '/vendor/autoload.php')) {
require_once __DIR__ . '/vendor/autoload.php';
$today_formatted = date('d.m.Y', strtotime($today));
$last_closing_formatted = date('d.m.Y', strtotime($last_closing_date));
foreach ($all_users as $recipient) {
$recipient_email = $recipient['email'];
$recipient_name = $recipient['username'] ?? 'Benutzer';
try {
$mail = new \PHPMailer\PHPMailer\PHPMailer(true);
$mail->CharSet = 'UTF-8';
$mail->isSMTP();
$mail->Host = SMTP_HOST;
$mail->SMTPAuth = true;
$mail->Username = SMTP_USERNAME;
$mail->Password = SMTP_PASSWORD;
$mail->SMTPSecure = SMTP_ENCRYPTION;
$mail->Port = SMTP_PORT;
$mail->setFrom(MAIL_FROM_ADDRESS, MAIL_FROM_NAME);
$text_body = "Hallo $recipient_name,\n\n";
$text_body .= "Der Jahresabschluss wurde am $today_formatted durchgeführt.\n";
$text_body .= "Strafen der abgeschlossenen Periode (vom $last_closing_formatted bis $today_formatted):\n\n";
if (empty($penalty_summary)) {
$text_body .= "Keine Strafen in diesem Zeitraum.\n";
} else {
foreach ($penalty_summary as $row) {
$text_body .= "- " . $row['username'] . ": " . (int)$row['penalty_count'] . " Strafen (1 € pro Strafe)\n";
}
}
$text_body .= "\nDein DoMiLi-Admin.";
$html_table_rows = '';
if (empty($penalty_summary)) {
$html_table_rows = '<tr><td colspan="3" style="text-align:center;">Keine Strafen in diesem Zeitraum.</td></tr>';
} else {
foreach ($penalty_summary as $row) {
$html_table_rows .= sprintf(
'<tr>
<td style="padding:8px;border:1px solid #ddd;">%s</td>
<td style="padding:8px;border:1px solid #ddd;text-align:center;">%d</td>
<td style="padding:8px;border:1px solid #ddd;text-align:right;">%s</td>
</tr>',
htmlspecialchars($row['username']),
(int)$row['penalty_count'],
number_format((float)($row['penalty_count']), 2, ',', '.') . ' €'
);
}
}
$html_body = "
<p>Hallo <strong>$recipient_name</strong>,</p>
<p>der Jahresabschluss wurde am <strong>$today_formatted</strong> durchgeführt.</p>
<p><strong>Strafen der abgeschlossenen Periode</strong> (vom $last_closing_formatted bis $today_formatted):</p>
<table style=\"border-collapse:collapse;width:100%;margin:15px 0;\">
<thead>
<tr>
<th style=\"padding:8px;border:1px solid #ddd;text-align:left;background:#f2f2f2;\">Mitglied</th>
<th style=\"padding:8px;border:1px solid #ddd;text-align:center;background:#f2f2f2;\">Anzahl Strafen</th>
<th style=\"padding:8px;border:1px solid #ddd;text-align:right;background:#f2f2f2;\">Betrag</th>
</tr>
</thead>
<tbody>
$html_table_rows
</tbody>
</table>
<p><em>Dein DoMiLi-Admin.</em></p>";
$mail->isHTML(true);
$mail->Subject = "DoMiLi Jahresabschluss abgeschlossen am $today_formatted";
$mail->Body = $html_body;
$mail->AltBody = $text_body;
$mail->addAddress($recipient_email);
$mail->send();
} catch (Exception $e) {
error_log("PHPMailer Fehler: " . $mail->ErrorInfo);
}
}
}
$close_success = "Jahresabschluss erfolgreich durchgeführt!";
} else {
$close_error = "Heute wurde bereits ein Abschluss durchgeführt.";
}
}
// --- DATEN FÜR ANZEIGE ---
$last_closing_date = '2020-01-01';
$result_last = mysqli_query($conn, "SELECT MAX(closing_date) AS last_date FROM penalty_closings");
if ($result_last) {
$row = mysqli_fetch_assoc($result_last);
if ($row && !is_null($row['last_date'])) {
$last_closing_date = $row['last_date'];
}
}
$cumulative_penalties = [];
$stmt_cum = mysqli_prepare($conn, "
SELECT u.username, COUNT(*) AS total_penalty_count
FROM meeting_teilnehmer mt
JOIN meetings m ON mt.meeting_id = m.id
JOIN users u ON mt.user_id = u.id
WHERE mt.attended = 1 AND mt.wore_color = 0 AND m.is_completed = 1
GROUP BY u.username
");
mysqli_stmt_execute($stmt_cum);
$result_cum = mysqli_stmt_get_result($stmt_cum);
while ($row = mysqli_fetch_assoc($result_cum)) {
$cumulative_penalties[$row['username']] = (int)$row['total_penalty_count'];
}
mysqli_stmt_close($stmt_cum);
$open_penalties = [];
$total_open = 0;
$penalty_amount = 1;
// SQL-Abfrage, um Strafen pro Benutzer seit dem letzten Reset-Datum zu zählen
// Strafe = anwesend (attended=1) aber Farbe nicht getragen (wore_color=0)
// NEUE BEDINGUNG: Nur Meetings einbeziehen, die als abgeschlossen markiert sind.
$sql_penalties = "
SELECT
u.username,
COUNT(mt.user_id) AS penalty_count
$stmt_open = mysqli_prepare($conn, "
SELECT u.username, COUNT(*) AS open_penalty_count
FROM meeting_teilnehmer mt
JOIN meetings m ON mt.meeting_id = m.id
JOIN users u ON mt.user_id = u.id
WHERE m.meeting_date >= ? AND mt.attended = 1 AND mt.wore_color = 0 AND m.is_completed = 1
GROUP BY u.username
ORDER BY penalty_count DESC
";
$stmt = mysqli_prepare($conn, $sql_penalties);
// Überprüfen, ob das Statement erfolgreich vorbereitet wurde
if ($stmt === false) {
die("Fehler bei der SQL-Abfrage: " . mysqli_error($conn));
");
mysqli_stmt_bind_param($stmt_open, "s", $last_closing_date);
mysqli_stmt_execute($stmt_open);
$result_open = mysqli_stmt_get_result($stmt_open);
while ($row = mysqli_fetch_assoc($result_open)) {
$open_penalties[$row['username']] = (int)$row['open_penalty_count'];
$total_open += $row['open_penalty_count'];
}
mysqli_stmt_bind_param($stmt, "s", $last_reset_date);
mysqli_stmt_execute($stmt);
$result = mysqli_stmt_get_result($stmt);
mysqli_stmt_close($stmt_open);
while ($row = mysqli_fetch_assoc($result)) {
$penalties_data[] = $row;
$total_penalties += $row['penalty_count'];
$total_due += $row['penalty_count'] * $penalty_amount;
$penalties_data = [];
$all_usernames = array_unique(array_merge(array_keys($cumulative_penalties), array_keys($open_penalties)));
foreach ($all_usernames as $username) {
$total = $cumulative_penalties[$username] ?? 0;
$open = $open_penalties[$username] ?? 0;
$penalties_data[] = [
'username' => $username,
'total_penalty_count' => $total,
'open_penalty_count' => $open
];
}
usort($penalties_data, fn($a, $b) => $b['open_penalty_count'] <=> $a['open_penalty_count']);
$total_penalties = array_sum(array_column($penalties_data, 'total_penalty_count'));
$total_due = $total_open * $penalty_amount;
$historical_table = [];
$closing_years = [];
$result_closings = mysqli_query($conn, "SELECT closing_date FROM penalty_closings ORDER BY closing_date");
while ($row = mysqli_fetch_assoc($result_closings)) {
$closing_date = new DateTime($row['closing_date']);
$year = (int)$closing_date->format('Y');
$month = (int)$closing_date->format('n');
$closed_year = ($month >= 7) ? $year : ($year - 1);
$closing_years[] = $closed_year;
}
$closing_years = array_unique($closing_years);
sort($closing_years);
if (!empty($closing_years)) {
foreach ($closing_years as $year) {
$year_start = "$year-01-01";
$year_end = "$year-12-31";
$stmt_hist = mysqli_prepare($conn, "
SELECT u.username, COUNT(*) AS penalty_count
FROM meeting_teilnehmer mt
JOIN meetings m ON mt.meeting_id = m.id
JOIN users u ON mt.user_id = u.id
WHERE m.meeting_date BETWEEN ? AND ?
AND mt.attended = 1 AND mt.wore_color = 0 AND m.is_completed = 1
GROUP BY u.username
");
mysqli_stmt_bind_param($stmt_hist, "ss", $year_start, $year_end);
mysqli_stmt_execute($stmt_hist);
$result_hist = mysqli_stmt_get_result($stmt_hist);
while ($row = mysqli_fetch_assoc($result_hist)) {
$username = $row['username'];
$count = (int)$row['penalty_count'];
if (!isset($historical_table[$username])) {
$historical_table[$username] = ['years' => [], 'total' => 0];
}
$historical_table[$username]['years'][] = $year;
$historical_table[$username]['total'] += $count;
}
mysqli_stmt_close($stmt_hist);
}
foreach ($historical_table as $username => $data) {
$historical_table[$username]['year_label'] = implode('/', $data['years']);
}
}
mysqli_stmt_close($stmt);
// Neue Statistik: Ranking nach bezahlten Strafen
// NEUE BEDINGUNG: Auch hier nur abgeschlossene Meetings einbeziehen.
$paid_stats = [];
$sql_paid = "
SELECT
u.username,
COUNT(mt.user_id) AS paid_count
SELECT u.username, COUNT(mt.user_id) AS paid_count
FROM meeting_teilnehmer mt
JOIN meetings m ON mt.meeting_id = m.id
JOIN users u ON mt.user_id = u.id
@@ -66,38 +276,58 @@ require_once 'inc/header.php';
?>
<div class="container mt-5">
<h2 class="mb-4">Kasse & Auswertung</h2>
<h2 class="mb-4">Kasse</h2>
<?php if (!empty($close_success)): ?>
<div class="alert alert-success alert-dismissible fade show" role="alert">
<?= htmlspecialchars($close_success) ?>
<button type="button" class="btn-close" data-bs-dismiss="alert"></button>
</div>
<?php endif; ?>
<?php if (!empty($close_error)): ?>
<div class="alert alert-warning alert-dismissible fade show" role="alert">
<?= htmlspecialchars($close_error) ?>
<button type="button" class="btn-close" data-bs-dismiss="alert"></button>
</div>
<?php endif; ?>
<div class="card shadow mb-4">
<div class="card-header bg-primary-subtle text-secondary">
<h4 class="mb-0">Strafen seit letztem Abschluss</h4>
<div class="card-header bg-primary-subtle text-secondary d-flex justify-content-between align-items-center">
<h4 class="mb-0">Strafen</h4>
<?php if ($is_admin): ?>
<form method="POST" class="d-flex align-items-center m-0" onsubmit="return confirm('Jahresabschluss durchführen?');">
<input type="hidden" name="action" value="close_year">
<button type="submit" class="btn btn-sm d-flex align-items-center justify-content-center p-0" style="height: 24px; font-size: 0.875rem;">
Jahresabschluss
<span class="material-symbols-outlined ms-1" style="font-size: 18px; line-height: 1;">add</span>
</button>
</form>
<?php endif; ?>
</div>
<div class="card-body">
<p class="fs-5 fw-bold text-center">
Gesamtzahl der Strafen: <span class="badge bg-danger"><?= $total_penalties ?></span><br>
Fälliger Gesamtbetrag: <span class="text-danger fw-bold"><?= number_format($total_due, 2, ',', '.'); ?> €</span>
Strafen gesamt: <span class="badge bg-danger"><?= $total_penalties ?></span><br>
offener Betrag: <span class="text-danger fw-bold"><?= number_format($total_due, 2, ',', '.') ?> €</span>
</p>
<?php if (empty($penalties_data)): ?>
<div class="alert alert-info text-center" role="alert">
Bisher wurden keine Strafen verzeichnet.
</div>
<div class="alert alert-info text-center">Keine Strafen erfasst.</div>
<?php else: ?>
<div class="table-responsive">
<table class="table table-striped table-hover">
<thead>
<tr>
<th>Benutzer</th>
<th>Anzahl Strafen</th>
<th>Fälliger Betrag</th>
<th>Strafen gesamt</th>
<th>offener Betrag</th>
</tr>
</thead>
<tbody>
<?php foreach ($penalties_data as $penalty): ?>
<?php foreach ($penalties_data as $p): ?>
<tr>
<td><?= htmlspecialchars($penalty['username']); ?></td>
<td><?= htmlspecialchars($penalty['penalty_count']); ?></td>
<td><?= number_format($penalty['penalty_count'] * $penalty_amount, 2, ',', '.'); ?> €</td>
<td><?= htmlspecialchars($p['username']) ?></td>
<td><?= htmlspecialchars($p['total_penalty_count']) ?></td>
<td><?= number_format($p['open_penalty_count'] * $penalty_amount, 2, ',', '.') ?> €</td>
</tr>
<?php endforeach; ?>
</tbody>
@@ -107,40 +337,79 @@ require_once 'inc/header.php';
</div>
</div>
<div class="card shadow mb-4">
<div class="card-header bg-primary-subtle text-secondary">
<h4 class="mb-0">Historische Strafen</h4>
</div>
<div class="card-body">
<div class="table-responsive">
<table class="table table-striped table-hover">
<thead>
<tr>
<th>Benutzer</th>
<th>Jahr(e)</th>
<th>Strafen</th>
</tr>
</thead>
<tbody>
<?php if (!empty($historical_table)): ?>
<?php foreach ($historical_table as $username => $data): ?>
<tr>
<td><?= htmlspecialchars($username) ?></td>
<td><?= htmlspecialchars($data['year_label']) ?></td>
<td><?= htmlspecialchars($data['total']) ?></td>
</tr>
<?php endforeach; ?>
<?php else: ?>
<tr>
<td colspan="3" class="text-center text-muted">—</td>
</tr>
<?php endif; ?>
</tbody>
</table>
</div>
</div>
</div>
<div class="card shadow mb-4">
<div class="card-header bg-primary-subtle text-secondary">
<h4 class="mb-0">Rechnungen</h4>
</div>
<div class="card-body">
<h5 class="card-title text-center">Ranking - Rechnung übernommen</h5>
<h5 class="card-title text-center">Ranking Rechnung übernommen</h5>
<div style="height: 40vh;">
<canvas id="paidChart"></canvas>
</div>
</div>
</div>
</div>
<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
<script>
document.addEventListener('DOMContentLoaded', function() {
// Daten für "Gezahlt"-Diagramm
const paidData = {
labels: <?= json_encode(array_column($paid_stats, 'username')); ?>,
datasets: [{
label: 'Gezahlte Strafen',
data: <?= json_encode(array_column($paid_stats, 'paid_count')); ?>,
backgroundColor: 'rgba(153, 102, 255, 0.8)',
borderColor: 'rgb(153, 102, 255)',
borderWidth: 1
}]
};
const rawLabels = <?php echo json_encode(array_column($paid_stats, 'username') ?: []); ?>;
const rawData = <?php echo json_encode(array_map('intval', array_column($paid_stats, 'paid_count') ?: [])); ?>;
const paidConfig = {
const ctx = document.getElementById('paidChart').getContext('2d');
if (rawLabels.length === 0 || rawData.length === 0) {
document.getElementById('paidChart').closest('.card-body').innerHTML =
'<p class="text-muted text-center mt-3">Keine bezahlten Rechnungen erfasst.</p>';
return;
}
new Chart(ctx, {
type: 'bar',
data: paidData,
data: {
labels: rawLabels,
datasets: [{
label: 'Gezahlte Strafen',
data: rawData,
backgroundColor: 'rgba(54, 162, 235, 0.6)',
borderColor: 'rgba(54, 162, 235, 1)',
borderWidth: 1
}]
},
options: {
indexAxis: 'y',
responsive: true,
@@ -156,31 +425,28 @@ require_once 'inc/header.php';
},
scales: {
x: {
beginAtZero: true
beginAtZero: true,
ticks: {
stepSize: 1,
callback: function(value) {
return Number.isInteger(value) ? value : null;
}
}
},
y: {
ticks: {
font: {
size: 10
size: 12
},
callback: function(value, index, ticks) {
const username = this.getLabelForValue(value);
const maxChars = 15;
if (username.length > maxChars) {
return username.substring(0, maxChars) + '...';
}
return username;
const label = this.getLabelForValue(value);
return label.length > 15 ? label.substring(0, 15) + '…' : label;
}
}
}
}
}
};
new Chart(
document.getElementById('paidChart'),
paidConfig
);
});
});
</script>