Android自定义控件实现简单滑动开关效果

本文实例为大家分享了Android自定义控件实现简单滑动开关的具体代码,供大家参考,具体内容如下

ToggleButton 滑动开关

项目概述

滑动开关是一个纯粹的自定义控件,上面的按钮会随着我们的左右滑动而滑动,并且在状态改变时通知用户,效果如下图1-9 所示,这也是应用中设置某些状态信息时最常见的控件,因此,我们有必要学习关于如何

自定义一个这样的滑动开关。

滑动开关UI

布局文件为activity_main.xml,代码如下:res/layout/activity_main.xml

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
                xmlns:itheima="http://schemas.android.com/apk/res/com.itheima.togglebuttondemo"
                xmlns:tools="http://schemas.android.com/tools"
                android:layout_width="match_parent"
                android:layout_height="match_parent">
    <com.itheima.togglebuttondemo.view.ToggleButton
        android:id="@+id/togglebutton"
        android:layout_width="wrap_content"
        android:layout_centerInParent="true"
        itheima:SwitchBtnBackgroud="@drawable/switch_background"
        itheima:SlidBtnBackgroud="@drawable/slide_button_background"
        itheima:CurrentState="false"
        android:layout_height="wrap_content"/>
</RelativeLayout>

在activity_main.xml 布局中引入如下命名空间:

xmlns:itheima=”http://schemas.android.com/apk/res/com.itheima.togglebuttondemo”,com.itheima.togglebuttondemo 是包名,itheima 是自定义的命名控件名,可以任取名字,也可以使用类名。

上面的布局主要是引入com.itheima.togglebuttondemo.view.ToggleButton 类和自定义属性的使用。添加自定义属性,在values 目录下创建attrs.xml 文件,具体代码如文件所示:

res/values/attrs.xml

<?xml version="1.0" encoding="utf-8"?>
<resources>

    <declare-styleable name="ToggleButton">
        <!-- 滑动开关背景图片属性-->
        <attr
            name="SwitchBtnBackgroud"
            format="reference"/>
        <!-- 滑动块背景图片属性-->
        <attr
            name="SlidBtnBackgroud"
            format="reference"/>
        <!-- 滑动开关的状态-->
        <attr
            name="CurrentState"
            format="boolean"/>
    </declare-styleable>
</resources>

attrs.xml 文件目录结构如下图所示:

滑动开关业务逻辑实现

下拉选择框activity 界面,MainActivity.java 代码如下:com/itheima/MySwitch/MainActivity

public class MainActivity extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ToggleButton togglebutton = (ToggleButton) findViewById(R.id.togglebutton);
        //设置滑动开关的背景图片
        // togglebutton.setSwitchBtnBackgroudResource(R.drawable.switch_background);
        //设置滑动块的背景图片
        // togglebutton.setSlidBtnBackgroudResource(R.drawable.slide_button_background);
        //设置滑动开关的默认状态
        // togglebutton.setCurrentState(true);
        //设置滑动开关状态改变监听
        Togglebutton.setToggleBtnStateChangeListener(new ToggleBtnStateChangeListener() {
            @Override
            public void onToggleBtnStateChange(boolean currentState) {
                if (currentState) {
                    Toast.makeText(getApplicationContext(), "开关打开", 0).show();
                }else{
                    Toast.makeText(getApplicationContext(), "开关关闭", 0).show();
                }
            }
        });
    }
}

自定义的滑动开关ToggleButton 类的实现,具体代码如文件所示:com/itheima/MySwitch/MainActivity

