近日在android项目要使用开关控件。可是android中自带的开关控件不太惬意,所以就打算通过自己定义View写一个开关控件

ios的开关控件当然就是我要仿照的目标。

先上图:

 

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/Center" alt=""> 

分析:

开关控件,中包括了两个部分,一个是一个圆,一个是圆角矩形,好了那我们仅仅要通过view来进行绘制这两部分就能够了

直接上代码:

package com.example.widget;

import com.example.switchbutton_master.R;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.RectF;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.widget.Toast; //自己定义开关控件
public class SwitchbuttonView extends View{
//开关圆圈的按钮
private Bitmap swithSilder;
//获取的开关圆圈的宽度
private int swithWidth;
//开关圆圈在x轴上的位置
private int swithShilerX; //在轻微调整大小的长度
private int baseHeight;
private int baseWidth;
//开关控件当前的状态,true:开,false:关
private Boolean currentState = false;
//画笔
private Paint paint; /** 开关状态改变监听 */
private OnToggleStateChangeListener mListener; public SwitchbuttonView(Context context, AttributeSet attrs) {
super(context, attrs);
initBitmap();
} private void initBitmap() {
swithSilder = BitmapFactory.decodeResource(getResources(), R.drawable.swich_slider_new);
swithWidth = swithSilder.getWidth();
baseHeight = swithWidth / 10;
baseWidth = swithWidth / 5; paint = new Paint();
//初始画笔的样式没不填充
paint.setStyle(Paint.Style.STROKE);
//画笔颜色为灰色
paint.setColor(Color.parseColor("#D3D1D1"));
// 去锯齿
paint.setAntiAlias(true);
//设置笔的颜色
paint.setStrokeWidth(2);
//设置开关圆圈的x轴位置
swithShilerX = -3;
} @Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
//canvas.drawColor(Color.WHITE);
//绘制开关的按钮
//绘制一个圆角的矩形,依据开关的大小进行绘制
RectF re3 = new RectF(0,0, swithWidth*2 - baseWidth, swithWidth - baseHeight);
canvas.drawRoundRect(re3, swithWidth/2, swithWidth/2, paint);
//在绘制一个控制按钮的图片,x轴为-3,是要让他向x轴的左側移动一点
canvas.drawBitmap(swithSilder,swithShilerX, 0, null);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
//设置控件的大小为开关圆圈的大小,必须设置。不然没有办法计算当前组件的时间大小
setMeasuredDimension(swithWidth*2 - baseWidth, swithWidth);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_UP: // 手指抬起
currentState = (currentState == true) ? false : true;
//Log.i("Switch", currentState+"");
if(currentState){
//设置所绘制矩形的状态
paint.setStyle(Paint.Style.FILL); //填充
paint.setColor(Color.GREEN); //绿色
//设置开关圆圈的位置
swithShilerX = swithWidth - baseHeight - 3;
}else{
paint.setStyle(Paint.Style.STROKE); //不填充
paint.setColor(Color.parseColor("#D3D1D1")); //灰色
swithShilerX = -3;
}
//调用回调方法,传递当前的状态
mListener.onToggleStateChange(currentState);
break;
}
invalidate(); // 刷新控件,该方法会调用onDraw(Canvas canvas)方法
return true; // 自己处理事件,不让父类负责消耗事件 } /**
* 对外设置监听方法
* @param listener
*/
public void setOnToggleStateChangeListener(OnToggleStateChangeListener listener) {
this.mListener = listener;
} //用于进行设置当开关的状态发生改变是同一时候上层调用这进行的处理操作
public interface OnToggleStateChangeListener{
/**
* 当开关状态改变回调此方法
* 当前开关的最新状态
*/
void onToggleStateChange(boolean b);
} }

关键点:

1. 我们在自己定义view的时候,使用的坐标都是我们当前的view视图的,不是相对于整个屏幕的

2. 绘制自己定义view关键点就是绘制(onDraw)和确定控件的大小(onMeasure)

通过上面的代码我的自己定义控件就建立了,然后就能够在界面中使用,这里要注意的是我们在activity要对该控件设置回调事件,详细使用方式:

<com.example.widget.SwitchbuttonView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/am_sbv"
/>
private SwitchbuttonView am_sbv;

	@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
am_sbv = (SwitchbuttonView) findViewById(R.id.am_sbv);
//为开关控件设置点击的回调事件
am_sbv.setOnToggleStateChangeListener(new OnToggleStateChangeListener(){ @Override
public void onToggleStateChange(boolean b) {
if(b){
Log.i("State", b+",开");
}else{
Log.i("State", b+",关");
}
} });
}

ok,这样一个自己定义的开关空间就完毕了,同志们假设有问题能够提出来。讨论一下

源代码下载

欢迎转载。

