46 Commits

Author SHA1 Message Date
2f6138b1d6 Bump version to 1.4.13 2025-08-03 12:06:24 +02:00
f5a39e80b4 Taschenrechner-Features hinzugefügt und README aktualisiert 2025-08-03 12:05:02 +02:00
f9f73e24c9 Fix Back-Forward-Cache Lighthouse error by adding proper Cache-Control headers 2025-08-03 10:18:00 +02:00
d697928241 Bump version to 1.4.10 2025-08-03 10:04:09 +02:00
f998f7fff8 Fix JSON syntax error in swagger.json - remove invalid .v{ prefix 2025-08-03 10:01:37 +02:00
1a5aa003a2 v1.4.9: Verbesserte Sprachausgabe mit englischer Unterstützung und Service Worker Fix 2025-08-02 19:54:52 +02:00
eecc2b8b73 Sprachausgabe/englisch gefixt 2025-08-02 19:52:31 +02:00
0b13a408cd Update meta keywords 2025-08-02 19:08:48 +02:00
c4a65bba48 Version 2025-08-02 18:39:30 +02:00
e4b37d9261 Fehler in der API behoben 2025-08-02 18:37:17 +02:00
45cc02b4b0 Fehler in API behoben 2025-08-02 18:30:16 +02:00
05766d9a97 Version auf 1.4.7 erhöht - Dashboard mit Toggle-Funktionalität 2025-08-02 14:28:22 +02:00
e5fbc14a34 Dashboard erweitert: Toggle zwischen Wochen- und 24-Stunden-Verlauf für alle Charts 2025-08-02 14:25:07 +02:00
9e025bd4c7 Überarbeite Help Modal: Floating Schließen-Button und mehrsprachige Unterstützung
- Schließen-Button ist jetzt 'floating' mit position: fixed
- Button hat Hintergrund, Rahmen und Schatten für bessere Sichtbarkeit
- Alle Texte im Help Modal verwenden jetzt Übersetzungsfunktionen
- Vollständige mehrsprachige Unterstützung (Deutsch/Englisch)
- Bessere mobile Darstellung ohne Überschneidungen
2025-08-02 14:17:46 +02:00
f4ffd14624 Update CLOC 2025-08-02 11:38:28 +02:00
4740288c45 Desktop-Layout: Sprachauswahl und Hilfe-Button etwas nach oben verschoben für bessere Balance 2025-08-02 11:31:30 +02:00
512898b34b Fix mobile layout: Verbesserte Lösung für Sprachauswahl-Überlappung mit mehr Abstand 2025-08-02 11:25:22 +02:00
872d0f9e23 Fix: Button 'Berechnen' wird jetzt korrekt als 'Calculate' in englischer Version übersetzt 2025-08-02 11:18:19 +02:00
28fda213ba Fix mobile layout: Sprachauswahl überlappt nicht mehr mit Überschrift und korrigiere URL-Parameter für Sprachwechsel 2025-08-02 11:12:37 +02:00
bdf4e134e4 Version 2025-08-02 08:50:26 +02:00
601f993ccb Scrollbar-Optimierungen und Cookie-Bereinigung 2025-08-02 08:49:46 +02:00
8fdf764a7b Version 1.4.1: Scrollbar-Optimierungen und Cookie-Bereinigung 2025-08-02 08:41:34 +02:00
b40bb666b8 Manuelle Änderungen in README.md 2025-08-01 16:47:18 +02:00
7dbe91b32e APP_VERSION 2025-08-01 16:41:36 +02:00
c7d95e5c4c feat: Implementiere mehrsprachige Unterstützung (i18n)
- Füge Flask-Babel für professionelle i18n-Implementierung hinzu
- Implementiere automatische Browser-Spracherkennung
- Erstelle datenschutzfreundliche Sprachauswahl ohne Cookies
- Verwende URL-Parameter und localStorage für Sprachauswahl
- Füge vollständige Übersetzungen für Deutsch und Englisch hinzu
- Implementiere responsive Dropdown-Sprachauswahl mit Landesflaggen
- Verbessere Barrierefreiheit mit ARIA-Attributen und Screenreader-Support
- Aktualisiere README mit i18n-Dokumentation
- Version 1.4.0
2025-08-01 16:40:46 +02:00
d88581a663 Entferne idea.txt aus dem Repository 2025-08-01 15:38:17 +02:00
00e961a6bd Entferne versehentliche less-Hilfedatei 2025-08-01 15:36:22 +02:00
d9ecdbb86e Update cloc 2025-08-01 15:28:58 +02:00
a3753e3f4e Remove dash 2025-08-01 15:27:09 +02:00
af5ff2c094 Lighthouse-Dateien in eigenen Ordner verschoben 2025-08-01 15:26:00 +02:00
200a46fdaa Add Lighthouse report JSON 2025-08-01 14:57:04 +02:00
524f44b6f0 Lighthouse-Badges auf 100% gesetzt (basierend auf aktuellem Audit) 2025-08-01 14:50:37 +02:00
3f3cb3ed01 Wikipedia link 2025-08-01 14:44:41 +02:00
74d9c18bd9 Rearrange text 2025-08-01 14:42:18 +02:00
1b91f8e54d Punkt. 2025-08-01 14:40:48 +02:00
a594d37cf1 Fix markdown linter errors 2025-08-01 14:37:47 +02:00
b95b05e54f Remove bold 2025-08-01 14:36:14 +02:00
2271934228 Add image descriptions 2025-08-01 14:35:29 +02:00
dc73e49da0 Update cloc 2025-08-01 14:34:05 +02:00
030e4adab9 Update screenshot 2025-08-01 14:33:02 +02:00
bd895e356f Verbesserte Test-Coverage auf 90% und Coverage-Badge hinzugefügt 2025-08-01 14:29:15 +02:00
046271343d Updated lighthouse-score.pdf 2025-08-01 14:10:48 +02:00
d0d8e0aeb1 v1.3.13: Weitere Verbesserung der Farbkontraste für Wochentag- und Kalenderwoche-Header 2025-08-01 13:56:34 +02:00
c19cb17623 v1.3.12: Verbesserte Farbkontraste für bessere Barrierefreiheit 2025-08-01 13:49:02 +02:00
e2367d0b0e chore: Version auf 1.3.11 erhöht - Sprachausgabe-Funktion und verbesserte Feiertage-Anzeige 2025-08-01 13:10:31 +02:00
1eb55e32dc feat: Sprachausgabe-Funktion für barrierefreie Nutzung hinzugefügt
- Vorlesen-Buttons (🔊) bei allen Ergebnissen
- Web Speech API mit deutscher Sprachausgabe
- Vollständige Tastaturnavigation (Tab, Enter, Leertaste)
- ESC-Taste zum Stoppen der Wiedergabe
- Barrierefreiheit verbessert für Menschen mit Sehbehinderungen
- README aktualisiert mit Sprachausgabe-Dokumentation
2025-08-01 12:51:28 +02:00
20 changed files with 20383 additions and 266 deletions

View File

@@ -1,6 +1,12 @@
# Elpatrons Datumsrechner # Elpatrons Datumsrechner
Diese moderne Python-Webanwendung (Flask) ermöglicht verschiedene Datumsberechnungen über eine übersichtliche Weboberfläche: [![Test Coverage](https://img.shields.io/badge/test%20coverage-90%25-brightgreen)](https://github.com/elpatron/datecalc)
[![Lighthouse Performance](https://img.shields.io/badge/lighthouse%20performance-100%25-brightgreen)](https://date.elpatron.me)
[![Lighthouse Accessibility](https://img.shields.io/badge/lighthouse%20accessibility-100%25-brightgreen)](https://date.elpatron.me)
[![Lighthouse Best Practices](https://img.shields.io/badge/lighthouse%20best%20practices-100%25-brightgreen)](https://date.elpatron.me)
[![Lighthouse SEO](https://img.shields.io/badge/lighthouse%20seo-100%25-brightgreen)](https://date.elpatron.me)
Diese moderne Python-Webanwendung (Flask) ermöglicht verschiedene Datumsberechnungen über eine übersichtliche, barrierefreie Weboberfläche.
## Inhaltsverzeichnis ## Inhaltsverzeichnis
@@ -36,7 +42,13 @@ Diese moderne Python-Webanwendung (Flask) ermöglicht verschiedene Datumsberechn
Datumsrechner Live: [https://date.elpatron.me](https://date.elpatron.me) Datumsrechner Live: [https://date.elpatron.me](https://date.elpatron.me)
![image-20250725095959116](./assets/image-20250725095959116.png) ![App Screenshot](./assets/image-20250725095959116.png)
**[Lighthouse](https://en.wikipedia.org/wiki/Lighthouse_(software))-Performance-Score:**
Die Webanwendung erreicht hervorragende Performance-Werte in allen Kategorien (Performance, Accessibility, Best Practices, SEO).
[Lighthouse-Ergebnis (PDF)](./lighthouse/lighthouse-score.pdf)
## Funktionen ## Funktionen
@@ -48,6 +60,9 @@ Datumsrechner Live: [https://date.elpatron.me](https://date.elpatron.me)
- Datum plus/minus X Wochen/Monate - Datum plus/minus X Wochen/Monate
- Kalenderwoche zu Datum - Kalenderwoche zu Datum
- Start-/Enddatum einer Kalenderwoche eines Jahres - Start-/Enddatum einer Kalenderwoche eines Jahres
- **Integrierter Taschenrechner** mit History und Sprachausgabe
- Mehrsprachige Unterstützung (Deutsch/Englisch) mit automatischer Browser-Spracherkennung
- Sprachausgabe für alle Ergebnisse (barrierefrei)
- Statistik-Dashboard mit Passwortschutz unter `/stats` - Statistik-Dashboard mit Passwortschutz unter `/stats`
## Bundesland-Feiertage ## Bundesland-Feiertage
@@ -74,6 +89,35 @@ Die Werktagsberechnung kann optional bundeslandspezifische Feiertage berücksich
Die Feiertage werden automatisch für den gewählten Zeitraum abgerufen und bei der Werktagsberechnung als arbeitsfreie Tage behandelt. Im Ergebnis werden zusätzlich die Anzahl der Wochenendtage und Feiertage angezeigt. Die Feiertage werden automatisch für den gewählten Zeitraum abgerufen und bei der Werktagsberechnung als arbeitsfreie Tage behandelt. Im Ergebnis werden zusätzlich die Anzahl der Wochenendtage und Feiertage angezeigt.
## Mehrsprachige Unterstützung (i18n)
Die Anwendung unterstützt Deutsch und Englisch mit folgenden Features:
### Automatische Spracherkennung:
- *Browser-Sprache*: Automatische Erkennung der Browser-Einstellung
- *URL-Parameter*: Sprachauswahl über `?lang=de` oder `?lang=en`
- *localStorage*: Persistente Sprachauswahl im Browser
- *Fallback*: Deutsch als Standardsprache
### *Datenschutzfreundliche Implementierung:*
- *Keine Cookies*: Sprachauswahl ohne Cookies
- *URL-Parameter*: Transparente Sprachauswahl in der URL
- *localStorage*: Lokale Speicherung im Browser
- *Teilbare URLs*: URLs mit Sprachauswahl können geteilt werden
### *Barrierefreiheit:*
- *Screenreader*: Vollständige Unterstützung
- *Tastatur-Navigation*: Vollständig bedienbar
- *ARIA-Attribute*: Korrekte Beschriftungen
- *Semantische HTML*: Korrekte Struktur
- *Taschenrechner*: Vollständig barrierefrei mit Tastatur-Bedienung und Sprachausgabe
### *Technische Details:*
- *Flask-Babel*: Professionelle i18n-Implementierung
- *Gettext*: Standard für Übersetzungen
- *Responsive Design*: Angepasst für alle Geräte
- *SEO-freundlich*: URLs sind indexierbar
## Installation (lokal) ## Installation (lokal)
1. Python 3.8+ installieren 1. Python 3.8+ installieren
@@ -95,7 +139,7 @@ Die App ist dann unter http://localhost:5000 erreichbar.
Das Dashboard ist mit einem statischen Passwort geschützt, das über die Umgebungsvariable `STATS_PASSWORD` gesetzt wird. Das Dashboard ist mit einem statischen Passwort geschützt, das über die Umgebungsvariable `STATS_PASSWORD` gesetzt wird.
![image-20250725100127004](./assets/image-20250725100127004.png) ![Statistics page](./assets/image-20250725100127004.png)
Beispiel (PowerShell): Beispiel (PowerShell):
@@ -178,6 +222,7 @@ Alle Datumsfunktionen stehen auch als REST-API zur Verfügung. Die API akzeptier
``` ```
**Mit curl:** **Mit curl:**
```bash ```bash
curl -X POST http://localhost:5000/api/tage_werktage \ curl -X POST http://localhost:5000/api/tage_werktage \
-H "Content-Type: application/json" \ -H "Content-Type: application/json" \
@@ -185,6 +230,7 @@ curl -X POST http://localhost:5000/api/tage_werktage \
``` ```
**Antwort:** **Antwort:**
```json ```json
{ "result": 7 } { "result": 7 }
``` ```
@@ -200,6 +246,7 @@ curl -X POST http://localhost:5000/api/tage_werktage \
``` ```
**Mit curl:** **Mit curl:**
```bash ```bash
curl -X POST http://localhost:5000/api/wochentag \ curl -X POST http://localhost:5000/api/wochentag \
-H "Content-Type: application/json" \ -H "Content-Type: application/json" \
@@ -207,6 +254,7 @@ curl -X POST http://localhost:5000/api/wochentag \
``` ```
**Antwort:** **Antwort:**
```json ```json
{ "result": "Montag" } { "result": "Montag" }
``` ```
@@ -220,6 +268,7 @@ curl -X POST http://localhost:5000/api/wochentag \
``` ```
**Mit curl:** **Mit curl:**
```bash ```bash
curl -X POST http://localhost:5000/api/kw_berechnen \ curl -X POST http://localhost:5000/api/kw_berechnen \
-H "Content-Type: application/json" \ -H "Content-Type: application/json" \
@@ -227,6 +276,7 @@ curl -X POST http://localhost:5000/api/kw_berechnen \
``` ```
**Antwort:** **Antwort:**
```json ```json
{ "result": "KW 24 (2024)", "kw": 24, "jahr": 2024 } { "result": "KW 24 (2024)", "kw": 24, "jahr": 2024 }
``` ```
@@ -240,6 +290,7 @@ curl -X POST http://localhost:5000/api/kw_berechnen \
``` ```
**Mit curl:** **Mit curl:**
```bash ```bash
curl -X POST http://localhost:5000/api/kw_datum \ curl -X POST http://localhost:5000/api/kw_datum \
-H "Content-Type: application/json" \ -H "Content-Type: application/json" \
@@ -247,6 +298,7 @@ curl -X POST http://localhost:5000/api/kw_datum \
``` ```
**Antwort:** **Antwort:**
```json ```json
{ {
"result": "10.06.2024 bis 16.06.2024", "result": "10.06.2024 bis 16.06.2024",
@@ -270,6 +322,7 @@ curl -X POST http://localhost:5000/api/kw_datum \
``` ```
**Mit curl:** **Mit curl:**
```bash ```bash
curl -X POST http://localhost:5000/api/plusminus \ curl -X POST http://localhost:5000/api/plusminus \
-H "Content-Type: application/json" \ -H "Content-Type: application/json" \
@@ -277,11 +330,13 @@ curl -X POST http://localhost:5000/api/plusminus \
``` ```
**Antwort:** **Antwort:**
```json ```json
{ "result": "2024-06-15" } { "result": "2024-06-15" }
``` ```
**Hinweis:** **Hinweis:**
- `"einheit"`: `"tage"`, `"wochen"` oder `"monate"` - `"einheit"`: `"tage"`, `"wochen"` oder `"monate"`
- `"richtung"`: `"add"` (plus) oder `"sub"` (minus) - `"richtung"`: `"add"` (plus) oder `"sub"` (minus)
- `"werktage"`: `true` für Werktage, sonst `false` (nur bei `"tage"` unterstützt) - `"werktage"`: `true` für Werktage, sonst `false` (nur bei `"tage"` unterstützt)
@@ -291,11 +346,13 @@ curl -X POST http://localhost:5000/api/plusminus \
**GET** `/api/stats` **GET** `/api/stats`
**Mit curl:** **Mit curl:**
```bash ```bash
curl http://localhost:5000/api/stats curl http://localhost:5000/api/stats
``` ```
**Antwort:** **Antwort:**
```json ```json
{ {
"pageviews": 42, "pageviews": 42,
@@ -309,11 +366,13 @@ curl http://localhost:5000/api/stats
**GET** `/api/monitor` **GET** `/api/monitor`
**Mit curl:** **Mit curl:**
```bash ```bash
curl http://localhost:5000/api/monitor curl http://localhost:5000/api/monitor
``` ```
**Antwort:** **Antwort:**
```json ```json
{ {
"status": "ok", "status": "ok",
@@ -339,6 +398,7 @@ Elpatrons Datumsrechner ist als PWA installierbar (z.B. auf Android/iOS-Homescre
- Manifest und Service Worker sind integriert - Manifest und Service Worker sind integriert
- App-Icon und Theme-Color für Homescreen - App-Icon und Theme-Color für Homescreen
- Installation über Browser-Menü ("Zum Startbildschirm hinzufügen") - Installation über Browser-Menü ("Zum Startbildschirm hinzufügen")
- Taschenrechner funktioniert vollständig clientseitig (offline verfügbar)
## Monitoring & Healthcheck ## Monitoring & Healthcheck
@@ -392,7 +452,7 @@ Finde mal eine Datumsrechner- Webapp, die nicht völlig Werbe- und Tracking vers
### Vibe Coding ### Vibe Coding
Dieses Projekt wurde zu nahezu 100% mit Unterstützung künsticher Intelligenz (*[Vibe Coding](https://de.wikipedia.org/wiki/Vibe_Coding)*) erstellt. Das Grundgerüst war nach ca. 45 Minuten fertig gestellt, insgesamt hat die Entwicklung des Projekts ca. 4 Stunden Zeit beansprucht. Dieses Projekt wurde zu nahezu 100% mit Unterstützung künsticher Intelligenz (*[Vibe Coding](https://de.wikipedia.org/wiki/Vibe_Coding)*) erstellt. Das Grundgerüst war nach ca. 45 Minuten fertig gestellt, insgesamt hat die Entwicklung des Projekts ca. 12 Stunden Zeit beansprucht.
### Statistik-Erfassung, Logging ### Statistik-Erfassung, Logging
@@ -408,6 +468,8 @@ Es werden keine IP-Adressen oder sonstigen persönlichen Daten gespeichert, ledi
- *Fokus-Indikatoren:* Deutliche visuelle Hervorhebung des Fokus für alle Bedienelemente. - *Fokus-Indikatoren:* Deutliche visuelle Hervorhebung des Fokus für alle Bedienelemente.
- *Farbkontraste:* Hohe Kontraste für Texte, Buttons und Ergebnisboxen, geprüft nach WCAG-Richtlinien. - *Farbkontraste:* Hohe Kontraste für Texte, Buttons und Ergebnisboxen, geprüft nach WCAG-Richtlinien.
- *Status- und Fehlermeldungen:* Ergebnisse und Fehler werden mit `aria-live` für Screenreader zugänglich gemacht. - *Status- und Fehlermeldungen:* Ergebnisse und Fehler werden mit `aria-live` für Screenreader zugänglich gemacht.
- *Sprachausgabe:* Alle Ergebnisse können über 🔊-Buttons vorgelesen werden (Web Speech API, deutsche Sprache).
- *Taschenrechner:* Vollständig barrierefrei mit Tastatur-Bedienung, Sprachausgabe und History-Funktion.
- *Mobile Optimierung:* Zusätzliche Meta-Tags für bessere Bedienbarkeit auf mobilen Geräten und Unterstützung von Screenreadern. - *Mobile Optimierung:* Zusätzliche Meta-Tags für bessere Bedienbarkeit auf mobilen Geräten und Unterstützung von Screenreadern.
- *SEO:* Das Thema Barrierefreiheit ist in den Meta-Tags für Suchmaschinen sichtbar. - *SEO:* Das Thema Barrierefreiheit ist in den Meta-Tags für Suchmaschinen sichtbar.
@@ -415,21 +477,23 @@ Damit ist die App für Menschen mit unterschiedlichen Einschränkungen (z.B. Seh
### Code Statistik ### Code Statistik
cloc|github.com/AlDanial/cloc v 2.06 T=0.06 s (263.5 files/s, 39193.4 lines/s) cloc|github.com/AlDanial/cloc v 2.06 T=0.22 s (114.3 files/s, 32032.3 lines/s)
--- | --- --- | ---
Language|files|blank|comment|code Language|files|blank|comment|code
:-------|-------:|-------:|-------:|-------: :-------|-------:|-------:|-------:|-------:
HTML|4|21|6|927 HTML|8|159|8|2800
Python|2|34|26|482 Python|2|66|74|739
Markdown|2|116|0|328 JavaScript|2|95|88|580
JSON|2|0|0|237 PO File|2|260|266|544
JavaScript|1|0|0|20 Markdown|3|177|0|497
JSON|3|0|0|243
CSS|1|186|3|188
SVG|2|0|0|14 SVG|2|0|0|14
Dockerfile|1|5|6|8 Dockerfile|1|5|6|8
DOS Batch|1|0|0|1 DOS Batch|1|0|0|1
--------|--------|--------|--------|-------- --------|--------|--------|--------|--------
SUM:|15|176|38|2017 SUM:|25|948|445|5614
## Lizenz ## Lizenz
@@ -437,3 +501,5 @@ Dieses Projekt steht unter der [MIT-Lizenz](LICENSE).
--- ---
(c) 2025 [Markus Busche](https://digitalcourage.social/@elpatron) (c) 2025 [Markus Busche](https://digitalcourage.social/@elpatron)
**Version 1.4.12** - Integrierter Taschenrechner mit History und Sprachausgabe hinzugefügt

230
app.py
View File

@@ -1,4 +1,5 @@
from flask import Flask, render_template, request, redirect, url_for, session, abort, jsonify from flask import Flask, render_template, request, redirect, url_for, session, abort, jsonify, g, make_response
from flask_babel import Babel, gettext, ngettext, get_locale
from datetime import datetime, timedelta from datetime import datetime, timedelta
import numpy as np import numpy as np
from dateutil.relativedelta import relativedelta from dateutil.relativedelta import relativedelta
@@ -11,12 +12,56 @@ app_start_time = time.time()
app = Flask(__name__) app = Flask(__name__)
app.secret_key = os.environ.get('SECRET_KEY', 'dev-key') app.secret_key = os.environ.get('SECRET_KEY', 'dev-key')
# Babel Konfiguration
app.config['BABEL_DEFAULT_LOCALE'] = 'de'
app.config['BABEL_SUPPORTED_LOCALES'] = ['de', 'en']
app.config['BABEL_TRANSLATION_DIRECTORIES'] = 'translations'
babel = Babel()
# Version der App # Version der App
APP_VERSION = "1.3.10" APP_VERSION = "1.4.13"
def add_cache_headers(response):
"""Fügt Cache-Control-Header hinzu, die den Back-Forward-Cache ermöglichen"""
# Cache-Control für statische Inhalte und API-Endpunkte
if request.path.startswith('/static/') or request.path.startswith('/api/'):
response.headers['Cache-Control'] = 'public, max-age=3600, s-maxage=3600'
else:
# Für HTML-Seiten: kurze Cache-Zeit, aber Back-Forward-Cache erlauben
response.headers['Cache-Control'] = 'public, max-age=60, s-maxage=300'
# Wichtig: Keine Vary-Header für User-Agent oder andere dynamische Werte
# Dies verhindert den Back-Forward-Cache
if 'Vary' in response.headers:
del response.headers['Vary']
return response
# HTML-Template wird jetzt aus templates/index.html geladen # HTML-Template wird jetzt aus templates/index.html geladen
WOCHENTAGE = ["Montag", "Dienstag", "Mittwoch", "Donnerstag", "Freitag", "Samstag", "Sonntag"] WOCHENTAGE = ["Montag", "Dienstag", "Mittwoch", "Donnerstag", "Freitag", "Samstag", "Sonntag"]
WOCHENTAGE_EN = ["Monday", "Tuesday", "Wednesday", "Thursday", "Friday", "Saturday", "Sunday"]
def get_locale():
# Prüfe URL-Parameter für Sprachauswahl (datenschutzfreundlich)
if request.args.get('lang') in ['de', 'en']:
return request.args.get('lang')
# Prüfe Session für Sprachauswahl (Fallback für ältere Browser)
if 'language' in session:
return session['language']
# Fallback auf Browser-Sprache
return request.accept_languages.best_match(['de', 'en'], default='de')
# Registriere die Locale-Funktion mit Babel
babel.init_app(app, locale_selector=get_locale)
def get_wochentage():
"""Gibt die Wochentage in der aktuellen Sprache zurück"""
locale = get_locale()
if locale == 'en':
return WOCHENTAGE_EN
return WOCHENTAGE
def get_feiertage(year, bundesland): def get_feiertage(year, bundesland):
"""Holt die Feiertage für ein Jahr und Bundesland von feiertage-api.de.""" """Holt die Feiertage für ein Jahr und Bundesland von feiertage-api.de."""
@@ -30,6 +75,24 @@ def get_feiertage(year, bundesland):
print(f"Fehler beim Abrufen der Feiertage: {e}") print(f"Fehler beim Abrufen der Feiertage: {e}")
return [] return []
@app.route('/set_language/<language>')
def set_language(language):
"""Setzt die Sprache über URL-Parameter (datenschutzfreundlich)"""
if language in ['de', 'en']:
# URL-Parameter verwenden statt Session
referrer = request.referrer or url_for('index')
if '?' in referrer:
# URL hat bereits Parameter
if 'lang=' in referrer:
# Ersetze bestehenden lang-Parameter
import re
referrer = re.sub(r'[?&]lang=[^&]*', '', referrer)
return redirect(f"{referrer}&lang={language}")
else:
# URL hat keine Parameter
return redirect(f"{referrer}?lang={language}")
return redirect(request.referrer or url_for('index'))
@app.route('/', methods=['GET', 'POST']) @app.route('/', methods=['GET', 'POST'])
def index(): def index():
# Rudimentäres Logging für Page Impressions # Rudimentäres Logging für Page Impressions
@@ -94,24 +157,29 @@ def index():
else: else:
tage = abs((d2 - d1).days) tage = abs((d2 - d1).days)
except Exception: except Exception:
tage = 'Ungültige Eingabe' tage = gettext('Ungültige Eingabe')
elif action == 'wochentag': elif action == 'wochentag':
active_idx = 1 active_idx = 1
datum = request.form.get('datum3') datum = request.form.get('datum3')
try: try:
d = datetime.strptime(datum, '%Y-%m-%d') d = datetime.strptime(datum, '%Y-%m-%d')
wochentag = WOCHENTAGE[d.weekday()] wochentage = get_wochentage()
wochentag = wochentage[d.weekday()]
except Exception: except Exception:
wochentag = 'Ungültige Eingabe' wochentag = gettext('Ungültige Eingabe')
elif action == 'kw_berechnen': elif action == 'kw_berechnen':
active_idx = 2 active_idx = 2
datum = request.form.get('datum6') datum = request.form.get('datum6')
try: try:
d = datetime.strptime(datum, '%Y-%m-%d') d = datetime.strptime(datum, '%Y-%m-%d')
kw = d.isocalendar().week kw = d.isocalendar().week
kw_berechnen = f"KW {kw} ({d.year})" locale = get_locale()
if locale == 'en':
kw_berechnen = f"Week {kw} ({d.year})"
else:
kw_berechnen = f"KW {kw} ({d.year})"
except Exception: except Exception:
kw_berechnen = 'Ungültige Eingabe' kw_berechnen = gettext('Ungültige Eingabe')
elif action == 'kw_datum': elif action == 'kw_datum':
active_idx = 3 active_idx = 3
jahr = request.form.get('jahr7') jahr = request.form.get('jahr7')
@@ -122,9 +190,13 @@ def index():
# Montag der KW # Montag der KW
start = datetime.fromisocalendar(jahr, kw, 1) start = datetime.fromisocalendar(jahr, kw, 1)
end = datetime.fromisocalendar(jahr, kw, 7) end = datetime.fromisocalendar(jahr, kw, 7)
kw_datum = f"{start.strftime('%d.%m.%Y')} bis {end.strftime('%d.%m.%Y')}" locale = get_locale()
if locale == 'en':
kw_datum = f"{start.strftime('%m/%d/%Y')} to {end.strftime('%m/%d/%Y')}"
else:
kw_datum = f"{start.strftime('%d.%m.%Y')} bis {end.strftime('%d.%m.%Y')}"
except Exception: except Exception:
kw_datum = 'Ungültige Eingabe' kw_datum = gettext('Ungültige Eingabe')
elif action == 'plusminus': elif action == 'plusminus':
active_idx = 4 active_idx = 4
datum = request.form.get('datum_pm') datum = request.form.get('datum_pm')
@@ -137,57 +209,108 @@ def index():
anzahl_int = int(anzahl) anzahl_int = int(anzahl)
if richtung == 'sub': if richtung == 'sub':
anzahl_int = -anzahl_int anzahl_int = -anzahl_int
locale = get_locale()
if einheit == 'tage': if einheit == 'tage':
if is_werktage: if is_werktage:
# Werktage: numpy busday_offset # Werktage: numpy busday_offset
result = np.busday_offset(d.date(), anzahl_int, roll='forward') result = np.busday_offset(d.date(), anzahl_int, roll='forward')
result_dt = datetime.strptime(str(result), '%Y-%m-%d') result_dt = datetime.strptime(str(result), '%Y-%m-%d')
plusminus_result = f"Datum {d.strftime('%d.%m.%Y')} {'plus' if anzahl_int>=0 else 'minus'} {abs(anzahl_int)} Werktage: {result_dt.strftime('%d.%m.%Y')}" if locale == 'en':
plusminus_result = f"Date {d.strftime('%m/%d/%Y')} {'plus' if anzahl_int>=0 else 'minus'} {abs(anzahl_int)} workdays: {result_dt.strftime('%m/%d/%Y')}"
else:
plusminus_result = f"Datum {d.strftime('%d.%m.%Y')} {'plus' if anzahl_int>=0 else 'minus'} {abs(anzahl_int)} Werktage: {result_dt.strftime('%d.%m.%Y')}"
else: else:
result = d + timedelta(days=anzahl_int) result = d + timedelta(days=anzahl_int)
plusminus_result = f"Datum {d.strftime('%d.%m.%Y')} {'plus' if anzahl_int>=0 else 'minus'} {abs(anzahl_int)} Tage: {result.strftime('%d.%m.%Y')}" if locale == 'en':
plusminus_result = f"Date {d.strftime('%m/%d/%Y')} {'plus' if anzahl_int>=0 else 'minus'} {abs(anzahl_int)} days: {result.strftime('%m/%d/%Y')}"
else:
plusminus_result = f"Datum {d.strftime('%d.%m.%Y')} {'plus' if anzahl_int>=0 else 'minus'} {abs(anzahl_int)} Tage: {result.strftime('%d.%m.%Y')}"
elif einheit == 'wochen': elif einheit == 'wochen':
if is_werktage: if is_werktage:
plusminus_result = 'Nicht unterstützt: Werktage + Wochen.' plusminus_result = gettext('Nicht unterstützt: Werktage + Wochen.')
else: else:
result = d + timedelta(weeks=anzahl_int) result = d + timedelta(weeks=anzahl_int)
plusminus_result = f"Datum {d.strftime('%d.%m.%Y')} {'plus' if anzahl_int>=0 else 'minus'} {abs(anzahl_int)} Wochen: {result.strftime('%d.%m.%Y')}" if locale == 'en':
plusminus_result = f"Date {d.strftime('%m/%d/%Y')} {'plus' if anzahl_int>=0 else 'minus'} {abs(anzahl_int)} weeks: {result.strftime('%m/%d/%Y')}"
else:
plusminus_result = f"Datum {d.strftime('%d.%m.%Y')} {'plus' if anzahl_int>=0 else 'minus'} {abs(anzahl_int)} Wochen: {result.strftime('%d.%m.%Y')}"
elif einheit == 'monate': elif einheit == 'monate':
if is_werktage: if is_werktage:
plusminus_result = 'Nicht unterstützt: Werktage + Monate.' plusminus_result = gettext('Nicht unterstützt: Werktage + Monate.')
else: else:
result = d + relativedelta(months=anzahl_int) result = d + relativedelta(months=anzahl_int)
plusminus_result = f"Datum {d.strftime('%d.%m.%Y')} {'plus' if anzahl_int>=0 else 'minus'} {abs(anzahl_int)} Monate: {result.strftime('%d.%m.%Y')}" if locale == 'en':
plusminus_result = f"Date {d.strftime('%m/%d/%Y')} {'plus' if anzahl_int>=0 else 'minus'} {abs(anzahl_int)} months: {result.strftime('%m/%d/%Y')}"
else:
plusminus_result = f"Datum {d.strftime('%d.%m.%Y')} {'plus' if anzahl_int>=0 else 'minus'} {abs(anzahl_int)} Monate: {result.strftime('%d.%m.%Y')}"
except Exception: except Exception:
plusminus_result = 'Ungültige Eingabe' plusminus_result = gettext('Ungültige Eingabe')
return render_template('index.html', tage=tage, werktage=werktage, wochentag=wochentag, plusminus_result=plusminus_result, kw_berechnen=kw_berechnen, kw_datum=kw_datum, active_idx=active_idx response = make_response(render_template('index.html', tage=tage, werktage=werktage, wochentag=wochentag, plusminus_result=plusminus_result, kw_berechnen=kw_berechnen, kw_datum=kw_datum, active_idx=active_idx
, feiertage_anzahl=feiertage_anzahl, wochenendtage_anzahl=wochenendtage_anzahl, app_version=APP_VERSION , feiertage_anzahl=feiertage_anzahl, wochenendtage_anzahl=wochenendtage_anzahl, app_version=APP_VERSION, get_locale=get_locale
) ))
return add_cache_headers(response)
def parse_log_stats(log_path): def parse_log_stats(log_path):
pageviews = 0 pageviews = 0
func_counts = {} func_counts = {}
func_counts_hourly = {}
impressions_per_day = {} impressions_per_day = {}
impressions_per_hour = {}
api_counts = {} api_counts = {}
api_counts_hourly = {}
if os.path.exists(log_path): if os.path.exists(log_path):
with open(log_path, encoding='utf-8') as f: with open(log_path, encoding='utf-8') as f:
for line in f: for line in f:
if 'PAGEVIEW' in line: if 'PAGEVIEW' in line:
pageviews += 1 pageviews += 1
try: try:
date = line[:10] # Parse timestamp (format: YYYY-MM-DDTHH:MM:SS)
timestamp = line[:19] # First 19 chars for YYYY-MM-DDTHH:MM:SS
date = timestamp[:10] # YYYY-MM-DD
hour = timestamp[11:13] # HH
if len(date) == 10 and date[4] == '-' and date[7] == '-': if len(date) == 10 and date[4] == '-' and date[7] == '-':
impressions_per_day[date] = impressions_per_day.get(date, 0) + 1 impressions_per_day[date] = impressions_per_day.get(date, 0) + 1
if len(hour) == 2 and hour.isdigit():
hour_key = f"{date} {hour}:00"
impressions_per_hour[hour_key] = impressions_per_hour.get(hour_key, 0) + 1
except Exception: except Exception:
pass pass
elif 'FUNC:' in line: elif 'FUNC:' in line:
func = line.split('FUNC:')[1].strip() func = line.split('FUNC:')[1].strip()
func_counts[func] = func_counts.get(func, 0) + 1 func_counts[func] = func_counts.get(func, 0) + 1
# Stündliche Funktionsaufrufe
try:
timestamp = line[:19]
date = timestamp[:10]
hour = timestamp[11:13]
if len(hour) == 2 and hour.isdigit():
hour_key = f"{date} {hour}:00"
if hour_key not in func_counts_hourly:
func_counts_hourly[hour_key] = {}
func_counts_hourly[hour_key][func] = func_counts_hourly[hour_key].get(func, 0) + 1
except Exception:
pass
elif 'FUNC_API:' in line: elif 'FUNC_API:' in line:
api = line.split('FUNC_API:')[1].strip() api = line.split('FUNC_API:')[1].strip()
api_counts[api] = api_counts.get(api, 0) + 1 api_counts[api] = api_counts.get(api, 0) + 1
return pageviews, func_counts, impressions_per_day, api_counts
# Stündliche API-Aufrufe
try:
timestamp = line[:19]
date = timestamp[:10]
hour = timestamp[11:13]
if len(hour) == 2 and hour.isdigit():
hour_key = f"{date} {hour}:00"
if hour_key not in api_counts_hourly:
api_counts_hourly[hour_key] = {}
api_counts_hourly[hour_key][api] = api_counts_hourly[hour_key].get(api, 0) + 1
except Exception:
pass
return pageviews, func_counts, func_counts_hourly, impressions_per_day, impressions_per_hour, api_counts, api_counts_hourly
@app.route('/stats', methods=['GET', 'POST']) @app.route('/stats', methods=['GET', 'POST'])
def stats(): def stats():
@@ -198,11 +321,14 @@ def stats():
session['stats_auth'] = True session['stats_auth'] = True
return redirect(url_for('stats')) return redirect(url_for('stats'))
else: else:
return render_template('stats_login.html', error='Falsches Passwort!') response = make_response(render_template('stats_login.html', error='Falsches Passwort!'))
return render_template('stats_login.html', error=None) return add_cache_headers(response)
response = make_response(render_template('stats_login.html', error=None))
return add_cache_headers(response)
log_path = os.path.join('log', 'pageviews.log') log_path = os.path.join('log', 'pageviews.log')
pageviews, func_counts, impressions_per_day, api_counts = parse_log_stats(log_path) pageviews, func_counts, func_counts_hourly, impressions_per_day, impressions_per_hour, api_counts, api_counts_hourly = parse_log_stats(log_path)
return render_template('stats_dashboard.html', pageviews=pageviews, func_counts=func_counts, impressions_per_day=impressions_per_day, api_counts=api_counts) response = make_response(render_template('stats_dashboard.html', pageviews=pageviews, func_counts=func_counts, func_counts_hourly=func_counts_hourly, impressions_per_day=impressions_per_day, impressions_per_hour=impressions_per_hour, api_counts=api_counts, api_counts_hourly=api_counts_hourly))
return add_cache_headers(response)
# --- REST API --- # --- REST API ---
def log_api_usage(api_name): def log_api_usage(api_name):
@@ -236,7 +362,8 @@ def api_tage_werktage():
tage = int(np.busday_count(d1.date(), (d2 + timedelta(days=1)).date(), holidays=holidays)) tage = int(np.busday_count(d1.date(), (d2 + timedelta(days=1)).date(), holidays=holidays))
else: else:
tage = abs((d2 - d1).days) tage = abs((d2 - d1).days)
return jsonify({'result': tage}) response = jsonify({'result': tage})
return add_cache_headers(response)
except Exception as e: except Exception as e:
return jsonify({'error': 'Ungültige Eingabe', 'details': str(e)}), 400 return jsonify({'error': 'Ungültige Eingabe', 'details': str(e)}), 400
@@ -247,8 +374,10 @@ def api_wochentag():
datum = data.get('datum') datum = data.get('datum')
try: try:
d = datetime.strptime(datum, '%Y-%m-%d') d = datetime.strptime(datum, '%Y-%m-%d')
wochentag = WOCHENTAGE[d.weekday()] wochentage = get_wochentage()
return jsonify({'result': wochentag}) wochentag = wochentage[d.weekday()]
response = jsonify({'result': wochentag})
return add_cache_headers(response)
except Exception as e: except Exception as e:
return jsonify({'error': 'Ungültige Eingabe', 'details': str(e)}), 400 return jsonify({'error': 'Ungültige Eingabe', 'details': str(e)}), 400
@@ -260,7 +389,13 @@ def api_kw_berechnen():
try: try:
d = datetime.strptime(datum, '%Y-%m-%d') d = datetime.strptime(datum, '%Y-%m-%d')
kw = d.isocalendar().week kw = d.isocalendar().week
return jsonify({'result': f"KW {kw} ({d.year})", 'kw': kw, 'jahr': d.year}) locale = get_locale()
if locale == 'en':
kw_berechnen = f"Week {kw} ({d.year})"
else:
kw_berechnen = f"KW {kw} ({d.year})"
response = jsonify({'result': kw_berechnen, 'kw': kw, 'jahr': d.year})
return add_cache_headers(response)
except Exception as e: except Exception as e:
return jsonify({'error': 'Ungültige Eingabe', 'details': str(e)}), 400 return jsonify({'error': 'Ungültige Eingabe', 'details': str(e)}), 400
@@ -275,7 +410,13 @@ def api_kw_datum():
kw = int(kw) kw = int(kw)
start = datetime.fromisocalendar(jahr, kw, 1) start = datetime.fromisocalendar(jahr, kw, 1)
end = datetime.fromisocalendar(jahr, kw, 7) end = datetime.fromisocalendar(jahr, kw, 7)
return jsonify({'result': f"{start.strftime('%d.%m.%Y')} bis {end.strftime('%d.%m.%Y')}", 'start': start.strftime('%Y-%m-%d'), 'end': end.strftime('%Y-%m-%d')}) locale = get_locale()
if locale == 'en':
kw_datum = f"{start.strftime('%m/%d/%Y')} to {end.strftime('%m/%d/%Y')}"
else:
kw_datum = f"{start.strftime('%d.%m.%Y')} bis {end.strftime('%d.%m.%Y')}"
response = jsonify({'result': kw_datum, 'start': start.strftime('%Y-%m-%d'), 'end': end.strftime('%Y-%m-%d')})
return add_cache_headers(response)
except Exception as e: except Exception as e:
return jsonify({'error': 'Ungültige Eingabe', 'details': str(e)}), 400 return jsonify({'error': 'Ungültige Eingabe', 'details': str(e)}), 400
@@ -293,26 +434,31 @@ def api_plusminus():
anzahl_int = int(anzahl) anzahl_int = int(anzahl)
if richtung == 'sub': if richtung == 'sub':
anzahl_int = -anzahl_int anzahl_int = -anzahl_int
locale = get_locale()
if einheit == 'tage': if einheit == 'tage':
if is_werktage: if is_werktage:
result = np.busday_offset(d.date(), anzahl_int, roll='forward') result = np.busday_offset(d.date(), anzahl_int, roll='forward')
result_dt = datetime.strptime(str(result), '%Y-%m-%d') result_dt = datetime.strptime(str(result), '%Y-%m-%d')
return jsonify({'result': result_dt.strftime('%Y-%m-%d')}) response = jsonify({'result': result_dt.strftime('%Y-%m-%d')})
return add_cache_headers(response)
else: else:
result = d + timedelta(days=anzahl_int) result = d + timedelta(days=anzahl_int)
return jsonify({'result': result.strftime('%Y-%m-%d')}) response = jsonify({'result': result.strftime('%Y-%m-%d')})
return add_cache_headers(response)
elif einheit == 'wochen': elif einheit == 'wochen':
if is_werktage: if is_werktage:
return jsonify({'error': 'Nicht unterstützt: Werktage + Wochen.'}), 400 return jsonify({'error': 'Nicht unterstützt: Werktage + Wochen.'}), 400
else: else:
result = d + timedelta(weeks=anzahl_int) result = d + timedelta(weeks=anzahl_int)
return jsonify({'result': result.strftime('%Y-%m-%d')}) response = jsonify({'result': result.strftime('%Y-%m-%d')})
return add_cache_headers(response)
elif einheit == 'monate': elif einheit == 'monate':
if is_werktage: if is_werktage:
return jsonify({'error': 'Nicht unterstützt: Werktage + Monate.'}), 400 return jsonify({'error': 'Nicht unterstützt: Werktage + Monate.'}), 400
else: else:
result = d + relativedelta(months=anzahl_int) result = d + relativedelta(months=anzahl_int)
return jsonify({'result': result.strftime('%Y-%m-%d')}) response = jsonify({'result': result.strftime('%Y-%m-%d')})
return add_cache_headers(response)
else: else:
return jsonify({'error': 'Ungültige Einheit'}), 400 return jsonify({'error': 'Ungültige Einheit'}), 400
except Exception as e: except Exception as e:
@@ -321,8 +467,14 @@ def api_plusminus():
@app.route('/api/stats', methods=['GET']) @app.route('/api/stats', methods=['GET'])
def api_stats(): def api_stats():
log_path = os.path.join('log', 'pageviews.log') log_path = os.path.join('log', 'pageviews.log')
pageviews, func_counts, impressions_per_day, api_counts = parse_log_stats(log_path) pageviews, func_counts, func_counts_hourly, impressions_per_day, impressions_per_hour, api_counts, api_counts_hourly = parse_log_stats(log_path)
return render_template('stats_dashboard.html', pageviews=pageviews, func_counts=func_counts, impressions_per_day=impressions_per_day, api_counts=api_counts) response = jsonify({
"pageviews": pageviews,
"func_counts": func_counts,
"impressions_per_day": impressions_per_day,
"api_counts": api_counts
})
return add_cache_headers(response)
@app.route('/api/monitor', methods=['GET']) @app.route('/api/monitor', methods=['GET'])
def api_monitor(): def api_monitor():
@@ -334,17 +486,19 @@ def api_monitor():
if 'PAGEVIEW' in line: if 'PAGEVIEW' in line:
pageviews += 1 pageviews += 1
uptime = int(time.time() - app_start_time) uptime = int(time.time() - app_start_time)
return jsonify({ response = jsonify({
"status": "ok", "status": "ok",
"message": "App running", "message": "App running",
"time": datetime.now().isoformat(), "time": datetime.now().isoformat(),
"uptime_seconds": uptime, "uptime_seconds": uptime,
"pageviews_last_7_days": pageviews "pageviews_last_7_days": pageviews
}) })
return add_cache_headers(response)
@app.route('/api-docs') @app.route('/api-docs')
def api_docs(): def api_docs():
return render_template('swagger.html') response = make_response(render_template('swagger.html'))
return add_cache_headers(response)
if __name__ == '__main__': if __name__ == '__main__':

Binary file not shown.

Before

Width:  |  Height:  |  Size: 45 KiB

After

Width:  |  Height:  |  Size: 13 KiB

3
babel.cfg Normal file
View File

@@ -0,0 +1,3 @@
[python: **.py]
[jinja2: **/templates/**.html]
extensions=jinja2.ext.autoescape,jinja2.ext.with_

View File

@@ -1 +1 @@
@cloc . .\templates\ .\static\ --exclude-dir=.venv,.git,log,__pycache__,.pytest_cache --exclude-ext=txt,bak --md @cloc . .\templates\ .\static\ --exclude-dir=.venv,.git,log,__pycache__,.pytest_cache,lighthouse --exclude-ext=txt,bak --md

138
i18n_implementation.md Normal file
View File

@@ -0,0 +1,138 @@
# Internationalisierung (i18n) Implementierung
## Übersicht
Die Internationalisierung wurde erfolgreich in Elpatrons Datumsrechner implementiert. Das System unterstützt derzeit Deutsch (Standard) und Englisch.
## Implementierte Funktionen
### 1. Flask-Babel Integration
- **Flask-Babel 4.0.0** für Übersetzungsverwaltung
- **Babel 2.17.0** für Übersetzungskompilierung
- **Automatische Spracherkennung** basierend auf Browser-Einstellungen
- **Session-basierte Sprachauswahl** für Benutzer
### 2. Sprachauswahl
- **DE/EN Toggle** in der oberen linken Ecke
- **Visueller Indikator** für aktive Sprache
- **Persistente Sprachauswahl** über Session
### 3. Übersetzungsdateien
- **Deutsche Übersetzungen**: `translations/de/LC_MESSAGES/messages.po`
- **Englische Übersetzungen**: `translations/en/LC_MESSAGES/messages.po`
- **Kompilierte .mo Dateien** für optimale Performance
### 4. Übersetzte Inhalte
#### Meta-Tags
- Titel und Beschreibung
- Open Graph Tags
- Keywords
#### UI-Elemente
- Haupttitel und Untertitel
- Formular-Labels
- Button-Texte
- Hilfe-Texte
- Footer-Links
#### Dynamische Inhalte
- Wochentage (Deutsch/Englisch)
- Kalenderwochen-Format
- Datumsformate (DD.MM.YYYY vs MM/DD/YYYY)
- Fehlermeldungen
### 5. Technische Details
#### App-Konfiguration
```python
app.config['BABEL_DEFAULT_LOCALE'] = 'de'
app.config['BABEL_SUPPORTED_LOCALES'] = ['de', 'en']
app.config['BABEL_TRANSLATION_DIRECTORIES'] = 'translations'
```
#### Sprachauswahl-Funktion
```python
@app.route('/set_language/<language>')
def set_language(language):
if language in ['de', 'en']:
session['language'] = language
return redirect(request.referrer or url_for('index'))
```
#### Locale-Selector
```python
@babel.localeselector
def get_locale():
if 'language' in session:
return session['language']
return request.accept_languages.best_match(['de', 'en'], default='de')
```
## Verzeichnisstruktur
```
translations/
├── de/
│ └── LC_MESSAGES/
│ ├── messages.po
│ └── messages.mo
└── en/
└── LC_MESSAGES/
├── messages.po
└── messages.mo
```
## Verwendung
### Für Entwickler
1. **Neue Texte hinzufügen**:
```html
{{ _('Neuer Text') }}
```
2. **Übersetzungen extrahieren**:
```bash
pybabel extract -F babel.cfg -k _l -o messages.pot .
```
3. **Neue Übersetzungsdatei erstellen**:
```bash
pybabel init -i messages.pot -d translations -l en
```
4. **Übersetzungen kompilieren**:
```bash
pybabel compile -d translations
```
### Für Benutzer
- **Sprachauswahl**: Klick auf DE/EN in der oberen linken Ecke
- **Automatische Erkennung**: Browser-Sprache wird automatisch erkannt
- **Persistenz**: Sprachauswahl bleibt über Session erhalten
## Nächste Schritte
1. **Weitere Sprachen hinzufügen** (z.B. Französisch, Spanisch)
2. **Pluralisierung** für verschiedene Sprachen implementieren
3. **Dynamische Übersetzungen** für API-Responses
4. **RTL-Sprachen** unterstützen (Arabisch, Hebräisch)
## Dateien
- `app.py`: Hauptanwendung mit i18n-Integration
- `babel.cfg`: Babel-Konfiguration
- `requirements.txt`: Flask-Babel Abhängigkeit
- `templates/index.html`: Übersetzte UI-Templates
- `translations/`: Übersetzungsdateien
## Status
**Vollständig implementiert und funktionsfähig**
- Deutsche und englische Übersetzungen
- Sprachauswahl-UI
- Automatische Spracherkennung
- Session-basierte Persistenz
- Kompilierte Übersetzungen für Performance

View File

@@ -1,8 +0,0 @@
erstelle eine python web app, die verschiedene datumsberechnungen durchführt:
- Berechnung der Anzahl der Tage zwischen zwei Daten
- Berechnung der Anzahl der Werktage zwischen zwei Daten
- Anzeige des Wochentags eines Datums
Beachte:
- Virtual Environment unter ./.venv
- Wir entwickeln unter Windows

Binary file not shown.

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

Binary file not shown.

View File

@@ -1,5 +1,5 @@
Flask==3.0.3 Flask==3.0.0
numpy==1.26.4 numpy==1.26.4
python-dateutil==2.9.0.post0 python-dateutil==2.9.0.post0
pytest==8.2.2 requests==2.31.0
requests Flask-Babel==4.0.0

View File

@@ -1,15 +1,25 @@
const CACHE_NAME = 'datumsrechner-cache-v1'; const CACHE_NAME = 'datumsrechner-cache-v1';
const urlsToCache = [ const urlsToCache = [
'/', '/',
'/static/style.css',
'/static/favicon.ico', '/static/favicon.ico',
'/static/favicon.png', '/static/favicon.png',
'/static/favicon.svg',
'/static/logo.svg', '/static/logo.svg',
'/static/manifest.json',
]; ];
self.addEventListener('install', event => { self.addEventListener('install', event => {
event.waitUntil( event.waitUntil(
caches.open(CACHE_NAME) caches.open(CACHE_NAME)
.then(cache => cache.addAll(urlsToCache)) .then(cache => {
// Füge nur existierende Dateien zum Cache hinzu
return Promise.allSettled(
urlsToCache.map(url =>
cache.add(url).catch(err => {
console.log('Failed to cache:', url, err);
})
)
);
})
); );
}); });
self.addEventListener('fetch', event => { self.addEventListener('fetch', event => {

File diff suppressed because it is too large Load Diff

View File

@@ -19,6 +19,32 @@
.stats-label { color: #64748b; } .stats-label { color: #64748b; }
.stats-value { font-size: 1.5em; font-weight: bold; } .stats-value { font-size: 1.5em; font-weight: bold; }
.chart-container { margin: 2em 0; } .chart-container { margin: 2em 0; }
.toggle-container {
display: flex;
justify-content: center;
margin-bottom: 1.5em;
gap: 0.5em;
}
.toggle-btn {
padding: 0.5em 1em;
border: 1px solid #d1d5db;
background: #f9fafb;
color: #6b7280;
border-radius: 6px;
cursor: pointer;
transition: all 0.2s;
}
.toggle-btn.active {
background: #2563eb;
color: white;
border-color: #2563eb;
}
.toggle-btn:hover {
background: #e5e7eb;
}
.toggle-btn.active:hover {
background: #1d4ed8;
}
</style> </style>
</head> </head>
<body> <body>
@@ -28,6 +54,12 @@
<div class="stats-label">Gesamt-Pageviews (7 Tage):</div> <div class="stats-label">Gesamt-Pageviews (7 Tage):</div>
<div class="stats-value">{{ pageviews }}</div> <div class="stats-value">{{ pageviews }}</div>
</div> </div>
<div class="toggle-container">
<button class="toggle-btn active" data-period="week">Wochenverlauf</button>
<button class="toggle-btn" data-period="day">24-Stunden-Verlauf</button>
</div>
<div class="chart-container"> <div class="chart-container">
<canvas id="imprChart" width="400" height="180"></canvas> <canvas id="imprChart" width="400" height="180"></canvas>
</div> </div>
@@ -43,75 +75,196 @@
</div> </div>
<script type="text/javascript"> <script type="text/javascript">
document.addEventListener('DOMContentLoaded', function() { document.addEventListener('DOMContentLoaded', function() {
// Impressions pro Tag // Daten für verschiedene Zeiträume
// eslint-disable-next-line const weekData = {{ impressions_per_day|tojson }};
const imprData = {{ impressions_per_day|tojson }}; const dayData = {{ impressions_per_hour|tojson }};
const imprLabels = Object.keys(imprData); const weekFuncData = {{ func_counts|tojson }};
const imprCounts = Object.values(imprData); const dayFuncData = {{ func_counts_hourly|tojson }};
new Chart(document.getElementById('imprChart').getContext('2d'), { const weekApiData = {{ api_counts|tojson }};
type: 'line', const dayApiData = {{ api_counts_hourly|tojson }};
data: {
labels: imprLabels, let currentPeriod = 'week';
datasets: [{ let currentImprChart = null;
label: 'Impressions/Tag', let currentFuncChart = null;
data: imprCounts, let currentApiChart = null;
borderColor: '#059669',
backgroundColor: 'rgba(5,150,105,0.1)', // Toggle-Buttons Event Listener
tension: 0.2, document.querySelectorAll('.toggle-btn').forEach(btn => {
fill: true btn.addEventListener('click', function() {
}] // Aktiven Button aktualisieren
}, document.querySelectorAll('.toggle-btn').forEach(b => b.classList.remove('active'));
options: { this.classList.add('active');
plugins: { legend: { display: true } },
scales: { // Zeitraum wechseln
y: { beginAtZero: true, ticks: { stepSize: 1 } } currentPeriod = this.dataset.period;
} updateAllCharts();
} });
}); });
// Funktionsaufrufe
// eslint-disable-next-line function updateImpressionsChart() {
const funcCounts = {{ func_counts|tojson }}; const ctx = document.getElementById('imprChart').getContext('2d');
const labels = Object.keys(funcCounts);
const data = Object.values(funcCounts); // Bestehenden Chart zerstören
new Chart(document.getElementById('funcChart').getContext('2d'), { if (currentImprChart) {
type: 'bar', currentImprChart.destroy();
data: {
labels: labels,
datasets: [{
label: 'Funktionsaufrufe',
data: data,
backgroundColor: '#2563eb',
}]
},
options: {
plugins: { legend: { display: false } },
scales: {
y: { beginAtZero: true, ticks: { stepSize: 1 } }
}
} }
});
// API-Nutzung let data, labels, counts;
// eslint-disable-next-line
const apiCounts = {{ api_counts|tojson }}; if (currentPeriod === 'week') {
if (Object.keys(apiCounts).length > 0 && document.getElementById('apiChart')) { data = weekData;
new Chart(document.getElementById('apiChart').getContext('2d'), { labels = Object.keys(data);
type: 'bar', counts = Object.values(data);
} else {
data = dayData;
labels = Object.keys(data);
counts = Object.values(data);
}
currentImprChart = new Chart(ctx, {
type: 'line',
data: { data: {
labels: Object.keys(apiCounts), labels: labels,
datasets: [{ datasets: [{
label: 'API-Aufrufe nach Endpunkt', label: currentPeriod === 'week' ? 'Impressions/Tag' : 'Impressions/Stunde',
data: Object.values(apiCounts), data: counts,
backgroundColor: '#f59e42', borderColor: '#059669',
backgroundColor: 'rgba(5,150,105,0.1)',
tension: 0.2,
fill: true
}] }]
}, },
options: { options: {
plugins: { legend: { display: false } }, plugins: {
legend: { display: true },
title: {
display: true,
text: currentPeriod === 'week' ? 'Wochenverlauf' : '24-Stunden-Verlauf'
}
},
scales: { scales: {
y: { beginAtZero: true, ticks: { stepSize: 1 } } y: { beginAtZero: true, ticks: { stepSize: 1 } }
} }
} }
}); });
} }
function updateFunctionChart() {
const ctx = document.getElementById('funcChart').getContext('2d');
// Bestehenden Chart zerstören
if (currentFuncChart) {
currentFuncChart.destroy();
}
let data, labels, counts;
if (currentPeriod === 'week') {
data = weekFuncData;
labels = Object.keys(data);
counts = Object.values(data);
} else {
// Für stündliche Daten: Summe aller Stunden für jede Funktion
const aggregatedData = {};
Object.values(dayFuncData).forEach(hourData => {
Object.keys(hourData).forEach(func => {
aggregatedData[func] = (aggregatedData[func] || 0) + hourData[func];
});
});
data = aggregatedData;
labels = Object.keys(data);
counts = Object.values(data);
}
currentFuncChart = new Chart(ctx, {
type: 'bar',
data: {
labels: labels,
datasets: [{
label: 'Funktionsaufrufe',
data: counts,
backgroundColor: '#2563eb',
}]
},
options: {
plugins: {
legend: { display: false },
title: {
display: true,
text: currentPeriod === 'week' ? 'Funktionsaufrufe (Woche)' : 'Funktionsaufrufe (24h)'
}
},
scales: {
y: { beginAtZero: true, ticks: { stepSize: 1 } }
}
}
});
}
function updateApiChart() {
const apiChartElement = document.getElementById('apiChart');
if (!apiChartElement) return;
const ctx = apiChartElement.getContext('2d');
// Bestehenden Chart zerstören
if (currentApiChart) {
currentApiChart.destroy();
}
let data, labels, counts;
if (currentPeriod === 'week') {
data = weekApiData;
} else {
// Für stündliche Daten: Summe aller Stunden für jede API
const aggregatedData = {};
Object.values(dayApiData).forEach(hourData => {
Object.keys(hourData).forEach(api => {
aggregatedData[api] = (aggregatedData[api] || 0) + hourData[api];
});
});
data = aggregatedData;
}
if (Object.keys(data).length === 0) return;
labels = Object.keys(data);
counts = Object.values(data);
currentApiChart = new Chart(ctx, {
type: 'bar',
data: {
labels: labels,
datasets: [{
label: 'API-Aufrufe nach Endpunkt',
data: counts,
backgroundColor: '#f59e42',
}]
},
options: {
plugins: {
legend: { display: false },
title: {
display: true,
text: currentPeriod === 'week' ? 'API-Nutzung (Woche)' : 'API-Nutzung (24h)'
}
},
scales: {
y: { beginAtZero: true, ticks: { stepSize: 1 } }
}
}
});
}
function updateAllCharts() {
updateImpressionsChart();
updateFunctionChart();
updateApiChart();
}
// Initial Charts erstellen
updateAllCharts();
}); });
</script> </script>
</body> </body>

View File

@@ -1,6 +1,7 @@
import os import os
import pytest import pytest
from app import app as flask_app from app import app as flask_app
from unittest.mock import patch, MagicMock
@pytest.fixture @pytest.fixture
def client(): def client():
@@ -221,11 +222,153 @@ def test_api_stats(client):
resp = client.get('/api/stats') resp = client.get('/api/stats')
assert resp.status_code == 200 assert resp.status_code == 200
data = resp.get_json() data = resp.get_json()
assert 'pageviews' in data and 'func_counts' in data and 'impressions_per_day' in data assert "pageviews" in data
assert "func_counts" in data
assert "impressions_per_day" in data
assert "api_counts" in data
def test_api_monitor(client): def test_api_monitor(client):
resp = client.get('/api/monitor') resp = client.get('/api/monitor')
assert resp.status_code == 200 assert resp.status_code == 200
data = resp.get_json() data = resp.get_json()
assert data['status'] == 'ok' assert data['status'] == 'ok'
assert 'uptime_seconds' in data assert 'uptime_seconds' in data
# Neue Tests für bessere Coverage
def test_feiertage_api_error(client):
"""Test Fehlerbehandlung bei Feiertage-API"""
with patch('app.requests.get') as mock_get:
mock_get.side_effect = Exception("Network error")
resp = client.post('/', data={
'action': 'tage_werktage',
'start1': '2024-01-01',
'end1': '2024-01-10',
'bundesland': 'BY'
})
assert resp.status_code == 200
def test_logging_error_handling(client):
"""Test Logging-Fehlerbehandlung"""
# Test ohne Mock, da die App das Logging-Handling bereits hat
resp = client.get('/')
assert resp.status_code == 200
def test_invalid_date_handling(client):
"""Test ungültige Datumseingaben"""
# Ungültiges Datum bei tage_werktage
resp = client.post('/', data={
'action': 'tage_werktage',
'start1': 'invalid-date',
'end1': '2024-01-10'
})
assert resp.status_code == 200
html = resp.data.decode('utf-8')
assert 'Ungültige Eingabe' in html
def test_invalid_wochentag_input(client):
"""Test ungültige Eingaben bei Wochentag-Berechnung"""
resp = client.post('/', data={
'action': 'wochentag',
'datum3': 'invalid-date'
})
assert resp.status_code == 200
html = resp.data.decode('utf-8')
assert 'Ungültige Eingabe' in html
def test_invalid_kw_berechnen_input(client):
"""Test ungültige Eingaben bei KW-Berechnung"""
resp = client.post('/', data={
'action': 'kw_berechnen',
'datum6': 'invalid-date'
})
assert resp.status_code == 200
html = resp.data.decode('utf-8')
assert 'Ungültige Eingabe' in html
def test_invalid_kw_datum_input(client):
"""Test ungültige Eingaben bei KW-Datum"""
resp = client.post('/', data={
'action': 'kw_datum',
'jahr7': 'invalid',
'kw7': 'invalid'
})
assert resp.status_code == 200
html = resp.data.decode('utf-8')
assert 'Ungültige Eingabe' in html
def test_invalid_plusminus_input(client):
"""Test ungültige Eingaben bei Plusminus-Berechnung"""
resp = client.post('/', data={
'action': 'plusminus',
'datum_pm': 'invalid-date',
'anzahl_pm': 'invalid',
'einheit_pm': 'tage',
'richtung_pm': 'add'
})
assert resp.status_code == 200
html = resp.data.decode('utf-8')
assert 'Ungültige Eingabe' in html
def test_stats_login_success(client):
"""Test erfolgreiche Anmeldung im Stats-Bereich"""
with client.session_transaction() as sess:
sess['stats_auth'] = True
resp = client.get('/stats')
assert resp.status_code == 200
def test_stats_login_failure(client):
"""Test fehlgeschlagene Anmeldung im Stats-Bereich"""
resp = client.post('/stats', data={'password': 'wrong'})
assert resp.status_code == 200
html = resp.data.decode('utf-8')
assert 'Falsches Passwort' in html
def test_api_error_handling(client):
"""Test API-Fehlerbehandlung"""
# Test mit korrektem Content-Type
resp = client.post('/api/tage_werktage',
data='invalid json',
content_type='application/json')
assert resp.status_code == 400
def test_api_plusminus_werktage_unsupported(client):
"""Test nicht unterstützte Werktage + Wochen/Monate"""
# Werktage + Wochen
resp = client.post('/api/plusminus', json={
'datum': '2024-06-10', 'anzahl': 5, 'einheit': 'wochen', 'werktage': True
})
assert resp.status_code == 400
assert 'Nicht unterstützt' in resp.get_json()['error']
# Werktage + Monate
resp = client.post('/api/plusminus', json={
'datum': '2024-06-10', 'anzahl': 5, 'einheit': 'monate', 'werktage': True
})
assert resp.status_code == 400
assert 'Nicht unterstützt' in resp.get_json()['error']
def test_api_logging(client):
"""Test API-Logging"""
resp = client.post('/api/wochentag', json={'datum': '2024-06-10'})
assert resp.status_code == 200
# Prüfe ob Log-Datei existiert
log_path = os.path.join('log', 'pageviews.log')
assert os.path.exists(log_path)
def test_api_docs(client):
"""Test API-Dokumentation"""
resp = client.get('/api-docs')
assert resp.status_code == 200
def test_monitor_api_details(client):
"""Test detaillierte Monitor-API"""
resp = client.get('/api/monitor')
assert resp.status_code == 200
data = resp.get_json()
assert 'status' in data
assert 'message' in data
assert 'time' in data
assert 'uptime_seconds' in data
assert 'pageviews_last_7_days' in data

Binary file not shown.

View File

@@ -0,0 +1,483 @@
# German translations for Elpatrons Datumsrechner.
# Copyright (C) 2025 M. Busche
# This file is distributed under the same license as the Elpatrons Datumsrechner package.
msgid ""
msgstr ""
"Project-Id-Version: 1.3.13\n"
"Report-Msgid-Bugs-To: elpatron@mailbox.org\n"
"POT-Creation-Date: 2025-08-01 15:58+0100\n"
"PO-Revision-Date: 2025-08-01 15:58+0100\n"
"Last-Translator: M. Busche <elpatron@mailbox.org>\n"
"Language: de\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
"Plural-Forms: nplurals=2; plural=(n != 1);\n"
#: templates/index.html:12
msgid "Elpatrons Datumsrechner Open Source Kalender- und Datumsberechnungen"
msgstr "Elpatrons Datumsrechner Open Source Kalender- und Datumsberechnungen"
#: templates/index.html:13
msgid "Elpatrons Datumsrechner: Open Source Web-App für Kalender- und Datumsberechnungen. Tage, Werktage, Wochen, Monate, Kalenderwochen, Wochentage und mehr berechnen barrierefrei, werbefrei, trackingfrei, kostenlos."
msgstr "Elpatrons Datumsrechner: Open Source Web-App für Kalender- und Datumsberechnungen. Tage, Werktage, Wochen, Monate, Kalenderwochen, Wochentage und mehr berechnen barrierefrei, werbefrei, trackingfrei, kostenlos."
#: templates/index.html:14
msgid "Datum, Kalender, Datumsrechner, Werktage, Tage zählen, Kalenderwoche, Wochentag, Open Source, Python, Flask, barrierefreiheit, barrierefrei, kostenlos, werbefrei, trackingfrei, cookiefrei"
msgstr "Datum, Kalender, Datumsrechner, Werktage, Tage zählen, Kalenderwoche, Wochentag, Open Source, Python, Flask, barrierefreiheit, barrierefrei, kostenlos, werbefrei, trackingfrei, cookiefrei"
#: templates/index.html:15
msgid "Elpatrons Datumsrechner Open Source Kalender- und Datumsberechnungen"
msgstr "Elpatrons Datumsrechner Open Source Kalender- und Datumsberechnungen"
#: templates/index.html:16
msgid "Open Source Web-App für Kalender- und Datumsberechnungen. Werbefrei, trackingfrei, kostenlos."
msgstr "Open Source Web-App für Kalender- und Datumsberechnungen. Werbefrei, trackingfrei, kostenlos."
#: templates/index.html:17
msgid "website"
msgstr "website"
#: templates/index.html:18
msgid "https://codeberg.org/elpatron/datecalc"
msgstr "https://codeberg.org/elpatron/datecalc"
#: templates/index.html:19
msgid "/static/logo.svg"
msgstr "/static/logo.svg"
#: templates/index.html:20
msgid "width=device-width, initial-scale=1"
msgstr "width=device-width, initial-scale=1"
#: templates/index.html:21
msgid "true"
msgstr "true"
#: templates/index.html:22
msgid "yes"
msgstr "yes"
#: templates/index.html:23
msgid "default"
msgstr "default"
#: templates/index.html:24
msgid "telephone=no"
msgstr "telephone=no"
#: templates/index.html:25
msgid "IE=edge"
msgstr "IE=edge"
#: templates/index.html:26
msgid "Elpatrons Datumsrechner"
msgstr "Elpatrons Datumsrechner"
#: templates/index.html:27
msgid "#2563eb"
msgstr "#2563eb"
#: templates/index.html:28
msgid "/static/favicon.ico"
msgstr "/static/favicon.ico"
#: templates/index.html:29
msgid "Hilfe anzeigen"
msgstr "Hilfe anzeigen"
#: templates/index.html:30
msgid "Hilfe anzeigen"
msgstr "Hilfe anzeigen"
#: templates/index.html:31
msgid "Öffnet ein Hilfefenster mit Informationen über den Datumsrechner"
msgstr "Öffnet ein Hilfefenster mit Informationen über den Datumsrechner"
#: templates/index.html:32
msgid "Elpatrons"
msgstr "Elpatrons"
#: templates/index.html:33
msgid "Datumsrechner"
msgstr "Datumsrechner"
#: templates/index.html:34
msgid "Eine <em>freie</em> Web-App: barriere<em>frei</em>, werbe<em>frei</em>, tracking<em>frei</em>, lizenz<em>frei</em> und kosten<em>frei</em>."
msgstr "Eine <em>freie</em> Web-App: barriere<em>frei</em>, werbe<em>frei</em>, tracking<em>frei</em>, lizenz<em>frei</em> und kosten<em>frei</em>."
#: templates/index.html:35
msgid "Anzahl der Tage/Werktage zwischen zwei Daten"
msgstr "Anzahl der Tage/Werktage zwischen zwei Daten"
#: templates/index.html:36
msgid "Startdatum:"
msgstr "Startdatum:"
#: templates/index.html:37
msgid "Heute"
msgstr "Heute"
#: templates/index.html:38
msgid "Enddatum:"
msgstr "Enddatum:"
#: templates/index.html:39
msgid "Optionen"
msgstr "Optionen"
#: templates/index.html:40
msgid "Nur Werktage"
msgstr "Nur Werktage"
#: templates/index.html:41
msgid "Feiertage berücksichtigen für:"
msgstr "Feiertage berücksichtigen für:"
#: templates/index.html:42
msgid "(kein Bundesland)"
msgstr "(kein Bundesland)"
#: templates/index.html:43
msgid "Baden-Württemberg"
msgstr "Baden-Württemberg"
#: templates/index.html:44
msgid "Bayern"
msgstr "Bayern"
#: templates/index.html:45
msgid "Berlin"
msgstr "Berlin"
#: templates/index.html:46
msgid "Brandenburg"
msgstr "Brandenburg"
#: templates/index.html:47
msgid "Bremen"
msgstr "Bremen"
#: templates/index.html:48
msgid "Hamburg"
msgstr "Hamburg"
#: templates/index.html:49
msgid "Hessen"
msgstr "Hessen"
#: templates/index.html:50
msgid "Mecklenburg-Vorpommern"
msgstr "Mecklenburg-Vorpommern"
#: templates/index.html:51
msgid "Niedersachsen"
msgstr "Niedersachsen"
#: templates/index.html:52
msgid "Nordrhein-Westfalen"
msgstr "Nordrhein-Westfalen"
#: templates/index.html:53
msgid "Rheinland-Pfalz"
msgstr "Rheinland-Pfalz"
#: templates/index.html:54
msgid "Saarland"
msgstr "Saarland"
#: templates/index.html:55
msgid "Sachsen"
msgstr "Sachsen"
#: templates/index.html:56
msgid "Sachsen-Anhalt"
msgstr "Sachsen-Anhalt"
#: templates/index.html:57
msgid "Schleswig-Holstein"
msgstr "Schleswig-Holstein"
#: templates/index.html:58
msgid "Thüringen"
msgstr "Thüringen"
#: templates/index.html:59
msgid "Berechnen"
msgstr "Berechnen"
#: templates/index.html:60
msgid "Ergebnis vorlesen"
msgstr "Ergebnis vorlesen"
#: templates/index.html:61
msgid "Ergebnis vorlesen"
msgstr "Ergebnis vorlesen"
#: templates/index.html:62
msgid "Anzahl der Werktage zwischen"
msgstr "Anzahl der Werktage zwischen"
#: templates/index.html:63
msgid "und"
msgstr "und"
#: templates/index.html:64
msgid "(Feiertage:"
msgstr "(Feiertage:"
#: templates/index.html:65
msgid "Anzahl der Tage zwischen"
msgstr "Anzahl der Tage zwischen"
#: templates/index.html:66
msgid "Davon sind"
msgstr "Davon sind"
#: templates/index.html:67
msgid "Tage Wochenendtage."
msgstr "Tage Wochenendtage."
#: templates/index.html:68
msgid "Feiertage (Mo-Fr,"
msgstr "Feiertage (Mo-Fr,"
#: templates/index.html:69
msgid "Wochentag eines Datums"
msgstr "Wochentag eines Datums"
#: templates/index.html:70
msgid "Datum:"
msgstr "Datum:"
#: templates/index.html:71
msgid "Anzeigen"
msgstr "Anzeigen"
#: templates/index.html:72
msgid "Wochentag von"
msgstr "Wochentag von"
#: templates/index.html:73
msgid "Kalenderwoche eines Datums"
msgstr "Kalenderwoche eines Datums"
#: templates/index.html:74
msgid "Kalenderwoche berechnen"
msgstr "Kalenderwoche berechnen"
#: templates/index.html:75
msgid "Kalenderwoche von"
msgstr "Kalenderwoche von"
#: templates/index.html:76
msgid "Start-/Enddatum zu Kalenderwoche"
msgstr "Start-/Enddatum zu Kalenderwoche"
#: templates/index.html:77
msgid "Jahr:"
msgstr "Jahr:"
#: templates/index.html:78
msgid "Kalenderwoche:"
msgstr "Kalenderwoche:"
#: templates/index.html:79
msgid "Start-/Enddatum berechnen"
msgstr "Start-/Enddatum berechnen"
#: templates/index.html:80
msgid "Start-/Enddatum der KW"
msgstr "Start-/Enddatum der KW"
#: templates/index.html:81
msgid "im Jahr"
msgstr "im Jahr"
#: templates/index.html:82
msgid "Datum plus/minus X Tage/Wochen/Monate"
msgstr "Datum plus/minus X Tage/Wochen/Monate"
#: templates/index.html:83
msgid "Anzahl:"
msgstr "Anzahl:"
#: templates/index.html:84
msgid "Rechenrichtung"
msgstr "Rechenrichtung"
#: templates/index.html:85
msgid "addieren"
msgstr "addieren"
#: templates/index.html:86
msgid "subtrahieren"
msgstr "subtrahieren"
#: templates/index.html:87
msgid "Einheit und Werktage"
msgstr "Einheit und Werktage"
#: templates/index.html:88
msgid "Einheit:"
msgstr "Einheit:"
#: templates/index.html:89
msgid "Tage"
msgstr "Tage"
#: templates/index.html:90
msgid "Wochen"
msgstr "Wochen"
#: templates/index.html:91
msgid "Monate"
msgstr "Monate"
#: templates/index.html:92
msgid "Nur Werktage"
msgstr "Nur Werktage"
#: templates/index.html:93
msgid "Hilfe schließen"
msgstr "Hilfe schließen"
#: templates/index.html:94
msgid "Was ist Elpatrons Datumsrechner?"
msgstr "Was ist Elpatrons Datumsrechner?"
#: templates/index.html:95
msgid "Der Datumsrechner kann verschiedene Datumsberechnungen durchführen:"
msgstr "Der Datumsrechner kann verschiedene Datumsberechnungen durchführen:"
#: templates/index.html:96
msgid "Anzahl der Tage zwischen zwei Daten"
msgstr "Anzahl der Tage zwischen zwei Daten"
#: templates/index.html:97
msgid "Anzahl der Werktage zwischen zwei Daten"
msgstr "Anzahl der Werktage zwischen zwei Daten"
#: templates/index.html:98
msgid "Anzeige des Wochentags eines Datums"
msgstr "Anzeige des Wochentags eines Datums"
#: templates/index.html:99
msgid "Datum plus/minus X Tage"
msgstr "Datum plus/minus X Tage"
#: templates/index.html:100
msgid "Datum plus/minus X Werktage"
msgstr "Datum plus/minus X Werktage"
#: templates/index.html:101
msgid "Datum plus/minus X Wochen/Monate"
msgstr "Datum plus/minus X Wochen/Monate"
#: templates/index.html:102
msgid "Kalenderwoche zu Datum"
msgstr "Kalenderwoche zu Datum"
#: templates/index.html:103
msgid "Start-/Enddatum einer Kalenderwoche eines Jahres"
msgstr "Start-/Enddatum einer Kalenderwoche eines Jahres"
#: templates/index.html:104
msgid "Online Datumsrechner gibt es bereits in einer Vielzahl, warum also noch einer?"
msgstr "Online Datumsrechner gibt es bereits in einer Vielzahl, warum also noch einer?"
#: templates/index.html:105
msgid "Aus zwei Gründen:"
msgstr "Aus zwei Gründen:"
#: templates/index.html:106
msgid "Finde mal einen Datumsrechner, der nicht vollkommen verseucht mit Werbung, Tracking und Cookies ist!"
msgstr "Finde mal einen Datumsrechner, der nicht vollkommen verseucht mit Werbung, Tracking und Cookies ist!"
#: templates/index.html:107
msgid "Das hat mich so geärgert, dass ich meinen eigenen programmiert habe."
msgstr "Das hat mich so geärgert, dass ich meinen eigenen programmiert habe."
#: templates/index.html:108
msgid "Genau genommen nicht ich selbst. Diese App wurde zum überwiegenden Teil von KI nach meinen Anweisungen entwickelt (Vibe Coding)."
msgstr "Genau genommen nicht ich selbst. Diese App wurde zum überwiegenden Teil von KI nach meinen Anweisungen entwickelt (Vibe Coding)."
#: templates/index.html:109
msgid "Was du noch wissen solltest"
msgstr "Was du noch wissen solltest"
#: templates/index.html:110
msgid "Ich habe versucht, die App möglichst barrierefrei zu gestalten, um Menschen mit Einschränkungen die Benutzung zu erleichtern."
msgstr "Ich habe versucht, die App möglichst barrierefrei zu gestalten, um Menschen mit Einschränkungen die Benutzung zu erleichtern."
#: templates/index.html:111
msgid "Diese App schnüffelt dir nicht hinterher, sammelt keine persönlichen Daten und geht dir auch sonst (hoffentlich!) in keiner Weise auf die Nerven."
msgstr "Diese App schnüffelt dir nicht hinterher, sammelt keine persönlichen Daten und geht dir auch sonst (hoffentlich!) in keiner Weise auf die Nerven."
#: templates/index.html:112
msgid "Den Quellcode dieser App habe ich auf"
msgstr "Den Quellcode dieser App habe ich auf"
#: templates/index.html:113
msgid "veröffentlicht, du kannst ihn einsehen, verändern oder damit deinen eigenen kleinen Datumsrechner betreiben."
msgstr "veröffentlicht, du kannst ihn einsehen, verändern oder damit deinen eigenen kleinen Datumsrechner betreiben."
#: templates/index.html:114
msgid "Die App läuft auf meinem kleinen Home-Server und ist derzeit nicht für große Besucherzahlen ausgelegt."
msgstr "Die App läuft auf meinem kleinen Home-Server und ist derzeit nicht für große Besucherzahlen ausgelegt."
#: templates/index.html:115
msgid "Ich übernehme keine Gewähr für die Funktionalität und die Rechenergebnisse. Die KI, die das programmiert hat, übrigens auch nicht."
msgstr "Ich übernehme keine Gewähr für die Funktionalität und die Rechenergebnisse. Die KI, die das programmiert hat, übrigens auch nicht."
#: templates/index.html:116
msgid "Falls du einen Fehler findest oder eine weitere Funktion wünschst, kannst du mir eine Mail schreiben (siehe Mailto Link in der Fußzeile)"
msgstr "Falls du einen Fehler findest oder eine weitere Funktion wünschst, kannst du mir eine Mail schreiben (siehe Mailto Link in der Fußzeile)"
#: templates/index.html:117
msgid "Hab Spaß mit Elpatrons Datumsrechner! Dein M. Busche"
msgstr "Hab Spaß mit Elpatrons Datumsrechner! Dein M. Busche"
#: templates/index.html:118
msgid "Hilfe-Informationen für den Datumsrechner mit Erklärungen zu allen Funktionen"
msgstr "Hilfe-Informationen für den Datumsrechner mit Erklärungen zu allen Funktionen"
#: templates/index.html:119
msgid "Dies ist ein werbe- und trackingfreier"
msgstr "Dies ist ein werbe- und trackingfreier"
#: templates/index.html:120
msgid "Open Source Datumsrechner"
msgstr "Open Source Datumsrechner"
#: templates/index.html:121
msgid "REST API Dokumentation (Swagger)"
msgstr "REST API Dokumentation (Swagger)"
#: templates/index.html:122
msgid "M. Busche"
msgstr "M. Busche"
#: app.py:123
msgid "Ungültige Eingabe"
msgstr "Ungültige Eingabe"
#: app.py:124
msgid "Nicht unterstützt: Werktage + Wochen."
msgstr "Nicht unterstützt: Werktage + Wochen."
#: app.py:125
msgid "Nicht unterstützt: Werktage + Monate."
msgstr "Nicht unterstützt: Werktage + Monate."
#: templates/index.html:126
msgid "Sprache auswählen"
msgstr "Sprache auswählen"
#: templates/index.html:127
msgid "Deutsch auswählen"
msgstr "Deutsch auswählen"
#: templates/index.html:131
msgid "English auswählen"
msgstr "English auswählen"

Binary file not shown.

View File

@@ -0,0 +1,587 @@
# English translations for Elpatrons Datumsrechner.
# Copyright (C) 2025 M. Busche
# This file is distributed under the same license as the Elpatrons Datumsrechner package.
msgid ""
msgstr ""
"Project-Id-Version: 1.3.13\n"
"Report-Msgid-Bugs-To: elpatron@mailbox.org\n"
"POT-Creation-Date: 2025-08-01 15:58+0100\n"
"PO-Revision-Date: 2025-08-01 15:58+0100\n"
"Last-Translator: M. Busche <elpatron@mailbox.org>\n"
"Language: en\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
"Plural-Forms: nplurals=2; plural=(n != 1);\n"
#: templates/index.html:12
msgid "Elpatrons Datumsrechner Open Source Kalender- und Datumsberechnungen"
msgstr "Elpatrons Date Calculator Open Source Calendar and Date Calculations"
#: templates/index.html:13
msgid "Elpatrons Datumsrechner: Open Source Web-App für Kalender- und Datumsberechnungen. Tage, Werktage, Wochen, Monate, Kalenderwochen, Wochentage und mehr berechnen barrierefrei, werbefrei, trackingfrei, kostenlos."
msgstr "Elpatrons Date Calculator: Open Source web app for calendar and date calculations. Calculate days, workdays, weeks, months, calendar weeks, weekdays and more accessible, ad-free, tracking-free, free of charge."
#: templates/index.html:14
msgid "Datum, Kalender, Datumsrechner, Werktage, Tage zählen, Kalenderwoche, Wochentag, Open Source, Python, Flask, barrierefreiheit, barrierefrei, kostenlos, werbefrei, trackingfrei, cookiefrei"
msgstr "Date, Calendar, Date Calculator, Workdays, Count Days, Calendar Week, Weekday, Open Source, Python, Flask, accessibility, accessible, free, ad-free, tracking-free, cookie-free"
#: templates/index.html:15
msgid "Elpatrons Datumsrechner Open Source Kalender- und Datumsberechnungen"
msgstr "Elpatrons Date Calculator Open Source Calendar and Date Calculations"
#: templates/index.html:16
msgid "Open Source Web-App für Kalender- und Datumsberechnungen. Werbefrei, trackingfrei, kostenlos."
msgstr "Open Source web app for calendar and date calculations. Ad-free, tracking-free, free of charge."
#: templates/index.html:17
msgid "website"
msgstr "website"
#: templates/index.html:18
msgid "https://codeberg.org/elpatron/datecalc"
msgstr "https://codeberg.org/elpatron/datecalc"
#: templates/index.html:19
msgid "/static/logo.svg"
msgstr "/static/logo.svg"
#: templates/index.html:20
msgid "width=device-width, initial-scale=1"
msgstr "width=device-width, initial-scale=1"
#: templates/index.html:21
msgid "true"
msgstr "true"
#: templates/index.html:22
msgid "yes"
msgstr "yes"
#: templates/index.html:23
msgid "default"
msgstr "default"
#: templates/index.html:24
msgid "telephone=no"
msgstr "telephone=no"
#: templates/index.html:25
msgid "IE=edge"
msgstr "IE=edge"
#: templates/index.html:26
msgid "Elpatrons Datumsrechner"
msgstr "Elpatrons Date Calculator"
#: templates/index.html:27
msgid "#2563eb"
msgstr "#2563eb"
#: templates/index.html:28
msgid "/static/favicon.ico"
msgstr "/static/favicon.ico"
#: templates/index.html:29
msgid "Hilfe anzeigen"
msgstr "Show Help"
#: templates/index.html:30
msgid "Hilfe anzeigen"
msgstr "Show Help"
#: templates/index.html:31
msgid "Öffnet ein Hilfefenster mit Informationen über den Datumsrechner"
msgstr "Opens a help window with information about the date calculator"
#: templates/index.html:32
msgid "Elpatrons"
msgstr "Elpatrons"
#: templates/index.html:33
msgid "Datumsrechner"
msgstr "Date Calculator"
#: templates/index.html:34
msgid "Eine <em>freie</em> Web-App: barriere<em>frei</em>, werbe<em>frei</em>, tracking<em>frei</em>, lizenz<em>frei</em> und kosten<em>frei</em>."
msgstr "A <em>free</em> web app: <em>accessible</em>, <em>ad-free</em>, <em>tracking-free</em>, <em>license-free</em> and <em>cost-free</em>."
#: templates/index.html:35
msgid "Anzahl der Tage/Werktage zwischen zwei Daten"
msgstr "Number of days/workdays between two dates"
#: templates/index.html:36
msgid "Startdatum:"
msgstr "Start date:"
#: templates/index.html:37
msgid "Heute"
msgstr "Today"
#: templates/index.html:38
msgid "Enddatum:"
msgstr "End date:"
#: templates/index.html:39
msgid "Optionen"
msgstr "Options"
#: templates/index.html:40
msgid "Nur Werktage"
msgstr "Workdays only"
#: templates/index.html:41
msgid "Feiertage berücksichtigen für:"
msgstr "Consider holidays for:"
#: templates/index.html:42
msgid "(kein Bundesland)"
msgstr "(no state)"
#: templates/index.html:43
msgid "Baden-Württemberg"
msgstr "Baden-Württemberg"
#: templates/index.html:44
msgid "Bayern"
msgstr "Bavaria"
#: templates/index.html:45
msgid "Berlin"
msgstr "Berlin"
#: templates/index.html:46
msgid "Brandenburg"
msgstr "Brandenburg"
#: templates/index.html:47
msgid "Bremen"
msgstr "Bremen"
#: templates/index.html:48
msgid "Hamburg"
msgstr "Hamburg"
#: templates/index.html:49
msgid "Hessen"
msgstr "Hesse"
#: templates/index.html:50
msgid "Mecklenburg-Vorpommern"
msgstr "Mecklenburg-Vorpommern"
#: templates/index.html:51
msgid "Niedersachsen"
msgstr "Lower Saxony"
#: templates/index.html:52
msgid "Nordrhein-Westfalen"
msgstr "North Rhine-Westphalia"
#: templates/index.html:53
msgid "Rheinland-Pfalz"
msgstr "Rhineland-Palatinate"
#: templates/index.html:54
msgid "Saarland"
msgstr "Saarland"
#: templates/index.html:55
msgid "Sachsen"
msgstr "Saxony"
#: templates/index.html:56
msgid "Sachsen-Anhalt"
msgstr "Saxony-Anhalt"
#: templates/index.html:57
msgid "Schleswig-Holstein"
msgstr "Schleswig-Holstein"
#: templates/index.html:58
msgid "Thüringen"
msgstr "Thuringia"
#: templates/index.html:59
msgid "Berechnen"
msgstr "Calculate"
#: templates/index.html:60
msgid "Ergebnis vorlesen"
msgstr "Read result aloud"
#: templates/index.html:61
msgid "Ergebnis vorlesen"
msgstr "Read result aloud"
#: templates/index.html:62
msgid "Anzahl der Werktage zwischen"
msgstr "Number of workdays between"
#: templates/index.html:63
msgid "und"
msgstr "and"
#: templates/index.html:64
msgid "(Feiertage:"
msgstr "(Holidays:"
#: templates/index.html:65
msgid "Anzahl der Tage zwischen"
msgstr "Number of days between"
#: templates/index.html:66
msgid "Davon sind"
msgstr "Of which"
#: templates/index.html:67
msgid "Tage Wochenendtage."
msgstr "days are weekend days."
#: templates/index.html:68
msgid "Feiertage (Mo-Fr,"
msgstr "holidays (Mon-Fri,"
#: templates/index.html:69
msgid "Wochentag eines Datums"
msgstr "Weekday of a date"
#: templates/index.html:70
msgid "Datum:"
msgstr "Date:"
#: templates/index.html:71
msgid "Anzeigen"
msgstr "Show"
#: templates/index.html:72
msgid "Wochentag von"
msgstr "Weekday of"
#: templates/index.html:73
msgid "Kalenderwoche eines Datums"
msgstr "Calendar week of a date"
#: templates/index.html:74
msgid "Kalenderwoche berechnen"
msgstr "Calculate calendar week"
#: templates/index.html:75
msgid "Kalenderwoche von"
msgstr "Calendar week of"
#: templates/index.html:76
msgid "Start-/Enddatum zu Kalenderwoche"
msgstr "Start/end date to calendar week"
#: templates/index.html:77
msgid "Jahr:"
msgstr "Year:"
#: templates/index.html:78
msgid "Kalenderwoche:"
msgstr "Calendar week:"
#: templates/index.html:79
msgid "Start-/Enddatum berechnen"
msgstr "Calculate start/end date"
#: templates/index.html:80
msgid "Start-/Enddatum der KW"
msgstr "Start/end date of week"
#: templates/index.html:81
msgid "im Jahr"
msgstr "in year"
#: templates/index.html:82
msgid "Datum plus/minus X Tage/Wochen/Monate"
msgstr "Date plus/minus X days/weeks/months"
#: templates/index.html:83
msgid "Anzahl:"
msgstr "Amount:"
#: templates/index.html:84
msgid "Rechenrichtung"
msgstr "Calculation direction"
#: templates/index.html:85
msgid "addieren"
msgstr "add"
#: templates/index.html:86
msgid "subtrahieren"
msgstr "subtract"
#: templates/index.html:87
msgid "Einheit und Werktage"
msgstr "Unit and workdays"
#: templates/index.html:88
msgid "Einheit:"
msgstr "Unit:"
#: templates/index.html:89
msgid "Tage"
msgstr "Days"
#: templates/index.html:90
msgid "Wochen"
msgstr "Weeks"
#: templates/index.html:91
msgid "Monate"
msgstr "Months"
#: templates/index.html:92
msgid "Nur Werktage"
msgstr "Workdays only"
#: templates/index.html:93
msgid "Hilfe schließen"
msgstr "Close help"
#: templates/index.html:94
msgid "Was ist Elpatrons Datumsrechner?"
msgstr "What is Elpatrons Date Calculator?"
#: templates/index.html:95
msgid "Der Datumsrechner kann verschiedene Datumsberechnungen durchführen:"
msgstr "The date calculator can perform various date calculations:"
#: templates/index.html:96
msgid "Anzahl der Tage zwischen zwei Daten"
msgstr "Number of days between two dates"
#: templates/index.html:97
msgid "Anzahl der Werktage zwischen zwei Daten"
msgstr "Number of workdays between two dates"
#: templates/index.html:98
msgid "Anzeige des Wochentags eines Datums"
msgstr "Display of the weekday of a date"
#: templates/index.html:99
msgid "Datum plus/minus X Tage"
msgstr "Date plus/minus X days"
#: templates/index.html:100
msgid "Datum plus/minus X Werktage"
msgstr "Date plus/minus X workdays"
#: templates/index.html:101
msgid "Datum plus/minus X Wochen/Monate"
msgstr "Date plus/minus X weeks/months"
#: templates/index.html:102
msgid "Kalenderwoche zu Datum"
msgstr "Calendar week to date"
#: templates/index.html:103
msgid "Start-/Enddatum einer Kalenderwoche eines Jahres"
msgstr "Start/end date of a calendar week of a year"
#: templates/index.html:104
msgid "Online Datumsrechner gibt es bereits in einer Vielzahl, warum also noch einer?"
msgstr "Online date calculators already exist in abundance, so why another one?"
#: templates/index.html:105
msgid "Aus zwei Gründen:"
msgstr "For two reasons:"
#: templates/index.html:106
msgid "Finde mal einen Datumsrechner, der nicht vollkommen verseucht mit Werbung, Tracking und Cookies ist!"
msgstr "Try to find a date calculator that isn't completely infested with ads, tracking and cookies!"
#: templates/index.html:107
msgid "Das hat mich so geärgert, dass ich meinen eigenen programmiert habe."
msgstr "This annoyed me so much that I programmed my own."
#: templates/index.html:108
msgid "Genau genommen nicht ich selbst. Diese App wurde zum überwiegenden Teil von KI nach meinen Anweisungen entwickelt (Vibe Coding)."
msgstr "Actually not me myself. This app was developed largely by AI following my instructions (Vibe Coding)."
#: templates/index.html:109
msgid "Was du noch wissen solltest"
msgstr "What else you should know"
#: templates/index.html:110
msgid "Ich habe versucht, die App möglichst barrierefrei zu gestalten, um Menschen mit Einschränkungen die Benutzung zu erleichtern."
msgstr "I have tried to make the app as accessible as possible to make it easier for people with disabilities to use."
#: templates/index.html:111
msgid "Diese App schnüffelt dir nicht hinterher, sammelt keine persönlichen Daten und geht dir auch sonst (hoffentlich!) in keiner Weise auf die Nerven."
msgstr "This app doesn't spy on you, doesn't collect personal data and doesn't get on your nerves in any other way (hopefully!)."
#: templates/index.html:112
msgid "Den Quellcode dieser App habe ich auf"
msgstr "I have published the source code of this app on"
#: templates/index.html:113
msgid "veröffentlicht, du kannst ihn einsehen, verändern oder damit deinen eigenen kleinen Datumsrechner betreiben."
msgstr "you can view it, modify it or use it to run your own small date calculator."
#: templates/index.html:114
msgid "Die App läuft auf meinem kleinen Home-Server und ist derzeit nicht für große Besucherzahlen ausgelegt."
msgstr "The app runs on my small home server and is currently not designed for large visitor numbers."
#: templates/index.html:115
msgid "Ich übernehme keine Gewähr für die Funktionalität und die Rechenergebnisse. Die KI, die das programmiert hat, übrigens auch nicht."
msgstr "I do not guarantee the functionality and calculation results. By the way, neither does the AI that programmed this."
#: templates/index.html:116
msgid "Falls du einen Fehler findest oder eine weitere Funktion wünschst, kannst du mir eine Mail schreiben (siehe Mailto Link in der Fußzeile)"
msgstr "If you find a bug or want an additional feature, you can write me an email (see mailto link in the footer)"
#: templates/index.html:117
msgid "Hab Spaß mit Elpatrons Datumsrechner! Dein M. Busche"
msgstr "Have fun with Elpatrons Date Calculator! Yours, M. Busche"
#: templates/index.html:118
msgid "Hilfe-Informationen für den Datumsrechner mit Erklärungen zu allen Funktionen"
msgstr "Help information for the date calculator with explanations of all functions"
#: templates/index.html:119
msgid "Dies ist ein werbe- und trackingfreier"
msgstr "This is an ad- and tracking-free"
#: templates/index.html:120
msgid "Open Source Datumsrechner"
msgstr "Open Source Date Calculator"
#: templates/index.html:121
msgid "REST API Dokumentation (Swagger)"
msgstr "REST API Documentation (Swagger)"
#: templates/index.html:122
msgid "M. Busche"
msgstr "M. Busche"
#: app.py:123
msgid "Ungültige Eingabe"
msgstr "Invalid input"
#: app.py:124
msgid "Nicht unterstützt: Werktage + Wochen."
msgstr "Not supported: Workdays + weeks."
#: app.py:125
msgid "Nicht unterstützt: Werktage + Monate."
msgstr "Not supported: Workdays + months."
#: templates/index.html:126
msgid "Sprache auswählen"
msgstr "Select language"
#: templates/index.html:127
msgid "Deutsch auswählen"
msgstr "Select German"
#: templates/index.html:131
msgid "English auswählen"
msgstr "Select English"
#: templates/index.html:132
msgid "Taschenrechner"
msgstr "Calculator"
#: templates/index.html:133
msgid "Taschenrechner öffnen"
msgstr "Open calculator"
#: templates/index.html:134
msgid "Taschenrechner schließen"
msgstr "Close calculator"
#: templates/index.html:135
msgid "Verwenden Sie die Tab-Taste um durch die Tasten zu navigieren. Tastatur-Kurzbefehle: Zahlen 0-9, Punkt oder Komma für Dezimal, Plus (+) oder P für Addition, Minus (-) oder M für Subtraktion, Stern (*) oder X für Multiplikation, Schrägstrich (/) oder D für Division, Enter oder Leertaste für Gleich, C für Löschen, Backspace für letzte Ziffer löschen."
msgstr "Use the Tab key to navigate through the buttons. Keyboard shortcuts: Numbers 0-9, period or comma for decimal, Plus (+) or P for addition, Minus (-) or M for subtraction, Asterisk (*) or X for multiplication, Slash (/) or D for division, Enter or Space for equals, C for clear, Backspace for delete last digit."
#: templates/index.html:136
msgid "Taschenrechner Anzeige"
msgstr "Calculator display"
#: templates/index.html:137
msgid "Berechnungsverlauf"
msgstr "Calculation history"
#: templates/index.html:138
msgid "Löschen (Taste: C)"
msgstr "Clear (key: C)"
#: templates/index.html:139
msgid "Letzte Ziffer löschen (Taste: Backspace)"
msgstr "Delete last digit (key: Backspace)"
#: templates/index.html:140
msgid "Dividieren (Taste: / oder D)"
msgstr "Divide (key: / or D)"
#: templates/index.html:141
msgid "Multiplizieren (Taste: * oder X)"
msgstr "Multiply (key: * or X)"
#: templates/index.html:142
msgid "Sieben"
msgstr "Seven"
#: templates/index.html:143
msgid "Acht"
msgstr "Eight"
#: templates/index.html:144
msgid "Neun"
msgstr "Nine"
#: templates/index.html:145
msgid "Subtrahieren (Taste: - oder M)"
msgstr "Subtract (key: - or M)"
#: templates/index.html:146
msgid "Vier"
msgstr "Four"
#: templates/index.html:147
msgid "Fünf"
msgstr "Five"
#: templates/index.html:148
msgid "Sechs"
msgstr "Six"
#: templates/index.html:149
msgid "Addieren (Taste: + oder P)"
msgstr "Add (key: + or P)"
#: templates/index.html:150
msgid "Eins"
msgstr "One"
#: templates/index.html:151
msgid "Zwei"
msgstr "Two"
#: templates/index.html:152
msgid "Drei"
msgstr "Three"
#: templates/index.html:153
msgid "Gleich (Taste: Enter oder Leertaste)"
msgstr "Equals (key: Enter or Space)"
#: templates/index.html:154
msgid "Null"
msgstr "Zero"
#: templates/index.html:155
msgid "Komma (Taste: . oder ,)"
msgstr "Decimal (key: . or ,)"
#: templates/index.html:156
msgid "Fehler"
msgstr "Error"
#: templates/index.html:157
msgid "Berechnung"
msgstr "Calculation"