[Android]一行代码创建Gif动态图

作者: lchad | 来源:发表于2017-03-26 19:56 被阅读519次

    开篇之前先看一下效果.以下示例程序Apk可以在本文底部找到下载地址.

    一脸懵逼.gif 是如何产生的?

    GIF.gif

    话说,那还是 2016 年的时候,由于自己比较喜欢用表情包斗图,所以为了方便,就开发了一个创建表情图片的 APP ,功能简单但是实用,我的很多朋友也都下载使用了,很多人跟我反馈说,如果可以做成动态图就好了.之后我 Google 了一番,发现并没有什么好的方案,于是这个功能就太监了.想说的就是拖了这么久,这个周末花了两天的时间把这个功能实现了.

    由于 Android 平台对Gif的支持很很好,没有现成的Java Api可以用,所以我借助NDK作为桥梁,通过C++语言实现对32-bit ARGB图片进行 Color Quantization,转化成Gif动态图(256色域).这个项目的色彩转换算法是基于 Gifflen @Bitmap color reduction and GIF encoding 的.

    通过对 Gifflen 的C++源码进行一些定制和修改,然后编译出native library(即 .so 文件),然后打包到 APK 中.我们可以很容易的在Android系统上创建一个Gif动图. 本项目的NDK部分代码是采用CMake的方式进行构建的.Android Studio2.2版本之后已经对NDK编程有了很好的支持.体验下来感觉很棒.


    使用方法

    1.添加动态链接库

    使用我编译好的动态链接库文件, 支持 arm-v8a, armaib, armabi-v7a, mips, mips64, x86, x86-64 等7个平台,你可以根据自己的需要添加对应的native library到项目中.

    2.添加Gifflen工具类

    应用层的所有接口都在 Gifflen.java 这个类中,你要做的就是在自己项目的app/main/java路径下创建com/lchad/gifflen文件夹(这个路径很重要一个字母都不能错),然后把Gifflen.java复制到这个特殊的路径下,注意,请一定要把Gifflen.java放到以上指定位置(即your-project-name/app/main/java/com/lchad/gifflen)下,否则会在运行时JNI会报错,找不到对应的jni方法.这一点需格外注意.

    3.配置读写存储的权限
    由于涉及到文件操作,需要事先获取Android的读写外部存储权限,在
    AndroidManifest.xml 中添加一下两行:

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

    如果是Android6.0及以上需注意动态申请权限.

     ActivityCompat.requestPermissions(MainActivity.this, new String[]{
                                Manifest.permission.WRITE_EXTERNAL_STORAGE,
                                Manifest.permission.READ_EXTERNAL_STORAGE},
                        REQUEST_PERMISSIONS);
    

    **4.初始化工具类 Gifflen **
    初始化采用Builder模式,使用起来十分清爽.

    Gifflen mGiffle = new Gifflen.Builder()
                            .color(mColor)   //色域范围是2~256,且必须是2的整数次幂.
                            .delay(mDelayTime) //每相邻两帧之间播放的时间间隔.
                            .quality(mQuality) //色彩量化时的quality值.
                            .width(500)     //生成Gif文件的宽度(像素).
                            .height(500)       //生成Gif文件的高度(像素).
                            .build();
    

    5.开始创建 Gif 图片

    • 从 File 列表创建
            List<File> files = getFileList();
            mGiffle.encode(320, 320, "target path", files);
            mGiffle.encode("target path", files); 
    
    • 从 Uri 列表创建
            List<Uri> uris = getUriList();
            mGiffle.encode(context, 500, 500, uris);
            mGiffle.encode(context, "target path", uris);
    
    • 从 TypeArray 创建
         TypeArra mDrawableList = getResources().obtainTypedArray(R.array.source);
         mGiffle.encode(MainActivity.this, "target path", 500, 500, mDrawableList);
         mGiffle.encode(MainActivity.this, "target path", mDrawableList);
    
    • 从 Bitmap 数组创建
            Bitmap[] bitmaps = getBitmaps();
            mGiffle.encode("target path", 500, 500, bitmaps);
            mGiffle.encode("target path", bitmaps);
        //注意:此时要考虑Bitmap的大小以及个数,否则可能会造成OOM.
    
    • 从 drawable id 数组创建
        int[] drawableIds = new int[]{
                    R.drawable.mengbi1,
                    R.drawable.mengbi2,
                    R.drawable.mengbi3};
            mGiffle.encode(context, "target path", 500, 500, drawableIds);
            mGiffle.encode(context, "target path", drawableIds);
    

    以上五种创建方式都支持重载(本质上都是对Bitmap进行操作),宽度和高度在encode()的时候都可以缺省,这时会使用创建Gifflen时传入的值,如果创建Gifflen时仍然没有传值,则会使用一个默认的值 320.

    示例Apk程序

    点我下载

    源码(GitHub) 欢迎Star,提Issue, PR

    Gifflen-Android

    相关文章

      网友评论

      • f385b313ecbf:这么好的东西,不得不赞👍
      • 静心风飞:你好,透明素材生成的GIF是黑色的。是不支持透明的GIf生成吗?
      • 逸云天:感谢楼主的分享。楼主可否分享一下编译好动态链接库文件的源文件:smile:
        lchad:有源码的
      • 我在等你回复可你没回:请问是用什么录制屏幕的?录制了好久
        lchad:用的是GifCam

      本文标题:[Android]一行代码创建Gif动态图

      本文链接:https://www.haomeiwen.com/subject/kkhkottx.html