mirror of
https://github.com/kuhyx/diet-guard.git
synced 2026-07-04 13:23:11 +02:00
Milestone 1 of the diet-app-as-wise-balloon plan: a phone-native way to log meals away from the PC, sharing the exact on-disk JSON shape diet_guard already uses (same field names, no translation layer). - lib/models/: 1:1 Dart mirrors of the Python dataclasses (Nutrition, FoodEntry, MealItem, FoodBankRecord, Slot), including the per-100g/ amount-eaten portion scaling that matches _resolve.resolve_nutrition's semantics exactly. - lib/services/log_storage_service.dart: plain-JSON persistence to food_log.json's exact shape (no sqflite -- the canonical format already is this JSON). - lib/services/foodbank_service.dart: ports _foodbank.py's upsert/fuzzy search logic for autocomplete. - lib/screens/: log_meal_screen.dart (single-item logging) and meal_builder_screen.dart (composite multi-item meals, logging full per-component macros via the new components field). Verified end-to-end on a physical device (BL9000): built, installed, logged a real meal through the UI. 77 Flutter tests passing, `flutter analyze` clean against very_good_analysis. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com> Claude-Session: https://claude.ai/code/session_01FU3f5KQ1GHXsbbSecfVEyF
61 lines
1.8 KiB
Dart
61 lines
1.8 KiB
Dart
import 'package:diet_guard_app/models/meal_item.dart';
|
|
import 'package:diet_guard_app/models/nutrition.dart';
|
|
import 'package:flutter_test/flutter_test.dart';
|
|
|
|
Nutrition _n(double kcal, double protein, double carbs, double fat, double g) =>
|
|
Nutrition(
|
|
kcal: kcal,
|
|
proteinG: protein,
|
|
carbsG: carbs,
|
|
fatG: fat,
|
|
grams: g,
|
|
source: 'manual',
|
|
);
|
|
|
|
void main() {
|
|
group('mealTotal', () {
|
|
test('sums every macro and the portion weight across items', () {
|
|
final items = [
|
|
MealItem(name: 'soup', nutrition: _n(100, 5, 10, 2, 200)),
|
|
MealItem(name: 'chicken', nutrition: _n(250, 30, 0, 10, 150)),
|
|
];
|
|
final total = mealTotal(items);
|
|
expect(total.kcal, 350);
|
|
expect(total.proteinG, 35);
|
|
expect(total.carbsG, 10);
|
|
expect(total.fatG, 12);
|
|
expect(total.grams, 350);
|
|
expect(total.source, mealSource);
|
|
});
|
|
|
|
test('returns all zeros for an empty meal', () {
|
|
final total = mealTotal(const []);
|
|
expect(total.kcal, 0);
|
|
expect(total.grams, 0);
|
|
expect(total.source, mealSource);
|
|
});
|
|
|
|
test('rounds the summed values to 0.1', () {
|
|
final items = [
|
|
MealItem(name: 'a', nutrition: _n(1.05, 1.05, 1.05, 1.05, 1.05)),
|
|
MealItem(name: 'b', nutrition: _n(1.05, 1.05, 1.05, 1.05, 1.05)),
|
|
];
|
|
final total = mealTotal(items);
|
|
expect(total.kcal, 2.1);
|
|
});
|
|
});
|
|
|
|
group('itemToComponent', () {
|
|
test('carries the item\'s name and full macros', () {
|
|
final item = MealItem(name: 'rice', nutrition: _n(200, 4, 44, 1, 150));
|
|
final component = itemToComponent(item);
|
|
expect(component.name, 'rice');
|
|
expect(component.kcal, 200);
|
|
expect(component.proteinG, 4);
|
|
expect(component.carbsG, 44);
|
|
expect(component.fatG, 1);
|
|
expect(component.grams, 150);
|
|
});
|
|
});
|
|
}
|