Android自定义控件仿ios下拉回弹效果

作者:十案圈圈 时间:2021-07-17 13:17:06 

网上有很多类似的文章,大多数还是继承listview来实现(主要是listview.addHeaderView()和listview.addFooterView在listview的首尾添加view,也可以用上面的两个listview自带函数实现下拉刷新的功能,在这里不准备介绍,有兴趣的朋友可以去自己试试)。

在本文主要是给android的线性布局(相对布局、帧布局)加上下拉或者上拉回弹得效果。在ios中我们经常能看到,在一个页面中即使是只有一个控件,这一个控件只占整个页面的1/10不到,但是当我们下拉整个页面的时候还是会有回弹的效果(在这里我们暂不考虑这样的页面是否美观,只是就怎么实现进行分析),显然在android中我们不会为了实现这个只有一个item(而且不会变多)的页面而去用listview(listview的使用还是相对比较繁琐),我们会直接使用线性布局或者相对布局这些简易一些的viewgroup来实现。所以在这里我也为线性布局加上了下拉或者上拉回弹得效果。

实现流程:

1.新建一个类继承LinearLayout

2.在构造方法中实例化Scroller(用于滑动),GestureDetector(网上有很多实现方法是复写onTouchEvent方法,把onTouchEvent方法写的很长,我不太喜欢这种方式,也推荐大家多用手势,很好用哦);

3.覆写computeScroll(),onTouchEvent(MotionEvent event)(在这里把触摸屏幕的处理交给GestureDetector)
4.在computeScroll()里面完成实际的滚动

在开始具体的实现之前,先得介绍几个要用到的比较重要的函数


mScroller.getCurrX() //获取mScroller当前水平滚动的位置
mScroller.getCurrY() //获取mScroller当前竖直滚动的位置
mScroller.getFinalX() //获取mScroller最终停止的水平位置
mScroller.getFinalY() //获取mScroller最终停止的竖直位置
mScroller.setFinalX(int newX) //设置mScroller最终停留的水平位置,没有动画效果,直接跳到目标位置
mScroller.setFinalY(int newY) //设置mScroller最终停留的竖直位置,没有动画效果,直接跳到目标位置

//滚动,startX, startY为开始滚动的位置,dx,dy为滚动的偏移量, duration为完成滚动的时间
mScroller.startScroll(int startX, int startY, int dx, int dy) //使用默认完成时间250ms
mScroller.startScroll(int startX, int startY, int dx, int dy, int duration)

mScroller.computeScrollOffset() //返回值为boolean,true说明滚动尚未完成,false说明滚动已经完成。这是一个很重要的方法,通常放在View.computeScroll()中,用来判断是否滚动是否结束。

上面的几个Scroller的方法,能够帮助我们实现滑动。

接下来还要介绍实现GestureDetector.OnGestureListener

因为我们在onTouchEvent中没有将MotionEvent.ACTION_UP交给GestureDetector,所以GestureDetector.OnGestureListener中的部分方法不会响应,还有在GestureDetector.OnGestureListener中要将down事件的返回值设为true,不然onscroll方法不会响应

接下来是具体实现:


public class SqqLinearLayout extends LinearLayout {
private Scroller mScroller;
private GestureDetector mGestureDetector;

public SqqLinearLayout (Context context) {
 this(context, null);
}

public SqqLinearLayout (Context context, AttributeSet attrs) {
 super(context, attrs);
 mScroller = new Scroller(context);
 mGestureDetector = new GestureDetector(context, new GestureListenerImpl());
}

//startScroll之后没有真正移动,会自动调用这个函数实现移动
@Override
public void computeScroll() {
 if (mScroller.computeScrollOffset()) {

scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
  //必须执行postInvalidate()从而调用computeScroll()
  //其实,在此调用invalidate();亦可
  postInvalidate();
 }
 super.computeScroll();
}

@Override
public boolean onTouchEvent(MotionEvent event) {
 switch (event.getAction()) {
 case MotionEvent.ACTION_UP :
 //手指抬起时回到最初位置
  prepareScroll(0, 0);
  break;
 default:
  //其余情况交给GestureDetector手势处理
  return mGestureDetector.onTouchEvent(event);
 }
 return super.onTouchEvent(event);
}

class GestureListenerImpl implements GestureDetector.OnGestureListener {
@Override
public boolean onDown(MotionEvent e) {
 return true;
}

@Override
public void onShowPress(MotionEvent e) {

}

@Override
public boolean onSingleTapUp(MotionEvent e) {
 return false;
}

@Override
public boolean onScroll(MotionEvent e1, MotionEvent e2,float distanceX, float distanceY) {
 int disY = (int) ((distanceY - 0.5)*0.65);
 beginScroll(0, disY);
 return false;
}

public void onLongPress(MotionEvent e) {

}

@Override
public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX,float velocityY) {
 return false;
}

}

