Button源码解析(API 26)
概述
说到Button,实在是再熟悉不过了,那就追追它的源码吧。
啥啥啥?代码这么简单,4个构造函数和2个重写函数就完啦?Button的秘密究竟在哪里,别着急,且听细细道来。
预热
总结一下Button的特性吧:
- 可以设置文字,所以继承TextView
- 有Focused,Pressed,Normal等状态,对应不同状态呈现不同UI,这个View里面都有,所以间接继承View
- 可以监听点击事件,这个也是View里的,所以间接继承View
这样看,实现一个Button好像很简单嘛(的确很简单),继承个TextView,然后对应不同状态设置不同的显示UI就好了,再简单点就用个selector,so esay。快去源码看看是不是这么回事吧,what?比我说的还简单,核心代码就一句话。然而真的这么简单吗?简单的东西用起来很爽,但是把问题简单化却很难,这背后的智慧才是我想要的,那么就来看看吧。
源码跟踪
以下代码取自Android API-26。这里仅列出主要代码(其实重要的就那么一句话):
public Button(Context context) {
this(context, null);
}
public Button(Context context, AttributeSet attrs) {
this(context, attrs, com.android.internal.R.attr.buttonStyle);
}
public Button(Context context, AttributeSet attrs, int defStyleAttr) {
this(context, attrs, defStyleAttr, 0);
}
public Button(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
super(context, attrs, defStyleAttr, defStyleRes);
}
前3个构造函数都会去调用第4个构造函数,自定过View的小伙伴对这些构造函数肯定不会陌生,还不太懂的就再听我唠叨一下吧。
我们自定义一个View的同时也会自定义它的一些属性,属性的定义一般是放在<resource>
下的<declare-styleable>
标签中的,但属性值又从哪里获取呢?一般有这么几种途径:
- 布局文件中定义,比如在
<Button>
中定义android:text=”hahaha”,这些属性值会在构造Button时传递给attrs参数(构造函数中的第二个参数) - 主题中定义,我们可以在AndroidManifest中指定主题,在构造Button时传递给defStyleAttr参数(构造函数中的第三个参数)
- 样式资源中定义,一般在
<resource>
标签下的<style>
中定义样式,在构造Button