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

作者:_爬虫 时间:2023-08-07 09:56:39 

本文实例为大家分享了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();
 }
}
}

来源:http://blog.csdn.net/qq_31694651/article/details/78884239

标签:Android,弹性滑动,Scroller
0
投稿

猜你喜欢

  • Flutter runApp到渲染上屏分析详解

    2023-06-27 12:09:45
  • SpringBoot Security前后端分离登录验证的实现

    2023-03-09 10:30:07
  • C语言编程C++动态内存分配示例讲解

    2023-11-02 18:00:12
  • Android 沉浸式改变小米魅族状态栏颜色的实例代码

    2023-11-18 15:13:16
  • C#实现窗体抖动的两种方法

    2021-10-06 10:20:52
  • java中重载,继承,重写和多态的区别

    2022-05-06 06:40:16
  • java控制台实现学生信息管理系统

    2023-11-29 02:05:20
  • Mybatis实现SQL存储流程详解

    2022-09-11 05:17:05
  • JAVA通过HttpClient发送HTTP请求的方法示例

    2023-08-24 18:45:47
  • c# 给pdf添加数字签名的步骤

    2022-04-05 17:58:01
  • Java编程泛型限定代码分享

    2023-11-09 17:46:32
  • 自定义时间格式转换代码分享

    2022-11-03 03:09:23
  • C#中的匿名函数、lambda表达式解读

    2023-08-28 07:07:39
  • Java读文件修改默认换行符的实现

    2023-11-29 08:24:32
  • springboot对接微信支付的完整流程(附前后端代码)

    2021-11-12 15:08:42
  • 使用maven创建web项目的方法步骤(图文)

    2022-12-08 04:36:41
  • Android本地验证码的生成代码

    2023-05-03 18:44:02
  • Spring容器初始化及问题解决方案

    2023-09-14 08:19:32
  • 最值得Java开发者收藏的网站

    2022-03-09 15:57:08
  • Android自定义控件案例汇总1(菜单、popupwindow、viewpager)

    2021-11-22 14:37:32
  • asp之家 软件编程 m.aspxhome.com