diet-guard/app/lib/widgets/macro_input_row.dart
Krzysztof kuhy Rudnicki ee5a7660cb Add Flutter companion app skeleton with local meal logging
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
2026-06-22 18:22:42 +02:00

125 lines
3.5 KiB
Dart

/// A row of macro entry fields (kcal/protein/carbs/fat/grams), with an
/// optional reference weight so a label's per-100g macros can be typed
/// directly and scaled to the amount actually eaten.
library;
import 'package:flutter/material.dart';
/// Text controllers for one macro-entry row, owned by the calling screen so
/// it can read/clear/prefill values around the row's lifecycle.
class MacroControllers {
/// Creates a fresh set of empty macro controllers.
MacroControllers()
: kcal = TextEditingController(),
protein = TextEditingController(),
carbs = TextEditingController(),
fat = TextEditingController(),
perGrams = TextEditingController(),
grams = TextEditingController();
/// Calories controller.
final TextEditingController kcal;
/// Protein (g) controller.
final TextEditingController protein;
/// Carbohydrate (g) controller.
final TextEditingController carbs;
/// Fat (g) controller.
final TextEditingController fat;
/// Reference weight (g) the typed macros are stated for, e.g. `100` for
/// a per-100g label. Blank means the macros already describe the full
/// eaten portion.
final TextEditingController perGrams;
/// Portion weight actually eaten (g). Blank assumes the eaten amount
/// equals [perGrams].
final TextEditingController grams;
/// Clears every field's text.
void clear() {
kcal.clear();
protein.clear();
carbs.clear();
fat.clear();
perGrams.clear();
grams.clear();
}
/// Disposes every controller.
void dispose() {
kcal.dispose();
protein.dispose();
carbs.dispose();
fat.dispose();
perGrams.dispose();
grams.dispose();
}
}
/// A labeled row of number-entry fields for calories, macros, and the
/// optional reference-weight-vs-eaten-weight split.
class MacroInputRow extends StatelessWidget {
/// Creates a [MacroInputRow] bound to [controllers].
const MacroInputRow({required this.controllers, super.key});
/// The text controllers this row reads from and writes to.
final MacroControllers controllers;
@override
Widget build(BuildContext context) {
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
children: [
Expanded(child: _macroField('kcal', controllers.kcal)),
const SizedBox(width: 8),
Expanded(
child: _macroField(
'macros per (g)',
controllers.perGrams,
helperText: 'e.g. 100 for a per-100g label',
),
),
],
),
const SizedBox(height: 8),
Row(
children: [
Expanded(child: _macroField('protein g', controllers.protein)),
const SizedBox(width: 8),
Expanded(child: _macroField('carbs g', controllers.carbs)),
const SizedBox(width: 8),
Expanded(child: _macroField('fat g', controllers.fat)),
],
),
const SizedBox(height: 8),
_macroField(
'amount eaten (g)',
controllers.grams,
helperText: "blank = same as 'macros per'",
),
],
);
}
Widget _macroField(
String label,
TextEditingController controller, {
String? helperText,
}) {
return TextField(
controller: controller,
keyboardType: const TextInputType.numberWithOptions(decimal: true),
decoration: InputDecoration(
labelText: label,
helperText: helperText,
isDense: true,
),
);
}
}