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

上一节初步了解了Android端的贝塞尔曲线,这一节就举个栗子练习一下,仿QQ未读消息气泡,是最经典的练习贝塞尔曲线的东东,效果如下

附上github源码地址:https://github.com/MonkeyMushroom/DragBubbleView

欢迎star~

大体思路就是画两个圆,一个黏连小球固定在一个点上,一个气泡小球跟随手指的滑动改变坐标。随着两个圆间距越来越大,黏连小球半径越来越小。当间距小于一定值,松开手指气泡小球会恢复原来位置;当间距超过一定值之后,黏连小球消失,气泡小球继续跟随手指移动,此时手指松开,气泡小球消失~

1、首先老一套~新建attrs.xml文件,编写自定义属性,新建DragBubbleView继承View,重写构造方法,获取自定义属性值,初始化Paint、Path等东东,重写onMeasure计算宽高,这里不再啰嗦~

2、在onSizeChanged方法中确定黏连小球和气泡小球的圆心坐标,这里我们取宽高的一半:

@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
 super.onSizeChanged(w, h, oldw, oldh);
 mBubbleCenterX = w / 2;
 mBubbleCenterY = h / 2;
 mCircleCenterX = mBubbleCenterX;
 mCircleCenterY = mBubbleCenterY;
}

3、经分析气泡小球有以下几个状态:默认、拖拽、移动、消失,我们这里定义一下,方便根据不同的状态分析不同情况:

/* 气泡的状态 */
private int mState;
/* 默认,无法拖拽 */
private static final int STATE_DEFAULT = 0x00;
/* 拖拽 */
private static final int STATE_DRAG = 0x01;
/* 移动 */
private static final int STATE_MOVE = 0x02;
/* 消失 */
private static final int STATE_DISMISS = 0x03;

4、重写onTouchEvent方法,其中d代表两圆圆心间距,maxD代表可拖拽的最大间距:

@Override
public boolean onTouchEvent(MotionEvent event) {
 switch (event.getAction()) {
 case MotionEvent.ACTION_DOWN:
 if (mState != STATE_DISMISS) {
 d = (float) Math.hypot(event.getX() - mBubbleCenterX, event.getY() - mBubbleCenterY);
 if (d < mBubbleRadius + maxD / 4) {
  //当指尖坐标在圆内的时候,才认为是可拖拽的
  //一般气泡比较小,增加(maxD/4)像素是为了更轻松的拖拽
  mState = STATE_DRAG;
 } else {
  mState = STATE_DEFAULT;
 }
 }
 break;
 case MotionEvent.ACTION_MOVE:
 if (mState != STATE_DEFAULT) {
 mBubbleCenterX = event.getX();
 mBubbleCenterY = event.getY();
 //计算气泡圆心与黏连小球圆心的间距
 d = (float) Math.hypot(mBubbleCenterX - mCircleCenterX, mBubbleCenterY - mCircleCenterY);
 //float d = (float) Math.sqrt(Math.pow(mBubbleCenterX - mCircleCenterX, 2)
 //+ Math.pow(mBubbleCenterY - mCircleCenterY, 2));
 if (mState == STATE_DRAG) {//如果可拖拽
  //间距小于可黏连的最大距离
  if (d < maxD - maxD / 4) {//减去(maxD/4)的像素大小,是为了让黏连小球半径到一个较小值快消失时直接消失
  mCircleRadius = mBubbleRadius - d / 8;//使黏连小球半径渐渐变小
  if (mOnBubbleStateListener != null) {
  mOnBubbleStateListener.onDrag();
  }
  } else {//间距大于于可黏连的最大距离
  mState = STATE_MOVE;//改为移动状态
  if (mOnBubbleStateListener != null) {
  mOnBubbleStateListener.onMove();
  }
  }
 }
 invalidate();
 }
 break;
 case MotionEvent.ACTION_UP:
 if (mState == STATE_DRAG) {//正在拖拽时松开手指,气泡恢复原来位置并颤动一下
 setBubbleRestoreAnim();
 } else if (mState == STATE_MOVE) {//正在移动时松开手指
 //如果在移动状态下间距回到两倍半径之内,我们认为用户不想取消该气泡
 if (d < 2 * mBubbleRadius) {//那么气泡恢复原来位置并颤动一下
  setBubbleRestoreAnim();
 } else {//气泡消失
  setBubbleDismissAnim();
 }
 }
 break;
 }
 return true;
}

