自定义View

一、新建一个class继承View

  public class MyView extends View {
    //第一个构造方法
  public MyView(Context context) {
          super(context);
  }
    //第二个构造方法
public MyView(Context context,  AttributeSet attrs) {
       super(context, attrs);
}
         //第三个构造方法
public MyView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
 }
}

1.第一个构造方法 MyView(Context context);

创建MyView时使用

    MyView view = new MyView(this);

2.第二个构造方法 MyView(Context context, AttributeSet attrs);

你把MyView添加到布局界面时系统自动调用
AttributeSet attrs 参数 : android布局属性 下面标注attrs都在这个参数里面

  <com.test.MyView
      android:id="@+id/myview"
     android:layout_width="match_parent"------attrs 
     android:layout_height="wrap_content"------attrs 
     android:layout_marginRight="16dp"-------attrs 
     android:layout_marginTop="16dp"-------attrs />

     MyView view =(MyView)findViewById(R.id.myview);

3.第三个构造方法 MyView(Context context, AttributeSet attrs, int defStyleAttr);

int defStyleAttr 参数:自定义属性
第三个构造函数不会被系统默认调用,而是需要我们自己去显式调用

二、修改构造方法

  public class MyView extends View {
    //第一个构造方法
  public MyView(Context context) {
        super(context);
  }
    //第二个构造方法
public MyView(Context context,  AttributeSet attrs) {
        this(context,null,0);//修改这个构造方法,使其调用第三个构造方法
}
         //第三个构造方法
public MyView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
 }
}

三、MyView的绘制过程 : Measure->Layout->Draw

---onMeasure方法:一个view要占界面的大小

---onLayout方法:控件放置位置

---onDraw方法:绘制

在MyView里重写这三个构造方法

 @Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
}

@Override
protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
    super.onLayout(changed, left, top, right, bottom);
}

@Override
protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
}

1.重写onMeasure();

widthMeasureSpec,heightMeasureSpec.是两个int类型的参数,但是他们并不简单的高宽,
每一个参数里都包含了两个数据mode、size。一个int数据怎么表示两个数据呢?
每个int数据有4个字节(32bit)组成,前2个bit表示mode,后30bit表示size。

我们可以通过MeasureSpec类获得mode和size

    int Mode = MeasureSpec.getMode(MeasureSpec);
    int Size = MeasureSpec.getSize(MeasureSpec);
Mode

-----MeasureSpec.UNSPECIFIED,父视图不对子视图施加任何限制,子视图可以得到任意想要的大小;
-----MeasureSpec.EXACTLY,父视图希望子视图的大小是size中指定的大小(match_parent或指定大小);
-----MeasureSpec.AT_MOST,子视图的大小最多是size中的大小(wrap_content)。


①当父控件是EXACTLY(match_parent或指定大小)时:
---------子空间为match_parent或指定大小,Mode为:EXACTLY
---------子空间为wrap_content或指定大小,Mode为:AT_MOST

②当父控件是AT_MOST(wrap_content)时:
---------子空间为指定大小,Mode为:EXACTLY
---------子空间为match_parent或wrap_content,Mode为:AT_MOST


我们要通过Mode来设置Size的大小,就是当MyView的控件为AT_MOST时(此时并不能完全的展示我们的view),就要给它设置一个值:private final int SIZE = 200;


然后通过setMeasuredDimension(widthMeasure,heightMeasure)方法设置正确的宽高。


 private final int SIZE = 200;
 @Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    int withMode = MeasureSpec.getMode(widthMeasureSpec);
    int withSize = MeasureSpec.getSize(widthMeasureSpec);
    int heiMode = MeasureSpec.getMode(heightMeasureSpec);
    int heiSize = MeasureSpec.getSize(heightMeasureSpec);
    int widthMeasure,heightMeasure;
    if(withMode==MeasureSpec.EXACTLY){
        widthMeasure=withSize;
    }else{
        widthMeasure= SIZE;
    }
    if(heiMode==MeasureSpec.EXACTLY){
        heightMeasure=heiSize;
    }else{
        heightMeasure=SIZE;
    }
    setMeasuredDimension(widthMeasure,heightMeasure);
}

2.重写onLayout();

在onLayout()我们值需要获得view的宽高。onLayout()实际使用时是设置子view的摆放位置,我
们只是单纯view,所以不需要编辑。

private int mHeight;
private int mWidth;
@Override
protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
    super.onLayout(changed, left, top, right, bottom);
    mHeight = getHeight();
    mWidth = getWidth();
}

