2023-06-10 12:44:02 +02:00
|
|
|
import 'dart:async';
|
|
|
|
|
|
|
|
import 'package:copy_with/copy_with.dart';
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
|
|
|
import 'package:kiwi/kiwi.dart';
|
|
|
|
import 'package:logging/logging.dart';
|
|
|
|
import 'package:nc_photos/account.dart';
|
|
|
|
import 'package:nc_photos/app_localizations.dart';
|
2023-07-28 18:48:50 +02:00
|
|
|
import 'package:nc_photos/bloc_util.dart';
|
2023-06-10 12:44:02 +02:00
|
|
|
import 'package:nc_photos/controller/account_controller.dart';
|
|
|
|
import 'package:nc_photos/controller/account_pref_controller.dart';
|
|
|
|
import 'package:nc_photos/di_container.dart';
|
2023-07-17 18:43:59 +02:00
|
|
|
import 'package:nc_photos/entity/person.dart';
|
2023-07-17 09:35:45 +02:00
|
|
|
import 'package:nc_photos/entity/pref.dart';
|
2023-06-10 12:44:02 +02:00
|
|
|
import 'package:nc_photos/exception_event.dart';
|
2023-07-17 18:43:59 +02:00
|
|
|
import 'package:nc_photos/help_utils.dart' as help_util;
|
2023-06-10 12:44:02 +02:00
|
|
|
import 'package:nc_photos/k.dart' as k;
|
|
|
|
import 'package:nc_photos/snack_bar_manager.dart';
|
2023-07-17 18:43:59 +02:00
|
|
|
import 'package:nc_photos/url_launcher_util.dart';
|
2023-06-10 12:44:02 +02:00
|
|
|
import 'package:nc_photos/widget/home.dart';
|
|
|
|
import 'package:nc_photos/widget/page_visibility_mixin.dart';
|
|
|
|
import 'package:nc_photos/widget/root_picker.dart';
|
|
|
|
import 'package:nc_photos/widget/settings/settings_list_caption.dart';
|
|
|
|
import 'package:nc_photos/widget/share_folder_picker.dart';
|
|
|
|
import 'package:nc_photos/widget/simple_input_dialog.dart';
|
|
|
|
import 'package:np_codegen/np_codegen.dart';
|
2024-05-08 18:12:18 +02:00
|
|
|
import 'package:np_ui/np_ui.dart';
|
2023-06-10 12:44:02 +02:00
|
|
|
import 'package:to_string/to_string.dart';
|
|
|
|
|
|
|
|
part 'account/bloc.dart';
|
|
|
|
part 'account/state_event.dart';
|
|
|
|
part 'account_settings.g.dart';
|
|
|
|
|
|
|
|
// typedef _BlocBuilder = BlocBuilder<_Bloc, _State>;
|
|
|
|
typedef _BlocListener = BlocListener<_Bloc, _State>;
|
|
|
|
typedef _BlocSelector<T> = BlocSelector<_Bloc, _State, T>;
|
|
|
|
|
2023-07-17 18:43:59 +02:00
|
|
|
enum AccountSettingsOption {
|
|
|
|
personProvider,
|
|
|
|
}
|
|
|
|
|
|
|
|
class AccountSettingsArguments {
|
|
|
|
const AccountSettingsArguments({
|
|
|
|
this.highlight,
|
|
|
|
});
|
|
|
|
|
|
|
|
final AccountSettingsOption? highlight;
|
|
|
|
}
|
|
|
|
|
2023-06-10 12:44:02 +02:00
|
|
|
class AccountSettings extends StatelessWidget {
|
2023-08-14 17:07:52 +02:00
|
|
|
static const routeName = "/settings/account";
|
2023-06-10 12:44:02 +02:00
|
|
|
|
2023-07-17 18:43:59 +02:00
|
|
|
static Route buildRoute(AccountSettingsArguments? args) => MaterialPageRoute(
|
|
|
|
builder: (_) => AccountSettings.fromArgs(args),
|
2023-06-10 12:44:02 +02:00
|
|
|
);
|
|
|
|
|
2023-07-17 18:43:59 +02:00
|
|
|
const AccountSettings({
|
|
|
|
super.key,
|
|
|
|
this.highlight,
|
|
|
|
});
|
|
|
|
|
|
|
|
AccountSettings.fromArgs(AccountSettingsArguments? args, {Key? key})
|
|
|
|
: this(
|
|
|
|
key: key,
|
|
|
|
highlight: args?.highlight,
|
|
|
|
);
|
2023-06-10 12:44:02 +02:00
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
final accountController = context.read<AccountController>();
|
|
|
|
return BlocProvider(
|
|
|
|
create: (_) => _Bloc(
|
|
|
|
container: KiwiContainer().resolve(),
|
|
|
|
account: accountController.account,
|
|
|
|
accountPrefController: accountController.accountPrefController,
|
2023-07-17 18:43:59 +02:00
|
|
|
highlight: highlight,
|
2023-06-10 12:44:02 +02:00
|
|
|
),
|
|
|
|
child: const _WrappedAccountSettings(),
|
|
|
|
);
|
|
|
|
}
|
2023-07-17 18:43:59 +02:00
|
|
|
|
|
|
|
final AccountSettingsOption? highlight;
|
2023-06-10 12:44:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
class _WrappedAccountSettings extends StatefulWidget {
|
|
|
|
const _WrappedAccountSettings();
|
|
|
|
|
|
|
|
@override
|
2023-07-23 07:08:30 +02:00
|
|
|
State<StatefulWidget> createState() => _WrappedAccountSettingsState();
|
2023-06-10 12:44:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@npLog
|
2023-07-23 07:08:30 +02:00
|
|
|
class _WrappedAccountSettingsState extends State<_WrappedAccountSettings>
|
|
|
|
with RouteAware, PageVisibilityMixin, TickerProviderStateMixin {
|
2023-07-22 16:26:51 +02:00
|
|
|
@override
|
|
|
|
void initState() {
|
|
|
|
super.initState();
|
2024-06-01 14:44:11 +02:00
|
|
|
_accountController = context.read();
|
2023-07-23 07:08:30 +02:00
|
|
|
|
|
|
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
|
|
|
_animationController.repeat(reverse: true);
|
|
|
|
});
|
2023-07-22 16:26:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void dispose() {
|
|
|
|
if (_bloc.state.shouldResync &&
|
|
|
|
_bloc.state.personProvider != PersonProvider.none) {
|
|
|
|
_log.fine("[dispose] Requesting to resync account");
|
2024-01-17 18:17:53 +01:00
|
|
|
_accountController.syncController.requestResync(
|
|
|
|
account: _bloc.state.account,
|
2024-06-01 14:44:11 +02:00
|
|
|
filesController: _accountController.filesController,
|
|
|
|
personsController: _accountController.personsController,
|
2024-01-17 18:17:53 +01:00
|
|
|
personProvider: _bloc.state.personProvider,
|
|
|
|
);
|
2023-07-22 16:26:51 +02:00
|
|
|
}
|
2023-07-23 07:08:30 +02:00
|
|
|
_animationController.dispose();
|
2023-07-22 16:26:51 +02:00
|
|
|
super.dispose();
|
|
|
|
}
|
|
|
|
|
2023-06-10 12:44:02 +02:00
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return Scaffold(
|
|
|
|
body: MultiBlocListener(
|
|
|
|
listeners: [
|
|
|
|
_BlocListener(
|
|
|
|
listenWhen: (previous, current) => previous.error != current.error,
|
|
|
|
listener: (context, state) {
|
|
|
|
if (state.error != null && isPageVisible()) {
|
|
|
|
if (state.error is _AccountConflictError) {
|
2024-06-19 08:58:08 +02:00
|
|
|
SnackBarManager().showSnackBar(SnackBar(
|
|
|
|
content: Text(
|
|
|
|
L10n.global().editAccountConflictFailureNotification),
|
|
|
|
duration: k.snackBarDurationNormal,
|
|
|
|
));
|
2023-06-10 12:44:02 +02:00
|
|
|
} else if (state.error is _WritePrefError) {
|
2024-06-19 08:58:08 +02:00
|
|
|
SnackBarManager().showSnackBar(SnackBar(
|
|
|
|
content:
|
|
|
|
Text(L10n.global().writePreferenceFailureNotification),
|
|
|
|
duration: k.snackBarDurationNormal,
|
|
|
|
));
|
2023-06-10 12:44:02 +02:00
|
|
|
} else {
|
2024-06-19 08:58:08 +02:00
|
|
|
SnackBarManager()
|
|
|
|
.showSnackBarForException(state.error!.error);
|
2023-06-10 12:44:02 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
),
|
|
|
|
],
|
2024-05-21 17:39:24 +02:00
|
|
|
child: _BlocSelector<bool>(
|
|
|
|
selector: (state) => state.shouldReload,
|
|
|
|
builder: (_, shouldReload) => PopScope(
|
|
|
|
canPop: !shouldReload,
|
|
|
|
child: CustomScrollView(
|
|
|
|
slivers: [
|
|
|
|
SliverAppBar(
|
|
|
|
pinned: true,
|
|
|
|
title: Text(L10n.global().settingsAccountTitle),
|
|
|
|
leading:
|
|
|
|
shouldReload ? const _DoneButton() : const BackButton(),
|
2023-06-10 12:44:02 +02:00
|
|
|
),
|
2024-05-21 17:39:24 +02:00
|
|
|
SliverList(
|
|
|
|
delegate: SliverChildListDelegate(
|
|
|
|
[
|
|
|
|
_BlocSelector<String?>(
|
|
|
|
selector: (state) => state.label,
|
|
|
|
builder: (context, state) => ListTile(
|
|
|
|
title: Text(L10n.global().settingsAccountLabelTitle),
|
|
|
|
subtitle: Text(state ??
|
|
|
|
L10n.global().settingsAccountLabelDescription),
|
|
|
|
onTap: () => _onLabelPressed(context),
|
|
|
|
),
|
2023-06-10 12:44:02 +02:00
|
|
|
),
|
2024-05-21 17:39:24 +02:00
|
|
|
_BlocSelector<Account>(
|
|
|
|
selector: (state) => state.account,
|
|
|
|
builder: (context, state) => ListTile(
|
|
|
|
title:
|
|
|
|
Text(L10n.global().settingsIncludedFoldersTitle),
|
|
|
|
subtitle:
|
|
|
|
Text(state.roots.map((e) => "/$e").join("; ")),
|
|
|
|
onTap: () => _onIncludedFoldersPressed(context),
|
|
|
|
),
|
2023-06-10 12:44:02 +02:00
|
|
|
),
|
2024-05-21 17:39:24 +02:00
|
|
|
_BlocSelector<String>(
|
|
|
|
selector: (state) => state.shareFolder,
|
|
|
|
builder: (context, state) => ListTile(
|
|
|
|
title: Text(L10n.global().settingsShareFolderTitle),
|
|
|
|
subtitle: Text("/$state"),
|
|
|
|
onTap: () => _onShareFolderPressed(context),
|
|
|
|
),
|
2023-06-10 12:44:02 +02:00
|
|
|
),
|
2024-05-21 17:39:24 +02:00
|
|
|
SettingsListCaption(
|
|
|
|
label: L10n.global().settingsServerAppSectionTitle,
|
|
|
|
),
|
|
|
|
_BlocSelector<PersonProvider>(
|
|
|
|
selector: (state) => state.personProvider,
|
|
|
|
builder: (context, state) {
|
|
|
|
if (_bloc.highlight ==
|
|
|
|
AccountSettingsOption.personProvider) {
|
|
|
|
return AnimatedBuilder(
|
|
|
|
animation: _highlightAnimation,
|
|
|
|
builder: (context, child) => ListTile(
|
|
|
|
title: Text(
|
|
|
|
L10n.global().settingsPersonProviderTitle),
|
|
|
|
subtitle: Text(state.toUserString()),
|
|
|
|
onTap: () => _onPersonProviderPressed(context),
|
|
|
|
tileColor: _highlightAnimation.value,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
return ListTile(
|
2023-07-23 07:08:30 +02:00
|
|
|
title: Text(
|
|
|
|
L10n.global().settingsPersonProviderTitle),
|
|
|
|
subtitle: Text(state.toUserString()),
|
|
|
|
onTap: () => _onPersonProviderPressed(context),
|
2024-05-21 17:39:24 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
2023-06-10 12:44:02 +02:00
|
|
|
),
|
2024-05-21 17:39:24 +02:00
|
|
|
],
|
|
|
|
),
|
2023-06-10 12:44:02 +02:00
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _onLabelPressed(BuildContext context) async {
|
|
|
|
final result = await showDialog<String>(
|
|
|
|
context: context,
|
|
|
|
builder: (context) => SimpleInputDialog(
|
|
|
|
titleText: L10n.global().settingsAccountLabelTitle,
|
|
|
|
buttonText: MaterialLocalizations.of(context).okButtonLabel,
|
2023-07-17 18:43:59 +02:00
|
|
|
initialText: _bloc.state.label ?? "",
|
2023-06-10 12:44:02 +02:00
|
|
|
),
|
|
|
|
);
|
|
|
|
if (!context.mounted || result == null) {
|
|
|
|
return;
|
|
|
|
}
|
2023-07-17 18:43:59 +02:00
|
|
|
_bloc.add(_SetLabel(result.isEmpty ? null : result));
|
2023-06-10 12:44:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _onIncludedFoldersPressed(BuildContext context) async {
|
|
|
|
final result = await Navigator.of(context).pushNamed<Account>(
|
|
|
|
RootPicker.routeName,
|
2023-07-17 18:43:59 +02:00
|
|
|
arguments: RootPickerArguments(_bloc.state.account),
|
2023-06-10 12:44:02 +02:00
|
|
|
);
|
|
|
|
if (result == null) {
|
|
|
|
return;
|
|
|
|
}
|
2023-07-17 18:43:59 +02:00
|
|
|
if (result == _bloc.state.account) {
|
2023-06-10 12:44:02 +02:00
|
|
|
// no changes, do nothing
|
|
|
|
_log.fine("[_onIncludedFoldersPressed] No changes");
|
|
|
|
return;
|
|
|
|
}
|
2023-07-17 18:43:59 +02:00
|
|
|
_bloc.add(_SetAccount(result));
|
2023-06-10 12:44:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _onShareFolderPressed(BuildContext context) async {
|
|
|
|
final result = await showDialog<String>(
|
|
|
|
context: context,
|
|
|
|
builder: (_) => _ShareFolderDialog(
|
2023-07-17 18:43:59 +02:00
|
|
|
account: _bloc.state.account,
|
|
|
|
initialValue: _bloc.state.shareFolder,
|
2023-06-10 12:44:02 +02:00
|
|
|
),
|
|
|
|
);
|
|
|
|
if (!context.mounted || result == null) {
|
|
|
|
return;
|
|
|
|
}
|
2023-07-17 18:43:59 +02:00
|
|
|
_bloc.add(_SetShareFolder(result));
|
2023-06-10 12:44:02 +02:00
|
|
|
}
|
|
|
|
|
2023-07-17 18:43:59 +02:00
|
|
|
Future<void> _onPersonProviderPressed(BuildContext context) async {
|
|
|
|
final result = await showDialog<PersonProvider>(
|
|
|
|
context: context,
|
|
|
|
builder: (_) => _PersonProviderDialog(
|
|
|
|
initialValue: _bloc.state.personProvider,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
if (!context.mounted || result == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_bloc.add(_SetPersonProvider(result));
|
2023-06-10 12:44:02 +02:00
|
|
|
}
|
2023-07-17 18:43:59 +02:00
|
|
|
|
|
|
|
late final _bloc = context.read<_Bloc>();
|
2023-07-22 16:26:51 +02:00
|
|
|
late final AccountController _accountController;
|
2023-07-23 07:08:30 +02:00
|
|
|
|
|
|
|
late final _animationController = AnimationController(
|
|
|
|
vsync: this,
|
|
|
|
duration: k.settingsHighlightDuration,
|
|
|
|
);
|
|
|
|
late final _highlightAnimation = ColorTween(
|
|
|
|
end: Theme.of(context).colorScheme.primaryContainer,
|
|
|
|
).animate(_animationController);
|
2023-06-10 12:44:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
class _DoneButton extends StatelessWidget {
|
|
|
|
const _DoneButton();
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return IconButton(
|
|
|
|
icon: const Icon(Icons.check),
|
|
|
|
tooltip: L10n.global().doneButtonTooltip,
|
|
|
|
onPressed: () {
|
|
|
|
final newAccount = context.read<_Bloc>().state.account;
|
|
|
|
context.read<AccountController>().setCurrentAccount(newAccount);
|
|
|
|
Navigator.of(context).pushNamedAndRemoveUntil(
|
|
|
|
Home.routeName,
|
|
|
|
(_) => false,
|
|
|
|
arguments: HomeArguments(newAccount),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class _ShareFolderDialog extends StatefulWidget {
|
|
|
|
const _ShareFolderDialog({
|
|
|
|
required this.account,
|
|
|
|
required this.initialValue,
|
|
|
|
});
|
|
|
|
|
|
|
|
@override
|
|
|
|
State<StatefulWidget> createState() => _ShareFolderDialogState();
|
|
|
|
|
|
|
|
final Account account;
|
|
|
|
final String initialValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
class _ShareFolderDialogState extends State<_ShareFolderDialog> {
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return AlertDialog(
|
|
|
|
title: Text(L10n.global().settingsShareFolderDialogTitle),
|
|
|
|
content: Form(
|
|
|
|
key: _formKey,
|
|
|
|
child: Column(
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
children: [
|
|
|
|
Text(L10n.global().settingsShareFolderDialogDescription),
|
|
|
|
const SizedBox(height: 8),
|
|
|
|
InkWell(
|
|
|
|
onTap: _onTextFieldPressed,
|
|
|
|
child: TextFormField(
|
2024-06-05 19:11:47 +02:00
|
|
|
style: TextStyle(
|
|
|
|
color: Theme.of(context).textTheme.bodyMedium!.color,
|
|
|
|
),
|
2023-06-10 12:44:02 +02:00
|
|
|
enabled: false,
|
|
|
|
controller: _controller,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
actions: [
|
|
|
|
TextButton(
|
|
|
|
onPressed: _onOkPressed,
|
|
|
|
child: Text(MaterialLocalizations.of(context).okButtonLabel),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _onTextFieldPressed() async {
|
|
|
|
final pick = await Navigator.of(context).pushNamed<String>(
|
|
|
|
ShareFolderPicker.routeName,
|
|
|
|
arguments: ShareFolderPickerArguments(widget.account, _path),
|
|
|
|
);
|
|
|
|
if (pick != null) {
|
|
|
|
_path = pick;
|
|
|
|
_controller.text = "/$pick";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _onOkPressed() {
|
|
|
|
Navigator.of(context).pop(_path);
|
|
|
|
}
|
|
|
|
|
|
|
|
final _formKey = GlobalKey<FormState>();
|
|
|
|
late final _controller =
|
|
|
|
TextEditingController(text: "/${widget.initialValue}");
|
|
|
|
late String _path = widget.initialValue;
|
|
|
|
}
|
2023-07-17 18:43:59 +02:00
|
|
|
|
|
|
|
@npLog
|
|
|
|
class _PersonProviderDialog extends StatelessWidget {
|
|
|
|
const _PersonProviderDialog({
|
|
|
|
required this.initialValue,
|
|
|
|
});
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return FancyOptionPicker(
|
|
|
|
title: Row(
|
|
|
|
children: [
|
|
|
|
Expanded(
|
|
|
|
child: Text(L10n.global().settingsPersonProviderTitle),
|
|
|
|
),
|
|
|
|
IconButton(
|
|
|
|
onPressed: () {
|
|
|
|
launch(help_util.peopleUrl);
|
|
|
|
},
|
|
|
|
tooltip: L10n.global().helpTooltip,
|
|
|
|
icon: const Icon(Icons.help_outline),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
items: PersonProvider.values
|
|
|
|
.map((provider) => FancyOptionPickerItem(
|
|
|
|
label: provider.toUserString(),
|
|
|
|
isSelected: provider == initialValue,
|
|
|
|
onSelect: () {
|
|
|
|
_log.info("[build] Set provider: ${provider.toUserString()}");
|
|
|
|
Navigator.of(context).pop(provider);
|
|
|
|
},
|
|
|
|
))
|
|
|
|
.toList(),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
final PersonProvider initialValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
extension on PersonProvider {
|
|
|
|
String toUserString() {
|
|
|
|
switch (this) {
|
|
|
|
case PersonProvider.none:
|
|
|
|
return "n/a";
|
|
|
|
case PersonProvider.faceRecognition:
|
|
|
|
return "Face Recognition";
|
|
|
|
case PersonProvider.recognize:
|
|
|
|
return "Recognize";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|