Scroller这个类在自己定义view中使用的还算是非常频繁的,和它名字一样。我们通常是在控制滑动的时候使用Scroller,以便让view滑动起来不那么生硬。在官方的解释上,Scroller是一个滑动辅助类,也就是说Scroller本身并不參与滑动,而是让我们的代码在Scroller的辅助下轻松的实现平滑滑动的效果。

既然Scroller仅仅是一个辅助类,那能不能利用它来辅助一些其它的功能呢? 当然能够,今天带来额Toggle就是利用Scroller来实现的一个平滑的开关button。

一、实现思路

Toggle须要三张图片,一个是背景图片、一个状态为开的图片、一个状态为关的图片。

因为不会美工,只使用photozoom缩放了三张图片,并非那么完美,各位看官凑活着看吧。


 

第一张图片是我们的背景图片。当然也是通过android:background=”@drawable/xxx”来设置的。第二张是状态为开的时候的图片,当然,最后一张就是关了。

背景图片不须要我们去绘制在view的draw方法里就能够帮我们绘制完毕了,我们仅仅须要在合适的时间和合适的位置将开关两张图片画上就可以。

怎样实现从开到关一个过渡的效果呢?当然要使用前面我们提到过的Scroller了,实现的过程是:当我们点击Toggle的时候,调用scroller.start方法,从左边移动到右边,然后切换到关闭状态这个图片就ok了。

实现思路就这么简单,接下载我们来看一下代码。

二、实现代码

public class Toggle extends View {
private int mNowX; // 当前滑块的x位置
private int mSmoothDuration = 500; private boolean isOpen; // 是否为打开状态 private Drawable mOpenDrawable; // 打开状态的图片
private Drawable mCloseDrawable; // 关闭状态的图片 private Scroller mScroller; public Toggle(Context context, AttributeSet attrs) {
this(context, attrs, 0);
} public Toggle(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr); mScroller = new Scroller(context, new LinearInterpolator()); // 获取两张图片
TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.toggle, defStyleAttr, 0);
mOpenDrawable = ta.getDrawable(R.styleable.toggle_drawable_open);
mCloseDrawable = ta.getDrawable(R.styleable.toggle_drawable_close);
ta.recycle();
} /**
* 状态改变时,保存一下isOpen变量
* 比如 屏幕旋转,防止在旋转后恢复原样了
*/
@Override
protected Parcelable onSaveInstanceState() {
Bundle bundle = new Bundle();
bundle.putBoolean("open", isOpen);
bundle.putParcelable("state", super.onSaveInstanceState());
return bundle;
} /**
* 获取保存的isOpen
*/
@Override
protected void onRestoreInstanceState(Parcelable state) {
if(state instanceof Bundle) {
Bundle bundle = (Bundle) state;
isOpen = bundle.getBoolean("open");
state = bundle.getParcelable("state");
} super.onRestoreInstanceState(state);
} /**
* 測量
*/
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
int widthMode = MeasureSpec.getMode(widthMeasureSpec);
int widthSize = MeasureSpec.getSize(widthMeasureSpec); /**
* 获取背景的宽高
*/
Drawable background = getBackground();
// background.getBounds().width()是在背景绘制完毕后才返回值
// 此时返回0
// background.getBounds().width();
// 获取宽度
int width = Math.max(background.getIntrinsicWidth(), 50);
// 获取高度
int height = Math.max(background.getIntrinsicHeight(), 20); // 父布局给指定了大小
if(widthMode == MeasureSpec.EXACTLY) {
width = widthSize;
}else if(widthMode == MeasureSpec.AT_MOST) { // 父布局给指定了最大限度
width = Math.min(width, widthSize);
} setMeasuredDimension(width, height);
// 假设“关闭” 则滑块的位置为当前view宽度-关闭图片宽度
if(!isOpen) mNowX = getMeasuredWidth() - mCloseDrawable.getIntrinsicWidth();
} @Override
protected void onDraw(Canvas canvas) {
// 依据isOpen获取当前要绘制的drawable
Drawable drawing = isOpen ? mOpenDrawable : mCloseDrawable;
// clip bounds
drawing.setBounds(mNowX, 0, mNowX + mOpenDrawable.getIntrinsicWidth(), getMeasuredHeight());
// draw on the canvas
drawing.draw(canvas);
} @Override
public void computeScroll() {
if(mScroller.computeScrollOffset()) {
mNowX = mScroller.getCurrX();
if(mScroller.isFinished()) {
isOpen = !isOpen;
if(isOpen) mNowX = 0;
else mNowX = getMeasuredWidth() - mCloseDrawable.getIntrinsicWidth();
} postInvalidate();
}
} public void toggle() {
mScroller.abortAnimation();
// open -> close
if(isOpen) {
mScroller.startScroll(0, 0, getMeasuredWidth() - mOpenDrawable.getIntrinsicWidth() ,
0, mSmoothDuration);
}else {
mScroller.startScroll(getMeasuredWidth() - mCloseDrawable.getIntrinsicWidth(), 0,
mOpenDrawable.getIntrinsicWidth()-getMeasuredWidth(), 0, mSmoothDuration);
} postInvalidate();
} /**
* 设置Scroller的Interpolator
* @param interpolator
*/
public void setInterpolator(Interpolator interpolator) {
mScroller = new Scroller(getContext(), interpolator);
} /**
* 设置动画完毕的时间间隔
* @param duration
*/
public void setSmoothDuration(int duration) {
mSmoothDuration = duration;
} public boolean isOpen() {
return isOpen;
} public void open() {
isOpen = true;
mNowX = 0;
postInvalidate();
} public void close() {
isOpen = false;
mNowX = getMeasuredWidth() - mCloseDrawable.getIntrinsicWidth();
postInvalidate();
}
}

