Backend:
- migration 005: expand menu_items.meal_type CHECK to all 6 types (second_breakfast, afternoon_snack, snack)
- ai/types.go: add Days and MealTypes to MenuRequest for partial generation
- openai/menu.go: parametrize GenerateMenu — use requested meal types and day count; add caloric fractions for all 6 meal types
- menu/repository.go: add UpsertItemsInTx for partial upsert (preserves existing slots); fix meal_type sort order in GetByWeek
- menu/handler.go: add dates+meal_types path to POST /ai/generate-menu; extract fetchImages/saveRecipes helpers; returns {"plans":[...]} for dates mode; backward-compatible with week mode
Client:
- PlanMenuSheet: bottom sheet with 4 planning horizon options
- PlanDatePickerSheet: adaptive sheet with date strip (single day/meal) or custom CalendarRangePicker (multi-day/week); sliding 7-day window for week mode
- menu_service.dart: add generateForDates
- menu_provider.dart: add PlanMenuService (generates + invalidates week providers), lastPlannedDateProvider
- home_screen.dart: add _PlanMenuButton card below quick actions; opens planning wizard
- l10n: 16 new keys for planning UI across all 12 languages
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
157 lines
6.3 KiB
Plaintext
157 lines
6.3 KiB
Plaintext
{
|
|
"@@locale": "ar",
|
|
"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": "سيقوم الذكاء الاصطناعي بإنشاء قائمة طعام تشمل الإفطار والغداء والعشاء لكامل الأسبوع",
|
|
"generatingMenu": "جارٍ إنشاء القائمة...",
|
|
"weekPlannedLabel": "تم تخطيط الأسبوع",
|
|
|
|
"planMenuButton": "تخطيط الوجبات",
|
|
"planMenuTitle": "ماذا تريد تخطيطه؟",
|
|
"planOptionSingleMeal": "وجبة واحدة",
|
|
"planOptionSingleMealDesc": "اختر اليوم ونوع الوجبة",
|
|
"planOptionDay": "يوم واحد",
|
|
"planOptionDayDesc": "جميع وجبات اليوم",
|
|
"planOptionDays": "عدة أيام",
|
|
"planOptionDaysDesc": "تخصيص الفترة",
|
|
"planOptionWeek": "أسبوع",
|
|
"planOptionWeekDesc": "7 أيام دفعة واحدة",
|
|
"planSelectDate": "اختر التاريخ",
|
|
"planSelectMealType": "نوع الوجبة",
|
|
"planSelectRange": "اختر الفترة",
|
|
"planGenerateButton": "تخطيط",
|
|
"planGenerating": "جارٍ إنشاء الخطة\u2026",
|
|
"planSuccess": "تم تخطيط القائمة!"
|
|
}
|