summaryrefslogtreecommitdiff
path: root/lib/pages
diff options
context:
space:
mode:
Diffstat (limited to 'lib/pages')
-rw-r--r--lib/pages/add_shift_page.dart142
-rw-r--r--lib/pages/agenda_page.dart118
-rw-r--r--lib/pages/home_page.dart62
-rw-r--r--lib/pages/logbook_page.dart172
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)),
+ ],
+ ),
+ ),
+ );
+ }
+}