span从入门到精通2 自定义drawable

2024-05-29 00:32

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

分析昨天的博客 span从入门到精通 第三方工具类GifDrawable 发现有个知识点有必要先梳理下要不大家可能看着博客都是懵逼的,这个知识点就是自定义drawable。
看看效果吧
drawable画圆

图片
首先我们先分析下源码里面drawable 是怎么被调用的我们先看看view类 的setBackgroundDrawable 这个方法

public void setBackgroundDrawable(Drawable background) {computeOpaqueFlags();if (background == mBackground) {return;}boolean requestLayout = false;mBackgroundResource = 0;/** Regardless of whether we're setting a new background or not, we want* to clear the previous drawable. setVisible first while we still have the callback set.*/if (mBackground != null) {if (isAttachedToWindow()) {mBackground.setVisible(false, false);}mBackground.setCallback(null);unscheduleDrawable(mBackground);}if (background != null) {Rect padding = sThreadLocal.get();if (padding == null) {padding = new Rect();sThreadLocal.set(padding);}resetResolvedDrawablesInternal();background.setLayoutDirection(getLayoutDirection());if (background.getPadding(padding)) {resetResolvedPaddingInternal();switch (background.getLayoutDirection()) {case LAYOUT_DIRECTION_RTL:mUserPaddingLeftInitial = padding.right;mUserPaddingRightInitial = padding.left;internalSetPadding(padding.right, padding.top, padding.left, padding.bottom);break;case LAYOUT_DIRECTION_LTR:default:mUserPaddingLeftInitial = padding.left;mUserPaddingRightInitial = padding.right;internalSetPadding(padding.left, padding.top, padding.right, padding.bottom);}mLeftPaddingDefined = false;mRightPaddingDefined = false;}// Compare the minimum sizes of the old Drawable and the new.  If there isn't an old or// if it has a different minimum size, we should layout againif (mBackground == null|| mBackground.getMinimumHeight() != background.getMinimumHeight()|| mBackground.getMinimumWidth() != background.getMinimumWidth()) {requestLayout = true;}// Set mBackground before we set this as the callback and start making other// background drawable state change calls. In particular, the setVisible call below// can result in drawables attempting to start animations or otherwise invalidate,// which requires the view set as the callback (us) to recognize the drawable as// belonging to it as per verifyDrawable.mBackground = background;if (background.isStateful()) {background.setState(getDrawableState());}if (isAttachedToWindow()) {background.setVisible(getWindowVisibility() == VISIBLE && isShown(), false);}applyBackgroundTint();// Set callback last, since the view may still be initializing.background.setCallback(this);if ((mPrivateFlags & PFLAG_SKIP_DRAW) != 0) {mPrivateFlags &= ~PFLAG_SKIP_DRAW;requestLayout = true;}} else {/* Remove the background */mBackground = null;if ((mViewFlags & WILL_NOT_DRAW) != 0&& (mDefaultFocusHighlight == null)&& (mForegroundInfo == null || mForegroundInfo.mDrawable == null)) {mPrivateFlags |= PFLAG_SKIP_DRAW;}/** When the background is set, we try to apply its padding to this* View. When the background is removed, we don't touch this View's* padding. This is noted in the Javadocs. Hence, we don't need to* requestLayout(), the invalidate() below is sufficient.*/// The old background's minimum size could have affected this// View's layout, so let's requestLayoutrequestLayout = true;}computeOpaqueFlags();if (requestLayout) {requestLayout();}mBackgroundSizeChanged = true;invalidate(true);invalidateOutline();}

这里面的意思大致分析一下就是首先判断设置的drawable是否跟以前的一样如果一样则不需要重新设置,如果旧的background不为空则清理下旧的background的数据,之后设置background的state,visible,callback等属性,最终调用requestLayout这个方法重新绘制布局也就是会走layout跟draw等方法,系统会在draw方法里面调用drawBackground这个处理我们来看drawBackground这个方法里面做了些什么处理

  private void drawBackground(Canvas canvas) {final Drawable background = mBackground;if (background == null) {return;}setBackgroundBounds();// Attempt to use a display list if requested.if (canvas.isHardwareAccelerated() && mAttachInfo != null&& mAttachInfo.mThreadedRenderer != null) {mBackgroundRenderNode = getDrawableRenderNode(background, mBackgroundRenderNode);final RenderNode renderNode = mBackgroundRenderNode;if (renderNode != null && renderNode.isValid()) {setBackgroundRenderNodeProperties(renderNode);((DisplayListCanvas) canvas).drawRenderNode(renderNode);return;}}final int scrollX = mScrollX;final int scrollY = mScrollY;if ((scrollX | scrollY) == 0) {background.draw(canvas);} else {canvas.translate(scrollX, scrollY);background.draw(canvas);canvas.translate(-scrollX, -scrollY);}}

这里我们只关注一句话background.draw(canvas);即当我们在绘制内容之前会调用drawable的draw方法绘制背景,因此drawable里面的draw方法是一个十分关键的代码它会将我们需要的图片文字等内容绘制到界面中。好了源码先分析到这里我们还是先看看基本使用吧。
好了废话先不多说了先看实现代码吧我们首先定义一个drawable类。(一个极其简单的类)

public class TestDrawable extends Drawable {private Paint paint;//画笔private int mWidth = 300;//图片宽与高的最小值public TestDrawable() {paint = new Paint();paint.setAntiAlias(true);paint.setColor(Color.YELLOW);paint.setStyle(Paint.Style.FILL);}public void setmWidth(int width){this.mWidth=width;invalidateSelf();//更新设置}@Overridepublic void draw(@NonNull Canvas canvas) {canvas.drawCircle(mWidth / 2, mWidth / 2, mWidth / 2, paint);}@Overridepublic void setAlpha(int i) {paint.setAlpha(i);invalidateSelf();//更新设置}@Overridepublic int getIntrinsicHeight() {return mWidth;}@Overridepublic int getIntrinsicWidth() {return mWidth;}@Overridepublic void setColorFilter(@Nullable ColorFilter colorFilter) {paint.setColorFilter(colorFilter);invalidateSelf();//更行设置}@Overridepublic int getOpacity() {return PixelFormat.TRANSLUCENT;}}

看到了吧是不是有一种恍然大明白的感觉
我们的设置极其简单许多都是固定模式化的东西
1.初始化paint这个不用我多说了吧不知道的自觉面壁去
2.setAlpha中设置paint的alpha
3.getIntrinsicHeight getIntrinsicWidth当布局为wrap_content的时候我们要画的内容的宽高
4.setColorFilter调用paint的setColorFilter处理
5.getOpacity一班返回PixelFormat.TRANSLUCENT即可
6.最最核心的处理draw方法这里我们画了一个圆大家也可以画其他图形,图片等处理
7.额外添加一个方法setmWidth重新定义控件的宽度这里要注意了invalidateSelf这个方法调用会重新绘制该布局
这下明白了吧下面就是外部的调用

TestDrawable drawable = new TestDrawable();
ImageView iv_main = findViewById(R.id.iv_main);
iv_main.setImageDrawable(drawable);
drawable.setmWidth(50);

一句iv_main.setImageDrawable将我们画的drawable中的圆画到了界面上是不是很神奇
当然大家如果需要做动画的话可以调用invalidateSelf不断改变我们画的内容

如果大家还是不太明白可以看看画图片是怎么处理的代码如下

public class TestDrawable extends Drawable {private Paint paint;//画笔private int mWidth;//图片宽与高的最小值private Bitmap bitmap;//位图public TestDrawable(Context context, int resID) {this(BitmapFactory.decodeResource(context.getResources(), resID));}public TestDrawable(Bitmap bitmap) {this.bitmap = bitmap;paint = new Paint();paint.setAntiAlias(true);//抗锯齿paint.setDither(true);//抖动,不同屏幕尺的使用保证图片质量///位图渲染器BitmapShader bitmapShader = new BitmapShader(bitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP);paint.setShader(bitmapShader);mWidth = Math.min(bitmap.getWidth(), bitmap.getHeight());}@Overridepublic void draw(@NonNull Canvas canvas) {canvas.drawCircle(mWidth / 2, mWidth / 2, mWidth / 2, paint);}@Overridepublic void setAlpha(int i) {paint.setAlpha(i);invalidateSelf();//更新设置}@Overridepublic int getIntrinsicHeight() {return bitmap.getHeight();}@Overridepublic int getIntrinsicWidth() {return bitmap.getWidth();}@Overridepublic void setColorFilter(@Nullable ColorFilter colorFilter) {paint.setColorFilter(colorFilter);invalidateSelf();//更行设置}@Overridepublic int getOpacity() {return PixelFormat.TRANSLUCENT;}}

代码也不多希望大家看完博客后能对drawable有个更深入的认识

github链接点击这里

这篇关于span从入门到精通2 自定义drawable的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

SpringBoot 自定义消息转换器使用详解

《SpringBoot自定义消息转换器使用详解》本文详细介绍了SpringBoot消息转换器的知识,并通过案例操作演示了如何进行自定义消息转换器的定制开发和使用,感兴趣的朋友一起看看吧... 目录一、前言二、SpringBoot 内容协商介绍2.1 什么是内容协商2.2 内容协商机制深入理解2.2.1 内容

python解析HTML并提取span标签中的文本

《python解析HTML并提取span标签中的文本》在网页开发和数据抓取过程中,我们经常需要从HTML页面中提取信息,尤其是span元素中的文本,span标签是一个行内元素,通常用于包装一小段文本或... 目录一、安装相关依赖二、html 页面结构三、使用 BeautifulSoup javascript

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

数论入门整理(updating)

一、gcd lcm 基础中的基础,一般用来处理计算第一步什么的,分数化简之类。 LL gcd(LL a, LL b) { return b ? gcd(b, a % b) : a; } <pre name="code" class="cpp">LL lcm(LL a, LL b){LL c = gcd(a, b);return a / c * b;} 例题:

Java 创建图形用户界面(GUI)入门指南(Swing库 JFrame 类)概述

概述 基本概念 Java Swing 的架构 Java Swing 是一个为 Java 设计的 GUI 工具包,是 JAVA 基础类的一部分,基于 Java AWT 构建,提供了一系列轻量级、可定制的图形用户界面(GUI)组件。 与 AWT 相比,Swing 提供了许多比 AWT 更好的屏幕显示元素,更加灵活和可定制,具有更好的跨平台性能。 组件和容器 Java Swing 提供了许多

【IPV6从入门到起飞】5-1 IPV6+Home Assistant(搭建基本环境)

【IPV6从入门到起飞】5-1 IPV6+Home Assistant #搭建基本环境 1 背景2 docker下载 hass3 创建容器4 浏览器访问 hass5 手机APP远程访问hass6 更多玩法 1 背景 既然电脑可以IPV6入站,手机流量可以访问IPV6网络的服务,为什么不在电脑搭建Home Assistant(hass),来控制你的设备呢?@智能家居 @万物互联

poj 2104 and hdu 2665 划分树模板入门题

题意: 给一个数组n(1e5)个数,给一个范围(fr, to, k),求这个范围中第k大的数。 解析: 划分树入门。 bing神的模板。 坑爹的地方是把-l 看成了-1........ 一直re。 代码: poj 2104: #include <iostream>#include <cstdio>#include <cstdlib>#include <al

MySQL-CRUD入门1

文章目录 认识配置文件client节点mysql节点mysqld节点 数据的添加(Create)添加一行数据添加多行数据两种添加数据的效率对比 数据的查询(Retrieve)全列查询指定列查询查询中带有表达式关于字面量关于as重命名 临时表引入distinct去重order by 排序关于NULL 认识配置文件 在我们的MySQL服务安装好了之后, 会有一个配置文件, 也就

自定义类型:结构体(续)

目录 一. 结构体的内存对齐 1.1 为什么存在内存对齐? 1.2 修改默认对齐数 二. 结构体传参 三. 结构体实现位段 一. 结构体的内存对齐 在前面的文章里我们已经讲过一部分的内存对齐的知识,并举出了两个例子,我们再举出两个例子继续说明: struct S3{double a;int b;char c;};int mian(){printf("%zd\n",s