mirror of
https://github.com/immich-app/immich.git
synced 2025-01-17 01:06:46 +01:00
Videos play in memories now
This commit is contained in:
parent
b0ff859cd6
commit
f14295e107
3 changed files with 108 additions and 55 deletions
|
@ -21,25 +21,33 @@ class VideoViewerPage extends HookConsumerWidget {
|
||||||
final Asset asset;
|
final Asset asset;
|
||||||
final bool isMotionVideo;
|
final bool isMotionVideo;
|
||||||
final Widget? placeholder;
|
final Widget? placeholder;
|
||||||
final VoidCallback onVideoEnded;
|
final VoidCallback? onVideoEnded;
|
||||||
final VoidCallback? onPlaying;
|
final VoidCallback? onPlaying;
|
||||||
final VoidCallback? onPaused;
|
final VoidCallback? onPaused;
|
||||||
|
final Duration hideControlsTimer;
|
||||||
|
final bool showControls;
|
||||||
|
final bool showDownloadingIndicator;
|
||||||
|
|
||||||
const VideoViewerPage({
|
const VideoViewerPage({
|
||||||
super.key,
|
super.key,
|
||||||
required this.asset,
|
required this.asset,
|
||||||
required this.isMotionVideo,
|
this.isMotionVideo = false,
|
||||||
required this.onVideoEnded,
|
this.onVideoEnded,
|
||||||
this.onPlaying,
|
this.onPlaying,
|
||||||
this.onPaused,
|
this.onPaused,
|
||||||
this.placeholder,
|
this.placeholder,
|
||||||
|
this.showControls = true,
|
||||||
|
this.hideControlsTimer = const Duration(seconds: 5),
|
||||||
|
this.showDownloadingIndicator = true,
|
||||||
});
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context, WidgetRef ref) {
|
Widget build(BuildContext context, WidgetRef ref) {
|
||||||
if (asset.isLocal && asset.livePhotoVideoId == null) {
|
if (asset.isLocal && asset.livePhotoVideoId == null) {
|
||||||
final AsyncValue<File> videoFile = ref.watch(_fileFamily(asset.local!));
|
final AsyncValue<File> videoFile = ref.watch(_fileFamily(asset.local!));
|
||||||
return videoFile.when(
|
return AnimatedSwitcher(
|
||||||
|
duration: const Duration(milliseconds: 200),
|
||||||
|
child: videoFile.when(
|
||||||
data: (data) => VideoPlayer(
|
data: (data) => VideoPlayer(
|
||||||
file: data,
|
file: data,
|
||||||
isMotionVideo: false,
|
isMotionVideo: false,
|
||||||
|
@ -49,12 +57,9 @@ class VideoViewerPage extends HookConsumerWidget {
|
||||||
Icons.image_not_supported_outlined,
|
Icons.image_not_supported_outlined,
|
||||||
color: context.primaryColor,
|
color: context.primaryColor,
|
||||||
),
|
),
|
||||||
loading: () => const Center(
|
loading: () => showDownloadingIndicator
|
||||||
child: SizedBox(
|
? const Center(child: ImmichLoadingIndicator())
|
||||||
width: 75,
|
: Container(),
|
||||||
height: 75,
|
|
||||||
child: CircularProgressIndicator.adaptive(),
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -74,15 +79,24 @@ class VideoViewerPage extends HookConsumerWidget {
|
||||||
onPaused: onPaused,
|
onPaused: onPaused,
|
||||||
onPlaying: onPlaying,
|
onPlaying: onPlaying,
|
||||||
placeholder: placeholder,
|
placeholder: placeholder,
|
||||||
|
hideControlsTimer: hideControlsTimer,
|
||||||
|
showControls: showControls,
|
||||||
|
showDownloadingIndicator: showDownloadingIndicator,
|
||||||
),
|
),
|
||||||
if (downloadAssetStatus == DownloadAssetStatus.loading)
|
AnimatedOpacity(
|
||||||
SizedBox(
|
duration: const Duration(milliseconds: 400),
|
||||||
|
opacity: (downloadAssetStatus == DownloadAssetStatus.loading &&
|
||||||
|
showDownloadingIndicator)
|
||||||
|
? 1.0
|
||||||
|
: 0.0,
|
||||||
|
child: SizedBox(
|
||||||
height: context.height,
|
height: context.height,
|
||||||
width: context.width,
|
width: context.width,
|
||||||
child: const Center(
|
child: const Center(
|
||||||
child: ImmichLoadingIndicator(),
|
child: ImmichLoadingIndicator(),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -102,7 +116,9 @@ class VideoPlayer extends StatefulWidget {
|
||||||
final String? jwtToken;
|
final String? jwtToken;
|
||||||
final File? file;
|
final File? file;
|
||||||
final bool isMotionVideo;
|
final bool isMotionVideo;
|
||||||
final VoidCallback onVideoEnded;
|
final VoidCallback? onVideoEnded;
|
||||||
|
final Duration hideControlsTimer;
|
||||||
|
final bool showControls;
|
||||||
|
|
||||||
final Function()? onPlaying;
|
final Function()? onPlaying;
|
||||||
final Function()? onPaused;
|
final Function()? onPaused;
|
||||||
|
@ -111,16 +127,23 @@ class VideoPlayer extends StatefulWidget {
|
||||||
/// usually, a thumbnail of the video
|
/// usually, a thumbnail of the video
|
||||||
final Widget? placeholder;
|
final Widget? placeholder;
|
||||||
|
|
||||||
|
final bool showDownloadingIndicator;
|
||||||
|
|
||||||
const VideoPlayer({
|
const VideoPlayer({
|
||||||
super.key,
|
super.key,
|
||||||
this.url,
|
this.url,
|
||||||
this.jwtToken,
|
this.jwtToken,
|
||||||
this.file,
|
this.file,
|
||||||
required this.onVideoEnded,
|
this.onVideoEnded,
|
||||||
required this.isMotionVideo,
|
required this.isMotionVideo,
|
||||||
this.onPlaying,
|
this.onPlaying,
|
||||||
this.onPaused,
|
this.onPaused,
|
||||||
this.placeholder,
|
this.placeholder,
|
||||||
|
this.hideControlsTimer = const Duration(
|
||||||
|
seconds: 5,
|
||||||
|
),
|
||||||
|
this.showControls = true,
|
||||||
|
this.showDownloadingIndicator = true,
|
||||||
});
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
|
@ -149,7 +172,7 @@ class _VideoPlayerState extends State<VideoPlayer> {
|
||||||
if (videoPlayerController.value.position ==
|
if (videoPlayerController.value.position ==
|
||||||
videoPlayerController.value.duration) {
|
videoPlayerController.value.duration) {
|
||||||
WakelockPlus.disable();
|
WakelockPlus.disable();
|
||||||
widget.onVideoEnded();
|
widget.onVideoEnded?.call();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -184,9 +207,9 @@ class _VideoPlayerState extends State<VideoPlayer> {
|
||||||
autoInitialize: true,
|
autoInitialize: true,
|
||||||
allowFullScreen: false,
|
allowFullScreen: false,
|
||||||
allowedScreenSleep: false,
|
allowedScreenSleep: false,
|
||||||
showControls: !widget.isMotionVideo,
|
showControls: widget.showControls && !widget.isMotionVideo,
|
||||||
customControls: const VideoPlayerControls(),
|
customControls: const VideoPlayerControls(),
|
||||||
hideControlsTimer: const Duration(seconds: 5),
|
hideControlsTimer: widget.hideControlsTimer,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -214,6 +237,7 @@ class _VideoPlayerState extends State<VideoPlayer> {
|
||||||
child: Stack(
|
child: Stack(
|
||||||
children: [
|
children: [
|
||||||
if (widget.placeholder != null) widget.placeholder!,
|
if (widget.placeholder != null) widget.placeholder!,
|
||||||
|
if (widget.showDownloadingIndicator)
|
||||||
const Center(
|
const Center(
|
||||||
child: ImmichLoadingIndicator(),
|
child: ImmichLoadingIndicator(),
|
||||||
),
|
),
|
||||||
|
|
|
@ -3,6 +3,7 @@ import 'dart:ui';
|
||||||
import 'package:cached_network_image/cached_network_image.dart';
|
import 'package:cached_network_image/cached_network_image.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:immich_mobile/extensions/build_context_extensions.dart';
|
import 'package:immich_mobile/extensions/build_context_extensions.dart';
|
||||||
|
import 'package:immich_mobile/modules/asset_viewer/views/video_viewer_page.dart';
|
||||||
import 'package:immich_mobile/shared/models/asset.dart';
|
import 'package:immich_mobile/shared/models/asset.dart';
|
||||||
import 'package:immich_mobile/shared/models/store.dart';
|
import 'package:immich_mobile/shared/models/store.dart';
|
||||||
import 'package:immich_mobile/shared/ui/immich_image.dart';
|
import 'package:immich_mobile/shared/ui/immich_image.dart';
|
||||||
|
@ -11,15 +12,15 @@ import 'package:openapi/api.dart';
|
||||||
|
|
||||||
class MemoryCard extends StatelessWidget {
|
class MemoryCard extends StatelessWidget {
|
||||||
final Asset asset;
|
final Asset asset;
|
||||||
final void Function() onTap;
|
|
||||||
final String title;
|
final String title;
|
||||||
final bool showTitle;
|
final bool showTitle;
|
||||||
|
final Function()? onVideoEnded;
|
||||||
|
|
||||||
const MemoryCard({
|
const MemoryCard({
|
||||||
required this.asset,
|
required this.asset,
|
||||||
required this.onTap,
|
|
||||||
required this.title,
|
required this.title,
|
||||||
required this.showTitle,
|
required this.showTitle,
|
||||||
|
this.onVideoEnded,
|
||||||
super.key,
|
super.key,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -59,14 +60,12 @@ class MemoryCard extends StatelessWidget {
|
||||||
child: Container(color: Colors.black.withOpacity(0.2)),
|
child: Container(color: Colors.black.withOpacity(0.2)),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
GestureDetector(
|
LayoutBuilder(
|
||||||
onTap: onTap,
|
|
||||||
child: LayoutBuilder(
|
|
||||||
builder: (context, constraints) {
|
builder: (context, constraints) {
|
||||||
// Determine the fit using the aspect ratio
|
// Determine the fit using the aspect ratio
|
||||||
BoxFit fit = BoxFit.fitWidth;
|
BoxFit fit = BoxFit.fitWidth;
|
||||||
if (asset.width != null && asset.height != null) {
|
if (asset.width != null && asset.height != null) {
|
||||||
final aspectRatio = asset.width! / asset.height!;
|
final aspectRatio = asset.height! / asset.width!;
|
||||||
final phoneAspectRatio =
|
final phoneAspectRatio =
|
||||||
constraints.maxWidth / constraints.maxHeight;
|
constraints.maxWidth / constraints.maxHeight;
|
||||||
// Look for a 25% difference in either direction
|
// Look for a 25% difference in either direction
|
||||||
|
@ -77,6 +76,7 @@ class MemoryCard extends StatelessWidget {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (asset.isImage) {
|
||||||
return Hero(
|
return Hero(
|
||||||
tag: 'memory-${asset.id}',
|
tag: 'memory-${asset.id}',
|
||||||
child: ImmichImage(
|
child: ImmichImage(
|
||||||
|
@ -88,8 +88,25 @@ class MemoryCard extends StatelessWidget {
|
||||||
preferredLocalAssetSize: 2048,
|
preferredLocalAssetSize: 2048,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
},
|
} else {
|
||||||
|
return Hero(
|
||||||
|
tag: 'memory-${asset.id}',
|
||||||
|
child: VideoViewerPage(
|
||||||
|
asset: asset,
|
||||||
|
showDownloadingIndicator: false,
|
||||||
|
placeholder: ImmichImage(
|
||||||
|
asset,
|
||||||
|
fit: fit,
|
||||||
|
type: ThumbnailFormat.JPEG,
|
||||||
|
preferredLocalAssetSize: 2048,
|
||||||
),
|
),
|
||||||
|
hideControlsTimer: const Duration(seconds: 2),
|
||||||
|
onVideoEnded: onVideoEnded,
|
||||||
|
showControls: false,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
),
|
),
|
||||||
if (showTitle)
|
if (showTitle)
|
||||||
Positioned(
|
Positioned(
|
||||||
|
|
|
@ -245,13 +245,25 @@ class MemoryPage extends HookConsumerWidget {
|
||||||
itemCount: memories[mIndex].assets.length,
|
itemCount: memories[mIndex].assets.length,
|
||||||
itemBuilder: (context, index) {
|
itemBuilder: (context, index) {
|
||||||
final asset = memories[mIndex].assets[index];
|
final asset = memories[mIndex].assets[index];
|
||||||
return Container(
|
return GestureDetector(
|
||||||
|
behavior: HitTestBehavior.translucent,
|
||||||
|
onTap: () {
|
||||||
|
toNextAsset(index);
|
||||||
|
},
|
||||||
|
child: Container(
|
||||||
color: Colors.black,
|
color: Colors.black,
|
||||||
child: MemoryCard(
|
child: MemoryCard(
|
||||||
asset: asset,
|
asset: asset,
|
||||||
onTap: () => toNextAsset(index),
|
|
||||||
title: memories[mIndex].title,
|
title: memories[mIndex].title,
|
||||||
showTitle: index == 0,
|
showTitle: index == 0,
|
||||||
|
onVideoEnded: () {
|
||||||
|
// If this is a live photo, don't go to
|
||||||
|
// next asset
|
||||||
|
if (asset.livePhotoVideoId == null) {
|
||||||
|
toNextAsset(index);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
|
|
Loading…
Reference in a new issue