热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android仿qq消息拖拽效果

这篇文章主要为大家详细介绍了Android仿qq消息拖拽效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了Android仿qq消息拖拽效果展示的具体代码,供大家参考,具体内容如下

这是一个仿qq消息拖拽效果,View和拖拽实现了分离,TextView、Button、Imageview等都可以实现相应的拖拽效果;在触发的地方调用

MessageBubbleView.attach(findViewById(R.id.text_view), new MessageBubbleView.BubbleDisappearListener() {
 @Override
 public void dismiss(View view) {
 Toast.makeText(MainActivity.this,"消失了",Toast.LENGTH_LONG).show();
 }
});

就可以了,第一个参数需要传入一个View,第二个参数需要出入BubbleDisappearListener的实现类进行消失监听回调;在attach();方法中也给传入的View设置了触摸监听事件;

/**
 * 绑定可以拖拽的控件
 *
 * @param view
 * @param disappearListener
 */
public static void attach(View view, BubbleDisappearListener disappearListener) {
 if (view == null) {
 return;
 }
 view.setOnTouchListener(new BubbleMessageTouchListener(view, view.getContext(),disappearListener));
}

BubbleMessageTouchListener类的话是用来处理触摸监听的类,先去看MessageBubbleView类,先去实现自定义view的效果,再去处理相应的触摸事件;

public class MessageBubbleView extends View {
 //两个圆的圆心
 private PointF mFixactionPoint;
 private PointF mDragPoint;
 //拖拽圆的半径
 private int mDragRadius = 15;
 //画笔
 private Paint mPaint;
 //固定圆的半径
 private int mFixactionRadius;
 //固定圆半径的初始值
 private int mFixactiOnRadiusMax= 12;
 //最小值
 private int mFixactiOnRadiusmin= 3;
 private Bitmap mDragBitmap;
 
 public MessageBubbleView(Context context) {
 this(context, null);
 }
 
 public MessageBubbleView(Context context, AttributeSet attrs) {
 this(context, attrs, 0);
 }
 
 public MessageBubbleView(Context context, AttributeSet attrs, int defStyleAttr) {
 super(context, attrs, defStyleAttr);
 mDragRadius = dip2px(mDragRadius);
 mFixactiOnRadiusMax= dip2px(mFixactionRadiusMax);
 mFixactiOnRadiusmin= dip2px(mFixactionRadiusmin);
 mPaint = new Paint();
 mPaint.setColor(Color.RED);
 mPaint.setAntiAlias(true);
 mPaint.setDither(true);
 }
 
 private int dip2px(int dip) {
 return (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, dip, getResources().getDisplayMetrics());
 }
}

首先是一些参数的定义及画笔的初始化,接下来就要在onDraw()方法中进行绘制,这里会涉及到两个圆的绘制,一个是固定圆,还有一个是拖拽圆,对于拖拽圆来说,确定x,y坐标及圆的半径就可以进行绘制了,相对来说简单些,对于固定圆来说,一开始有一个初始值,半径是随着距离的增大而减小,小到一定程度就消失;

@Override
protected void onDraw(Canvas canvas) {
 if (mDragPoint == null || mFixactiOnPoint== null) {
 return;
 }
 //画两个圆
 //绘制拖拽圆
 canvas.drawCircle(mDragPoint.x, mDragPoint.y, mDragRadius, mPaint);
 //绘制固定圆 有一个初始大小,而且半径是随着距离的增大而减小,小到一定程度就消失
 Path bezeierPath = getBezeierPath();
 if (bezeierPath != null) {
 canvas.drawCircle(mFixactionPoint.x, mFixactionPoint.y, mFixactionRadius, mPaint);
 //绘制贝塞尔曲线
 canvas.drawPath(bezeierPath, mPaint);
 }
 if (mDragBitmap != null) {
 //绘制图片 位置也是手指一动的位置 中心位置才是手指拖动的位置
 canvas.drawBitmap(mDragBitmap, mDragPoint.x - mDragBitmap.getWidth() / 2, mDragPoint.y - mDragBitmap.getHeight() / 2, null);
 }
}

绘制了拖拽圆和固定圆后,就需要将两个圆连接起来,连接两个圆的路径的绘制就需要使用三阶贝塞尔曲线来实现;

看过去,需要求p0、p1、p2、p3,这几个点的左边,对于c0、c1的坐标,拖拽圆和固定圆的半径都是知道的,可以先求出c0到c1的距离,对于p0、p1、p2、p3坐标可以通过三角函数求得,再利用Path路径进行绘制;

