Files
flutter/lib/models/workouts/session.dart
Roland Geider 6a2158a397 Reload the session page in the gym mode
This makes sure that we have the current state and don't try to create a new session
which already exists
2025-12-18 11:59:26 +01:00

137 lines
4.4 KiB
Dart

/*
* This file is part of wger Workout Manager <https://github.com/wger-project>.
* Copyright (c) 2025 wger Team
*
* wger Workout Manager is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
import 'package:clock/clock.dart';
import 'package:flutter/material.dart';
import 'package:json_annotation/json_annotation.dart';
import 'package:logging/logging.dart';
import 'package:wger/helpers/consts.dart';
import 'package:wger/helpers/json.dart';
import 'package:wger/l10n/generated/app_localizations.dart';
import 'package:wger/models/workouts/log.dart';
part 'session.g.dart';
const IMPRESSION_MAP = {1: 'bad', 2: 'neutral', 3: 'good'};
@JsonSerializable()
class WorkoutSession {
final _logger = Logger('WorkoutSession');
@JsonKey(required: true)
int? id;
@JsonKey(required: true, name: 'routine')
late int? routineId;
@JsonKey(required: true, name: 'day')
int? dayId;
@JsonKey(required: true, toJson: dateToYYYYMMDD)
late DateTime date;
@JsonKey(required: true, fromJson: int.parse, toJson: numToString)
late int impression;
@JsonKey(required: false, defaultValue: '')
late String notes;
@JsonKey(required: true, name: 'time_start', toJson: timeToString, fromJson: stringToTimeNull)
late TimeOfDay? timeStart;
@JsonKey(required: true, name: 'time_end', toJson: timeToString, fromJson: stringToTimeNull)
late TimeOfDay? timeEnd;
@JsonKey(required: false, includeToJson: false, defaultValue: [])
List<Log> logs = [];
WorkoutSession({
this.id,
this.dayId,
required this.routineId,
this.impression = DEFAULT_IMPRESSION,
this.notes = '',
this.timeStart,
this.timeEnd,
this.logs = const <Log>[],
DateTime? date,
}) {
this.date = date ?? clock.now();
}
Duration? get duration {
if (timeStart == null || timeEnd == null) {
return null;
}
final now = DateTime.now();
final startDate = DateTime(now.year, now.month, now.day, timeStart!.hour, timeStart!.minute);
final endDate = DateTime(now.year, now.month, now.day, timeEnd!.hour, timeEnd!.minute);
return endDate.difference(startDate);
}
String durationTxt(BuildContext context) {
final duration = this.duration;
if (duration == null) {
return '-/-';
}
return AppLocalizations.of(
context,
).durationHoursMinutes(duration.inHours, duration.inMinutes.remainder(60));
}
String durationTxtWithStartEnd(BuildContext context) {
final duration = this.duration;
if (duration == null) {
return '-/-';
}
final startTime = MaterialLocalizations.of(context).formatTimeOfDay(timeStart!);
final endTime = MaterialLocalizations.of(context).formatTimeOfDay(timeEnd!);
return '${durationTxt(context)} ($startTime - $endTime)';
}
/// Get total volume of the session for metric and imperial units
/// (i.e. sets that have "repetitions" as units and weight in kg or lbs).
/// Other combinations such as "seconds" are ignored.
Map<String, Object> get volume {
final volumeMetric = logs.fold<double>(0, (sum, log) => sum + log.volume(metric: true));
final volumeImperial = logs.fold<double>(0, (sum, log) => sum + log.volume(metric: false));
return {'metric': volumeMetric, 'imperial': volumeImperial};
}
// Boilerplate
factory WorkoutSession.fromJson(Map<String, dynamic> json) => _$WorkoutSessionFromJson(json);
Map<String, dynamic> toJson() => _$WorkoutSessionToJson(this);
String impressionAsString(BuildContext context) {
if (impression == 1) {
return AppLocalizations.of(context).impressionBad;
} else if (impression == 2) {
return AppLocalizations.of(context).impressionNeutral;
} else if (impression == 3) {
return AppLocalizations.of(context).impressionGood;
}
_logger.warning('Unknown impression value: $impression');
return AppLocalizations.of(context).impressionGood;
}
}