Flutter使用Android原生播放器详解
作者:白玉梁 发布时间:2023-04-11 20:23:00
标签:Flutter,播放器,Android,原生
接上篇:播放器-IOS(Swift)篇
安卓端原生播放器的接入思路与ios基本一致,所以本篇就不废话了,直接上代码:
创建插件VideoViewPlugin实现FlutterPlugin:
package io.flutter.plugins.videoplayer;
import android.util.Log;
import androidx.annotation.NonNull;
import io.flutter.embedding.engine.plugins.FlutterPlugin;
import io.flutter.embedding.engine.plugins.activity.ActivityAware;
import io.flutter.embedding.engine.plugins.activity.ActivityPluginBinding;
public class VideoViewPlugin implements FlutterPlugin, ActivityAware {
private final static String TAG = "VideoViewPlugin";
FlutterPluginBinding fpBinding;
@Override
public void onAttachedToEngine(@NonNull FlutterPluginBinding binding) {
fpBinding = binding;
Log.e(TAG, "onAttachedToEngine");
}
@Override
public void onDetachedFromEngine(@NonNull FlutterPluginBinding binding) {
Log.e(TAG, "onDetachedFromEngine");
}
@Override
public void onAttachedToActivity(@NonNull ActivityPluginBinding apBinding) {
fpBinding.getPlatformViewRegistry().registerViewFactory("plugins.my_video_player/view", new VideoViewFactory(fpBinding, apBinding));
Log.e(TAG, "onAttachedToActivity");
}
@Override
public void onDetachedFromActivityForConfigChanges() {
Log.e(TAG, "onDetachedFromActivityForConfigChanges");
}
@Override
public void onReattachedToActivityForConfigChanges(@NonNull ActivityPluginBinding binding) {
Log.e(TAG, "onReattachedToActivityForConfigChanges");
}
@Override
public void onDetachedFromActivity() {
Log.e(TAG, "onDetachedFromActivity");
}
}
由于引用视频播放器时需要用到Activity的context,所以实现了ActivityAware接口,在onAttachedToActivity方法中注册PlatformViewFactory!
创建VideoViewFactory实现PlatformViewFactory:
package io.flutter.plugins.videoplayer;
import android.content.Context;
import io.flutter.Log;
import io.flutter.embedding.engine.plugins.FlutterPlugin;
import io.flutter.embedding.engine.plugins.activity.ActivityPluginBinding;
import io.flutter.plugin.common.StandardMessageCodec;
import io.flutter.plugin.platform.PlatformView;
import io.flutter.plugin.platform.PlatformViewFactory;
public class VideoViewFactory extends PlatformViewFactory {
private final static String TAG = "VideoViewFactory";
private final FlutterPlugin.FlutterPluginBinding fpBinding;
private final ActivityPluginBinding apBinding;
public VideoViewFactory(FlutterPlugin.FlutterPluginBinding fpBinding, ActivityPluginBinding apBinding) {
super(StandardMessageCodec.INSTANCE);
Log.e(TAG, "VideoViewFactory");
this.fpBinding = fpBinding;
this.apBinding = apBinding;
}
@Override
public PlatformView create(Context context, int viewId, Object args) {
Log.e(TAG, "PlatformView-create:" + args.toString());
return new VideoViewPlayer(args.toString(), fpBinding, apBinding);
}
}
创建VideoViewPlayer,实现PlatformView和MethodChannel.MethodCallHandler:
package io.flutter.plugins.videoplayer;
import android.app.Activity;
import android.content.Context;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import com.th.kjjl_flutter.R;
import com.videoplayer.player.VideoView;
import io.flutter.embedding.engine.plugins.FlutterPlugin;
import io.flutter.embedding.engine.plugins.activity.ActivityPluginBinding;
import io.flutter.plugin.common.MethodCall;
import io.flutter.plugin.common.MethodChannel;
import io.flutter.plugin.platform.PlatformView;
public class VideoViewPlayer implements PlatformView, MethodChannel.MethodCallHandler {
private final static String TAG = "VideoPlayerView";
Activity context;
private VideoView videoView;
private MethodChannel methodChannel;
VideoViewPlayer(String viewId, FlutterPlugin.FlutterPluginBinding fpBinding, ActivityPluginBinding apBinding) {
this.context = apBinding.getActivity();
videoView = (VideoView) LayoutInflater.from(context).inflate(R.layout.video_player, null);
methodChannel = new MethodChannel(fpBinding.getBinaryMessenger(), "my_video_player_" + viewId);
methodChannel.setMethodCallHandler(this);
}
@Override
public void onMethodCall(@NonNull MethodCall call, @NonNull MethodChannel.Result result) {
Log.e(TAG, "call.method>>" + call.method);
switch (call.method) {
case "setUrl":
String url = call.arguments.toString();
videoView.initVideoController(context);
videoView.setUrl(url);
break;
case "start":
videoView.start();
break;
case "pause":
videoView.pause();
break;
case "release":
videoView.pause();
videoView.release();
break;
case "stopFullScreen":
videoView.stopFullScreen();
break;
default:
result.notImplemented();
}
}
@Nullable
@Override
public View getView() {
return videoView;
}
@Override
public void dispose() {
Log.e(TAG, "dispose>>");
videoView.pause();
videoView.release();
methodChannel.setMethodCallHandler(null);
methodChannel = null;
}
}
其中的VideoView即引用的第三方播放器库,你可以根据自己情况,使用常见的安卓端开源播放器如GSY,饺子,DKPlayer等!
video_player.xml:
<?xml version="1.0" encoding="utf-8"?>
<com.videoplayer.player.VideoView xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/videoView"
android:layout_width="match_parent"
android:layout_height="match_parent" />
在MainActivity中注册插件:
class MainActivity : FlutterActivity() {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
flutterEngine!!.plugins.add(VideoViewPlugin())
}
}
注意:不要在GeneratedPluginRegistrant中去注册插件包括IOS,这个类由系统自动生成其它第三方插件注册代码!安卓端在MainActivity中注册,IOS端在AppDelegate中注册!
flutter中的引用以及通信方法,上一篇已经写了,本篇就不再重复了!需要注意的是,插件名,插件id,methodChannel等,安卓,IOS和Flutter三端一定要一致!
来源:https://blog.csdn.net/baiyuliang2013/article/details/129228342
0
投稿
猜你喜欢
- 写在前面现在,越来越多的App里面使用了模糊效果,这种模糊效果称之为高斯模糊。大家都知道,在Android平台上进行模糊渲染是一个相当耗CP
- 应用情景:很多标准的方法都是利用Object.Equals方法来做对比的,例如LIst.Remove假设 某些情景下我们希望引用类型判断&a
- 采取的方法是Fragment+FragmentTabHost组件来实现这种常见的app主页面的效果首先给出main.xml文件
- 使用 ResponseEntity 实现文件上传和下载在 static 下新建一个 img ,并且我放了一张图片在里面,然后重新 maven
- Join子句一、简介使用join子句可以将来自不同源序列并且在对象模型中没有直接关系的元素相关联,唯一的要求是每个源中的元素需要共享某个可以
- 一、单链表(Linked List)简介二、单链表的各种操作1.单链表的创建和遍历2.单链表的按顺序插入节点 以及节点的修改3.单链表节点的
- 一、为何使用内部类内部类提供了更好的封装,只有外部类能访问内部类内部类可以独立继承一个接口,不受外部类是否继承接口影响内部类中的属性和方法即
- 可以用于简单的过期订单取消支付、7天自动收货场景中1、Spring Boot整合redis 参考https://www.jb51.net/a
- 本文实例讲述了Android编程实现状态保存的方法。分享给大家供大家参考,具体如下:1、当我们正在发短信的时候,已经写了几百字了,这时突然来
- 通过这篇文章通过实例代码向大家介绍了Spring实例化bean的几种方法,接下来看看具体内容吧。1.使用类构造器实现实例化(bean的自身构
- 获取自定义菜单查询返回的结果有乱码解决方法:string Posturl = "https://api.weixin.qq.com
- 概述Spring针对Java Transaction API (JTA)、JDBC、Hibernate和Java Persistence A
- Java中有两种处理异常的方式,分别是用throws抛出异常、用try、catch捕获异常。try-catch在Javatry-catch语
- 标准函数标准函数就是在Standard.kt文件中定义的函数,任何Kotlin代码都可以自由地调用所有的标准函数let函数就属于是一个标准函
- 水印种类及功能介绍 PDF水印分为两种:文本水印和图片水印。文本水印一般被用在商业领域,提醒读者该文档是受版权保护的,其他人不能抄
- 还是有一些小问题....懒得改了,但大体思路还是清晰的,首先定义一个运算符栈,一个数栈。关于优先级,两个括号(,)优先级最低,其次是+、-,
- Java操作Redis的方式有下面两种:一、jedis(1)maven配置<dependency> <grou
- C#事件sender的小用法开WPF新坑了,看了WPF的炫酷界面,再看看winForm实在是有些惨不忍睹(逃)。后面会开始写一些短的学习笔记
- 注意:导包的时候API 11之前: android.text.ClipboardManagerAPI 11之后: android.conte
- 本文实例讲述了C#中Arraylist的sort函数用法。分享给大家供大家参考。具体如下:ArrayList的sort函数有几种比较常用的重