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();
}
}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();
if(!isOpen) mNowX = getMeasuredWidth() - mCloseDrawable.getIntrinsicWidth();由于关闭状态下,我们的滑动要位于view的右边,所以,在这里推断一下,假设是关闭状态,则初始化mNowX为视图的宽度减去Drawable的宽度,也就是Drawable正好位于视图的右边。
// 依据isOpen获取当前要绘制的drawable Drawable drawing = isOpen ? mOpenDrawable : mCloseDrawable; // clip bounds drawing.setBounds(mNowX, 0, mNowX + mOpenDrawable.getIntrinsicWidth(), getMeasuredHeight()); // draw on the canvas drawing.draw(canvas);
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();
}@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();
}
}原文:http://www.cnblogs.com/yxysuanfa/p/7147366.html