import 'dart:io'; import 'dart:ui'; import 'package:croppy/croppy.dart'; import 'package:easy_localization/easy_localization.dart'; import 'package:flutter/cupertino.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:gap/gap.dart'; import 'package:image_picker/image_picker.dart'; import 'package:material_symbols_icons/symbols.dart'; import 'package:path/path.dart' show basename; import 'package:provider/provider.dart'; import 'package:styled_widget/styled_widget.dart'; import 'package:surface/providers/sn_attachment.dart'; import 'package:surface/providers/sn_network.dart'; import 'package:surface/providers/userinfo.dart'; import 'package:surface/widgets/account/account_image.dart'; import 'package:surface/widgets/dialog.dart'; import 'package:surface/widgets/loading_indicator.dart'; import 'package:surface/widgets/universal_image.dart'; class ProfileEditScreen extends StatefulWidget { const ProfileEditScreen({super.key}); @override State<ProfileEditScreen> createState() => _ProfileEditScreenState(); } class _ProfileEditScreenState extends State<ProfileEditScreen> { final _imagePicker = ImagePicker(); final _usernameController = TextEditingController(); final _nicknameController = TextEditingController(); final _firstNameController = TextEditingController(); final _lastNameController = TextEditingController(); final _descriptionController = TextEditingController(); final _birthdayController = TextEditingController(); String? _avatar; String? _banner; DateTime? _birthday; bool _isBusy = false; static const _kDateFormat = 'y/M/d'; void _syncWidget() async { final ua = context.read<UserProvider>(); final prof = ua.user!; _usernameController.text = prof.name; _nicknameController.text = prof.nick; _descriptionController.text = prof.description; _firstNameController.text = prof.profile!.firstName; _lastNameController.text = prof.profile!.lastName; _avatar = prof.avatar; _banner = prof.banner; if (prof.profile!.birthday != null) { _birthdayController.text = DateFormat(_kDateFormat).format( prof.profile!.birthday!.toLocal(), ); } } void _selectBirthday() async { await showCupertinoModalPopup<DateTime?>( context: context, builder: (BuildContext context) => Container( height: 216, padding: const EdgeInsets.only(top: 6.0), margin: EdgeInsets.only( bottom: MediaQuery.of(context).viewInsets.bottom, ), color: Theme.of(context).colorScheme.surface, child: SafeArea( top: false, child: CupertinoDatePicker( initialDateTime: _birthday?.toLocal(), mode: CupertinoDatePickerMode.date, use24hFormat: true, onDateTimeChanged: (DateTime newDate) { setState(() { _birthday = newDate; _birthdayController.text = DateFormat(_kDateFormat).format(_birthday!); }); }, ), ), ), ); } Future<void> _updateImage(String place) async { final image = await _imagePicker.pickImage(source: ImageSource.gallery); if (image == null) return; if (!mounted) return; final ImageProvider imageProvider = kIsWeb ? NetworkImage(image.path) : FileImage(File(image.path)); final aspectRatios = place == 'banner' ? [CropAspectRatio(width: 16, height: 7)] : [CropAspectRatio(width: 1, height: 1)]; final result = (!kIsWeb && (Platform.isIOS || Platform.isMacOS)) ? await showCupertinoImageCropper( // ignore: use_build_context_synchronously context, allowedAspectRatios: aspectRatios, imageProvider: imageProvider, ) : await showMaterialImageCropper( // ignore: use_build_context_synchronously context, allowedAspectRatios: aspectRatios, imageProvider: imageProvider, ); if (result == null) return; if (!mounted) return; final attach = context.read<SnAttachmentProvider>(); setState(() => _isBusy = true); final rawBytes = (await result.uiImage.toByteData(format: ImageByteFormat.png))! .buffer .asUint8List(); try { final attachment = await attach.directUploadOne( rawBytes, basename(image.path), 'avatar', null, mimetype: 'image/png', ); if (!mounted) return; final sn = context.read<SnNetworkProvider>(); await sn.client.put( '/cgi/id/users/me/$place', data: {'attachment': attachment.rid}, ); if (!mounted) return; final ua = context.read<UserProvider>(); await ua.refreshUser(); if (!mounted) return; context.showSnackbar('accountProfileEditApplied'.tr()); _syncWidget(); } catch (err) { if (!mounted) return; context.showErrorDialog(err); } finally { setState(() => _isBusy = false); } } void _updateUserInfo() async { setState(() => _isBusy = true); final sn = context.read<SnNetworkProvider>(); try { await sn.client.put( '/cgi/id/users/me', data: { 'nick': _nicknameController.value.text, 'description': _descriptionController.value.text, 'first_name': _firstNameController.value.text, 'last_name': _lastNameController.value.text, 'birthday': _birthday?.toUtc().toIso8601String(), }, ); if (!mounted) return; final ua = context.read<UserProvider>(); await ua.refreshUser(); if (!mounted) return; context.showSnackbar('accountProfileEditApplied'.tr()); _syncWidget(); } catch (err) { context.showErrorDialog(err); } finally { setState(() => _isBusy = false); } } @override void initState() { super.initState(); _syncWidget(); } @override void dispose() { _usernameController.dispose(); _nicknameController.dispose(); _firstNameController.dispose(); _lastNameController.dispose(); _descriptionController.dispose(); _birthdayController.dispose(); super.dispose(); } @override Widget build(BuildContext context) { const double padding = 24; final sn = context.read<SnNetworkProvider>(); return SingleChildScrollView( child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ LoadingIndicator(isActive: _isBusy), const Gap(24), Stack( clipBehavior: Clip.none, children: [ Material( elevation: 0, child: InkWell( child: ClipRRect( borderRadius: const BorderRadius.all(Radius.circular(8)), child: AspectRatio( aspectRatio: 16 / 9, child: Container( color: Theme.of(context).colorScheme.surfaceContainerHigh, child: _banner != null ? AutoResizeUniversalImage( sn.getAttachmentUrl(_banner!), fit: BoxFit.cover, ) : const SizedBox.shrink(), ), ), ), onTap: () { _updateImage('banner'); }, ), ), Positioned( bottom: -28, left: 16, child: Material( elevation: 2, borderRadius: const BorderRadius.all(Radius.circular(40)), child: InkWell( child: AccountImage(content: _avatar, radius: 40), onTap: () { _updateImage('avatar'); }, ), ), ), ], ).padding(horizontal: padding), const Gap(8 + 28), Column( children: [ TextField( readOnly: true, controller: _usernameController, decoration: InputDecoration( border: const UnderlineInputBorder(), labelText: 'fieldUsername'.tr(), helperText: 'fieldUsernameCannotEditHint'.tr(), ), ), const Gap(4), TextField( controller: _nicknameController, decoration: InputDecoration( border: const UnderlineInputBorder(), labelText: 'fieldNickname'.tr(), ), ), const Gap(4), Row( children: [ Flexible( flex: 1, child: TextField( controller: _firstNameController, decoration: InputDecoration( border: const UnderlineInputBorder(), labelText: 'fieldFirstName'.tr(), ), ), ), const Gap(8), Flexible( flex: 1, child: TextField( controller: _lastNameController, decoration: InputDecoration( border: const UnderlineInputBorder(), labelText: 'fieldLastName'.tr(), ), ), ), ], ), const Gap(4), TextField( controller: _descriptionController, keyboardType: TextInputType.multiline, maxLines: null, minLines: 3, decoration: InputDecoration( border: const UnderlineInputBorder(), labelText: 'fieldDescription'.tr(), ), ), const Gap(4), TextField( controller: _birthdayController, readOnly: true, decoration: InputDecoration( border: const UnderlineInputBorder(), labelText: 'fieldBirthday'.tr(), ), onTap: () => _selectBirthday(), ), ], ).padding(horizontal: padding + 8), const Gap(12), Row( mainAxisAlignment: MainAxisAlignment.end, children: [ ElevatedButton.icon( onPressed: _isBusy ? null : _updateUserInfo, icon: const Icon(Symbols.save), label: Text('apply').tr(), ), ], ).padding(horizontal: padding), ], ), ); } }