Files
food-ai/client/lib/l10n/app_it.arb
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

177 lines
6.2 KiB
Plaintext

{
"@@locale": "it",
"appTitle": "FoodAI",
"greetingMorning": "Buongiorno",
"greetingAfternoon": "Buon pomeriggio",
"greetingEvening": "Buonasera",
"caloriesUnit": "kcal",
"gramsUnit": "g",
"goalLabel": "obiettivo:",
"consumed": "Consumato",
"remaining": "Rimanente",
"exceeded": "Superato",
"proteinLabel": "Proteine",
"fatLabel": "Grassi",
"carbsLabel": "Carboidrati",
"today": "Oggi",
"yesterday": "Ieri",
"mealsSection": "Pasti",
"addDish": "Aggiungi piatto",
"scanDish": "Scansiona",
"menu": "Menu",
"dishHistory": "Cronologia piatti",
"recommendCook": "Consigliamo di cucinare",
"camera": "Fotocamera",
"gallery": "Galleria",
"analyzingPhoto": "Analisi foto in corso...",
"inQueue": "Sei in coda",
"queuePosition": "Posizione {position}",
"@queuePosition": {
"placeholders": {
"position": {
"type": "int"
}
}
},
"processing": "Elaborazione...",
"upgradePrompt": "Salta la coda? Aggiorna →",
"recognitionFailed": "Riconoscimento fallito. Riprova.",
"dishRecognition": "Riconoscimento piatti",
"all": "Tutti",
"dishRecognized": "Piatto riconosciuto",
"recognizing": "Riconoscimento in corso…",
"recognitionError": "Errore di riconoscimento",
"dishResultTitle": "Piatto riconosciuto",
"selectDish": "Seleziona un piatto",
"dishNotRecognized": "Piatto non riconosciuto",
"tryAgain": "Riprova",
"nutritionApproximate": "I valori nutrizionali sono approssimativi — stimati dalla foto.",
"portion": "Porzione",
"mealType": "Tipo di pasto",
"dateLabel": "Data",
"addToJournal": "Aggiungi al diario",
"addFailed": "Aggiunta fallita. Riprova.",
"historyTitle": "Cronologia riconoscimenti",
"historyLoadError": "Impossibile caricare la cronologia",
"retry": "Riprova",
"noHistory": "Nessun riconoscimento ancora",
"profileTitle": "Profilo",
"edit": "Modifica",
"bodyParams": "PARAMETRI CORPOREI",
"goalActivity": "OBIETTIVO & ATTIVITÀ",
"nutrition": "NUTRIZIONE",
"settings": "IMPOSTAZIONI",
"height": "Altezza",
"weight": "Peso",
"age": "Età",
"gender": "Sesso",
"genderMale": "Maschio",
"genderFemale": "Femmina",
"goalLoss": "Perdita di peso",
"goalMaintain": "Mantenimento",
"goalGain": "Aumento muscolare",
"activityLow": "Bassa",
"activityMedium": "Media",
"activityHigh": "Alta",
"calorieGoal": "Obiettivo calorico",
"mealTypes": "Tipi di pasto",
"formulaNote": "Calcolato con la formula di Mifflin-St Jeor",
"language": "Lingua",
"notSet": "Non impostato",
"calorieHint": "Inserisci i parametri corporei per calcolare l'obiettivo calorico",
"logout": "Disconnetti",
"editProfile": "Modifica profilo",
"cancel": "Annulla",
"save": "Salva",
"nameLabel": "Nome",
"heightCm": "Altezza (cm)",
"weightKg": "Peso (kg)",
"birthDate": "Data di nascita",
"nameRequired": "Inserisci il nome",
"profileUpdated": "Profilo aggiornato",
"profileSaveFailed": "Salvataggio fallito",
"mealTypeBreakfast": "Colazione",
"mealTypeSecondBreakfast": "Seconda colazione",
"mealTypeLunch": "Pranzo",
"mealTypeAfternoonSnack": "Merenda",
"mealTypeDinner": "Cena",
"mealTypeSnack": "Spuntino",
"navHome": "Home",
"navProducts": "Prodotti",
"navRecipes": "Ricette",
"addFromReceiptOrPhoto": "Aggiungi da scontrino o foto",
"chooseMethod": "Scegli il metodo",
"photoReceipt": "Fotografa scontrino",
"photoReceiptSubtitle": "Riconosciamo tutti i prodotti dallo scontrino",
"photoProducts": "Fotografa i prodotti",
"photoProductsSubtitle": "Frigo, tavolo, scaffale — fino a 3 foto",
"addPackagedFood": "Aggiungi alimento confezionato",
"scanBarcode": "Scansiona codice a barre",
"portionWeightG": "Peso della porzione (g)",
"productNotFound": "Prodotto non trovato",
"enterManually": "Inserisci manualmente",
"perHundredG": "per 100 g",
"searchFoodHint": "Cerca prodotti e piatti...",
"recentlyUsedLabel": "Usati di recente",
"productsSection": "Prodotti",
"dishesSection": "Piatti",
"noResultsForQuery": "Nessun risultato per \"{query}\"",
"@noResultsForQuery": {
"placeholders": {
"query": {
"type": "String"
}
}
},
"servingsLabel": "Porzioni",
"addToDiary": "Aggiungi al diario",
"scanDishPhoto": "Scansiona foto",
"planningForDate": "",
"@planningForDate": {
"placeholders": {
"date": {
"type": "String"
}
}
},
"markAsEaten": "Segna come mangiato",
"plannedMealLabel": "Pianificato",
"generateWeekLabel": "Pianifica la settimana",
"generateWeekSubtitle": "L'AI creerà un menu con colazione, pranzo e cena per tutta la settimana",
"generatingMenu": "Generazione menu...",
"dayPlannedLabel": "Giorno pianificato",
"planMenuButton": "Pianifica i pasti",
"planMenuTitle": "Cosa pianificare?",
"planOptionSingleMeal": "Un pasto",
"planOptionSingleMealDesc": "Scegli giorno e tipo di pasto",
"planOptionDay": "Un giorno",
"planOptionDayDesc": "Tutti i pasti di un giorno",
"planOptionDays": "Più giorni",
"planOptionDaysDesc": "Personalizza il periodo",
"planOptionWeek": "Una settimana",
"planOptionWeekDesc": "7 giorni in una volta",
"planSelectDate": "Seleziona data",
"planSelectMealType": "Tipo di pasto",
"planSelectRange": "Seleziona periodo",
"planGenerateButton": "Pianifica",
"planGenerating": "Generazione piano…",
"planSuccess": "Menu pianificato!",
"planProductsTitle": "Prodotti per il menu",
"planProductsSubtitle": "L'AI terrà conto dei prodotti selezionati nella generazione delle ricette",
"planProductsEmpty": "Nessun prodotto aggiunto",
"planProductsEmptyMessage": "Aggiungi prodotti che hai in casa — l'AI suggerirà ricette con quello che hai già",
"planProductsAddProducts": "Aggiungi prodotti",
"planProductsContinue": "Continua",
"planProductsSkip": "Salta la selezione dei prodotti",
"planProductsSkipNoProducts": "Pianifica senza prodotti",
"planProductsSelectAll": "Seleziona tutto",
"planProductsDeselectAll": "Deseleziona tutto",
"recentScans": "Scansioni recenti",
"seeAllScans": "Vedi tutto",
"productJobHistoryTitle": "Cronologia scansioni",
"jobTypeReceipt": "Scontrino",
"jobTypeProducts": "Prodotti",
"scanSubmitting": "Invio...",
"processingProducts": "Elaborazione..."
}