Compare commits

..

No commits in common. "fixed-bottom-bar" and "master" have entirely different histories.

9 changed files with 185 additions and 262 deletions

View File

@ -1,7 +1,6 @@
import 'package:calorimeter/food_entry/food_entry_bloc.dart'; import 'package:calorimeter/food_entry/food_entry_bloc.dart';
import 'package:calorimeter/perdate/perdate_pageview_controller.dart'; import 'package:calorimeter/perdate/perdate_pageview.dart';
import 'package:calorimeter/storage/storage.dart'; import 'package:calorimeter/storage/storage.dart';
import 'package:calorimeter/utils/date_time_helper.dart';
import 'package:calorimeter/utils/settings_bloc.dart'; import 'package:calorimeter/utils/settings_bloc.dart';
import 'package:calorimeter/utils/theme_bloc.dart'; import 'package:calorimeter/utils/theme_bloc.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
@ -19,8 +18,14 @@ void main() async {
var storage = await FoodStorage.create(); var storage = await FoodStorage.create();
await storage.buildFoodLookupDatabase(); await storage.buildFoodLookupDatabase();
timeNow = DateTime.now().copyWith(
hour: 0,
isUtc: true,
minute: 0,
second: 0,
millisecond: 0,
microsecond: 0);
timeNow = DateTimeHelper.now();
entriesForToday = await storage.getEntriesForDate(timeNow); entriesForToday = await storage.getEntriesForDate(timeNow);
var kcalLimit = await storage.readLimit(); var kcalLimit = await storage.readLimit();
@ -82,8 +87,14 @@ class MainApp extends StatelessWidget {
GoRoute( GoRoute(
path: '/', path: '/',
builder: (context, state) { builder: (context, state) {
return PerDatePageViewController( return PerDatePageview(
initialDate: DateTimeHelper.now(), initalDate: DateTime.now().copyWith(
hour: 0,
minute: 0,
second: 0,
millisecond: 0,
microsecond: 0,
),
); );
}, },
), ),

View File

@ -1,50 +1,76 @@
import 'dart:developer';
import 'package:calorimeter/perdate/perdate_pageview_controller.dart';
import 'package:calorimeter/perdate/perdate_widget.dart'; import 'package:calorimeter/perdate/perdate_widget.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:provider/provider.dart';
class PerDatePageView extends StatelessWidget { class PerDatePageview extends StatefulWidget {
// this is the date for which the PerDate widget will be shown on screen // this is the date for which the PerDate widget will be shown on screen
// left of it will be yesterday's PerDate widget // left of it will be yesterday's PerDate widget
// right of it will be tomorrow's PerDate widget // right of it will be tomorrow's PerDate widget
final DateTime initialDate; final DateTime initalDate;
final PageController pageController; const PerDatePageview({required this.initalDate, super.key});
const PerDatePageView({ @override
required this.initialDate, State<PerDatePageview> createState() => _PerDatePageviewState();
required this.pageController, }
super.key,
}); class _PerDatePageviewState extends State<PerDatePageview> {
late PageController pageController;
late DateTime displayedDate;
late List<int> visitedIndexes = [];
final int initialOffset = 36500000;
//TODO: that is just ugly
bool backButtonWasPressed = false;
@override
void initState() {
super.initState();
pageController = PageController(initialPage: initialOffset);
displayedDate = widget.initalDate;
visitedIndexes.add(initialOffset);
}
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
log("PerDatePageView's build()"); return BackButtonListener(
return PageView.builder( onBackButtonPressed: () async {
reverse: true, if (visitedIndexes.length == 1) {
controller: pageController, return false;
onPageChanged: (value) { }
log("onPageChanged() with value $value");
var diff = value - pageController.initialPage; visitedIndexes.removeLast();
var newDate = initialDate.subtract(Duration(days: diff));
log("newDate = $newDate");
context backButtonWasPressed = true;
.read<PageViewStateProvider>() pageController.jumpToPage(visitedIndexes.last);
.addVisitedindexIfNotVisitedByBackButton(value);
context.read<PageViewStateProvider>().setDisplayedDate(newDate);
},
itemBuilder: (context, index) {
log("itemBuilder() called with index $index");
var dateToBuildWidgetFor = initialDate
.subtract(Duration(days: index - pageController.initialPage));
return PerDateWidget( return true;
key: ValueKey(dateToBuildWidgetFor.toString()), },
date: dateToBuildWidgetFor, child: PageView.builder(
); reverse: true,
}); controller: pageController,
onPageChanged: (value) {
if (backButtonWasPressed) {
backButtonWasPressed = false;
return;
}
visitedIndexes.add(value);
},
itemBuilder: (context, index) {
var dateToBuildWidgetFor =
displayedDate.subtract(Duration(days: index - initialOffset));
return PerDateWidget(
key: ValueKey(dateToBuildWidgetFor.toString()),
date: dateToBuildWidgetFor.copyWith(isUtc: true),
onDateSelected: (dateSelected) {
if (dateSelected == null) return;
var diff = dateSelected.difference(dateToBuildWidgetFor);
var newIndex = index - diff.inDays;
pageController.jumpToPage(newIndex);
});
}),
);
} }
} }