如果控件外面有嵌套ListView、RecyclerView等拦截焦点的控件,那就在ACTION_DOWN中请求父控件不拦截事件:

getParent().requestDisallowInterceptTouchEvent(true);

然后ACTION_UP再把事件还回去:

getParent().requestDisallowInterceptTouchEvent(false);

5、在onDraw方法中画圆、画贝赛尔曲线、画消息个数文本:

@Override
protected void onDraw(Canvas canvas) {
 super.onDraw(canvas);
 //画拖拽气泡
 canvas.drawCircle(mBubbleCenterX, mBubbleCenterY, mBubbleRadius, mBubblePaint);

 if (mState == STATE_DRAG && d < maxD - 48) {
 //画黏连小圆
 canvas.drawCircle(mCircleCenterX, mCircleCenterY, mCircleRadius, mBubblePaint);
 //计算二阶贝塞尔曲线做需要的起点、终点和控制点坐标
 calculateBezierCoordinate();
 //画二阶贝赛尔曲线
 mBezierPath.reset();
 mBezierPath.moveTo(mCircleStartX, mCircleStartY);
 mBezierPath.quadTo(mControlX, mControlY, mBubbleEndX, mBubbleEndY);
 mBezierPath.lineTo(mBubbleStartX, mBubbleStartY);
 mBezierPath.quadTo(mControlX, mControlY, mCircleEndX, mCircleEndY);
 mBezierPath.close();
 canvas.drawPath(mBezierPath, mBubblePaint);
 }
 //画消息个数的文本
 if (mState != STATE_DISMISS && !TextUtils.isEmpty(mText)) {
 mTextPaint.getTextBounds(mText, 0, mText.length(), mTextRect);
 canvas.drawText(mText, mBubbleCenterX - mTextRect.width() / 2, mBubbleCenterY + mTextRect.height() / 2, mTextPaint);
 }
}

其中计算二阶贝塞尔曲线做需要的起点、终点和控制点坐标,顺序是moveTo A, quadTo B, lineTo C, quadTo D, close
先来张示意图:

再上代码

/**
 * 计算二阶贝塞尔曲线做需要的起点、终点和控制点坐标
 */
private void calculateBezierCoordinate(){
 //计算控制点坐标,为两圆圆心连线的中点
 mControlX = (mBubbleCenterX + mCircleCenterX) / 2;
 mControlY = (mBubbleCenterY + mCircleCenterY) / 2;
 //计算两条二阶贝塞尔曲线的起点和终点
 float sin = (mBubbleCenterY - mCircleCenterY) / d;
 float cos = (mBubbleCenterX - mCircleCenterX) / d;
 mCircleStartX = mCircleCenterX - mCircleRadius * sin;
 mCircleStartY = mCircleCenterY + mCircleRadius * cos;
 mBubbleEndX = mBubbleCenterX - mBubbleRadius * sin;
 mBubbleEndY = mBubbleCenterY + mBubbleRadius * cos;
 mBubbleStartX = mBubbleCenterX + mBubbleRadius * sin;
 mBubbleStartY = mBubbleCenterY - mBubbleRadius * cos;
 mCircleEndX = mCircleCenterX + mCircleRadius * sin;
 mCircleEndY = mCircleCenterY - mCircleRadius * cos;
}

6、气泡复原的动画,使用估值器计算坐标

/**
 * 设置气泡复原的动画
 */
private void setBubbleRestoreAnim() {
 ValueAnimator anim = ValueAnimator.ofObject(new PointFEvaluator(),
 new PointF(mBubbleCenterX, mBubbleCenterY),
 new PointF(mCircleCenterX, mCircleCenterY));
 anim.setDuration(200);
 //使用OvershootInterpolator差值器达到颤动效果
 anim.setInterpolator(new OvershootInterpolator(5));
 anim.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
 @Override
 public void onAnimationUpdate(ValueAnimator animation) {
 PointF curPoint = (PointF) animation.getAnimatedValue();
 mBubbleCenterX = curPoint.x;
 mBubbleCenterY = curPoint.y;
 invalidate();
 }
 });
 anim.addListener(new AnimatorListenerAdapter() {
 @Override
 public void onAnimationEnd(Animator animation) {
 //动画结束后状态改为默认
 mState = STATE_DEFAULT;
 if (mOnBubbleStateListener != null) {
 mOnBubbleStateListener.onRestore();
 }
 }
 });
 anim.start();
}
/**
 * PointF动画估值器
 */
