mirror of
https://gitlab.com/nkming2/nc-photos.git
synced 2025-01-23 17:26:18 +01:00
261 lines
7.9 KiB
Dart
261 lines
7.9 KiB
Dart
|
part of '../home_photos2.dart';
|
||
|
|
||
|
class _ContentList extends StatelessWidget {
|
||
|
const _ContentList();
|
||
|
|
||
|
@override
|
||
|
Widget build(BuildContext context) {
|
||
|
return _BlocSelector<int>(
|
||
|
selector: (state) => state.zoom,
|
||
|
builder: (context, zoom) => _ContentListBody(
|
||
|
maxCrossAxisExtent: photo_list_util.getThumbSize(zoom).toDouble(),
|
||
|
isNeedVisibilityInfo: true,
|
||
|
),
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
class _ScalingList extends StatelessWidget {
|
||
|
const _ScalingList();
|
||
|
|
||
|
@override
|
||
|
Widget build(BuildContext context) {
|
||
|
return _BlocBuilder(
|
||
|
buildWhen: (previous, current) => previous.scale != current.scale,
|
||
|
builder: (context, state) {
|
||
|
if (state.scale == null) {
|
||
|
return const SizedBox.shrink();
|
||
|
}
|
||
|
int nextZoom;
|
||
|
if (state.scale! > 1) {
|
||
|
nextZoom = state.zoom + 1;
|
||
|
} else {
|
||
|
nextZoom = state.zoom - 1;
|
||
|
}
|
||
|
nextZoom = nextZoom.clamp(-1, 2);
|
||
|
return _ContentListBody(
|
||
|
maxCrossAxisExtent: photo_list_util.getThumbSize(nextZoom).toDouble(),
|
||
|
isNeedVisibilityInfo: false,
|
||
|
);
|
||
|
},
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
@npLog
|
||
|
class _ContentListBody extends StatelessWidget {
|
||
|
const _ContentListBody({
|
||
|
required this.maxCrossAxisExtent,
|
||
|
required this.isNeedVisibilityInfo,
|
||
|
});
|
||
|
|
||
|
@override
|
||
|
Widget build(BuildContext context) {
|
||
|
return _BlocBuilder(
|
||
|
buildWhen: (previous, current) =>
|
||
|
previous.transformedItems != current.transformedItems ||
|
||
|
previous.selectedItems != current.selectedItems,
|
||
|
builder: (context, state) => SelectableItemList<_Item>(
|
||
|
maxCrossAxisExtent: maxCrossAxisExtent,
|
||
|
items: state.transformedItems,
|
||
|
itemBuilder: (context, index, item) {
|
||
|
final w = item.buildWidget(context);
|
||
|
if (isNeedVisibilityInfo) {
|
||
|
return VisibilityDetector(
|
||
|
key: Key("${_log.fullName}.$index"),
|
||
|
onVisibilityChanged: (info) {
|
||
|
if (context.mounted) {
|
||
|
if (info.visibleFraction >= 0.2) {
|
||
|
context
|
||
|
.addEvent(_AddVisibleItem(_VisibleItem(index, item)));
|
||
|
} else {
|
||
|
context.addEvent(
|
||
|
_RemoveVisibleItem(_VisibleItem(index, item)));
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
child: w,
|
||
|
);
|
||
|
} else {
|
||
|
return w;
|
||
|
}
|
||
|
},
|
||
|
staggeredTileBuilder: (_, item) => item.staggeredTile,
|
||
|
selectedItems: state.selectedItems,
|
||
|
onSelectionChange: (_, selected) {
|
||
|
context.addEvent(_SetSelectedItems(items: selected.cast()));
|
||
|
},
|
||
|
onItemTap: (context, index, _) {
|
||
|
if (state.transformedItems[index] is! _FileItem) {
|
||
|
return;
|
||
|
}
|
||
|
final actualIndex = index -
|
||
|
state.transformedItems
|
||
|
.sublist(0, index)
|
||
|
.where((e) => e is! _FileItem)
|
||
|
.length;
|
||
|
Navigator.of(context).pushNamed(
|
||
|
Viewer.routeName,
|
||
|
arguments: ViewerArguments(
|
||
|
context.bloc.account,
|
||
|
state.transformedItems
|
||
|
.whereType<_FileItem>()
|
||
|
.map((e) => e.file)
|
||
|
.toList(),
|
||
|
actualIndex,
|
||
|
),
|
||
|
);
|
||
|
},
|
||
|
onMaxExtentChange: (value) {
|
||
|
context.addEvent(_SetContentListMaxExtent(value));
|
||
|
},
|
||
|
),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
final double maxCrossAxisExtent;
|
||
|
final bool isNeedVisibilityInfo;
|
||
|
}
|
||
|
|
||
|
class _MemoryCollectionList extends StatelessWidget {
|
||
|
const _MemoryCollectionList();
|
||
|
|
||
|
@override
|
||
|
Widget build(BuildContext context) {
|
||
|
return SliverToBoxAdapter(
|
||
|
child: SizedBox(
|
||
|
height: _MemoryCollectionItem.height,
|
||
|
child: _BlocSelector<List<Collection>>(
|
||
|
selector: (state) => state.memoryCollections,
|
||
|
builder: (context, memoryCollections) => ListView.separated(
|
||
|
scrollDirection: Axis.horizontal,
|
||
|
padding: const EdgeInsets.symmetric(horizontal: 8),
|
||
|
itemCount: memoryCollections.length,
|
||
|
itemBuilder: (context, index) {
|
||
|
final c = memoryCollections[index];
|
||
|
return _MemoryCollectionItemView(
|
||
|
coverUrl: c.getCoverUrl(
|
||
|
k.photoThumbSize,
|
||
|
k.photoThumbSize,
|
||
|
isKeepAspectRatio: true,
|
||
|
),
|
||
|
label: c.name,
|
||
|
onTap: () {
|
||
|
Navigator.of(context).pushNamed(
|
||
|
CollectionBrowser.routeName,
|
||
|
arguments: CollectionBrowserArguments(c),
|
||
|
);
|
||
|
},
|
||
|
);
|
||
|
},
|
||
|
separatorBuilder: (context, index) => const SizedBox(width: 8),
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
class _MemoryCollectionItemView extends StatelessWidget {
|
||
|
static const width = 96.0;
|
||
|
static const height = width * 1.15;
|
||
|
|
||
|
const _MemoryCollectionItemView({
|
||
|
required this.coverUrl,
|
||
|
required this.label,
|
||
|
this.onTap,
|
||
|
});
|
||
|
|
||
|
@override
|
||
|
Widget build(BuildContext context) {
|
||
|
return Align(
|
||
|
alignment: AlignmentDirectional.topStart,
|
||
|
child: ClipRRect(
|
||
|
borderRadius: BorderRadius.circular(8),
|
||
|
child: SizedBox(
|
||
|
width: width,
|
||
|
height: height,
|
||
|
child: Stack(
|
||
|
fit: StackFit.expand,
|
||
|
children: [
|
||
|
PhotoListImage(
|
||
|
account: context.bloc.account,
|
||
|
previewUrl: coverUrl,
|
||
|
padding: const EdgeInsets.all(0),
|
||
|
),
|
||
|
Positioned.fill(
|
||
|
child: Container(
|
||
|
decoration: const BoxDecoration(
|
||
|
gradient: LinearGradient(
|
||
|
begin: Alignment.center,
|
||
|
end: Alignment.bottomCenter,
|
||
|
colors: [Colors.transparent, Colors.black87],
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
Positioned.fill(
|
||
|
child: Align(
|
||
|
alignment: AlignmentDirectional.bottomStart,
|
||
|
child: Padding(
|
||
|
padding: const EdgeInsets.all(4),
|
||
|
child: Text(
|
||
|
label,
|
||
|
style: Theme.of(context).textTheme.labelLarge!.copyWith(
|
||
|
color: Theme.of(context).onDarkSurface,
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
if (onTap != null)
|
||
|
Positioned.fill(
|
||
|
child: Material(
|
||
|
type: MaterialType.transparency,
|
||
|
child: InkWell(
|
||
|
onTap: onTap,
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
],
|
||
|
),
|
||
|
),
|
||
|
),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
final String? coverUrl;
|
||
|
final String label;
|
||
|
final VoidCallback? onTap;
|
||
|
}
|
||
|
|
||
|
class _ScrollLabel extends StatelessWidget {
|
||
|
const _ScrollLabel();
|
||
|
|
||
|
@override
|
||
|
Widget build(BuildContext context) {
|
||
|
return _BlocSelector<Set<_VisibleItem>>(
|
||
|
selector: (state) => state.visibleItems,
|
||
|
builder: (context, visibleItems) {
|
||
|
final firstVisibleItem =
|
||
|
visibleItems.sorted().firstWhereOrNull((e) => e.item is _FileItem);
|
||
|
final date = firstVisibleItem?.item.as<_FileItem>()?.file.fdDateTime;
|
||
|
if (date == null) {
|
||
|
return const SizedBox.shrink();
|
||
|
}
|
||
|
final text = DateFormat(DateFormat.YEAR_ABBR_MONTH,
|
||
|
Localizations.localeOf(context).languageCode)
|
||
|
.format(date.toLocal());
|
||
|
return Padding(
|
||
|
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
|
||
|
child: DefaultTextStyle(
|
||
|
style: Theme.of(context).textTheme.titleMedium!.copyWith(
|
||
|
color: Theme.of(context).colorScheme.onInverseSurface),
|
||
|
child: Text(text),
|
||
|
),
|
||
|
);
|
||
|
},
|
||
|
);
|
||
|
}
|
||
|
}
|