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 < mFixactionRadiusmin) {
  //超过一定距离不需要绘制贝塞尔曲线和圆
  return null;
 }
 Path path = new Path();
 //求斜率
 float dy = (mDragPoint.y - mFixactionPoint.y);
 float dx = (mDragPoint.x - mFixactionPoint.x);
 float tanA = dy / dx;
 //求角a
 double arcTanA = Math.atan(tanA);
 //p0
 float p0x = (float) (mFixactionPoint.x + mFixactionRadius * Math.sin(arcTanA));
 float p0y = (float) (mFixactionPoint.y - mFixactionRadius * Math.cos(arcTanA));
 //p1
 float p1x = (float) (mDragPoint.x + mDragRadius * Math.sin(arcTanA));
 float p1y = (float) (mDragPoint.y - mDragRadius * Math.cos(arcTanA));
 //p2
 float p2x = (float) (mDragPoint.x - mDragRadius * Math.sin(arcTanA));
 float p2y = (float) (mDragPoint.y + mDragRadius * Math.cos(arcTanA));
 //p3
 float p3x = (float) (mFixactionPoint.x - mFixactionRadius * Math.sin(arcTanA));
 float p3y = (float) (mFixactionPoint.y + mFixactionRadius * Math.cos(arcTanA));

 //拼装贝塞尔曲线
 path.moveTo(p0x, p0y);
 //两个点,第一个是控制点,第二个是p1的位置
 PointF controlPoint = getControlPoint();
 //绘制第一条
 path.quadTo(controlPoint.x, controlPoint.y, p1x, p1y);

 //绘制第二条
 path.lineTo(p2x, p2y);
 path.quadTo(controlPoint.x, controlPoint.y, p3x, p3y);
 //闭合
 path.close();
 return path;
 }

 public PointF getControlPoint() {
 //控制点选取的为圆心的中心点
 PointF controlPoint = new PointF();
 controlPoint.x = (mDragPoint.x + mFixactionPoint.x) / 2;
 controlPoint.y = (mDragPoint.y + mFixactionPoint.y) / 2;
 return controlPoint;
 }

接下来就是处理手势触摸了,手势触摸主要是在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));
 }

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

<?xml version="1.0" encoding="utf-8"?>
<animation-list xmlns:android="http://schemas.android.com/apk/res/android"
 android:oneshot="true" >

 <item
 android:drawable="@drawable/pop1"
 android:duration="100"/>
 <item
 android:drawable="@drawable/pop2"
 android:duration="100"/>
 <item
 android:drawable="@drawable/pop3"
 android:duration="100"/>
 <item
 android:drawable="@drawable/pop4"
 android:duration="100"/>
 <item
 android:drawable="@drawable/pop5"
 android:duration="100"/>

</animation-list>

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

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

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

(0)

