Flutter动态照片处理插件motion_photos的使用
Flutter动态照片处理插件motion_photos的使用
简介
motion_photos
是一个用于检测和提取动态照片(Motion Photos)中视频内容的Flutter插件。该插件由 ente.io 开发。
相关博客
功能
isMotionPhoto
: 检测给定文件是否为动态照片。getMotionVideoIndex
: 提取动态照片的起始和结束索引。getMotionVideo
: 返回动态照片中的视频内容的字节数据 (Uint8List
)。getMotionVideoFile
: 提取并返回动态照片中的视频内容的MP4文件。
使用入门
添加依赖
-
通过终端添加依赖:
flutter pub get motion_photos
-
或者在
pubspec.yaml
文件中添加以下内容:dependencies: flutter: sdk: flutter motion_photos:
示例代码
完整示例Demo
import 'dart:io';
import 'package:flutter/material.dart';
import 'package:motion_photos/motion_photos.dart';
import 'package:file_picker/file_picker.dart';
import 'package:video_player/video_player.dart';
void main() {
runApp(const MyApp());
}
class MyApp extends StatelessWidget {
const MyApp({super.key});
[@override](/user/override)
Widget build(BuildContext context) {
return MaterialApp(
title: 'Motion Photo Example (from ente.io team)',
debugShowCheckedModeBanner: false,
theme: ThemeData(
primarySwatch: Colors.deepPurple,
),
home: const MyHomePage(title: 'Motion Photo Example'),
);
}
}
class MyHomePage extends StatefulWidget {
const MyHomePage({super.key, required this.title});
final String title;
[@override](/user/override)
State<MyHomePage> createState() => _MyHomePageState();
}
class _MyHomePageState extends State<MyHomePage> {
late MotionPhotos motionPhotos;
bool? _isMotionPhoto;
VideoIndex? videoIndex;
bool isPicked = false;
late VideoPlayerController _controller;
Future<void> _pickFromGallery() async {
try {
FilePickerResult? result = await FilePicker.platform.pickFiles(
type: FileType.image,
allowMultiple: false,
allowCompression: false,
);
if (result != null) {
final path = result.files.single.path!;
motionPhotos = MotionPhotos(path);
_isMotionPhoto = await motionPhotos.isMotionPhoto();
if (_isMotionPhoto!) {
videoIndex = await motionPhotos.getMotionVideoIndex();
}
setState(() {
isPicked = true;
});
}
} catch (e) {
print('Error: $e');
}
}
Future<Widget> _playVideo() async {
if (isPicked && (_isMotionPhoto ?? false)) {
try {
File file = await motionPhotos.getMotionVideoFile();
_controller = VideoPlayerController.file(file)
..initialize().then((_) {
setState(() {});
_controller.play();
});
return VideoPlayer(_controller);
} catch (e) {
return Text(e.toString(), style: const TextStyle(color: Colors.red));
}
}
return const SizedBox.shrink();
}
String printIsMotionPhoto() {
if (isPicked && _isMotionPhoto != null) {
return _isMotionPhoto! ? 'Yes' : 'No';
}
return 'TBA';
}
String printVideoIndex() {
if (isPicked && videoIndex != null) {
return '''
Start Index: ${videoIndex!.start}
End Index: ${videoIndex!.end}
Video Size: ${videoIndex!.videoLength}
''';
}
return 'NA';
}
[@override](/user/override)
void dispose() {
_controller.dispose();
super.dispose();
}
[@override](/user/override)
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(widget.title),
),
body: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Text(
'Is MotionPhoto: ${printIsMotionPhoto()}',
style: const TextStyle(fontSize: 16, fontWeight: FontWeight.w600),
),
const SizedBox(height: 20),
const Text('Video Info', style: TextStyle(fontSize: 16, fontWeight: FontWeight.w600)),
Text(printVideoIndex()),
const SizedBox(height: 20),
Container(
color: Colors.transparent,
width: double.infinity,
height: 300,
child: FutureBuilder<Widget>(
future: _playVideo(),
builder: (context, snapshot) {
if (snapshot.connectionState == ConnectionState.done && snapshot.hasData) {
return snapshot.data!;
} else {
return const Center(child: CircularProgressIndicator());
}
},
),
),
],
),
floatingActionButton: FloatingActionButton(
onPressed: _pickFromGallery,
child: const Icon(Icons.image),
),
);
}
}
更多关于Flutter动态照片处理插件motion_photos的使用的实战系列教程也可以访问 https://www.itying.com/category-92-b0.html
1 回复