代码量不是非常多,并且非常清晰。以下我们就来分析分析几个方法。


三、代码分析

首先我们在第二个构造方法中。获取了两个Drawable,分别相应了开和关时的图片。
TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.toggle, defStyleAttr, 0);
mOpenDrawable = ta.getDrawable(R.styleable.toggle_drawable_open);
mCloseDrawable = ta.getDrawable(R.styleable.toggle_drawable_close);
ta.recycle();

继续走代码,onSaveInstanceState和onRestoreInstanceState这两个方法中做的工作就是将isopen保存和恢复了。


onMeasure方法中。首先获取背景的高度和宽度,注意这里不能使用Drawable.getBounds().width()。由于这种方法仅仅有在Drawable绘制完成后才会返回值,此前都是返回的0。所以我们使用Drawable.getIntrinsicWidth()来获取Drawable的真实宽度。继续代码,以下的就是一个简单的測量流程了。再来看看最后一行代码
if(!isOpen) mNowX = getMeasuredWidth() - mCloseDrawable.getIntrinsicWidth();

由于关闭状态下,我们的滑动要位于view的右边,所以,在这里推断一下,假设是关闭状态,则初始化mNowX为视图的宽度减去Drawable的宽度,也就是Drawable正好位于视图的右边。



onDraw里主要就是依据状态和位置来绘制滑块了:
// 依据isOpen获取当前要绘制的drawable
Drawable drawing = isOpen ? mOpenDrawable : mCloseDrawable;
// clip bounds
drawing.setBounds(mNowX, 0, mNowX + mOpenDrawable.getIntrinsicWidth(), getMeasuredHeight());
// draw on the canvas
drawing.draw(canvas);

代码非常easy, so, 跳过。


接下来再来看看toggle方法。toggle方法是提供给外部调用的,该方法依据如今的状态来调用Scroller.startScroll()方法。
public void toggle() {
<span style="white-space:pre"> </span>mScroller.abortAnimation();
// open -> close
if(isOpen) {
mScroller.startScroll(0, 0, getMeasuredWidth() - mOpenDrawable.getIntrinsicWidth() ,
0, mSmoothDuration);
}else {
mScroller.startScroll(getMeasuredWidth() - mCloseDrawable.getIntrinsicWidth(), 0,
mOpenDrawable.getIntrinsicWidth()-getMeasuredWidth(), 0, mSmoothDuration);
} postInvalidate();
}

能够看到。在该方法中并没有去改变isOpen的值,那isOpen是在什么时候改变的呢?答案是在Scroller停止的时候,来看看重载的computeScroll()的代码.

@Override
public void computeScroll() {
if(mScroller.computeScrollOffset()) {
mNowX = mScroller.getCurrX();
if(mScroller.isFinished()) {
isOpen = !isOpen;
if(isOpen) mNowX = 0;
else mNowX = getMeasuredWidth() - mCloseDrawable.getIntrinsicWidth();
} postInvalidate();
}
}

能够看到,在if(mScroller.isFinished())中我们改变了isOpen的值,当然,最后别忘了postInvalidate一下,以刷新当前的视图。

到如今为止。Toggle的关键代码已经解说完了。以下我们来看看效果吧:


使用Scroller制作滑块开关ToggleButton的更多相关文章

  1. mui 滑块开关 进度条 以及如何获取值

    <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name ...

  2. WPF 仿IPhone滑块开关 样式 - CheckBox

    原文:WPF 仿IPhone滑块开关 样式 - CheckBox <Style x:Key="CheckRadioFocusVisual"> <Setter Pr ...

  3. 自定义开关ToggleButton

    package com.example.test;import android.os.Bundle;import android.app.Activity;import android.view.Me ...

  4. 使用css3 制作switch开关

    使用css3来实现switch开关的效果: html代码: <!--switch开关--><div class="switch-btn"> <inpu ...

  5. 自定义开关ToggleButton的使用

    [代码]: toggleMe.setOnCheckedChangeListener(new CompoundButton.OnCheckedChangeListener() { @Override p ...

  6. Arduino IDE搭建ESP8266开发环境,文件下载过慢解决方法 | ESP-01制作WiFi开关教程,改造宿舍灯

    1. Arduino IDE配置ESP8266环境 参考:https://www.jianshu.com/p/cb0274d612b5 首先从 Arduino 官网 下载最新版本的 Arduino I ...

  7. android开关控件Switch和ToggleButton

    序:今天项目中用到了开关按钮控件,查阅了一些资料特地写了这篇博客记录下. 1.Switch <Switch android:id="@+id/bt" android:layo ...

  8. 6.Android之switch和togglebutton按钮学习

    Switch和ToggleButton按钮在手机上也经常看到,比如手机设置里面wlan,蓝牙,gps开关等. 首先在工具上拖进一个switch和togglebutton开关按钮,如图 生成xml代码如 ...

  9. 纯css的滑块开关按钮

    之前在项目中使用滑块开关按钮,纯css写的,不考虑兼容低版本浏览器,先说下原理: 使用 checkbox 的 选中 checked 属性改变css 伪类样式, 一定要使用-webkit-appeara ...

随机推荐

  1. js编码处理(转)

    1.       使用 JS 中的 encodeURIComponent 或 encodeURI 方法. 说明: encodeURIComponent(String) 对传递参数进行设置.不编码字符有 ...

  2. React入门介绍(2)- React Component-React组件

    React Component-React组件 允许用户自由封装组件是React非常突出的特性,用户可将自己创建的组件像普通的HTML标签一样插入页面,React.CreateClass方法就是用来创 ...

  3. 题解 洛谷P4198/BZOJ2957【楼房重建】

    每个楼房,还有修改操作.简单的想到用线段树来维护信息. 显然线段树只需要维护y/x即可,对于每一个楼房,能看见的条件就是前面楼房的y/x的严格小于当前楼房的y/x. 线段树的区间修改不再赘述. 那么怎 ...

  4. <Spring Cloud>入门一 Eureka Server

    1.搭建父工程 主要是添加版本依赖,此处版本是: spring-boot  : 2.0.8.RELEASE spring-cloud : Finchley.SR2 <?xml version=& ...

  5. 简述FTP主动模式与被动模式

    1 FTP工作模式 2 不同模式FTP面临的问题 3 主动模式的FTP连接建立连接主要步骤 客户端打开一个随机的端口(端口号大于1024,在这里,我们称它为x),同时一个FTP进程连接至服务器的21号 ...

  6. xtrbackup备份mysql

    mysqldump备份方式是采用逻辑备份,但是它最大的缺陷就是备份和恢复速度慢对于一个小于50G的数据库而言,这个速度还是能接受的,但如果数据库非常大,那再使用mysqldump备份就不太适合了. x ...

  7. 对Spring框架的理解(转)

    ①  spring框架是一个开源而轻量级的框架,是一个IOC和AOP容器 ② spring的核心就是控制反转(IOC)和面向切面编程(AOP) ③  控制反转(IOC):是面向对象编程中的一种设计原则 ...

  8. 使用idea搭建ssh项目

    参考: https://www.cnblogs.com/getchen/p/8036709.html 需要自己提前在数据库中建好表 然后连接数据库通过侧边栏的persistence来生成实体类和相应的 ...

  9. HDU 4473

    题目大意: 给定一个long long 型的数 n,找到一共有多少对a,b,使比n小的某一个数的是a*b的倍数 这样我们可以理解为 存在a*b*c <= n,令 a <= b <= ...

  10. bzoj 3224 NOI2004郁闷的出纳员

    NOI2004郁闷的出纳员 2013年12月26日6,1818 输入描述 Input Description 第一行有两个非负整数n和min.n表示下面有多少条命令,min表示工资下界. 接下来的n行 ...