mirror of
https://github.com/ente-io/ente.git
synced 2025-07-13 10:28:42 +00:00
369 lines
11 KiB
Dart
369 lines
11 KiB
Dart
import "dart:async";
|
|
import "dart:io";
|
|
|
|
import "package:flutter/cupertino.dart";
|
|
import "package:flutter/material.dart";
|
|
import "package:logging/logging.dart";
|
|
import "package:media_kit/media_kit.dart";
|
|
import "package:media_kit_video/media_kit_video.dart";
|
|
import "package:native_video_player/native_video_player.dart";
|
|
import "package:photos/core/constants.dart";
|
|
import "package:photos/core/event_bus.dart";
|
|
import "package:photos/events/file_swipe_lock_event.dart";
|
|
import "package:photos/events/pause_video_event.dart";
|
|
import "package:photos/generated/l10n.dart";
|
|
import "package:photos/models/file/extensions/file_props.dart";
|
|
import "package:photos/models/file/file.dart";
|
|
import "package:photos/services/files_service.dart";
|
|
import "package:photos/theme/colors.dart";
|
|
import "package:photos/theme/ente_theme.dart";
|
|
import "package:photos/ui/actions/file/file_actions.dart";
|
|
import "package:photos/ui/viewer/file/thumbnail_widget.dart";
|
|
import "package:photos/utils/dialog_util.dart";
|
|
import "package:photos/utils/exif_util.dart";
|
|
import "package:photos/utils/file_util.dart";
|
|
import "package:photos/utils/toast_util.dart";
|
|
|
|
class VideoWidgetNative extends StatefulWidget {
|
|
final EnteFile file;
|
|
final String? tagPrefix;
|
|
final Function(bool)? playbackCallback;
|
|
const VideoWidgetNative(
|
|
this.file, {
|
|
this.tagPrefix,
|
|
this.playbackCallback,
|
|
super.key,
|
|
});
|
|
|
|
@override
|
|
State<VideoWidgetNative> createState() => _VideoWidgetNativeState();
|
|
}
|
|
|
|
class _VideoWidgetNativeState extends State<VideoWidgetNative>
|
|
with WidgetsBindingObserver {
|
|
final Logger _logger = Logger("VideoWidgetNew");
|
|
static const verticalMargin = 72.0;
|
|
late final player = Player();
|
|
VideoController? controller;
|
|
final _progressNotifier = ValueNotifier<double?>(null);
|
|
late StreamSubscription<bool> playingStreamSubscription;
|
|
bool _isAppInFG = true;
|
|
late StreamSubscription<PauseVideoEvent> pauseVideoSubscription;
|
|
bool _isFileSwipeLocked = false;
|
|
late final StreamSubscription<FileSwipeLockEvent>
|
|
_fileSwipeLockEventSubscription;
|
|
|
|
String? _filePath;
|
|
double? aspectRatio;
|
|
|
|
@override
|
|
void initState() {
|
|
_logger.info(
|
|
'initState for ${widget.file.generatedID} with tag ${widget.file.tag} and name ${widget.file.displayName}',
|
|
);
|
|
super.initState();
|
|
WidgetsBinding.instance.addObserver(this);
|
|
if (widget.file.isRemoteFile) {
|
|
_loadNetworkVideo();
|
|
_setFileSizeIfNull();
|
|
} else if (widget.file.isSharedMediaToAppSandbox) {
|
|
final localFile = File(getSharedMediaFilePath(widget.file));
|
|
if (localFile.existsSync()) {
|
|
_setFilePathForNativePlayer(localFile.path);
|
|
} else if (widget.file.uploadedFileID != null) {
|
|
_loadNetworkVideo();
|
|
}
|
|
} else {
|
|
widget.file.getAsset.then((asset) async {
|
|
if (asset == null || !(await asset.exists)) {
|
|
if (widget.file.uploadedFileID != null) {
|
|
_loadNetworkVideo();
|
|
}
|
|
} else {
|
|
// ignore: unawaited_futures
|
|
getFile(widget.file, isOrigin: true).then((file) {
|
|
_setFilePathForNativePlayer(file!.path);
|
|
file.delete();
|
|
});
|
|
}
|
|
});
|
|
}
|
|
playingStreamSubscription = player.stream.playing.listen((event) {
|
|
if (widget.playbackCallback != null && mounted) {
|
|
widget.playbackCallback!(event);
|
|
}
|
|
});
|
|
|
|
pauseVideoSubscription = Bus.instance.on<PauseVideoEvent>().listen((event) {
|
|
player.pause();
|
|
});
|
|
_fileSwipeLockEventSubscription =
|
|
Bus.instance.on<FileSwipeLockEvent>().listen((event) {
|
|
setState(() {
|
|
_isFileSwipeLocked = event.shouldSwipeLock;
|
|
});
|
|
});
|
|
}
|
|
|
|
@override
|
|
void didChangeAppLifecycleState(AppLifecycleState state) {
|
|
if (state == AppLifecycleState.resumed) {
|
|
_isAppInFG = true;
|
|
} else {
|
|
_isAppInFG = false;
|
|
}
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
_fileSwipeLockEventSubscription.cancel();
|
|
pauseVideoSubscription.cancel();
|
|
removeCallBack(widget.file);
|
|
_progressNotifier.dispose();
|
|
WidgetsBinding.instance.removeObserver(this);
|
|
playingStreamSubscription.cancel();
|
|
player.dispose();
|
|
super.dispose();
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return Hero(
|
|
tag: widget.tagPrefix! + widget.file.tag,
|
|
child: GestureDetector(
|
|
onVerticalDragUpdate: _isFileSwipeLocked
|
|
? null
|
|
: (d) => {
|
|
if (d.delta.dy > dragSensitivity)
|
|
{
|
|
Navigator.of(context).pop(),
|
|
}
|
|
else if (d.delta.dy < (dragSensitivity * -1))
|
|
{
|
|
showDetailsSheet(context, widget.file),
|
|
},
|
|
},
|
|
child: _filePath == null
|
|
? _getLoadingWidget()
|
|
: Center(
|
|
child: AspectRatio(
|
|
aspectRatio: aspectRatio ?? 1,
|
|
child: NativeVideoPlayerView(
|
|
onViewReady: (controller) async {
|
|
final videoSource = await VideoSource.init(
|
|
path: _filePath!,
|
|
type: VideoSourceType.file,
|
|
);
|
|
await controller.loadVideoSource(videoSource);
|
|
await controller.play();
|
|
},
|
|
),
|
|
),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
void _loadNetworkVideo() {
|
|
getFileFromServer(
|
|
widget.file,
|
|
progressCallback: (count, total) {
|
|
if (!mounted) {
|
|
return;
|
|
}
|
|
_progressNotifier.value = count / (widget.file.fileSize ?? total);
|
|
if (_progressNotifier.value == 1) {
|
|
if (mounted) {
|
|
showShortToast(context, S.of(context).decryptingVideo);
|
|
}
|
|
}
|
|
},
|
|
).then((file) {
|
|
if (file != null) {
|
|
_setFilePathForNativePlayer(file.path);
|
|
}
|
|
}).onError((error, stackTrace) {
|
|
showErrorDialog(context, "Error", S.of(context).failedToDownloadVideo);
|
|
});
|
|
}
|
|
|
|
void _setFileSizeIfNull() {
|
|
if (widget.file.fileSize == null && widget.file.canEditMetaInfo) {
|
|
FilesService.instance
|
|
.getFileSize(widget.file.uploadedFileID!)
|
|
.then((value) {
|
|
widget.file.fileSize = value;
|
|
if (mounted) {
|
|
setState(() {});
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
Widget _getLoadingWidget() {
|
|
return Stack(
|
|
children: [
|
|
_getThumbnail(),
|
|
Container(
|
|
color: Colors.black12,
|
|
constraints: const BoxConstraints.expand(),
|
|
),
|
|
Center(
|
|
child: SizedBox.fromSize(
|
|
size: const Size.square(20),
|
|
child: ValueListenableBuilder(
|
|
valueListenable: _progressNotifier,
|
|
builder: (BuildContext context, double? progress, _) {
|
|
return progress == null || progress == 1
|
|
? const CupertinoActivityIndicator(
|
|
color: Colors.white,
|
|
)
|
|
: CircularProgressIndicator(
|
|
backgroundColor: Colors.black,
|
|
value: progress,
|
|
valueColor: const AlwaysStoppedAnimation<Color>(
|
|
Color.fromRGBO(45, 194, 98, 1.0),
|
|
),
|
|
);
|
|
},
|
|
),
|
|
),
|
|
),
|
|
],
|
|
);
|
|
}
|
|
|
|
Widget _getThumbnail() {
|
|
return Container(
|
|
color: Colors.black,
|
|
constraints: const BoxConstraints.expand(),
|
|
child: ThumbnailWidget(
|
|
widget.file,
|
|
fit: BoxFit.contain,
|
|
),
|
|
);
|
|
}
|
|
|
|
void _setFilePathForNativePlayer(String url) {
|
|
if (mounted) {
|
|
setState(() {
|
|
_filePath = url;
|
|
});
|
|
_setAspectRatioFromVideoProps().then((_) {
|
|
setState(() {});
|
|
});
|
|
}
|
|
}
|
|
|
|
Future<void> _setAspectRatioFromVideoProps() async {
|
|
final videoProps = await getVideoPropsAsync(File(_filePath!));
|
|
if (videoProps != null) {
|
|
if (videoProps.width != null && videoProps.height != null) {
|
|
if (videoProps.width != null && videoProps.height != 0) {
|
|
aspectRatio = videoProps.width! / videoProps.height!;
|
|
} else {
|
|
_logger.info("Video props height or width is 0");
|
|
aspectRatio = 1;
|
|
}
|
|
} else {
|
|
_logger.info("Video props width and height are null");
|
|
aspectRatio = 1;
|
|
}
|
|
} else {
|
|
_logger.info("Video props are null");
|
|
aspectRatio = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
class PausePlayAndDuration extends StatefulWidget {
|
|
final Player? player;
|
|
const PausePlayAndDuration(this.player, {super.key});
|
|
|
|
@override
|
|
State<PausePlayAndDuration> createState() => _PausePlayAndDurationState();
|
|
}
|
|
|
|
class _PausePlayAndDurationState extends State<PausePlayAndDuration> {
|
|
Color backgroundColor = fillStrongLight;
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return GestureDetector(
|
|
onTapDown: (details) {
|
|
setState(() {
|
|
backgroundColor = fillMutedDark;
|
|
});
|
|
},
|
|
onTapUp: (details) {
|
|
Future.delayed(const Duration(milliseconds: 175), () {
|
|
if (mounted) {
|
|
setState(() {
|
|
backgroundColor = fillStrongLight;
|
|
});
|
|
}
|
|
});
|
|
},
|
|
onTapCancel: () {
|
|
Future.delayed(const Duration(milliseconds: 175), () {
|
|
if (mounted) {
|
|
setState(() {
|
|
backgroundColor = fillStrongLight;
|
|
});
|
|
}
|
|
});
|
|
},
|
|
onTap: () => widget.player!.playOrPause(),
|
|
child: AnimatedContainer(
|
|
duration: const Duration(milliseconds: 150),
|
|
curve: Curves.easeInBack,
|
|
padding: const EdgeInsets.symmetric(vertical: 4, horizontal: 8),
|
|
decoration: BoxDecoration(
|
|
color: backgroundColor,
|
|
border: Border.all(
|
|
color: strokeFaintDark,
|
|
width: 1,
|
|
),
|
|
borderRadius: BorderRadius.circular(24),
|
|
),
|
|
child: AnimatedSize(
|
|
duration: const Duration(seconds: 2),
|
|
curve: Curves.easeInOutExpo,
|
|
child: Row(
|
|
children: [
|
|
StreamBuilder(
|
|
builder: (context, snapshot) {
|
|
final bool isPlaying = snapshot.data ?? false;
|
|
return AnimatedSwitcher(
|
|
duration: const Duration(milliseconds: 350),
|
|
switchInCurve: Curves.easeInOutCirc,
|
|
switchOutCurve: Curves.easeInOutCirc,
|
|
child: Icon(
|
|
key: ValueKey(
|
|
isPlaying ? "pause_button" : "play_button",
|
|
),
|
|
isPlaying
|
|
? Icons.pause_rounded
|
|
: Icons.play_arrow_rounded,
|
|
color: backdropBaseLight,
|
|
size: 24,
|
|
),
|
|
);
|
|
},
|
|
initialData: widget.player?.state.playing,
|
|
stream: widget.player?.stream.playing,
|
|
),
|
|
const SizedBox(width: 8),
|
|
MaterialPositionIndicator(
|
|
style: getEnteTextTheme(context).tiny.copyWith(
|
|
color: textBaseDark,
|
|
),
|
|
),
|
|
const SizedBox(width: 10),
|
|
],
|
|
),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
}
|