import 'package:diameter/localization_keys.dart'; import 'package:diameter/screens/log/log_filter_dialog.dart'; import 'package:diameter/screens/reports/export.dart'; import 'package:diameter/utils/dialog_utils.dart'; import 'package:diameter/models/glucose_target.dart'; import 'package:diameter/models/log_bolus.dart'; import 'package:diameter/models/log_entry.dart'; import 'package:diameter/models/log_meal.dart'; import 'package:diameter/models/settings.dart'; import 'package:diameter/navigation.dart'; import 'package:diameter/screens/log/log_entry/log_entry.dart'; import 'package:diameter/utils/date_time_utils.dart'; import 'package:flutter/material.dart'; import 'dart:math' as math; import 'package:flutter_translate/flutter_translate.dart'; class LogScreen extends StatefulWidget { static const String routeName = '/log'; const LogScreen({Key? key}) : super(key: key); @override _LogScreenState createState() => _LogScreenState(); } class _LogScreenState extends State { late List _logEntries; final ScrollController _scrollController = ScrollController(); final TextEditingController _dateController = TextEditingController(text: ''); String? swipeDirection; late DateTime _date; @override void initState() { super.initState(); _date = DateTime.now(); _dateController.text = DateTimeUtils.displayDate(_date); reload(); } @override void dispose() { _scrollController.dispose(); _dateController.dispose(); super.dispose(); } void reload({String? message}) { setState(() { _logEntries = LogEntry.getAllForDate(_date); }); setState(() { if (message != null) { var snackBar = SnackBar( content: Text(message), duration: const Duration(seconds: 2), ); ScaffoldMessenger.of(context) ..removeCurrentSnackBar() ..showSnackBar(snackBar); } }); } void onDelete(LogEntry logEntry) { LogEntry.remove(logEntry.id); reload(message: translate(LocalizationKeys.log_deleted)); } void handleDeleteAction(LogEntry logEntry) async { if (Settings.get().showConfirmationDialogOnDelete) { DialogUtils.showConfirmationDialog( context: context, onConfirm: () => onDelete(logEntry), message: translate(LocalizationKeys.log_confirmDelete), ); } else { onDelete(logEntry); } } void onChangeDate(DateTime? date) { if (date != null) { setState(() { _date = DateTime(date.year, date.month, date.day); _dateController.text = DateTimeUtils.displayDate(date); }); reload(); } } void onChangeFilter( {DateTime? startDate, DateTime? endDate, int? minMgPerDl, int? maxMgPerDl, double? minMmolPerL, double? maxMmolPerL, int? mealId, String? mealName, String? note}) { setState(() { _logEntries = LogEntry.getAllByFilter( startDate: startDate, endDate: endDate, minMgPerDl: minMgPerDl, maxMgPerDl: maxMgPerDl, minMmolPerL: minMmolPerL, maxMmolPerL: maxMmolPerL, mealId: mealId, mealName: mealName, note: note, ); }); } @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar( title: Text(translate(LocalizationKeys.log_title)), actions: [ IconButton( onPressed: () => onChangeDate(DateTime.now()), icon: const Icon(Icons.today)), IconButton( onPressed: () => showDialog( context: context, builder: (context) => LogFilterDialog( date: _date, onApplyFilter: onChangeFilter, )), icon: const Icon(Icons.filter_list)), IconButton( onPressed: () => showDialog( context: context, builder: (context) => ExportDialog(date: _date)), icon: const Icon(Icons.file_download)), IconButton(onPressed: reload, icon: const Icon(Icons.refresh)), ], ), drawer: const Navigation(currentLocation: LogScreen.routeName), body: GestureDetector( onPanUpdate: (details) { swipeDirection = details.delta.dx < 0 ? 'left' : 'right'; }, onPanEnd: (details) { if (swipeDirection == null) { return; } if (swipeDirection == 'right' && !_date.isAtSameMomentAs(DateTime(2000, 1, 1))) { onChangeDate(_date.subtract(const Duration(days: 1))); } if (swipeDirection == 'left' && _date.add(const Duration(days: 1)).isBefore(DateTime.now())) { onChangeDate(_date.add(const Duration(days: 1))); } }, child: Column( children: [ Row( children: [ IconButton( onPressed: _date.isAtSameMomentAs(DateTime(2000, 1, 1)) ? null : () => onChangeDate(_date.subtract(const Duration(days: 1))), icon: const Icon(Icons.arrow_back), ), Expanded( child: GestureDetector( onTap: () async { final newTime = await showDatePicker( context: context, initialDate: _date, firstDate: DateTime(2000, 1, 1), lastDate: DateTime.now().add(const Duration(days: 365)), ); onChangeDate(newTime); }, child: Text( DateTimeUtils.displayDate(_date).toUpperCase(), style: Theme.of(context).textTheme.subtitle2, textAlign: TextAlign.center, ), ), ), IconButton( onPressed: _date .add(const Duration(days: 1)) .isBefore(DateTime.now()) ? () => onChangeDate(_date.add(const Duration(days: 1))) : null, icon: const Icon(Icons.arrow_forward), ), ], ), Expanded( child: _logEntries.isNotEmpty ? Scrollbar( controller: _scrollController, child: ListView.builder( controller: _scrollController, padding: const EdgeInsets.all(10.0), shrinkWrap: true, itemCount: _logEntries.length, itemBuilder: (context, index) { LogEntry logEntry = _logEntries[index]; double bolus = LogBolus.getTotalBolusForEntry(logEntry.id); double carbs = LogMeal.getTotalCarbsForEntry(logEntry.id); TextStyle glucoseStyle = TextStyle( color: GlucoseTarget.getColorForGlucose( mgPerDl: logEntry.mgPerDl ?? 0, mmolPerL: logEntry.mmolPerL ?? 0)); return Card( child: ListTile( onTap: () { Navigator.push( context, MaterialPageRoute( builder: (context) => LogEntryScreen(id: logEntry.id), ), ).then((result) => reload(message: result?[0])); }, title: Row( mainAxisAlignment: MainAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start, children: [ Expanded( child: Text( DateTimeUtils.displayTime(logEntry.time), ), ), Expanded( child: Column( children: logEntry.mgPerDl != null && (Settings.glucoseMeasurement == GlucoseMeasurement .mgPerDl || Settings.glucoseDisplayMode == GlucoseDisplayMode.both || Settings.glucoseDisplayMode == GlucoseDisplayMode .bothForList) ? [ Row( mainAxisAlignment: MainAxisAlignment.center, children: [ Text( logEntry.mgPerDl .toString(), style: glucoseStyle), logEntry.glucoseTrend != null ? Transform.rotate( angle: logEntry .glucoseTrend! * math.pi / 180, child: Icon( Icons.arrow_upward, color: glucoseStyle .color, size: 16.0, ), ) : Container(), ], ), const Text( 'mg/dl', textScaleFactor: 0.75, ), ] : [], ), ), Expanded( child: Column( children: logEntry.mmolPerL != null && (Settings.glucoseMeasurement == GlucoseMeasurement .mmolPerL || Settings.glucoseDisplayMode == GlucoseDisplayMode.both || Settings.glucoseDisplayMode == GlucoseDisplayMode .bothForList) ? [ Row( mainAxisAlignment: MainAxisAlignment.center, children: [ Text( logEntry.mmolPerL .toString(), style: glucoseStyle), logEntry.glucoseTrend != null ? Transform.rotate( angle: logEntry .glucoseTrend! * math.pi / 180, child: Icon( Icons.arrow_upward, color: glucoseStyle .color, size: 16.0, ), ) : Container(), ], ), const Text( 'mmol/l', textScaleFactor: 0.75, ), ] : [], ), ), Expanded( child: Column( children: (bolus > 0) ? [ Text( bolus.toStringAsPrecision(3)), Text(translate(LocalizationKeys.general_suffixes_units), textScaleFactor: 0.75), ] : [], ), ), Expanded( child: Column( children: (carbs > 0) ? [ Text( carbs.toStringAsPrecision(3)), Text(translate( LocalizationKeys.general_suffixes_carbs, args: { "nutritionMeasurementSuffix": Settings.nutritionMeasurementSuffix, } ), textScaleFactor: 0.75), ] : [], ), ), ], ), trailing: Row( mainAxisSize: MainAxisSize.min, children: [ IconButton( onPressed: () => handleDeleteAction(logEntry), icon: const Icon(Icons.delete, color: Colors.blue), ) ], ), ), ); }, ), ) : Center( child: Text( translate(LocalizationKeys.log_empty) ), ), ), ], ), ), floatingActionButton: FloatingActionButton( onPressed: () { Navigator.push( context, MaterialPageRoute( builder: (context) { return _date.isAtSameMomentAs(DateTimeUtils.today()) ? const LogEntryScreen() : LogEntryScreen( suggestedDate: _date, ); }, ), ).then((result) => reload(message: result?[0])); }, child: const Icon(Icons.add), ), ); } }