diff options
Diffstat (limited to 'lib/pages')
| -rw-r--r-- | lib/pages/add_shift_page.dart | 142 | ||||
| -rw-r--r-- | lib/pages/agenda_page.dart | 118 | ||||
| -rw-r--r-- | lib/pages/home_page.dart | 62 | ||||
| -rw-r--r-- | lib/pages/logbook_page.dart | 172 |
4 files changed, 494 insertions, 0 deletions
diff --git a/lib/pages/add_shift_page.dart b/lib/pages/add_shift_page.dart new file mode 100644 index 0000000..64b6142 --- /dev/null +++ b/lib/pages/add_shift_page.dart @@ -0,0 +1,142 @@ +import 'package:flutter/material.dart'; +import 'package:training_planner/pages/agenda_page.dart'; +import 'package:training_planner/pages/home_page.dart'; +import 'package:training_planner/shift.dart'; +import 'package:training_planner/main.dart'; +import 'package:training_planner/style/style.dart'; + +class AddShiftPage extends StatefulWidget { + final int pageNr; + final int pageIndex; + final DateTime mondayOfWeek; + @override + _AddShiftPageState createState() => _AddShiftPageState(); + + const AddShiftPage({ + Key? key, + required this.pageNr, + required this.pageIndex, + required this.mondayOfWeek, + }) : super(key: key); +} + +class _AddShiftPageState extends State<AddShiftPage> { + @override + initState() { + super.initState(); + } + + String dropdownValue = 'Maandag'; + List<bool> isSelected = [false, true, false]; + + void addShift() { + DateTime startDate = widget.mondayOfWeek; + switch (dropdownValue) { + case 'Dinsdag': + startDate = startDate.add(Duration(days: 1)); + break; + case 'Woensdag': + startDate = startDate.add(Duration(days: 2)); + break; + case 'Donderdag': + startDate = startDate.add(Duration(days: 3)); + break; + case 'Vrijdag': + startDate = startDate.add(Duration(days: 4)); + break; + case 'Zaterdag': + startDate = startDate.add(Duration(days: 5)); + break; + case 'Zondag': + startDate = startDate.add(Duration(days: 6)); + break; + } + + shiftProvider.addShift(Shift( + start: startDate, + type: isSelected[0] + ? ShiftType.Dagrit + : isSelected[1] + ? ShiftType.Avondrit + : ShiftType.Terugscannen)); + + Navigator.pop(context, true); + + // Previous page will not refresh without this. + Navigator.pushAndRemoveUntil( + context, + MaterialPageRoute(builder: (context) => HomePage()), + (Route<dynamic> route) => false, + ); + } + + @override + Widget build(BuildContext context) { + return Scaffold( + appBar: AppBar( + title: Text('Toevoegen aan planning #' + widget.pageNr.toString()), + backgroundColor: Style.background, + foregroundColor: Style.titleColor, + ), + body: Center( + child: Column( + children: [ + Padding(padding: const EdgeInsets.all(20)), + Container( + width: 200, + child: DropdownButton<String>( + value: dropdownValue, + icon: const Icon(Icons.arrow_downward), + elevation: 16, + style: const TextStyle(color: Colors.black), + underline: Container( + height: 2, + color: Colors.black, + ), + onChanged: (String? newValue) { + setState(() { + dropdownValue = newValue!; + }); + }, + items: <String>[ + 'Maandag', + 'Dinsdag', + 'Woensdag', + 'Donderdag', + 'Vrijdag', + 'Zaterdag', + 'Zondag' + ].map<DropdownMenuItem<String>>((String value) { + return DropdownMenuItem<String>( + value: value, + child: Text(value), + ); + }).toList(), + ), + ), + Padding(padding: const EdgeInsets.all(20)), + ToggleButtons( + children: <Widget>[ + Padding( + padding: const EdgeInsets.all(15), child: Text('Dagrit')), + Padding( + padding: const EdgeInsets.all(15), child: Text('Avondrit')), + Padding( + padding: const EdgeInsets.all(15), + child: Text('Terugscan')), + ], + onPressed: (int index) { + setState(() { + isSelected[index] = !isSelected[index]; + }); + }, + isSelected: isSelected, + ), + Padding( + padding: const EdgeInsets.all(20), + ), + TextButton(onPressed: () => {addShift()}, child: Text('Toevoegen')), + ], + ))); + } +} diff --git a/lib/pages/agenda_page.dart b/lib/pages/agenda_page.dart new file mode 100644 index 0000000..55c7360 --- /dev/null +++ b/lib/pages/agenda_page.dart @@ -0,0 +1,118 @@ +import 'package:carousel_slider/carousel_slider.dart'; +import 'package:flutter/material.dart'; +import 'package:in_date_utils/in_date_utils.dart' as DateUtilities; +import 'package:training_planner/pages/add_shift_page.dart'; +import 'package:training_planner/shift.dart'; +import 'package:training_planner/style/style.dart'; +import 'package:training_planner/utils/date.dart'; +import 'package:training_planner/widgets/agenda_week.dart'; +import 'package:flutter_local_notifications/flutter_local_notifications.dart'; + +class AgendaPage extends StatefulWidget { + @override + _AgendaPageState createState() => _AgendaPageState(); + + const AgendaPage({Key? key}) : super(key: key); +} + +class _AgendaPageState extends State<AgendaPage> { + int weekToStartAt = 0; + List<Widget> weeks = []; + List<int> weekNrs = []; + List<DateTime> dateTimes = []; + int currentSelectedPageIndex = 0; + int currentSelectedPageNr = 0; + DateTime currentSelectedWeek = DateTime.now(); + + @override + initState() { + super.initState(); + + weeks = getWeeks(); + + currentSelectedPageIndex = weekToStartAt; + currentSelectedPageNr = weekNrs[weekToStartAt]; + } + + List<Widget> getWeeks() { + List<Widget> result = []; + List<int> weekNrs = []; + DateTime startDate = + DateUtilities.DateUtils.firstDayOfWeek(DateTime(2020, 1, 1)); + DateTime today = DateTime.now(); + DateTime firstDayOfCurrentWeek = + DateUtilities.DateUtils.firstDayOfWeek(today); + int difference = today.difference(startDate).inDays; + + int totalWeeks = (difference / 7.0).ceil() + 4; + + for (int i = 0; i < totalWeeks; i++) { + DateTime mondayOfWeek = startDate.add(Duration(days: 7 * i)); + int weekNr = DateUtilities.DateUtils.getWeekNumber(mondayOfWeek); + + bool isCurrentWeek = false; + if (mondayOfWeek == firstDayOfCurrentWeek) { + weekToStartAt = i; + isCurrentWeek = true; + } + + result.add( + AgendaWeek( + weekNr: weekNr, + mondayOfWeek: mondayOfWeek, + isCurrentWeek: isCurrentWeek, + ), + ); + weekNrs.add(weekNr); + dateTimes.add(mondayOfWeek); + } + this.weekNrs = weekNrs; + + return result; + } + + @override + Widget build(BuildContext context) { + return Scaffold( + body: Padding( + padding: const EdgeInsets.all(0), + child: CarouselSlider( + options: CarouselOptions( + onPageChanged: (index, _) { + currentSelectedPageIndex = index; + currentSelectedPageNr = weekNrs[index]; + currentSelectedWeek = dateTimes[index]; + }, + height: MediaQuery.of(context).size.height - 163, + viewportFraction: 1.0, + enlargeCenterPage: false, + enableInfiniteScroll: false, + initialPage: weekToStartAt, // Week nr + ), + items: getWeeks(), + ), + ), + floatingActionButton: FloatingActionButton( + onPressed: () { + Navigator.push( + context, + MaterialPageRoute( + builder: (context) => AddShiftPage( + pageNr: currentSelectedPageNr, + pageIndex: currentSelectedPageIndex, + mondayOfWeek: currentSelectedWeek, + )), + ).then((val) => { + setState(() { + weeks = getWeeks(); + currentSelectedPageIndex = weekToStartAt; + currentSelectedPageNr = weekNrs[weekToStartAt]; + }) + }); + }, + backgroundColor: Style.titleColor, + child: const Icon(Icons.add), + ), + ); + } +} diff --git a/lib/pages/home_page.dart b/lib/pages/home_page.dart new file mode 100644 index 0000000..821caed --- /dev/null +++ b/lib/pages/home_page.dart @@ -0,0 +1,62 @@ +import 'package:flutter/material.dart'; +import 'package:training_planner/pages/agenda_page.dart'; +import 'package:training_planner/pages/logbook_page.dart'; +import 'package:training_planner/shift.dart'; +import 'package:training_planner/main.dart'; +import 'package:training_planner/style/style.dart'; + +class HomePage extends StatefulWidget { + @override + _HomePageState createState() => _HomePageState(); + + const HomePage({Key? key}) : super(key: key); +} + +class _HomePageState extends State<HomePage> { + int _selectedIndex = 0; + + static final List<Widget> _widgetOptions = <Widget>[ + new AgendaPage(), + new LogbookPage(), + ]; + + @override + initState() { + super.initState(); + } + + void _onItemTapped(int index) { + setState(() { + _selectedIndex = index; + }); + } + + @override + Widget build(BuildContext context) { + return Scaffold( + appBar: AppBar( + title: const Text('Werkschema'), + backgroundColor: Style.background, + foregroundColor: Style.titleColor, + ), + body: Center( + child: _widgetOptions.elementAt(_selectedIndex), + ), + bottomNavigationBar: BottomNavigationBar( + items: const <BottomNavigationBarItem>[ + BottomNavigationBarItem( + icon: Icon(Icons.assignment_ind_sharp), + label: 'Agenda', + ), + BottomNavigationBarItem( + icon: Icon(Icons.book), + label: 'Logboek', + ), + ], + currentIndex: _selectedIndex, + selectedItemColor: Style.titleColor, + onTap: _onItemTapped, + ), + ); + } +} diff --git a/lib/pages/logbook_page.dart b/lib/pages/logbook_page.dart new file mode 100644 index 0000000..bd9bde2 --- /dev/null +++ b/lib/pages/logbook_page.dart @@ -0,0 +1,172 @@ +import 'package:carousel_slider/carousel_slider.dart'; +import 'package:flutter/material.dart'; +import 'package:in_date_utils/in_date_utils.dart' as DateUtilities; +import 'package:training_planner/main.dart'; +import 'package:training_planner/pages/add_shift_page.dart'; +import 'package:training_planner/shift.dart'; +import 'package:training_planner/style/style.dart'; +import 'package:training_planner/utils/date.dart'; +import 'package:training_planner/widgets/agenda_week.dart'; +import 'package:flutter_local_notifications/flutter_local_notifications.dart'; + +class LogbookPage extends StatefulWidget { + @override + _LogbookPageState createState() => _LogbookPageState(); + + const LogbookPage({Key? key}) : super(key: key); +} + +class MonthData { + DateTime firstDayOfMonth; + List<Shift> shifts; + Duration totalWorkedTime = Duration(); + double expectedSalary = 0; + + void calculateData() { + totalWorkedTime = Duration(); + for (var shift in shifts) { + totalWorkedTime += shift.getElapsedSessionTime(); + } + if (shifts.isNotEmpty) { + expectedSalary = totalWorkedTime.inMinutes * shifts[0].getMinutePayRate(); + } + } + + MonthData({required this.firstDayOfMonth, required this.shifts}) { + calculateData(); + } +} + +class _LogbookPageState extends State<LogbookPage> { + List<MonthData> months = []; + + void updateMonthData(MonthData month, Shift shift) { + month.shifts.add(shift); + month.calculateData(); + } + + void sortShifts(List<Shift> shifts) { + for (var shift in shifts) { + DateTime firstDayOfMonth = + DateUtilities.DateUtils.firstDayOfMonth(shift.start); + + bool found = false; + for (var month in months) { + if (month.firstDayOfMonth == firstDayOfMonth) { + updateMonthData(month, shift); + found = true; + } + } + + if (!found) { + months + .add(MonthData(firstDayOfMonth: firstDayOfMonth, shifts: [shift])); + } + } + + months.sort((a, b) => b.firstDayOfMonth.compareTo(a.firstDayOfMonth)); + } + + @override + initState() { + super.initState(); + + shiftProvider.getPastShifts().then( + (value) => setState( + () { + List<Shift> allShifts = value; + sortShifts(allShifts); + }, + ), + ); + } + + List<Widget> createMonthDataWidgets() { + List<Widget> result = []; + + for (var month in months) { + result.add(Padding( + padding: const EdgeInsets.only(bottom: 8, left: 10, right: 10), + child: Container( + decoration: BoxDecoration( + border: Border.all(color: Color.fromARGB(255, 140, 140, 180)), + color: Color.fromARGB(255, 180, 180, 200), + borderRadius: BorderRadius.all(Radius.circular(8))), + child: Padding( + padding: const EdgeInsets.all(8), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text( + DateHelper.getMonthName(month.firstDayOfMonth.month) + + ' ' + + month.firstDayOfMonth.year.toString(), + style: TextStyle( + color: Colors.white, fontWeight: FontWeight.bold), + ), + Padding(padding: EdgeInsets.only(left: 5, bottom: 5, right: 5)), + Text('Gewerkt: ' + + month.totalWorkedTime.inHours.toString() + + ' uur'), + Text('Verdiend: ' + + month.expectedSalary.toStringAsFixed(2) + + (month.shifts.isNotEmpty + ? ' (' + + (month.shifts[0].getMinutePayRate() * 60) + .toStringAsFixed(2) + + '/uur)' + : '')), + Padding(padding: EdgeInsets.only(left: 5, bottom: 5, right: 5)), + ], + ), + ), + ), + )); + } + + return result; + } + + @override + Widget build(BuildContext context) { + var monthDataWidgets = createMonthDataWidgets(); + + return ShaderMask( + shaderCallback: (Rect rect) { + return LinearGradient( + begin: Alignment.topCenter, + end: Alignment.bottomCenter, + colors: [ + Style.background, + Colors.transparent, + Colors.transparent, + Style.background + ], + stops: [ + 0.0, + 0.05, + 0.95, + 1.0 + ], // 10% purple, 80% transparent, 10% purple + ).createShader(rect); + }, + blendMode: BlendMode.dstOut, + child: SafeArea( + child: CustomScrollView( + physics: null, + slivers: [ + SliverPadding(padding: EdgeInsets.only(top: 20)), + SliverList( + delegate: SliverChildBuilderDelegate( + (BuildContext context, int index) { + return monthDataWidgets[index]; + }, + childCount: monthDataWidgets.length, + )), + SliverPadding(padding: EdgeInsets.only(top: 20)), + ], + ), + ), + ); + } +} |
