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>
177 lines
9.4 KiB
Plaintext
177 lines
9.4 KiB
Plaintext
{
|
|
"@@locale": "hi",
|
|
"appTitle": "FoodAI",
|
|
"greetingMorning": "सुप्रभात",
|
|
"greetingAfternoon": "नमस्ते",
|
|
"greetingEvening": "शुभ संध्या",
|
|
"caloriesUnit": "कैलोरी",
|
|
"gramsUnit": "ग्रा",
|
|
"goalLabel": "लक्ष्य:",
|
|
"consumed": "सेवन किया",
|
|
"remaining": "शेष",
|
|
"exceeded": "अधिक",
|
|
"proteinLabel": "प्रोटीन",
|
|
"fatLabel": "वसा",
|
|
"carbsLabel": "कार्बोहाइड्रेट",
|
|
"today": "आज",
|
|
"yesterday": "कल",
|
|
"mealsSection": "भोजन",
|
|
"addDish": "व्यंजन जोड़ें",
|
|
"scanDish": "स्कैन करें",
|
|
"menu": "मेनू",
|
|
"dishHistory": "व्यंजन इतिहास",
|
|
"recommendCook": "पकाने की सिफारिश",
|
|
"camera": "कैमरा",
|
|
"gallery": "गैलरी",
|
|
"analyzingPhoto": "फ़ोटो का विश्लेषण हो रहा है...",
|
|
"inQueue": "आप कतार में हैं",
|
|
"queuePosition": "स्थिति {position}",
|
|
"@queuePosition": {
|
|
"placeholders": {
|
|
"position": {
|
|
"type": "int"
|
|
}
|
|
}
|
|
},
|
|
"processing": "प्रसंस्करण हो रहा है...",
|
|
"upgradePrompt": "कतार छोड़ें? अपग्रेड करें →",
|
|
"recognitionFailed": "पहचान विफल। पुनः प्रयास करें।",
|
|
"dishRecognition": "व्यंजन पहचान",
|
|
"all": "सभी",
|
|
"dishRecognized": "व्यंजन पहचाना गया",
|
|
"recognizing": "पहचान हो रही है…",
|
|
"recognitionError": "पहचान में त्रुटि",
|
|
"dishResultTitle": "व्यंजन पहचाना गया",
|
|
"selectDish": "व्यंजन चुनें",
|
|
"dishNotRecognized": "व्यंजन नहीं पहचाना",
|
|
"tryAgain": "पुनः प्रयास करें",
|
|
"nutritionApproximate": "पोषण मूल्य अनुमानित हैं — फ़ोटो से अनुमानित।",
|
|
"portion": "हिस्सा",
|
|
"mealType": "भोजन का प्रकार",
|
|
"dateLabel": "तिथि",
|
|
"addToJournal": "डायरी में जोड़ें",
|
|
"addFailed": "जोड़ने में विफल। पुनः प्रयास करें।",
|
|
"historyTitle": "पहचान इतिहास",
|
|
"historyLoadError": "इतिहास लोड करने में विफल",
|
|
"retry": "पुनः प्रयास",
|
|
"noHistory": "अभी तक कोई पहचान नहीं",
|
|
"profileTitle": "प्रोफ़ाइल",
|
|
"edit": "संपादित करें",
|
|
"bodyParams": "शरीर के पैरामीटर",
|
|
"goalActivity": "लक्ष्य और गतिविधि",
|
|
"nutrition": "पोषण",
|
|
"settings": "सेटिंग्स",
|
|
"height": "ऊंचाई",
|
|
"weight": "वज़न",
|
|
"age": "आयु",
|
|
"gender": "लिंग",
|
|
"genderMale": "पुरुष",
|
|
"genderFemale": "महिला",
|
|
"goalLoss": "वज़न घटाना",
|
|
"goalMaintain": "वज़न बनाए रखना",
|
|
"goalGain": "मांसपेशी बढ़ाना",
|
|
"activityLow": "कम",
|
|
"activityMedium": "मध्यम",
|
|
"activityHigh": "अधिक",
|
|
"calorieGoal": "कैलोरी लक्ष्य",
|
|
"mealTypes": "भोजन के प्रकार",
|
|
"formulaNote": "मिफ्लिन-सेंट जेओर सूत्र का उपयोग करके गणना",
|
|
"language": "भाषा",
|
|
"notSet": "सेट नहीं",
|
|
"calorieHint": "कैलोरी लक्ष्य की गणना के लिए शरीर के पैरामीटर दर्ज करें",
|
|
"logout": "लॉग आउट",
|
|
"editProfile": "प्रोफ़ाइल संपादित करें",
|
|
"cancel": "रद्द करें",
|
|
"save": "सहेजें",
|
|
"nameLabel": "नाम",
|
|
"heightCm": "ऊंचाई (सेमी)",
|
|
"weightKg": "वज़न (किग्रा)",
|
|
"birthDate": "जन्म तिथि",
|
|
"nameRequired": "नाम दर्ज करें",
|
|
"profileUpdated": "प्रोफ़ाइल अपडेट हुई",
|
|
"profileSaveFailed": "सहेजने में विफल",
|
|
"mealTypeBreakfast": "नाश्ता",
|
|
"mealTypeSecondBreakfast": "दूसरा नाश्ता",
|
|
"mealTypeLunch": "दोपहर का भोजन",
|
|
"mealTypeAfternoonSnack": "शाम का नाश्ता",
|
|
"mealTypeDinner": "रात का खाना",
|
|
"mealTypeSnack": "स्नैक",
|
|
"navHome": "होम",
|
|
"navProducts": "उत्पाद",
|
|
"navRecipes": "रेसिपी",
|
|
"addFromReceiptOrPhoto": "रसीद या फ़ोटो से जोड़ें",
|
|
"chooseMethod": "तरीका चुनें",
|
|
"photoReceipt": "रसीद की फ़ोटो",
|
|
"photoReceiptSubtitle": "रसीद से सभी उत्पाद पहचानें",
|
|
"photoProducts": "उत्पादों की फ़ोटो",
|
|
"photoProductsSubtitle": "फ्रिज, टेबल, शेल्फ — 3 फ़ोटो तक",
|
|
"addPackagedFood": "पैकेज्ड फूड जोड़ें",
|
|
"scanBarcode": "बारकोड स्कैन करें",
|
|
"portionWeightG": "हिस्से का वजन (ग्राम)",
|
|
"productNotFound": "उत्पाद नहीं मिला",
|
|
"enterManually": "मैन्युअल दर्ज करें",
|
|
"perHundredG": "प्रति 100 ग्राम",
|
|
"searchFoodHint": "उत्पाद और व्यंजन खोजें...",
|
|
"recentlyUsedLabel": "हाल ही में उपयोग किए गए",
|
|
"productsSection": "उत्पाद",
|
|
"dishesSection": "व्यंजन",
|
|
"noResultsForQuery": "\"{query}\" के लिए कुछ नहीं मिला",
|
|
"@noResultsForQuery": {
|
|
"placeholders": {
|
|
"query": {
|
|
"type": "String"
|
|
}
|
|
}
|
|
},
|
|
"servingsLabel": "सर्विंग",
|
|
"addToDiary": "डायरी में जोड़ें",
|
|
"scanDishPhoto": "फ़ोटो स्कैन करें",
|
|
"planningForDate": "",
|
|
"@planningForDate": {
|
|
"placeholders": {
|
|
"date": {
|
|
"type": "String"
|
|
}
|
|
}
|
|
},
|
|
"markAsEaten": "खाया हुआ चिह्नित करें",
|
|
"plannedMealLabel": "नियोजित",
|
|
"generateWeekLabel": "सप्ताह की योजना बनाएं",
|
|
"generateWeekSubtitle": "AI पूरे सप्ताह के लिए नाश्ता, दोपहर का खाना और रात के खाने के साथ मेनू बनाएगा",
|
|
"generatingMenu": "मेनू बना रहे हैं...",
|
|
"dayPlannedLabel": "दिन की योजना बनाई गई",
|
|
"planMenuButton": "भोजन की योजना बनाएं",
|
|
"planMenuTitle": "क्या योजना बनानी है?",
|
|
"planOptionSingleMeal": "एक भोजन",
|
|
"planOptionSingleMealDesc": "दिन और भोजन का प्रकार चुनें",
|
|
"planOptionDay": "एक दिन",
|
|
"planOptionDayDesc": "एक दिन के सभी भोजन",
|
|
"planOptionDays": "कई दिन",
|
|
"planOptionDaysDesc": "अवधि अनुकूलित करें",
|
|
"planOptionWeek": "एक सप्ताह",
|
|
"planOptionWeekDesc": "एक बार में 7 दिन",
|
|
"planSelectDate": "तारीख चुनें",
|
|
"planSelectMealType": "भोजन का प्रकार",
|
|
"planSelectRange": "अवधि चुनें",
|
|
"planGenerateButton": "योजना बनाएं",
|
|
"planGenerating": "योजना बना रहे हैं…",
|
|
"planSuccess": "मेनू की योजना बनाई गई!",
|
|
"planProductsTitle": "मेनू के लिए उत्पाद",
|
|
"planProductsSubtitle": "AI रेसिपी बनाते समय चुने हुए उत्पादों को ध्यान में रखेगा",
|
|
"planProductsEmpty": "कोई उत्पाद नहीं जोड़ा गया",
|
|
"planProductsEmptyMessage": "घर पर उपलब्ध उत्पाद जोड़ें — AI आपके पास पहले से मौजूद चीज़ों से रेसिपी सुझाएगा",
|
|
"planProductsAddProducts": "उत्पाद जोड़ें",
|
|
"planProductsContinue": "जारी रखें",
|
|
"planProductsSkip": "उत्पाद चयन छोड़ें",
|
|
"planProductsSkipNoProducts": "उत्पादों के बिना योजना बनाएं",
|
|
"planProductsSelectAll": "सभी चुनें",
|
|
"planProductsDeselectAll": "सभी हटाएं",
|
|
"recentScans": "हाल के स्कैन",
|
|
"seeAllScans": "सभी देखें",
|
|
"productJobHistoryTitle": "स्कैन इतिहास",
|
|
"jobTypeReceipt": "रसीद",
|
|
"jobTypeProducts": "उत्पाद",
|
|
"scanSubmitting": "सबमिट हो रहा है...",
|
|
"processingProducts": "प्रोसेस हो रहा है..."
|
|
}
|