Android自定义View弹性滑动Scroller详解

 更新时间:2017年12月25日 11:11:45   作者:_爬虫  
这篇文章主要为大家详细介绍了Android自定义View弹性滑动Scroller,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了Android弹性滑动类Scroller的具体代码,供大家参考,具体内容如下

Scroller是什么

Scroller就是一个滑动帮助类。它并不可以使View真正的滑动,而是配合scrollTo/ScrollBy让view产生缓慢的滑动,产生动画的效果,其实和属性动画是同一个原理。在我看来,Scroller跟属性动画的平移的效果是一样的。

如何使用

 //①实例一个Scroller,它有三个构造方法如下
  //public Scroller (Context context)
  //public Scroller (Context context, Interpolator interpolator)//传入一个时间插值器
  //public Scroller (Context context, Interpolator interpolator, boolean flywheel)
  Scroller mScroller=new Scroller(context);

  //②使用Scroller
  //startScroll()传入一些参数:开始位置,结束位置,开始时间滑动到结束位置的完成时间。
  mScrooler.startScroll(int startX,int startY,int endx,int endY,int duration);
  invalidate();//在ViewGroup中,invalidate()方法会导致computeScroll()方法的执行

  //③在computeScroll()的方法中判断:mScroller是否结束,如果没有结束就调用scrollTo()让view处于正确的位置
  @Override
  public void computeScroll() {
   //computeScrollOffset()判断是否还在滚动,如果还在滚动,会获取到某一时刻view应该所在的位置,刷新Scroller中mCurrX,mCurrY的值,并且return true;
   if (mScroller.computeScrollOffset()) {
    scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
    //更新界面
    postInvalidate();
   }
   super.computeScroll();
  }

使用示例:

package com.liujian.chart;

/**
 * Scroller练习,一个简单的ViewPager
 * @author : liujian
 * @since : 2017/12/17
 */

public class ScrollLayout extends ViewGroup {
 private Scroller mScroller;
 //当前设备滑动的最小距离
 private int mTouchSlop;

 private int leftBorder;//布局内容的左边界
 private int rightBorder;//布局内容的右边界

 private float mRawXDown;
 private float mRawXMove;
 private float mRawXLastMove;

 public ScrollLayout(Context context) {
  super(context);
  initView(context);
 }

 public ScrollLayout(Context context, @Nullable AttributeSet attrs) {
  super(context, attrs);
  initView(context);
 }

 public ScrollLayout(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
  super(context, attrs, defStyleAttr);
  initView(context);
 }

 private void initView(Context context) {
  mTouchSlop = ViewConfiguration.get(getContext()).getScaledTouchSlop();
  mScroller = new Scroller(getContext());
 }

