教你搞定Android自定义View

Android App开发过程中,很多时候会遇到系统框架中提供的控件无法满足我们产品的设计需求,那么这时候我们可以选择先Google下有没有比较成熟的开源项目可以让我们用,当然现在Github上面的项目非常丰富,能够满足我们绝不多数的开发需求,但是在使用这些炫酷的第三方控件时,我们也要想一想,我们是不是也可以发挥自己的想象力,动手实现自己想要的控件,尽可能掌控实现的细节!

View

Android所有的控件都是View或者View的子类,它其实表示的就是屏幕上的一块矩形区域,用一个Rect来表示,left,top表示View相对于它的parent View的起点,width,height表示View自己的宽高,通过这4个字段就能确定View在屏幕上的位置,确定位置后就可以开始绘制View的内容了。

View绘制过程

View的绘制可以分为下面三个过程:

  • Measure View会先做一次测量,算出自己需要占用多大的面积。View的Measure过程给我们暴露了一个接口onMeasure,方法的定义是这样的,

    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {}

    View类已经提供了一个基本的onMeasure实现,

    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec), getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec)); } public static int getDefaultSize(int size, int measureSpec) { int result = size; int specMode = MeasureSpec.getMode(measureSpec); int specSize = MeasureSpec.getSize(measureSpec); switch (specMode) { case MeasureSpec.UNSPECIFIED: result = size; break; case MeasureSpec.AT_MOST: case MeasureSpec.EXACTLY: result = specSize; break; } return result; }

    其中invoke了setMeasuredDimension()方法,设置了measure过程中View的宽高,getSuggestedMinimumWidth()返回View的最小Width,Height也有对应的方法。插几句,MeasureSpec类是View类的一个内部静态类,它定义了三个常量UNSPECIFIED、AT_MOST、EXACTLY,其实我们可以这样理解它,它们分别对应LayoutParams中match_parent、wrap_content、xxxdp。我们可以重写onMeasure来重新定义View的宽高。

  • Layout Layout过程对于View类非常简单,同样View给我们暴露了onLayout方法

    protected void onLayout(boolean changed, int left, int top, int right, int bottom) { }

    因为我们现在讨论的是View,没有子View需要排列,所以这一步其实我们不需要做额外的工作。插一句,对ViewGroup类,onLayout方法中,我们需要将所有子View的大小宽高设置好,这个我们下一篇会详细说。

  • Draw Draw过程,就是在canvas上画出我们需要的View样式。同样View给我们暴露了onDraw方法

    protected void onDraw(Canvas canvas) { }

    默认View类的onDraw没有一行代码,但是提供给我们了一张空白的画布,举个例子,就像一张画卷一样,我们就是画家,能画出什么样的效果,完全取决我们。

    View中还有三个比较重要的方法

  • requestLayout View重新调用一次layout过程。

  • invalidate View重新调用一次draw过程

  • forceLayout 标识View在下一次重绘,需要重新调用layout过程。

自定义属性

整个View的绘制流程我们已经介绍完了,还有一个很重要的知识,自定义控件属性,我们都知道View已经有一些基本的属性,比如layout_width,layout_height,background等,我们往往需要定义自己的属性,那么具体可以这么做。

  • 1.在values文件夹下,打开attrs.xml,其实这个文件名称可以是任意的,写在这里更规范一点,表示里面放的全是view的属性。
  • 2.因为我们下面的实例会用到2个长度,一个颜色值的属性,所以我们这里先创建3个属性。

    <declare-styleable name="rainbowbar"><attr name="rainbowbar_hspace" format="dimension"></attr> <attr name="rainbowbar_vspace" format="dimension"></attr> <attr name="rainbowbar_color" format="color"></attr> </declare-styleable>

那么到底怎么用呢,我们会看一个实例。

实现一个比较简单的Google彩虹进度条。

为了简单起见,这里我只用一种颜色,多种颜色就留给大家了,我们直接上代码。

