安卓漫漫路之实现简单的弹幕.

直播和看视频中越来越火的控件---弹幕(Danmaku)

本文即介绍怎样实现简单的弹幕效果:咱们使用的是哔哩哔哩开源的弹幕效果库 DanmakuFlameMaster.


必需:首先咱们在项目主工程app/build.gradle中的dependencies闭包中添加如下依赖:

	compile 'com.github.ctiao:DanmakuFlameMaster:0.5.3'

这样我们就将DanmakuFlameMaster库引入到当前项目中了.

开始展示Demo的代码给大家:


首先看咱们的布局文件:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="#000">  

    <VideoView
        android:id="@+id/video_view"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true"/>

    <master.flame.danmaku.ui.widget.DanmakuView
        android:id="@+id/danmaku_view"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />

    <LinearLayout
        android:id="@+id/operation_layout"
        android:layout_width="match_parent"
        android:layout_height="50dp"
        android:layout_alignParentBottom="true"
        android:background="#fff"
        android:visibility="gone">

        <EditText
            android:id="@+id/edit_text"
            android:layout_width="0dp"
            android:layout_height="match_parent"
            android:layout_weight="1"
            />

        <Button
            android:id="@+id/send"
            android:layout_width="wrap_content"
            android:layout_height="match_parent"
            android:text="发送" />
    </LinearLayout>

</RelativeLayout>
.
背景色为黑色,承载我是用的Video来播放本地或者网络视频,DanmakuView即是咱们的弹幕控件,LinearLayout内包含输入框和发送按钮.当然这些都是最普通的控件,最终的界面当然是由您来设定.

接下来看咱们的MainActivity的代码,如下:
.
import android.graphics.Color;
import android.os.Build;
import android.os.Bundle;
import android.os.Environment;
import android.support.v7.app.AppCompatActivity;
import android.text.TextUtils;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.LinearLayout;
import android.widget.VideoView;

import java.util.Random;

import master.flame.danmaku.controller.DrawHandler;
import master.flame.danmaku.danmaku.model.BaseDanmaku;
import master.flame.danmaku.danmaku.model.DanmakuTimer;
import master.flame.danmaku.danmaku.model.IDanmakus;
import master.flame.danmaku.danmaku.model.android.DanmakuContext;
import master.flame.danmaku.danmaku.model.android.Danmakus;
import master.flame.danmaku.danmaku.parser.BaseDanmakuParser;
import master.flame.danmaku.ui.widget.DanmakuView;

