span从入门到精通1 第三方工具类GifDrawable

2024-05-29 00:32

本文主要是介绍span从入门到精通1 第三方工具类GifDrawable,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

好久没有写博客了这次介绍下span的使用希望大家少走弯路这次我们从工具类GifDrawable说起
先看基本效果图吧
这里写图片描述

大家看不太懂不要慌我会把代码传到GitHub上后面会附带GitHub链接

首先先看基本的使用然后再逐步分析

GifDrawable gifDrawable = new RefreshGifDrawable(getResources(), R.drawable.a);
GlidePreDrawable glidePreDrawable = new GlideReusePreDrawable();
GlideApp.with(this).load("http://5b0988e595225.cdn.sohucs.com/images/20170919/1ce5d4c52c24432e9304ef942b764d37.gif").placeholder(gifDrawable).into(new DrawableTarget(glidePreDrawable));
CharSequence charSequence = DrawableUtil.getDrawableText("[c]", glidePreDrawable);
spEditText.insertSpecialStr(charSequence, false, charSequence, null);

里面有几个关键的元素GifDrawable ,RefreshGifDrawable,DrawableTarget,spEditText在接下来的过程里我们会对这几个类做着重介绍。

由于全面介绍可能大家,看起来会头疼这里我们针对使用流程进行分析。
首先我们来看new RefreshGifDrawable(getResources(), R.drawable.a)这个东西是什么东东都有哪些处理(前提声明 R.drawable.a是一个gif图片)
代码如下

    public RefreshGifDrawable(@NonNull Resources res,int id) throws Resources.NotFoundException, IOException {super(res, id);}

很简单吧我们再看他这个super做了些什么

public GifDrawable(@NonNull Resources res, @RawRes @DrawableRes int id) throws NotFoundException, IOException {this(res.openRawResourceFd(id));final float densityScale = GifViewUtils.getDensityScale(res, id);mScaledHeight = (int) (mNativeInfoHandle.getHeight() * densityScale);mScaledWidth = (int) (mNativeInfoHandle.getWidth() * densityScale);}

我们先从this(res.openRawResourceFd(id));这里看起res.openRawResourceFd(id)是Android系统处理将我们的gif当成二进制对象处理返回AssetFileDescriptor这个对象我们再看this的处理

public GifDrawable(@NonNull AssetFileDescriptor afd) throws IOException {this(new GifInfoHandle(afd), null, null, true);}

这里new了一个GifInfoHandle我们再看GifInfoHandle都做了些什么好我们继续看他的构造处理