public class ToggleButton extends View {
    private Bitmap  switchBitmap;//滑动开关的背景图片
    private Bitmap  slidBitmap;//滑动块的背景图片
    private boolean currentState;
    private int     currentX;//手指触摸点的X 值
    private boolean isTouching = false;
    private ToggleBtnStateChangeListener mToggleBtnStateChangeListener;//状态改变监听器
    //在xml 中引用该控件时,调用该方法
    public ToggleButton(Context context, AttributeSet attrs) {
        super(context, attrs);
        String namespace = "http://schemas.android.com/apk/res/com.itheima.togglebuttondemo";
        currentState = attrs.getAttributeBooleanValue(namespace, "CurrentState", false);
        int switchBtnBackgroudId =
                attrs.getAttributeResourceValue(namespace, "SwitchBtnBackgroud", -1);
        int slidBtnBackgroudId =
                attrs.getAttributeResourceValue(namespace, "SlidBtnBackgroud", -1);
        setSwitchBtnBackgroudResource(switchBtnBackgroudId);
        setSlidBtnBackgroudResource(slidBtnBackgroudId);
    }
    //在代码中创建该控件时,调用该构造方法
    public ToggleButton(Context context) {
        super(context);
    }
    //设置滑动开关的背景图片
    public void setSwitchBtnBackgroudResource(int switchBackground) {
        switchBitmap = BitmapFactory.decodeResource(getResources(), switchBackground);
    }
    // 为了可以高度自定义和增强可扩展性,我们可以给其创建一个自定义控件底部背景了一个方法
    // 设置滑动块的背景图片
    public void setSlidBtnBackgroudResource(int slideButtonBackground) {
        slidBitmap = BitmapFactory.decodeResource(getResources(), slideButtonBackground);
    }
    //设置滑动开关的默认状态
    public void setCurrentState(boolean b) {
        currentState = b;
    }
    // 1、测量滑动开关的宽高
    // 测量控件的宽高
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        setMeasuredDimension(switchBitmap.getWidth(), switchBitmap.getHeight());
    }
    // 2、绘制,画出我们的滑动开关
    //canvas:画布,将图形绘制在canvas,才能显示到屏幕上
    @Override
    protected void onDraw(Canvas canvas) {
        //绘制滑动开关的背景图片
        canvas.drawBitmap(switchBitmap, 0, 0, null);
        //绘制滑动块的背景图片
        if(isTouching){//手指触摸的时候,根据currentx 的值来绘制滑动块
            //根据手指的X 值,来绘制滑动块图片
            int left = currentX - slidBitmap.getWidth()/2;
            if(left < 0){//设置左边界
                left = 0;
            }else if(left > (switchBitmap.getWidth() - slidBitmap.getWidth())){//设置右边界
                left = switchBitmap.getWidth() - slidBitmap.getWidth();
            }
            canvas.drawBitmap(slidBitmap, left, 0, null);
        }else{ // 手指离开控件的时候,根据状态来绘制滑动块
            // 根据状态值,来绘制滑动块
            if(currentState){ //当前为true,开关打开,滑动块显示在最右边
                canvas.drawBitmap(slidBitmap,switchBitmap.getWidth() - slidBitmap.getWidth(),
                        0, null);
            }else{//当前为false,开关关闭,滑动块显示在最左边
                canvas.drawBitmap(slidBitmap, 0, 0, null);
            }
        }
    }
    //当控件被触摸后,会调用该方法
    @Override
    public boolean onTouchEvent(MotionEvent event) {
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN://手指按下
                isTouching = true;
                currentX = (int) event.getX();
                break;
            case MotionEvent.ACTION_MOVE://手指滑动
                isTouching= true;
                currentX = (int) event.getX();
                break;
            case MotionEvent.ACTION_UP://手指抬起
                isTouching = false;
                currentX = (int) event.getX();
                int center = switchBitmap.getWidth()/2;
                //当滑动块中心点大于滑动开关背景图片的中心线时,显示到右边,当前状态为true
                boolean state = currentState;
                currentState = currentX > center;
                if(mToggleBtnStateChangeListener !=null&&state != currentState ){
                    mToggleBtnStateChangeListener.onToggleBtnStateChange(currentState);
                }
                break;
            default:
                break;
        }
        invalidate(); //强制让控件重新绘制,ondraw;
        return true; //自己处理触摸事件
    }
    public void setToggleBtnStateChangeListener(ToggleBtnStateChangeListenerlistener){
        this.mToggleBtnStateChangeListener = listener;
    }
    // 定义滑动开关状态改变的回调接口
    public interface ToggleBtnStateChangeListener{
        void onToggleBtnStateChange(boolean currentState);
    }
}

运行程序,效果图如图1-11 所示。

知识点总结

1.通过setMeasuredDimension 方法,来设置自定义控件的宽高,见ToggleButton 类第42 行
2.View 可以通过invalidate()方法强制让自己重新绘制,见ToggleButton 类第96 行
3.View 通过实现onTouchEvent 方法来处理手指触摸事件,见ToggleButton 类第72 行

