博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
使用MotionLayout实现高德地图bottomSheets效果
阅读量:6159 次
发布时间:2019-06-21

本文共 6358 字,大约阅读时间需要 21 分钟。

高德效果

  • 搜到的一个效果,附上链接,用的behavior,我没下源码看,因为我只是想尝试另外一种方式。具体效果暂不知

以下是我用motionlayout实现的效果,没有达到丝滑流畅,优化就看小伙伴你了

缘由

  • 使用高德地图的时候看着这种体验很好,随后就想试试怎么达到类似效果
  • 最近正在看MotionLayout的东西,正好就尝试尝试

MotionLayout

  • 系列教会你如何使用MotionLayout
  • 这里不做过多描述,总结一下在xml文件夹下创建xxscene.xml 主要用于描述场景动画的关键帧和view状态变化等
  • xxscene.xml内容包括 主要为3个关键内容:
  1. Transition 过渡

constraintSetStart:启动约束场景

constraintSetEnd:结束约束场景

app:dragDirection="dragUp" 拽动(拖拉)

  1. KeyFrameSet关键帧集合

KeyAttribute关键帧

app:framePosition 位置,进度

app:target="@id/xxx 被描述的view id

  1. ConstraintSet 约束集合
...
...
复制代码

拆解过程

  • 高德地图是上拉之后是三段式的,如图所示

  • MotionLayout就只有一个初始约束和结束约束,没有中间约束,如何实现这种三段式效果?
  • 答: 使用progress, MotionLayout自带进度
  • 有进度,什么时候执行下一步操作,什么时候又执行上一步操作?
  • 答: 根据手势,我们可以判断用户下一步是往上拉还是下拉,设定阶段阀值,超过进入下一步,未超过回到之前一步
  • 高德地图中,只有手触碰到bottomview的时候手势才有效果,所以还需要判断touch事件是否在view范围内

拆解完毕

实现过程

  • 设置阀值
/**     * 初始位置     */    public final static float PROGRESS_START = 0f;    /**     * 顶部阀值      */    public final static float PROGRESS_TOP = 0.9f;    /**     * 低部阀值      */    public final static float PROGRESS_BOTTOM = 0.1f;    /**     * 中间位置      */    public final static float PROGRESS_MIDDLE = 0.6f;    /**     * 结束位置      */    public final static float PROGRESS_END = 1.0f;复制代码
  • 重写MotionLayoutonTouchEvent事件 ,使用hasMiddle布尔值判断是否有中间状态
