Android读书笔记-----自定义View(2)实现

(内容来自于Android开发艺术探索以及csdn的博主Hongyang)

一、自定义View的分类

1.继承View重写onDraw方法
实现一些特定的View效果。一些不能通过组合控件实现的View.
2.继承特定的View(比如TextView)
一般是为了扩展一些现有View的功能
3.继承特定的ViewGroup(例如LinerLayout)
4.继承ViewGroup

二、自定义View的注意点

1.如果需要控件支持wrap_content,需要用户在onMeasure方法里对自定义AT_MOST类型做处理。
2.如果要控件支持padding,需要再onDraw方法中对padding做处理。如果是继承ViewGroup 则需要再onLayout,onMeasure中考虑margin,和padding。
3.View的滑动冲突要考虑好
4.尽量不要在View中加入Handler
5.View中有线程或者动画需要及时停止。

三、绘制View时常常用到的参数canvas paint
这里写图片描述
这里写图片描述

本段有内容来自于鸿洋的自定义View博客
http://blog.csdn.net/lmj623565791/article/details/24252901 如果有不知道他的一定要去看看!!!!大神!!!

自定义View的方法

步骤是:自定义View属性,解析属性,测量控件,绘制控件
1.自定义View的属性
1、自定义View的属性,首先在res/values/ 下建立一个attrs.xml ,声明我们抛给外部使用的属性。
这里的name不能随便命名,一定要和你定义的View名称相同,既View的类名。
两种写法

<resources>
<declare-styleable name="CustomerTextView">     <attr name="textsize" format="dimension"></attr>  <attr name="text" format="string"></attr>
 attr name="textColor" format="color"></attr>
</declare-styleable>
</resources>

第二种写法 鸿洋写的 不知道为什么我这用会报错…

<?xml version="1.0" encoding="utf-8"?>  
<resources>  
<attr name="titleText" format="string" />  
<attr name="titleTextColor" format="color" />  <attr name="titleTextSize" format="dimension" />  
//引用属性
<declare-styleable name="CustomTitleView">  
  <attr name="titleText" />  
  <attr name="titleTextColor" />  
  <attr name="titleTextSize" />  
 </declare-styleable>  
</resources>  

第二部在View的构造方法中解析属性
注意 TypedArray对象用完一定要recycle掉
一定要在其他的构造方法中,调用你的你解析参数的构造方法。
this(context, attrs, 0);
this(context,null);

   public CustomerView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
  //解析自定义的属性
          TypedArray a = context.getTheme().obtainStyledAttributes(attrs, R.styleable.CustomTitleView, defStyle, 0); 
        int count_view = a.getIndexCount();
for (int i = 0; i < count_view; i++) {
      //循环得到每一次的属性 attr=R.styleable.CustomerTextView_。。。。。
            int attr = a.getIndex(i);
            switch (attr) {
 case R.styleable.CustomerTextView_text:
    textTitle = a.getString(attr);
        break;
       caseR.styleable.CustomerTextView_textColor:
    textColor = a.getInt(attr, Color.BLACK);
        break;

    case R.styleable.CustomerTextView_textsize:
 //设置字体的大小
textSize =a.getDimensionPixelSize(attr,
 (int)TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_SP, 16, getResources().getDisplayMetrics()));
                    break;
            }
     //用完之后一定要循环
  a.recycle();

2.onMeasure方法 测量自定义的控件
注意:一定要对wrap_content方法对应的AT_MOST类型做处理,否则wrap_content不生效。
setMeasuredDimension(width, height);最后写 设定View的宽高。

 @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        //得到外部设置的长宽的 类型 大小
        // EXACTLY:一般是设置了明确的值或者是MATCH_PARENT,AT_MOST:表示子布局限制在一个最大值内,一般为WARP_CONTENT
        // UNSPECIFIED:表示子布局想要多大就多大,很少使用
        int heightModel = MeasureSpec.getMode(heightMeasureSpec);
        int heightSize = MeasureSpec.getSize(heightMeasureSpec);
        int widthModel = MeasureSpec.getMode(widthMeasureSpec);
        int widthSize = MeasureSpec.getSize(widthMeasureSpec);
        int width;
        int height;
        //如果外部设定了精确的值,那么就用精确的值
        if (heightModel == MeasureSpec.EXACTLY) {
            height = heightSize;
        } else {
   paint.setTextSize(textSize);
            //将字写到rect上
paint.getTextBounds(textTitle, 0, textTitle.length(), rect);
   float textheight = rect.height();
   height = (int) (getPaddingTop() + textheight + getPaddingBottom());
        }
if (widthModel == MeasureSpec.EXACTLY) {
            width = widthSize;
} else {

     paint.setTextSize(textSize);
     paint.getTextBounds(textTitle, 0, textTitle.length(), rect);
    float textwidth = rect.width();
     width = (int) (textwidth + getPaddingLeft() + getPaddingRight());
        }

  //设置计算出的值
 setMeasuredDimension(width, height);
    }

onDraw方法绘制控件
注意onDraw方法使用时,如果加入了padidng要考虑padding的因素

 protected void onDraw(Canvas canvas)
    {
        paint.setColor(Color.YELLOW);
        canvas.drawRect(0, 0, getMeasuredWidth(), getMeasuredHeight(), paint);

        paint.setColor(textColor);
        canvas.drawText(textTitle, getWidth() / 2 - rect.width() / 2, getHeight() / 2 + rect.height() / 2, paint);
    }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值