Initial commit

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-06-03 18:15:11 +02:00
commit c65c9f1751
19 changed files with 3633 additions and 0 deletions
+3
View File
@@ -0,0 +1,3 @@
node_modules/
.env
*.log
+10
View File
@@ -0,0 +1,10 @@
# Default ignored files
/shelf/
/workspace.xml
# Editor-based HTTP Client requests
/httpRequests/
# Ignored default folder with query files
/queries/
# Datasource local storage ignored files
/dataSources/
/dataSources.local.xml
+8
View File
@@ -0,0 +1,8 @@
<?xml version="1.0" encoding="UTF-8"?>
<module type="WEB_MODULE" version="4">
<component name="NewModuleRootManager">
<content url="file://$MODULE_DIR$" />
<orderEntry type="inheritedJdk" />
<orderEntry type="sourceFolder" forTests="false" />
</component>
</module>
+10
View File
@@ -0,0 +1,10 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="MaterialThemeProjectNewConfig">
<option name="metadata">
<MTProjectMetadataState>
<option name="userId" value="49989f6b:19e8e0f7dcd:-7fff" />
</MTProjectMetadataState>
</option>
</component>
</project>
+8
View File
@@ -0,0 +1,8 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="ProjectModuleManager">
<modules>
<module fileurl="file://$PROJECT_DIR$/.idea/jobbi-bewerbung.iml" filepath="$PROJECT_DIR$/.idea/jobbi-bewerbung.iml" />
</modules>
</component>
</project>
+110
View File
@@ -0,0 +1,110 @@
# Bewerbungs-Tracker
Lokale Web-Anwendung zur Verwaltung von Stellenbewerbungen optimiert für die monatliche Nachweispflicht beim Jobcenter (Grundsicherung / Bürgergeld).
## Funktionen
- **Bewerbungen verwalten** Anlegen, Bearbeiten, Löschen
- **Monatsansicht** Filter nach Monat und Jahr
- **Statistiken** Gesamt, Positiv, Absagen, Ausstehend
- **PDF-Export** Professionelles Dokument mit Ihren Daten für das Jobcenter
- **Dunkelmodus** Standard oder per Schalter umschaltbar
- **Datenschutz** Alle Daten bleiben lokal auf Ihrem Rechner (SQLite)
## Installation
### Voraussetzungen
- Node.js ≥ 18 (https://nodejs.org)
- npm (im Lieferumfang von Node.js)
- Build-Tools (für `better-sqlite3`):
- **Linux/Mac:** `build-essential` / Xcode Command Line Tools
- **Windows:** `windows-build-tools` oder Visual Studio Build Tools
### Schritte
```bash
# 1. In das Projektverzeichnis wechseln
cd bewerbungs-tracker
# 2. Abhängigkeiten installieren
npm install
# 3. Server starten
npm start
```
Die Anwendung ist dann unter **http://localhost:3000** erreichbar.
### Entwicklungsmodus (Auto-Reload)
```bash
npm run dev
```
## Projektstruktur
```
bewerbungs-tracker/
├── server.js # Express-Server mit allen Routen
├── package.json
├── views/
│ ├── index.ejs # Übersichtsseite
│ ├── einstellungen.ejs # Einstellungsseite
│ └── partials/
│ ├── header.ejs # HTML-Head + Navigation
│ └── footer.ejs # Abschlusselemente + Scripts
├── public/
│ ├── css/style.css # Tailwind-Utility-Klassen
│ └── js/main.js # Dark Mode, Modals, PDF-Generierung
└── data/
└── bewerbungen.db # SQLite-Datenbank (wird automatisch erstellt)
```
## Datenbank-Schema
```sql
-- Bewerbungen
CREATE TABLE bewerbungen (
id INTEGER PRIMARY KEY AUTOINCREMENT,
datum DATE NOT NULL,
firma TEXT NOT NULL,
stelle TEXT NOT NULL,
art TEXT,
status TEXT,
notizen TEXT,
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
updated_at DATETIME DEFAULT CURRENT_TIMESTAMP
);
-- Benutzerprofil (wird im PDF verwendet)
CREATE TABLE settings (
id INTEGER PRIMARY KEY CHECK (id = 1),
name TEXT,
adresse TEXT,
kundennummer TEXT
);
```
## PDF-Export
1. Öffnen Sie **Einstellungen** und tragen Sie Ihren Namen, Adresse und Kundennummer ein.
2. Filtern Sie auf der Übersicht den gewünschten Monat.
3. Klicken Sie auf **„PDF exportieren"**.
Das PDF enthält:
- Briefkopf mit Ihren persönlichen Daten
- Titel „Bewerbungsaktivitäten Monat Jahr"
- Zusammenfassungssatz
- Tabelle aller Bewerbungen des Monats
- Unterschriftszeile mit Datum
## Port ändern
```bash
PORT=8080 npm start
```
## Lizenz
MIT
View File
Binary file not shown.
Binary file not shown.
Binary file not shown.
+1724
View File
File diff suppressed because it is too large Load Diff
+19
View File
@@ -0,0 +1,19 @@
{
"name": "bewerbungs-tracker",
"version": "1.0.0",
"description": "Bewerbungs-Tracker für die monatliche Jobcenter-Berichterstattung",
"main": "server.js",
"scripts": {
"start": "node server.js",
"dev": "nodemon server.js"
},
"dependencies": {
"better-sqlite3": "^12.10.0",
"ejs": "^3.1.10",
"express": "^4.22.2",
"method-override": "^3.0.0"
},
"devDependencies": {
"nodemon": "^3.0.2"
}
}
+822
View File
@@ -0,0 +1,822 @@
/* ── Bewerbungs-Tracker — Design System ──────────────────────────────────── */
@import url('https://fonts.googleapis.com/css2?family=Barlow+Semi+Condensed:wght@600;700;800&family=DM+Sans:ital,opsz,wght@0,9..40,400;0,9..40,500;0,9..40,600;1,9..40,400&family=DM+Mono:wght@400;500&display=swap');
/* ── Variables ───────────────────────────────────────────────────────────── */
:root {
--bg: #f3f5fb;
--surface: #ffffff;
--surface-2: #edf0f7;
--surface-3: #e4e9f4;
--border: #dde2ef;
--border-2: #cad2e4;
--text: #18202f;
--text-2: #4a556b;
--text-muted: #8896b0;
--accent: #e07b00;
--accent-dim: rgba(224,123,0,0.1);
--accent-hover: #c96e00;
--green: #16a34a;
--green-dim: rgba(22,163,74,0.1);
--red: #dc2626;
--red-dim: rgba(220,38,38,0.1);
--amber: #d97706;
--amber-dim: rgba(217,119,6,0.1);
--blue: #2563eb;
--blue-dim: rgba(37,99,235,0.1);
--purple: #7c3aed;
--purple-dim: rgba(124,58,237,0.1);
--shadow-sm: 0 1px 3px rgba(0,0,0,0.06), 0 1px 2px rgba(0,0,0,0.04);
--shadow: 0 4px 16px rgba(0,0,0,0.08);
--shadow-xl: 0 20px 60px rgba(0,0,0,0.14);
--radius: 6px;
--radius-lg: 10px;
--radius-xl: 14px;
--font-sans: 'DM Sans', system-ui, -apple-system, sans-serif;
--font-display: 'Barlow Semi Condensed', system-ui, sans-serif;
--font-mono: 'DM Mono', 'Fira Mono', monospace;
}
html.dark {
--bg: #0c0e14;
--surface: #131720;
--surface-2: #192030;
--surface-3: #1f2840;
--border: #232d42;
--border-2: #2c3a56;
--text: #d0d9ea;
--text-2: #7d8ea8;
--text-muted: #435069;
--accent: #f08c00;
--accent-dim: rgba(240,140,0,0.12);
--accent-hover: #e07b00;
--green: #22c55e;
--green-dim: rgba(34,197,94,0.12);
--red: #f87171;
--red-dim: rgba(248,113,113,0.12);
--amber: #fbbf24;
--amber-dim: rgba(251,191,36,0.12);
--blue: #60a5fa;
--blue-dim: rgba(96,165,250,0.12);
--purple: #a78bfa;
--purple-dim: rgba(167,139,250,0.12);
--shadow-sm: 0 1px 3px rgba(0,0,0,0.4);
--shadow: 0 4px 16px rgba(0,0,0,0.5);
--shadow-xl: 0 20px 60px rgba(0,0,0,0.7);
}
/* ── Reset ───────────────────────────────────────────────────────────────── */
*, *::before, *::after { box-sizing: border-box; }
body { margin: 0; }
/* ── Base ────────────────────────────────────────────────────────────────── */
body {
font-family: var(--font-sans);
background: var(--bg);
color: var(--text);
min-height: 100vh;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
transition: background 0.2s, color 0.2s;
}
/* Ambient gradient for dark mode */
html.dark body::before {
content: '';
position: fixed;
inset: 0;
z-index: 0;
pointer-events: none;
background:
radial-gradient(ellipse 70% 50% at 5% 0%, rgba(240,140,0,0.05) 0%, transparent 65%),
radial-gradient(ellipse 50% 40% at 95% 100%, rgba(96,165,250,0.04) 0%, transparent 65%);
}
/* ── Navigation ──────────────────────────────────────────────────────────── */
.app-nav {
position: sticky;
top: 0;
z-index: 40;
height: 56px;
display: flex;
align-items: center;
padding: 0 1.5rem;
gap: 1.5rem;
background: var(--surface);
border-bottom: 1px solid var(--border);
}
.nav-brand {
display: flex;
align-items: center;
gap: 0.5rem;
text-decoration: none;
font-family: var(--font-display);
font-weight: 700;
font-size: 1.05rem;
letter-spacing: 0.01em;
color: var(--text);
flex-shrink: 0;
}
/* Animated dot */
.brand-dot {
width: 8px;
height: 8px;
border-radius: 50%;
background: var(--accent);
flex-shrink: 0;
animation: pulse-dot 3s ease-in-out infinite;
}
@keyframes pulse-dot {
0%, 100% { box-shadow: 0 0 0 0 var(--accent-dim); }
50% { box-shadow: 0 0 0 5px transparent; }
}
.nav-links {
display: flex;
align-items: center;
gap: 2px;
flex: 1;
}
.nav-link {
padding: 0.3125rem 0.6875rem;
border-radius: var(--radius);
font-size: 0.875rem;
font-weight: 500;
color: var(--text-2);
text-decoration: none;
transition: color 0.15s, background 0.15s;
}
.nav-link:hover { color: var(--text); background: var(--surface-2); }
.nav-link.active { color: var(--accent); background: var(--accent-dim); }
/* Pill toggle for dark mode */
.dark-toggle {
position: relative;
width: 42px;
height: 24px;
border-radius: 12px;
background: var(--surface-3);
border: 1px solid var(--border-2);
cursor: pointer;
transition: background 0.2s, border-color 0.2s;
flex-shrink: 0;
padding: 0;
outline: none;
}
.dark-toggle:focus-visible {
box-shadow: 0 0 0 2px var(--accent);
}
.dark-toggle::after {
content: '';
position: absolute;
top: 3px;
left: 3px;
width: 16px;
height: 16px;
border-radius: 50%;
background: var(--text-muted);
transition: transform 0.22s cubic-bezier(0.4,0,0.2,1), background 0.2s;
}
html.dark .dark-toggle { background: var(--accent-dim); border-color: var(--accent); }
html.dark .dark-toggle::after {
transform: translateX(18px);
background: var(--accent);
}
/* ── Page wrapper ────────────────────────────────────────────────────────── */
.page-wrapper {
position: relative;
z-index: 1;
max-width: 88rem;
margin: 0 auto;
padding: 2rem 1.5rem 3rem;
}
/* ── Page header ─────────────────────────────────────────────────────────── */
.page-header {
display: flex;
flex-wrap: wrap;
align-items: flex-start;
justify-content: space-between;
gap: 1rem;
margin-bottom: 1.75rem;
}
.page-title {
font-family: var(--font-display);
font-weight: 700;
font-size: 1.875rem;
letter-spacing: 0.01em;
line-height: 1.15;
color: var(--text);
}
.page-meta {
font-size: 0.8125rem;
color: var(--text-muted);
margin-top: 0.3rem;
display: flex;
align-items: center;
gap: 0.5rem;
}
.page-meta .sep { opacity: 0.4; }
.page-meta .tag {
font-family: var(--font-mono);
font-size: 0.7rem;
background: var(--surface-2);
border: 1px solid var(--border);
padding: 0.1rem 0.5rem;
border-radius: 4px;
color: var(--text-2);
}
/* ── Filter bar ──────────────────────────────────────────────────────────── */
.filter-bar {
display: flex;
align-items: center;
gap: 0.5rem;
flex-wrap: wrap;
}
/* ── Stats ───────────────────────────────────────────────────────────────── */
.stats-grid {
display: grid;
grid-template-columns: repeat(2, 1fr);
gap: 0.875rem;
margin-bottom: 1.5rem;
}
@media (min-width: 560px) {
.stats-grid { grid-template-columns: repeat(4, 1fr); }
}
.stat-card {
background: var(--surface);
border: 1px solid var(--border);
border-radius: var(--radius-xl);
padding: 1.125rem 1.25rem 1rem;
position: relative;
overflow: hidden;
animation: fadeUp 0.45s cubic-bezier(0.22,1,0.36,1) both;
}
.stat-card:nth-child(1) { animation-delay: 0.04s; }
.stat-card:nth-child(2) { animation-delay: 0.09s; }
.stat-card:nth-child(3) { animation-delay: 0.14s; }
.stat-card:nth-child(4) { animation-delay: 0.19s; }
/* Bottom color strip */
.stat-card::after {
content: '';
position: absolute;
bottom: 0; left: 0; right: 0;
height: 2px;
border-radius: 0 0 var(--radius-xl) var(--radius-xl);
}
.stat-card.s-total::after { background: var(--accent); }
.stat-card.s-positiv::after { background: var(--green); }
.stat-card.s-absage::after { background: var(--red); }
.stat-card.s-pending::after { background: var(--amber); }
.stat-label {
font-size: 0.65rem;
font-weight: 600;
text-transform: uppercase;
letter-spacing: 0.1em;
color: var(--text-muted);
margin-bottom: 0.375rem;
}
.stat-value {
font-family: var(--font-display);
font-weight: 800;
font-size: 2.5rem;
line-height: 1;
letter-spacing: 0.01em;
color: var(--text);
font-variant-numeric: tabular-nums;
}
.stat-card.s-total .stat-value { color: var(--accent); }
.stat-card.s-positiv .stat-value { color: var(--green); }
.stat-card.s-absage .stat-value { color: var(--red); }
.stat-card.s-pending .stat-value { color: var(--amber); }
/* ── Action bar ──────────────────────────────────────────────────────────── */
.action-bar {
display: flex;
align-items: center;
justify-content: space-between;
margin-bottom: 0.75rem;
}
.entry-count {
font-size: 0.6875rem;
font-weight: 700;
text-transform: uppercase;
letter-spacing: 0.1em;
color: var(--text-muted);
}
/* ── Data card ───────────────────────────────────────────────────────────── */
.data-card {
background: var(--surface);
border: 1px solid var(--border);
border-radius: var(--radius-xl);
overflow: hidden;
animation: fadeUp 0.45s 0.24s cubic-bezier(0.22,1,0.36,1) both;
}
/* ── Table ───────────────────────────────────────────────────────────────── */
.data-table {
width: 100%;
border-collapse: collapse;
}
.data-table thead th {
padding: 0.6875rem 1rem;
text-align: left;
font-size: 0.65rem;
font-weight: 700;
text-transform: uppercase;
letter-spacing: 0.1em;
color: var(--text-muted);
background: var(--surface-2);
border-bottom: 1px solid var(--border);
white-space: nowrap;
}
.data-table thead th:first-child { border-radius: 0; }
.data-table tbody tr {
border-bottom: 1px solid var(--border);
transition: background 0.1s;
}
.data-table tbody tr:last-child { border-bottom: none; }
.data-table tbody tr:hover { background: var(--surface-2); }
/* Row actions: only visible on hover */
.data-table tbody tr .row-actions { opacity: 0; transition: opacity 0.15s; }
.data-table tbody tr:hover .row-actions { opacity: 1; }
.data-table td {
padding: 0.8125rem 1rem;
font-size: 0.875rem;
vertical-align: middle;
}
.td-date {
font-family: var(--font-mono);
font-size: 0.75rem;
color: var(--text-muted);
white-space: nowrap;
letter-spacing: 0.02em;
}
.td-firma { font-weight: 600; color: var(--text); }
.td-stelle { color: var(--text-2); }
.td-art {
font-family: var(--font-mono);
font-size: 0.7rem;
color: var(--text-muted);
}
.td-notizen {
max-width: 14rem;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
font-size: 0.75rem;
color: var(--text-muted);
}
/* ── Status chip ─────────────────────────────────────────────────────────── */
.status-chip {
display: inline-flex;
align-items: center;
gap: 0.375rem;
font-size: 0.8rem;
font-weight: 500;
white-space: nowrap;
}
.status-chip::before {
content: '';
width: 6px;
height: 6px;
border-radius: 50%;
flex-shrink: 0;
}
.st-gesendet { color: var(--blue); }
.st-gesendet::before { background: var(--blue); }
.st-eingang { color: var(--amber); }
.st-eingang::before { background: var(--amber); }
.st-vorstellung { color: var(--purple); }
.st-vorstellung::before { background: var(--purple); }
.st-absage { color: var(--red); }
.st-absage::before { background: var(--red); }
.st-einstellung { color: var(--green); }
.st-einstellung::before { background: var(--green); box-shadow: 0 0 5px var(--green); }
.st-keine { color: var(--text-muted); }
.st-keine::before { background: var(--text-muted); }
/* ── Row action buttons ──────────────────────────────────────────────────── */
.row-btn {
display: inline-flex;
align-items: center;
justify-content: center;
width: 30px;
height: 30px;
border-radius: var(--radius);
border: none;
background: transparent;
cursor: pointer;
transition: background 0.12s, color 0.12s;
color: var(--text-muted);
}
.row-btn:hover { background: var(--surface-3); color: var(--text); }
.row-btn.btn-edit:hover { background: var(--blue-dim); color: var(--blue); }
.row-btn.btn-del:hover { background: var(--red-dim); color: var(--red); }
/* ── Empty state ─────────────────────────────────────────────────────────── */
.empty-state {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
padding: 5rem 2rem;
text-align: center;
}
.empty-icon {
width: 52px;
height: 52px;
border-radius: 50%;
background: var(--surface-2);
border: 1px solid var(--border);
display: flex;
align-items: center;
justify-content: center;
margin-bottom: 1.125rem;
color: var(--text-muted);
}
.empty-title { font-weight: 600; color: var(--text-2); margin-bottom: 0.3rem; font-size: 0.9375rem; }
.empty-sub { font-size: 0.8125rem; color: var(--text-muted); }
/* ── Buttons ─────────────────────────────────────────────────────────────── */
.btn {
display: inline-flex;
align-items: center;
justify-content: center;
gap: 0.375rem;
padding: 0.5rem 1rem;
border-radius: var(--radius);
font-family: var(--font-sans);
font-size: 0.875rem;
font-weight: 600;
line-height: 1;
cursor: pointer;
border: 1px solid transparent;
transition: background 0.13s, border-color 0.13s, color 0.13s, opacity 0.13s, transform 0.1s;
text-decoration: none;
white-space: nowrap;
user-select: none;
}
.btn:active { transform: scale(0.975); }
.btn:disabled { opacity: 0.45; cursor: not-allowed; transform: none !important; }
.btn-primary {
background: var(--accent);
color: #0d0d0d;
border-color: var(--accent);
}
.btn-primary:hover:not(:disabled) {
background: var(--accent-hover);
border-color: var(--accent-hover);
}
.btn-secondary {
background: var(--surface-2);
color: var(--text-2);
border-color: var(--border);
}
.btn-secondary:hover:not(:disabled) {
background: var(--surface-3);
border-color: var(--border-2);
}
.btn-danger {
background: var(--red-dim);
color: var(--red);
border-color: transparent;
}
.btn-danger:hover:not(:disabled) {
background: var(--red);
color: #fff;
}
.btn-sm {
padding: 0.375rem 0.75rem;
font-size: 0.8125rem;
}
/* ── Form controls ───────────────────────────────────────────────────────── */
.form-group {
display: flex;
flex-direction: column;
gap: 0.375rem;
}
.form-label {
font-size: 0.8125rem;
font-weight: 600;
color: var(--text-2);
letter-spacing: -0.01em;
}
.form-label .req { color: var(--red); margin-left: 2px; }
.form-hint {
font-size: 0.75rem;
color: var(--text-muted);
margin-top: 0.2rem;
}
.form-input,
.form-select,
.form-textarea {
width: 100%;
padding: 0.5625rem 0.75rem;
font-family: var(--font-sans);
font-size: 0.875rem;
line-height: 1.5;
color: var(--text);
background: var(--surface);
border: 1px solid var(--border-2);
border-radius: var(--radius);
outline: none;
transition: border-color 0.15s, box-shadow 0.15s;
-webkit-appearance: none;
-moz-appearance: none;
}
.form-input::placeholder,
.form-textarea::placeholder { color: var(--text-muted); }
.form-input:focus,
.form-select:focus,
.form-textarea:focus {
border-color: var(--accent);
box-shadow: 0 0 0 3px var(--accent-dim);
}
.form-select {
cursor: pointer;
padding-right: 2.25rem;
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' viewBox='0 0 24 24' stroke='%237d8ea8'%3E%3Cpath stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='M19 9l-7 7-7-7'/%3E%3C/svg%3E");
background-repeat: no-repeat;
background-position: right 0.625rem center;
background-size: 1em;
}
.form-textarea { resize: none; line-height: 1.65; }
.form-grid {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 1rem;
}
.form-grid .span-2 { grid-column: span 2; }
@media (max-width: 500px) {
.form-grid { grid-template-columns: 1fr; }
.form-grid .span-2 { grid-column: span 1; }
}
/* ── Modal ───────────────────────────────────────────────────────────────── */
.modal-backdrop {
position: fixed;
inset: 0;
z-index: 50;
display: flex;
align-items: center;
justify-content: center;
padding: 1rem;
background: rgba(0, 0, 0, 0);
backdrop-filter: blur(0px);
visibility: hidden;
opacity: 0;
transition:
background 0.22s ease,
backdrop-filter 0.22s ease,
opacity 0.22s ease,
visibility 0s 0.22s;
}
.modal-backdrop.is-open {
background: rgba(0,0,0,0.72);
backdrop-filter: blur(5px);
visibility: visible;
opacity: 1;
transition:
background 0.22s ease,
backdrop-filter 0.22s ease,
opacity 0.22s ease,
visibility 0s 0s;
}
.modal-box {
width: 100%;
max-width: 36rem;
background: var(--surface);
border: 1px solid var(--border-2);
border-radius: var(--radius-xl);
box-shadow: var(--shadow-xl);
display: flex;
flex-direction: column;
max-height: 90vh;
overflow-y: auto;
transform: scale(0.94) translateY(14px);
opacity: 0;
transition: transform 0.28s cubic-bezier(0.32,0.72,0,1), opacity 0.22s ease;
}
.modal-backdrop.is-open .modal-box {
transform: scale(1) translateY(0);
opacity: 1;
}
.modal-box.compact { max-width: 26rem; }
.modal-head {
display: flex;
align-items: center;
justify-content: space-between;
padding: 1.25rem 1.5rem 1.125rem;
border-bottom: 1px solid var(--border);
flex-shrink: 0;
}
.modal-title {
font-family: var(--font-display);
font-weight: 700;
font-size: 1.125rem;
letter-spacing: 0.01em;
color: var(--text);
}
.modal-body { padding: 1.25rem 1.5rem; }
.modal-foot {
display: flex;
align-items: center;
justify-content: flex-end;
gap: 0.625rem;
padding: 1rem 1.5rem;
border-top: 1px solid var(--border);
flex-shrink: 0;
}
.modal-close {
width: 28px;
height: 28px;
display: flex;
align-items: center;
justify-content: center;
border-radius: var(--radius);
background: transparent;
border: none;
cursor: pointer;
color: var(--text-muted);
transition: background 0.12s, color 0.12s;
padding: 0;
}
.modal-close:hover { background: var(--surface-2); color: var(--text); }
/* ── Alert ───────────────────────────────────────────────────────────────── */
.alert {
display: flex;
align-items: center;
gap: 0.625rem;
padding: 0.75rem 1rem;
border-radius: var(--radius-lg);
font-size: 0.875rem;
margin-bottom: 1.25rem;
animation: fadeUp 0.3s ease-out both;
}
.alert-error {
background: var(--red-dim);
border: 1px solid rgba(248,113,113,0.25);
color: var(--red);
}
.alert-success {
background: var(--green-dim);
border: 1px solid rgba(34,197,94,0.25);
color: var(--green);
}
/* ── Settings ────────────────────────────────────────────────────────────── */
.settings-card {
background: var(--surface);
border: 1px solid var(--border);
border-radius: var(--radius-xl);
padding: 2rem 2rem 1.75rem;
animation: fadeUp 0.4s ease-out both;
}
.settings-form > * + * { margin-top: 1.25rem; }
.info-box {
padding: 1rem 1.125rem;
border-radius: var(--radius-lg);
background: var(--blue-dim);
border: 1px solid rgba(96,165,250,0.2);
font-size: 0.8125rem;
color: var(--blue);
animation: fadeUp 0.4s 0.1s ease-out both;
}
html:not(.dark) .info-box {
background: rgba(37,99,235,0.06);
border-color: rgba(37,99,235,0.18);
color: var(--blue);
}
.info-box strong {
display: block;
font-weight: 600;
margin-bottom: 0.3rem;
font-size: 0.875rem;
}
/* ── Footer ──────────────────────────────────────────────────────────────── */
.app-footer {
position: relative;
text-align: center;
padding: 1.5rem;
font-size: 0.75rem;
color: var(--text-muted);
border-top: 1px solid var(--border);
}
/* ── Animations ──────────────────────────────────────────────────────────── */
@keyframes fadeUp {
from { opacity: 0; transform: translateY(10px); }
to { opacity: 1; transform: translateY(0); }
}
/* ── Scrollbar (webkit) ──────────────────────────────────────────────────── */
::-webkit-scrollbar { width: 6px; height: 6px; }
::-webkit-scrollbar-track { background: transparent; }
::-webkit-scrollbar-thumb { background: var(--border-2); border-radius: 3px; }
::-webkit-scrollbar-thumb:hover { background: var(--text-muted); }
/* ── Selection ───────────────────────────────────────────────────────────── */
::selection { background: var(--accent-dim); color: var(--text); }
+267
View File
@@ -0,0 +1,267 @@
'use strict';
// ── Dark mode toggle ──────────────────────────────────────────────────────────
document.getElementById('darkModeToggle').addEventListener('click', () => {
const isDark = document.documentElement.classList.toggle('dark');
localStorage.setItem('darkMode', isDark ? 'true' : 'false');
});
// ── Stat counter animation ────────────────────────────────────────────────────
(function animateStats() {
const DURATION = 700;
document.querySelectorAll('.stat-value[data-count]').forEach((el, i) => {
const target = parseInt(el.dataset.count, 10) || 0;
if (target === 0) return;
const delay = 80 + i * 60;
setTimeout(() => {
const start = performance.now();
const tick = (now) => {
const p = Math.min((now - start) / DURATION, 1);
// ease-out expo
const eased = p === 1 ? 1 : 1 - Math.pow(2, -10 * p);
el.textContent = Math.round(eased * target);
if (p < 1) requestAnimationFrame(tick);
};
requestAnimationFrame(tick);
}, delay);
});
})();
// ── Load bewerbungen from embedded JSON ───────────────────────────────────────
let BEWERBUNGEN = [];
const dataEl = document.getElementById('bewerbungenData');
if (dataEl) {
try { BEWERBUNGEN = JSON.parse(dataEl.textContent); } catch (_) {}
}
// ── Modal system ──────────────────────────────────────────────────────────────
function openModal(id) {
const el = document.getElementById(id);
el.classList.add('is-open');
document.body.style.overflow = 'hidden';
setTimeout(() => {
const first = el.querySelector(
'input[type="date"], input[type="text"]:not([type="hidden"]), select, textarea'
);
if (first) first.focus();
}, 60);
}
function closeModal(id) {
const el = document.getElementById(id);
el.classList.remove('is-open');
document.body.style.overflow = '';
}
function closeOnBackdrop(event, id) {
if (event.target === event.currentTarget) closeModal(id);
}
document.addEventListener('keydown', (e) => {
if (e.key !== 'Escape') return;
['bewerbungModal', 'deleteModal'].forEach(id => {
const el = document.getElementById(id);
if (el && el.classList.contains('is-open')) closeModal(id);
});
});
// ── Add modal ─────────────────────────────────────────────────────────────────
function openAddModal() {
const form = document.getElementById('bewerbungForm');
form.reset();
form.action = '/bewerbungen';
document.getElementById('modalTitle').textContent = 'Neue Bewerbung';
document.getElementById('datum').value = todayISO();
openModal('bewerbungModal');
}
// ── Edit modal ────────────────────────────────────────────────────────────────
function openEditModal(id) {
const b = BEWERBUNGEN.find(x => x.id === id);
if (!b) return;
const form = document.getElementById('bewerbungForm');
form.action = `/bewerbungen/${id}?_method=PUT`;
document.getElementById('modalTitle').textContent = 'Bewerbung bearbeiten';
document.getElementById('datum').value = b.datum || '';
document.getElementById('firma').value = b.firma || '';
document.getElementById('stelle').value = b.stelle || '';
document.getElementById('art').value = b.art || '';
document.getElementById('status').value = b.status || '';
document.getElementById('notizen').value = b.notizen || '';
openModal('bewerbungModal');
}
// ── Delete modal ──────────────────────────────────────────────────────────────
function openDeleteModal(id, firma, stelle) {
document.getElementById('deleteInfo').textContent = `${firma} ${stelle}`;
document.getElementById('deleteForm').action = `/bewerbungen/${id}?_method=DELETE`;
openModal('deleteModal');
}
// ── PDF generation ────────────────────────────────────────────────────────────
async function generatePDF(monat, jahr) {
const btn = event.currentTarget;
const saved = btn.innerHTML;
btn.disabled = true;
btn.innerHTML = '<span style="opacity:.7">Wird erstellt …</span>';
try {
const res = await fetch(`/api/pdf-daten?monat=${monat}&jahr=${jahr}`);
if (!res.ok) throw new Error(`Server ${res.status}`);
buildPDF(await res.json(), monat, jahr);
} catch (err) {
alert('PDF-Erstellung fehlgeschlagen:\n' + err.message);
} finally {
btn.disabled = false;
btn.innerHTML = saved;
}
}
function buildPDF({ bewerbungen, settings }, monat, jahr) {
const { jsPDF } = window.jspdf;
const doc = new jsPDF({ orientation: 'portrait', unit: 'mm', format: 'a4' });
const PW = doc.internal.pageSize.getWidth(); // 210 mm
const PH = doc.internal.pageSize.getHeight(); // 297 mm
const monatName = MONATE_DE[monat - 1];
const count = bewerbungen.length;
// Accent stripe
doc.setFillColor(224, 123, 0);
doc.rect(0, 0, PW, 2.5, 'F');
// ── User info (top right) ──────────────────────────────────────────────────
doc.setFontSize(9);
doc.setTextColor(120, 120, 120);
let uy = 10;
const ux = PW - 12;
if (settings?.name) {
doc.setFont('helvetica', 'bold');
doc.text(settings.name, ux, uy, { align: 'right' });
uy += 5;
doc.setFont('helvetica', 'normal');
}
if (settings?.adresse) {
settings.adresse.split('\n').forEach(line => {
doc.text(line.trim(), ux, uy, { align: 'right' });
uy += 4.5;
});
}
if (settings?.kundennummer) {
uy += 1;
doc.text(`Kundennr.: ${settings.kundennummer}`, ux, uy, { align: 'right' });
}
// ── Title ──────────────────────────────────────────────────────────────────
doc.setFontSize(17);
doc.setFont('helvetica', 'bold');
doc.setTextColor(24, 32, 47);
const titleText = `Bewerbungsaktivitäten ${monatName} ${jahr}`;
doc.text(titleText, 12, 14);
// Divider — only under the title, never touching the address block on the right
const titleWidth = doc.getTextWidth(titleText);
doc.setDrawColor(220, 226, 237);
doc.setLineWidth(0.35);
doc.line(12, 18.5, 12 + titleWidth + 6, 18.5);
// ── Summary ────────────────────────────────────────────────────────────────
doc.setFontSize(10);
doc.setFont('helvetica', 'normal');
doc.setTextColor(74, 85, 107);
const pl = count !== 1 ? 'n' : '';
doc.text(
`Im Monat ${monatName} ${jahr} habe ich mich insgesamt auf ${count} Stelle${pl} beworben.`,
12, 25
);
// ── Table ──────────────────────────────────────────────────────────────────
doc.autoTable({
startY: 31,
head: [['Datum', 'Firma / Unternehmen', 'Stelle / Position', 'Art', 'Status', 'Notizen']],
body: bewerbungen.map(b => [
formatDateDE(b.datum),
b.firma || '',
b.stelle || '',
b.art || '',
b.status || '',
b.notizen ? b.notizen.slice(0, 120) : ''
]),
margin: { left: 12, right: 12 },
styles: {
fontSize: 7,
cellPadding: 2.8,
textColor: [24, 32, 47],
lineColor: [221, 226, 239],
lineWidth: 0.25,
font: 'helvetica'
},
headStyles: {
fillColor: [224, 123, 0],
textColor: [255, 255, 255],
fontStyle: 'bold',
halign: 'left',
fontSize: 7
},
alternateRowStyles: { fillColor: [248, 250, 254] },
columnStyles: {
0: { cellWidth: 26, overflow: 'hidden' },
1: { cellWidth: 38 },
2: { cellWidth: 38 },
3: { cellWidth: 28 },
4: { cellWidth: 32 },
5: { cellWidth: 'auto' }
},
didDrawPage: ({ pageNumber }) => {
doc.setFontSize(7.5);
doc.setTextColor(160, 170, 185);
doc.text(`Seite ${pageNumber}`, PW / 2, PH - 4.5, { align: 'center' });
}
});
// ── Declaration line ───────────────────────────────────────────────────────
const finalY = doc.lastAutoTable.finalY;
const declY = finalY + 10;
doc.setFontSize(8.5);
doc.setTextColor(100, 110, 130);
doc.setFont('helvetica', 'italic');
doc.text(
'Ich erkläre hiermit, dass die vorstehenden Angaben vollständig und wahrheitsgemäß sind.',
12, declY
);
doc.save(`Bewerbungen_${monatName}_${jahr}.pdf`);
}
// ── Utilities ─────────────────────────────────────────────────────────────────
function todayISO() {
const d = new Date();
return [
d.getFullYear(),
String(d.getMonth() + 1).padStart(2, '0'),
String(d.getDate()).padStart(2, '0')
].join('-');
}
function formatDateDE(s) {
if (!s) return '';
const p = s.split('-');
return p.length === 3 ? `${p[2]}.${p[1]}.${p[0]}` : s;
}
+234
View File
@@ -0,0 +1,234 @@
'use strict';
const express = require('express');
const path = require('path');
const fs = require('fs');
const methodOverride = require('method-override');
const Database = require('better-sqlite3');
const app = express();
const PORT = process.env.PORT || 3000;
// ── Database bootstrap ────────────────────────────────────────────────────────
const dataDir = path.join(__dirname, 'data');
if (!fs.existsSync(dataDir)) fs.mkdirSync(dataDir, { recursive: true });
const db = new Database(path.join(dataDir, 'bewerbungen.db'));
db.pragma('journal_mode = WAL');
db.exec(`
CREATE TABLE IF NOT EXISTS bewerbungen (
id INTEGER PRIMARY KEY AUTOINCREMENT,
datum DATE NOT NULL,
firma TEXT NOT NULL,
stelle TEXT NOT NULL,
art TEXT,
status TEXT,
notizen TEXT,
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
updated_at DATETIME DEFAULT CURRENT_TIMESTAMP
);
CREATE TABLE IF NOT EXISTS settings (
id INTEGER PRIMARY KEY CHECK (id = 1),
name TEXT,
adresse TEXT,
kundennummer TEXT
);
INSERT OR IGNORE INTO settings (id) VALUES (1);
`);
// ── Constants ─────────────────────────────────────────────────────────────────
const ART_OPTIONEN = [
'E-Mail', 'Online-Portal', 'Indeed', 'StepStone', 'Firmenwebsite',
'Post', 'Initiativbewerbung', 'Arbeitsagentur', 'Sonstiges'
];
const STATUS_OPTIONEN = [
'Gesendet', 'Eingangsbestätigung', 'Vorstellungsgespräch',
'Absage', 'Einstellung', 'Keine Rückmeldung'
];
const MONATE = [
'Januar', 'Februar', 'März', 'April', 'Mai', 'Juni',
'Juli', 'August', 'September', 'Oktober', 'November', 'Dezember'
];
// ── Helpers ───────────────────────────────────────────────────────────────────
function sanitize(val, maxLen = 2000) {
if (val === null || val === undefined) return '';
return String(val).trim().slice(0, maxLen);
}
function safeJson(obj) {
return JSON.stringify(obj)
.replace(/</g, '\\u003c')
.replace(/>/g, '\\u003e')
.replace(/&/g, '\\u0026');
}
function getSettings() {
return db.prepare('SELECT * FROM settings WHERE id = 1').get() || {};
}
// ── Middleware ────────────────────────────────────────────────────────────────
app.use(express.urlencoded({ extended: true }));
app.use(express.json());
app.use(methodOverride('_method'));
app.use(express.static(path.join(__dirname, 'public')));
app.set('view engine', 'ejs');
app.set('views', path.join(__dirname, 'views'));
// ── Routes ────────────────────────────────────────────────────────────────────
// GET / Übersicht
app.get('/', (req, res) => {
const now = new Date();
const monat = Math.min(12, Math.max(1, parseInt(req.query.monat) || (now.getMonth() + 1)));
const jahr = parseInt(req.query.jahr) || now.getFullYear();
const monatStr = String(monat).padStart(2, '0');
const vonDatum = `${jahr}-${monatStr}-01`;
const bisDatum = `${jahr}-${monatStr}-31`;
const bewerbungen = db.prepare(`
SELECT * FROM bewerbungen
WHERE datum BETWEEN ? AND ?
ORDER BY datum DESC, id DESC
`).all(vonDatum, bisDatum);
// Statistics
const stats = { gesamt: bewerbungen.length, positiv: 0, absagen: 0, ausstehend: 0 };
for (const b of bewerbungen) {
if (b.status === 'Vorstellungsgespräch' || b.status === 'Einstellung') stats.positiv++;
else if (b.status === 'Absage') stats.absagen++;
else stats.ausstehend++;
}
// Years for filter dropdown
const dbJahre = db.prepare(
`SELECT DISTINCT strftime('%Y', datum) AS j FROM bewerbungen ORDER BY j DESC`
).all().map(r => parseInt(r.j)).filter(Boolean);
const jahre = [...new Set([...dbJahre, now.getFullYear()])].sort((a, b) => b - a);
res.render('index', {
bewerbungen,
bewerbungenJson: safeJson(bewerbungen),
stats,
monat,
jahr,
monate: MONATE,
jahre,
artOptionen: ART_OPTIONEN,
statusOptionen: STATUS_OPTIONEN,
settings: getSettings(),
monatName: MONATE[monat - 1],
fehler: req.query.fehler || null,
currentPage: 'uebersicht'
});
});
// POST /bewerbungen Neu anlegen
app.post('/bewerbungen', (req, res) => {
const { datum, firma, stelle, art, status, notizen } = req.body;
if (!datum || !firma || !stelle) {
const d = new Date();
return res.redirect(`/?fehler=pflichtfelder&monat=${d.getMonth() + 1}&jahr=${d.getFullYear()}`);
}
db.prepare(`
INSERT INTO bewerbungen (datum, firma, stelle, art, status, notizen)
VALUES (?, ?, ?, ?, ?, ?)
`).run(
sanitize(datum, 20),
sanitize(firma),
sanitize(stelle),
sanitize(art),
sanitize(status),
sanitize(notizen)
);
const d = new Date(datum + 'T00:00:00');
res.redirect(`/?monat=${d.getMonth() + 1}&jahr=${d.getFullYear()}`);
});
// PUT /bewerbungen/:id Aktualisieren
app.put('/bewerbungen/:id', (req, res) => {
const id = parseInt(req.params.id);
const { datum, firma, stelle, art, status, notizen, monat, jahr } = req.body;
if (!datum || !firma || !stelle) {
return res.redirect(`/?fehler=pflichtfelder&monat=${monat}&jahr=${jahr}`);
}
db.prepare(`
UPDATE bewerbungen
SET datum=?, firma=?, stelle=?, art=?, status=?, notizen=?, updated_at=CURRENT_TIMESTAMP
WHERE id=?
`).run(
sanitize(datum, 20),
sanitize(firma),
sanitize(stelle),
sanitize(art),
sanitize(status),
sanitize(notizen),
id
);
res.redirect(`/?monat=${monat || 1}&jahr=${jahr || new Date().getFullYear()}`);
});
// DELETE /bewerbungen/:id Löschen
app.delete('/bewerbungen/:id', (req, res) => {
db.prepare('DELETE FROM bewerbungen WHERE id=?').run(parseInt(req.params.id));
const { monat, jahr } = req.body;
const now = new Date();
res.redirect(`/?monat=${monat || now.getMonth() + 1}&jahr=${jahr || now.getFullYear()}`);
});
// GET /einstellungen
app.get('/einstellungen', (req, res) => {
res.render('einstellungen', {
settings: getSettings(),
gespeichert: req.query.gespeichert === '1',
currentPage: 'einstellungen'
});
});
// POST /einstellungen
app.post('/einstellungen', (req, res) => {
const { name, adresse, kundennummer } = req.body;
db.prepare(`
INSERT OR REPLACE INTO settings (id, name, adresse, kundennummer)
VALUES (1, ?, ?, ?)
`).run(sanitize(name), sanitize(adresse), sanitize(kundennummer));
res.redirect('/einstellungen?gespeichert=1');
});
// GET /api/pdf-daten JSON für clientseitige PDF-Generierung
app.get('/api/pdf-daten', (req, res) => {
const monat = Math.min(12, Math.max(1, parseInt(req.query.monat) || 1));
const jahr = parseInt(req.query.jahr) || new Date().getFullYear();
const ms = String(monat).padStart(2, '0');
const bewerbungen = db.prepare(`
SELECT * FROM bewerbungen
WHERE datum BETWEEN ? AND ?
ORDER BY datum ASC, id ASC
`).all(`${jahr}-${ms}-01`, `${jahr}-${ms}-31`);
res.json({ bewerbungen, settings: getSettings(), monat, jahr });
});
// ── Start ─────────────────────────────────────────────────────────────────────
app.listen(PORT, () => {
console.log(`\n✓ Bewerbungs-Tracker läuft auf → http://localhost:${PORT}\n`);
});
+62
View File
@@ -0,0 +1,62 @@
<%- include('partials/header') %>
<div style="max-width:42rem;">
<div class="page-header" style="margin-bottom:1.5rem;">
<div>
<h1 class="page-title">Einstellungen</h1>
<p class="page-meta">Diese Daten erscheinen automatisch in jedem PDF-Export.</p>
</div>
</div>
<% if (gespeichert) { %>
<div class="alert alert-success">
<svg width="16" height="16" fill="none" stroke="currentColor" viewBox="0 0 24 24" style="flex-shrink:0">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z"/>
</svg>
Einstellungen wurden gespeichert.
</div>
<% } %>
<div class="settings-card">
<form method="POST" action="/einstellungen" class="settings-form">
<div class="form-group">
<label class="form-label" for="name">Vollständiger Name</label>
<input type="text" id="name" name="name" maxlength="500" class="form-input"
value="<%= settings.name || '' %>"
placeholder="Vorname Nachname">
<p class="form-hint">Erscheint als Absender und in der Unterschriftszeile des PDFs.</p>
</div>
<div class="form-group">
<label class="form-label" for="adresse">Adresse</label>
<textarea id="adresse" name="adresse" rows="3" maxlength="1000" class="form-textarea"
placeholder="Musterstraße 1&#10;12345 Musterstadt"><%= settings.adresse || '' %></textarea>
<p class="form-hint">Mehrzeilig erscheint im Briefkopf des PDFs.</p>
</div>
<div class="form-group">
<label class="form-label" for="kundennummer">Jobcenter-Kundennummer</label>
<input type="text" id="kundennummer" name="kundennummer" maxlength="100" class="form-input"
value="<%= settings.kundennummer || '' %>"
placeholder="z.B. BG-12345678">
<p class="form-hint">Ihre Kundennummer beim Jobcenter / der Agentur für Arbeit.</p>
</div>
<div style="display:flex;justify-content:flex-end;padding-top:.5rem;">
<button type="submit" class="btn btn-primary">Einstellungen speichern</button>
</div>
</form>
</div>
<div class="info-box" style="margin-top:1rem;">
<strong>PDF-Hinweis</strong>
Das exportierte PDF enthält Ihren Briefkopf mit den obigen Daten, einen Zusammenfassungssatz,
eine vollständige Bewerbungstabelle sowie eine Bestätigungs- und Unterschriftszeile.
Starten Sie den Export über „PDF exportieren" auf der Übersichtsseite.
</div>
</div>
<%- include('partials/footer') %>
+298
View File
@@ -0,0 +1,298 @@
<%- include('partials/header') %>
<%# ── Embed server data for client-side use ──────────────────────────────── %>
<script type="application/json" id="bewerbungenData"><%- bewerbungenJson %></script>
<script>
const CURRENT_MONAT = <%= monat %>;
const CURRENT_JAHR = <%= jahr %>;
const MONATE_DE = <%- JSON.stringify(monate) %>;
</script>
<%# ── Page header ────────────────────────────────────────────────────────── %>
<div class="page-header">
<div>
<h1 class="page-title">Bewerbungsübersicht</h1>
<div class="page-meta">
<span><%= monatName %> <%= jahr %></span>
<% if (settings && settings.kundennummer) { %>
<span class="sep">&middot;</span>
<span class="tag">KdNr. <%= settings.kundennummer %></span>
<% } %>
</div>
</div>
<%# Filter + PDF %>
<form method="GET" action="/" class="filter-bar">
<select name="monat" class="form-select btn-sm" style="width:auto;">
<% monate.forEach((m, i) => { %>
<option value="<%= i + 1 %>" <%= (i + 1 === monat) ? 'selected' : '' %>><%= m %></option>
<% }) %>
</select>
<select name="jahr" class="form-select btn-sm" style="width:auto;">
<% jahre.forEach(j => { %>
<option value="<%= j %>" <%= (j === jahr) ? 'selected' : '' %>><%= j %></option>
<% }) %>
</select>
<button type="submit" class="btn btn-secondary btn-sm">Filtern</button>
<button type="button" class="btn btn-primary btn-sm" onclick="generatePDF(<%= monat %>, <%= jahr %>)">
<svg width="14" height="14" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
d="M12 10v6m0 0l-3-3m3 3l3-3m2 8H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
PDF exportieren
</button>
</form>
</div>
<%# ── Error banner ───────────────────────────────────────────────────────── %>
<% if (fehler === 'pflichtfelder') { %>
<div class="alert alert-error">
<svg width="16" height="16" fill="none" stroke="currentColor" viewBox="0 0 24 24" style="flex-shrink:0">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
d="M12 9v2m0 4h.01m-6.938 4h13.856c1.54 0 2.502-1.667 1.732-3L13.732 4c-.77-1.333-2.694-1.333-3.464 0L3.34 16c-.77 1.333.192 3 1.732 3z"/>
</svg>
Bitte alle Pflichtfelder ausfüllen: Datum, Firma und Stelle sind erforderlich.
</div>
<% } %>
<%# ── Stats grid ─────────────────────────────────────────────────────────── %>
<div class="stats-grid">
<div class="stat-card s-total">
<div class="stat-label">Gesamt</div>
<div class="stat-value" data-count="<%= stats.gesamt %>"><%= stats.gesamt %></div>
</div>
<div class="stat-card s-positiv">
<div class="stat-label">Positiv</div>
<div class="stat-value" data-count="<%= stats.positiv %>"><%= stats.positiv %></div>
</div>
<div class="stat-card s-absage">
<div class="stat-label">Absagen</div>
<div class="stat-value" data-count="<%= stats.absagen %>"><%= stats.absagen %></div>
</div>
<div class="stat-card s-pending">
<div class="stat-label">Ausstehend</div>
<div class="stat-value" data-count="<%= stats.ausstehend %>"><%= stats.ausstehend %></div>
</div>
</div>
<%# ── Action bar ─────────────────────────────────────────────────────────── %>
<div class="action-bar">
<span class="entry-count"><%= bewerbungen.length %> Eintr<%= bewerbungen.length === 1 ? 'ag' : 'äge' %></span>
<button class="btn btn-primary btn-sm" onclick="openAddModal()">
<svg width="14" height="14" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2.5" d="M12 4v16m8-8H4"/>
</svg>
Neue Bewerbung
</button>
</div>
<%# ── Data table ─────────────────────────────────────────────────────────── %>
<div class="data-card">
<% if (bewerbungen.length === 0) { %>
<div class="empty-state">
<div class="empty-icon">
<svg width="22" height="22" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="1.5"
d="M9 12h6m-6 4h6m2 5H7a2 2 0 01-2-2V5a2 2 0 012-2h5.586a1 1 0 01.707.293l5.414 5.414a1 1 0 01.293.707V19a2 2 0 01-2 2z"/>
</svg>
</div>
<p class="empty-title">Keine Bewerbungen für <%= monatName %> <%= jahr %></p>
<p class="empty-sub">Klicken Sie auf „Neue Bewerbung", um Ihren ersten Eintrag hinzuzufügen.</p>
</div>
<% } else { %>
<div class="overflow-x-auto">
<table class="data-table">
<thead>
<tr>
<th>Datum</th>
<th>Firma</th>
<th>Stelle</th>
<th>Art</th>
<th>Status</th>
<th class="hidden md:table-cell">Notizen</th>
<th style="text-align:right; padding-right:1.25rem;">Aktionen</th>
</tr>
</thead>
<tbody>
<% bewerbungen.forEach(b => { %>
<tr>
<td class="td-date">
<%= b.datum ? b.datum.split('-').reverse().join('.') : '' %>
</td>
<td class="td-firma"><%= b.firma %></td>
<td class="td-stelle"><%= b.stelle %></td>
<td>
<% if (b.art) { %>
<span class="td-art"><%= b.art %></span>
<% } %>
</td>
<td>
<% if (b.status) { %>
<span class="status-chip <%= statusClass(b.status) %>"><%= b.status %></span>
<% } %>
</td>
<td class="td-notizen hidden md:table-cell"><%= b.notizen || '' %></td>
<td>
<div class="row-actions flex items-center justify-end gap-1" style="padding-right:0.25rem;">
<button class="row-btn btn-edit" title="Bearbeiten"
onclick="openEditModal(<%= b.id %>)">
<svg width="15" height="15" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
d="M11 5H6a2 2 0 00-2 2v11a2 2 0 002 2h11a2 2 0 002-2v-5m-1.414-9.414a2 2 0 112.828 2.828L11.828 15H9v-2.828l8.586-8.586z"/>
</svg>
</button>
<button class="row-btn btn-del" title="Löschen"
onclick="openDeleteModal(<%= b.id %>, '<%= b.firma.replace(/\\/g, '\\\\').replace(/'/g, "\\'") %>', '<%= b.stelle.replace(/\\/g, '\\\\').replace(/'/g, "\\'") %>')">
<svg width="15" height="15" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
d="M19 7l-.867 12.142A2 2 0 0116.138 21H7.862a2 2 0 01-1.995-1.858L5 7m5 4v6m4-6v6m1-10V4a1 1 0 00-1-1h-4a1 1 0 00-1 1v3M4 7h16"/>
</svg>
</button>
</div>
</td>
</tr>
<% }) %>
</tbody>
</table>
</div>
<% } %>
</div>
<%# ══════════════════════════════════════════════════════════════════════════ %>
<%# Modal: Neue / Bewerbung bearbeiten %>
<%# ══════════════════════════════════════════════════════════════════════════ %>
<div id="bewerbungModal" class="modal-backdrop" onclick="closeOnBackdrop(event,'bewerbungModal')">
<div class="modal-box" role="dialog" aria-modal="true" aria-labelledby="modalTitle">
<div class="modal-head">
<h2 id="modalTitle" class="modal-title">Neue Bewerbung</h2>
<button class="modal-close" onclick="closeModal('bewerbungModal')" aria-label="Schließen">
<svg width="16" height="16" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12"/>
</svg>
</button>
</div>
<form id="bewerbungForm" method="POST" action="/bewerbungen">
<input type="hidden" name="monat" value="<%= monat %>">
<input type="hidden" name="jahr" value="<%= jahr %>">
<div class="modal-body">
<div class="form-grid">
<div class="form-group">
<label class="form-label" for="datum">Datum <span class="req">*</span></label>
<input type="date" id="datum" name="datum" required class="form-input">
</div>
<div class="form-group">
<label class="form-label" for="art">Art der Bewerbung</label>
<select id="art" name="art" class="form-select">
<option value="">— bitte wählen —</option>
<% artOptionen.forEach(a => { %>
<option value="<%= a %>"><%= a %></option>
<% }) %>
</select>
</div>
<div class="form-group span-2">
<label class="form-label" for="firma">Firma / Unternehmen <span class="req">*</span></label>
<input type="text" id="firma" name="firma" required maxlength="500" class="form-input"
placeholder="z.B. Muster GmbH">
</div>
<div class="form-group span-2">
<label class="form-label" for="stelle">Stelle / Position <span class="req">*</span></label>
<input type="text" id="stelle" name="stelle" required maxlength="500" class="form-input"
placeholder="z.B. Fachinformatiker / Softwareentwickler">
</div>
<div class="form-group span-2">
<label class="form-label" for="status">Status</label>
<select id="status" name="status" class="form-select">
<option value="">— bitte wählen —</option>
<% statusOptionen.forEach(s => { %>
<option value="<%= s %>"><%= s %></option>
<% }) %>
</select>
</div>
<div class="form-group span-2">
<label class="form-label" for="notizen">Notizen</label>
<textarea id="notizen" name="notizen" rows="3" maxlength="2000" class="form-textarea"
placeholder="Ansprechpartner, Referenznummer, Gehaltsvorstellung …"></textarea>
</div>
</div>
</div>
<div class="modal-foot">
<button type="button" class="btn btn-secondary" onclick="closeModal('bewerbungModal')">Abbrechen</button>
<button type="submit" class="btn btn-primary">Speichern</button>
</div>
</form>
</div>
</div>
<%# ══════════════════════════════════════════════════════════════════════════ %>
<%# Modal: Löschen bestätigen %>
<%# ══════════════════════════════════════════════════════════════════════════ %>
<div id="deleteModal" class="modal-backdrop" onclick="closeOnBackdrop(event,'deleteModal')">
<div class="modal-box compact" role="dialog" aria-modal="true">
<div class="modal-head">
<h2 class="modal-title">Eintrag löschen</h2>
<button class="modal-close" onclick="closeModal('deleteModal')" aria-label="Schließen">
<svg width="16" height="16" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M6 18L18 6M6 6l12 12"/>
</svg>
</button>
</div>
<div class="modal-body">
<div class="flex gap-3">
<div class="flex-shrink-0 flex items-center justify-center w-10 h-10 rounded-full"
style="background:var(--red-dim);">
<svg width="18" height="18" fill="none" stroke="var(--red)" viewBox="0 0 24 24">
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2"
d="M12 9v2m0 4h.01m-6.938 4h13.856c1.54 0 2.502-1.667 1.732-3L13.732 4c-.77-1.333-2.694-1.333-3.464 0L3.34 16c-.77 1.333.192 3 1.732 3z"/>
</svg>
</div>
<div>
<p style="font-size:.9375rem;font-weight:600;color:var(--text);margin-bottom:.375rem;">
Sicher löschen?
</p>
<p id="deleteInfo" style="font-size:.875rem;color:var(--text-2);margin-bottom:.5rem;"></p>
<p style="font-size:.75rem;color:var(--text-muted);">
Diese Aktion kann nicht rückgängig gemacht werden.
</p>
</div>
</div>
</div>
<form id="deleteForm" method="POST">
<input type="hidden" name="monat" value="<%= monat %>">
<input type="hidden" name="jahr" value="<%= jahr %>">
<div class="modal-foot">
<button type="button" class="btn btn-secondary" onclick="closeModal('deleteModal')">Abbrechen</button>
<button type="submit" class="btn btn-danger">Endgültig löschen</button>
</div>
</form>
</div>
</div>
<%# Server-side helper: map status string → CSS class ─────────────────────── %>
<% function statusClass(s) {
const map = {
'Gesendet': 'st-gesendet',
'Eingangsbestätigung': 'st-eingang',
'Vorstellungsgespräch': 'st-vorstellung',
'Absage': 'st-absage',
'Einstellung': 'st-einstellung',
'Keine Rückmeldung': 'st-keine'
};
return map[s] || 'st-keine';
} %>
<%- include('partials/footer') %>
+9
View File
@@ -0,0 +1,9 @@
</div><%# /page-wrapper %>
<footer class="app-footer">
Bewerbungs-Tracker &mdash; Lokale Bewerbungsverwaltung
</footer>
<script src="/js/main.js"></script>
</body>
</html>
+49
View File
@@ -0,0 +1,49 @@
<!DOCTYPE html>
<html lang="de">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Bewerbungs-Tracker</title>
<%# Synchronously apply saved dark mode preference before paint to prevent flash %>
<script>
(function () {
if (localStorage.getItem('darkMode') !== 'false') {
document.documentElement.classList.add('dark');
}
})();
</script>
<%# Tailwind CDN — used for layout utilities (flex, grid, overflow, responsive) %>
<script src="https://cdn.tailwindcss.com"></script>
<script>
tailwind.config = {
darkMode: 'class',
corePlugins: { preflight: false }
};
</script>
<%# jsPDF + autoTable for client-side PDF generation %>
<script src="https://cdnjs.cloudflare.com/ajax/libs/jspdf/2.5.1/jspdf.umd.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/jspdf-autotable/3.8.2/jspdf.plugin.autotable.min.js"></script>
<link rel="stylesheet" href="/css/style.css">
</head>
<body>
<nav class="app-nav">
<a href="/" class="nav-brand">
<span class="brand-dot"></span>
Bewerbungs-Tracker
</a>
<div class="nav-links">
<a href="/" class="nav-link <%= (typeof currentPage !== 'undefined' && currentPage === 'uebersicht') ? 'active' : '' %>">Übersicht</a>
<a href="/einstellungen" class="nav-link <%= (typeof currentPage !== 'undefined' && currentPage === 'einstellungen') ? 'active' : '' %>">Einstellungen</a>
</div>
<button id="darkModeToggle" class="dark-toggle" title="Darkmodus umschalten" aria-label="Darkmodus umschalten"></button>
</nav>
<div class="page-wrapper">