Files
food-ai/client/lib/l10n/app_localizations_de.dart
dbastrikin c7317c4335 feat: async product/receipt recognition via Kafka
Backend:
- Migration 002: product_recognition_jobs table with JSONB images column
  and job_type CHECK ('receipt' | 'products')
- New Kafka topics: ai.products.paid / ai.products.free
- ProductJob model, ProductJobRepository (mirrors dish job pattern)
- itemEnricher extracted from Handler — shared by HTTP handler and worker
- ProductSSEBroker: PG LISTEN on product_job_update channel
- ProductWorkerPool: 5 workers, branches on job_type to call
  RecognizeReceipt or RecognizeProducts per image in parallel
- Handler: RecognizeReceipt and RecognizeProducts now return 202 Accepted
  instead of blocking; 4 new endpoints: GET /ai/product-jobs,
  /product-jobs/history, /product-jobs/{id}, /product-jobs/{id}/stream
- cmd/worker: extended to run ProductWorkerPool alongside dish WorkerPool
- cmd/server: wires productJobRepository + productSSEBroker; both SSE
  brokers started in App.Start()

Flutter client:
- ProductJobCreated, ProductJobResult, ProductJobSummary, ProductJobEvent
  models + submitReceiptRecognition/submitProductsRecognition/stream methods
- Shared _openSseStream helper eliminates duplicate SSE parsing loop
- ScanScreen: replace blocking AI calls with async submit + navigate to
  ProductJobWatchScreen
- ProductJobWatchScreen: watches SSE stream, navigates to /scan/confirm
  when done, shows error on failure
- ProductsScreen: prepends _RecentScansSection (hidden when empty); compact
  horizontal list of recent scans with "See all" → history
- ProductJobHistoryScreen: full list of all product recognition jobs
- New routes: /scan/product-job-watch, /products/job-history
- L10n: 7 new keys in all 12 ARB files

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-03-23 23:01:30 +02:00

479 lines
9.8 KiB
Dart