View File

@ -1,147 +0,0 @@
import 'dart:developer';
import 'package:barcode_scan2/barcode_scan2.dart';
import 'package:calorimeter/food_entry/food_entry_bloc.dart';
import 'package:calorimeter/perdate/perdate_pageview.dart';
import 'package:calorimeter/utils/app_drawer.dart';
import 'package:calorimeter/utils/calendar_floating_button.dart';
import 'package:calorimeter/utils/date_time_helper.dart';
import 'package:calorimeter/utils/rectangular_notch_shape.dart';
import 'package:calorimeter/utils/scan_food_floating_button.dart';
import 'package:calorimeter/utils/sum_widget.dart';
import 'package:calorimeter/utils/theme_switcher_button.dart';
import 'package:flutter/material.dart';
import 'package:intl/intl.dart';
import 'package:provider/provider.dart';
class PerDatePageViewController extends StatelessWidget {
// this is the date for which the PerDate widget will be shown on screen
// left of it will be yesterday's PerDate widget
// right of it will be tomorrow's PerDate widget
final DateTime initialDate;
final PageController pageController;
static final int initialOffset = 36500000;
const PerDatePageViewController._(
{required this.initialDate, required this.pageController});
factory PerDatePageViewController({required initialDate}) {
return PerDatePageViewController._(
initialDate: initialDate,
pageController: PageController(initialPage: initialOffset));
}
@override
Widget build(BuildContext context) {
return ChangeNotifierProvider(
create: (context) => PageViewStateProvider(
initialDate: initialDate,
initialOffset: initialOffset,
),
child: Builder(builder: (context) {
return BackButtonListener(
onBackButtonPressed: () async {
context.read<PageViewStateProvider>().backButtonWasPressed = true;
var visitedIndexes =
context.read<PageViewStateProvider>().visitedIndexes;
if (visitedIndexes.length == 1) {
return false;
}
visitedIndexes.removeLast();
pageController.jumpToPage(visitedIndexes.last);
return true;
},
child: Scaffold(
appBar: AppBar(
title: Builder(builder: (context) {
return Text(DateFormat.yMMMMd('de').format(
context.watch<PageViewStateProvider>().displayedDate));
}),
actions: const [ThemeSwitcherButton()],
),
bottomNavigationBar: BottomAppBar(
shape: const RectangularNotchShape(),
color: Theme.of(context).colorScheme.secondary,
child: Builder(builder: (context) {
return SumWidget(
date: context.watch<PageViewStateProvider>().displayedDate);
}),
),
drawer: const AppDrawer(),
floatingActionButton: OverflowBar(children: [
ScanFoodFAB(
onPressed: () {
var result = BarcodeScanner.scan();
context.read<FoodEntryBloc>().add(
BarcodeScanned(
scanResultFuture: result,
forDate: context
.read<PageViewStateProvider>()
.displayedDate,
),
);
},
),
const SizedBox(width: 8),
CalendarFAB(
startFromDate: DateTimeHelper.now(),
onDateSelected: (dateSelected) {
if (dateSelected == null) return;
var dateDiff = dateSelected.difference(initialDate).inDays;
log("dateDiff = $dateDiff");
pageController.jumpToPage(initialOffset - dateDiff);
},
),
]),
floatingActionButtonLocation:
FloatingActionButtonLocation.endDocked,
body: PerDatePageView(
pageController: pageController,
initialDate: initialDate,
),
),
);
}),
);
}
}
class PageViewStateProvider with ChangeNotifier {
DateTime _displayedDate;
final List<int> _visitedIndexes;
bool _backButtonWasPressed = false;
PageViewStateProvider({required DateTime initialDate, int initialOffset = 0})
: _displayedDate = initialDate,
_visitedIndexes = [] {
_visitedIndexes.add(initialOffset);
}
set backButtonWasPressed(val) => _backButtonWasPressed = val;
get backButtonWasPressed => _backButtonWasPressed;
get displayedDate => _displayedDate;
void setDisplayedDate(date) {
_displayedDate = date;
notifyListeners();
}
get visitedIndexes => _visitedIndexes;
void addVisitedIndex(int index) {
_visitedIndexes.add(index);
}
void addVisitedindexIfNotVisitedByBackButton(int index) {
if (_backButtonWasPressed) {
_backButtonWasPressed = false;
return;
}
addVisitedIndex(index);
}
}