3.重写onDraw();

方法里只有一个参数Canvas,一个画布,我们就是需要在这个画布上进行绘制


Canvas类的使用:
----drawRect(RectF rect, Paint paint) //绘制区域,参数RectF(类)一个矩形区域
----drawPath(Path path, Paint paint) //绘制绘复杂图形,参数一为Path路径对象
----drawBitmap(Bitmap bitmap, Rect src, Rect dst, Paint paint) //贴图,参数一就是我们常规的Bitmap对象
----drawLine(float startX, float startY, float stopX, float stopY, Paintpaint) //画线
----drawPoint(float x, float y, Paint paint) //画点
----drawText(String text, float x, floaty, Paint paint) //渲染文本
----drawOval(RectF oval, Paint paint)//画椭圆
----drawCircle(float cx, float cy, float radius,Paint paint)// 绘制圆
----drawArc(RectF oval, float startAngle, float sweepAngle, boolean useCenter, Paint paint)//画弧


我们以比较简单的圆来举例:
float cx:圆的中心点x轴上位置。float cy:圆的中心点y轴上位置。float radius:半径。Paint paint:paint对象。

    Paint mPaint = new Paint();
    mPaint.setColor(custom_background);//设置背景颜色
    mPaint.setAntiAlias(true);//抗锯齿
    mPaint.setStyle(Paint.Style.FILL);//设置风格
     //  1.Paint.Style.STROKE:描边 。
         2.Paint.Style.FILL_AND_STROKE:描边并填充 。
         3.Paint.Style.FILL:填充 。

@Override
protected void onDraw(Canvas canvas) {
    canvas.drawCircle(mWidth/2, mHeight/2, custom_size * scale, mPaint);

}

我们如果想在xml布局文件里设置半径和颜色就需要用到,自定义属性了。

四、MyView的自定义属性

1.在values/attrs.xml 文件里定义一个name为自己定义view名字的declare-styleable

 <resources>
      <declare-styleable name="MyView">
              <attr name="background_color" format="color"/>
              <attr name="size" format="dimension"/>
     </declare-styleable>
</resources>

2.xml文件里使用自己定义的属性了。
----注意需要加上命名空间: xmlns:app="http://schemas.android.com/apk/res-auto"

<com.test.MyView
        android:id="@+id/myview"
        xmlns:app="http://schemas.android.com/apk/res-auto"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginRight="16dp"
        android:layout_marginTop="16dp"
        app:size="24dp"
        app:background_color="@color/colorAccent"/>

3.需要重写第三个构造函数,使其onDraw()时获得size(半径)和background_color(颜色)

  private int custom_size;
  private int custom_background;
  private final int SIZE = 200;
  private final int DEFAULT_COLOR = Color.BLUE;

  public MyView(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
   //获取自定义属性的类
    TypedArray a = context.obtainStyledAttributes(attrs, MyView, defStyleAttr, R.style.AppTheme);
  //获得自定义属性大小,SIZE在没有设置时默认值,
    custom_size = a.getDimensionPixelSize(MyView_size, SIZE);
   //获得自定义颜色,DEFAULT_COLOR在没有设置时默认值,
    custom_background = a.getColor(MyView_background_color, DEFAULT_COLOR);
    a.recycle();
}

完成。

五、让MyView动起来--动画。

    private ValueAnimator mAnimator;
   public void startAnimation() {
    mAnimator = ValueAnimator.ofFloat(1, 2);
    mAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
        @Override
        public void onAnimationUpdate(ValueAnimator animation) {
            scale = (float) animation.getAnimatedValue();
            postInvalidate();//让view重新绘制
        }
    });

    // 重复次数 -1表示无限循环
    mAnimator.setRepeatCount(-1);

    // 重复模式, RESTART: 重新开始 REVERSE:恢复初始状态再开始
    mAnimator.setRepeatMode(ValueAnimator.REVERSE);
    mAnimator.start();
}

还要重写一个构造方法,在view消失时关闭动画

   @Override
   protected void onDetachedFromWindow() {
    super.onDetachedFromWindow();
    // 关闭动画
    mAnimator.end();
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,417评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,921评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,850评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,945评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,069评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,188评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,239评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,994评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,409评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,735评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,898评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,578评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,205评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,916评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,156评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,722评论 2 363
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,781评论 2 351

推荐阅读更多精彩内容