2021-08-07 22:34:44 +02:00
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
2021-11-19 19:29:56 +01:00
|
|
|
import 'package:flutter_typeahead/flutter_typeahead.dart';
|
2021-12-03 18:31:27 +01:00
|
|
|
import 'package:kiwi/kiwi.dart';
|
2021-08-07 22:34:44 +02:00
|
|
|
import 'package:logging/logging.dart';
|
2021-11-25 10:47:49 +01:00
|
|
|
import 'package:mutex/mutex.dart';
|
2021-08-07 22:34:44 +02:00
|
|
|
import 'package:nc_photos/account.dart';
|
|
|
|
import 'package:nc_photos/app_localizations.dart';
|
2021-11-19 19:29:56 +01:00
|
|
|
import 'package:nc_photos/async_util.dart' as async_util;
|
2021-08-07 22:34:44 +02:00
|
|
|
import 'package:nc_photos/bloc/list_sharee.dart';
|
2021-11-19 19:29:56 +01:00
|
|
|
import 'package:nc_photos/bloc/search_suggestion.dart';
|
2021-11-12 22:13:02 +01:00
|
|
|
import 'package:nc_photos/ci_string.dart';
|
2021-12-03 18:31:27 +01:00
|
|
|
import 'package:nc_photos/di_container.dart';
|
2021-10-18 12:46:06 +02:00
|
|
|
import 'package:nc_photos/entity/album.dart';
|
2021-08-07 22:34:44 +02:00
|
|
|
import 'package:nc_photos/entity/sharee.dart';
|
|
|
|
import 'package:nc_photos/exception_util.dart' as exception_util;
|
|
|
|
import 'package:nc_photos/k.dart' as k;
|
|
|
|
import 'package:nc_photos/snack_bar_manager.dart';
|
2021-10-18 12:46:06 +02:00
|
|
|
import 'package:nc_photos/use_case/share_album_with_user.dart';
|
|
|
|
import 'package:nc_photos/use_case/unshare_album_with_user.dart';
|
|
|
|
import 'package:nc_photos/widget/album_share_outlier_browser.dart';
|
2022-01-28 21:06:19 +01:00
|
|
|
import 'package:nc_photos/widget/dialog_scaffold.dart';
|
2021-08-07 22:34:44 +02:00
|
|
|
|
|
|
|
class ShareAlbumDialog extends StatefulWidget {
|
2021-10-18 12:46:06 +02:00
|
|
|
ShareAlbumDialog({
|
2021-08-07 22:34:44 +02:00
|
|
|
Key? key,
|
|
|
|
required this.account,
|
2021-10-18 12:46:06 +02:00
|
|
|
required this.album,
|
|
|
|
}) : assert(album.albumFile != null),
|
|
|
|
super(key: key);
|
2021-08-07 22:34:44 +02:00
|
|
|
|
|
|
|
@override
|
|
|
|
createState() => _ShareAlbumDialogState();
|
|
|
|
|
|
|
|
final Account account;
|
2021-10-18 12:46:06 +02:00
|
|
|
final Album album;
|
2021-08-07 22:34:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
class _ShareAlbumDialogState extends State<ShareAlbumDialog> {
|
2022-07-05 22:20:24 +02:00
|
|
|
_ShareAlbumDialogState() {
|
|
|
|
final c = KiwiContainer().resolve<DiContainer>();
|
|
|
|
assert(require(c));
|
|
|
|
_c = c;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool require(DiContainer c) =>
|
|
|
|
DiContainer.has(c, DiType.albumRepo) &&
|
|
|
|
DiContainer.has(c, DiType.shareRepo);
|
|
|
|
|
2021-08-07 22:34:44 +02:00
|
|
|
@override
|
|
|
|
initState() {
|
|
|
|
super.initState();
|
2021-11-25 10:47:49 +01:00
|
|
|
_album = widget.album;
|
|
|
|
_items = _album.shares
|
2021-11-19 19:29:56 +01:00
|
|
|
?.map((s) =>
|
|
|
|
_ShareItem(s.userId, s.displayName ?? s.userId.toString()))
|
|
|
|
.toList() ??
|
|
|
|
[];
|
|
|
|
_initBloc();
|
2021-08-07 22:34:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
build(BuildContext context) {
|
2022-11-12 10:55:33 +01:00
|
|
|
return DialogScaffold(
|
|
|
|
canPop: _processingSharee.isEmpty,
|
|
|
|
body: BlocListener<ListShareeBloc, ListShareeBlocState>(
|
|
|
|
bloc: _shareeBloc,
|
|
|
|
listener: _onShareeStateChange,
|
|
|
|
child: Builder(
|
|
|
|
builder: _buildContent,
|
2021-08-21 21:28:44 +02:00
|
|
|
),
|
2021-08-07 22:34:44 +02:00
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-11-19 19:29:56 +01:00
|
|
|
Widget _buildContent(BuildContext context) {
|
2022-01-28 21:06:19 +01:00
|
|
|
return SimpleDialog(
|
|
|
|
title: Text(L10n.global().shareAlbumDialogTitle),
|
|
|
|
children: [
|
|
|
|
..._items.map((i) => _buildItem(context, i)),
|
|
|
|
_buildCreateShareItem(context),
|
|
|
|
],
|
2021-08-07 22:34:44 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-11-19 19:29:56 +01:00
|
|
|
Widget _buildItem(BuildContext context, _ShareItem share) {
|
|
|
|
final isProcessing = _processingSharee.any((s) => s == share.shareWith);
|
2021-08-07 22:34:44 +02:00
|
|
|
final Widget trailing;
|
|
|
|
if (isProcessing) {
|
2022-11-12 10:55:33 +01:00
|
|
|
trailing = const Padding(
|
|
|
|
padding: EdgeInsetsDirectional.only(end: 12),
|
2021-09-15 13:14:37 +02:00
|
|
|
child: SizedBox(
|
|
|
|
width: 24,
|
|
|
|
height: 24,
|
2022-11-12 10:55:33 +01:00
|
|
|
child: CircularProgressIndicator(),
|
2021-09-15 13:14:37 +02:00
|
|
|
),
|
2021-08-07 22:34:44 +02:00
|
|
|
);
|
|
|
|
} else {
|
|
|
|
trailing = Checkbox(
|
2021-11-19 19:29:56 +01:00
|
|
|
value: true,
|
|
|
|
onChanged: (_) {},
|
2021-08-07 22:34:44 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
return SimpleDialogOption(
|
2022-07-08 16:52:18 +02:00
|
|
|
onPressed: isProcessing ? () {} : () => _onShareItemPressed(share),
|
2021-08-07 22:34:44 +02:00
|
|
|
child: ListTile(
|
2021-11-19 19:29:56 +01:00
|
|
|
title: Text(share.displayName),
|
|
|
|
subtitle: Text(share.shareWith.toString()),
|
2021-08-07 22:34:44 +02:00
|
|
|
// pass through the tap event
|
|
|
|
trailing: IgnorePointer(
|
|
|
|
child: trailing,
|
|
|
|
),
|
|
|
|
),
|
2021-11-19 19:29:56 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Widget _buildCreateShareItem(BuildContext context) {
|
|
|
|
return Padding(
|
|
|
|
padding: const EdgeInsets.symmetric(horizontal: 40),
|
|
|
|
child: TypeAheadField<Sharee>(
|
|
|
|
textFieldConfiguration: TextFieldConfiguration(
|
|
|
|
controller: _searchController,
|
2021-11-25 13:39:19 +01:00
|
|
|
decoration: InputDecoration(
|
|
|
|
hintText: L10n.global().addUserInputHint,
|
2021-11-19 19:29:56 +01:00
|
|
|
),
|
|
|
|
),
|
|
|
|
suggestionsCallback: _onSearch,
|
|
|
|
itemBuilder: (context, suggestion) => ListTile(
|
|
|
|
title: Text(suggestion.label),
|
|
|
|
subtitle: Text(suggestion.shareWith.toString()),
|
|
|
|
),
|
|
|
|
onSuggestionSelected: _onSearchSuggestionSelected,
|
|
|
|
hideOnEmpty: true,
|
|
|
|
hideOnLoading: true,
|
|
|
|
autoFlipDirection: true,
|
|
|
|
),
|
2021-08-07 22:34:44 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-11-19 19:29:56 +01:00
|
|
|
void _onShareeStateChange(BuildContext context, ListShareeBlocState state) {
|
|
|
|
if (state is ListShareeBlocSuccess) {
|
|
|
|
_transformShareeItems(state.items);
|
|
|
|
} else if (state is ListShareeBlocFailure) {
|
2021-08-21 21:28:44 +02:00
|
|
|
SnackBarManager().showSnackBar(SnackBar(
|
2021-08-29 13:51:43 +02:00
|
|
|
content: Text(exception_util.toUserString(state.exception)),
|
2021-08-21 21:28:44 +02:00
|
|
|
duration: k.snackBarDurationNormal,
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-19 19:29:56 +01:00
|
|
|
Future<void> _onShareItemPressed(_ShareItem share) async {
|
2021-08-07 22:34:44 +02:00
|
|
|
setState(() {
|
2021-11-19 19:29:56 +01:00
|
|
|
_processingSharee.add(share.shareWith);
|
2021-08-07 22:34:44 +02:00
|
|
|
});
|
2021-11-19 19:29:56 +01:00
|
|
|
try {
|
|
|
|
if (await _removeShare(share)) {
|
|
|
|
if (mounted) {
|
|
|
|
setState(() {
|
|
|
|
_items.remove(share);
|
|
|
|
_onShareItemListUpdated();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
if (mounted) {
|
|
|
|
setState(() {
|
|
|
|
_processingSharee.remove(share.shareWith);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<Iterable<Sharee>> _onSearch(String pattern) async {
|
|
|
|
_suggestionBloc.add(SearchSuggestionBlocSearchEvent(pattern.toCi()));
|
|
|
|
await Future.delayed(const Duration(milliseconds: 500));
|
|
|
|
await async_util
|
|
|
|
.wait(() => _suggestionBloc.state is! SearchSuggestionBlocLoading);
|
|
|
|
if (_suggestionBloc.state is SearchSuggestionBlocSuccess) {
|
|
|
|
return _suggestionBloc.state.results;
|
2021-08-07 22:34:44 +02:00
|
|
|
} else {
|
2021-11-19 19:29:56 +01:00
|
|
|
return [];
|
2021-08-07 22:34:44 +02:00
|
|
|
}
|
2021-11-19 19:29:56 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> _onSearchSuggestionSelected(Sharee sharee) async {
|
|
|
|
_searchController.clear();
|
|
|
|
final item = _ShareItem(sharee.shareWith, sharee.label);
|
|
|
|
var isGood = false;
|
2021-08-07 22:34:44 +02:00
|
|
|
setState(() {
|
2021-11-19 19:29:56 +01:00
|
|
|
_items.add(item);
|
|
|
|
_onShareItemListUpdated();
|
|
|
|
_processingSharee.add(sharee.shareWith);
|
2021-08-07 22:34:44 +02:00
|
|
|
});
|
2021-11-19 19:29:56 +01:00
|
|
|
try {
|
|
|
|
isGood = await _createShare(sharee);
|
|
|
|
} finally {
|
|
|
|
if (mounted) {
|
|
|
|
setState(() {
|
|
|
|
if (!isGood) {
|
|
|
|
_items.remove(item);
|
|
|
|
_onShareItemListUpdated();
|
|
|
|
}
|
|
|
|
_processingSharee.remove(sharee.shareWith);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void _onShareItemListUpdated() {
|
|
|
|
if (_shareeBloc.state is ListShareeBlocSuccess) {
|
|
|
|
_transformShareeItems(_shareeBloc.state.items);
|
|
|
|
}
|
2021-08-07 22:34:44 +02:00
|
|
|
}
|
|
|
|
|
2021-10-18 12:46:06 +02:00
|
|
|
void _onFixPressed() {
|
|
|
|
Navigator.of(context).pushNamed(AlbumShareOutlierBrowser.routeName,
|
2021-11-25 10:47:49 +01:00
|
|
|
arguments: AlbumShareOutlierBrowserArguments(widget.account, _album));
|
2021-10-18 12:46:06 +02:00
|
|
|
}
|
|
|
|
|
2021-11-19 19:29:56 +01:00
|
|
|
void _transformShareeItems(List<Sharee> sharees) {
|
|
|
|
final candidates = sharees
|
|
|
|
.where((s) =>
|
2022-07-11 20:14:42 +02:00
|
|
|
s.shareWith != widget.account.userId &&
|
2021-11-19 19:29:56 +01:00
|
|
|
// remove users already shared with
|
|
|
|
!_items.any((i) => i.shareWith == s.shareWith))
|
|
|
|
.toList();
|
|
|
|
_suggestionBloc
|
|
|
|
.add(SearchSuggestionBlocUpdateItemsEvent<Sharee>(candidates));
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<bool> _createShare(Sharee sharee) async {
|
2021-10-18 12:46:06 +02:00
|
|
|
var hasFailure = false;
|
|
|
|
try {
|
2021-11-25 10:47:49 +01:00
|
|
|
_album = await _editMutex.protect(() async {
|
2022-07-05 22:20:24 +02:00
|
|
|
return await ShareAlbumWithUser(_c.shareRepo, _c.albumRepo)(
|
2021-11-25 10:47:49 +01:00
|
|
|
widget.account,
|
|
|
|
_album,
|
|
|
|
sharee,
|
|
|
|
onShareFileFailed: (_) {
|
|
|
|
hasFailure = true;
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
2021-10-18 12:46:06 +02:00
|
|
|
} catch (e, stackTrace) {
|
|
|
|
_log.shout(
|
|
|
|
"[_createShare] Failed while ShareAlbumWithUser", e, stackTrace);
|
|
|
|
SnackBarManager().showSnackBar(SnackBar(
|
|
|
|
content: Text(exception_util.toUserString(e)),
|
|
|
|
duration: k.snackBarDurationNormal,
|
|
|
|
));
|
2021-11-19 19:29:56 +01:00
|
|
|
return false;
|
2021-10-18 12:46:06 +02:00
|
|
|
}
|
2021-11-19 19:29:56 +01:00
|
|
|
SnackBarManager().showSnackBar(SnackBar(
|
|
|
|
content: Text(hasFailure
|
|
|
|
? L10n.global()
|
|
|
|
.shareAlbumSuccessWithErrorNotification(sharee.shareWith)
|
|
|
|
: L10n.global().shareAlbumSuccessNotification(sharee.shareWith)),
|
|
|
|
action: hasFailure
|
|
|
|
? SnackBarAction(
|
|
|
|
label: L10n.global().fixButtonLabel,
|
|
|
|
onPressed: _onFixPressed,
|
|
|
|
)
|
|
|
|
: null,
|
|
|
|
duration: k.snackBarDurationNormal,
|
|
|
|
));
|
|
|
|
return true;
|
2021-10-18 12:46:06 +02:00
|
|
|
}
|
|
|
|
|
2021-11-19 19:29:56 +01:00
|
|
|
Future<bool> _removeShare(_ShareItem share) async {
|
2021-10-18 12:46:06 +02:00
|
|
|
var hasFailure = false;
|
|
|
|
try {
|
2021-11-25 10:47:49 +01:00
|
|
|
_album = await _editMutex.protect(() async {
|
2021-12-03 18:31:27 +01:00
|
|
|
return await UnshareAlbumWithUser(
|
|
|
|
KiwiContainer().resolve<DiContainer>())(
|
2021-11-25 10:47:49 +01:00
|
|
|
widget.account,
|
|
|
|
_album,
|
|
|
|
share.shareWith,
|
|
|
|
onUnshareFileFailed: (_) {
|
|
|
|
hasFailure = true;
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
2021-10-18 12:46:06 +02:00
|
|
|
} catch (e, stackTrace) {
|
|
|
|
_log.shout(
|
|
|
|
"[_removeShare] Failed while UnshareAlbumWithUser", e, stackTrace);
|
|
|
|
SnackBarManager().showSnackBar(SnackBar(
|
|
|
|
content: Text(exception_util.toUserString(e)),
|
|
|
|
duration: k.snackBarDurationNormal,
|
|
|
|
));
|
2021-11-19 19:29:56 +01:00
|
|
|
return false;
|
2021-10-18 12:46:06 +02:00
|
|
|
}
|
2021-11-19 19:29:56 +01:00
|
|
|
SnackBarManager().showSnackBar(SnackBar(
|
|
|
|
content: Text(hasFailure
|
|
|
|
? L10n.global()
|
|
|
|
.unshareAlbumSuccessWithErrorNotification(share.shareWith)
|
|
|
|
: L10n.global().unshareAlbumSuccessNotification(share.shareWith)),
|
|
|
|
action: hasFailure
|
|
|
|
? SnackBarAction(
|
|
|
|
label: L10n.global().fixButtonLabel,
|
|
|
|
onPressed: _onFixPressed,
|
|
|
|
)
|
|
|
|
: null,
|
|
|
|
duration: k.snackBarDurationNormal,
|
|
|
|
));
|
|
|
|
return true;
|
2021-10-18 12:46:06 +02:00
|
|
|
}
|
|
|
|
|
2021-11-19 19:29:56 +01:00
|
|
|
Future<void> _initBloc() async {
|
|
|
|
if (_shareeBloc.state is ListShareeBlocSuccess) {
|
2022-06-20 13:49:58 +02:00
|
|
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
2021-11-19 19:29:56 +01:00
|
|
|
setState(() {
|
|
|
|
_onShareeStateChange(context, _shareeBloc.state);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
_log.info("[_initBloc] Initialize bloc");
|
|
|
|
_shareeBloc.add(ListShareeBlocQuery(widget.account));
|
|
|
|
}
|
|
|
|
}
|
2021-08-07 22:34:44 +02:00
|
|
|
|
2022-07-05 22:20:24 +02:00
|
|
|
late final DiContainer _c;
|
|
|
|
|
2021-11-19 19:29:56 +01:00
|
|
|
late final _shareeBloc = ListShareeBloc.of(widget.account);
|
|
|
|
final _suggestionBloc = SearchSuggestionBloc<Sharee>(
|
|
|
|
itemToKeywords: (item) => [item.shareWith, item.label.toCi()],
|
|
|
|
);
|
|
|
|
|
2021-11-25 10:47:49 +01:00
|
|
|
late Album _album;
|
|
|
|
final _editMutex = Mutex();
|
2021-11-19 19:29:56 +01:00
|
|
|
late final List<_ShareItem> _items;
|
|
|
|
final _processingSharee = <CiString>[];
|
|
|
|
final _searchController = TextEditingController();
|
2021-08-07 22:34:44 +02:00
|
|
|
|
|
|
|
static final _log =
|
|
|
|
Logger("widget.share_album_dialog._ShareAlbumDialogState");
|
|
|
|
}
|
2021-11-19 19:29:56 +01:00
|
|
|
|
|
|
|
class _ShareItem {
|
|
|
|
_ShareItem(this.shareWith, this.displayName);
|
|
|
|
|
|
|
|
final CiString shareWith;
|
|
|
|
final String displayName;
|
|
|
|
}
|