View File

@ -1,11 +1,22 @@
import 'package:barcode_scan2/barcode_scan2.dart';
import 'package:calorimeter/utils/scan_food_floating_button.dart';
import 'package:calorimeter/utils/app_drawer.dart';
import 'package:calorimeter/food_entry/food_entry_bloc.dart'; import 'package:calorimeter/food_entry/food_entry_bloc.dart';
import 'package:calorimeter/perdate/entry_list.dart'; import 'package:calorimeter/perdate/entry_list.dart';
import 'package:calorimeter/storage/storage.dart';
import 'package:calorimeter/utils/calendar_floating_button.dart';
import 'package:calorimeter/utils/rectangular_notch_shape.dart';
import 'package:calorimeter/utils/sum_widget.dart';
import 'package:calorimeter/utils/theme_switcher_button.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:intl/intl.dart';
class PerDateWidget extends StatefulWidget { class PerDateWidget extends StatefulWidget {
final DateTime date; final DateTime date;
const PerDateWidget({super.key, required this.date}); final Function(DateTime?) onDateSelected;
const PerDateWidget(
{super.key, required this.date, required this.onDateSelected});
@override @override
State<PerDateWidget> createState() => _PerDateWidgetState(); State<PerDateWidget> createState() => _PerDateWidgetState();
@ -13,6 +24,9 @@ class PerDateWidget extends StatefulWidget {
class _PerDateWidgetState extends State<PerDateWidget> class _PerDateWidgetState extends State<PerDateWidget>
with AutomaticKeepAliveClientMixin<PerDateWidget> { with AutomaticKeepAliveClientMixin<PerDateWidget> {
late FoodStorage storage;
List<FoodEntryState> entries = [];
@override @override
void initState() { void initState() {
context context
@ -21,6 +35,11 @@ class _PerDateWidgetState extends State<PerDateWidget>
super.initState(); super.initState();
} }
@override
void dispose() {
super.dispose();
}
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
super.build(context); super.build(context);
@ -31,10 +50,43 @@ class _PerDateWidgetState extends State<PerDateWidget>
showNewSnackbarWith(context, pageState.errorString!); showNewSnackbarWith(context, pageState.errorString!);
} }
}, },
builder: (context, pageState) { builder: (context, globalState) {
return FoodEntryList( return Scaffold(
entries: pageState.foodEntries[widget.date] ?? [], appBar: AppBar(
date: widget.date); title: Text(DateFormat.yMMMMd('de').format(widget.date)),
actions: const [ThemeSwitcherButton()],
),
body: FoodEntryList(
entries: globalState.foodEntries[widget.date] ?? [],
date: widget.date),
bottomNavigationBar: BottomAppBar(
shape: const RectangularNotchShape(),
color: Theme.of(context).colorScheme.secondary,
child: SumWidget(
foodEntries: globalState.foodEntries[widget.date] ?? [])),
drawer: const AppDrawer(),
floatingActionButton: OverflowBar(children: [
ScanFoodFloatingButton(
onPressed: () {
var result = BarcodeScanner.scan();
context.read<FoodEntryBloc>().add(
BarcodeScanned(
scanResultFuture: result,
forDate: widget.date,
),
);
},
),
const SizedBox(width: 8),
CalendarFloatingButton(
startFromDate: widget.date,
onDateSelected: (dateSelected) {
widget.onDateSelected(dateSelected);
},
),
]),
floatingActionButtonLocation:
FloatingActionButtonLocation.endDocked);
}, },
); );
} }

View File