GifInfoHandle(AssetFileDescriptor afd) throws IOException {try {gifInfoPtr = openFd(afd.getFileDescriptor(), afd.getStartOffset());} finally {try {afd.close();} catch (IOException ignored) {//no-op}}}

通过这里我们看到他调用了openFd方法并返回gifInfoPtr 最终关闭afd流释放资源接下来我们需要分析openFd方法(感觉这个可能是核心处理之一)我们接下来继续看

static native long openFd(FileDescriptor fd, long offset) throws GifIOException;

擦这居然是个jni方法好吧做到这里我们只能看其他地方了,我们在看他的返回值

    /*** Pointer to native structure. Access must be synchronized, heap corruption may occur otherwise* when {@link #recycle()} is called during another operation.*/private volatile long gifInfoPtr;

大致意思是返回的是一个指向native的一个底层指针,我们再捋一遍这个流程,将我们的res文件转换成二进制解析对象在经过处理返回一个底层指针。我们再回去继续看。

public GifDrawable(@NonNull AssetFileDescriptor afd) throws IOException {this(new GifInfoHandle(afd), null, null, true);}

我们再继续分析下这个this又调的什么处理(注意这里可能是初始化的关键点)

GifDrawable(GifInfoHandle gifInfoHandle, final GifDrawable oldDrawable, ScheduledThreadPoolExecutor executor, boolean isRenderingTriggeredOnDraw) {mIsRenderingTriggeredOnDraw = isRenderingTriggeredOnDraw;mExecutor = executor != null ? executor : GifRenderingExecutor.getInstance();mNativeInfoHandle = gifInfoHandle;Bitmap oldBitmap = null;if (oldDrawable != null) {synchronized (oldDrawable.mNativeInfoHandle) {if (!oldDrawable.mNativeInfoHandle.isRecycled()&& oldDrawable.mNativeInfoHandle.getHeight() >= mNativeInfoHandle.getHeight()&& oldDrawable.mNativeInfoHandle.getWidth() >= mNativeInfoHandle.getWidth()) {oldDrawable.shutdown();oldBitmap = oldDrawable.mBuffer;oldBitmap.eraseColor(Color.TRANSPARENT);}}}if (oldBitmap == null) {mBuffer = Bitmap.createBitmap(mNativeInfoHandle.getWidth(), mNativeInfoHandle.getHeight(), Bitmap.Config.ARGB_8888);} else {mBuffer = oldBitmap;}mBuffer.setHasAlpha(!gifInfoHandle.isOpaque());mSrcRect = new Rect(0, 0, mNativeInfoHandle.getWidth(), mNativeInfoHandle.getHeight());mInvalidationHandler = new InvalidationHandler(this);mRenderTask.doWork();mScaledWidth = mNativeInfoHandle.getWidth();mScaledHeight = mNativeInfoHandle.getHeight();}

第一行给mIsRenderingTriggeredOnDraw赋值根据前面传过来的值为true即我们再绘制时需要触发渲染(这个目前还没用到)
第二行初始化mExecutor传入值为空为默认渲染线程池
第三行处理mNativeInfoHandle = gifInfoHandle;给底层处理赋值
oldDrawable为空不执行(假如为空里面的处理大致意思就是如果回收了重置bitmap的资源)

if (oldBitmap == null) {mBuffer = Bitmap.createBitmap(mNativeInfoHandle.getWidth(), mNativeInfoHandle.getHeight(), Bitmap.Config.ARGB_8888);} else {mBuffer = oldBitmap;}

这块儿的处理就是根据底层类提供的指针获取图片的宽高并以ARGB_8888的方式加载获取一个bitmap对象。
setHasAlpha(boolean hasAlpha) 这一块的处理是判断是否有alpha如果图片是不透明的画的更快些。
接下来处理初始化mSrcRect确定图片的外切矩形
初始化mInvalidationHandler
执行mRenderTask
初始化mScaledWidth mScaledHeight
最后执行下面代码

final float densityScale = GifViewUtils.getDensityScale(res, id);
mScaledHeight = (int) (mNativeInfoHandle.getHeight() * densityScale);
mScaledWidth = (int) (mNativeInfoHandle.getWidth() * densityScale);

好到了这里我们确认下初始化的内容
1.初始化了常量mIsRenderingTriggeredOnDraw即在绘制时需要触发渲染
2.初始化mExecutor 执行渲染的线程池
3.初始化获取底层处理对象mNativeInfoHandle通过指针获取底层gif的内容
4.初始化Bitmap对象mBuffer
5.初始化mSrcRect确定图像外围
6.初始化mInvalidationHandler
7.执行mRenderTask
8.初始化mScaledWidth, mScaledHeight这里后面还要根据屏幕密度resize宽高
到这里我们解析gif所需要的工具 应该都齐了接下来我们继续往下看

我们再看下面
GifDrawable gifDrawable = new RefreshGifDrawable(getResources().getAssets(), “timg.gif”)
这里的处理给上面差不多这里我就不再过多说明了
做到这里我们在执行下下面的方法
GlideApp.with(this).load(gifDrawable)
// .diskCacheStrategy(DiskCacheStrategy.ALL)
.placeholder(d1).into(new DrawableTarget(glidePreDrawable));

如果这个target是一个ImageView的话我们将会将这个gif图设置到ImageView中去这样的话我们的操作就结束了 ,我会在下一篇文章中对这些内容进行更细致的分析,祝大家生活愉快。

gitHub链接点击这里

这篇关于span从入门到精通1 第三方工具类GifDrawable的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1012133

相关文章

Python办公自动化实战之打造智能邮件发送工具

《Python办公自动化实战之打造智能邮件发送工具》在数字化办公场景中,邮件自动化是提升工作效率的关键技能,本文将演示如何使用Python的smtplib和email库构建一个支持图文混排,多附件,多... 目录前言一、基础配置:搭建邮件发送框架1.1 邮箱服务准备1.2 核心库导入1.3 基础发送函数二、

基于Python实现一个图片拆分工具

《基于Python实现一个图片拆分工具》这篇文章主要为大家详细介绍了如何基于Python实现一个图片拆分工具,可以根据需要的行数和列数进行拆分,感兴趣的小伙伴可以跟随小编一起学习一下... 简单介绍先自己选择输入的图片,默认是输出到项目文件夹中,可以自己选择其他的文件夹,选择需要拆分的行数和列数,可以通过

Python使用pip工具实现包自动更新的多种方法

《Python使用pip工具实现包自动更新的多种方法》本文深入探讨了使用Python的pip工具实现包自动更新的各种方法和技术,我们将从基础概念开始,逐步介绍手动更新方法、自动化脚本编写、结合CI/C... 目录1. 背景介绍1.1 目的和范围1.2 预期读者1.3 文档结构概述1.4 术语表1.4.1 核

Python使用OpenCV实现获取视频时长的小工具

《Python使用OpenCV实现获取视频时长的小工具》在处理视频数据时,获取视频的时长是一项常见且基础的需求,本文将详细介绍如何使用Python和OpenCV获取视频时长,并对每一行代码进行深入解析... 目录一、代码实现二、代码解析1. 导入 OpenCV 库2. 定义获取视频时长的函数3. 打开视频文

从入门到精通MySQL联合查询

《从入门到精通MySQL联合查询》:本文主要介绍从入门到精通MySQL联合查询,本文通过实例代码给大家介绍的非常详细,需要的朋友可以参考下... 目录摘要1. 多表联合查询时mysql内部原理2. 内连接3. 外连接4. 自连接5. 子查询6. 合并查询7. 插入查询结果摘要前面我们学习了数据库设计时要满

Linux中压缩、网络传输与系统监控工具的使用完整指南

《Linux中压缩、网络传输与系统监控工具的使用完整指南》在Linux系统管理中,压缩与传输工具是数据备份和远程协作的桥梁,而系统监控工具则是保障服务器稳定运行的眼睛,下面小编就来和大家详细介绍一下它... 目录引言一、压缩与解压:数据存储与传输的优化核心1. zip/unzip:通用压缩格式的便捷操作2.

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

sqlite3 命令行工具使用指南

《sqlite3命令行工具使用指南》本文系统介绍sqlite3CLI的启动、数据库操作、元数据查询、数据导入导出及输出格式化命令,涵盖文件管理、备份恢复、性能统计等实用功能,并说明命令分类、SQL语... 目录一、启动与退出二、数据库与文件操作三、元数据查询四、数据操作与导入导出五、查询输出格式化六、实用功

解析C++11 static_assert及与Boost库的关联从入门到精通

《解析C++11static_assert及与Boost库的关联从入门到精通》static_assert是C++中强大的编译时验证工具,它能够在编译阶段拦截不符合预期的类型或值,增强代码的健壮性,通... 目录一、背景知识:传统断言方法的局限性1.1 assert宏1.2 #error指令1.3 第三方解决

从入门到精通MySQL 数据库索引(实战案例)

《从入门到精通MySQL数据库索引(实战案例)》索引是数据库的目录,提升查询速度,主要类型包括BTree、Hash、全文、空间索引,需根据场景选择,建议用于高频查询、关联字段、排序等,避免重复率高或... 目录一、索引是什么?能干嘛?核心作用:二、索引的 4 种主要类型(附通俗例子)1. BTree 索引(