2024-01-05 06:20:55 +01:00
|
|
|
import 'package:auto_route/auto_route.dart';
|
2023-06-27 23:00:20 +02:00
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:flutter/services.dart';
|
|
|
|
import 'package:flutter_hooks/flutter_hooks.dart';
|
|
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
|
|
import 'package:immich_mobile/modules/memories/models/memory.dart';
|
2024-01-31 00:47:47 +01:00
|
|
|
import 'package:immich_mobile/modules/memories/ui/memory_bottom_info.dart';
|
2023-06-27 23:00:20 +02:00
|
|
|
import 'package:immich_mobile/modules/memories/ui/memory_card.dart';
|
2024-01-31 00:47:47 +01:00
|
|
|
import 'package:immich_mobile/modules/memories/ui/memory_epilogue.dart';
|
2023-07-22 21:51:25 +02:00
|
|
|
import 'package:immich_mobile/shared/models/asset.dart';
|
|
|
|
import 'package:immich_mobile/shared/ui/immich_image.dart';
|
|
|
|
import 'package:openapi/api.dart' as api;
|
2023-06-27 23:00:20 +02:00
|
|
|
|
2024-01-15 17:50:33 +01:00
|
|
|
@RoutePage()
|
2023-06-27 23:00:20 +02:00
|
|
|
class MemoryPage extends HookConsumerWidget {
|
|
|
|
final List<Memory> memories;
|
|
|
|
final int memoryIndex;
|
|
|
|
|
|
|
|
const MemoryPage({
|
|
|
|
required this.memories,
|
|
|
|
required this.memoryIndex,
|
|
|
|
super.key,
|
|
|
|
});
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context, WidgetRef ref) {
|
|
|
|
final memoryPageController = usePageController(initialPage: memoryIndex);
|
|
|
|
final memoryAssetPageController = usePageController();
|
|
|
|
final currentMemory = useState(memories[memoryIndex]);
|
|
|
|
final currentAssetPage = useState(0);
|
|
|
|
final assetProgress = useState(
|
|
|
|
"${currentAssetPage.value + 1}|${currentMemory.value.assets.length}",
|
|
|
|
);
|
|
|
|
const bgColor = Colors.black;
|
|
|
|
|
|
|
|
toNextMemory() {
|
|
|
|
memoryPageController.nextPage(
|
|
|
|
duration: const Duration(milliseconds: 500),
|
|
|
|
curve: Curves.easeIn,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
toNextAsset(int currentAssetIndex) {
|
2023-07-22 21:51:25 +02:00
|
|
|
if (currentAssetIndex + 1 < currentMemory.value.assets.length) {
|
|
|
|
// Go to the next asset
|
|
|
|
memoryAssetPageController.nextPage(
|
|
|
|
curve: Curves.easeInOut,
|
|
|
|
duration: const Duration(milliseconds: 500),
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
// Go to the next memory since we are at the end of our assets
|
|
|
|
toNextMemory();
|
|
|
|
}
|
2023-06-27 23:00:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
updateProgressText() {
|
|
|
|
assetProgress.value =
|
|
|
|
"${currentAssetPage.value + 1}|${currentMemory.value.assets.length}";
|
|
|
|
}
|
|
|
|
|
2023-07-22 21:51:25 +02:00
|
|
|
/// Downloads and caches the image for the asset at this [currentMemory]'s index
|
|
|
|
precacheAsset(int index) async {
|
|
|
|
// Guard index out of range
|
|
|
|
if (index < 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-07-31 20:14:17 +02:00
|
|
|
// Context might be removed due to popping out of Memory Lane during Scroll handling
|
|
|
|
if (!context.mounted) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-07-22 21:51:25 +02:00
|
|
|
late Asset asset;
|
|
|
|
if (index < currentMemory.value.assets.length) {
|
|
|
|
// Uses the next asset in this current memory
|
|
|
|
asset = currentMemory.value.assets[index];
|
|
|
|
} else {
|
|
|
|
// Precache the first asset in the next memory if available
|
|
|
|
final currentMemoryIndex = memories.indexOf(currentMemory.value);
|
|
|
|
|
|
|
|
// Guard no memory found
|
|
|
|
if (currentMemoryIndex == -1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
final nextMemoryIndex = currentMemoryIndex + 1;
|
|
|
|
// Guard no next memory
|
|
|
|
if (nextMemoryIndex >= memories.length) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the first asset from the next memory
|
|
|
|
asset = memories[nextMemoryIndex].assets.first;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Gets the thumbnail url and precaches it
|
|
|
|
final precaches = <Future<dynamic>>[];
|
|
|
|
|
|
|
|
precaches.add(
|
|
|
|
ImmichImage.precacheAsset(
|
|
|
|
asset,
|
|
|
|
context,
|
|
|
|
type: api.ThumbnailFormat.WEBP,
|
2024-01-30 16:24:31 +01:00
|
|
|
size: 2048,
|
2023-07-22 21:51:25 +02:00
|
|
|
),
|
|
|
|
);
|
|
|
|
precaches.add(
|
|
|
|
ImmichImage.precacheAsset(
|
|
|
|
asset,
|
|
|
|
context,
|
|
|
|
type: api.ThumbnailFormat.JPEG,
|
2024-01-30 16:24:31 +01:00
|
|
|
size: 2048,
|
2023-07-22 21:51:25 +02:00
|
|
|
),
|
|
|
|
);
|
|
|
|
|
|
|
|
await Future.wait(precaches);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Precache the next page right away if we are on the first page
|
|
|
|
if (currentAssetPage.value == 0) {
|
|
|
|
Future.delayed(const Duration(milliseconds: 200))
|
|
|
|
.then((_) => precacheAsset(1));
|
|
|
|
}
|
|
|
|
|
2023-06-27 23:00:20 +02:00
|
|
|
onAssetChanged(int otherIndex) {
|
|
|
|
HapticFeedback.selectionClick();
|
|
|
|
currentAssetPage.value = otherIndex;
|
2023-07-22 21:51:25 +02:00
|
|
|
precacheAsset(otherIndex + 1);
|
2023-06-27 23:00:20 +02:00
|
|
|
updateProgressText();
|
|
|
|
}
|
|
|
|
|
2023-07-22 06:56:49 +02:00
|
|
|
/* Notification listener is used instead of OnPageChanged callback since OnPageChanged is called
|
|
|
|
* when the page in the **center** of the viewer changes. We want to reset currentAssetPage only when the final
|
|
|
|
* page during the end of scroll is different than the current page
|
|
|
|
*/
|
|
|
|
return NotificationListener<ScrollNotification>(
|
|
|
|
onNotification: (ScrollNotification notification) {
|
2023-07-31 20:14:17 +02:00
|
|
|
// Calculate OverScroll manually using the number of pixels away from maxScrollExtent
|
|
|
|
// maxScrollExtend contains the sum of horizontal pixels of all assets for depth = 1
|
|
|
|
// or sum of vertical pixels of all memories for depth = 0
|
|
|
|
if (notification is ScrollUpdateNotification) {
|
2024-01-31 00:47:47 +01:00
|
|
|
final isEpiloguePage =
|
|
|
|
(memoryPageController.page?.floor() ?? 0) >= memories.length;
|
2023-07-31 20:14:17 +02:00
|
|
|
|
2024-01-31 00:47:47 +01:00
|
|
|
final offset = notification.metrics.pixels;
|
|
|
|
if (isEpiloguePage &&
|
|
|
|
(offset > notification.metrics.maxScrollExtent + 150)) {
|
|
|
|
context.popRoute();
|
2023-07-31 20:14:17 +02:00
|
|
|
return true;
|
2023-07-22 06:56:49 +02:00
|
|
|
}
|
|
|
|
}
|
2024-01-31 00:47:47 +01:00
|
|
|
|
2023-07-22 06:56:49 +02:00
|
|
|
return false;
|
|
|
|
},
|
|
|
|
child: Scaffold(
|
|
|
|
backgroundColor: bgColor,
|
|
|
|
body: SafeArea(
|
|
|
|
child: PageView.builder(
|
2023-07-31 20:14:17 +02:00
|
|
|
physics: const BouncingScrollPhysics(
|
|
|
|
parent: AlwaysScrollableScrollPhysics(),
|
|
|
|
),
|
2023-07-22 06:56:49 +02:00
|
|
|
scrollDirection: Axis.vertical,
|
|
|
|
controller: memoryPageController,
|
2024-01-31 00:47:47 +01:00
|
|
|
onPageChanged: (pageNumber) {
|
|
|
|
HapticFeedback.mediumImpact();
|
|
|
|
if (pageNumber < memories.length) {
|
|
|
|
currentMemory.value = memories[pageNumber];
|
|
|
|
}
|
|
|
|
|
|
|
|
currentAssetPage.value = 0;
|
|
|
|
|
|
|
|
updateProgressText();
|
|
|
|
},
|
|
|
|
itemCount: memories.length + 1,
|
2023-07-22 06:56:49 +02:00
|
|
|
itemBuilder: (context, mIndex) {
|
2024-01-31 00:47:47 +01:00
|
|
|
// Build last page
|
|
|
|
if (mIndex == memories.length) {
|
|
|
|
return MemoryEpilogue(
|
|
|
|
onStartOver: () => memoryPageController.animateToPage(
|
|
|
|
0,
|
|
|
|
duration: const Duration(seconds: 1),
|
|
|
|
curve: Curves.easeInOut,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
2023-07-22 06:56:49 +02:00
|
|
|
// Build horizontal page
|
|
|
|
return Column(
|
|
|
|
children: [
|
|
|
|
Expanded(
|
|
|
|
child: PageView.builder(
|
2023-07-31 20:14:17 +02:00
|
|
|
physics: const BouncingScrollPhysics(
|
|
|
|
parent: AlwaysScrollableScrollPhysics(),
|
|
|
|
),
|
2023-07-22 06:56:49 +02:00
|
|
|
controller: memoryAssetPageController,
|
|
|
|
onPageChanged: onAssetChanged,
|
|
|
|
scrollDirection: Axis.horizontal,
|
|
|
|
itemCount: memories[mIndex].assets.length,
|
|
|
|
itemBuilder: (context, index) {
|
|
|
|
final asset = memories[mIndex].assets[index];
|
|
|
|
return Container(
|
|
|
|
color: Colors.black,
|
|
|
|
child: MemoryCard(
|
|
|
|
asset: asset,
|
|
|
|
onTap: () => toNextAsset(index),
|
2024-01-05 06:20:55 +01:00
|
|
|
onClose: () => context.popRoute(),
|
2023-07-22 06:56:49 +02:00
|
|
|
rightCornerText: assetProgress.value,
|
|
|
|
title: memories[mIndex].title,
|
|
|
|
showTitle: index == 0,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
2023-06-27 23:00:20 +02:00
|
|
|
),
|
2024-01-31 00:47:47 +01:00
|
|
|
MemoryBottomInfo(memory: memories[mIndex]),
|
2023-07-22 06:56:49 +02:00
|
|
|
],
|
|
|
|
);
|
|
|
|
},
|
|
|
|
),
|
2023-06-27 23:00:20 +02:00
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|