Android 深入理解Android中的自定义属性
版权声明:本文为博主原创文章,未经博主允许不得转载。
转载请标明出处:
http://blog.csdn.net/lmj623565791/article/details/45022631;
本文出自:【张鸿洋的博客】
1、引言
对于自定义属性,大家肯定都不陌生,遵循以下几步,就可以实现:
- 自定义一个CustomView(extends View )类
- 编写values/attrs.xml,在其中编写styleable和item等标签元素
- 在布局文件中CustomView使用自定义的属性(注意namespace)
- 在CustomView的构造方法中通过TypedArray获取
ps:如果你对上述几个步骤不熟悉,建议先熟悉下,再继续~
那么,我有几个问题:
- 以上步骤是如何奏效的?
- styleable 的含义是什么?可以不写嘛?我自定义属性,我声明属性就好了,为什么一定要写个styleable呢?
- 如果系统中已经有了语义比较明确的属性,我可以直接使用嘛?
- 构造方法中的有个参数叫做
AttributeSet
(eg: MyTextView(Context context, AttributeSet attrs) )这个参数看名字就知道包含的是参数的数组,那么我能不能通过它去获取我的自定义属性呢? - TypedArray是什么鬼?从哪冒出来的,就要我去使用?
恩,针对这几个问题,大家可以考虑下,如何回答呢?还是说:老子会背上述4个步骤就够了~~
2、常见的例子
接下来通过例子来回答上述问题,问题的回答顺序不定~~大家先看一个常见的例子,即上述几个步骤的代码化。
- 自定义属性的声明文件
<code class="language-xml hljs has-numbering"> <span class="hljs-pi"><?xml version="1.0" encoding="utf-8"?></span> <span class="hljs-tag"><<span class="hljs-title">resources</span>></span> <span class="hljs-tag"><<span class="hljs-title">declare-styleable</span> <span class="hljs-attribute">name</span>=<span class="hljs-value">"test"</span>></span> <span class="hljs-tag"><<span class="hljs-title">attr</span> <span class="hljs-attribute">name</span>=<span class="hljs-value">"text"</span> <span class="hljs-attribute">format</span>=<span class="hljs-value">"string"</span> /></span> <span class="hljs-tag"><<span class="hljs-title">attr</span> <span class="hljs-attribute">name</span>=<span class="hljs-value">"testAttr"</span> <span class="hljs-attribute">format</span>=<span class="hljs-value">"integer"</span> /></span> <span class="hljs-tag"></<span class="hljs-title">declare-styleable</span>></span> <span class="hljs-tag"></<span class="hljs-title">resources</span>></span></code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li></ul>
- 自定义View类
<code class="language-java hljs has-numbering"><span class="hljs-keyword">package</span> com.example.test; <span class="hljs-keyword">import</span> android.content.Context; <span class="hljs-keyword">import</span> android.content.res.TypedArray; <span class="hljs-keyword">import</span> android.util.AttributeSet; <span class="hljs-keyword">import</span> android.util.Log; <span class="hljs-keyword">import</span> android.view.View; <span class="hljs-keyword">public</span> <span class="hljs-class"><span class="hljs-keyword">class</span> <span class="hljs-title">MyTextView</span> <span class="hljs-keyword">extends</span> <span class="hljs-title">View</span> {</span> <span class="hljs-keyword">private</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">final</span> String TAG = MyTextView.class.getSimpleName(); <span class="hljs-keyword">public</span> <span class="hljs-title">MyTextView</span>(Context context, AttributeSet attrs) { <span class="hljs-keyword">super</span>(context, attrs); TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.test); String text = ta.getString(R.styleable.test_testAttr); <span class="hljs-keyword">int</span> textAttr = ta.getInteger(R.styleable.test_text, -<span class="hljs-number">1</span>); Log.e(TAG, <span class="hljs-string">"text = "</span> + text + <span class="hljs-string">" , textAttr = "</span> + textAttr); ta.recycle(); } }</code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li></ul>
- 布局文件中使用
<code class="language-xml hljs has-numbering"><span class="hljs-tag"><<span class="hljs-title">RelativeLayout</span> <span class="hljs-attribute">xmlns:android</span>=<span class="hljs-value">"http://schemas.android.com/apk/res/android"</span> <span class="hljs-attribute">xmlns:tools</span>=<span class="hljs-value">"http://schemas.android.com/tools"</span> <span class="hljs-attribute">xmlns:zhy</span>=<span class="hljs-value">"http://schemas.android.com/apk/res/com.example.test"</span> <span class="hljs-attribute">android:layout_width</span>=<span class="hljs-value">"match_parent"</span> <span class="hljs-attribute">android:layout_height</span>=<span class="hljs-value">"match_parent"</span> ></span> <span class="hljs-tag"><<span class="hljs-title">com.example.test.MyTextView </span> <span class="hljs-attribute">android:layout_width</span>=<span class="hljs-value">"100dp"</span> <span class="hljs-attribute">android:layout_height</span>=<span class="hljs-value">"200dp"</span> <span class="hljs-attribute">zhy:testAttr</span>=<span class="hljs-value">"520"</span> <span class="hljs-attribute">zhy:text</span>=<span class="hljs-value">"helloworld"</span> /></span> <span class="hljs-tag"></<span class="hljs-title">RelativeLayout</span>></span></code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li></ul>
ok,大家花3s扫一下,运行结果为:
<code class="language-java hljs has-numbering"> MyTextView: text = helloworld , textAttr = <span class="hljs-number">520</span></code><ul style="display: block;" class="pre-numbering"><li>1</li></ul>
应该都不意外吧,注意下,我的styleable的name写的是test,所以说这里并不要求一定是自定义View的名字。
3、AttributeSet与TypedArray
下面考虑:
构造方法中的有个参数叫做
AttributeSet
(eg: MyTextView(Context context, AttributeSet attrs) )这个参数看名字就知道包含的是参数的集合,那么我能不能通过它去获取我的自定义属性呢?
首先AttributeSet
中的确保存的是该View声明的所有的属性,并且外面的确可以通过它去获取(自定义的)属性,怎么做呢?
其实看下AttributeSet
的方法就明白了,下面看代码。
<code class="language-java hljs has-numbering"><span class="hljs-keyword">public</span> <span class="hljs-title">MyTextView</span>(Context context, AttributeSet attrs) { <span class="hljs-keyword">super</span>(context, attrs); <span class="hljs-keyword">int</span> count = attrs.getAttributeCount(); <span class="hljs-keyword">for</span> (<span class="hljs-keyword">int</span> i = <span class="hljs-number">0</span>; i < count; i++) { String attrName = attrs.getAttributeName(i); String attrVal = attrs.getAttributeValue(i); Log.e(TAG, <span class="hljs-string">"attrName = "</span> + attrName + <span class="hljs-string">" , attrVal = "</span> + attrVal); } <span class="hljs-comment">// ==>use typedarray ...</span> } </code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li></ul>
输出:
<code class="language-java hljs has-numbering">MyTextView(<span class="hljs-number">4136</span>): attrName = layout_width , attrVal = <span class="hljs-number">100.0</span>dip MyTextView(<span class="hljs-number">4136</span>): attrName = layout_height , attrVal = <span class="hljs-number">200.0</span>dip MyTextView(<span class="hljs-number">4136</span>): attrName = text , attrVal = helloworld MyTextView(<span class="hljs-number">4136</span>): attrName = testAttr , attrVal = <span class="hljs-number">520</span> </code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li></ul>
结合上面的布局文件,你发现了什么?
我擦,果然很神奇,真的获得所有的属性,恩,没错,通过AttributeSet
可以获得布局文件中定义的所有属性的key和value(还有一些方法,自己去尝试),那么是不是说TypedArray这个鬼可以抛弃了呢?答案是:NO!
。
现在关注下一个问题:
TypedArray是什么鬼?从哪冒出来的,就要我去使用?
我们简单修改下,布局文件中的MyTextView的属性。
<code class="language-xml hljs has-numbering"><span class="hljs-tag"><<span class="hljs-title">com.example.test.MyTextView </span> <span class="hljs-attribute">android:layout_width</span>=<span class="hljs-value">"@dimen/dp100"</span> <span class="hljs-attribute">android:layout_height</span>=<span class="hljs-value">"@dimen/dp200"</span> <span class="hljs-attribute">zhy:testAttr</span>=<span class="hljs-value">"520"</span> <span class="hljs-attribute">zhy:text</span>=<span class="hljs-value">"@string/hello_world"</span> /></span></code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li></ul>
现在再次运行的结果是:
<code class="language-java hljs has-numbering">MyTextView(<span class="hljs-number">4692</span>): attrName = layout_width , attrVal = @<span class="hljs-number">2131165234</span> MyTextView(<span class="hljs-number">4692</span>): attrName = layout_height , attrVal = @<span class="hljs-number">2131165235</span> MyTextView(<span class="hljs-number">4692</span>): attrName = text , attrVal = @<span class="hljs-number">2131361809</span> MyTextView(<span class="hljs-number">4692</span>): attrName = testAttr , attrVal = <span class="hljs-number">520</span> >>use typedarray MyTextView(<span class="hljs-number">4692</span>): text = Hello world! , textAttr = <span class="hljs-number">520</span></code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li></ul>
发现了什么?通过AttributeSet
获取的值,如果是引用都变成了@+数字的字符串。你说,这玩意你能看懂么?那么你看看最后一行使用TypedArray获取的值,是不是瞬间明白了什么。
TypedArray
其实是用来简化我们的工作的,比如上例,如果布局中的属性的值是引用类型(比如:@dimen/dp100),如果使用AttributeSet
去获得最终的像素值,那么需要第一步拿到id,第二步再去解析id。而TypedArray正是帮我们简化了这个过程。
贴一下:如果通过AttributeSet
获取最终的像素值的过程:
<code class="language-java hljs has-numbering"><span class="hljs-keyword">int</span> widthDimensionId = attrs.getAttributeResourceValue(<span class="hljs-number">0</span>, -<span class="hljs-number">1</span>); Log.e(TAG, <span class="hljs-string">"layout_width= "</span>+getResources().getDimension(widthDimensionId)); </code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li></ul>
ok,现在别人问你TypedArray存在的意义,你就可以告诉他了。
4、declare-styleable
我们已经解决了两个问题,接下来,我们看看布局文件,我们有一个属性叫做:zhy:text
。
总所周知,系统提供了一个属性叫做:android:text
,那么我觉得直接使用android:text
更nice,这样的话,考虑问题:
如果系统中已经有了语义比较明确的属性,我可以直接使用嘛?
答案是可以的,怎么做呢?
直接在attrs.xml中使用android:text
属性。
<code class="language-xml hljs has-numbering"> <span class="hljs-tag"><<span class="hljs-title">declare-styleable</span> <span class="hljs-attribute">name</span>=<span class="hljs-value">"test"</span>></span> <span class="hljs-tag"><<span class="hljs-title">attr</span> <span class="hljs-attribute">name</span>=<span class="hljs-value">"android:text"</span> /></span> <span class="hljs-tag"><<span class="hljs-title">attr</span> <span class="hljs-attribute">name</span>=<span class="hljs-value">"testAttr"</span> <span class="hljs-attribute">format</span>=<span class="hljs-value">"integer"</span> /></span> <span class="hljs-tag"></<span class="hljs-title">declare-styleable</span>></span></code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li></ul>
注意,这里我们是使用已经定义好的属性,不需要去添加format
属性(注意声明和使用的区别,差别就是有没有format)。
然后在类中这么获取:ta.getString(R.styleable.test_android_text)
;布局文件中直接android:text="@string/hello_world"
即可。
这里提一下,系统中定义的属性,其实和我们自定义属性的方式类似,你可以在sdk/platforms/android-xx/data/res/values
该目录下看到系统中定义的属性。然后你可以在系统提供的View(eg:TextView)的构造方法中发现TypedArray获取属性的代码(自己去看一下)。
ok,接下来,我在想,既然declare-styleable
这个标签的name都能随便写,这么随意的话,那么考虑问题:
styleable 的含义是什么?可以不写嘛?我自定义属性,我声明属性就好了,为什么一定要写个styleable呢?
其实的确是可以不写的,怎么做呢?
- 首先删除declare-styleable的标签
那么现在的attrs.xml为:
<code class="language-xml hljs has-numbering"><span class="hljs-pi"><?xml version="1.0" encoding="utf-8"?></span> <span class="hljs-tag"><<span class="hljs-title">resources</span>></span> <span class="hljs-tag"><<span class="hljs-title">attr</span> <span class="hljs-attribute">name</span>=<span class="hljs-value">"testAttr"</span> <span class="hljs-attribute">format</span>=<span class="hljs-value">"integer"</span> /></span> <span class="hljs-tag"></<span class="hljs-title">resources</span>></span></code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li></ul>
哟西,so清爽~
* MyTextView实现
<code class="language-java hljs has-numbering"><span class="hljs-keyword">package</span> com.example.test; <span class="hljs-keyword">import</span> android.content.Context; <span class="hljs-keyword">import</span> android.content.res.TypedArray; <span class="hljs-keyword">import</span> android.util.AttributeSet; <span class="hljs-keyword">import</span> android.util.Log; <span class="hljs-keyword">import</span> android.view.View; <span class="hljs-keyword">public</span> <span class="hljs-class"><span class="hljs-keyword">class</span> <span class="hljs-title">MyTextView</span> <span class="hljs-keyword">extends</span> <span class="hljs-title">View</span> {</span> <span class="hljs-keyword">private</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">final</span> String TAG = MyTextView.class.getSimpleName(); <span class="hljs-keyword">private</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">final</span> <span class="hljs-keyword">int</span>[] mAttr = { android.R.attr.text, R.attr.testAttr }; <span class="hljs-keyword">private</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">final</span> <span class="hljs-keyword">int</span> ATTR_ANDROID_TEXT = <span class="hljs-number">0</span>; <span class="hljs-keyword">private</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">final</span> <span class="hljs-keyword">int</span> ATTR_TESTATTR = <span class="hljs-number">1</span>; <span class="hljs-keyword">public</span> <span class="hljs-title">MyTextView</span>(Context context, AttributeSet attrs) { <span class="hljs-keyword">super</span>(context, attrs); <span class="hljs-comment">// ==>use typedarray</span> TypedArray ta = context.obtainStyledAttributes(attrs, mAttr); String text = ta.getString(ATTR_ANDROID_TEXT); <span class="hljs-keyword">int</span> textAttr = ta.getInteger(ATTR_TESTATTR, -<span class="hljs-number">1</span>); <span class="hljs-comment">//输出 text = Hello world! , textAttr = 520</span> Log.e(TAG, <span class="hljs-string">"text = "</span> + text + <span class="hljs-string">" , textAttr = "</span> + textAttr); ta.recycle(); } } </code>
貌似多了些代码,可以看到我们声明了一个int数组,数组中的元素就是我们想要获取的attr的id。并且我们根据元素的在数组中的位置,定义了一些整形的常量代表其下标,然后通过TypedArray
进行获取。
可以看到,我们原本的:
<code class="language-xml hljs has-numbering">R.styleable.test => mAttr R.styleable.test_text => ATTR_ANDROID_TEXT(0) R.styleable.test_testAttr => ATTR_TESTATTR(1)</code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li></ul>
那么其实呢?android在其内部也会这么做,按照传统的写法,它会在R.java生成如下代码:
<code class="language-java hljs has-numbering"><span class="hljs-keyword">public</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">final</span> <span class="hljs-class"><span class="hljs-keyword">class</span> <span class="hljs-title">attr</span> {</span> <span class="hljs-keyword">public</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">final</span> <span class="hljs-keyword">int</span> testAttr=<span class="hljs-number">0x7f0100a9</span>; } <span class="hljs-keyword">public</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">final</span> <span class="hljs-class"><span class="hljs-keyword">class</span> <span class="hljs-title">styleable</span> {</span> <span class="hljs-keyword">public</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">final</span> <span class="hljs-keyword">int</span> test_android_text = <span class="hljs-number">0</span>; <span class="hljs-keyword">public</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">final</span> <span class="hljs-keyword">int</span> test_testAttr = <span class="hljs-number">1</span>; <span class="hljs-keyword">public</span> <span class="hljs-keyword">static</span> <span class="hljs-keyword">final</span> <span class="hljs-keyword">int</span>[] test = { <span class="hljs-number">0x0101014f</span>, <span class="hljs-number">0x7f0100a9</span> }; }</code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li></ul>
ok,根据上述你应该发现了什么。styleale的出现系统可以为我们完成很多常量(int[]数组,下标常量)等的编写,简化我们的开发工作(想想如果一堆属性,自己编写常量,你得写成什么样的代码)。那么大家肯定还知道declare-styleable
的name属性,一般情况下写的都是我们自定义View的类名。主要为了直观的表达,该declare-styleable
的属性,都是改View所用的。
其实了解该原理是有用的,详见:Android 自定义控件 优雅实现元素间的分割线
ok,现在5个问题,回答了4个,第一个问题:
自定义属性的几个步骤是如何奏效的?
恩,上述以及基本涵盖了这个问题的答案,大家自己总结,所以:略。
总结下今天的博客。
- attrs.xml里面的declare-styleable以及item,android会根据其在R.java中生成一些常量方便我们使用(aapt干的),本质上,我们可以不声明declare-styleable仅仅声明所需的属性即可。
- 我们在View的构造方法中,可以通过
AttributeSet
去获得自定义属性的值,但是比较麻烦,而TypedArray
可以很方便的便于我们去获取。 - 我们在自定义View的时候,可以使用系统已经定义的属性。
近期的更新计划:自定义View的一些细节相关的Blog(重点会在交互上),Android最佳实践相关的文章,framework相关的一些文章,敬请期待
自定义控件在android中无处不见,自定义控件给了我们很大的方便。比如说,一个视图为imageview ,imagebutton ,textview 等诸多控件的组合,用的地方有很多,我们不可能每次都来写3个的组合,既浪费时间,效率又低。在这种情况下,我们就可以自定义一个view来替换他们,不仅提升了效率并且在xml中运用也是相当的美观。
一、控件自定义属性介绍
以下示例中代码均在values/attrs.xml 中定义,属性均可随意命名。
1. reference:参考某一资源ID。
示例:- <declare-styleable name = "名称">
- <attr name = "background" format = "reference" />
- <attr name = "src" format = "reference" />
- </declare-styleable>
示例:
- <declare-styleable name = "名称">
- <attr name = "textColor" format = "color" />
- </declare-styleable>
示例:
- <declare-styleable name = "名称">
- <attr name = "focusable" format = "boolean" />
- </declare-styleable>
示例:
- <declare-styleable name = "名称">
- <attr name = "layout_width" format = "dimension" />
- </declare-styleable>
示例:
- <declare-styleable name = "名称">
- <attr name = "fromAlpha" format = "float" />
- <attr name = "toAlpha" format = "float" />
- </declare-styleable>
示例:
- <declare-styleable name = "名称">
- <attr name = "frameDuration" format="integer" />
- <attr name = "framesCount" format="integer" />
- </declare-styleable>
示例:
- <declare-styleable name = "名称">
- <attr name = "text" format = "string" />
- </declare-styleable>
示例:
- <declare-styleable name="名称">
- <attr name = "pivotX" format = "fraction" />
- <attr name = "pivotY" format = "fraction" />
- </declare-styleable>
示例:
- <declare-styleable name="名称">
- <attr name="orientation">
- <enum name="horizontal" value="0" />
- <enum name="vertical" value="1" />
- </attr>
- </declare-styleable>
10. flag:位或运算。
示例:- <declare-styleable name="名称">
- <attr name="windowSoftInputMode">
- <flag name = "stateUnspecified" value = "0" />
- <flag name = "stateUnchanged" value = "1" />
- <flag name = "stateHidden" value = "2" />
- <flag name = "stateAlwaysHidden" value = "3" />
- </attr>
- </declare-styleable>
示例:
- <declare-styleable name = "名称">
- <attr name = "background" format = "reference|color" />
- </declare-styleable>
-------------------------------------------------------------------------------------------
二、属性的使用以及自定义控件的实现
1、构思控件的组成元素,思考所需自定义的属性。
比如:我要做一个 <带阴影的按钮,按钮正下方有文字说明>(类似9宫格按钮)
新建values/attrs.xml
- <?xml version="1.0" encoding="utf-8"?>
- <resources>
- <declare-styleable name="custom_view">
- <attr name="custom_id" format="integer" />
- <attr name="src" format="reference" />
- <attr name="background" format="reference" />
- <attr name="text" format="string" />
- <attr name="textColor" format="color" />
- <attr name="textSize" format="dimension" />
- </declare-styleable>
- </resources>
以上,所定义为custom_view,custom_id为按钮id,src为按钮,background为阴影背景,text为按钮说明,textColor为字体颜色,textSize为字体大小。
2、怎么自定义控件呢,怎么使用这些属性呢?话不多说请看代码,CustomView :
- package com.nanlus.custom;
- import com.nanlus.custom.R;
- import android.content.Context;
- import android.content.res.TypedArray;
- import android.graphics.Color;
- import android.graphics.drawable.Drawable;
- import android.util.AttributeSet;
- import android.view.Gravity;
- import android.view.View;
- import android.view.View.OnClickListener;
- import android.widget.FrameLayout;
- import android.widget.ImageButton;
- import android.widget.ImageView;
- import android.widget.TextView;
- public class CustomView extends FrameLayout implements OnClickListener {
- private CustomListener customListener = null;
- private Drawable mSrc = null, mBackground = null;
- private String mText = "";
- private int mTextColor = 0;
- private float mTextSize = 20;
- private int mCustomId = 0;
- private ImageView mBackgroundView = null;
- private ImageButton mButtonView = null;
- private TextView mTextView = null;
- private LayoutParams mParams = null;
- public CustomView(Context context) {
- super(context);
- }
- public CustomView(Context context, AttributeSet attrs) {
- super(context, attrs);
- TypedArray a = context.obtainStyledAttributes(attrs,
- R.styleable.custom_view);
- mSrc = a.getDrawable(R.styleable.custom_view_src);
- mBackground = a.getDrawable(R.styleable.custom_view_background);
- mText = a.getString(R.styleable.custom_view_text);
- mTextColor = a.getColor(R.styleable.custom_view_textColor,
- Color.WHITE);
- mTextSize = a.getDimension(R.styleable.custom_view_textSize, 20);
- mCustomId = a.getInt(R.styleable.custom_view_custom_id, 0);
- mTextView = new TextView(context);
- mTextView.setTextSize(mTextSize);
- mTextView.setTextColor(mTextColor);
- mTextView.setText(mText);
- mTextView.setGravity(Gravity.CENTER);
- mTextView.setLayoutParams(new LayoutParams(LayoutParams.WRAP_CONTENT,
- LayoutParams.WRAP_CONTENT));
- mButtonView = new ImageButton(context);
- mButtonView.setImageDrawable(mSrc);
- mButtonView.setBackgroundDrawable(null);
- mButtonView.setLayoutParams(new LayoutParams(LayoutParams.WRAP_CONTENT,
- LayoutParams.WRAP_CONTENT));
- mButtonView.setOnClickListener(this);
- mBackgroundView = new ImageView(context);
- mBackgroundView.setImageDrawable(mBackground);
- mBackgroundView.setLayoutParams(new LayoutParams(
- LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT));
- addView(mBackgroundView);
- addView(mButtonView);
- addView(mTextView);
- this.setOnClickListener(this);
- a.recycle();
- }
- @Override
- protected void onAttachedToWindow() {
- super.onAttachedToWindow();
- mParams = (LayoutParams) mButtonView.getLayoutParams();
- if (mParams != null) {
- mParams.gravity = Gravity.CENTER_HORIZONTAL | Gravity.TOP;
- mButtonView.setLayoutParams(mParams);
- }
- mParams = (LayoutParams) mBackgroundView.getLayoutParams();
- if (mParams != null) {
- mParams.gravity = Gravity.CENTER_HORIZONTAL | Gravity.TOP;
- mBackgroundView.setLayoutParams(mParams);
- }
- mParams = (LayoutParams) mTextView.getLayoutParams();
- if (mParams != null) {
- mParams.gravity = Gravity.CENTER_HORIZONTAL | Gravity.BOTTOM;
- mTextView.setLayoutParams(mParams);
- }
- }
- public void setCustomListener(CustomListener l) {
- customListener = l;
- }
- @Override
- public void onClick(View v) {
- if (customListener != null) {
- customListener.onCuscomClick(v, mCustomId);
- }
- }
- public interface CustomListener {
- void onCuscomClick(View v, int custom_id);
- }
- }
3、自定义控件的使用
话不多说,请看代码,main.xml:
- <?xml version="1.0" encoding="utf-8"?>
- <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
- xmlns:nanlus="http://schemas.android.com/apk/res/com.nanlus.custom"
- android:layout_width="fill_parent"
- android:layout_height="fill_parent" >
- <LinearLayout
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_centerHorizontal="true"
- android:layout_centerVertical="true"
- android:orientation="horizontal" >
- <com.nanlus.custom.CustomView
- android:id="@+id/custom1"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:layout_weight="1"
- nanlus:background="@drawable/background"
- nanlus:custom_id="1"
- nanlus:src="@drawable/style_button"
- nanlus:text="按钮1" >
- </com.nanlus.custom.CustomView>
- </LinearLayout>
- </RelativeLayout>
在这里需要解释一下,
xmlns:nanlus="http://schemas.android.com/apk/res/com.nanlus.custom"
nanlus为在xml中的前缀,com.nanlus.custom为包名
4、在Activity中,直接上代码
- package com.nanlus.custom;
- import android.os.Bundle;
- import android.view.View;
- import android.widget.ImageButton;
- import android.widget.ImageView;
- import android.widget.TextView;
- import android.widget.Toast;
- import com.nanlus.BaseActivity;
- import com.nanlus.custom.R;
- import com.nanlus.custom.CustomView.CustomListener;
- public class CustomActivity extends BaseActivity implements CustomListener {
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
- ((CustomView) this.findViewById(R.id.custom1)).setCustomListener(this);
- }
- @Override
- public void onCuscomClick(View v, int custom_id) {
- switch (custom_id) {
- case 1:
- Toast.makeText(this, "hello !!!", Toast.LENGTH_LONG).show();
- break;
- default:
- break;
- }
- }
- }
至此,自定义控件属性以及运用讲解完毕,谢谢大家。本文链接: http://blog.csdn.net/nanlus/article/details/8219868
-
顶
- 2
-
踩
- 0
我的同类文章
- •iostream与iostream.h乱弹琴2014-07-14
- •Android 手机上获取物理唯一标识码2013-03-14
- •Android 横屏时禁止输入法全屏2012-10-04
- •Android的Zipalign优化2012-08-24
- •Intent的FLAG_ACTIVITY_CLEAR_TOP和FLAG_ACTIVITY_REORDER_TO_FRONT小结2012-08-01
- •Android JNI之Android.mk、Application.mk2014-01-25
- •关于ListView顶部和底部滚(拖)动出现阴影解决方案2012-11-11
- •2.3.1 Launcher编译与修改2012-09-22
- •Android 源码下载、编译以及编译自己的apk2012-08-15
-
猜你在找
- Qt基础与Qt on Android入门
- Android入门实战教程
- Android底层技术:Java层系统服务(Android Service)
- 精讲精练_参悟Android核心技术
- Android开源项目实践之UI篇
核心技术类目
- 个人资料
-
- 访问:103816次
- 积分:1111
- 等级:
- 排名:千里之外
- 原创:18篇
- 转载:8篇
- 译文:0篇
- 评论:23条
- 文章搜索
- 阅读排行
- 黑苹果教程 win7+virtualbox安装Mac os搭建完美越狱环境(26259)
- cocos2d-x与ISO内存管理(9650)
- 关于ListView顶部和底部滚(拖)动出现阴影解决方案(9042)
- Cocos2dx3.1-Android环境搭建初体验(6538)
- Cocos2dx-Android初体验(6352)
- Android 手机上获取物理唯一标识码(5501)
- Android 横屏时禁止输入法全屏(4543)
- Android 源码下载、编译以及编译自己的apk(4018)
- Android的Zipalign优化(3993)
- tom猫变声原理解析(3611)
- 评论排行
- 推荐文章
- 最新评论
- Cocos2dx-Android初体验
sinat_34388146:大神我有点问题想问你 281927610 我Q。方便就加你询问下
- 关于ListView顶部和底部滚(拖)动出现阴影解决方案
qq_16548203:请教阴影部分的颜色的怎么改变啊,
- 关于ListView顶部和底部滚(拖)动出现阴影解决方案
vivian陈薇:好像在 xml 里边设置的 scrollmode 那个为 never,好像还是没起作用
- 检测某个应用是否开机启动
fei91818:为什么获取到的返回值一直是0呢
- 关于ListView顶部和底部滚(拖)动出现阴影解决方案
ddxss2012:。。。直接一行mListView.setOverScrollMode(View.OVER_SCROL...
- Cocos2dx3.1-Android环境搭建初体验
纯洁码农:@qimingxingyuliyang:python setup.py
- Cocos2dx3.1-Android环境搭建初体验
qimingxingyuliyang:请问单击setup.py出现闪退是什么原因
- Cocos2dx-Android初体验
超逸_流:我想说。。。怎么比官网的还麻烦?
- Cocos2dx3.1-Android环境搭建初体验
prehonor:@lqs1173783069:jni文件夹下的android.mk文件有修改吗?比如:LOCAL_W...
- Cocos2dx3.1-Android环境搭建初体验
prehonor:2,3,4楼,你的jni文件夹下的android.mk相应的修改了吗?比如下面这个用到的静态库添加了...
暂无评论