- Rewrite receipt OCR prompt: completes truncated names, preserves fat% and flavour attributes, extracts weight/volume from line, infers typical package sizes for solid goods with quantity_confidence field - Add quantity_confidence to RecognizedItem, EnrichedItem, and ProductJobResultItem; propagate through item enricher and worker - Replace per-item create loop with single POST /user-products/batch call from RecognitionConfirmScreen - Rebuild RecognitionConfirmScreen: amber qty border for low quantity_confidence, tappable product name → catalog picker, sort items by confidence, full L10n (no hardcoded strings) - Add timestamps (HH:mm / d MMM HH:mm) to recent scan chips - Show close-app hint on ProductJobWatchScreen (queued + processing) - Refresh recentProductJobsProvider on watch screen init so new job appears without a manual pull-to-refresh - App-level WidgetsBindingObserver refreshes product and dish job lists on resume, fixing stale lists after background/foreground transitions - Add 9 new L10n keys across all 12 locales Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
579 lines
14 KiB
Dart
579 lines
14 KiB
Dart
// ignore: unused_import
|
|
import 'package:intl/intl.dart' as intl;
|
|
import 'app_localizations.dart';
|
|
|
|
// ignore_for_file: type=lint
|
|
|
|
/// The translations for Arabic (`ar`).
|
|
class AppLocalizationsAr extends AppLocalizations {
|
|
AppLocalizationsAr([String locale = 'ar']) : super(locale);
|
|
|
|
@override
|
|
String get appTitle => 'FoodAI';
|
|
|
|
@override
|
|
String get greetingMorning => 'صباح الخير';
|
|
|
|
@override
|
|
String get greetingAfternoon => 'مساء الخير';
|
|
|
|
@override
|
|
String get greetingEvening => 'مساء النور';
|
|
|
|
@override
|
|
String get caloriesUnit => 'سعرة';
|
|
|
|
@override
|
|
String get gramsUnit => 'غ';
|
|
|
|
@override
|
|
String get goalLabel => 'الهدف:';
|
|
|
|
@override
|
|
String get consumed => 'المستهلك';
|
|
|
|
@override
|
|
String get remaining => 'المتبقي';
|
|
|
|
@override
|
|
String get exceeded => 'تجاوز';
|
|
|
|
@override
|
|
String get proteinLabel => 'بروتين';
|
|
|
|
@override
|
|
String get fatLabel => 'دهون';
|
|
|
|
@override
|
|
String get carbsLabel => 'كربوهيدرات';
|
|
|
|
@override
|
|
String get today => 'اليوم';
|
|
|
|
@override
|
|
String get yesterday => 'أمس';
|
|
|
|
@override
|
|
String get mealsSection => 'الوجبات';
|
|
|
|
@override
|
|
String get addDish => 'إضافة طبق';
|
|
|
|
@override
|
|
String get scanDish => 'مسح';
|
|
|
|
@override
|
|
String get menu => 'القائمة';
|
|
|
|
@override
|
|
String get dishHistory => 'سجل الأطباق';
|
|
|
|
@override
|
|
String get recommendCook => 'نوصي بطهي';
|
|
|
|
@override
|
|
String get camera => 'الكاميرا';
|
|
|
|
@override
|
|
String get gallery => 'المعرض';
|
|
|
|
@override
|
|
String get analyzingPhoto => 'تحليل الصورة...';
|
|
|
|
@override
|
|
String get inQueue => 'أنت في قائمة الانتظار';
|
|
|
|
@override
|
|
String queuePosition(int position) {
|
|
return 'الموضع $position';
|
|
}
|
|
|
|
@override
|
|
String get processing => 'جارٍ المعالجة...';
|
|
|
|
@override
|
|
String get upgradePrompt => 'تخطي قائمة الانتظار؟ ترقية →';
|
|
|
|
@override
|
|
String get recognitionFailed => 'فشل التعرف. حاول مرة أخرى.';
|
|
|
|
@override
|
|
String get dishRecognition => 'التعرف على الأطباق';
|
|
|
|
@override
|
|
String get all => 'الكل';
|
|
|
|
@override
|
|
String get dishRecognized => 'تم التعرف على الطبق';
|
|
|
|
@override
|
|
String get recognizing => 'جارٍ التعرف…';
|
|
|
|
@override
|
|
String get recognitionError => 'خطأ في التعرف';
|
|
|
|
@override
|
|
String get dishResultTitle => 'تم التعرف على الطبق';
|
|
|
|
@override
|
|
String get selectDish => 'اختر طبقًا';
|
|
|
|
@override
|
|
String get dishNotRecognized => 'لم يتم التعرف على الطبق';
|
|
|
|
@override
|
|
String get tryAgain => 'حاول مرة أخرى';
|
|
|
|
@override
|
|
String get nutritionApproximate =>
|
|
'القيم الغذائية تقريبية — مقدَّرة من الصورة.';
|
|
|
|
@override
|
|
String get portion => 'الحصة';
|
|
|
|
@override
|
|
String get mealType => 'نوع الوجبة';
|
|
|
|
@override
|
|
String get dateLabel => 'التاريخ';
|
|
|
|
@override
|
|
String get addToJournal => 'إضافة إلى السجل';
|
|
|
|
@override
|
|
String get addFailed => 'فشل الإضافة. حاول مرة أخرى.';
|
|
|
|
@override
|
|
String get historyTitle => 'سجل التعرف';
|
|
|
|
@override
|
|
String get historyLoadError => 'فشل تحميل السجل';
|
|
|
|
@override
|
|
String get retry => 'إعادة المحاولة';
|
|
|
|
@override
|
|
String get noHistory => 'لا توجد تعرفات بعد';
|
|
|
|
@override
|
|
String get profileTitle => 'الملف الشخصي';
|
|
|
|
@override
|
|
String get edit => 'تعديل';
|
|
|
|
@override
|
|
String get bodyParams => 'معاملات الجسم';
|
|
|
|
@override
|
|
String get goalActivity => 'الهدف والنشاط';
|
|
|
|
@override
|
|
String get nutrition => 'التغذية';
|
|
|
|
@override
|
|
String get settings => 'الإعدادات';
|
|
|
|
@override
|
|
String get height => 'الطول';
|
|
|
|
@override
|
|
String get weight => 'الوزن';
|
|
|
|
@override
|
|
String get age => 'العمر';
|
|
|
|
@override
|
|
String get gender => 'الجنس';
|
|
|
|
@override
|
|
String get genderMale => 'ذكر';
|
|
|
|
@override
|
|
String get genderFemale => 'أنثى';
|
|
|
|
@override
|
|
String get goalLoss => 'خسارة الوزن';
|
|
|
|
@override
|
|
String get goalMaintain => 'الحفاظ على الوزن';
|
|
|
|
@override
|
|
String get goalGain => 'بناء العضلات';
|
|
|
|
@override
|
|
String get activityLow => 'منخفض';
|
|
|
|
@override
|
|
String get activityMedium => 'متوسط';
|
|
|
|
@override
|
|
String get activityHigh => 'مرتفع';
|
|
|
|
@override
|
|
String get calorieGoal => 'هدف السعرات';
|
|
|
|
@override
|
|
String get mealTypes => 'أنواع الوجبات';
|
|
|
|
@override
|
|
String get formulaNote => 'محسوب بمعادلة ميفلين سانت جيور';
|
|
|
|
@override
|
|
String get language => 'اللغة';
|
|
|
|
@override
|
|
String get notSet => 'غير محدد';
|
|
|
|
@override
|
|
String get calorieHint => 'أدخل معاملات الجسم لحساب هدف السعرات';
|
|
|
|
@override
|
|
String get logout => 'تسجيل الخروج';
|
|
|
|
@override
|
|
String get editProfile => 'تعديل الملف الشخصي';
|
|
|
|
@override
|
|
String get cancel => 'إلغاء';
|
|
|
|
@override
|
|
String get save => 'حفظ';
|
|
|
|
@override
|
|
String get nameLabel => 'الاسم';
|
|
|
|
@override
|
|
String get heightCm => 'الطول (سم)';
|
|
|
|
@override
|
|
String get weightKg => 'الوزن (كغ)';
|
|
|
|
@override
|
|
String get birthDate => 'تاريخ الميلاد';
|
|
|
|
@override
|
|
String get nameRequired => 'أدخل الاسم';
|
|
|
|
@override
|
|
String get profileUpdated => 'تم تحديث الملف الشخصي';
|
|
|
|
@override
|
|
String get profileSaveFailed => 'فشل الحفظ';
|
|
|
|
@override
|
|
String get mealTypeBreakfast => 'الإفطار';
|
|
|
|
@override
|
|
String get mealTypeSecondBreakfast => 'الإفطار الثاني';
|
|
|
|
@override
|
|
String get mealTypeLunch => 'الغداء';
|
|
|
|
@override
|
|
String get mealTypeAfternoonSnack => 'وجبة العصر';
|
|
|
|
@override
|
|
String get mealTypeDinner => 'العشاء';
|
|
|
|
@override
|
|
String get mealTypeSnack => 'وجبة خفيفة';
|
|
|
|
@override
|
|
String get navHome => 'الرئيسية';
|
|
|
|
@override
|
|
String get navProducts => 'المنتجات';
|
|
|
|
@override
|
|
String get navRecipes => 'الوصفات';
|
|
|
|
@override
|
|
String get addFromReceiptOrPhoto => 'إضافة من الإيصال أو الصورة';
|
|
|
|
@override
|
|
String get scanScreenTitle => 'المسح والتعرف';
|
|
|
|
@override
|
|
String get barcodeScanSubtitle => 'ابحث عن منتج بالباركود';
|
|
|
|
@override
|
|
String get chooseMethod => 'اختر الطريقة';
|
|
|
|
@override
|
|
String get photoReceipt => 'تصوير الإيصال';
|
|
|
|
@override
|
|
String get photoReceiptSubtitle => 'التعرف على جميع المنتجات من الإيصال';
|
|
|
|
@override
|
|
String get photoProducts => 'تصوير المنتجات';
|
|
|
|
@override
|
|
String get photoProductsSubtitle => 'الثلاجة، الطاولة، الرف — حتى 3 صور';
|
|
|
|
@override
|
|
String get addPackagedFood => 'إضافة منتج معبأ';
|
|
|
|
@override
|
|
String get scanBarcode => 'مسح الباركود';
|
|
|
|
@override
|
|
String get portionWeightG => 'وزن الحصة (جم)';
|
|
|
|
@override
|
|
String get productNotFound => 'المنتج غير موجود';
|
|
|
|
@override
|
|
String get enterManually => 'أدخل يدوياً';
|
|
|
|
@override
|
|
String get perHundredG => 'لكل 100 جم';
|
|
|
|
@override
|
|
String get searchFoodHint => 'البحث عن المنتجات والأطباق...';
|
|
|
|
@override
|
|
String get recentlyUsedLabel => 'المستخدمة مؤخراً';
|
|
|
|
@override
|
|
String get productsSection => 'المنتجات';
|
|
|
|
@override
|
|
String get dishesSection => 'الأطباق';
|
|
|
|
@override
|
|
String noResultsForQuery(String query) {
|
|
return 'لم يتم العثور على نتائج لـ \"$query\"';
|
|
}
|
|
|
|
@override
|
|
String get servingsLabel => 'حصص';
|
|
|
|
@override
|
|
String get addToDiary => 'إضافة إلى اليومية';
|
|
|
|
@override
|
|
String get scanDishPhoto => 'مسح الصورة';
|
|
|
|
@override
|
|
String planningForDate(String date) {
|
|
return '';
|
|
}
|
|
|
|
@override
|
|
String get markAsEaten => 'وضع علامة كمأكول';
|
|
|
|
@override
|
|
String get plannedMealLabel => 'مخطط';
|
|
|
|
@override
|
|
String get generateWeekLabel => 'تخطيط الأسبوع';
|
|
|
|
@override
|
|
String get generateWeekSubtitle =>
|
|
'سيقوم الذكاء الاصطناعي بإنشاء قائمة طعام تشمل الإفطار والغداء والعشاء لكامل الأسبوع';
|
|
|
|
@override
|
|
String get generatingMenu => 'جارٍ إنشاء القائمة...';
|
|
|
|
@override
|
|
String get dayPlannedLabel => 'تم تخطيط اليوم';
|
|
|
|
@override
|
|
String get planMenuButton => 'تخطيط الوجبات';
|
|
|
|
@override
|
|
String get planMenuTitle => 'ماذا تريد تخطيطه؟';
|
|
|
|
@override
|
|
String get planOptionSingleMeal => 'وجبة واحدة';
|
|
|
|
@override
|
|
String get planOptionSingleMealDesc => 'اختر اليوم ونوع الوجبة';
|
|
|
|
@override
|
|
String get planOptionDay => 'يوم واحد';
|
|
|
|
@override
|
|
String get planOptionDayDesc => 'جميع وجبات اليوم';
|
|
|
|
@override
|
|
String get planOptionDays => 'عدة أيام';
|
|
|
|
@override
|
|
String get planOptionDaysDesc => 'تخصيص الفترة';
|
|
|
|
@override
|
|
String get planOptionWeek => 'أسبوع';
|
|
|
|
@override
|
|
String get planOptionWeekDesc => '7 أيام دفعة واحدة';
|
|
|
|
@override
|
|
String get planSelectDate => 'اختر التاريخ';
|
|
|
|
@override
|
|
String get planSelectMealType => 'نوع الوجبة';
|
|
|
|
@override
|
|
String get planSelectRange => 'اختر الفترة';
|
|
|
|
@override
|
|
String get planGenerateButton => 'تخطيط';
|
|
|
|
@override
|
|
String get planGenerating => 'جارٍ إنشاء الخطة…';
|
|
|
|
@override
|
|
String get planSuccess => 'تم تخطيط القائمة!';
|
|
|
|
@override
|
|
String get planProductsTitle => 'مكونات القائمة';
|
|
|
|
@override
|
|
String get planProductsSubtitle =>
|
|
'سيأخذ الذكاء الاصطناعي المكونات المختارة بعين الاعتبار عند إنشاء الوصفات';
|
|
|
|
@override
|
|
String get planProductsEmpty => 'لم يتم إضافة منتجات';
|
|
|
|
@override
|
|
String get planProductsEmptyMessage =>
|
|
'أضف المنتجات الموجودة لديك في المنزل — سيقترح الذكاء الاصطناعي وصفات مما لديك بالفعل';
|
|
|
|
@override
|
|
String get planProductsAddProducts => 'إضافة منتجات';
|
|
|
|
@override
|
|
String get planProductsContinue => 'متابعة';
|
|
|
|
@override
|
|
String get planProductsSkip => 'تخطي اختيار المنتجات';
|
|
|
|
@override
|
|
String get planProductsSkipNoProducts => 'التخطيط بدون منتجات';
|
|
|
|
@override
|
|
String get planProductsSelectAll => 'تحديد الكل';
|
|
|
|
@override
|
|
String get planProductsDeselectAll => 'إلغاء تحديد الكل';
|
|
|
|
@override
|
|
String get recentScans => 'عمليات المسح الأخيرة';
|
|
|
|
@override
|
|
String get seeAllScans => 'عرض الكل';
|
|
|
|
@override
|
|
String get productJobHistoryTitle => 'سجل المسح';
|
|
|
|
@override
|
|
String get jobTypeReceipt => 'إيصال';
|
|
|
|
@override
|
|
String get jobTypeProducts => 'منتجات';
|
|
|
|
@override
|
|
String get scanSubmitting => 'جارٍ الإرسال...';
|
|
|
|
@override
|
|
String get processingProducts => 'جارٍ المعالجة...';
|
|
|
|
@override
|
|
String get clearAllProducts => 'مسح الكل';
|
|
|
|
@override
|
|
String get clearAllConfirmTitle => 'مسح جميع المنتجات؟';
|
|
|
|
@override
|
|
String get clearAllConfirmMessage => 'سيتم حذف جميع المنتجات نهائياً.';
|
|
|
|
@override
|
|
String get addManually => 'يدويًا';
|
|
|
|
@override
|
|
String get scan => 'مسح ضوئي';
|
|
|
|
@override
|
|
String get addProduct => 'إضافة';
|
|
|
|
@override
|
|
String get searchProducts => 'البحث عن المنتجات';
|
|
|
|
@override
|
|
String get searchProductsHint => 'اكتب اسم المنتج أو أضف يدويًا';
|
|
|
|
@override
|
|
String noSearchResults(String query) {
|
|
return 'لا توجد نتائج لـ\"$query\"';
|
|
}
|
|
|
|
@override
|
|
String get quantity => 'الكمية';
|
|
|
|
@override
|
|
String get storageDays => 'أيام التخزين';
|
|
|
|
@override
|
|
String get addToShelf => 'إضافة إلى المخزن';
|
|
|
|
@override
|
|
String get errorGeneric => 'حدث خطأ ما';
|
|
|
|
@override
|
|
String get nutritionOptional => 'القيم الغذائية لكل 100 جم (اختياري)';
|
|
|
|
@override
|
|
String get calories => 'سعرات حرارية';
|
|
|
|
@override
|
|
String get protein => 'بروتين';
|
|
|
|
@override
|
|
String get fat => 'دهون';
|
|
|
|
@override
|
|
String get carbs => 'كربوهيدرات';
|
|
|
|
@override
|
|
String get fiber => 'ألياف';
|
|
|
|
@override
|
|
String get productAddedToShelf => 'تمت الإضافة إلى المخزن';
|
|
|
|
@override
|
|
String recognitionFoundProducts(int count) {
|
|
return 'تم العثور على $count منتجات';
|
|
}
|
|
|
|
@override
|
|
String get recognitionAddAll => 'إضافة الكل';
|
|
|
|
@override
|
|
String get recognitionAddToStock => 'إضافة إلى المخزن';
|
|
|
|
@override
|
|
String recognitionAdded(int count) {
|
|
return 'تمت إضافة $count منتجات';
|
|
}
|
|
|
|
@override
|
|
String get recognitionProductsFailed => 'فشل إضافة المنتجات';
|
|
|
|
@override
|
|
String get recognitionEmpty => 'لم يتم العثور على منتجات';
|
|
|
|
@override
|
|
String recognitionConfidence(int percent) {
|
|
return '$percent% ثقة';
|
|
}
|
|
|
|
@override
|
|
String get recognitionReplaceProduct => 'استبدال المنتج';
|
|
|
|
@override
|
|
String get scanJobCloseHint =>
|
|
'يمكنك إغلاق التطبيق — سيظهر هذا المسح في عمليات المسح الأخيرة في شاشة المنتجات';
|
|
}
|