自定义控件 - CustomToggleButton

自定义控件 - CustomToggleButton

需求

  1. 绘图:滑块 + 背景
  2. 自定义属性:isOpen - 布局或代码控制开关
  3. 动作:触摸滑动开关,点击开关
  4. 监听:OnToggleChangeListener - 开关状态改变监听器

构建

  • 拷贝滑块和背景到资源目录
  • 自定义控件继承 View

自定义参数

  • 新建attrs.xml,定义参数isOpen,类型boolean
<resources>
    <declare-styleable name="CustomToggleButton">
        <attr name="isOpen" format="boolean"/>
    </declare-styleable>
</resources>
  • 构造方法中获取参数isOpen的初始值
TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.CustomToggleButton);
isOpen = typedArray.getBoolean(R.styleable.CustomToggleButton_isOpen, false);
typedArray.recycle();

加载数据

  • 获取滑块和背景的Bitmap
background = BitmapFactory.decodeResource(getResources(), R.drawable.ctb_switch_background);
button = BitmapFactory.decodeResource(getResources(), R.drawable.ctb_slide_button);

初始化

  • 获取滑动最大范围
  • 获取初始滑动距离
  • 经过测量后,大小数值会变化,需要重新初始化
// max = background.getWidth() - button.getWidth();
// left = isOpen ? max : 0;
scaledTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop(); // 滑动临界值

加载完成

  • 加载完成时调用onFinishInflate()

测量

设置控件大小

  • 调用setMeasuredDimension设置控件大小,以背景大小为准
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    int width = measureDimension(widthMeasureSpec, background.getWidth());
    int height = measureDimension(heightMeasureSpec, background.getHeight());
    setMeasuredDimension(width, height); // 测量完成后设置控件大小
}
  • measureSpec 为32位二进制数,前2位表示Mode,后30位表示Size
  • 一般测量方法
private int measureDimension(int measureSpec, int content) {
    int result = 0;
    int mode = MeasureSpec.getMode(measureSpec);
    int size = MeasureSpec.getSize(measureSpec);
    switch (mode) {
        case MeasureSpec.UNSPECIFIED:  // 未指定,例如ScrollView
            result = content;
            break;
        case MeasureSpec.EXACTLY:  // 确定值,match_parent和确定的数值
            result = size;
            break;
        case MeasureSpec.AT_MOST:  // 最大值,wrap_content
            result = Math.min(content, size); // 在最大可用值和内容的大小中取最小值
            break;
    }
    return result;
}

缩放

  • 测量完成,调用onSizeChanged(int w, int h, int oldw, int oldh)
  • 调整控件各部分大小
@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
    super.onSizeChanged(w, h, oldw, oldh);
    button = Bitmap.createScaledBitmap(button, button.getWidth() * w / background.getWidth(), button.getHeight() * h / background.getHeight(), true);
    background = Bitmap.createScaledBitmap(background, w, h, true);

    // 此时需初始化和大小相关的变量
    max = background.getWidth() - button.getWidth();
    left = isOpen ? max : 0;
}

布局

  • 调用onLayout(boolean changed, int left, int top, int right, int bottom),对子View进行布局

绘图

  • 调用传入的canvas绘制背景和控件
protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    canvas.drawBitmap(background, 0, 0, null);
    canvas.drawBitmap(button, left, 0, null);
}

动作

  • MotionEvent.ACTION_DOWN/MotionEvent.ACTION_MOVE/MotionEvent.ACTION_UP 分别代表触摸移动和放开
  • return true消费事件
public boolean onTouchEvent(MotionEvent event) {
     switch (event.getAction()) {
         case MotionEvent.ACTION_DOWN:
             break;
         case MotionEvent.ACTION_MOVE:
             break;
         case MotionEvent.ACTION_UP:
             break;
     }
     return true;
 }

滑块滑动开关

  • getX()/getY() 相对View左上角坐标; getRawX()/getRawY() 相对屏幕左上角坐标
  • MotionEvent.ACTION_DOWN
    • 中获取初始按压坐标
  • MotionEvent.ACTION_MOVE
    • 中获取移动坐标
    • 叠加移动距离
    • 限制滑动范围
    • 重绘
  • MotionEvent.ACTION_UP
    • 移动距离和一半最大距离比较确定开关
    • 重绘
public boolean onTouchEvent(MotionEvent event) {
    switch (event.getAction()) {
        case MotionEvent.ACTION_DOWN:
            startX = event.getX();
            break;
        case MotionEvent.ACTION_MOVE:
            float moveX = event.getX();
            float dx = moveX - startX;
            left += dx;

            // 限制范围
            if (left < 0) {
                left = 0;
            }
            if (left > max) {
                left = max;
            }

            // 移动重绘
            invalidate();
            startX = moveX;
            break;
        case MotionEvent.ACTION_UP:
            isOpen = left > max / 2;
            change();
            break;
    }
    return true;
}

private void change() {
    // 按开关重绘
    left = isOpen ? max : 0;
    invalidate();
}

点击空白开关

  • 通过时间和距离区分点击和滑动
  • 通过isOpen状态和释放坐标判断点击位置
public boolean onTouchEvent(MotionEvent event) {
    switch (event.getAction()) {
        case MotionEvent.ACTION_DOWN:
            startX = event.getX();
            startTime = SystemClock.uptimeMillis();
            break;
        case MotionEvent.ACTION_MOVE:
            ...
            break;
        case MotionEvent.ACTION_UP:
            float upX = event.getX();
            if (upX - startX < scaledTouchSlop && SystemClock.uptimeMillis() - startTime < 500) {
                // 点击
                if (isOpen) {
                    // 状态开,点击关
                    isOpen = !(upX > 0 && upX < max);
                } else {
                    // 状态关,点击开
                    isOpen = upX > button.getWidth() && upX < background.getWidth();
                }
            } else {
                // 滑动
                isOpen = left > max / 2;
            }

            change();
            break;
    }
    return true;
}

监听

  • 创建监听器
  • 比对按压和释放的开关状态判断开关状态是否改变
public boolean onTouchEvent(MotionEvent event) {
    switch (event.getAction()) {
        case MotionEvent.ACTION_DOWN:
            ...
            isCurrent = isOpen;
            break;
        ...
    }
    return true;
}

private void change() {
    // 按开关重绘
    left = isOpen ? max : 0;
    invalidate();

    // 有改变回调监听
    if (isCurrent != isOpen && listener != null) {
        listener.change(isOpen);
    }
}

// 监听器
private OnStateChangeListener listener;

public interface OnStateChangeListener {
    void change(boolean isOpen);
}

public void setOnStateChangeListener(OnStateChangeListener listener) {
    this.listener = listener;
}

开关方法

  • 3个方法,开、关和判断状态
public boolean isOpen() {
    return isOpen;
}

public void open() {
    isCurrent = isOpen;
    isOpen = true;
    change();
}

public void close() {
    isCurrent = isOpen;
    isOpen = false;
    change();
}

使用

布局

<com.library.customtogglebutton.CustomToggleButton
    android:id="@+id/custom"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    app:isOpen="true" />

监听

CustomToggleButton customToggleButton = (CustomToggleButton) findViewById(R.id.custom);
customToggleButton.setOnStateChangeListener(new CustomToggleButton.OnStateChangeListener() {
    @Override
    public void change(boolean isOpen) {
        Toast.makeText(CtbActivity.this, isOpen ? "开" : "关", Toast.LENGTH_SHORT).show();
    }
});

开关

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

推荐阅读更多精彩内容