This commit is contained in:
2026-03-23 02:09:14 +01:00
parent 705329d3c2
commit d8d46ed8e9
61 changed files with 6054 additions and 3116 deletions

157
public/js/pages/start.js Normal file
View File

@@ -0,0 +1,157 @@
import { apiGet, isAuthRedirectError } from '../api.js';
import { guard } from '../core/auth-guard.js';
import {
ticketStatusLabel,
ticketPriorityLabel,
eventTypeLabel,
eventTypeBadgeClass,
statusBadgeClass,
priorityBadgeClass,
} from '../core/constants.js';
import { esc, formatDateTime, extrasName } from '../core/utils.js';
import { bindAttachmentPreview } from '../core/attachment-preview.js';
import {
eventInhaltHtml,
sortEventsChronologicalWithAttachmentsLast,
} from '../core/ticket-events.js';
const loadingEl = document.getElementById('page-loading');
const mainEl = document.getElementById('page-main');
const errEl = document.getElementById('page-error');
const listEl = document.getElementById('home-ticket-list');
const emptyEl = document.getElementById('home-empty');
const tpl = document.getElementById('tpl-home-ticket');
function showError(msg) {
loadingEl.hidden = true;
mainEl.hidden = true;
errEl.hidden = false;
errEl.textContent = msg;
}
function renderEventBoxes(events) {
const evChrono = sortEventsChronologicalWithAttachmentsLast(events);
if (evChrono.length === 0) {
return '<p class="muted home-ticket-no-events">Noch keine Ereignisse in der Historie.</p>';
}
return evChrono
.map(
(ev) => `
<div class="home-event-box">
<div class="home-event-box-header">
<span class="badge event-type-badge ${eventTypeBadgeClass[ev.type] || ''}">${esc(eventTypeLabel[ev.type] || ev.type)}</span>
<time class="home-event-box-time" datetime="${esc(ev.createdAt)}">${esc(formatDateTime(ev.createdAt))}</time>
</div>
<div class="home-event-box-body">${eventInhaltHtml(ev)}</div>
</div>`,
)
.join('');
}
function fillTicketCard(node, t, events) {
const id = t.id;
const detailId = `home-ticket-detail-${id}`;
node.dataset.ticketId = id;
const btn = node.querySelector('.home-ticket-collapse-btn');
const panel = node.querySelector('.home-ticket-collapsible');
panel.id = detailId;
btn.setAttribute('aria-controls', detailId);
const mn = t.machine ? extrasName(t.machine) : '';
const machineLabel = t.machine
? `${esc(t.machine.seriennummer)}${mn ? ` · ${esc(mn)}` : ''}`
: '';
const standort = t.machine ? esc(t.machine.standort) : '—';
const headEl = node.querySelector('.home-ticket-head');
headEl.classList.toggle('home-ticket-head-overdue', Boolean(t.isOverdue));
const titleA = node.querySelector('.js-ticket-link');
titleA.href = `/ticket.html?id=${encodeURIComponent(id)}`;
titleA.textContent = t.title;
const st = node.querySelector('.js-status');
st.textContent = ticketStatusLabel[t.status];
st.className = `badge js-status ${statusBadgeClass[t.status] || ''}`;
const pr = node.querySelector('.js-priority');
pr.textContent = ticketPriorityLabel[t.priority];
pr.className = `badge js-priority ${priorityBadgeClass[t.priority] || ''}`;
const metaM = node.querySelector('.js-meta-machine');
metaM.innerHTML = t.machine
? `<span class="muted">Maschine:</span> ${machineLabel}`
: '<span class="muted">Keine Maschine</span>';
node.querySelector('.js-meta-standort').textContent = standort;
node.querySelector('.js-meta-created').textContent = formatDateTime(t.createdAt);
node.querySelector('.js-meta-updated').textContent = formatDateTime(t.updatedAt);
const openA = node.querySelector('.js-ticket-open');
openA.href = `/ticket.html?id=${encodeURIComponent(id)}`;
node.querySelector('.js-desc').textContent = t.description;
const mBlock = node.querySelector('.js-machine-block');
if (t.machine) {
const m = t.machine;
mBlock.innerHTML = `<p class="home-ticket-machine"><strong>Maschine:</strong> <a href="/machine.html?id=${esc(m.id)}">${esc(m.seriennummer)}</a>${mn ? ` · ${esc(mn)}` : ''} &nbsp;·&nbsp; <strong>Typ:</strong> ${esc(m.typ)}</p>`;
} else {
mBlock.innerHTML = '';
}
node.querySelector('.js-events').innerHTML = renderEventBoxes(events);
const chev = btn.querySelector('.home-ticket-chevron');
btn.onclick = () => {
const willShow = panel.hidden;
panel.hidden = !willShow;
btn.setAttribute('aria-expanded', String(willShow));
chev.textContent = willShow ? '▲' : '▼';
};
}
async function run() {
const tickets = await apiGet('/tickets?open=1');
const eventsLists =
tickets.length === 0
? []
: await Promise.all(tickets.map((t) => apiGet(`/tickets/${t.id}/events`)));
const openCount = tickets.filter((t) => t.status === 'OPEN').length;
const waitingCount = tickets.filter((t) => t.status === 'WAITING').length;
document.getElementById('kpi-open').textContent = `${openCount} Offen`;
document.getElementById('kpi-waiting').textContent = `${waitingCount} Wartend`;
document.getElementById('kpi-total').textContent = `gesamt: ${tickets.length}`;
listEl.innerHTML = '';
if (tickets.length === 0) {
emptyEl.hidden = false;
} else {
emptyEl.hidden = true;
tickets.forEach((t, i) => {
const frag = tpl.content.cloneNode(true);
const article = frag.querySelector('.home-ticket-card');
fillTicketCard(article, t, eventsLists[i] || []);
listEl.appendChild(article);
});
}
}
async function init() {
const st = await guard({ activeNav: 'start' });
if (!st) return;
loadingEl.hidden = true;
mainEl.hidden = false;
bindAttachmentPreview(document.body);
try {
await run();
} catch (e) {
if (isAuthRedirectError(e)) return;
showError(e.message || 'Fehler');
}
}
init();