@ -1,7 +1,6 @@
import 'dart:io'; import 'dart:io';
import 'package:calorimeter/food_entry/food_entry_bloc.dart'; import 'package:calorimeter/food_entry/food_entry_bloc.dart';
import 'package:calorimeter/utils/date_time_helper.dart';
import 'package:path_provider/path_provider.dart'; import 'package:path_provider/path_provider.dart';
import 'package:universal_platform/universal_platform.dart'; import 'package:universal_platform/universal_platform.dart';
@ -158,7 +157,7 @@ class FoodStorage {
// get a list of dates of the last 365 days // get a list of dates of the last 365 days
var dates = List<DateTime>.generate(365, (idx) { var dates = List<DateTime>.generate(365, (idx) {
var durationToPast = Duration(days: idx); var durationToPast = Duration(days: idx);
return DateTimeHelper.now().subtract(durationToPast); return DateTime.now().subtract(durationToPast);
}); });
for (var date in dates.reversed) { for (var date in dates.reversed) {

View File

@ -1,11 +1,10 @@
import 'package:calorimeter/utils/date_time_helper.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
class CalendarFAB extends StatelessWidget { class CalendarFloatingButton extends StatelessWidget {
final DateTime startFromDate; final DateTime startFromDate;
final Function(DateTime?) onDateSelected; final Function(DateTime?) onDateSelected;
const CalendarFAB( const CalendarFloatingButton(
{super.key, required this.startFromDate, required this.onDateSelected}); {super.key, required this.startFromDate, required this.onDateSelected});
@override @override
@ -16,18 +15,17 @@ class CalendarFAB extends StatelessWidget {
locale: const Locale('de'), locale: const Locale('de'),
context: context, context: context,
initialDate: startFromDate, initialDate: startFromDate,
currentDate: DateTimeHelper.now(), currentDate: DateTime.now(),
firstDate: firstDate: DateTime.now().subtract(const Duration(days: 365 * 10)),
DateTimeHelper.now().subtract(const Duration(days: 365 * 10)), lastDate: DateTime.now().add(const Duration(days: 365 * 10)),
lastDate: DateTimeHelper.now().add(const Duration(days: 365 * 10)),
); );
if (!context.mounted) return; if (!context.mounted) return;
onDateSelected(datePicked?.copyWith(isUtc: true)); onDateSelected(datePicked);
}, },
heroTag: "calendarFAB", heroTag: "calendarFAB",
child: const Icon(Icons.date_range), child: const Icon(Icons.today),
); );
} }
} }

View File

@ -1,12 +0,0 @@
class DateTimeHelper {
static DateTime now() {
return DateTime.now().copyWith(
isUtc: true,
hour: 0,
minute: 0,
second: 0,
millisecond: 0,
microsecond: 0,
);
}
}

View File

@ -1,8 +1,8 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
class ScanFoodFAB extends StatelessWidget { class ScanFoodFloatingButton extends StatelessWidget {
final Function() onPressed; final Function() onPressed;
const ScanFoodFAB({super.key, required this.onPressed}); const ScanFoodFloatingButton({super.key, required this.onPressed});
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {

View File

@ -4,57 +4,53 @@ import 'package:calorimeter/food_entry/food_entry_bloc.dart';
import 'package:calorimeter/utils/settings_bloc.dart'; import 'package:calorimeter/utils/settings_bloc.dart';
class SumWidget extends StatelessWidget { class SumWidget extends StatelessWidget {
final DateTime date; final List<FoodEntryState> foodEntries;
const SumWidget({required this.foodEntries, super.key});
const SumWidget({super.key, required this.date});
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return BlocBuilder<FoodEntryBloc, GlobalEntryState>( return BlocBuilder<SettingsDataBloc, SettingsState>(
builder: (context, entryState) { builder: (context, state) {
return BlocBuilder<SettingsDataBloc, SettingsState>( var sum = 0.0;
builder: (context, settingsState) { for (var entry in foodEntries) {
var sum = 0.0; sum += entry.kcalPer100 / 100 * entry.mass;
for (var entry in entryState.foodEntries[date] ?? []) { }
sum += entry.kcalPer100 / 100 * entry.mass; var diff = state.kcalLimit - sum;
} var diffLimit = state.kcalLimit ~/ 4;
var diff = settingsState.kcalLimit - sum;
var diffLimit = settingsState.kcalLimit ~/ 4;
var textColor = Theme.of(context).colorScheme.onSecondary; var textColor = Theme.of(context).colorScheme.onSecondary;
var newTextColor = textColor; var newTextColor = textColor;
var brightness = Theme.of(context).brightness; var brightness = Theme.of(context).brightness;
switch (brightness) { switch (brightness) {
case Brightness.dark: case Brightness.dark:
if (diff < 0) { if (diff < 0) {
newTextColor = Colors.red[900]!; newTextColor = Colors.red[900]!;
} else if (diff < diffLimit) { } else if (diff < diffLimit) {
newTextColor = Colors.orange[900]!; newTextColor = Colors.orange[900]!;
} }
break; break;
case Brightness.light: case Brightness.light:
if (diff < 0) { if (diff < 0) {
newTextColor = Colors.redAccent; newTextColor = Colors.redAccent;
} else if (diff < diffLimit) { } else if (diff < diffLimit) {
newTextColor = Colors.orangeAccent; newTextColor = Colors.orangeAccent;
} }
break; break;
} }
return Align( return Align(
alignment: Alignment.centerLeft, alignment: Alignment.centerLeft,
child: Text( child: Text(
'kcal heute: ${sum.ceil().toString()}/${settingsState.kcalLimit.ceil()}', 'kcal heute: ${sum.ceil().toString()}/${state.kcalLimit.ceil()}',
style: Theme.of(context) style: Theme.of(context)
.textTheme .textTheme
.bodyLarge! .bodyLarge!
.copyWith(color: newTextColor), .copyWith(color: newTextColor),
), ),
); );
}, },
); );
});
} }
} }