蓝色的进度条
蓝色的进度条
public class RainbowBar extends View {//progress bar colorint barColor = Color.parseColor("#1E88E5"); //every bar segment width int hSpace = Utils.dpToPx(80, getResources()); //every bar segment height int vSpace = Utils.dpToPx(4, getResources()); //space among bars int space = Utils.dpToPx(10, getResources()); float startX = 0; float delta = 10f; Paint mPaint; public RainbowBar(Context context) { super(context); } public RainbowBar(Context context, AttributeSet attrs) { this(context, attrs, 0); } public RainbowBar(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); //read custom attrs TypedArray t = context.obtainStyledAttributes(attrs, R.styleable.rainbowbar, 0, 0); hSpace = t.getDimensionPixelSize(R.styleable.rainbowbar_rainbowbar_hspace, hSpace); vSpace = t.getDimensionPixelOffset(R.styleable.rainbowbar_rainbowbar_vspace, vSpace); barColor = t.getColor(R.styleable.rainbowbar_rainbowbar_color, barColor); t.recycle(); // we should always recycle after used mPaint = new Paint(); mPaint.setAntiAlias(true); mPaint.setColor(barColor); mPaint.setStrokeWidth(vSpace); } ....... }

View有了三个构造方法需要我们重写,这里介绍下三个方法会被调用的场景,

  • 第一个方法,一般我们这样使用时会被调用,View view = new View(context);
  • 第二个方法,当我们在xml布局文件中使用View时,会在inflate布局时被调用, <View  layout_width="match_parent"  layout_height="match_parent"/>。
  • 第三个方法,跟第二种类似,但是增加style属性设置,这时inflater布局时会调用第三个构造方法。 <View  style="@styles/MyCustomStyle" layout_width="match_parent"  layout_height="match_parent"/>。

上面大家可能会感觉到有点困惑的是,我把初始化读取自定义属性hspace,vspace,和barcolor的代码写在第三个构造方法里面,但是我RainbowBar在线性布局中没有加style属性(),那按照我们上面的解释,inflate布局时应该会invoke第二个构造方法啊,但是我们在第二个构造方法里面调用了第三个构造方法,this(context, attrs, 0); 所以在第三个构造方法中读取自定义属性,没有问题,这是一点小细节,避免代码冗余-,-

Draw

因为我们这里不用关注measrue和layout过程,直接重写onDraw方法即可。