// ignore: unused_import
import 'package:intl/intl.dart' as intl;
import 'app_localizations.dart';
// ignore_for_file: type=lint
/// The translations for German (`de`).
class AppLocalizationsDe extends AppLocalizations {
AppLocalizationsDe([String locale = 'de']) : super(locale);
@override
String get appTitle => 'FoodAI';
@override
String get greetingMorning => 'Guten Morgen';
@override
String get greetingAfternoon => 'Guten Tag';
@override
String get greetingEvening => 'Guten Abend';
@override
String get caloriesUnit => 'kcal';
@override
String get gramsUnit => 'g';
@override
String get goalLabel => 'Ziel:';
@override
String get consumed => 'Verzehrt';
@override
String get remaining => 'Verbleibend';
@override
String get exceeded => 'Überschritten';
@override
String get proteinLabel => 'Protein';
@override
String get fatLabel => 'Fett';
@override
String get carbsLabel => 'Kohlenhydrate';
@override
String get today => 'Heute';
@override
String get yesterday => 'Gestern';
@override
String get mealsSection => 'Mahlzeiten';
@override
String get addDish => 'Gericht hinzufügen';
@override
String get scanDish => 'Scannen';
@override
String get menu => 'Menü';
@override
String get dishHistory => 'Gerichtverlauf';
@override
String get recommendCook => 'Wir empfehlen zu kochen';
@override
String get camera => 'Kamera';
@override
String get gallery => 'Galerie';
@override
String get analyzingPhoto => 'Foto wird analysiert...';
@override
String get inQueue => 'Sie sind in der Warteschlange';
@override
String queuePosition(int position) {
return 'Position $position';
}
@override
String get processing => 'Verarbeitung...';
@override
String get upgradePrompt => 'Warteschlange überspringen? Upgrade →';
@override
String get recognitionFailed => 'Erkennung fehlgeschlagen. Erneut versuchen.';
@override
String get dishRecognition => 'Gerichterkennung';
@override
String get all => 'Alle';
@override
String get dishRecognized => 'Gericht erkannt';
@override
String get recognizing => 'Wird erkannt…';
@override
String get recognitionError => 'Erkennungsfehler';
@override
String get dishResultTitle => 'Gericht erkannt';
@override
String get selectDish => 'Gericht auswählen';
@override
String get dishNotRecognized => 'Gericht nicht erkannt';
@override
String get tryAgain => 'Erneut versuchen';
@override
String get nutritionApproximate =>
'Nährwerte sind ungefähr — aus dem Foto geschätzt.';
@override
String get portion => 'Portion';
@override
String get mealType => 'Mahlzeittyp';
@override
String get dateLabel => 'Datum';
@override
String get addToJournal => 'Zum Tagebuch hinzufügen';
@override
String get addFailed => 'Hinzufügen fehlgeschlagen. Erneut versuchen.';
@override
String get historyTitle => 'Erkennungsverlauf';
@override
String get historyLoadError => 'Verlauf konnte nicht geladen werden';
@override
String get retry => 'Wiederholen';
@override
String get noHistory => 'Noch keine Erkennungen';
@override
String get profileTitle => 'Profil';
@override
String get edit => 'Bearbeiten';
@override
String get bodyParams => 'KÖRPERPARAMETER';
@override
String get goalActivity => 'ZIEL & AKTIVITÄT';
@override
String get nutrition => 'ERNÄHRUNG';
@override
String get settings => 'EINSTELLUNGEN';
@override
String get height => 'Größe';
@override
String get weight => 'Gewicht';
@override
String get age => 'Alter';
@override
String get gender => 'Geschlecht';
@override
String get genderMale => 'Männlich';
@override
String get genderFemale => 'Weiblich';
@override
String get goalLoss => 'Gewichtsverlust';
@override
String get goalMaintain => 'Gewicht halten';
@override
String get goalGain => 'Muskelaufbau';
@override
String get activityLow => 'Niedrig';
@override
String get activityMedium => 'Mittel';
@override
String get activityHigh => 'Hoch';
@override
String get calorieGoal => 'Kalorienziel';
@override
String get mealTypes => 'Mahlzeittypen';
@override
String get formulaNote => 'Berechnet mit der Mifflin-St Jeor Formel';
@override
String get language => 'Sprache';
@override
String get notSet => 'Nicht festgelegt';
@override
String get calorieHint =>
'Körperparameter eingeben, um das Kalorienziel zu berechnen';
@override
String get logout => 'Abmelden';
@override
String get editProfile => 'Profil bearbeiten';
@override
String get cancel => 'Abbrechen';
@override
String get save => 'Speichern';
@override
String get nameLabel => 'Name';
@override
String get heightCm => 'Größe (cm)';
@override
String get weightKg => 'Gewicht (kg)';
@override
String get birthDate => 'Geburtsdatum';
@override
String get nameRequired => 'Name eingeben';
@override
String get profileUpdated => 'Profil aktualisiert';
@override
String get profileSaveFailed => 'Speichern fehlgeschlagen';
@override
String get mealTypeBreakfast => 'Frühstück';
@override
String get mealTypeSecondBreakfast => 'Zweites Frühstück';
@override
String get mealTypeLunch => 'Mittagessen';
@override
String get mealTypeAfternoonSnack => 'Nachmittagssnack';
@override
String get mealTypeDinner => 'Abendessen';
@override
String get mealTypeSnack => 'Snack';
@override
String get navHome => 'Startseite';
@override
String get navProducts => 'Produkte';
@override
String get navRecipes => 'Rezepte';
@override
String get addFromReceiptOrPhoto => 'Aus Kassenbon oder Foto hinzufügen';
@override
String get chooseMethod => 'Methode wählen';
@override
String get photoReceipt => 'Kassenbon fotografieren';
@override
String get photoReceiptSubtitle => 'Alle Produkte vom Kassenbon erkennen';
@override
String get photoProducts => 'Produkte fotografieren';
@override
String get photoProductsSubtitle =>
'Kühlschrank, Tisch, Regal — bis zu 3 Fotos';
@override
String get addPackagedFood => 'Verpacktes Lebensmittel hinzufügen';
@override
String get scanBarcode => 'Barcode scannen';
@override
String get portionWeightG => 'Portionsgewicht (g)';
@override
String get productNotFound => 'Produkt nicht gefunden';
@override
String get enterManually => 'Manuell eingeben';
@override
String get perHundredG => 'pro 100 g';
@override
String get searchFoodHint => 'Produkte und Gerichte suchen...';
@override
String get recentlyUsedLabel => 'Zuletzt verwendet';
@override
String get productsSection => 'Produkte';
@override
String get dishesSection => 'Gerichte';
@override
String noResultsForQuery(String query) {
return 'Keine Ergebnisse für \"$query\"';
}
@override
String get servingsLabel => 'Portionen';
@override
String get addToDiary => 'Zum Tagebuch hinzufügen';
@override
String get scanDishPhoto => 'Foto scannen';
@override
String planningForDate(String date) {
return '';
}
@override
String get markAsEaten => 'Als gegessen markieren';
@override
String get plannedMealLabel => 'Geplant';
@override
String get generateWeekLabel => 'Woche planen';
@override
String get generateWeekSubtitle =>
'KI erstellt einen Menüplan mit Frühstück, Mittagessen und Abendessen für die ganze Woche';
@override
String get generatingMenu => 'Menü wird erstellt...';
@override
String get dayPlannedLabel => 'Tag geplant';
@override
String get planMenuButton => 'Mahlzeiten planen';
@override
String get planMenuTitle => 'Was planen?';
@override
String get planOptionSingleMeal => 'Einzelne Mahlzeit';
@override
String get planOptionSingleMealDesc => 'Tag und Mahlzeittyp wählen';
@override
String get planOptionDay => 'Ein Tag';
@override
String get planOptionDayDesc => 'Alle Mahlzeiten für einen Tag';
@override
String get planOptionDays => 'Mehrere Tage';
@override
String get planOptionDaysDesc => 'Zeitraum anpassen';
@override
String get planOptionWeek => 'Eine Woche';
@override
String get planOptionWeekDesc => '7 Tage auf einmal';
@override
String get planSelectDate => 'Datum wählen';
@override
String get planSelectMealType => 'Mahlzeittyp';
@override
String get planSelectRange => 'Zeitraum wählen';
@override
String get planGenerateButton => 'Planen';
@override
String get planGenerating => 'Plan wird erstellt…';
@override
String get planSuccess => 'Menü geplant!';
@override
String get planProductsTitle => 'Zutaten für den Speiseplan';
@override
String get planProductsSubtitle =>
'Die KI berücksichtigt die ausgewählten Produkte bei der Rezeptgenerierung';
@override
String get planProductsEmpty => 'Keine Produkte hinzugefügt';
@override
String get planProductsEmptyMessage =>
'Füge Produkte hinzu, die du zu Hause hast — die KI schlägt Rezepte aus deinen Vorräten vor';
@override
String get planProductsAddProducts => 'Produkte hinzufügen';
@override
String get planProductsContinue => 'Weiter';
@override
String get planProductsSkip => 'Produktauswahl überspringen';
@override
String get planProductsSkipNoProducts => 'Ohne Produkte planen';
@override
String get planProductsSelectAll => 'Alle auswählen';
@override
String get planProductsDeselectAll => 'Alle abwählen';
@override
String get recentScans => 'Letzte Scans';
@override
String get seeAllScans => 'Alle';
@override
String get productJobHistoryTitle => 'Scan-Verlauf';
@override
String get jobTypeReceipt => 'Kassenbon';
@override
String get jobTypeProducts => 'Produkte';
@override
String get scanSubmitting => 'Wird gesendet...';
@override
String get processingProducts => 'Verarbeitung...';
}