public class PointFEvaluator implements TypeEvaluator<PointF> {

 @Override
 public PointF evaluate(float fraction, PointF startPointF, PointF endPointF) {
 float x = startPointF.x + fraction * (endPointF.x - startPointF.x);
 float y = startPointF.y + fraction * (endPointF.y - startPointF.y);
 return new PointF(x, y);
 }
}

7、顺便来个气泡状态的监听器,方便外部调用监听其状态:

/**
 * 气泡状态的监听器
 */
public interface OnBubbleStateListener {
 /**
 * 拖拽气泡
 */
 void onDrag();

 /**
 * 移动气泡
 */
 void onMove();

 /**
 * 气泡恢复原来位置
 */
 void onRestore();

 /**
 * 气泡消失
 */
 void onDismiss();
}

/**
 * 设置气泡状态的监听器
 */
public void setOnBubbleStateListener(OnBubbleStateListener onBubbleStateListener) {
 mOnBubbleStateListener = onBubbleStateListener;
}

8、关于气泡爆炸的动画,思路就是放几张图片到drawable里,然后动态计数重绘,在onDraw中调用canvas.drawBitmap()方法,具体如下:

/* 气泡爆炸的图片id数组 */
private int[] mExplosionDrawables = {R.drawable.explosion_one, R.drawable.explosion_two
 , R.drawable.explosion_three, R.drawable.explosion_four, R.drawable.explosion_five};
/* 气泡爆炸的bitmap数组 */
private Bitmap[] mExplosionBitmaps;
/* 气泡爆炸当前进行到第几张 */
private int mCurExplosionIndex;
/* 气泡爆炸动画是否开始 */
private boolean mIsExplosionAnimStart = false;

在构造方法中:

mExplosionPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mExplosionPaint.setFilterBitmap(true);
mExplosionRect = new Rect();
mExplosionBitmaps = new Bitmap[mExplosionDrawables.length];
for (int i = 0; i < mExplosionDrawables.length; i++) {
 //将气泡爆炸的drawable转为bitmap
 Bitmap bitmap = BitmapFactory.decodeResource(getResources(), mExplosionDrawables[i]);
 mExplosionBitmaps[i] = bitmap;
}

然后在手指抬起的时候使用如下动画:

/**
 * 设置气泡消失的动画
 */
private void setBubbleDismissAnim() {
 mState = STATE_DISMISS;//气泡改为消失状态
 mIsExplosionAnimStart = true;
 if (mOnBubbleStateListener != null) {
 mOnBubbleStateListener.onDismiss();
 }
 //做一个int型属性动画,从0开始,到气泡爆炸图片数组个数结束
 ValueAnimator anim = ValueAnimator.ofInt(0, mExplosionDrawables.length);
 anim.setInterpolator(new LinearInterpolator());
 anim.setDuration(500);
 anim.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
 @Override
 public void onAnimationUpdate(ValueAnimator animation) {
 //拿到当前的值并重绘
 mCurExplosionIndex = (int) animation.getAnimatedValue();
 invalidate();
 }
 });
 anim.addListener(new AnimatorListenerAdapter() {
 @Override
 public void onAnimationEnd(Animator animation) {
 //动画结束后改变状态
 mIsExplosionAnimStart = false;
 }
 });
 anim.start();
}

最后在onDraw中:

if (mIsExplosionAnimStart && mCurExplosionIndex < mExplosionDrawables.length) {
 //设置气泡爆炸图片的位置
 mExplosionRect.set((int) (mBubbleCenterX - mBubbleRadius), (int) (mBubbleCenterY - mBubbleRadius)
 , (int) (mBubbleCenterX + mBubbleRadius), (int) (mBubbleCenterY + mBubbleRadius));
 //根据当前进行到爆炸气泡的位置index来绘制爆炸气泡bitmap
 canvas.drawBitmap(mExplosionBitmaps[mCurExplosionIndex], null, mExplosionRect, mExplosionPaint);
}