public class MainActivity extends AppCompatActivity {
    private boolean showDanmaku;
    //弹幕控件
    private DanmakuView danmakuView;
    //DanmakuContext  字体实例
    private DanmakuContext danmakuContext;
    private BaseDanmakuParser parser = new BaseDanmakuParser() {
        @Override
        protected IDanmakus parse() {
            return new Danmakus();
        }
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //初始化VideoView控件
        final VideoView videoView = (VideoView) findViewById(R.id.video_view);
        //指定好VideoView的本地路径地址  SD卡根目录的xxx.mp4文件
        videoView.setVideoPath(Environment.getExternalStorageDirectory() + "/xxx.mp4");
        //访问网络视频
        //Uri uri = Uri.parse("http://clips.vorwaerts-gmbh.de/big_buck_bunny.mp4");
        //设置视频控制器
        //videoView.setMediaController(new MediaController(this));
        //设置视频路径
        // videoView.setVideoURI(uri);
        //开始播放
        videoView.start();

        new android.os.Handler().postDelayed(new Runnable() {
            @Override
            public void run() {
                videoView.pause();
            }
        }, 0);

        //初始化弹幕控件
        danmakuView = (DanmakuView) findViewById(R.id.danmaku_view);
        //默认为true 在模拟器上运行有问题
        danmakuView.enableDanmakuDrawingCache(true);
        //看源码得知是一个接口    怎么实现还是要咱们去重写其中的方法
        danmakuView.setCallback(new DrawHandler.Callback() {
            @Override
            public void prepared() {
                //把变量置为 true
                showDanmaku = true;
                //开始运行弹幕控件
                danmakuView.start();
                //随机生成一些弹幕内容以供测试
                generateSomeDanmaku();
            }

            @Override
            public void updateTimer(DanmakuTimer timer) {

            }

            @Override
            public void danmakuShown(BaseDanmaku danmaku) {

            }

            @Override
            public void drawingFinished() {

            }
        });
        //调用  DanmakuContext.create() 完成DanmakuContext的实例化.
        danmakuContext = DanmakuContext.create();
        danmakuView.prepare(parser, danmakuContext);
        //初始化含有输入框和按钮的线性布局
        final LinearLayout operationLayout = (LinearLayout) findViewById(R.id.operation_layout);
        //初始化发送按钮
        final Button send = (Button) findViewById(R.id.send);
        //输入框输入内容
        final EditText editText = (EditText) findViewById(R.id.edit_text);
        //适时的让输入框显现或隐藏
        danmakuView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                if (operationLayout.getVisibility() == View.GONE) {
                    operationLayout.setVisibility(View.VISIBLE);
                } else {
                    operationLayout.setVisibility(View.GONE);
                }
            }
        });
        //发送按钮的点击事件
        send.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                String content = editText.getText().toString();
                if (!TextUtils.isEmpty(content)) {
                    //因为是自己的内容,所以传一个true过去,方法内部会判断这个变量
                    addDanmaku(content, true);
                    //再把输入框置为空
                    editText.setText("");
                }
            }
        });
        //获取到窗体的顶级父类并设置状态栏的显示隐藏
        getWindow().getDecorView().setOnSystemUiVisibilityChangeListener(new View.OnSystemUiVisibilityChangeListener() {
            @Override
            public void onSystemUiVisibilityChange(int visibility) {
                if (visibility == View.SYSTEM_UI_FLAG_VISIBLE) { //显示状态栏,Activity不全屏显示
                    onWindowFocusChanged(true);
                }
            }
        });
    }

    /**
     * 向弹幕View中添加一条弹幕 
     *
     * @param content    弹幕的具体内容
     * @param withBorder 弹幕是否有边框
     */
    private void addDanmaku(String content, boolean withBorder) {
        //BaseDanmaku 您可以点击进入查看源码实现
        // 弹幕的相关设置:弹幕优先级  颜色  时长  文本  Z轴  Y轴  阴影  描边
        //                下划线  内边距  宽度  高度  存活时间  是否是直播弹幕
        BaseDanmaku danmaku = danmakuContext.mDanmakuFactory.createDanmaku(BaseDanmaku.TYPE_SCROLL_RL);
        danmaku.text = content;                  //文本
        danmaku.padding = 5;                     //内边距
        danmaku.textSize = sp2px(20);           //字体大小
        danmaku.textColor = Color.WHITE;       //文本颜色
        danmaku.setTime(danmakuView.getCurrentTime()); //显示时长 偏移时间
        //如果是true 证明是自己的弹幕,那么就可以更改自己想要的颜色了
        if (withBorder) {
            danmaku.borderColor = Color.GREEN;
        }
        //调用底层代码 把弹幕内容添加到LinkedList<Long> mDrawTimes;
        danmakuView.addDanmaku(danmaku);
    }

    /**
     * 随机生成一些弹幕内容以供测试
     */
    private void generateSomeDanmaku() {
        new Thread(new Runnable() {
            @Override
            public void run() {
                while (showDanmaku) {
                    int time = new Random().nextInt(300);
                    String content = "" + time + time;
                    addDanmaku(content, false);
                    try {
                        Thread.sleep(time);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            }
        }).start();
    }

    /**
     * sp转px的方法。
     */
    public int sp2px(float spValue) {
        final float fontScale = getResources().getDisplayMetrics().scaledDensity;
        return (int) (spValue * fontScale + 0.5f);
    }

    @Override//表示Activity正在停止.
    protected void onPause() {
        super.onPause();
        //如果弹幕控件不为空 && 弹幕控件的线程还存活
        if (danmakuView != null && danmakuView.isPrepared()) {
            //暂停运行弹幕控件
            danmakuView.pause();
        }
    }

    @Override//表示Activity前台并可与用户交互.
    protected void onResume() {
        super.onResume();
        if (danmakuView != null && danmakuView.isPrepared() && danmakuView.isPaused()) {
            danmakuView.resume();
        }
    }

    @Override//表示Activity即将被销毁.
    protected void onDestroy() {
        super.onDestroy();
        //把变量置为false
        showDanmaku = false;
        //如果弹幕控件还存在.调用release(); 底层调用stop(),并把底层的LinkedList<Long> mDrawTimes 置为空;
        if (danmakuView != null) {
            danmakuView.release();
            danmakuView = null;
        }
    }

    @Override//都说这个函数才会使用户可以与应用真正开始进行交互.
    public void onWindowFocusChanged(boolean hasFocus) {
        super.onWindowFocusChanged(hasFocus);
        //看了底层后得知  Build.VERSION.SDK_INT == 20 ;
        if (hasFocus && Build.VERSION.SDK_INT >= 19) {
            View decorView = getWindow().getDecorView();
            decorView.setSystemUiVisibility(
                        //这个标志来帮助你的应用维持一个稳定的布局.
                     View.SYSTEM_UI_FLAG_LAYOUT_STABLE
                        //Activity全屏显示,但状态栏不会被隐藏覆盖,状态栏依然可见,Activity顶端布局部分会被状态遮住。
                   | View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION
                        //ctivity全屏显示,但状态栏不会被隐藏覆盖,状态栏依然可见,Activity顶端布局部分会被状态遮住.
                   | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
                        //ctivity全屏显示,但状态栏不会被隐藏覆盖,状态栏依然可见,Activity顶端布局部分会被状态遮住.
                   | View.SYSTEM_UI_FLAG_HIDE_NAVIGATION
                        //Activity全屏显示,且状态栏被隐藏覆盖掉.
                   | View.SYSTEM_UI_FLAG_FULLSCREEN
                        //安卓4.4 新增.
                   | View.SYSTEM_UI_FLAG_IMMERSIVE_STICKY);
        }
    }
}





如上,在安卓设备上,一个简单的弹幕Demo就可以实现了.


项目直达下载通道为: Demo_Danmaku


如有问题请多指正,您的指正使我更正确的前行.


  • 3
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 8
    评论
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值