自定义实现MIUI的拖动视差效果(阻尼效果)

在MIUI上有一些界面在拖动的时候有一个视差效果:



在可以滚动的视图中,内容滚动到顶部时继续下拉,整个视图就有一个竖直方向拉伸的视差效果。滚动到底部继续上拉,也有同样的效果。

滚动视图可能是ScrollViewRecyclerView,要实现这样的效果,需要自定义并拦截Touch事件,重新处理事件逻辑。

RecyclerView为例,我们自定义一个ParallaxRecyclerView,复写onInterceptTouchEvent方法:

@Override
public boolean onInterceptTouchEvent(MotionEvent event) {
    int action = MotionEventCompat.getActionMasked(event);
    if (isRestoring && action == MotionEvent.ACTION_DOWN) {
        isRestoring = false;
    }
    if (!isEnabled() || isRestoring || (!isScrollToTop() && !isScrollToBottom())) {
        return super.onInterceptTouchEvent(event);
    }
    switch (action) {
        case MotionEvent.ACTION_DOWN: {
            mActivePointerId = event.getPointerId(0);
            isBeingDragged = false;
            float initialMotionY = getMotionEventY(event);
            if (initialMotionY == -1) {
                return super.onInterceptTouchEvent(event);
            }
            mInitialMotionY = initialMotionY;
            break;
        }
        case MotionEvent.ACTION_MOVE: {
            if (mActivePointerId == MotionEvent.INVALID_POINTER_ID) {
                return super.onInterceptTouchEvent(event);
            }
            final float y = getMotionEventY(event);
            if (y == -1f) {
                return super.onInterceptTouchEvent(event);
            }
            if (isScrollToTop() && !isScrollToBottom()) {
                // 在顶部不在底部
                float yDiff = y - mInitialMotionY;
                if (yDiff > mTouchSlop && !isBeingDragged) {
                    isBeingDragged = true;
                }
            } else if (!isScrollToTop() && isScrollToBottom()) {
                // 在底部不在顶部
                float yDiff = mInitialMotionY - y;
                if (yDiff > mTouchSlop && !isBeingDragged) {
                    isBeingDragged = true;
                }
            } else if (isScrollToTop() && isScrollToBottom()) {
                // 在底部也在顶部
                float yDiff = y - mInitialMotionY;
                if (Math.abs(yDiff) > mTouchSlop && !isBeingDragged) {
                    isBeingDragged = true;
                }
            } else {
                // 不在底部也不在顶部
                return super.onInterceptTouchEvent(event);
            }
            break;
        }
        case MotionEventCompat.ACTION_POINTER_UP:
            onSecondaryPointerUp(event);
            break;
        case MotionEvent.ACTION_UP:
        case MotionEvent.ACTION_CANCEL:
            mActivePointerId = MotionEvent.INVALID_POINTER_ID;
            isBeingDragged = false;
            break;
    }
    return isBeingDragged || super.onInterceptTouchEvent(event);
}

滚动RecyclerView到达顶部或者底部继续拖动时,需要拦截Touch事件。所以在MotionEvent.ACTION_MOVE时需要判断当前RecyclerView是否在顶部或者底部。需要注意的是,当RecyclerView中的item没有填充满整视图时,RecyclerView的状态既是在顶部也是在底部。

private boolean isScrollToTop() {
    return !ViewCompat.canScrollVertically(this, -1);
}

private boolean isScrollToBottom() {
    return !ViewCompat.canScrollVertically(this, 1);
}

mActivePointerId表示在多点触控是当前活动手指的id,mInitialMotionY为手指按下时的Y坐标。

当达到顶部或底部继续拖动时,根据当前的位置(isScrollToTop()isScrollToBottom())和ACTION_MOVE时的移动距离yDiff来判断是否需要拦截:在顶部时向上拖动并且yDiff>mTouchSlop就需要拦截,底部时向下拖动同样yDiff>mTouchSlop也需要拦截,同时在顶部和底部时满足Math.abs(yDiff)>mTouchSlop也需要拦截。需要拦截都是在没有被拖动(!isBeingDragged)的情况下。

RecyclerViev既没有在顶部也没有在底部时,说明item滚动到中间,可以上下继续滚动,不需要拦截,交给super.onInterceptTouchEvent(event)来处理。同时其它不需要拦截的情况也都交给super来处理。

onSecondaryPointerUp(event)为当第二个手指离开屏幕是需要重新设置mActivePointerId:

private void onSecondaryPointerUp(MotionEvent event) {
    final int pointerIndex = MotionEventCompat.getActionIndex(event);
    final int pointerId = event.getPointerId(pointerIndex);
    if (pointerId == mActivePointerId) {
        int newPointerIndex = pointerIndex == 0 ? 1 : 0;
        mActivePointerId = event.getPointerId(newPointerIndex);
    }
}

拦截到TouchEvent,在onTouchEven中处理,实现拖动视差效果:

@Override
public boolean onTouchEvent(MotionEvent event) {
    switch (MotionEventCompat.getActionMasked(event)) {
        case MotionEvent.ACTION_DOWN:
            mActivePointerId = event.getPointerId(0);
            isBeingDragged = false;
            break;
        case MotionEvent.ACTION_MOVE: { 
            float y = getMotionEventY(event);
            if (isScrollToTop() && !isScrollToBottom()) {
                // 在顶部不在底部
                mDistance = y - mInitialMotionY;
                if (mDistance < 0) {
                    return super.onTouchEvent(event);
                }
                mScale = calculateRate(mDistance);
                pull(mScale);
                return true;
            } else if (!isScrollToTop() && isScrollToBottom()) {
                // 在底部不在顶部
                mDistance = mInitialMotionY - y;
                if (mDistance < 0) {
                    return super.onTouchEvent(event);
                }
                mScale = calculateRate(mDistance);
                push(mScale);
                return true;
            } else if (isScrollToTop() && isScrollToBottom()) {
                // 在底部也在顶部
                mDistance = y - mInitialMotionY;
                if (mDistance > 0) {
                    mScale = calculateRate(mDistance);
                    pull(mScale);
                } else {
                    mScale = calculateRate(-mDistance);
                    push(mScale);
                }
                return true;
            } else {
                // 不在底部也不在顶部
                return super.onTouchEvent(event);
            }
        }
        case MotionEventCompat.ACTION_POINTER_DOWN:
            mActivePointerId = event.getPointerId(MotionEventCompat.getActionIndex(event));
            break;
        case MotionEventCompat.ACTION_POINTER_UP:
            onSecondaryPointerUp(event);
            break;
        case MotionEvent.ACTION_UP: 
        case MotionEvent.ACTION_CANCEL: {
            if (isScrollToTop() && !isScrollToBottom()) {
                animateRestore(true);
            } else if (!isScrollToTop() && isScrollToBottom()) {
                animateRestore(false);
            } else if (isScrollToTop() && isScrollToBottom()) {
                if (mDistance > 0) {
                    animateRestore(true);
                } else {
                    animateRestore(false);
                }
            } else {
                return super.onTouchEvent(event);
            }
            break;
        }
    }
    return super.onTouchEvent(event);
}

代码虽然有点长,但是逻辑很简单,在拦截到ACTION_MOVE事件后,同样根据顶部或底部位置以及滚动的距离mDistance来确定是否消费掉该事件。不需要消费的直接给``super.onTouchEvent(event)来处理,需要消费的话根据mDistance来计算出缩放的比例mScale,再通过pull(mScale)push(mScale)`来缩放。

private float calculateRate(float distance) {
    int screenHeight = getResources().getDisplayMetrics().heightPixels;
    float originalDragPercent = distance / screenHeight;
    float dragPercent = Math.min(1f, originalDragPercent);
    float rate = 2f * dragPercent - (float) Math.pow(dragPercent, 2f);
    return 1 + rate / 5f;
}

mScale的计算是一个二次函数,当拖动距离越大时,mScale的变化程度越小,这样使得拖动时有一个张力效果。

private void pull(float scale) {
    this.setPivotY(0);
    this.setScaleY(scale);
}

private void push(float scale) {
    this.setPivotY(this.getHeight());
    this.setScaleY(scale);
}

ACTION_UP时,需要将缩放的视图通过动画还原到初始状态。这里也需要判断位置,因为不同位置的的缩放中心点不一样。同时即在顶部也在底部时是根mDistance的正负值来判断拖动的方向。

private void animateRestore(final boolean isPullRestore) {
    ValueAnimator animator = ValueAnimator.ofFloat(mScale, 1f);
    animator.setDuration(300);
    animator.setInterpolator(new DecelerateInterpolator(2f));
    animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
        @Override
        public void onAnimationUpdate(ValueAnimator animation) {
            float value = (float) animation.getAnimatedValue();
            if (isPullRestore) {
                pull(value);
            } else {
                push(value);
            }
        }
    });
    animator.addListener(new Animator.AnimatorListener() {
        @Override
        public void onAnimationStart(Animator animation) {
            isRestoring = true;
        }

        @Override
        public void onAnimationEnd(Animator animation) {
            isRestoring = false;
        }

        @Override
        public void onAnimationCancel(Animator animation) {

        }

        @Override
        public void onAnimationRepeat(Animator animation) {

        }
    });
    animator.start();
}

这样就OK了,如果需要实现ScrollViewListViewGridView也是一样的逻辑,源码中已经有了ParallaxScrollView的实现,看下最终效果图:

ParallaxRecyclerView
ParallaxScrollView

源码:https://github.com/xiaoyanger0825/Parallax

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,172评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,346评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,788评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,299评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,409评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,467评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,476评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,262评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,699评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,994评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,167评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,499评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,149评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,387评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,028评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,055评论 2 352

推荐阅读更多精彩内容