9、在布局文件中使用该控件,并使用自定义属性:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
 xmlns:monkey="http://schemas.android.com/apk/res-auto"
 xmlns:tools="http://schemas.android.com/tools"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
 android:clipChildren="false"
 tools:context=".MainActivity">

 <com.monkey.dragpopview.DragBubbleView
 android:id="@+id/dragBubbleView"
 android:layout_width="wrap_content"
 android:layout_height="wrap_content"
 android:layout_centerInParent="true"
 monkey:bubbleColor="#ff0000"
 monkey:bubbleRadius="12dp"
 monkey:text="99+"
 monkey:textColor="#ffffff"
 monkey:textSize="12sp" />

</RelativeLayout>

其中 android:clipChildren=”false” 这个属性可以使根布局下的子控件超出本身控件范围的大小,加上这个属性就可以满屏幕随意拖拽而不必拘泥于它本身的大小了,炒鸡方便~

还有如果觉得在属性中设置消息个数不方便,需要在代码中动态获取数据并设置的话,只要在DragBubbleView中添加一个方法即可

public void setText(String text){
 mText = text;
 invalidate();
}

10、在MainActivity中:

 DragBubbleView dragBubbleView = (DragBubbleView) findViewById(R.id.dragBubbleView);
 dragBubbleView.setText("99+");
 dragBubbleView.setOnBubbleStateListener(new DragBubbleView.OnBubbleStateListener() {
 @Override
 public void onDrag() {
 Log.e("---> ", "拖拽气泡");
 }

 @Override
 public void onMove() {
 Log.e("---> ", "移动气泡");
 }

 @Override
 public void onRestore() {
 Log.e("---> ", "气泡恢复原来位置");
 }

 @Override
 public void onDismiss() {
 Log.e("---> ", "气泡消失");
 }
 });

总结

这次既练习了自定义View,还囊括了贝赛尔曲线,坐标的计算一定要画图,简单直观。

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

(0)

