Android_开发_Day28_自定义控件

Android_开发Day28自定义控件

目的:

学会组合一些系统控件成一个控件,以满足某些需要

技术:

<1> 自定义控件的三种方式:

自定义控件有三种方式:
方式一:组合方式 也就是用系统的控件进行组合,拼接
方式二:继承方式 用一个自定义控件类来继承系统的控件
方式三:自绘方式 自己去画控件

<2> 组合方式自定义控件:

组合方式就是将系统已有的控件进行组合,一般就是先创造一个容器布局,然后在这个容器布局里面添加控件,死的控件就用xml配置,活的控件就用代码来添加,然后完成之后将该容器看成一个整体,像添加其余控件一样把它添加到相应的位置即可。

<3> 继承方式自定义控件:

继承方式自定义控件较组合方式的优点就是能够封装,组合方式做的控件简单但是难以移植,而继承方式恰好用到了类的思想,能够高效的移植。因此继承方式定义控件其实就是自己创造一个类来继承系统类,因此自定义控件首先就要选好继承的对象,根据你要实现的组合控件的类型来选择布局,往往要继承的也就是这些布局。
自定义控件的基本步骤如下:
1.创建一个类继承系统控件
2.重写里面的构造方法,一般就是实现前三个:PageController(Context context) 代码创建时需要用, PageController(Context context, AttributeSet attrs) xml创建时系统会自动调用该方法,PageController(Context context, AttributeSet attrs, int defStyleAttr)
xml创建时还有样式就会调用该方法
3.更改构造方法依次访问参数多的那个
4.实现功能->实现功能的地方:
(1) 创建控件就默认有了的功能 在构造方法里面写
(2) 用户设置的 对应的属性的set方法里面写
(3) 数据源 接口里面写
以上是功能实现常选的地方,当然也可以根据实际情况更改,但是一般情况是这样。

<4> 用xml来创建自己的控件:

先写好你自己的控件,用继承的方式来创建,然后再在values文件夹下新建一个valueresource file,具体操作如下图:


新建.png

接下来往里面写resource下敲出关键词:<declare-styleable然后设置name属性和你的自定义的控件的名字一样,配置属性,如下代码:

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <!--声明在那个控件上添加属性-->
    <declare-styleable name="PageController">
        <attr name="padding" format="integer"/>
        <attr name="resID" format="reference|color"/>
        <attr name="numbeOfPage" format="integer"/>
    </declare-styleable>
</resources>
<!--
自定义属性的步骤
1.创建一个value资源文件
2.使用declare-styleable关键字修饰 name的值为自己定义的类的名称
3.关键子attr 添加属性name和对应的值的类型
  integer整数
-->

有了这个接下来就是要如何从xml里面去得到该值了,这时就要从构造方法入手了,还记得构造方法三个各自的参数与不同在哪里吗,没错,就是参数最多的那个构造方法,参数attrs就是xml里面传来的所有属性的值组成的集合,是由系统调用的,因此第一步就是要依次遍历取出数组元素,然后第二步就是要去为属性赋值。参考如下代码:

if (attrs != null){
            TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.PageController);
            //属性名和没有的时候的默认值
            padding = typedArray.getInt(R.styleable.PageController_padding, 0);
            resID = typedArray.getResourceId(R.styleable.PageController_resource, 0);
            numberOfPages = typedArray.getInt(R.styleable.PageController_numberOfPage, 0);
            setNumberOfPages(numberOfPages);
        }

需要一个TypedArray对象来保存这些值,然后依次根据属性的数据类型来取。

技术如何使用:

做一个简单的组合控件,banner下面的点点点,第一,用最老土的方式来做就是直接在代码中组合,每个点是一个ImageView,用两种不同的状态来表示点的选中与否,第二种就是学会在此基础上封装,用一个类来管理,因此需要将其继承一个布局,这里LinearLayout就比较和适,该类代码参考如下:

package com.example.customattr;

import android.animation.ObjectAnimator;
import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Color;
import android.util.AttributeSet;
import android.view.Gravity;
import android.view.MotionEvent;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.LinearLayout;

