mirror of
https://github.com/immich-app/immich.git
synced 2024-12-19 00:32:49 +02:00
69091e5309
* Videos play in memories now Remove auto video advance * to next asset after video is done playing --------- Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
253 lines
7.3 KiB
Dart
253 lines
7.3 KiB
Dart
import 'dart:io';
|
|
|
|
import 'package:auto_route/auto_route.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
import 'package:chewie/chewie.dart';
|
|
import 'package:immich_mobile/extensions/build_context_extensions.dart';
|
|
import 'package:immich_mobile/modules/asset_viewer/models/image_viewer_page_state.model.dart';
|
|
import 'package:immich_mobile/modules/asset_viewer/providers/image_viewer_page_state.provider.dart';
|
|
import 'package:immich_mobile/modules/asset_viewer/ui/video_player_controls.dart';
|
|
import 'package:immich_mobile/shared/models/asset.dart';
|
|
import 'package:immich_mobile/shared/models/store.dart';
|
|
import 'package:immich_mobile/shared/ui/immich_loading_indicator.dart';
|
|
import 'package:photo_manager/photo_manager.dart';
|
|
import 'package:video_player/video_player.dart';
|
|
import 'package:wakelock_plus/wakelock_plus.dart';
|
|
|
|
@RoutePage()
|
|
// ignore: must_be_immutable
|
|
class VideoViewerPage extends HookConsumerWidget {
|
|
final Asset asset;
|
|
final bool isMotionVideo;
|
|
final Widget? placeholder;
|
|
final VoidCallback? onVideoEnded;
|
|
final VoidCallback? onPlaying;
|
|
final VoidCallback? onPaused;
|
|
final Duration hideControlsTimer;
|
|
final bool showControls;
|
|
final bool showDownloadingIndicator;
|
|
|
|
const VideoViewerPage({
|
|
super.key,
|
|
required this.asset,
|
|
this.isMotionVideo = false,
|
|
this.onVideoEnded,
|
|
this.onPlaying,
|
|
this.onPaused,
|
|
this.placeholder,
|
|
this.showControls = true,
|
|
this.hideControlsTimer = const Duration(seconds: 5),
|
|
this.showDownloadingIndicator = true,
|
|
});
|
|
|
|
@override
|
|
Widget build(BuildContext context, WidgetRef ref) {
|
|
if (asset.isLocal && asset.livePhotoVideoId == null) {
|
|
final AsyncValue<File> videoFile = ref.watch(_fileFamily(asset.local!));
|
|
return AnimatedSwitcher(
|
|
duration: const Duration(milliseconds: 200),
|
|
child: videoFile.when(
|
|
data: (data) => VideoPlayer(
|
|
file: data,
|
|
isMotionVideo: false,
|
|
onVideoEnded: () {},
|
|
),
|
|
error: (error, stackTrace) => Icon(
|
|
Icons.image_not_supported_outlined,
|
|
color: context.primaryColor,
|
|
),
|
|
loading: () => showDownloadingIndicator
|
|
? const Center(child: ImmichLoadingIndicator())
|
|
: Container(),
|
|
),
|
|
);
|
|
}
|
|
final downloadAssetStatus =
|
|
ref.watch(imageViewerStateProvider).downloadAssetStatus;
|
|
final String videoUrl = isMotionVideo
|
|
? '${Store.get(StoreKey.serverEndpoint)}/asset/file/${asset.livePhotoVideoId}'
|
|
: '${Store.get(StoreKey.serverEndpoint)}/asset/file/${asset.remoteId}';
|
|
|
|
return Stack(
|
|
children: [
|
|
VideoPlayer(
|
|
url: videoUrl,
|
|
accessToken: Store.get(StoreKey.accessToken),
|
|
isMotionVideo: isMotionVideo,
|
|
onVideoEnded: onVideoEnded,
|
|
onPaused: onPaused,
|
|
onPlaying: onPlaying,
|
|
placeholder: placeholder,
|
|
hideControlsTimer: hideControlsTimer,
|
|
showControls: showControls,
|
|
showDownloadingIndicator: showDownloadingIndicator,
|
|
),
|
|
AnimatedOpacity(
|
|
duration: const Duration(milliseconds: 400),
|
|
opacity: (downloadAssetStatus == DownloadAssetStatus.loading &&
|
|
showDownloadingIndicator)
|
|
? 1.0
|
|
: 0.0,
|
|
child: SizedBox(
|
|
height: context.height,
|
|
width: context.width,
|
|
child: const Center(
|
|
child: ImmichLoadingIndicator(),
|
|
),
|
|
),
|
|
),
|
|
],
|
|
);
|
|
}
|
|
}
|
|
|
|
final _fileFamily =
|
|
FutureProvider.family<File, AssetEntity>((ref, entity) async {
|
|
final file = await entity.file;
|
|
if (file == null) {
|
|
throw Exception();
|
|
}
|
|
return file;
|
|
});
|
|
|
|
class VideoPlayer extends StatefulWidget {
|
|
final String? url;
|
|
final String? accessToken;
|
|
final File? file;
|
|
final bool isMotionVideo;
|
|
final VoidCallback? onVideoEnded;
|
|
final Duration hideControlsTimer;
|
|
final bool showControls;
|
|
|
|
final Function()? onPlaying;
|
|
final Function()? onPaused;
|
|
|
|
/// The placeholder to show while the video is loading
|
|
/// usually, a thumbnail of the video
|
|
final Widget? placeholder;
|
|
|
|
final bool showDownloadingIndicator;
|
|
|
|
const VideoPlayer({
|
|
super.key,
|
|
this.url,
|
|
this.accessToken,
|
|
this.file,
|
|
this.onVideoEnded,
|
|
required this.isMotionVideo,
|
|
this.onPlaying,
|
|
this.onPaused,
|
|
this.placeholder,
|
|
this.hideControlsTimer = const Duration(
|
|
seconds: 5,
|
|
),
|
|
this.showControls = true,
|
|
this.showDownloadingIndicator = true,
|
|
});
|
|
|
|
@override
|
|
State<VideoPlayer> createState() => _VideoPlayerState();
|
|
}
|
|
|
|
class _VideoPlayerState extends State<VideoPlayer> {
|
|
late VideoPlayerController videoPlayerController;
|
|
ChewieController? chewieController;
|
|
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
initializePlayer();
|
|
|
|
videoPlayerController.addListener(() {
|
|
if (videoPlayerController.value.isInitialized) {
|
|
if (videoPlayerController.value.isPlaying) {
|
|
WakelockPlus.enable();
|
|
widget.onPlaying?.call();
|
|
} else if (!videoPlayerController.value.isPlaying) {
|
|
WakelockPlus.disable();
|
|
widget.onPaused?.call();
|
|
}
|
|
|
|
if (videoPlayerController.value.position ==
|
|
videoPlayerController.value.duration) {
|
|
WakelockPlus.disable();
|
|
widget.onVideoEnded?.call();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
Future<void> initializePlayer() async {
|
|
try {
|
|
videoPlayerController = widget.file == null
|
|
? VideoPlayerController.networkUrl(
|
|
Uri.parse(widget.url!),
|
|
httpHeaders: {"x-immich-user-token": widget.accessToken ?? ""},
|
|
)
|
|
: VideoPlayerController.file(widget.file!);
|
|
|
|
await videoPlayerController.initialize();
|
|
_createChewieController();
|
|
setState(() {});
|
|
} catch (e) {
|
|
debugPrint("ERROR initialize video player $e");
|
|
}
|
|
}
|
|
|
|
_createChewieController() {
|
|
chewieController = ChewieController(
|
|
controlsSafeAreaMinimum: const EdgeInsets.only(
|
|
bottom: 100,
|
|
),
|
|
showOptions: true,
|
|
showControlsOnInitialize: false,
|
|
videoPlayerController: videoPlayerController,
|
|
autoPlay: true,
|
|
autoInitialize: true,
|
|
allowFullScreen: false,
|
|
allowedScreenSleep: false,
|
|
showControls: widget.showControls && !widget.isMotionVideo,
|
|
customControls: const VideoPlayerControls(),
|
|
hideControlsTimer: widget.hideControlsTimer,
|
|
);
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
super.dispose();
|
|
videoPlayerController.pause();
|
|
videoPlayerController.dispose();
|
|
chewieController?.dispose();
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
if (chewieController?.videoPlayerController.value.isInitialized == true) {
|
|
return SizedBox(
|
|
height: context.height,
|
|
width: context.width,
|
|
child: Chewie(
|
|
controller: chewieController!,
|
|
),
|
|
);
|
|
} else {
|
|
return SizedBox(
|
|
height: context.height,
|
|
width: context.width,
|
|
child: Center(
|
|
child: Stack(
|
|
children: [
|
|
if (widget.placeholder != null) widget.placeholder!,
|
|
if (widget.showDownloadingIndicator)
|
|
const Center(
|
|
child: ImmichLoadingIndicator(),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
);
|
|
}
|
|
}
|
|
}
|