1概述

目前的app的动画效果是越来越炫了,很多主流app的图片预览返回都有类似功能,比较常见的是ios自带相册,微信朋友圈等等。自己项目中也有类似功能,最近整理了一下这个功能的代码,做个笔记记录,有兴趣的朋友可以点击源码。

网上已经有对应功能的三方库了,MyDragPhotoView 和 DragPhotoView等等。

但是他们都是继承view去实现的。如果我想用到其他View/ViewGroup上,就不是很方便,需要重新自定义view,然后复制黏贴和修改。因此就有了这篇文章,这个效果可以加在任意View或者ViewGroup上。

效果图:

一般经常用到这个功能的,莫非就是图片预览和视频预览了。

视频控件效果

图片控件效果

2使用步骤

1.activity主题设为透明

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)

3原理

很简单,就是touch事件传递,相信大家都已经滚瓜烂熟了。

大概步骤:

1.检测是否有拦截

2.ACTION_DOWN事件,初始化数据

3.ACTION_MOVE事件,如果多手指或者手指Id不一致,则复原,否则开始移动,同时更新拖拽View/ViewGroup的位置和大小。

4.ACTION_UP事件,判断是否超过指定的最大距离,如果超过,开始关闭动画,否则开始复原动画

核心代码如下:

public boolean handleEvent(MotionEvent event) {

if (dragCloseListener != null && dragCloseListener.intercept()) {

//拦截

Logger.d("action dispatch--->");

isSwipingToClose = false;

return false;

} else {

//不拦截

if (event.getAction() == MotionEvent.ACTION_DOWN) {

Logger.d("action down--->");

//初始化数据

lastPointerId = event.getPointerId(0);

reset(event);

} else if (event.getAction() == MotionEvent.ACTION_MOVE) {

Logger.d("action move--->" + event.getPointerCount() + "---" + isSwipingToClose);

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;

Logger.d("action move---> start close");

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);

childV.setScaleY(percent);

return true;

}

} else if (event.getAction() == MotionEvent.ACTION_UP) {

Logger.d("action up--->" + isSwipingToClose);

//手指抬起事件

if (isSwipingToClose) {

if (mCurrentTranslationY > maxExitY) {

if (isShareElementMode) {

//会执行共享元素的离开动画

if (dragCloseListener != null) {

dragCloseListener.dragClose(true);

}

} else {

//会执行定制的离开动画

exitWithTranslation(mCurrentTranslationY);

}

} else {

resetCallBackAnimation();

}

isSwipingToClose = false;

return true;

}

}

}

return false;

}

源码:

参考资料