android自己定义开关控件的更多相关文章

  1. android 仿ios开关控件

    ios一些控件还是挺美丽的,可是对android程序猿来说可能比較苦逼,由于ios一些看起来简单的效果对android来说可能就没那么简单了,可是没办法非常多产品都是拿ios的一些控件叫android ...

  2. android 自己定义组合控件

    自己定义控件是一些android程序猿感觉非常难攻破的难点,起码对我来说是这种,可是我们能够在网上找一些好的博客关于自己定义控件好好拿过来学习研究下,多练,多写点也能找到感觉,把一些原理弄懂,今天就讲 ...

  3. 小米3系统计算器自己定义开关控件-MySwitchView

    1.前言             在android4.0以后,有switch控件.相似于iPhone上面滑块的效果.可是仅仅能用在4.0以后的系统中.之前的平台.就无法使用这种控件. 近段时间.看到了 ...

  4. android:自己定义组合控件Weight(高仿猫眼底部菜单条)

    在我们实际开发其中.会碰见一些布局结构类似或者同样的界面.比如应用的设置界面.tabbutton界面等. 这时候.对于刚開始学习的人来说,xml里面一个个绘制出来也许是最初的想法.可能随着经验的积累, ...

  5. Android 使用shape定义不同控件的的颜色、背景色、边框色

    Android 使用shape定义不同控件的的颜色.背景色.边框色 设置按钮的右边框和底边框颜色为红色,边框大小为3dp: 在drawable新建一个 buttonstyle.xml的文件,内容如下: ...

  6. Android常用酷炫控件(开源项目)github地址汇总

    转载一个很牛逼的控件收集帖... 第一部分 个性化控件(View) 主要介绍那些不错个性化的 View,包括 ListView.ActionBar.Menu.ViewPager.Gallery.Gri ...

  7. Android 常用炫酷控件(开源项目)git地址汇总

    第一部分 个性化控件(View) 主要介绍那些不错个性化的 View,包括 ListView.ActionBar.Menu.ViewPager.Gallery.GridView.ImageView.P ...

  8. Android中通过WebView控件实现与JavaScript方法相互调用的地图应用

    在Android中通过WebView控件,可以实现要加载的页面与Android方法相互调用,我们要实现WebView中的addJavascriptInterface方法,这样html才能调用andro ...

  9. [Android学习笔记]组合控件的使用

    组合控件的使用 开发过程中,多个UI控件需要协同工作,相互交互之后,才可完成一个完整的业务需求,此时可把这些控件封装成为一个整体,相互之间的交互逻辑封装其中,外部调用可无需关心内部逻辑,只需获取处理后 ...

随机推荐

  1. 注册苹果开发者账号和iOS9打包上线

    链接: http://www.jianshu.com/p/507ca4e5fde0 http://blog.csdn.net/a283127993/article/details/45828175 i ...

  2. iOS学习笔记20-地图(二)MapKit框架

    一.地图开发介绍 从iOS6.0开始地图数据不再由谷歌驱动,而是改用自家地图,当然在国内它的数据是由高德地图提供的. 在iOS中进行地图开发主要有三种方式: 利用MapKit框架进行地图开发,利用这种 ...

  3. hihoCoder 1367 等式填空

    明确题意 等号左边是由'+'和'?'组成的算式,其中处于某个整数(即便这个整数只有一位)首位的'?'可以填入1-9中的某个数字,其余'?'可以填入0-9中的某个数字. SOURCE 这里未明确等号左边 ...

  4. 刷题总结——切蛋糕(ssoj)

    题目: 切蛋糕 (cake.cpp/c/pas) [问题描述] BG 有一块细长的蛋糕,长度为�. 有一些人要来BG 家里吃蛋糕, BG把蛋糕切成了若干块(整数长度),然后分给这些人.为了公平,每个人 ...

  5. 源码分析 脱壳神器ZjDroid工作原理

    0. 神器ZjDroid Xposed框架的另外一个功能就是实现应用的简单脱壳,其实说是Xposed的作用其实也不是,主要是模块编写的好就可以了,主要是利用Xposed的牛逼Hook技术实现的,下面就 ...

  6. Apache Sqoop - Overview Apache Sqoop 概述

    使用Hadoop来分析和处理数据需要将数据加载到集群中并且将它和企业生产数据库中的其他数据进行结合处理.从生产系统加载大块数据到Hadoop中或者从大型集群的map reduce应用中获得数据是个挑战 ...

  7. P3102 [USACO14FEB]秘密代码Secret Code

    题目描述 Farmer John has secret message that he wants to hide from his cows; the message is a string of ...

  8. CentOS配置DHCP服务器

    知识储备 bootp (boot protocol) 早前用于无盘工作站,dhcp的前身 IP初次分配完成,以后固定mac和IP绑定关系 dhcp基础 获取IP步骤 step1: Client dhc ...

  9. awk 对简单文本处理试水

    #juanjuan是一个文件 [root@localhost c_test]# cat juanjuan , , bffd97d0 , , bffd97cc , , bffd97c8 , , #-F ...

  10. ci框架——辅助函数

    辅助函数:application/helper下面.命名要求为***_helper.php;这样在调用的时候直接$this->load->helper('***');若想给自定义的辅助函数 ...