 @Override
 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
  super.onMeasure(widthMeasureSpec, heightMeasureSpec);
  //为ScrollLayout中的某一个子View给出一个建议的测量大小和测量模式
  measureChildren(widthMeasureSpec, heightMeasureSpec);
 }

 @Override
 protected void onLayout(boolean changed, int l, int t, int r, int b) {
  int childCount = getChildCount();
  for (int i = 0; i < childCount; i++) {
   View view = getChildAt(i);
   view.layout(i * view.getMeasuredWidth(), 0, (i + 1) * view.getMeasuredWidth(), view.getMeasuredHeight());
  }
  leftBorder = getChildAt(0).getLeft();
  rightBorder = getChildAt(getChildCount() - 1).getRight();
 }

 @Override
 public boolean onInterceptTouchEvent(MotionEvent ev) {
  switch (ev.getAction()) {
   case MotionEvent.ACTION_DOWN:
    mRawXDown = ev.getRawX();
    mRawXLastMove = mRawXDown;
    break;
   case MotionEvent.ACTION_MOVE:
    mRawXMove = ev.getRawX();
    mRawXLastMove = mRawXMove;
    float distance = Math.abs(mRawXMove - mRawXDown);
    //左右滑动时,拦截子view的触摸事件
    if (distance > mTouchSlop) {
     return true;
    }
    break;
   case MotionEvent.ACTION_UP:
    break;
  }
  return super.onInterceptTouchEvent(ev);
 }

 @SuppressLint("ClickableViewAccessibility")
 @Override
 public boolean onTouchEvent(MotionEvent event) {
  switch (event.getAction()) {
   case MotionEvent.ACTION_MOVE:
    mRawXMove = event.getRawX();
    int distanceX = (int) (mRawXLastMove - mRawXMove);
    //对边界异常情况的处理
    if (getScrollX() + distanceX < leftBorder) {
     scrollBy(leftBorder, 0);
    }
    if (getScrollX() + getWidth() + distanceX > rightBorder) {
     scrollBy(rightBorder - getWidth(), 0);
    }
    scrollBy(distanceX, 0);
    mRawXLastMove = mRawXMove;
    break;
   case MotionEvent.ACTION_UP:
    //当前所在的page页面
    int targetIndex = (getScrollX() + getWidth() / 2) / getWidth();
    int dx = targetIndex * getWidth() - getScrollX();
    Log.i("TAG", "dx: " + dx);
    Log.i("TAG", "getScrollX: " + getScrollX());
    Log.i("TAG", "getWidth: " + getWidth());

    // 第二步,调用startScroll()方法来初始化滚动数据并刷新界面
    mScroller.startScroll(getScrollX(), 0, dx, 0);
    invalidate();
    break;
  }
  return super.onTouchEvent(event);
 }

 @Override
 public void computeScroll() {
  if (mScroller.computeScrollOffset()) {
   scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
   invalidate();
  }
 }
}

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

相关文章

  • Android实现多级树形选择列表

    Android实现多级树形选择列表

    这篇文章主要为大家详细介绍了Android实现多级树形选择列表,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-04-04
  • Android导航栏功能项的显示与屏蔽介绍

    Android导航栏功能项的显示与屏蔽介绍

    大家好,本篇文章主要讲的是Android导航栏功能项的显示与屏蔽介绍,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览
    2021-12-12
  • Android编程实现自定义输入法功能示例【输入密码时防止第三方窃取】

    Android编程实现自定义输入法功能示例【输入密码时防止第三方窃取】

    这篇文章主要介绍了Android编程实现自定义输入法功能,可实习输入密码时防止第三方窃取的效果,结合实例形式详细分析了Android布局、控件及输入法相关操作技巧,需要的朋友可以参考下
    2017-01-01
  • android电话模拟器(示例代码)

    android电话模拟器(示例代码)

    本篇文章我将为大家介绍一下android电话模拟器(示例代码),需要的朋友可以过来参考下,希望对大家有所帮助
    2013-11-11
  • Android主项目与Module中R类的区别详解

    Android主项目与Module中R类的区别详解

    这篇文章主要给大家介绍了关于Android主项目与Module中R类的区别的相关资料,文中通过示例代码介绍的非常详细,对各位Android开发者们具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧。
    2018-02-02
  • Android编程实现仿心跳动画效果的方法

    Android编程实现仿心跳动画效果的方法

    这篇文章主要介绍了Android编程实现仿心跳动画效果的方法,实例分析了Android基于线程实现动画过度效果的相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-11-11
  • android onTouchEvent处理机制总结(必看)

    android onTouchEvent处理机制总结(必看)

    下面小编就为大家带来一篇android onTouchEvent处理机制总结(必看)小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-04-04
  • Android ActionBarActivity设置全屏无标题的方法总结

    Android ActionBarActivity设置全屏无标题的方法总结

    这篇文章主要介绍了Android ActionBarActivity设置全屏无标题的相关资料,需要的朋友可以参考下
    2017-07-07
  • android 实现侧边弹窗特效代码

    android 实现侧边弹窗特效代码

    侧边弹窗是在左边,需要定位好位置,实现原理其实就是进出动效,用位移加透明度效果来控制,下面通过代码给大家介绍android 实现侧边弹窗,需要的朋友参考下吧
    2021-06-06
  • Android画图实现MPAndroidchart折线图示例详解

    Android画图实现MPAndroidchart折线图示例详解

    这篇文章主要为大家介绍了Android画图实现MPAndroidchart折线图示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-07-07

最新评论