import 'dart:io'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:chewie/chewie.dart'; import 'package:flutter_gen/gen_l10n/l10n.dart'; import 'package:flutter_linkify/flutter_linkify.dart'; import 'package:matrix/matrix.dart'; import 'package:path_provider/path_provider.dart'; import 'package:universal_html/html.dart' as html; import 'package:video_player/video_player.dart'; import 'package:fluffychat/config/app_config.dart'; import 'package:fluffychat/pages/chat/events/image_bubble.dart'; import 'package:fluffychat/utils/file_description.dart'; import 'package:fluffychat/utils/localized_exception_extension.dart'; import 'package:fluffychat/utils/matrix_sdk_extensions/event_extension.dart'; import 'package:fluffychat/utils/platform_infos.dart'; import 'package:fluffychat/utils/url_launcher.dart'; import 'package:fluffychat/widgets/blur_hash.dart'; import '../../../utils/error_reporter.dart'; class EventVideoPlayer extends StatefulWidget { final Event event; final Color? textColor; final Color? linkColor; const EventVideoPlayer( this.event, { this.textColor, this.linkColor, super.key, }); @override EventVideoPlayerState createState() => EventVideoPlayerState(); } class EventVideoPlayerState extends State { ChewieController? _chewieController; VideoPlayerController? _videoPlayerController; bool _isDownloading = false; // The video_player package only doesn't support Windows and Linux. final _supportsVideoPlayer = !PlatformInfos.isWindows && !PlatformInfos.isLinux; void _downloadAction() async { if (!_supportsVideoPlayer) { widget.event.saveFile(context); return; } setState(() => _isDownloading = true); try { final videoFile = await widget.event.downloadAndDecryptAttachment(); // Dispose the controllers if we already have them. _disposeControllers(); late VideoPlayerController videoPlayerController; // Create the VideoPlayerController from the contents of videoFile. if (kIsWeb) { final blob = html.Blob([videoFile.bytes]); final networkUri = Uri.parse(html.Url.createObjectUrlFromBlob(blob)); videoPlayerController = VideoPlayerController.networkUrl(networkUri); } else { final tempDir = await getTemporaryDirectory(); final fileName = Uri.encodeComponent( widget.event.attachmentOrThumbnailMxcUrl()!.pathSegments.last, ); final file = File('${tempDir.path}/${fileName}_${videoFile.name}'); if (await file.exists() == false) { await file.writeAsBytes(videoFile.bytes); } videoPlayerController = VideoPlayerController.file(file); } _videoPlayerController = videoPlayerController; await videoPlayerController.initialize(); // Create a ChewieController on top. _chewieController = ChewieController( videoPlayerController: videoPlayerController, useRootNavigator: !kIsWeb, autoPlay: true, autoInitialize: true, ); } on IOException catch (e) { ScaffoldMessenger.of(context).showSnackBar( SnackBar( content: Text(e.toLocalizedString(context)), ), ); } catch (e, s) { ErrorReporter(context, 'Unable to play video').onErrorCallback(e, s); } finally { setState(() => _isDownloading = false); } } void _disposeControllers() { _chewieController?.dispose(); _videoPlayerController?.dispose(); _chewieController = null; _videoPlayerController = null; } @override void dispose() { _disposeControllers(); super.dispose(); } static const String fallbackBlurHash = 'L5H2EC=PM+yV0g-mq.wG9c010J}I'; @override Widget build(BuildContext context) { final theme = Theme.of(context); final hasThumbnail = widget.event.hasThumbnail; final blurHash = (widget.event.infoMap as Map) .tryGet('xyz.amorgan.blurhash') ?? fallbackBlurHash; final fileDescription = widget.event.fileDescription; final textColor = widget.textColor; final linkColor = widget.linkColor; const width = 300.0; final chewieController = _chewieController; return Column( mainAxisSize: MainAxisSize.min, spacing: 8, children: [ Material( color: Colors.black, borderRadius: BorderRadius.circular(AppConfig.borderRadius), child: SizedBox( height: width, child: chewieController != null ? Center(child: Chewie(controller: chewieController)) : Stack( children: [ if (hasThumbnail) Center( child: ImageBubble( widget.event, tapToView: false, textColor: widget.textColor, ), ) else BlurHash( blurhash: blurHash, width: width, height: width, ), Center( child: IconButton( style: IconButton.styleFrom( backgroundColor: theme.colorScheme.surface, ), icon: _isDownloading ? const SizedBox( width: 24, height: 24, child: CircularProgressIndicator.adaptive( strokeWidth: 2, ), ) : _supportsVideoPlayer ? const Icon(Icons.play_circle_outlined) : const Icon(Icons.file_download_outlined), tooltip: _isDownloading ? L10n.of(context).loadingPleaseWait : L10n.of(context).videoWithSize( widget.event.sizeString ?? '?MB', ), onPressed: _isDownloading ? null : _downloadAction, ), ), ], ), ), ), if (fileDescription != null && textColor != null && linkColor != null) SizedBox( width: width, child: Linkify( text: fileDescription, textScaleFactor: MediaQuery.textScalerOf(context).scale(1), style: TextStyle( color: textColor, fontSize: AppConfig.fontSizeFactor * AppConfig.messageFontSize, ), options: const LinkifyOptions(humanize: false), linkStyle: TextStyle( color: linkColor, fontSize: AppConfig.fontSizeFactor * AppConfig.messageFontSize, decoration: TextDecoration.underline, decorationColor: linkColor, ), onOpen: (url) => UrlLauncher(context, url.url).launchUrl(), ), ), ], ); } }