自定义控件之自定义属性

当我们使用自定义属性来自定义控件时,一般分为以下几个步骤进行设置:

1、在res 文件的values 里面创建attrs.xml,见文件【1-10】attrs.xml
2、在attrs.xml,里面定义我们需要的属性,见文件【1-10】attrs.xml 代码
3、在布局文件中使用自定义的属性,注意要添加命名空间,见文件【1-9】activity_main.xml 第2 行
4、在构造方法中来获取设置的属性数据,见文件【1-9】见ToggleButton 类第8~19 行

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android编程实现滑动开关组件功能【附源码下载】

    本文实例讲述了Android编程实现滑动开关组件功能.分享给大家供大家参考,具体如下: 由于Android并未提供滑动开关之类的组件,所以我们需要自己去实现一个自定义的视图组件来实现滑动开关效果. 这里有一个示例代码,它包括三个类:开关组件视图.状态监听接口.MainActivity 我们先来看看整个demo的效果图: 我们先来看看视图组件的完整代码,代码都已经注释: package com.bear.swtichbuttondemo; import java.util.ArrayList; i

  • Android自定义控件实现滑动开关效果

    自定义开关控件 Android自定义控件一般有三种方式 1.继承Android固有的控件,在Android原生控件的基础上,进行添加功能和逻辑. 2.继承ViewGroup,这类自定义控件是可以往自己的布局里面添加其他的子控件的. 3.继承View,这类自定义控件没有跟原生的控件有太多的相似的地方,也不需要在自己的肚子里添加其他的子控件. ToggleView自定义开关控件表征上没有跟Android原生的控件有什么相似的地方,而且在滑动的效果上也没有沿袭Android原生的地方,所以我们的自定义

  • Android开发进阶自定义控件之滑动开关实现方法【附demo源码下载】

    本文实例讲述了Android开发进阶自定义控件之滑动开关实现方法.分享给大家供大家参考,具体如下: 自定义开关控件 Android自定义控件一般有三种方式 1.继承Android固有的控件,在Android原生控件的基础上,进行添加功能和逻辑. 2.继承ViewGroup,这类自定义控件是可以往自己的布局里面添加其他的子控件的. 3.继承View,这类自定义控件没有跟原生的控件有太多的相似的地方,也不需要在自己的肚子里添加其他的子控件. ToggleView自定义开关控件表征上没有跟Androi

  • Android studio实现滑动开关

    大家好,今天刚学会使用Android Studio实现滑动开关的效果,自己感觉还可以,和大家分享一下,如果觉得可以的可以拿去,然后再给我点个赞,谢谢.本人也是学Android studiok开发不久,是个菜鸟,各位大佬觉得有不好的地方,可以讨论一下,共同学习,一起进步. 实现效果 下面是代码,代码写的比较粗糙,注释没有写的很好,需要用的可以自己移植一下 package com.example.biansheng2; import android.content.Context; import a

  • Android开发仿IOS滑动开关实现代码

    Android开发仿IOS滑动开关实现代码 Android与iOS相比,ios好多控件都是自带的,而android需要使用自定义来实现.今天说的是ios的滑动开关,我层看到好多博客都是通过自定义ToggleButton实现的.这里我通过自定义view来实现他的效果. 首先在onsizechange里把2个半圆和一个矩形绘制出来. width = w; height = h; left = top = 0; right = width; bottom = height * 0.8f; cx = (

  • Android自定义控件实现简单滑动开关效果

    本文实例为大家分享了Android自定义控件实现简单滑动开关的具体代码,供大家参考,具体内容如下 ToggleButton 滑动开关 项目概述 滑动开关是一个纯粹的自定义控件,上面的按钮会随着我们的左右滑动而滑动,并且在状态改变时通知用户,效果如下图1-9 所示,这也是应用中设置某些状态信息时最常见的控件,因此,我们有必要学习关于如何 自定义一个这样的滑动开关. 滑动开关UI 布局文件为activity_main.xml,代码如下:res/layout/activity_main.xml <Re

  • Android自定义控件实现水波纹效果

    本文实例为大家分享了Android自定义控件实现水波纹的具体代码,供大家参考,具体内容如下 示例代码: MainActivity.java package com.example.mhy.shuibowen; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; public class MainActivity extends AppCompatActivity { @Override protec

  • Android自定义控件实现简单的轮播图控件

    最近要做一个轮播图的效果,网上看了几篇文章,基本上都能找到实现,效果还挺不错,但是在写的时候感觉每次都要单独去重新在Activity里写一堆代码.于是自己封装了一下.本篇轮播图实现原理原文出处:循环广告位组件的实现,这里只是做了下封装成一个控件,不必每次重复写代码了. 效果图: 实现分析 轮播图的功能就是实现左右滑动的广告.图片信息展示,那我们就用ViewPager来实现,由于考虑到用户体验,我们还需要在下面加一个指示器来标示滑动到了第几张轮播图.指示器我们可以用一个线性布局来根据要展示的轮播图

  • Android自定义控件实现简单写字板功能

    先来看看效果图 就是简单的根据手指写下的轨迹去画出内容 一.实现 之前一篇文章里提到了android官方给出的自定义控件需要考虑以下几点: 创建View 处理View的布局 绘制View 与用户进行交互 优化已定义的View 就按照这个步骤来完成今天的自定义控件 1.创建View 上篇提到创建View这一步的时候要考虑的就是很简单的自定义属性的声明.使用. 今天的控件可以有一些什么自定义属性呢?要实现写字板,其实就是三个东西:写字板的颜色.笔的颜色.笔的粗细.所以接下来自定义属性. <?xml

  • Android自定义控件仿QQ抽屉效果

    其实网上类似的实现已经很多了,原理也并不难,只是网上各种demo运行下来,多少都有一些问题.折腾了半天,决定自己实现一个. 首先我们看看实现效果: 对比网上各类demo,这次要实现的主要表现在以下几点: 1.侧滑显示抽屉view 2.侧滑抽屉隐藏view控件点击事件 3.单击任意item隐藏显示的抽屉view 4.滑动list隐藏显示的抽屉view 5.增加SwipeLayout点击事件和Swipe touch事件判断处理 6.优化快速划开多个抽屉隐藏view时多个SwipeLayout滑动状态

  • Android自定义控件实现望远镜效果

    Android自定义控件今天要讲到的就是望远镜效果,那么什么是望远镜效果,我们不妨看看下方的动图,看完后,相信大家就有一定的认识了. 1.着色器 对于这种效果来说,其实实现起来挺简单的,但我们将会用到在三维软件中的着色器Shader,它是用来给空白图形上色的.用过PS的人,相信大家都知道里面有一个印章工具,印章的样式可以是图像,颜色,渐变色等.在Android里面,Shader的效果其实与他类似. public Shader setShader(Shader shader) 上面是Shader的

  • android实现简单仪表盘效果

    本文实例为大家分享了android实现简单仪表盘效果的具体代码,供大家参考,具体内容如下 实现这个效果: 中间的文字很好写,外层的进度条就需要自定义控件了,代码如下: public class CirCleProgressBar extends View { private Paint circlePaint; private Paint textPaint; private int circleColor;//圆弧颜色 private int circleBgColor;//圆弧背景颜色 pr

  • Android自定义控件eBook实现翻书效果实例详解

    本文实例讲述了Android自定义控件eBook实现翻书效果的方法.分享给大家供大家参考,具体如下: 效果图: Book.java文件: package com.book; import android.app.Activity; import android.os.Bundle; import android.util.Log; import android.view.View; import android.widget.ImageView; public class Book extend

  • 简单实用的Android UI微博动态点赞效果

    说起空间动态.微博的点赞效果,网上也是很泛滥,各种实现与效果一大堆.而详细实现的部分,讲述的也是参差不齐,另一方面估计也有很多大侠也不屑一顾,觉得完全没必要单独开篇来写和讲解吧.毕竟,也就是两个view和一些简单的动画效果罢了. 单若是只讲这些,我自然也是不愿花这番功夫的.虽然自己很菜,可也不甘于太菜.所以偶尔看到些好东西,可以延伸学写下,我还是很情愿拿出来用用,顺带秀一秀逼格什么的. 不扯太多,先说说今天实现点赞效果用到的自以为不错的两个点: Checkable 用来扩展View实现选中状态切

随机推荐