相关推荐

  • Android仿QQ未读消息--红点拖拽删除【源代码】

    本Demo是一款仿qq未读消息拖拽删除的例子,继承RelativeLayout的WaterDrop实现了圆形图标功能.继承ImageView的CircleImageView圆形图片功能.效果非常不错.很适合有圆形设计的哥们.效果图片如下 CircleImageView核心代码 private void updateShaderMatrix() { float scale; float dx = 0; float dy = 0; mShaderMatrix.set(null); if (mBitm

  • Android贝塞尔曲线实现消息拖拽消失

    写在前头 写消息拖拽效果的文章不少,但是大部分都把自定义View写死了,我们要实现的是传入一个View,每个View都可以实现拖拽消失爆炸的效果,当然我也是站在巨人的肩膀上来学习的.但个人觉得程序员本就应该敢于学习和借鉴. 源码地址:源码Github地址 效果图 分析(用到的知识点):  (1)ValueAnimator (数值生成器) 用于生成数值,可以设置差值器来改变数字的变化幅度. (2)ObjectAnimator (动画生成器) 用于生成各种属性,布局动画,同样也可以设置差值器来改变效

  • Android仿QQ消息提示点拖拽功能

    很久以前,发现QQ有一个很有趣的功能,就是未读消息的红点是可以拖拽的,而且在任何地方都可以随意拖拽,并且有一个弹性的动画,非常有趣,而且也是一个非常方便的功能,于是总想仿制一个,虽说仿制,但也只是他的拖拽功能,弹性效果还是能力有限. 不多说 先上效果 一个自定义的view 使用方式也很简单 <com.weizhenbin.show.widget.VanishView android:layout_width="30dp" android:layout_height="3

  • Android贝塞尔曲线初步学习第二课 仿QQ未读消息气泡拖拽黏连效果

    上一节初步了解了Android端的贝塞尔曲线,这一节就举个栗子练习一下,仿QQ未读消息气泡,是最经典的练习贝塞尔曲线的东东,效果如下 附上github源码地址:https://github.com/MonkeyMushroom/DragBubbleView 欢迎star~ 大体思路就是画两个圆,一个黏连小球固定在一个点上,一个气泡小球跟随手指的滑动改变坐标.随着两个圆间距越来越大,黏连小球半径越来越小.当间距小于一定值,松开手指气泡小球会恢复原来位置:当间距超过一定值之后,黏连小球消失,气泡小球

  • Android使用贝塞尔曲线仿QQ聊天消息气泡拖拽效果

    本文实例为大家分享了Android仿QQ聊天消息气泡拖拽效果展示的具体代码,供大家参考,具体内容如下 先画圆,都会吧.代码如下: public class Bezier extends View { private final Paint mGesturePaint = new Paint(); private final Path mPath = new Path(); private float mX1 = 100, mY1 = 150; private float mX2 = 300, m

  • android自定义view之模拟qq消息拖拽删除效果

    这个模拟功能的实现主要依靠了PATH和二阶贝塞尔曲线.首先上一张图来简单看一下: 这个模拟功能有以下几个特点: 在开始的时候点击圆以外的区域不会触发拖动事件 点击圆的时候可以拖拽,此时会有一个拉伸效果,连接大圆和小圆 拉伸到一定距离(自己设定)以后两个圆会断开,此时即使再拖拽进距离之内的时候也不会再产生已经断开的连接 在距离之内松手的时候会回弹会原位置,并伴有一个弹跳动画 介绍了这么多,看过我前边文章的朋友应该会有一个基本思路. 暴露接口 这个模拟功能共分为三部分,一个是那个小圆,固定的位置,一

  • Android仿qq消息拖拽效果

    本文实例为大家分享了Android仿qq消息拖拽效果展示的具体代码,供大家参考,具体内容如下 这是一个仿qq消息拖拽效果,View和拖拽实现了分离,TextView.Button.Imageview等都可以实现相应的拖拽效果:在触发的地方调用 MessageBubbleView.attach(findViewById(R.id.text_view), new MessageBubbleView.BubbleDisappearListener() { @Override public void d

  • Android自定义ListView实现仿QQ可拖拽列表功能

    我们大致的思路,其实是这样子的,也是我的设想,我们可以先去实现一个简单的ListView的数据,但是他的Adapter,我们可以用系统封装好的,然后传递进去一个实体类,最后自定义一个listview去操作,所以我们先把准备的工作做好,比如? list_item.xml <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.a

  • Android自定义View实现拖拽效果

    腾讯QQ有那种红点拖动效果,今天就来实现一个简单的自定义View拖动效果,再回到原处,并非完全仿QQ红点拖动. 先来看一下效果图 简单说一下实现步骤 1.创建一个类继承View 2.绘制出一个小球 3.重写onTouchEvent,来根据手指放下,移动,抬起,来控制小球 4.直接在布局中引用 先贴一张图看下View的坐标系 下面就贴一下代码,最后会给出源码 public class CustomView extends View { private int lastX; private int

  • Android RecycleView实现Item拖拽效果

    基于公司产品的优化需求,其中一个需求涉及到RecycleView的拖拽,以及拖拽后item位置的持久化,目的是可以用户自定义界面偏好,并在用户下次进入本界面后,之前设置的偏好仍然有效.我写了一个小Demo用作演示效果. 先看效果(只看效果,不看颜值) 步骤1.建接口文件ItemTouchHelperViewHolder,该接口文件中描述的是选中和放开当前Item调用的方法. public interface ItemTouchHelperViewHolder { void onItemSelec

  • js实现QQ面板拖拽效果(慕课网DOM事件探秘)(全)

    QQ面板拖拽,效果如图 JavaScript代码如下: function getByClass(clsName, parent) { var oParent = parent ? document.getElementById(parent) : document, eles = [], elements = oParent.getElementsByTagName('*'); for (var i = 0, l = elements.length; i < l; i++) { if (elem

  • Android仿QQ讨论组头像效果

    本文实例为大家分享了Android仿QQ讨论组头像展示的具体代码,供大家参考,具体内容如下 一.效果图 二.实现 基本实现过程: 1.将原图片读取为bitmap 2.在Canvas画布上计算出图片位置,并绘制新的图片. (ps:计算位置对我来说是难点,花了好长时间): 三.源码 1.布局文件 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http:/

  • Android仿QQ消息提示实现弹出式对话框

    本文在<7种形式的Android Dialog使用实例>在这篇文章的基础进行学习,具体内容如下 1.概述 android原生控件向来以丑著称(新推出的Material Design当另说),因此几乎所有的应用都会特殊定制自己的UI样式.而其中弹出式提示框的定制尤为常见,本篇我们将从模仿QQ退出提示框来看一下常见的几种自定义提示框的实现方式. 这里使用的几种弹出框实现方法概括为以下几种: 自定义Dialog 自定义PopupWindow 自定义Layout View Activity的Dialo

  • Android仿QQ微信侧滑删除效果

    仿QQ侧滑删除效果图 1.自定义listview public class DragDelListView extends ListView { private boolean moveable=false; private boolean closed=true; private float mDownX,mDownY; private int mTouchPosition,oldPosition=-1; private DragDelItem mTouchView,oldView; priv

随机推荐