相关推荐

  • Android 未读消息的红点显示

    在很多APP里面,经常会看到未读消息的小红点,如下图: 这个功能用到的是一个控件,叫做BadgeView. BadgeView的用法很简单,直接把jar文件导入包中就可以使用,使用也很简单, badgeView = new BadgeView(this); //btn是控件 badgeView.setTargetView(btn); //设置相对位置 badgeView.setBadgeMargin(0, 5, 15, 0); //设置显示未读消息条数 badgeView.setBadgeCou

  • Android仿微信底部菜单栏功能显示未读消息数量

    底部菜单栏很重要,我看了一下很多应用软件都是用了底部菜单栏,这里使用了tabhost做了一种通用的(就是可以像微信那样显示未读消息数量的,虽然之前也做过但是layout下的xml写的太臃肿,这里去掉了很多不必要的层,个人看起来还是不错的,所以贴出来方便以后使用). 先看一下做出来之后的效果: 以后使用的时候就可以换成自己项目的图片和字体了,主框架不用变哈哈, 首先是要布局layout下xml文件 main.xml: <?xml version="1.0" encoding=&qu

  • 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仿QQ滑动弹出菜单标记已读、未读消息

    在上一篇<Android仿微信滑动弹出编辑.删除菜单效果.增加下拉刷新功能>里,已经带着大家学习如何使用SwipeMenuListView这一开源库实现滑动列表弹出菜单,接下来,将进一步学习,如何为不同的list item呈现不同的菜单,此处我们做一个实例:Android 高仿QQ滑动弹出菜单标记已读.未读消息,看下效果图: 1. 创建项目,并导入SwipeMenuListView类库 2. 创建消息实体bean: public class Msg { public int id; publi

  • Android 给应用程序的icon添加未读消息个数提示(红圈内数字)

    最近在做一个可以查看未读消息的功能,需要在界面中的Tab页的标签icon的右上角添加一个未读消息提示的功能. 先上个效果图出来,比较直观明白需求: 思路上似乎有两种:  1. 直接把底图和红圆圈的图片用相对布局进行排列,在代码中动态更改红圆中的TextView的数字,并且识别一下各种情况下红圆的显示或者隐藏.这种方法比较直观.  2. 采用canvas画出圆和数字. 由于项目采用的是Tab页的形式,其中的RadioButton不适合采用相对布局.故我采用了第二种方式.将绘制的过程写成了工具方法.

  • Android 桌面图标右上角显示未读消息数字

    背景: 在Android原生系统中,众所周知不支持桌面图标显示未读消息提醒的数字,虽然第三方控件BadgeView可以实现应用内的数字提醒.但对于系统的图标,特别是app的logo图标很难实现数字标志,即使是绘图的方式不断修改,但这种方式天生弊端,实用性很差.但幸运的是,一些强大的手机厂商(小米,三星,索尼)提供了私有的API,但也带来了难度,API的不同就意味着代码量的增加和兼容性问题更加突出. 现在我们来看看他们是如何实现的: 实现原理: 首先我们要明白 并不是应用本身处理对启动图标进行修改

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

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

  • Android贝塞尔曲线初步学习第一课

    贝塞尔曲线有一阶.二阶.三阶.N阶 一阶就是一条直线,有起点终点,没有控制点,对应方法就是 canvas.drawLine(float startX, float startY, float stopX, float stopY, @NonNull Paint paint) ; 二阶为曲线,有起点终点,一个控制点,对应方法就是 path.quadTo(float x1, float y1, float x2, float y2); 其中x1.y1为控制点坐标, x2.y2为终点坐标,效果如下:

  • Android贝塞尔曲线初步学习第三课 Android实现添加至购物车的运动轨迹

    不知上一节高仿QQ未读消息气泡大家还喜欢么,今天继续练习贝赛尔曲线,这一节我们通过贝赛尔曲线和属性动画估值器实现添加至购物车的运动轨迹,效果如下: 1.新建自定义View,重写构造方法,初始化Paint.Path: 2.确定起始点.终止点.控制点坐标,这里我们直接固定: @Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(w, h, oldw, oldh);

  • 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贝塞尔曲线实现直播点赞效果

    本文实例为大家分享了Android实现直播点赞效果的具体代码,供大家参考,具体内容如下 效果展示 原理分析 点赞效果最主要的难点和原理在于贝塞尔曲线动画的生成,我们通过图片主要讲解贝塞尔曲线动画 1.需要找到贝塞尔曲线的四个点 2.通过三级贝塞尔曲线的公式计算,获取贝塞尔曲线的轨迹路径点 3.通过设置点赞图片X,Y坐标,从而形成点赞的效果 实现步骤 1.初始化变量 //1.继承RelativeLayout public class ChristmasView extends RelativeLa

  • android贝塞尔曲线实现波浪效果

    本文实例为大家分享了android贝塞尔曲线实现波浪效果的具体代码,供大家参考,具体内容如下 因为手机录制gif不知道下什么软件好,所以暂时就先忽略效果图了 我在屏幕外多画了1.5个波浪,延伸至屏幕内.然后不断的循环,向右边移动.就有一种波浪的效果. 所以现在只需要画出左边的波长,然后再通过循环添加所有的波长即可. 第一个曲线已经确定了控制点和终点的坐标, 第二条曲线也可以很明显的看出来终点是在x轴的0点坐标,Y轴不变,而控制点是在负的波长的1/4的位置 有了上下曲线以后,其他的就可以直接通过循

  • Android贝塞尔曲线实现手指轨迹

    本文实例为大家分享了Android贝塞尔曲线实现手指轨迹的具体代码,供大家参考,具体内容如下 1.使用贝塞尔曲线前 MyView.java public class MyView extends View { // 实例一个路径对象 private Path mPath = new Path(); public MyView(Context context) { super(context); // TODO Auto-generated constructor stub } public My

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

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

  • Android贝塞尔曲线实现加入购物车抛物线动画

    本文实例为大家分享了Android贝塞尔曲线实现加入购物车抛物线动画的具体代码,供大家参考,具体内容如下 先上图看效果 步骤: a.确定动画的起终点b.在起终点之间使用二次贝塞尔曲线填充起终点之间的点的轨迹c.设置属性动画,ValueAnimator插值器,获取中间点的坐标d.将执行动画的控件的x.y坐标设为上面得到的中间点坐标e.开启属性动画f.当动画结束时的操作 获取控件在屏幕中的绝对坐标: int[] parentLocation = new int[2]; mRLayout.getLoc

随机推荐