/**
 * 获取贝塞尔的路径
 *
 * @return
 */
 public Path getBezeierPath() {
 //计算两个点的距离
 double distance = getDistance(mDragPoint, mFixactionPoint);
 mFixactiOnRadius= (int) (mFixactionRadiusMax - distance / 14);
 if (mFixactionRadius 

接下来就是处理手势触摸了,手势触摸主要是在BubbleMessageTouchListener类中的onTouch()方法中进行处理;

@Override
 public boolean onTouch(View v, MotionEvent event) {
 switch (event.getAction()) {
  case MotionEvent.ACTION_DOWN:
  //在windowManager上面搞一个view,
  mWindowManager.addView(mMessageBubbleView, mParams);
  //初始化贝塞尔view的点
  //需要获取屏幕的位置 不是相对于父布局的位置 还需要减掉状态栏的高度
  //将页面做为全屏的可以将其拖拽到状态栏上面
  //保证固定圆的中心在view的中心
  int[] location = new int[2];
  mStateView.getLocationOnScreen(location);
  Bitmap bitmapByView = getBitmapByView(mStateView);
  mMessageBubbleView.initPoint(location[0] + mStateView.getWidth() / 2, location[1] + mStateView.getHeight() / 2 - BubbleUtils.getStatusBarHeight(mContext));
  //给消息拖拽设置一个bitmap
  mMessageBubbleView.setDragBitmap(bitmapByView);
  //首先将自己隐藏
  mStateView.setVisibility(View.INVISIBLE);
  break;
  case MotionEvent.ACTION_MOVE:
  mMessageBubbleView.updataDragPoint(event.getRawX(), event.getRawY());
  break;
  case MotionEvent.ACTION_UP:
  //拖动如果贝塞尔曲线没有消失就回弹
  //拖动如果贝塞尔曲线消失就爆炸
  mMessageBubbleView.handleActionUp();
  break;
 }
 return true;
 }

在按下拖拽的时候,为了能让View能拖拽到手机屏幕上的任意一点,是在该view添加到了WindowManager上,

public BubbleMessageTouchListener(View mStateView, Context context,MessageBubbleView.BubbleDisappearListener disappearListener) {
 this.mStateView = mStateView;
 this.mCOntext= context;
 this.disappearListener=disappearListener;
 mWindowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
 mMessageBubbleView = new MessageBubbleView(context);
 //设置监听
 mMessageBubbleView.setMessageBubbleListener(this);
 mParams = new WindowManager.LayoutParams();
 //设置背景透明
 mParams.format = PixelFormat.TRANSLUCENT;
 
 mBombFrame = new FrameLayout(mContext);
 mBombImageView = new ImageView(mContext);
 mBombImageView.setLayoutParams(new FrameLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT));
 mBombFrame.addView(mBombImageView);
 }

在按下的时候需要初始化坐标点及设置相应的背景;

/**
 * 初始化位置
 *
 * @param downX
 * @param downY
 */
 public void initPoint(float downX, float downY) {
 mFixactiOnPoint= new PointF(downX, downY);
 mDragPoint = new PointF(downX, downY);
 }
 /**
 * @param bitmap
 */
 public void setDragBitmap(Bitmap bitmap) {
 this.mDragBitmap = bitmap;
 }

对于ACTION_MOVE手势移动来说,只需要去不断更新移动的坐标就可以了;

/**
 * 更新当前拖拽点的位置
 *
 * @param moveX
 * @param moveY
 */
 public void updataDragPoint(float moveX, float moveY) {
 mDragPoint.x = moveX;
 mDragPoint.y = moveY;
 //不断绘制
 invalidate();
 }

对于ACTION_UP手势松开的话,处理就要麻烦些,这里需要判断拖拽的距离,如果拖拽的距离在规定的距离内就反弹,如果超过规定的距离就消失,并伴随相应的动画效果;

/**
 * 处理手指松开
 */
 public void handleActionUp() {
 if (mFixactionRadius > mFixactionRadiusmin) {
  //拖动如果贝塞尔曲线没有消失就回弹
  //ValueAnimator 值变化的动画 从0-->1的变化
  ValueAnimator animator = ObjectAnimator.ofFloat(1);
  animator.setDuration(250);
  final PointF start = new PointF(mDragPoint.x, mDragPoint.y);
  final PointF end = new PointF(mFixactionPoint.x, mFixactionPoint.y);
  animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
  @Override
  public void onAnimationUpdate(ValueAnimator animation) {
   float animatedValue = (float) animation.getAnimatedValue();
//   int percent = (int) animatedValue;
   PointF pointF = BubbleUtils.getPointByPercent(start, end, animatedValue);
   //更新当前拖拽点
   updataDragPoint(pointF.x, pointF.y);
  }
  });
  animator.setInterpolator(new OvershootInterpolator(5f));
  animator.start();
  //通知TouchListener移除当前View然后显示静态的view
  animator.addListener(new AnimatorListenerAdapter() {
  @Override
  public void onAnimationEnd(Animator animation) {
   super.onAnimationEnd(animation);
   if(mListener!=null){
   mListener.restore();
   }
  }
  });
 } else {
  //拖动如果贝塞尔曲线消失就爆炸
  if(mListener!=null){
  mListener.dimiss(mDragPoint);
  }
 }
 }

