mirror of
https://gitlab.com/nkming2/nc-photos.git
synced 2025-01-23 09:16:19 +01:00
343 lines
11 KiB
Dart
343 lines
11 KiB
Dart
import 'package:collection/collection.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
|
import 'package:intl/intl.dart';
|
|
import 'package:kiwi/kiwi.dart';
|
|
import 'package:logging/logging.dart';
|
|
import 'package:nc_photos/account.dart';
|
|
import 'package:nc_photos/app_localizations.dart';
|
|
import 'package:nc_photos/bloc/list_sharing.dart';
|
|
import 'package:nc_photos/di_container.dart';
|
|
import 'package:nc_photos/entity/album.dart';
|
|
import 'package:nc_photos/entity/album/data_source.dart';
|
|
import 'package:nc_photos/entity/collection/builder.dart';
|
|
import 'package:nc_photos/entity/file.dart';
|
|
import 'package:nc_photos/entity/file/data_source.dart';
|
|
import 'package:nc_photos/entity/pref.dart';
|
|
import 'package:nc_photos/entity/share.dart';
|
|
import 'package:nc_photos/exception_util.dart' as exception_util;
|
|
import 'package:nc_photos/k.dart' as k;
|
|
import 'package:nc_photos/object_extension.dart';
|
|
import 'package:nc_photos/or_null.dart';
|
|
import 'package:nc_photos/snack_bar_manager.dart';
|
|
import 'package:nc_photos/use_case/import_potential_shared_album.dart';
|
|
import 'package:nc_photos/widget/collection_browser.dart';
|
|
import 'package:nc_photos/widget/empty_list_indicator.dart';
|
|
import 'package:nc_photos/widget/network_thumbnail.dart';
|
|
import 'package:nc_photos/widget/shared_file_viewer.dart';
|
|
import 'package:np_codegen/np_codegen.dart';
|
|
import 'package:np_ui/np_ui.dart';
|
|
|
|
part 'sharing_browser.g.dart';
|
|
|
|
class SharingBrowserArguments {
|
|
SharingBrowserArguments(this.account);
|
|
|
|
final Account account;
|
|
}
|
|
|
|
/// Show a list of all shares associated with this account
|
|
class SharingBrowser extends StatefulWidget {
|
|
static const routeName = "/sharing-browser";
|
|
|
|
static Route buildRoute(SharingBrowserArguments args) => MaterialPageRoute(
|
|
builder: (context) => SharingBrowser.fromArgs(args),
|
|
);
|
|
|
|
const SharingBrowser({
|
|
Key? key,
|
|
required this.account,
|
|
}) : super(key: key);
|
|
|
|
SharingBrowser.fromArgs(SharingBrowserArguments args, {Key? key})
|
|
: this(
|
|
key: key,
|
|
account: args.account,
|
|
);
|
|
|
|
@override
|
|
createState() => _SharingBrowserState();
|
|
|
|
final Account account;
|
|
}
|
|
|
|
@npLog
|
|
class _SharingBrowserState extends State<SharingBrowser> {
|
|
@override
|
|
initState() {
|
|
super.initState();
|
|
_importPotentialSharedAlbum().whenComplete(() {
|
|
_initBloc();
|
|
});
|
|
AccountPref.of(widget.account).run((obj) {
|
|
if (obj.hasNewSharedAlbumOr()) {
|
|
obj.setNewSharedAlbum(false);
|
|
}
|
|
});
|
|
}
|
|
|
|
@override
|
|
build(BuildContext context) {
|
|
return Scaffold(
|
|
body: BlocListener<ListSharingBloc, ListSharingBlocState>(
|
|
bloc: _bloc,
|
|
listener: (context, state) => _onStateChange(context, state),
|
|
child: BlocBuilder<ListSharingBloc, ListSharingBlocState>(
|
|
bloc: _bloc,
|
|
builder: (context, state) => _buildContent(context, state),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
void _initBloc() {
|
|
if (_bloc.state is ListSharingBlocInit) {
|
|
_log.info("[_initBloc] Initialize bloc");
|
|
} else {
|
|
// process the current state
|
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
|
if (mounted) {
|
|
setState(() {
|
|
_onStateChange(context, _bloc.state);
|
|
});
|
|
}
|
|
});
|
|
}
|
|
_reqQuery();
|
|
}
|
|
|
|
Widget _buildContent(BuildContext context, ListSharingBlocState state) {
|
|
if ((state is ListSharingBlocSuccess || state is ListSharingBlocFailure) &&
|
|
state.items.isEmpty) {
|
|
return _buildEmptyContent(context);
|
|
} else {
|
|
return Stack(
|
|
children: [
|
|
CustomScrollView(
|
|
slivers: [
|
|
SliverAppBar(
|
|
title: Text(L10n.global().collectionSharingLabel),
|
|
floating: true,
|
|
),
|
|
SliverList(
|
|
delegate: SliverChildBuilderDelegate(
|
|
(context, index) => _buildItem(context, _items[index]),
|
|
childCount: _items.length,
|
|
),
|
|
),
|
|
],
|
|
),
|
|
if (state is ListSharingBlocLoading)
|
|
const Align(
|
|
alignment: Alignment.bottomCenter,
|
|
child: LinearProgressIndicator(),
|
|
),
|
|
],
|
|
);
|
|
}
|
|
}
|
|
|
|
Widget _buildEmptyContent(BuildContext context) {
|
|
return Column(
|
|
children: [
|
|
AppBar(
|
|
title: Text(L10n.global().collectionSharingLabel),
|
|
elevation: 0,
|
|
),
|
|
Expanded(
|
|
child: EmptyListIndicator(
|
|
icon: Icons.share_outlined,
|
|
text: L10n.global().listEmptyText,
|
|
),
|
|
),
|
|
],
|
|
);
|
|
}
|
|
|
|
Widget _buildFileItem(BuildContext context, List<ListSharingItem> shares) {
|
|
assert(shares.first is ListSharingFile);
|
|
final dateStr =
|
|
_getDateFormat(context).format(shares.first.share.stime.toLocal());
|
|
final firstItem = shares.first as ListSharingFile;
|
|
return _ListTile(
|
|
leading: shares.first.share.itemType == ShareItemType.folder
|
|
? const SizedBox(
|
|
height: _leadingSize,
|
|
width: _leadingSize,
|
|
child: Icon(Icons.folder, size: 32),
|
|
)
|
|
: NetworkRectThumbnail(
|
|
account: widget.account,
|
|
imageUrl: NetworkRectThumbnail.imageUrlForFile(
|
|
widget.account, firstItem.file),
|
|
dimension: _leadingSize,
|
|
errorBuilder: (_) => const Icon(Icons.folder, size: 32),
|
|
),
|
|
label: shares.first.share.filename,
|
|
description: shares.first.share.uidOwner == widget.account.userId
|
|
? L10n.global().fileLastSharedDescription(dateStr)
|
|
: L10n.global().fileLastSharedByOthersDescription(
|
|
shares.first.share.displaynameOwner, dateStr),
|
|
trailing: (shares.any((element) => element.share.url?.isNotEmpty == true))
|
|
? const Icon(Icons.link)
|
|
: null,
|
|
onTap: () {
|
|
Navigator.of(context).pushNamed(SharedFileViewer.routeName,
|
|
arguments: SharedFileViewerArguments(
|
|
widget.account,
|
|
firstItem.file,
|
|
shares.map((e) => e.share).toList(),
|
|
));
|
|
},
|
|
);
|
|
}
|
|
|
|
Widget _buildAlbumItem(BuildContext context, List<ListSharingItem> shares) {
|
|
assert(shares.first is ListSharingAlbum);
|
|
final dateStr =
|
|
_getDateFormat(context).format(shares.first.share.stime.toLocal());
|
|
final firstItem = shares.first as ListSharingAlbum;
|
|
final cover = firstItem.album.coverProvider.getCover(firstItem.album);
|
|
return _ListTile(
|
|
leading: cover == null
|
|
? const SizedBox(
|
|
height: _leadingSize,
|
|
width: _leadingSize,
|
|
child: Icon(Icons.photo_album, size: 32),
|
|
)
|
|
: NetworkRectThumbnail(
|
|
account: widget.account,
|
|
imageUrl:
|
|
NetworkRectThumbnail.imageUrlForFile(widget.account, cover),
|
|
dimension: _leadingSize,
|
|
errorBuilder: (_) => const Icon(Icons.photo_album, size: 32),
|
|
),
|
|
label: firstItem.album.name,
|
|
description: shares.first.share.uidOwner == widget.account.userId
|
|
? L10n.global().fileLastSharedDescription(dateStr)
|
|
: L10n.global().albumLastSharedByOthersDescription(
|
|
shares.first.share.displaynameOwner, dateStr),
|
|
trailing: const Icon(Icons.photo_album_outlined),
|
|
onTap: () =>
|
|
_onAlbumShareItemTap(context, shares.first as ListSharingAlbum),
|
|
);
|
|
}
|
|
|
|
Widget _buildItem(BuildContext context, List<ListSharingItem> shares) {
|
|
if (shares.first is ListSharingFile) {
|
|
return _buildFileItem(context, shares);
|
|
} else if (shares.first is ListSharingAlbum) {
|
|
return _buildAlbumItem(context, shares);
|
|
} else {
|
|
throw StateError("Unknown item type: ${shares.first.runtimeType}");
|
|
}
|
|
}
|
|
|
|
void _onStateChange(BuildContext context, ListSharingBlocState state) {
|
|
if (state is ListSharingBlocInit) {
|
|
_items = [];
|
|
} else if (state is ListSharingBlocSuccess ||
|
|
state is ListSharingBlocLoading) {
|
|
_transformItems(state.items);
|
|
} else if (state is ListSharingBlocFailure) {
|
|
_transformItems(state.items);
|
|
SnackBarManager().showSnackBar(SnackBar(
|
|
content: Text(exception_util.toUserString(state.exception)),
|
|
duration: k.snackBarDurationNormal,
|
|
));
|
|
}
|
|
}
|
|
|
|
void _onAlbumShareItemTap(BuildContext context, ListSharingAlbum share) {
|
|
Navigator.of(context).pushNamed(
|
|
CollectionBrowser.routeName,
|
|
arguments: CollectionBrowserArguments(
|
|
CollectionBuilder.byAlbum(widget.account, share.album),
|
|
),
|
|
);
|
|
}
|
|
|
|
void _transformItems(List<ListSharingItem> items) {
|
|
// group shares of the same file
|
|
final map = <String, List<ListSharingItem>>{};
|
|
for (final i in items) {
|
|
final isSharedByMe = (i.share.uidOwner == widget.account.userId);
|
|
final groupKey = "${i.share.path}?$isSharedByMe";
|
|
map[groupKey] ??= <ListSharingItem>[];
|
|
map[groupKey]!.add(i);
|
|
}
|
|
// sort the sub-lists
|
|
for (final list in map.values) {
|
|
list.sort((a, b) => b.share.stime.compareTo(a.share.stime));
|
|
}
|
|
// then sort the map and convert it to list
|
|
_items = map.entries
|
|
.sorted((a, b) =>
|
|
b.value.first.share.stime.compareTo(a.value.first.share.stime))
|
|
.map((e) => e.value)
|
|
.toList();
|
|
}
|
|
|
|
void _reqQuery() {
|
|
_bloc.add(ListSharingBlocQuery(widget.account));
|
|
}
|
|
|
|
Future<List<Album>> _importPotentialSharedAlbum() async {
|
|
final c = KiwiContainer().resolve<DiContainer>().copyWith(
|
|
// don't want the potential albums to be cached at this moment
|
|
fileRepo: OrNull(const FileRepo(FileWebdavDataSource())),
|
|
albumRepo: OrNull(AlbumRepo(AlbumRemoteDataSource())),
|
|
);
|
|
try {
|
|
return await ImportPotentialSharedAlbum(c)(
|
|
widget.account, AccountPref.of(widget.account));
|
|
} catch (e, stackTrace) {
|
|
_log.shout(
|
|
"[_importPotentialSharedAlbum] Failed while ImportPotentialSharedAlbum",
|
|
e,
|
|
stackTrace);
|
|
return [];
|
|
}
|
|
}
|
|
|
|
late final _bloc = ListSharingBloc.of(widget.account);
|
|
|
|
var _items = <List<ListSharingItem>>[];
|
|
}
|
|
|
|
class _ListTile extends StatelessWidget {
|
|
const _ListTile({
|
|
required this.leading,
|
|
required this.label,
|
|
required this.description,
|
|
this.trailing,
|
|
required this.onTap,
|
|
});
|
|
|
|
@override
|
|
build(BuildContext context) {
|
|
return UnboundedListTile(
|
|
leading: leading,
|
|
title: Text(
|
|
label,
|
|
maxLines: 1,
|
|
overflow: TextOverflow.ellipsis,
|
|
),
|
|
subtitle: Text(description),
|
|
trailing: trailing,
|
|
onTap: onTap,
|
|
);
|
|
}
|
|
|
|
final Widget leading;
|
|
final String label;
|
|
final String description;
|
|
final Widget? trailing;
|
|
final VoidCallback onTap;
|
|
}
|
|
|
|
const _leadingSize = 56.0;
|
|
|
|
DateFormat _getDateFormat(BuildContext context) => DateFormat(
|
|
DateFormat.YEAR_ABBR_MONTH_DAY,
|
|
Localizations.localeOf(context).languageCode);
|