安卓开发仿微信图片拖拽_使用Android 模仿微信朋友圈图片拖拽返回相关推荐

  1. 安卓开发学习日记第三天_新手怪button_莫韵乐的欢乐笔记

    安卓开发学习日记第三天--新手怪button (不是buttercup,虽然里面好像也有button,心中已经响起那段音乐了) 前情提要: 第一天学习日记之安装Android Studio3.6 第二 ...

  2. 安卓开发学习日记第四天_会爬就会跑_莫韵乐的欢乐笔记

    安卓开发学习日记第四天_会爬就会跑 前情提要 安卓开发学习日记第一天Android Studio3.6安装 安卓开发学习日记第二天_破坏陷阱卡之sync的坑 安卓开发学习日记第三天_新手怪button ...

  3. 视频教程-企业微信支付JAVA版_向员工收款-微信开发

    企业微信支付JAVA版_向员工收款 微信企业号星级会员.10多年软件从业经历,国家级软件项目负责人,主要从事软件研发.软件企业员工技能培训.已经取得计算机技术与软件资格考试(软考)--"信息 ...

  4. Android Studio 是谷歌基于IntelliJ IDEA开发的安卓开发工具,有点类似 Eclipse ADT,Android Studio 提供了集成的 Android 开发工具用于开发和调

    Android Studio 是谷歌基于IntelliJ IDEA开发的安卓开发工具,有点类似 Eclipse ADT,Android Studio 提供了集成的 Android 开发工具用于开发和调 ...

  5. android 仿微信录音,Android 模仿微信长按录音功能

    Android 模仿微信长按录音功能 布局就两个button这里不多介绍了. 下面是activity内的代码 package com.example.forevercloud.myapplicatio ...

  6. android手机微信收藏功能实现,Android模仿微信收藏文件的标签处理功能

    最近需要用到微信的标签功能(如下图所示).该功能可以添加已有标签,也可以自定义标签.也可以删除已编辑菜单.研究了一番.发现还是挺有意思的,模拟实现相关功能. 该功能使用类似FlowLayout的功能. ...

  7. android让字体左右对齐,Android 模仿微信读书文字左右对齐效果

    原标题:Android 模仿微信读书文字左右对齐效果 本文作者 作者:Amter https://www.jianshu.com/p/020786e22a6f 左右对齐的文字效果,很常见,在大多数文章 ...

  8. 安卓开发仿微信图片拖拽_仿微信朋友圈发表图片拖拽和删除功能

    原标题:仿微信朋友圈发表图片拖拽和删除功能 中国联通在香港公布了上市公司2017年中期业绩.2017年上半年,公司主要业绩指标持续向好,收入稳步回升,服务收入达到人民币1,241.1亿元,同比增长3. ...

  9. 安卓开发仿微信图片拖拽_Android 仿微信朋友圈发表图片拖拽和删除功能

    朋友圈实现原理 我们使用 Android Device Monitor 来分析朋友圈发布图片的界面实现原理.如果需要分析其他应用的界面实现也是采用这种方法哦. 打开 Android Device Mo ...

最新文章

  1. 未来30年,这20项技术将颠覆人类生活
  2. 网络回溯分析技术八大应用之运维评估 网络运维的真正价值
  3. VCAP-DCA 510 经验分享
  4. spring.net nhibernate 分布布式事务(下)
  5. node.js升级后原来的Ionic项目跑不起来了解决方法
  6. 数据结构与算法-数组
  7. LeetCode——贪心思想
  8. [TODO]Python拾遗(三)
  9. MongoDB 教程索引 (附有视频)
  10. Android集成支付宝支付
  11. Atitit 稳定性提升的艺术 之技术解决之道 目录 1. 2. 为什么会发生稳定性问题 1 1 2. 大原则 1 2.1. 尽快释放资源类似php最好的稳定性 1 2.2. Nginx 负载均衡
  12. 小程序开发者服务器设计,微信小程序的设计思路和理念
  13. linux如何磁盘管理工具下载,Linux_Linux系统的磁盘空间管理工具, 1.fdisk -lDis - phpStudy...
  14. “新政”不断的智慧城市,如何安全迎接新变革?
  15. mysql中chr_ASCII码对应表chr(9)、chr(10)、chr(13)、chr(32)、chr(34)、chr(39)
  16. 我是如何面试iOS开发者的?
  17. App Thinning(应用瘦身)功能介绍
  18. mysql命令行导入csv_MySQL命令行导入CSV文件
  19. 2020.3.13 美国数学大联盟杯赛复赛成绩出来了
  20. nodejs模拟登陆旧版正方教务系统

热门文章

  1. C语言之字符串处理函数
  2. 【uni-app】深度作用选择器解决修改checkbox样式无效问题
  3. MySQL查看存储过程
  4. core identity mysql_Asp.Net Core Identity 4 改成 MySql/MariaDB
  5. PPT怎么在剪映_剪映怎么剪辑音乐?添加、分割、导入音乐,图文详解!
  6. 我设计了一个牛逼的本地缓存!
  7. 万字讲解API网关的来龙去脉
  8. 蓝桥杯练习系统习题解答-入门训练
  9. 使用javabean类用户注册
  10. Java监听器用法(三):外部类监听器