public class PageController extends LinearLayout {

    private int numberOfPages;
    private int resID;//不同状态下显示的形状和颜色
    private int padding;//间距
    private int currentPages;//记录当前的页数
    private PageChangeListener pageChangeListener;//回调对象

    public PageController(Context context) {
        this(context, null);
    }

    public PageController(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public PageController(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        setOrientation(HORIZONTAL);
        setGravity(Gravity.CENTER_HORIZONTAL);
        if (attrs != null){
            TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.PageController);
            //属性名和没有的时候的默认值
            padding = typedArray.getInt(R.styleable.PageController_padding, 0);
            resID = typedArray.getResourceId(R.styleable.PageController_resource, 0);
            numberOfPages = typedArray.getInt(R.styleable.PageController_numberOfPage, 0);
            setNumberOfPages(numberOfPages);
        }
    }

    /**
     *
     */
    public int getNumberOfPages() {
        return numberOfPages;
    }

    public void setNumberOfPages(int numberOfPages) {
        this.numberOfPages = numberOfPages;
        for (int i = 0; i < numberOfPages; i++) {
            ImageView dot = new ImageView(getContext());
            dot.setBackgroundResource(resID);
            LayoutParams params = new LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);
            params.gravity = Gravity.CENTER_VERTICAL;
            if (i > 0)params.leftMargin = padding;
            else dot.setEnabled(false);
            addView(dot, params);
        }
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        if (event.getAction() == MotionEvent.ACTION_DOWN){
            int current = currentPages;
            if (event.getX() > getWidth()*0.5){
                //右边
                if (current == numberOfPages-1){
                    current = 0;
                }else {
                    current++;
                }
            }else {
                //左边
                if (current == 0){
                    current = numberOfPages-1;
                }else {
                    current--;
                }
            }
            setCurrentPages(current);
        }
        return true;
    }


    public int getCurrentPages() {
        return currentPages;
    }

    public void setCurrentPages(int currentPages) {
        //将上一次的还原为默认状态
        getChildAt(this.currentPages).setEnabled(true);
        //修改页数
        this.currentPages = currentPages;
        //将这一次的设置为选中状态
        getChildAt(currentPages).setEnabled(false);
        showAnimation((ImageView) getChildAt(this.currentPages));
        if (pageChangeListener != null) {
            pageChangeListener.pageDidChanged(currentPages);
        }
    }

    public void setPadding(int padding) {
        this.padding = padding;
    }

    public void setResID(int resID) {
        this.resID = resID;
    }

    //定义一个接口
    public interface PageChangeListener{
        void pageDidChanged(int currentPage);
    }
    public void addPageChangeListener(PageChangeListener listener){
        this.pageChangeListener = listener;
    }
    //写一个拉伸动画
    public void showAnimation(ImageView item){
        ObjectAnimator scale = ObjectAnimator.ofFloat(item, "scaleX", 1, 1.65f, 1);
        scale.setDuration(400);
        scale.start();
    }
}

上面的控件时在此基础从上加了动画,如果想让该控件能在xml里面使用,那就需要new 一个valueresource file 了,代码就是上面的。

总结:

自定义控件时用到了类的封装思想,因此以后写代码的时候想让别人能轻松的使用你的东西,类何尝不是已给很好的选择呢

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

推荐阅读更多精彩内容

  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,101评论 1 32
  • ¥开启¥ 【iAPP实现进入界面执行逐一显】 〖2017-08-25 15:22:14〗 《//首先开一个线程,因...
    小菜c阅读 6,415评论 0 17
  • 6、View的绘制 (1)当测量好一个View之后,我们就可以简单的重写 onDraw()方法,并在 Canvas...
    b5e7a6386c84阅读 1,893评论 0 3
  • 导语 当系统控件不能满足我们的需求的时候,这时候我们就需要自定义控件,根据我们的需求来定制一个能满足我们需求的控件...
    一个有故事的程序员阅读 6,395评论 2 14
  • 【Android 动画】 动画分类补间动画(Tween动画)帧动画(Frame 动画)属性动画(Property ...
    Rtia阅读 6,153评论 1 38