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
52 lines
1.4 KiB
Dart
52 lines
1.4 KiB
Dart
/// Shows today's 08:00/12:00/16:00/20:00 slot status.
|
|
library;
|
|
|
|
import 'package:diet_guard_app/models/slot.dart';
|
|
import 'package:flutter/material.dart';
|
|
|
|
/// Renders each of today's meal slots as logged / due / upcoming.
|
|
class SlotStatusBar extends StatelessWidget {
|
|
/// Creates a [SlotStatusBar] for [now] given [loggedSlots] satisfied so
|
|
/// far today.
|
|
const SlotStatusBar({
|
|
required this.now,
|
|
required this.loggedSlots,
|
|
super.key,
|
|
});
|
|
|
|
/// Reference time used to decide which slots have elapsed.
|
|
final DateTime now;
|
|
|
|
/// Slot hours already satisfied by today's log.
|
|
final Set<int> loggedSlots;
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
final elapsed = elapsedSlots(now).toSet();
|
|
return Wrap(
|
|
spacing: 8,
|
|
runSpacing: 4,
|
|
children: daySlots().map((slot) {
|
|
final label = slotLabel(slot);
|
|
final String status;
|
|
final Color color;
|
|
if (loggedSlots.contains(slot)) {
|
|
status = 'logged';
|
|
color = Colors.green;
|
|
} else if (elapsed.contains(slot)) {
|
|
status = 'DUE';
|
|
color = Colors.red;
|
|
} else {
|
|
status = 'upcoming';
|
|
color = Colors.grey;
|
|
}
|
|
return Chip(
|
|
label: Text('$label $status'),
|
|
backgroundColor: color.withValues(alpha: 0.15),
|
|
labelStyle: TextStyle(color: color),
|
|
);
|
|
}).toList(),
|
|
);
|
|
}
|
|
}
|