而在MessageBubbleListener接口监听中需要对void restore();和void dimiss(PointF pointf);进行相应的监听处理,在拖拽距离在规定距离内的话就会去回调restore()方法;

@Override
 public void restore() {
 //把消息的view移除
 mWindowManager.removeView(mMessageBubbleView);
 //将原来的View显示
 mStateView.setVisibility(View.VISIBLE);
 }

如果拖拽的距离大于规定的距离就会去回调void dimiss(PointF pointf);方法:

 @Override
 public void dimiss(PointF pointF) {
 //要去执行爆炸动画 帧动画
 //原来的view肯定要移除
 mWindowManager.removeView(mMessageBubbleView);
 //要在WindowManager添加一个爆炸动画
 mWindowManager.addView(mBombFrame, mParams);
 //设置背景
 mBombImageView.setBackgroundResource(R.drawable.anim_bubble_pop);
 AnimationDrawable drawable = (AnimationDrawable) mBombImageView.getBackground();
 //设置位置
 mBombImageView.setX(pointF.x-drawable.getIntrinsicWidth()/2);
 mBombImageView.setY(pointF.y-drawable.getIntrinsicHeight()/2);
 //开启动画
 drawable.start();
 //执行完毕后要移除掉mBombFrame
 mBombImageView.postDelayed(new Runnable() {
  @Override
  public void run() {
  //移除
  mWindowManager.removeView(mBombFrame);
  //通知该view消失了
  if(disappearListener!=null){
   disappearListener.dismiss(mMessageBubbleView);
  }
  }
 }, getAnimationDrawableTime(drawable));
 }

在拖拽消失后的那个消失动画是使用帧动画来实现的;

<&#63;xml version="1.0" encoding="utf-8"&#63;>

 
 
 
 
 
 
 

这样子效果就差不多ok了。

源码地址:仿qq消息拖拽效果

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • 本文介绍了使用kotlin实现动画效果的方法,包括上下移动、放大缩小、旋转等功能。通过代码示例演示了如何使用ObjectAnimator和AnimatorSet来实现动画效果,并提供了实现抖动效果的代码。同时还介绍了如何使用translationY和translationX来实现上下和左右移动的效果。最后还提供了一个anim_small.xml文件的代码示例,可以用来实现放大缩小的效果。 ... [详细]
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • 本文讲述了如何通过代码在Android中更改Recycler视图项的背景颜色。通过在onBindViewHolder方法中设置条件判断,可以实现根据条件改变背景颜色的效果。同时,还介绍了如何修改底部边框颜色以及提供了RecyclerView Fragment layout.xml和项目布局文件的示例代码。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • Android系统移植与调试之如何修改Android设备状态条上音量加减键在横竖屏切换的时候的显示于隐藏
    本文介绍了如何修改Android设备状态条上音量加减键在横竖屏切换时的显示与隐藏。通过修改系统文件system_bar.xml实现了该功能,并分享了解决思路和经验。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • 本文介绍了C#中数据集DataSet对象的使用及相关方法详解,包括DataSet对象的概述、与数据关系对象的互联、Rows集合和Columns集合的组成,以及DataSet对象常用的方法之一——Merge方法的使用。通过本文的阅读,读者可以了解到DataSet对象在C#中的重要性和使用方法。 ... [详细]
  • HDFS2.x新特性
    一、集群间数据拷贝scp实现两个远程主机之间的文件复制scp-rhello.txtroothadoop103:useratguiguhello.txt推pushscp-rr ... [详细]
  • flowable工作流 流程变量_信也科技工作流平台的技术实践
    1背景随着公司业务发展及内部业务流程诉求的增长,目前信息化系统不能够很好满足期望,主要体现如下:目前OA流程引擎无法满足企业特定业务流程需求,且移动端体 ... [详细]
author-avatar
他给我留下的美好_813
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有