 //draw be invoke numbers.
int index = 0;
@Override
protected void onDraw(Canvas canvas) { super.onDraw(canvas); //get screen width float sw = this.getMeasuredWidth(); if (startX >= sw + (hSpace + space) - (sw % (hSpace + space))) { startX = 0; } else { startX += delta; } float start = startX; // draw latter parse while (start < sw) { canvas.drawLine(start, 5, start + hSpace, 5, mPaint); start += (hSpace + space); } start = startX - space - hSpace; // draw front parse while (start >= -hSpace) { canvas.drawLine(start, 5, start + hSpace, 5, mPaint); start -= (hSpace + space); } if (index >= 700000) { index = 0; } invalidate(); } //布局文件 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" android:layout_width="match_parent" android:layout_height="match_parent" android:gravity="center" android:layout_marginTop="40dp" android:orientation="vertical" > <com.sw.demo.widget.RainbowBar android:layout_width="match_parent" android:layout_height="wrap_content" app:rainbowbar_color="@android:color/holo_blue_bright" app:rainbowbar_hspace="80dp" app:rainbowbar_vspace="10dp" ></com.sw.demo.widget.RainbowBar> </LinearLayout>

其实就是调用canvas的drawLine方法,然后每次将draw的起点向前推进,在方法的结尾,我们调用了invalidate方法,上面我们已经说明了,这个方法会让View重新调用onDraw方法,所以就达到我们的进度条一直在向前绘制的效果。下面是最后的显示效果,制作成gif时好像有色差,但是真实效果是蓝色的。我们只写了短短的几十行代码,自定义View并不是我们想象中那么难,下一篇我们会继续ViewGroup的绘制流程学习。

rainbow_bar_demo.gif
rainbow_bar_demo.gif
文/ALIOUS(简书作者) 原文链接:http://www.jianshu.com/p/84cee705b0d3 著作权归作者所有,转载请联系作者获得授权,并标注“简书作者”。

转载于:https://www.cnblogs.com/Free-Thinker/p/5573232.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/430467.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

将游戏成绩传到排名页面html,用野狗开发实时游戏排行榜

创建wilddog应用填写应用名称和应用ID就可以创建了。应用ID需要全网唯一创建成功之后就可以在控制面板看到应用了.1.引入SDK2.创建引用ref Wilddog("https://.wilddogio.com/")//将替换成申请的应用IDref Wilddog("https://fullstack-top-demo.wilddogio.com/…

波士顿大学计算机与传媒专业,波士顿大学传媒专业好吗

波士顿大学传媒专业是美国历史上最早的可以授予公共关系学位的大学&#xff0c;它是世界历史上培养得最多的公共关系学者的一所学校。在美国所有新闻传媒专业的排名来讲&#xff0c;波士顿大学传媒专业全美排名也是在前10名的&#xff0c;而在全世界的排名上则是位于前50名的超…

html5 deckview,六本木Hills出現超大型巨人?進擊的巨人展FINAL×頂樓Sky Deck的VR體驗...

戶外複合型度假區「相模湖森林度假遊樂園」將從2019年7月20日(六)&#xff5e;8月30日(五)這段期間展開超人氣動畫《進擊的巨人》與超人氣競技設施「MUSCLE MONSTER」的合作企劃「進擊的企劃『MUSCLE MONSTER』」活動&#xff01;這次的活動是為了紀念7月5日(五)&#xff5e;9月…

surface系列平板电脑属于微型计算机,什么是Surface平板电脑 微软的Surface平板电脑泽怎么样...

什么是Surface平板电脑?微软的Surface平板电脑是什么?Surface平板电脑分两个版本一个是运行Windows RT系统&#xff0c;搭载ARM处理器(类似于大多手机和平板电脑)&#xff0c;9.3毫米(比iPad薄一点)&#xff0c;676g(比iPad重一点)&#xff0c;配有10.6英寸ClearType高清显示…

用户注意到用户计算机中千兆位网卡,为何你电脑上的千兆网卡跑不到千兆?

这几天谈了不少网络方面的知识和教程&#xff0c;基本上已经脱离百兆的温饱线&#xff0c;走向了高大上的千兆小康生活……事实上&#xff0c;对于目前的硬件环境而言&#xff0c;无论是路由器、网卡、光纤、网线&#xff0c;仅从带宽而言&#xff0c;达到千兆毫无难度&#xf…

安卓学习日记:初识Android Studio · java环境配置和AS安装

工欲善其事&#xff0c;必先利其器。要进行安卓开发&#xff0c;必须要有一款上手的开发利器。查阅了相关资料后&#xff0c;了解到&#xff0c;现在主流的安卓开发工具是&#xff1a; 1. Eclipse Android SDK 2. Android Studio 因为安卓的开发语言是 java&#xff0c;所以…

sql 递归查询

1、既然要谈到sql&#xff0c;数据库表是必须的 2、数据结构 3、获取某个节点的所有子节点 传统的写法&#xff08;sql2000&#xff09; 很麻烦&#xff0c;暂且就不写了 来看看CTE的写法 CREATE PROC sp_getTreeById(TreeId int) AS BEGIN WITH cteTree AS (SELECT * FROM Tuz…

计算机画画教程,【推荐】初学者电脑画画教程

电脑绘画教程&#xff0c;今天跟大家分享一下&#xff0c;如何学电脑绘画以及关于零基础学板绘是不是一定要美术基础的问题~~这也是很多学习电脑绘画小伙伴的疑虑。如果你想快速提升自己的绘画水平&#xff0c;并且绝对的不怕辛苦&#xff0c;欢迎挑战轻微课魔鬼特训班&#xf…

python 中调用shell命令

subprocess模块 根据Python官方文档说明&#xff0c;subprocess模块用于取代上面这些模块。有一个用Python实现的并行ssh工具—mssh&#xff0c;代码很简短&#xff0c;不过很有意思&#xff0c;它在线程中调用subprocess启动子进程来干活。 [python] view plaincopy >>&…

一台计算机数据丢失与恢复,如何在不丢失数据的情况下将iPhone与多台计算机同步-万兴数据恢复-万兴恢复专家...

第2部分&#xff1a;将iPhone与多台计算机与iTunes同步如果用户对iPhone有很强的控制欲&#xff0c;并且不想尝试使用任何新软件来同步需求&#xff0c;那么iTunes也可用于将iPhone与多台计算机同步。虽然在第一时间&#xff0c;这可能听起来和iTunes的职能相悖&#xff0c;但实…

iOS OC语言: Block底层实现原理

来源http://www.wtoutiao.com/p/11dgbk4.html 先来简单介绍一下Block Block是什么&#xff1f; 苹果推荐的类型&#xff0c;效率高&#xff0c;在运行中保存代码。用来封装和保存代码&#xff0c;有点像函数&#xff0c;Block可以在任何时候执行。 Block和函数的相似性&#xf…

计算机如何取消还原卡,如何关闭硬件还原卡?

2009-11-06有什么方法可以让电脑每次关机都可以自动删1)打开控制面板/性能维护/管理工具/双击本地安全策略&#xff0c;在右侧选“关机清理虚拟内存页面文件”双击他选“已启用”按应用重启即可。2)建议下载超级兔子是免费的小巧著名的软件&#xff0c;选择清理系统垃圾(全选)&…

服务器虚拟主机推荐,免费的虚拟主机推荐

写在前面&#xff1a;免费虚拟主机免费虚拟主机是指IDC服务商“免费”为站长提供网页寄存服务。免费虚拟主机--用于制作免费个人网站&#xff0c;是学习网页设计的好方法。虚拟主机就是指网站空间&#xff0c;是在网络服务器上划分出一定的磁盘空间供用户放置站点、应用组件等&…

js/css文件修改后浏览器本地缓存解决

本文实例讲述了让html页面随js的修改来更新缓存的实现方法。分享给大家供大家参考。具体实现方法如下&#xff1a; 很多朋友都会碰到这样的情况&#xff1a;如果我们页面加载了js的话下次打开时也会是调用这个js缓存文件&#xff0c;但对于我们修改后调试和发布是非常的不方便了…

js中报错 ajax不存在,AJAX

AJAX : Asynchronous JavaScript and XML 异步JavaScript和XML (XML更多的是被JSON格式替代使用)AJAX 只做一件事情&#xff1a;异步获取数据&#xff0c;更为重要的是还是JS对返回的数据进行操作。异步获取数据极大地改善web与用户的数据交互(如下图左侧为传统web,右侧为借…

iOS开发内购图文教程

2015年最全的内购图文教程&#xff0c;首先是填各种资料&#xff0c;最后是代码&#xff0c;废话不多说&#xff0c;直接上图 第一部分协议 第一步.png第二步.jpg第三步.jpg第四步.png第五步.png第六步.png第七步.jpg第八步.jpg第九步.jpg第十步.pngCNAPS CODE 查询地址https:/…

cisco服务器维修,面向终端的AMP控制台的思科维护的排除列表更改

简介本文档介绍添加到思科维护的例外项的更改。思科维护的例外项由思科创建和维护&#xff0c;以便在面向终端的高级恶意软件防护(AMP)连接器和防病毒、安全或其他软件之间提供更好的兼容性&#xff0c;这些例外项可以添加到应用的新版本。作者&#xff1a;思科工程师Caly Hess…

服务器怎么用光驱装系统教程,使用光驱重装系统详细教程

电脑系统在用了很长一段时间后总会出现卡顿或者其他的问题&#xff0c;这时很多小伙伴后就会选择进行系统的重装&#xff0c;系统的重装也是有多种方法的&#xff0c;比如使用U盘进行重装&#xff0c;或者使用光驱&#xff0c;而今天小编要给大家分享的就是使用光驱重装系统的详…

struts2 Eclipse 中集成strust2开发框架实例

下面通过建立一个小的实例具体来说明Eclipse 集成struts2,以下实例采用的为 struts2 版本为 struts2 2.2.3.1 为应用. 1. 下载struts2的开发包 第一步: 在浏览器中输入 http://apache.org 第二步:在apche的页面项目中选择struct 点击连接进入相关页面 第三步: 点击download选择…

电脑电池,笔记本电池校正,教您怎样校正笔记本电脑电池

只要留个心&#xff0c;就会发现身边的很多人都在使用着笔记本电脑&#xff0c;不过笔记本电脑使用久了&#xff0c;电池可能会出现虚电的情况&#xff0c;导致我们的笔记本的续航时间变短了&#xff0c;这时候需要我们手动对笔记本电池进行校准&#xff0c;为此&#xff0c;小…