true
2.初始化
DragCloseHelper dragCloseHelper = new DragCloseHelper(this);
3.如果是共享元素启动的页面,需要如下设置(强烈建议和共享元素一起使用,否则是没有灵魂的)
dragCloseHelper.setShareElementMode(true);
4.设置需要进行拖拽的View/ViewGroup,以及背景ViewGroup(必须要设置背景色)
dragCloseHelper.setDragCloseViews(parentV, childV);
5.设置监听
dragCloseHelper.setDragCloseListener(new DragCloseHelper.DragCloseListener() {
@Override
public boolean intercept() {
//默认false 不拦截。比如图片在放大状态,是不需要执行拖拽动画的等等。
return false;
}
@Override
public void dragStart() {
//拖拽开始。可以在此额外处理一些逻辑
}
@Override
public void dragging(float percent) {
//拖拽中。percent当前的进度,取值0-1,可以在此额外处理一些逻辑
}
@Override
public void dragCancel() {
//拖拽取消,会自动复原。可以在此额外处理一些逻辑
}
@Override
public void dragClose(boolean isShareElementMode) {
//拖拽关闭,如果是共享元素的页面,需要执行activity的onBackPressed方法,注意如果使用finish方法,则返回的时候没有共享元素的返回动画
if (isShareElementMode) {
onBackPressed();
}
}
});
6.处理touch事件
@Override
public boolean dispatchTouchEvent(MotionEvent event) {
if (dragCloseHelper.handleEvent(event)) {
return true;
} else {
return super.dispatchTouchEvent(event);
}
}
7.可以自定义最大拖拽距离和最小缩放尺寸
setMaxExitY(int maxExitY)
setMinScale(@FloatRange(from = 0.1f, to = 1.0f) float minScale)
原理:
很简单,就是touch事件传递,相信大家都已经滚瓜烂熟了。
大概步骤:
1.检测是否有拦截
2.ACTION_DOWN事件,初始化数据
3.ACTION_MOVE事件,如果多手指或者手指Id不一致,则复原,否则开始移动,同时更新拖拽View/ViewGroup的位置和大小。
4.ACTION_UP事件,判断是否超过指定的最大距离,如果超过,开始关闭动画,否则开始复原动画
核心代码如下:
/**
- 处理touch事件
- @param event
- @return
*/
public boolean handleEvent(MotionEvent event) {
if (dragCloseListener != null && dragCloseListener.intercept()) {
//拦截
isSwipingToClose = false;
return false;
} else {
//不拦截
if (event.getAction() == MotionEvent.ACTION_DOWN) {
//初始化数据
lastPointerId = event.getPointerId(0);
reset(event);
} else if (event.getAction() == MotionEvent.ACTION_MOVE) {
if (event.getPointerCount() > 1) {
//如果有多个手指
if (isSwipingToClose) {
//已经开始滑动关闭,恢复原状,否则需要派发事件
isSwipingToClose = false;
resetCallBackAnimation();
return true;
}
reset(event);
return false;
}
if (lastPointerId != event.getPointerId(0)) {
//手指不一致,恢复原状
if (isSwipingToClose) {
resetCallBackAnimation();
}
reset(event);
return true;
}
float currentY = event.getY();
float currentX = event.getX();
if (isSwipingToClose || Math.abs(currentY - mLastY) > 2 * viewConfiguration.getScaledTouchSlop()) {
//已经触发或者开始触发,更新view
mLastY = currentY;
mLastX = currentX;
float currentRawY = event.getRawY();
float currentRawX = event.getRawX();
if (!isSwipingToClose) {
//准备开始
isSwipingToClose = true;
if (dragCloseListener != null) {
dragCloseListener.dragStart();
}
}
//已经开始,更新view
mCurrentTranslationY = currentRawY - mLastRawY + mLastTranslationY;
mCurrentTranslationX = currentRawX - mLastRawX + mLastTranslationX;
float percent = 1 - Math.abs(mCurrentTranslationY / (maxExitY + childV.getHeight()));
if (percent > 1) {
percent = 1;
} else if (percent < 0) {
percent = 0;
}
parentV.getBackground().mutate().setAlpha((int) (percent * 255));
if (dragCloseListener != null) {
dragCloseListener.dragging(percent);
}
childV.setTranslationY(mCurrentTranslationY);
childV.setTranslationX(mCurrentTranslationX);
if (percent < minScale) {
percent = minScale;
}
childV.setScaleX(percent);
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

学习交流
群内有许多来自一线的技术大牛,也有在小厂或外包公司奋斗的码农,我们致力打造一个平等,高质量的Android交流圈子,不一定能短期就让每个人的技术突飞猛进,但从长远来说,眼光,格局,长远发展的方向才是最重要的。
35岁中年危机大多是因为被短期的利益牵着走,过早压榨掉了价值,如果能一开始就树立一个正确的长远的职业规划。35岁后的你只会比周围的人更值钱。
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!
是最重要的。
35岁中年危机大多是因为被短期的利益牵着走,过早压榨掉了价值,如果能一开始就树立一个正确的长远的职业规划。35岁后的你只会比周围的人更值钱。
《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!