这个控件的实现思路 类似于 PullToReresh
用法也很简单,将我们写好的自定义控件 作为父控件 即可 使用方式 参考PullToReresh
强调 出处 感谢该作者 – 小瓶盖_tl – 大神多谢啦
将 onTach () 方法的参数 介绍一下
常用事件 | 功能作用 |
---|---|
ACTION_DOWN | 表示按下了屏幕,第一个执行也是必然执行的方法。 |
ACTION_MOVE | 表示为移动手势,会不断的执行直到触摸停止。 |
ACTION_UP | 表示为离开屏幕,触摸停止的时候执行。 |
ACTION_CANCEL | 表示取消手势,不会由用户产生,而是由程序产生的。 |
package ash.com.myapplication;
import android.content.Context;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewConfiguration;
import android.view.ViewGroup;
import android.widget.AbsListView;
import android.widget.LinearLayout;
import android.widget.ScrollView;
import android.widget.Scroller;
/*
*▶ 作者: ash ◀
* 2019/1/19 20:48 01
*(◕ᴗ◕✿)
*
*页面功能-1: 【实现滑动功能】 ✔
*
*/
public class SildingFinishLayout extends LinearLayout implements View.OnTouchListener {
private ViewGroup mParentView; //SildingFinishLayout布局的父布局
private View touchView; //处理滑动逻辑的View
private int mTouchSlop; //滑动的最小距离
private int downX; //按下点的X坐标
private int downY; //按下点的Y坐标
private int tempX; //临时存储X坐标
private Scroller mScroller; //滑动类
private int viewWidth; //SildingFinishLayout的宽度
private boolean isSilding; //记录是否正在滑动
private OnSildingFinishListener onSildingFinishListener; //自定义滑动监听
private boolean isFinish;
public SildingFinishLayout(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
public SildingFinishLayout(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
mTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
mScroller = new Scroller(context);
}
@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
super.onLayout(changed, l, t, r, b);
if (changed) {
// 获取SildingFinishLayout所在布局的父布局
mParentView = (ViewGroup) this.getParent();
viewWidth = this.getWidth();
}
}
/*
* 设置OnSildingFinishListener, 在onSildingFinish()方法中finish Activity
* @param onSildingFinishListener
*/
public void setOnSildingFinishListener(
OnSildingFinishListener onSildingFinishListener) {
this.onSildingFinishListener = onSildingFinishListener;
}
/*
* 设置Touch的View
* @param touchView
*/
public void setTouchView(View touchView) {
this.touchView = touchView;
touchView.setOnTouchListener(this);
}
public View getTouchView() {
return touchView;
}
/*
* 滚动出界面
*/
private void scrollRight() {
final int delta = (viewWidth + mParentView.getScrollX());
// 调用startScroll方法来设置一些滚动的参数,我们在computeScroll()方法中调用scrollTo来滚动item
mScroller.startScroll(mParentView.getScrollX(), 0, -delta + 1, 0,
Math.abs(delta));
postInvalidate();
}
/*
* 滚动到起始位置
*/
private void scrollOrigin() {
int delta = mParentView.getScrollX();
mScroller.startScroll(mParentView.getScrollX(), 0, -delta, 0,
Math.abs(delta));
postInvalidate();
}
/*
* touch的View是否是AbsListView, 例如ListView, GridView等其子类
* @return
*/
private boolean isTouchOnAbsListView() {
return touchView instanceof AbsListView ? true : false;
}
/*
* touch的view是否是ScrollView或者其子类
* @return
*/
private boolean isTouchOnScrollView() {
return touchView instanceof ScrollView ? true : false;
}
//该方法是 由 View.OnTouchListener 接口重写而来
@Override
public boolean onTouch(View v, MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
downX = tempX = (int) event.getRawX(); // getRawX : 获取点击事件距离整个屏幕左边的距离,即绝对坐标
downY = (int) event.getRawY(); // getRawY : 获取点击事件距离整个屏幕顶边的距离,即绝对坐标
break;
case MotionEvent.ACTION_MOVE:
int moveX = (int) event.getRawX();
int deltaX = tempX - moveX;
tempX = moveX;
if (Math.abs(moveX - downX) > mTouchSlop
&& Math.abs((int) event.getRawY() - downY) < mTouchSlop) {
isSilding = true;
// 若touchView是AbsListView,
// 则当手指滑动,取消item的点击事件,不然我们滑动也伴随着item点击事件的发生
if (isTouchOnAbsListView()) {
MotionEvent cancelEvent = MotionEvent.obtain(event);
cancelEvent
.setAction(MotionEvent.ACTION_CANCEL
| (event.getActionIndex() << MotionEvent.ACTION_POINTER_INDEX_SHIFT));
v.onTouchEvent(cancelEvent);
}
}
if (moveX - downX >= 0 && isSilding) {
mParentView.scrollBy(deltaX, 0);
// 屏蔽在滑动过程中ListView ScrollView等自己的滑动事件
if (isTouchOnScrollView() || isTouchOnAbsListView()) {
return true;
}
}
break;
case MotionEvent.ACTION_UP:
isSilding = false;
if (mParentView.getScrollX() <= -viewWidth / 2) {
isFinish = true;
scrollRight();
} else {
scrollOrigin();
isFinish = false;
}
break;
}
// 假如touch的view是AbsListView或者ScrollView 我们处理完上面自己的逻辑之后
// 再交给AbsListView, ScrollView自己处理其自己的逻辑
if (isTouchOnScrollView() || isTouchOnAbsListView()) {
return v.onTouchEvent(event);
}
// 注意! 这里如果直接返回true 会将 子控件的点击事件拦截 (原作者出处此处返回值为true)
return super.onTouchEvent(event);
}
//通过重新绘制 不断调用computeScrolls
@Override
public void computeScroll() {
// 调用startScroll的时候scroller.computeScrollOffset()返回true,
if (mScroller.computeScrollOffset()) {
//getCurrX 获取当前滑动的坐标
//getCurrY 获取当前滑动的坐标
mParentView.scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
postInvalidate();
if (mScroller.isFinished()) {
//再次调用接口回调方法
if (onSildingFinishListener != null && isFinish) {
onSildingFinishListener.onSildingFinish();
}
}
}
}
//接口
public interface OnSildingFinishListener {
void onSildingFinish();
}
}
布局文件
Activity中的用法 参考上边的 布局文件的ID等信息
//调用这个监听
mSildingFinishLayout.setOnSildingFinishListener(new SildingFinishLayout.OnSildingFinishListener() {
@Override
public void onSildingFinish() {
MainActivity.this.finish();
}
});
mSildingFinishLayout.setTouchView(mBtn);