亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > Android > 正文

Android仿qq消息拖拽效果

2019-10-21 21:26:31
字體:
來源:轉載
供稿:網友

本文實例為大家分享了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坐標及圓的半徑就可以進行繪制了,相對來說簡單些,對于固定圓來說,一開始有一個初始值,半徑是隨著距離的增大而減小,小到一定程度就消失;

@Overrideprotected 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); }}

繪制了拖拽圓和固定圓后,就需要將兩個圓連接起來,連接兩個圓的路徑的繪制就需要使用三階貝塞爾曲線來實現;

Android,qq消息,拖拽

看過去,需要求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消息拖拽效果

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VEVB武林網。


注:相關教程知識閱讀請移步到Android開發頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
成人在线中文字幕| 欧美一级淫片aaaaaaa视频| 中文字幕亚洲综合久久筱田步美| 亚洲欧美中文日韩v在线观看| 精品久久久久久国产| 91视频-88av| 久久99精品久久久久久琪琪| 欧亚精品中文字幕| 久久精品国产一区二区三区| 91在线观看免费高清完整版在线观看| 2019国产精品自在线拍国产不卡| 蜜月aⅴ免费一区二区三区| 欧美激情综合色| 久久99精品国产99久久6尤物| 97超级碰碰碰| 国产精品对白刺激| 国产成人高清激情视频在线观看| 久久久成人av| 姬川优奈aav一区二区| 日韩高清av在线| 日本中文字幕不卡免费| 亚洲欧美精品一区| 91精品国产九九九久久久亚洲| 97碰在线观看| 中文字幕亚洲无线码在线一区| 国产在线久久久| 成人久久久久爱| 国产精品久久77777| 97在线精品视频| 色婷婷综合久久久久中文字幕1| 欧洲亚洲在线视频| 欧美一区二区三区……| 国产日韩在线播放| 欧美乱大交xxxxx| 成人网在线视频| 一区二区欧美久久| 91久久综合亚洲鲁鲁五月天| 91精品国产综合久久久久久蜜臀| 欧美日韩成人网| 69av视频在线播放| 国产欧美日韩免费| 91深夜福利视频| 日韩欧美精品在线观看| 国语自产在线不卡| 欧美日韩ab片| 亚洲一区美女视频在线观看免费| 亚洲欧洲成视频免费观看| 国产精品久久久久9999| 久久久久久久亚洲精品| 九九视频直播综合网| 国产男女猛烈无遮挡91| 亚洲国产另类 国产精品国产免费| 不卡av在线网站| 欧美激情亚洲视频| 亚洲欧美综合图区| 亚洲午夜女主播在线直播| 欧美激情喷水视频| 欧美激情a∨在线视频播放| 久久国产精品影视| 精品无码久久久久久国产| 日韩欧美高清在线视频| 亚洲第一天堂无码专区| 欧美性xxxx极品高清hd直播| 久久99热精品| 精品国产区一区二区三区在线观看| 日韩av在线播放资源| 日韩免费av片在线观看| 91免费国产网站| 国产成人精品一区二区| 狠狠色狠狠色综合日日小说| 97成人精品区在线播放| 91黑丝在线观看| 国产精品爽黄69天堂a| 91免费看国产| 91成人性视频| 黄色一区二区三区| 亚洲福利视频二区| 亚洲97在线观看| 欧美激情第6页| 久久久久久噜噜噜久久久精品| 7777免费精品视频| 久久夜色精品国产亚洲aⅴ| 国产成人在线精品| 亚洲视频欧美视频| 国产欧美日韩精品丝袜高跟鞋| 国产视频福利一区| 亚洲成avwww人| 91av视频在线| 中文字幕精品国产| 欧美日韩免费一区| 一区二区欧美在线| 欧美一区二区大胆人体摄影专业网站| 欧美激情一区二区三区在线视频观看| 亚洲国产成人在线视频| 国产一区私人高清影院| 欧美激情亚洲视频| 懂色av影视一区二区三区| 92国产精品视频| 2020欧美日韩在线视频| 国产精品久久一| 亚洲综合自拍一区| 亚洲福利影片在线| 国产精品美女久久| 成人精品视频99在线观看免费| 欧美在线观看网址综合| 欧美日韩综合视频| 在线国产精品播放| 国产成人精品视频在线| 欧美中文字幕在线观看| 欧美日韩亚洲网| 成人精品久久av网站| 欧美大片欧美激情性色a∨久久| 亚洲free性xxxx护士hd| 国产欧美在线播放| 欧美日韩亚洲国产一区| 欧美韩国理论所午夜片917电影| 久久久精品999| 尤物yw午夜国产精品视频| 久久久亚洲网站| 欧美成人h版在线观看| 午夜精品久久久久久久男人的天堂| 欧美激情精品久久久久久久变态| 国产精品视频不卡| 在线免费观看羞羞视频一区二区| 中文字幕日韩精品在线观看| 日韩av中文在线| 色哟哟亚洲精品一区二区| 性欧美视频videos6一9| 久久久久久69| 国产91精品青草社区| 96pao国产成视频永久免费| 国产v综合ⅴ日韩v欧美大片| 日韩中文字幕亚洲| 日本最新高清不卡中文字幕| 久久久久久18| 亚洲人成电影网站色…| 国产精品美腿一区在线看| 国产精品久久色| 懂色av中文一区二区三区天美| 97视频在线观看网址| 亚洲免费精彩视频| 亚洲国产精品专区久久| 久久精品久久久久久国产 免费| 国产成人涩涩涩视频在线观看| 久久人人看视频| 国产一区二区三区精品久久久| 亚洲福利视频网| 91精品国产91久久久久久| 久久久久久久网站| 国产精品日韩欧美综合| 日韩在线视频线视频免费网站| 91九色综合久久| 久久99精品久久久久久噜噜| 亚洲人成电影在线播放| 一区二区在线视频播放| 日韩av中文字幕在线| 亚洲成人黄色在线| 国产精品日本精品| 韩国三级电影久久久久久| 亚洲精品国产欧美| 日韩精品福利网站| 98精品在线视频| 日韩视频免费在线观看| 国产成人综合精品在线|