mirror of
https://github.com/wger-project/flutter.git
synced 2026-02-18 00:17:48 +01:00
The translations are not stored in a synthetic package anymore, but the files are just copied to a specific folder so it can be more easily imported Also, flutter_barcode_scanner was replaced by simple_barcode_scanner since the first hasn't been updated in 3 years and was already causing problems for some time
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();
|
|
}
|
|
},
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
}
|