Compare commits
	
		
			No commits in common. "4aa6926f5ca953eeaf4aeba9be1be9638b607f09" and "fb278e448796cad1f32b192b76b61df8fe92443d" have entirely different histories.
		
	
	
		
			4aa6926f5c
			...
			fb278e4487
		
	
		
|  | @ -1,6 +1,6 @@ | ||||||
| import 'dart:io'; | import 'dart:io'; | ||||||
| 
 | 
 | ||||||
| import 'package:fmassive/gym_sets.dart'; | import 'package:fmassive/gym_set.dart'; | ||||||
| import 'package:fmassive/main.dart'; | import 'package:fmassive/main.dart'; | ||||||
| import 'package:fmassive/plans.dart'; | import 'package:fmassive/plans.dart'; | ||||||
| import 'package:moor/ffi.dart'; | import 'package:moor/ffi.dart'; | ||||||
|  | @ -22,18 +22,12 @@ class MyDatabase extends _$MyDatabase { | ||||||
|   @override |   @override | ||||||
|   MigrationStrategy get migration => MigrationStrategy( |   MigrationStrategy get migration => MigrationStrategy( | ||||||
|         onCreate: (Migrator m) async { |         onCreate: (Migrator m) async { | ||||||
|           print('Creating...'); |  | ||||||
|           await m.createAll(); |           await m.createAll(); | ||||||
|           var data = await (db.select(db.settings)..limit(1)).get(); |           var data = await (db.select(db.settings)..limit(1)).get(); | ||||||
|           if (data.isEmpty) await db.into(db.settings).insert(defaultSettings); |           if (data.isEmpty) await db.into(db.settings).insert(defaultSettings); | ||||||
|         }, |         }, | ||||||
|         onUpgrade: (Migrator m, int from, int to) async { |         onUpgrade: (Migrator m, int from, int to) async { | ||||||
|           if (from == 1) { |           if (from == 1) { | ||||||
|             await m.create(db.gymSets); |  | ||||||
|             await db.customInsert(''' |  | ||||||
|               INSERT INTO gym_sets(id, name, reps, weight, created, unit, hidden, image, sets, minutes, seconds, steps)  |  | ||||||
|                 SELECT id, name, reps, weight, created, unit, hidden, image, sets, minutes, seconds, steps FROM sets |  | ||||||
|             '''); |  | ||||||
|             await m.addColumn(settings, settings.darkColor); |             await m.addColumn(settings, settings.darkColor); | ||||||
|             await db.customStatement(''' |             await db.customStatement(''' | ||||||
|               UPDATE settings SET dark_color = darkColor |               UPDATE settings SET dark_color = darkColor | ||||||
|  |  | ||||||
|  | @ -1248,8 +1248,8 @@ class $GymSetsTable extends GymSets with TableInfo<$GymSetsTable, GymSet> { | ||||||
| class Plan extends DataClass implements Insertable<Plan> { | class Plan extends DataClass implements Insertable<Plan> { | ||||||
|   final int id; |   final int id; | ||||||
|   final String days; |   final String days; | ||||||
|   final String exercises; |   final String workouts; | ||||||
|   Plan({required this.id, required this.days, required this.exercises}); |   Plan({required this.id, required this.days, required this.workouts}); | ||||||
|   factory Plan.fromData(Map<String, dynamic> data, GeneratedDatabase db, |   factory Plan.fromData(Map<String, dynamic> data, GeneratedDatabase db, | ||||||
|       {String? prefix}) { |       {String? prefix}) { | ||||||
|     final effectivePrefix = prefix ?? ''; |     final effectivePrefix = prefix ?? ''; | ||||||
|  | @ -1258,8 +1258,8 @@ class Plan extends DataClass implements Insertable<Plan> { | ||||||
|           .mapFromDatabaseResponse(data['${effectivePrefix}id'])!, |           .mapFromDatabaseResponse(data['${effectivePrefix}id'])!, | ||||||
|       days: const StringType() |       days: const StringType() | ||||||
|           .mapFromDatabaseResponse(data['${effectivePrefix}days'])!, |           .mapFromDatabaseResponse(data['${effectivePrefix}days'])!, | ||||||
|       exercises: const StringType() |       workouts: const StringType() | ||||||
|           .mapFromDatabaseResponse(data['${effectivePrefix}exercises'])!, |           .mapFromDatabaseResponse(data['${effectivePrefix}workouts'])!, | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|   @override |   @override | ||||||
|  | @ -1267,7 +1267,7 @@ class Plan extends DataClass implements Insertable<Plan> { | ||||||
|     final map = <String, Expression>{}; |     final map = <String, Expression>{}; | ||||||
|     map['id'] = Variable<int>(id); |     map['id'] = Variable<int>(id); | ||||||
|     map['days'] = Variable<String>(days); |     map['days'] = Variable<String>(days); | ||||||
|     map['exercises'] = Variable<String>(exercises); |     map['workouts'] = Variable<String>(workouts); | ||||||
|     return map; |     return map; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  | @ -1275,7 +1275,7 @@ class Plan extends DataClass implements Insertable<Plan> { | ||||||
|     return PlansCompanion( |     return PlansCompanion( | ||||||
|       id: Value(id), |       id: Value(id), | ||||||
|       days: Value(days), |       days: Value(days), | ||||||
|       exercises: Value(exercises), |       workouts: Value(workouts), | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  | @ -1285,7 +1285,7 @@ class Plan extends DataClass implements Insertable<Plan> { | ||||||
|     return Plan( |     return Plan( | ||||||
|       id: serializer.fromJson<int>(json['id']), |       id: serializer.fromJson<int>(json['id']), | ||||||
|       days: serializer.fromJson<String>(json['days']), |       days: serializer.fromJson<String>(json['days']), | ||||||
|       exercises: serializer.fromJson<String>(json['exercises']), |       workouts: serializer.fromJson<String>(json['workouts']), | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
|   @override |   @override | ||||||
|  | @ -1294,69 +1294,69 @@ class Plan extends DataClass implements Insertable<Plan> { | ||||||
|     return <String, dynamic>{ |     return <String, dynamic>{ | ||||||
|       'id': serializer.toJson<int>(id), |       'id': serializer.toJson<int>(id), | ||||||
|       'days': serializer.toJson<String>(days), |       'days': serializer.toJson<String>(days), | ||||||
|       'exercises': serializer.toJson<String>(exercises), |       'workouts': serializer.toJson<String>(workouts), | ||||||
|     }; |     }; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   Plan copyWith({int? id, String? days, String? exercises}) => Plan( |   Plan copyWith({int? id, String? days, String? workouts}) => Plan( | ||||||
|         id: id ?? this.id, |         id: id ?? this.id, | ||||||
|         days: days ?? this.days, |         days: days ?? this.days, | ||||||
|         exercises: exercises ?? this.exercises, |         workouts: workouts ?? this.workouts, | ||||||
|       ); |       ); | ||||||
|   @override |   @override | ||||||
|   String toString() { |   String toString() { | ||||||
|     return (StringBuffer('Plan(') |     return (StringBuffer('Plan(') | ||||||
|           ..write('id: $id, ') |           ..write('id: $id, ') | ||||||
|           ..write('days: $days, ') |           ..write('days: $days, ') | ||||||
|           ..write('exercises: $exercises') |           ..write('workouts: $workouts') | ||||||
|           ..write(')')) |           ..write(')')) | ||||||
|         .toString(); |         .toString(); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   @override |   @override | ||||||
|   int get hashCode => Object.hash(id, days, exercises); |   int get hashCode => Object.hash(id, days, workouts); | ||||||
|   @override |   @override | ||||||
|   bool operator ==(Object other) => |   bool operator ==(Object other) => | ||||||
|       identical(this, other) || |       identical(this, other) || | ||||||
|       (other is Plan && |       (other is Plan && | ||||||
|           other.id == this.id && |           other.id == this.id && | ||||||
|           other.days == this.days && |           other.days == this.days && | ||||||
|           other.exercises == this.exercises); |           other.workouts == this.workouts); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| class PlansCompanion extends UpdateCompanion<Plan> { | class PlansCompanion extends UpdateCompanion<Plan> { | ||||||
|   final Value<int> id; |   final Value<int> id; | ||||||
|   final Value<String> days; |   final Value<String> days; | ||||||
|   final Value<String> exercises; |   final Value<String> workouts; | ||||||
|   const PlansCompanion({ |   const PlansCompanion({ | ||||||
|     this.id = const Value.absent(), |     this.id = const Value.absent(), | ||||||
|     this.days = const Value.absent(), |     this.days = const Value.absent(), | ||||||
|     this.exercises = const Value.absent(), |     this.workouts = const Value.absent(), | ||||||
|   }); |   }); | ||||||
|   PlansCompanion.insert({ |   PlansCompanion.insert({ | ||||||
|     this.id = const Value.absent(), |     this.id = const Value.absent(), | ||||||
|     required String days, |     required String days, | ||||||
|     required String exercises, |     required String workouts, | ||||||
|   })  : days = Value(days), |   })  : days = Value(days), | ||||||
|         exercises = Value(exercises); |         workouts = Value(workouts); | ||||||
|   static Insertable<Plan> custom({ |   static Insertable<Plan> custom({ | ||||||
|     Expression<int>? id, |     Expression<int>? id, | ||||||
|     Expression<String>? days, |     Expression<String>? days, | ||||||
|     Expression<String>? exercises, |     Expression<String>? workouts, | ||||||
|   }) { |   }) { | ||||||
|     return RawValuesInsertable({ |     return RawValuesInsertable({ | ||||||
|       if (id != null) 'id': id, |       if (id != null) 'id': id, | ||||||
|       if (days != null) 'days': days, |       if (days != null) 'days': days, | ||||||
|       if (exercises != null) 'exercises': exercises, |       if (workouts != null) 'workouts': workouts, | ||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   PlansCompanion copyWith( |   PlansCompanion copyWith( | ||||||
|       {Value<int>? id, Value<String>? days, Value<String>? exercises}) { |       {Value<int>? id, Value<String>? days, Value<String>? workouts}) { | ||||||
|     return PlansCompanion( |     return PlansCompanion( | ||||||
|       id: id ?? this.id, |       id: id ?? this.id, | ||||||
|       days: days ?? this.days, |       days: days ?? this.days, | ||||||
|       exercises: exercises ?? this.exercises, |       workouts: workouts ?? this.workouts, | ||||||
|     ); |     ); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  | @ -1369,8 +1369,8 @@ class PlansCompanion extends UpdateCompanion<Plan> { | ||||||
|     if (days.present) { |     if (days.present) { | ||||||
|       map['days'] = Variable<String>(days.value); |       map['days'] = Variable<String>(days.value); | ||||||
|     } |     } | ||||||
|     if (exercises.present) { |     if (workouts.present) { | ||||||
|       map['exercises'] = Variable<String>(exercises.value); |       map['workouts'] = Variable<String>(workouts.value); | ||||||
|     } |     } | ||||||
|     return map; |     return map; | ||||||
|   } |   } | ||||||
|  | @ -1380,7 +1380,7 @@ class PlansCompanion extends UpdateCompanion<Plan> { | ||||||
|     return (StringBuffer('PlansCompanion(') |     return (StringBuffer('PlansCompanion(') | ||||||
|           ..write('id: $id, ') |           ..write('id: $id, ') | ||||||
|           ..write('days: $days, ') |           ..write('days: $days, ') | ||||||
|           ..write('exercises: $exercises') |           ..write('workouts: $workouts') | ||||||
|           ..write(')')) |           ..write(')')) | ||||||
|         .toString(); |         .toString(); | ||||||
|   } |   } | ||||||
|  | @ -1403,13 +1403,13 @@ class $PlansTable extends Plans with TableInfo<$PlansTable, Plan> { | ||||||
|   late final GeneratedColumn<String?> days = GeneratedColumn<String?>( |   late final GeneratedColumn<String?> days = GeneratedColumn<String?>( | ||||||
|       'days', aliasedName, false, |       'days', aliasedName, false, | ||||||
|       type: const StringType(), requiredDuringInsert: true); |       type: const StringType(), requiredDuringInsert: true); | ||||||
|   final VerificationMeta _exercisesMeta = const VerificationMeta('exercises'); |   final VerificationMeta _workoutsMeta = const VerificationMeta('workouts'); | ||||||
|   @override |   @override | ||||||
|   late final GeneratedColumn<String?> exercises = GeneratedColumn<String?>( |   late final GeneratedColumn<String?> workouts = GeneratedColumn<String?>( | ||||||
|       'exercises', aliasedName, false, |       'workouts', aliasedName, false, | ||||||
|       type: const StringType(), requiredDuringInsert: true); |       type: const StringType(), requiredDuringInsert: true); | ||||||
|   @override |   @override | ||||||
|   List<GeneratedColumn> get $columns => [id, days, exercises]; |   List<GeneratedColumn> get $columns => [id, days, workouts]; | ||||||
|   @override |   @override | ||||||
|   String get aliasedName => _alias ?? 'plans'; |   String get aliasedName => _alias ?? 'plans'; | ||||||
|   @override |   @override | ||||||
|  | @ -1428,11 +1428,11 @@ class $PlansTable extends Plans with TableInfo<$PlansTable, Plan> { | ||||||
|     } else if (isInserting) { |     } else if (isInserting) { | ||||||
|       context.missing(_daysMeta); |       context.missing(_daysMeta); | ||||||
|     } |     } | ||||||
|     if (data.containsKey('exercises')) { |     if (data.containsKey('workouts')) { | ||||||
|       context.handle(_exercisesMeta, |       context.handle(_workoutsMeta, | ||||||
|           exercises.isAcceptableOrUnknown(data['exercises']!, _exercisesMeta)); |           workouts.isAcceptableOrUnknown(data['workouts']!, _workoutsMeta)); | ||||||
|     } else if (isInserting) { |     } else if (isInserting) { | ||||||
|       context.missing(_exercisesMeta); |       context.missing(_workoutsMeta); | ||||||
|     } |     } | ||||||
|     return context; |     return context; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | @ -2,15 +2,15 @@ import 'package:flutter/material.dart'; | ||||||
| 
 | 
 | ||||||
| class Days extends StatefulWidget { | class Days extends StatefulWidget { | ||||||
|   final ValueChanged<String> onChanged; |   final ValueChanged<String> onChanged; | ||||||
|   final String days; |  | ||||||
| 
 | 
 | ||||||
|   const Days({required this.onChanged, required this.days, super.key}); |   const Days({required this.onChanged, super.key}); | ||||||
| 
 | 
 | ||||||
|   @override |   @override | ||||||
|   createState() => _DaysState(); |   createState() => _DaysState(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| class _DaysState extends State<Days> { | class _DaysState extends State<Days> { | ||||||
|  |   final List<bool> _selections = List.generate(7, (_) => false); | ||||||
|   final List<String> _days = [ |   final List<String> _days = [ | ||||||
|     'Monday', |     'Monday', | ||||||
|     'Tuesday', |     'Tuesday', | ||||||
|  | @ -20,19 +20,14 @@ class _DaysState extends State<Days> { | ||||||
|     'Saturday', |     'Saturday', | ||||||
|     'Sunday' |     'Sunday' | ||||||
|   ]; |   ]; | ||||||
|   late List<bool> _selections; |  | ||||||
| 
 |  | ||||||
|   @override |  | ||||||
|   void initState() { |  | ||||||
|     super.initState(); |  | ||||||
|     final dayList = widget.days.split(','); |  | ||||||
|     _selections = _days.map((day) => dayList.contains(day)).toList(); |  | ||||||
|   } |  | ||||||
| 
 | 
 | ||||||
|   String _getSelectedDaysString() { |   String _getSelectedDaysString() { | ||||||
|     List<String> selectedDays = []; |     List<String> selectedDays = []; | ||||||
|     for (int i = 0; i < _selections.length; i++) |     for (int i = 0; i < _selections.length; i++) { | ||||||
|       if (_selections[i]) selectedDays.add(_days[i]); |       if (_selections[i]) { | ||||||
|  |         selectedDays.add(_days[i]); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|     return selectedDays.join(","); |     return selectedDays.join(","); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  | @ -45,10 +40,7 @@ class _DaysState extends State<Days> { | ||||||
| 
 | 
 | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     return Column(children: [ |     return Column( | ||||||
|       Text('Days', style: Theme.of(context).textTheme.headlineSmall), |  | ||||||
|       Expanded( |  | ||||||
|         child: ListView( |  | ||||||
|       children: List.generate(7, (index) { |       children: List.generate(7, (index) { | ||||||
|         return SwitchListTile( |         return SwitchListTile( | ||||||
|           title: Text(_days[index]), |           title: Text(_days[index]), | ||||||
|  | @ -56,8 +48,6 @@ class _DaysState extends State<Days> { | ||||||
|           onChanged: (value) => _updateSelections(index), |           onChanged: (value) => _updateSelections(index), | ||||||
|         ); |         ); | ||||||
|       }), |       }), | ||||||
|         ), |     ); | ||||||
|       ), |  | ||||||
|     ]); |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -2,7 +2,6 @@ import 'package:flutter/material.dart'; | ||||||
| import 'package:flutter/material.dart' as material; | import 'package:flutter/material.dart' as material; | ||||||
| import 'package:fmassive/database.dart'; | import 'package:fmassive/database.dart'; | ||||||
| import 'package:fmassive/days.dart'; | import 'package:fmassive/days.dart'; | ||||||
| import 'package:fmassive/exercises.dart'; |  | ||||||
| import 'package:fmassive/main.dart'; | import 'package:fmassive/main.dart'; | ||||||
| import 'package:moor_flutter/moor_flutter.dart'; | import 'package:moor_flutter/moor_flutter.dart'; | ||||||
| 
 | 
 | ||||||
|  | @ -16,69 +15,37 @@ class EditPlanPage extends StatefulWidget { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| class _EditPlanPageState extends State<EditPlanPage> { | class _EditPlanPageState extends State<EditPlanPage> { | ||||||
|  |   final TextEditingController _daysController = TextEditingController(); | ||||||
|  |   final TextEditingController _workoutsController = TextEditingController(); | ||||||
|   late PlansCompanion plan; |   late PlansCompanion plan; | ||||||
|   List<String?> names = []; |   final daysNode = FocusNode(); | ||||||
| 
 |   final workoutsNode = FocusNode(); | ||||||
|   Future<List<String?>> getDistinctNames() async { |  | ||||||
|     final names = await (db.gymSets.selectOnly(distinct: true) |  | ||||||
|           ..addColumns([db.gymSets.name])) |  | ||||||
|         .get(); |  | ||||||
|     return names.map((name) => name.read(db.gymSets.name)).toList(); |  | ||||||
|   } |  | ||||||
| 
 | 
 | ||||||
|   @override |   @override | ||||||
|   void initState() { |   void initState() { | ||||||
|     super.initState(); |     super.initState(); | ||||||
| 
 |  | ||||||
|     getDistinctNames().then((value) { |  | ||||||
|       setState(() { |  | ||||||
|         names = value; |  | ||||||
|       }); |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     plan = widget.plan; |     plan = widget.plan; | ||||||
|  |     _daysController.text = plan.days.value; | ||||||
|  |     _workoutsController.text = plan.workouts.value; | ||||||
|  |     if (plan.id.present) | ||||||
|  |       workoutsNode.requestFocus(); | ||||||
|  |     else | ||||||
|  |       daysNode.requestFocus(); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   @override |   @override | ||||||
|   dispose() { |   dispose() { | ||||||
|  |     daysNode.dispose(); | ||||||
|  |     workoutsNode.dispose(); | ||||||
|     super.dispose(); |     super.dispose(); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   Future<bool?> _showConfirmationDialog(BuildContext context) { |  | ||||||
|     return showDialog<bool>( |  | ||||||
|       context: context, |  | ||||||
|       barrierDismissible: false, |  | ||||||
|       builder: (BuildContext context) { |  | ||||||
|         return AlertDialog( |  | ||||||
|           title: const Text('Confirm Delete'), |  | ||||||
|           content: const Text('Are you sure you want to delete this plan?'), |  | ||||||
|           actions: <Widget>[ |  | ||||||
|             ElevatedButton( |  | ||||||
|               child: const Text('Yes'), |  | ||||||
|               onPressed: () { |  | ||||||
|                 Navigator.pop(context, true); // showDialog() returns true |  | ||||||
|               }, |  | ||||||
|             ), |  | ||||||
|             ElevatedButton( |  | ||||||
|               child: const Text('No'), |  | ||||||
|               onPressed: () { |  | ||||||
|                 Navigator.pop(context, false); // showDialog() returns false |  | ||||||
|               }, |  | ||||||
|             ), |  | ||||||
|           ], |  | ||||||
|         ); |  | ||||||
|       }, |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   @override |   @override | ||||||
|   Widget build(BuildContext context) { |   Widget build(BuildContext context) { | ||||||
|     List<Widget> actions = []; |     List<Widget> actions = []; | ||||||
|     if (widget.plan.id.present) |     if (widget.plan.id.present) | ||||||
|       actions.add(IconButton( |       actions.add(IconButton( | ||||||
|           onPressed: () async { |           onPressed: () async { | ||||||
|             bool? confirm = await _showConfirmationDialog(context); |  | ||||||
|             if (!confirm!) return; |  | ||||||
|             await db.plans.deleteOne(widget.plan); |             await db.plans.deleteOne(widget.plan); | ||||||
|             if (!mounted) return; |             if (!mounted) return; | ||||||
|             Navigator.pop(context); |             Navigator.pop(context); | ||||||
|  | @ -92,11 +59,38 @@ class _EditPlanPageState extends State<EditPlanPage> { | ||||||
|         padding: const EdgeInsets.all(16.0), |         padding: const EdgeInsets.all(16.0), | ||||||
|         child: material.Column( |         child: material.Column( | ||||||
|           crossAxisAlignment: CrossAxisAlignment.start, |           crossAxisAlignment: CrossAxisAlignment.start, | ||||||
|           children: getChildren, |           children: [ | ||||||
|  |             Days(onChanged: (days) { | ||||||
|  |               setState(() { | ||||||
|  |                 days = days; | ||||||
|  |               }); | ||||||
|  |             }), | ||||||
|  |             TextFormField( | ||||||
|  |               controller: _workoutsController, | ||||||
|  |               focusNode: workoutsNode, | ||||||
|  |               onTap: () { | ||||||
|  |                 _workoutsController.selection = TextSelection( | ||||||
|  |                     baseOffset: 0, | ||||||
|  |                     extentOffset: _workoutsController.text.length); | ||||||
|  |               }, | ||||||
|  |               decoration: const InputDecoration(labelText: 'Workouts'), | ||||||
|  |               onChanged: (value) { | ||||||
|  |                 setState(() { | ||||||
|  |                   plan = plan.copyWith(workouts: Value(value)); | ||||||
|  |                 }); | ||||||
|  |               }, | ||||||
|  |             ), | ||||||
|  |           ], | ||||||
|         ), |         ), | ||||||
|       ), |       ), | ||||||
|       floatingActionButton: FloatingActionButton( |       floatingActionButton: FloatingActionButton( | ||||||
|         onPressed: () async { |         onPressed: () async { | ||||||
|  |           if (_daysController.text.isEmpty) { | ||||||
|  |             ScaffoldMessenger.of(context).showSnackBar( | ||||||
|  |                 const SnackBar(content: Text('Please enter days'))); | ||||||
|  |             daysNode.requestFocus(); | ||||||
|  |             return; | ||||||
|  |           } | ||||||
|           if (plan.id.present) |           if (plan.id.present) | ||||||
|             await db.update(db.plans).replace(plan); |             await db.update(db.plans).replace(plan); | ||||||
|           else |           else | ||||||
|  | @ -108,32 +102,4 @@ class _EditPlanPageState extends State<EditPlanPage> { | ||||||
|       ), |       ), | ||||||
|     )); |     )); | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   List<Widget> get getChildren { |  | ||||||
|     return [ |  | ||||||
|       Expanded( |  | ||||||
|         child: Days( |  | ||||||
|           onChanged: (days) { |  | ||||||
|             setState(() { |  | ||||||
|               plan = plan.copyWith(days: Value(days)); |  | ||||||
|             }); |  | ||||||
|           }, |  | ||||||
|           days: plan.days.value, |  | ||||||
|         ), |  | ||||||
|       ), |  | ||||||
|       names.isNotEmpty |  | ||||||
|           ? Expanded( |  | ||||||
|               child: Exercises( |  | ||||||
|                 onChanged: (exercises) { |  | ||||||
|                   setState(() { |  | ||||||
|                     plan = plan.copyWith(exercises: Value(exercises)); |  | ||||||
|                   }); |  | ||||||
|                 }, |  | ||||||
|                 exercises: plan.exercises.value, |  | ||||||
|                 names: names, |  | ||||||
|               ), |  | ||||||
|             ) |  | ||||||
|           : const CircularProgressIndicator(), |  | ||||||
|     ]; |  | ||||||
|   } |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,60 +0,0 @@ | ||||||
| import 'package:flutter/material.dart'; |  | ||||||
| 
 |  | ||||||
| class Exercises extends StatefulWidget { |  | ||||||
|   final ValueChanged<String> onChanged; |  | ||||||
|   final String? exercises; |  | ||||||
|   final List<String?> names; |  | ||||||
| 
 |  | ||||||
|   const Exercises( |  | ||||||
|       {required this.onChanged, |  | ||||||
|       required this.exercises, |  | ||||||
|       super.key, |  | ||||||
|       required this.names}); |  | ||||||
| 
 |  | ||||||
|   @override |  | ||||||
|   createState() => _ExercisesState(); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| class _ExercisesState extends State<Exercises> { |  | ||||||
|   late List<bool> _selections; |  | ||||||
| 
 |  | ||||||
|   @override |  | ||||||
|   initState() { |  | ||||||
|     super.initState(); |  | ||||||
|     final exercises = widget.exercises?.split(','); |  | ||||||
|     _selections = |  | ||||||
|         widget.names.map((name) => exercises?.contains(name) ?? false).toList(); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   String _getExercises() { |  | ||||||
|     List<String> exercises = []; |  | ||||||
|     for (int i = 0; i < _selections.length; i++) |  | ||||||
|       if (_selections[i]) exercises.add(widget.names[i] ?? ''); |  | ||||||
|     return exercises.join(","); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   void _updateSelections(int index) { |  | ||||||
|     setState(() { |  | ||||||
|       _selections[index] = !_selections[index]; |  | ||||||
|       widget.onChanged(_getExercises()); |  | ||||||
|     }); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   @override |  | ||||||
|   Widget build(BuildContext context) { |  | ||||||
|     return Column(children: [ |  | ||||||
|       Text('Exercises', style: Theme.of(context).textTheme.headlineSmall), |  | ||||||
|       Expanded( |  | ||||||
|         child: ListView( |  | ||||||
|           children: List.generate(widget.names.length, (index) { |  | ||||||
|             return SwitchListTile( |  | ||||||
|               title: Text(widget.names[index] ?? ''), |  | ||||||
|               value: _selections[index], |  | ||||||
|               onChanged: (value) => _updateSelections(index), |  | ||||||
|             ); |  | ||||||
|           }), |  | ||||||
|         ), |  | ||||||
|       ), |  | ||||||
|     ]); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  | @ -1,61 +0,0 @@ | ||||||
| import 'package:flutter/material.dart'; |  | ||||||
| import 'package:fmassive/database.dart'; |  | ||||||
| import 'package:fmassive/edit_plan.dart'; |  | ||||||
| import 'package:fmassive/main.dart'; |  | ||||||
| import 'package:moor_flutter/moor_flutter.dart'; |  | ||||||
| 
 |  | ||||||
| class PlanList extends StatelessWidget { |  | ||||||
|   const PlanList({ |  | ||||||
|     super.key, |  | ||||||
|     required this.plans, |  | ||||||
|   }); |  | ||||||
| 
 |  | ||||||
|   final List<Plan> plans; |  | ||||||
| 
 |  | ||||||
|   @override |  | ||||||
|   Widget build(BuildContext context) { |  | ||||||
|     return ListView.builder( |  | ||||||
|       itemCount: plans.length, |  | ||||||
|       itemBuilder: (context, index) { |  | ||||||
|         return ListTile( |  | ||||||
|             title: Text(plans[index].days.replaceAll(',', ', ')), |  | ||||||
|             subtitle: Text(plans[index].exercises), |  | ||||||
|             onLongPress: () => showDialog( |  | ||||||
|                   context: context, |  | ||||||
|                   builder: (BuildContext context) { |  | ||||||
|                     return AlertDialog( |  | ||||||
|                       title: const Text('Delete set'), |  | ||||||
|                       content: Text( |  | ||||||
|                           'Are you sure you want to delete ${plans[index].days}?'), |  | ||||||
|                       actions: <Widget>[ |  | ||||||
|                         ElevatedButton( |  | ||||||
|                           child: const Text('Cancel'), |  | ||||||
|                           onPressed: () { |  | ||||||
|                             Navigator.of(context).pop(); |  | ||||||
|                           }, |  | ||||||
|                         ), |  | ||||||
|                         ElevatedButton( |  | ||||||
|                           child: const Text('Delete'), |  | ||||||
|                           onPressed: () async { |  | ||||||
|                             final navigator = Navigator.of(context); |  | ||||||
|                             await db.plans.deleteOne(plans[index]); |  | ||||||
|                             navigator.pop(); |  | ||||||
|                           }, |  | ||||||
|                         ), |  | ||||||
|                       ], |  | ||||||
|                     ); |  | ||||||
|                   }, |  | ||||||
|                 ), |  | ||||||
|             onTap: () async { |  | ||||||
|               await Navigator.push( |  | ||||||
|                 context, |  | ||||||
|                 MaterialPageRoute( |  | ||||||
|                   builder: (context) => |  | ||||||
|                       EditPlanPage(plan: plans[index].toCompanion(false)), |  | ||||||
|                 ), |  | ||||||
|               ); |  | ||||||
|             }); |  | ||||||
|       }, |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  | @ -3,5 +3,5 @@ import 'package:moor/moor.dart'; | ||||||
| class Plans extends Table { | class Plans extends Table { | ||||||
|   IntColumn get id => integer().autoIncrement()(); |   IntColumn get id => integer().autoIncrement()(); | ||||||
|   TextColumn get days => text()(); |   TextColumn get days => text()(); | ||||||
|   TextColumn get exercises => text()(); |   TextColumn get workouts => text()(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -2,7 +2,6 @@ import 'package:flutter/material.dart'; | ||||||
| import 'package:fmassive/database.dart'; | import 'package:fmassive/database.dart'; | ||||||
| import 'package:fmassive/edit_plan.dart'; | import 'package:fmassive/edit_plan.dart'; | ||||||
| import 'package:fmassive/main.dart'; | import 'package:fmassive/main.dart'; | ||||||
| import 'package:fmassive/plan_list.dart'; |  | ||||||
| import 'package:moor/moor.dart'; | import 'package:moor/moor.dart'; | ||||||
| 
 | 
 | ||||||
| class PlansPage extends StatelessWidget { | class PlansPage extends StatelessWidget { | ||||||
|  | @ -66,20 +65,27 @@ class _PlansPageState extends State<_PlansPage> { | ||||||
|             builder: (context, snapshot) { |             builder: (context, snapshot) { | ||||||
|               final plans = snapshot.data; |               final plans = snapshot.data; | ||||||
| 
 | 
 | ||||||
|               if (snapshot.hasError) |  | ||||||
|                 return Center( |  | ||||||
|                   child: Padding( |  | ||||||
|                       padding: const EdgeInsets.all(16.0), |  | ||||||
|                       child: Text( |  | ||||||
|                         'Error: ${snapshot.error}', |  | ||||||
|                         style: Theme.of(context).textTheme.headlineSmall, |  | ||||||
|                       )), |  | ||||||
|                 ); |  | ||||||
| 
 |  | ||||||
|               if (plans == null) |               if (plans == null) | ||||||
|                 return const Center(child: CircularProgressIndicator()); |                 return const Center(child: CircularProgressIndicator()); | ||||||
| 
 | 
 | ||||||
|               return PlanList(plans: plans); |               return ListView.builder( | ||||||
|  |                 itemCount: plans.length, | ||||||
|  |                 itemBuilder: (context, index) { | ||||||
|  |                   return ListTile( | ||||||
|  |                       title: Text(plans[index].days.replaceAll(',', ', ')), | ||||||
|  |                       subtitle: | ||||||
|  |                           Text(plans[index].workouts.replaceAll(',', ', ')), | ||||||
|  |                       onTap: () async { | ||||||
|  |                         await Navigator.push( | ||||||
|  |                           context, | ||||||
|  |                           MaterialPageRoute( | ||||||
|  |                             builder: (context) => EditPlanPage( | ||||||
|  |                                 plan: plans[index].toCompanion(false)), | ||||||
|  |                           ), | ||||||
|  |                         ); | ||||||
|  |                       }); | ||||||
|  |                 }, | ||||||
|  |               ); | ||||||
|             }), |             }), | ||||||
|         floatingActionButton: FloatingActionButton( |         floatingActionButton: FloatingActionButton( | ||||||
|             onPressed: () async { |             onPressed: () async { | ||||||
|  | @ -87,8 +93,8 @@ class _PlansPageState extends State<_PlansPage> { | ||||||
|                 context, |                 context, | ||||||
|                 MaterialPageRoute( |                 MaterialPageRoute( | ||||||
|                   builder: (context) => const EditPlanPage( |                   builder: (context) => const EditPlanPage( | ||||||
|                       plan: PlansCompanion( |                       plan: | ||||||
|                           days: Value(''), exercises: Value(''))), |                           PlansCompanion(days: Value(''), workouts: Value(''))), | ||||||
|                 ), |                 ), | ||||||
|               ); |               ); | ||||||
|             }, |             }, | ||||||
|  |  | ||||||
|  | @ -95,9 +95,11 @@ class _SettingsPageState extends State<_SettingsPage> { | ||||||
|                                 await file.copy(to); |                                 await file.copy(to); | ||||||
|                                 print('Migrating...'); |                                 print('Migrating...'); | ||||||
|                                 db = MyDatabase(); |                                 db = MyDatabase(); | ||||||
|                                 // ignore: invalid_use_of_protected_member, invalid_use_of_visible_for_testing_member |  | ||||||
|                                 final migrator = db.createMigrator(); |                                 final migrator = db.createMigrator(); | ||||||
|                                 await migrator.createAll(); |                                 await migrator.createAll(); | ||||||
|  |                                 await db.customInsert(''' | ||||||
|  |                                   INSERT INTO gym_sets(id, name, reps, weight, created, unit, hidden, image, sets, minutes, seconds, steps)  | ||||||
|  |                                     SELECT id, name, reps, weight, created, unit, hidden, image, sets, minutes, seconds, steps FROM sets'''); | ||||||
|                                 print('Migrated.'); |                                 print('Migrated.'); | ||||||
|                               }, |                               }, | ||||||
|                               child: const Text("Import"))); |                               child: const Text("Import"))); | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue
	
	Block a user