mirror of
https://github.com/wger-project/flutter.git
synced 2026-02-18 00:17:48 +01:00
# Conflicts: # Gemfile.lock # android/settings.gradle # fastlane/report.xml # integration_test/1_dashboard.dart # integration_test/2_workout.dart # integration_test/3_gym_mode.dart # integration_test/5_nutritional_plan.dart # integration_test/6_weight.dart # lib/helpers/i18n.dart # lib/helpers/ui.dart # lib/main.dart # lib/models/nutrition/meal.g.dart # lib/models/nutrition/nutritional_plan.dart # lib/providers/auth.dart # lib/providers/nutrition.dart # lib/screens/add_exercise_screen.dart # lib/screens/auth_screen.dart # lib/screens/dashboard.dart # lib/screens/exercises_screen.dart # lib/screens/home_tabs_screen.dart # lib/screens/log_meal_screen.dart # lib/screens/log_meals_screen.dart # lib/screens/nutritional_plan_screen.dart # lib/screens/routine_list_screen.dart # lib/screens/workout_plan_screen.dart # lib/widgets/add_exercise/steps/step1basics.dart # lib/widgets/add_exercise/steps/step5images.dart # lib/widgets/add_exercise/steps/step_2_variations.dart # lib/widgets/add_exercise/steps/step_3_description.dart # lib/widgets/add_exercise/steps/step_4_translations.dart # lib/widgets/core/about.dart # lib/widgets/core/settings.dart # lib/widgets/dashboard/calendar.dart # lib/widgets/dashboard/widgets.dart # lib/widgets/exercises/exercises.dart # lib/widgets/gallery/overview.dart # lib/widgets/measurements/categories_card.dart # lib/widgets/measurements/charts.dart # lib/widgets/measurements/entries.dart # lib/widgets/measurements/forms.dart # lib/widgets/measurements/helpers.dart # lib/widgets/nutrition/charts.dart # lib/widgets/nutrition/forms.dart # lib/widgets/nutrition/helpers.dart # lib/widgets/nutrition/meal.dart # lib/widgets/nutrition/nutritional_diary_table.dart # lib/widgets/nutrition/nutritional_plan_detail.dart # lib/widgets/nutrition/nutritional_plans_list.dart # lib/widgets/nutrition/widgets.dart # lib/widgets/routines/charts.dart # lib/widgets/routines/workout_logs.dart # lib/widgets/weight/forms.dart # lib/widgets/weight/weight_overview.dart # lib/widgets/workouts/app_bar.dart # lib/widgets/workouts/day.dart # lib/widgets/workouts/forms.dart # lib/widgets/workouts/gym_mode.dart # lib/widgets/workouts/workout_plan_detail.dart # lib/widgets/workouts/workout_plans_list.dart # linux/flutter/generated_plugin_registrant.cc # linux/flutter/generated_plugins.cmake # pubspec.lock # pubspec.yaml # test/auth/auth_screen_test.dart # test/core/settings_test.dart # test/core/settings_test.mocks.dart # test/nutrition/nutritional_meal_form_test.mocks.dart # test/nutrition/nutritional_meal_item_form_test.dart # test/nutrition/nutritional_plan_form_test.mocks.dart # test/nutrition/nutritional_plan_screen_test.dart # test/nutrition/nutritional_plans_screen_test.dart # test/routine/repetition_unit_form_widget_test.dart # test/routine/routine_screen_test.dart # test/routine/routines_screen_test.dart # test/routine/weight_unit_form_widget_test.dart # test/workout/gym_mode_screen_test.dart # test/workout/workout_day_form_test.dart # test/workout/workout_form_test.dart # test/workout/workout_set_form_test.dart
215 lines
6.9 KiB
Dart
215 lines
6.9 KiB
Dart
/*
|
|
* This file is part of wger Workout Manager <https://github.com/wger-project>.
|
|
* Copyright (C) 2020, 2021 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.
|
|
*
|
|
* wger Workout Manager 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 'dart:io';
|
|
|
|
import 'package:flutter/material.dart';
|
|
import 'package:image_picker/image_picker.dart';
|
|
import 'package:provider/provider.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/gallery/image.dart' as gallery;
|
|
import 'package:wger/providers/gallery.dart';
|
|
|
|
class ImageForm extends StatefulWidget {
|
|
late final gallery.Image _image;
|
|
|
|
ImageForm([gallery.Image? image]) {
|
|
_image = image ?? gallery.Image.emtpy();
|
|
}
|
|
|
|
@override
|
|
_ImageFormState createState() => _ImageFormState();
|
|
}
|
|
|
|
class _ImageFormState extends State<ImageForm> {
|
|
final _form = GlobalKey<FormState>();
|
|
|
|
XFile? _file;
|
|
|
|
final dateController = TextEditingController();
|
|
final TextEditingController descriptionController = TextEditingController();
|
|
|
|
@override
|
|
void dispose() {
|
|
dateController.dispose();
|
|
descriptionController.dispose();
|
|
super.dispose();
|
|
}
|
|
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
|
|
dateController.text = dateToYYYYMMDD(widget._image.date)!;
|
|
descriptionController.text = widget._image.description;
|
|
}
|
|
|
|
void _showPicker(ImageSource source) async {
|
|
final picker = ImagePicker();
|
|
final file = await picker.pickImage(source: source);
|
|
|
|
setState(() {
|
|
_file = file;
|
|
});
|
|
}
|
|
|
|
/// Returns widget with current picture, depending on whether the user is
|
|
/// editing an existing entry or adding a new one. A text message is shown if
|
|
/// neither is available
|
|
Widget getPicture() {
|
|
// An image file was selected, use it
|
|
if (_file != null) {
|
|
return Image(image: FileImage(File(_file!.path)));
|
|
}
|
|
|
|
// We are editing an existing entry
|
|
if (widget._image.url != null) {
|
|
return Image.network(widget._image.url!);
|
|
}
|
|
|
|
// No picture available, show a message to the user
|
|
return Column(
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
children: [
|
|
Text(AppLocalizations.of(context).selectImage),
|
|
const SizedBox(height: 8),
|
|
const Icon(Icons.photo_camera),
|
|
],
|
|
);
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return Form(
|
|
key: _form,
|
|
child: Column(
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
Expanded(
|
|
child: GestureDetector(
|
|
onTap: () async {
|
|
showModalBottomSheet(
|
|
context: context,
|
|
builder: (context) {
|
|
return SizedBox(
|
|
height: 150,
|
|
child: Column(
|
|
children: [
|
|
ListTile(
|
|
onTap: () {
|
|
Navigator.of(context).pop();
|
|
_showPicker(ImageSource.camera);
|
|
},
|
|
leading: const Icon(Icons.photo_camera),
|
|
title: Text(AppLocalizations.of(context).takePicture),
|
|
),
|
|
ListTile(
|
|
onTap: () {
|
|
Navigator.of(context).pop();
|
|
_showPicker(ImageSource.gallery);
|
|
},
|
|
leading: const Icon(Icons.photo_library),
|
|
title: Text(AppLocalizations.of(context).chooseFromLibrary),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
},
|
|
);
|
|
},
|
|
child: getPicture(),
|
|
),
|
|
),
|
|
TextFormField(
|
|
key: const Key('field-date'),
|
|
decoration: InputDecoration(
|
|
labelText: AppLocalizations.of(context).date,
|
|
suffixIcon: const Icon(Icons.calendar_today),
|
|
),
|
|
readOnly: true,
|
|
// Stop keyboard from appearing
|
|
controller: dateController,
|
|
onTap: () async {
|
|
// Stop keyboard from appearing
|
|
FocusScope.of(context).requestFocus(FocusNode());
|
|
|
|
// Show Date Picker Here
|
|
final pickedDate = await showDatePicker(
|
|
context: context,
|
|
initialDate: widget._image.date,
|
|
firstDate: DateTime(DateTime.now().year - 10),
|
|
lastDate: DateTime.now(),
|
|
);
|
|
|
|
dateController.text = dateToYYYYMMDD(pickedDate)!;
|
|
},
|
|
onSaved: (newValue) {
|
|
widget._image.date = DateTime.parse(newValue!);
|
|
},
|
|
validator: (value) {
|
|
if (widget._image.id == null && _file == null) {
|
|
return AppLocalizations.of(context).selectImage;
|
|
}
|
|
|
|
return null;
|
|
},
|
|
),
|
|
TextFormField(
|
|
key: const Key('field-description'),
|
|
decoration: InputDecoration(labelText: AppLocalizations.of(context).description),
|
|
minLines: 3,
|
|
maxLines: 10,
|
|
controller: descriptionController,
|
|
onSaved: (newValue) {
|
|
widget._image.description = newValue!;
|
|
},
|
|
),
|
|
ElevatedButton(
|
|
key: const Key(SUBMIT_BUTTON_KEY_NAME),
|
|
child: Text(AppLocalizations.of(context).save),
|
|
onPressed: () async {
|
|
// Validate and save
|
|
final isValid = _form.currentState!.validate();
|
|
if (!isValid) {
|
|
return;
|
|
}
|
|
_form.currentState!.save();
|
|
|
|
if (widget._image.id == null) {
|
|
Provider.of<GalleryProvider>(
|
|
context,
|
|
listen: false,
|
|
).addImage(widget._image, _file!);
|
|
Navigator.of(context).pop();
|
|
} else {
|
|
Provider.of<GalleryProvider>(
|
|
context,
|
|
listen: false,
|
|
).editImage(widget._image, _file);
|
|
Navigator.of(context).pop();
|
|
}
|
|
},
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
}
|