当前位置: 首页 > news >正文

flutter实现视频播放器,可根据指定视频地址播放、设置声音,进度条拖动,下载等

需要装依赖:

  gallery_saver: ^2.3.2video_player: ^2.8.3

AndroidManifest.xml

<uses-permission android:name="android.permission.INTERNET"/>

实现代码

import 'dart:async';
import 'dart:io';import 'package:flutter/material.dart';
import 'package:gallery_saver/gallery_saver.dart';
import 'package:path_provider/path_provider.dart';
import 'package:video_player/video_player.dart';
import 'package:http/http.dart' as http;void main() => runApp(const VideoApp());class VideoApp extends StatefulWidget {const VideoApp({super.key});@override_VideoAppState createState() => _VideoAppState();
}class _VideoAppState extends State<VideoApp> {late VideoPlayerController _controller;bool isMuted = false;double _currentSliderValue = 0.0;Timer? _timer;double _playbackSpeed = 1.0;@overridevoid initState() {super.initState();_controller = VideoPlayerController.networkUrl(Uri.parse('你的播放地址'))..initialize().then((_) {setState(() {});});_controller.setLooping(true);_controller.play();_timer = Timer.periodic(Duration(seconds: 1), (Timer timer) {if (_controller.value.isPlaying) {setState(() {_currentSliderValue = _controller.value.position.inMilliseconds.toDouble();});}});}Future<void> downloadAndSaveVideo(String url) async {var httpClient = http.Client();var response = await httpClient.get(Uri.parse(url));var bytes = response.bodyBytes;var tempDir = await getTemporaryDirectory();var tempFile = File('${tempDir.path}/temp_video.mp4');await tempFile.writeAsBytes(bytes);final result = await GallerySaver.saveVideo(tempFile.path);print('Video saved to gallery: $result,path: ${tempFile.path}');}@overrideWidget build(BuildContext context) {return MaterialApp(title: 'Video Demo',home: Scaffold(appBar: AppBar(title: Text('Video Demo'),),body: Center(child: Column(mainAxisAlignment: MainAxisAlignment.center,children: <Widget>[_controller.value.isInitialized? AspectRatio(aspectRatio: _controller.value.aspectRatio,child: VideoPlayer(_controller),):const  CircularProgressIndicator(),const SizedBox(height: 5),ElevatedButton(onPressed: () {setState(() {isMuted = !isMuted;_controller.setVolume(isMuted ? 0.0 : 1.0);});},child: Text(isMuted ? '静音模式' : '开启声音模式'),),ElevatedButton(onPressed: () {setState(() {if (_controller.value.isPlaying) {_controller.pause();} else {_controller.play();}});},child: Text(_controller.value.isPlaying ? '暂停' : '播放'),),Slider(value: _currentSliderValue,min: 0.0,max: _controller.value.duration.inMilliseconds.toDouble(),onChanged: (double value) {setState(() {_currentSliderValue = value;_controller.seekTo(Duration(milliseconds: value.toInt()));});},),const SizedBox(height: 5),Text('Playback Speed: ${_playbackSpeed.toStringAsFixed(1)}x'),Slider(value: _playbackSpeed,min: 0.5,max: 2.0,divisions: 15,label: '${_playbackSpeed.toStringAsFixed(1)}x',onChanged: (double value) {setState(() {_playbackSpeed = value;_controller.setPlaybackSpeed(_playbackSpeed);});},),ElevatedButton(onPressed: () {downloadAndSaveVideo(_controller.dataSource);},child: Text('下载'),),],),),),);}@overridevoid dispose() {super.dispose();_controller.dispose();_timer?.cancel();}
}

http://www.lryc.cn/news/324818.html

相关文章:

  • 微服务(基础篇-001-介绍、Eureka)
  • mac 解决随机出现的蓝色框
  • 深入理解与使用go之函数与方法--使用
  • 【QT问题】 Qt信号函数如果重名,调用怎么处理
  • 登山小分队(dfs,模拟)
  • Luminar Neo:重塑图像编辑新纪元,Mac与Win双平台畅享创意之旅
  • 计算机二级Python题库深度解析与备考策略
  • 微信商家转账到零钱:实用指南,涵盖开通、使用与常见问题
  • [精选]Kimi到底是什么,将带来什么?
  • MySQL学习笔记------SQL(2)
  • 【循环神经网络rnn】一篇文章讲透
  • KW音乐搜索参数
  • SpringBoot3+Vue3项目的阿里云部署--将后端以及前端项目打包
  • MySQL 存储引擎
  • perl:打开文件夹,选择视频文件,并播放
  • 分布式链上随机数和keyless account
  • 【项目设计】基于MVC的负载均衡式的在线OJ
  • MRC是谁?- 媒体评级委员会 Media Rating Council
  • 反序列化漏洞简单知识
  • Es之正排索引与倒排索引
  • wordpress将图片默认连接到媒体文件
  • Java学习笔记 | Java基础语法 | 03 | 流程控制语句
  • 记录新人的web3之旅
  • 由浅到深认识Java语言(9):Eclipse IDE简介
  • 游戏引擎中的地形系统
  • 【论文精读】OTA: Optimal Transport Assignment for Object Detection(物体探测的最优传输分配)
  • 无极低码SQL模板引擎使用教程示例,自己手撸一个sql模板引擎进行动态sql生成。
  • Python学习(一)
  • Day62:WEB攻防-PHP反序列化CLI框架类PHPGGC生成器TPYiiLaravel等利用
  • 运动想象 (MI) 迁移学习系列 (14) : EEGNet-Fine tuning