//滚动到目标位置
protected void prepareScroll(int fx, int fy) {
 int dx = fx - mScroller.getFinalX();
 int dy = fy - mScroller.getFinalY();
 beginScroll(dx, dy,1000); //经测试1s是不错的
}

//设置滚动的相对偏移
protected void beginScroll(int dx, int dy) {
 mScroller.startScroll(mScroller.getFinalX(), mScroller.getFinalY(), dx, dy);

//必须执行invalidate()从而调用computeScroll()
 //invalidate();
//上面一句注释掉好像也没什么影响,暂时没有发现
}

//设置滚动的相对偏移
protected void beginScroll(int dx, int dy,int duration) {
 mScroller.startScroll(mScroller.getFinalX(), mScroller.getFinalY(), dx, dy,duration);

//必须执行invalidate()从而调用computeScroll()
 //invalidate();
//上面一句注释掉好像也没什么影响,暂时没有发现
}

}

上面实现了线性布局的下拉回弹效果,相对布局的实现和上面一样,只是继承的是RelativeLayout。所以抱着不写重复代码的准则,在下一篇我会做个优化,将线性布局和相对布局的下拉刷新写到一个类中,具体的线性布局和相对布局作为参数或者其他的形式。当然这还只是个想法,不知道能不能很好的实现。

项目下载地址:Android自定义控件仿ios下拉回弹效果

来源:https://blog.csdn.net/u012806692/article/details/50618239

标签:Android,ios,下拉回弹
0
投稿

猜你喜欢

  • Spring中的事务隔离级别和传播行为

    2022-07-21 18:18:24
  • Java数据结构之链表(动力节点之Java学院整理)

    2022-08-29 16:40:18
  • Android自定义View实现五子棋游戏

    2021-10-16 23:39:27
  • Unity查找游戏物体的六种方式详解

    2021-07-09 18:23:17
  • Java 栈和队列的相互转换详解

    2021-09-22 05:00:12
  • java 微信随机红包算法代码实例

    2022-12-03 12:29:08
  • 为SpringBoot服务添加HTTPS证书的方法

    2023-10-11 03:03:22
  • C#中WPF颜色对话框控件的实现

    2023-04-13 06:09:41
  • SSH框架网上商城项目第2战之基本增删查改、Service和Action的抽取

    2023-06-21 19:16:23
  • c#中oracle的to_date函数使用方法

    2021-09-06 10:21:17
  • Java使用Optional实现优雅避免空指针异常

    2023-06-05 15:35:39
  • C# 文字代码页 文字编码的代码页名称速查表

    2023-12-13 04:03:54
  • java多线程通过CompletableFuture组装异步计算单元

    2023-07-19 10:15:42
  • Java中的拦截器、过滤器、监听器用法详解

    2021-08-08 16:26:03
  • 用JAVA实现杨辉三角实例

    2023-08-28 16:45:23
  • Ubuntu 14.04下创建Genymotion安卓虚拟机的步骤详解

    2023-11-04 19:17:54
  • Android RecyclerView选择多个item的实现代码

    2022-06-09 10:56:56
  • C语言运算符优先级列表(超详细)

    2023-07-04 08:17:54
  • Android 游戏开发入门简单示例

    2023-05-02 07:29:56
  • Android Studio实现简易计算器设计

    2022-08-22 18:41:55
  • asp之家 软件编程 m.aspxhome.com