@Override    public boolean onTouchEvent(MotionEvent event) {        float progress = getProgress();        View viewGroup = findViewById(R.id.content);        Rect mRect = new Rect();        if (!mTouchStared) {            viewGroup.getHitRect(mRect);            mTouchStared = mRect.contains((int) event.getX(), (int) event.getY());        }        float endY;        if (hasMiddle) {            switch (event.getActionMasked()) {                case MotionEvent.ACTION_CANCEL:                    mTouchStared = false;                    break;                case MotionEvent.ACTION_DOWN:                    startY = event.getY();                    break;                case MotionEvent.ACTION_UP:                    endY = event.getY();                    //手势向下                    if ((endY - startY) > 0) {                        if (progress >= PROGRESS_TOP) {                            mTouchStared = false;                            handleProgress(PROGRESS_END);                        }                        if (progress < PROGRESS_TOP && progress >= PROGRESS_MIDDLE) {                            handleProgress(PROGRESS_MIDDLE);                        }                        if (progress < PROGRESS_MIDDLE) {                            handleProgress(PROGRESS_START);                        }                        //手势向上                    } else {                        if (progress <= PROGRESS_BOTTOM) {                            handleProgress(PROGRESS_START);                        }                        if (progress > PROGRESS_BOTTOM && progress <= PROGRESS_MIDDLE) {                            handleProgress(PROGRESS_MIDDLE);                        }                        if (progress > PROGRESS_MIDDLE) {                            mTouchStared = false;                            handleProgress(PROGRESS_END);                        }                    }                    return mTouchStared;            }        } else {            if (event.getActionMasked() == MotionEvent.ACTION_CANCEL || event.getActionMasked() == MotionEvent.ACTION_UP) {                mTouchStared = false;                return super.onTouchEvent(event);            }        }        return mTouchStared && super.onTouchEvent(event);    }复制代码
  • bottom_scene.xml
  1. 上拉超过顶部阀值PROGRESS_TOP之后标题出现在屏幕内,其余时候出现在屏幕外即可;
  2. 初始状态这里把scaleXscaleY设为0.9结束设为了1,仅仅是为了过渡好看,你可以不用设置随意修改即可
  3. 背景色过渡,最开始透明,结束为白色背景。中间过渡关键帧95变纯白背景

结果和改进

  • 设置允许中间状态后,之后进入下一步的过程,如图,过于生硬

  • 改进方向
  1. 动画应该是匀速的,然而setProgress(pro);却是一步直达;
  2. 设置时间间隔匀速达到最后的进度即可,源码已详细注释。改进之后见最上面效果图;
private void handleProgress(float progress) {        //如果需要设置的进度和当前进度相同不做处理        if (progress == getProgress()){            return;        }        //动画播放时间底值        long time = 200;        //进度间隔 >0 说明上拉 < 0说明下滑        float interval = progress - getProgress();        long startTime, endTime;        if (interval > 0) {            startTime = (long) (getProgress() * time);            endTime = (long) (progress * time);        } else {            endTime = (long) (getProgress() * time);            startTime = (long) (progress * time);        }        if (timeDisposable != null){            timeDisposable.dispose();        }        //startTime 初始时间 endTime - startTime为次数 0为延迟时间 3为间隔 单位TimeUnit.MILLISECONDS 毫秒        timeDisposable = Observable.intervalRange(startTime, endTime - startTime, 0, 3, TimeUnit.MILLISECONDS)                .observeOn(Schedulers.io())                .compose(((BaseActivity) getContext()).getProvider().bindToLifecycle())                .observeOn(AndroidSchedulers.mainThread())                .map(new Function
() { @Override public Long apply(Long aLong) throws Exception { //下滑需要反向 if (interval < 0) { long interStart = aLong - startTime; return endTime - interStart; } return aLong; } }) .subscribe(new Consumer
() { @Override public void accept(Long aLong) throws Exception { float pro = (Float.valueOf(aLong) / time); setProgress(pro); } }); }复制代码

源码已放入sugar demo中,sugar是我会长期维护的一个库⬇️⬇️⬇️

About me

  • Email:
  • 掘金:
  • 简书:
  • apkbus:
  • github:

License

Copyright 2019, wobiancao            Licensed under the Apache License, Version 2.0 (the "License");   you may not use this file except in compliance with the License.   You may obtain a copy of the License at         http://www.apache.org/licenses/LICENSE-2.0    Unless required by applicable law or agreed to in writing, software   distributed under the License is distributed on an "AS IS" BASIS,   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.   See the License for the specific language governing permissions and   limitations under the License.   复制代码

转载于:https://juejin.im/post/5cf634405188253d271539be

你可能感兴趣的文章
[译]Hour 7 Teach.Yourself.WPF.in.24.Hours
查看>>
C++ constexpr
查看>>
mac linux netstat -n
查看>>
The Basics
查看>>
Linux/U-Boot Git Repo
查看>>
python了解
查看>>
在写HTML和CSS时的黄金规范
查看>>
【php】用filter_var实现的简单参数验证
查看>>
springMVC解析视图
查看>>
CSS之字体相关样式
查看>>
PHP两个字符串比较(人为出错),两字符串类型和数据表面相等,但strcmp()结果不为0...
查看>>
洛谷P3006 [USACO11JAN]瓶颈Bottleneck(堆模拟)
查看>>
onchange事件与onpropertychange事件的区别
查看>>
【Linux】 诊断工具-strace
查看>>
2015-11-04 报表 (asp.net 部分)
查看>>
编辑器Sublime Text 2
查看>>
php性能测试工具
查看>>
动态规划(DP)
查看>>
bzoj3611
查看>>
Linux下配置MySQL需要注意的几点
查看>>