mirror of
https://github.com/kuhyx/diet-guard.git
synced 2026-07-04 15:23:16 +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
87 lines
2.3 KiB
Dart
87 lines
2.3 KiB
Dart
import 'package:diet_guard_app/models/slot.dart';
|
|
import 'package:flutter_test/flutter_test.dart';
|
|
|
|
void main() {
|
|
group('daySlots', () {
|
|
test('returns the four fixed hourly slots', () {
|
|
expect(daySlots(), [8, 12, 16, 20]);
|
|
});
|
|
});
|
|
|
|
group('withinEnforcementWindow', () {
|
|
test('false before the day start hour', () {
|
|
expect(withinEnforcementWindow(DateTime(2026, 6, 22, 7, 59)), isFalse);
|
|
});
|
|
|
|
test('true at the day start hour', () {
|
|
expect(withinEnforcementWindow(DateTime(2026, 6, 22, 8, 0)), isTrue);
|
|
});
|
|
|
|
test('true just before the eating end hour', () {
|
|
expect(withinEnforcementWindow(DateTime(2026, 6, 22, 21, 59)), isTrue);
|
|
});
|
|
|
|
test('false at the eating end hour (exclusive)', () {
|
|
expect(withinEnforcementWindow(DateTime(2026, 6, 22, 22, 0)), isFalse);
|
|
});
|
|
});
|
|
|
|
group('elapsedSlots', () {
|
|
test('empty outside the enforcement window', () {
|
|
expect(elapsedSlots(DateTime(2026, 6, 22, 23, 0)), isEmpty);
|
|
});
|
|
|
|
test('only the 8 slot right at day start', () {
|
|
expect(elapsedSlots(DateTime(2026, 6, 22, 8, 0)), [8]);
|
|
});
|
|
|
|
test('8 and 12 mid-afternoon before 16', () {
|
|
expect(elapsedSlots(DateTime(2026, 6, 22, 15, 59)), [8, 12]);
|
|
});
|
|
|
|
test('all four slots once 20:00 has passed', () {
|
|
expect(elapsedSlots(DateTime(2026, 6, 22, 21, 0)), [8, 12, 16, 20]);
|
|
});
|
|
});
|
|
|
|
group('missingSlots', () {
|
|
test('excludes already-logged elapsed slots', () {
|
|
expect(
|
|
missingSlots(DateTime(2026, 6, 22, 17, 0), {8}),
|
|
[12, 16],
|
|
);
|
|
});
|
|
|
|
test('empty once every elapsed slot is logged', () {
|
|
expect(
|
|
missingSlots(DateTime(2026, 6, 22, 17, 0), {8, 12, 16}),
|
|
isEmpty,
|
|
);
|
|
});
|
|
});
|
|
|
|
group('currentSlot', () {
|
|
test('null outside the enforcement window', () {
|
|
expect(currentSlot(DateTime(2026, 6, 22, 6, 0)), isNull);
|
|
});
|
|
|
|
test('returns the most recently elapsed slot', () {
|
|
expect(currentSlot(DateTime(2026, 6, 22, 17, 41)), 16);
|
|
});
|
|
|
|
test('returns 8 right at day start', () {
|
|
expect(currentSlot(DateTime(2026, 6, 22, 8, 0)), 8);
|
|
});
|
|
});
|
|
|
|
group('slotLabel', () {
|
|
test('pads single-digit hours', () {
|
|
expect(slotLabel(8), '08:00');
|
|
});
|
|
|
|
test('formats double-digit hours', () {
|
|
expect(slotLabel(20), '20:00');
|
|
});
|
|
});
|
|
}
|