Files
flutter/lib/widgets/gallery/forms.dart
Roland Geider 93a11a16e4 Recreate generated files and reformat the repo
Seems the default dart formatting rules changed recently, and this was causing
huge diffs in branches
2025-10-09 17:39:02 +02:00

219 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();
}
},
),
],
),
);
}
}