<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
本文範例為大家分享了Android自定義View實現QQ訊息氣泡的具體程式碼,供大家參考,具體內容如下
效果圖:
原理:
控制元件原始碼:
public class DragView extends View { private int defaultZoomSize = 8; //初始化圓的大小 private int initRadius; //圓1的圓心位置 private PointF center1; private PointF center2; private PointF point1; private PointF point2; private PointF point3; private PointF point4; private int mWidth; private int mHeight; private float realZoomSize; private float currentRadius; private float minRadiusScale = 1 / 2f; private Paint paint; private Path path; private Bitmap bitmap; @DragStatus private int mDragStatus; public DragView(Context context) { this(context, null); } public DragView(Context context, @Nullable AttributeSet attrs) { this(context, attrs, 0); } public DragView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); paint = new Paint(); paint.setColor(Color.BLUE); paint.setStyle(Paint.Style.FILL); paint.setStrokeWidth(4); paint.setAntiAlias(true); path = new Path(); center1 = new PointF(); center2 = new PointF(); point1 = new PointF(); point2 = new PointF(); point3 = new PointF(); point4 = new PointF(); bitmap = BitmapFactory.decodeResource(context.getResources(), R.mipmap.icon_pot); initRadius = Math.min(bitmap.getWidth(), bitmap.getHeight()) / 2; Log.e("zhen", "解析bitmap: " + bitmap.getWidth() + " * " + bitmap.getHeight() + " * " + initRadius); } @Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(w, h, oldw, oldh); mWidth = w; mHeight = h; center1.set(mWidth / 2, mHeight / 2); Log.d("zhen", "圓心位置:x" + center1.x + " y: " + center1.y); } private boolean isSelected = false; @Override public boolean onTouchEvent(MotionEvent event) { float x = event.getX(); float y = event.getY(); switch (event.getAction()) { case MotionEvent.ACTION_DOWN: if (Math.sqrt(Math.pow(x - center1.x, 2) + Math.pow(y - center1.y, 2)) < initRadius && mDragStatus == DragStatus.NORMAL) { inAnimation = false; isSelected = true; Log.e("zhen", "選中狀態"); } break; case MotionEvent.ACTION_MOVE: if (isSelected) { // Log.d("zhen", "拖動距離: " + dragDistance); if (mDragStatus != DragStatus.DRAG_BACK && mDragStatus != DragStatus.DRAG_TO) { mDragStatus = DragStatus.DRAG_MOVE; center2.set(x, y); float dragDistance = (float) (Math.sqrt(Math.pow(center2.x - center1.x, 2) + Math.pow(center2.y - center1.y, 2))); //多少倍圓的大小 realZoomSize = dragDistance / initRadius; invalidate(); } } break; case MotionEvent.ACTION_UP: if (isSelected) { if (realZoomSize <= defaultZoomSize) { //回彈,改變center2.x, center2.y直到等於center1.x, center1.y doAnimation(DragStatus.DRAG_BACK, center2, center1); } } isSelected = false; break; } return true; } @Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); //圓的半徑改變 currentRadius = initRadius * (1 + (minRadiusScale - 1) / defaultZoomSize * realZoomSize); if (realZoomSize > defaultZoomSize) { //圓縮小為一半,去往目的地,就應該消失了 doAnimation(DragStatus.DRAG_TO, center1, center2); } //中間矩形 // paint.setColor(Color.BLACK); float angle = (float) Math.atan((center2.y - center1.y) / (center2.x - center1.x)); float sinValue; float cosValue; float controlX; float controlY; sinValue = (float) Math.abs((currentRadius * Math.sin(angle))); cosValue = (float) Math.abs((currentRadius * Math.cos(angle))); controlX = (center1.x + center2.x) / 2; controlY = (center1.y + center2.y) / 2; point1.set(center1.x - sinValue, center1.y - cosValue); point2.set(center1.x + sinValue, center1.y + cosValue); point3.set(center2.x - sinValue, center2.y - cosValue); point4.set(center2.x + sinValue, center2.y + cosValue); path.reset(); switch (mDragStatus) { case DragStatus.NORMAL: currentRadius = initRadius; //原始圖片 canvas.drawBitmap(bitmap, center1.x - initRadius, center1.y - initRadius, paint); //起始位置的圓 // paint.setColor(Color.RED); // canvas.drawCircle(center1.x, center1.y, currentRadius, paint); break; case DragStatus.DRAG_MOVE: //拖動過程中 path.moveTo(point1.x, point1.y); path.lineTo(point2.x, point2.y); path.quadTo(controlX, controlY, point4.x, point4.y); path.lineTo(point3.x, point3.y); path.quadTo(controlX, controlY, point1.x, point1.y); canvas.drawPath(path, paint); //起始位置的圓 paint.setColor(Color.RED); canvas.drawCircle(center1.x, center1.y, currentRadius, paint); //結束位置的圓 // paint.setColor(Color.BLUE); // canvas.drawCircle(center2.x, center2.y, currentRadius, paint); //原始圖片 canvas.drawBitmap(bitmap, center2.x - initRadius, center2.y - initRadius, paint); break; case DragStatus.DRAG_BACK: //改變center2.x, center2.y直到等於center1.x, center1.y path.reset(); path.moveTo(point1.x, point1.y); path.quadTo(center2.x, center2.y, point2.x, point2.y); canvas.drawPath(path, paint); //起始位置的圓 // paint.setColor(Color.RED); // canvas.drawCircle(center1.x, center1.y, currentRadius, paint); //原始圖片 canvas.drawBitmap(bitmap, center1.x - initRadius, center1.y - initRadius, paint); break; case DragStatus.DRAG_TO: //改變center1.x, center1.y,直到等於center2.x, center2.y path.reset(); path.moveTo(point3.x, point3.y); path.quadTo(center1.x, center1.y, point4.x, point4.y); canvas.drawPath(path, paint); // //起始位置的圓 // paint.setColor(Color.RED); // canvas.drawCircle(center1.x, center1.y, currentRadius, paint); // //結束位置的圓 // paint.setColor(Color.BLUE); // canvas.drawCircle(center2.x, center2.y, currentRadius, paint); //原始圖片 canvas.drawBitmap(bitmap, center2.x - initRadius, center2.y - initRadius, paint); break; } // Log.d("zhen", "dragStatus: " + mDragStatus + " 圓1:" + center1 + " 圓2:" + center2 + " 半徑: " + currentRadius); // Log.w("zhen", "dragStatus: " + mDragStatus + " point3:" + point3 + " point4" + point4 + " sinValue " + sinValue + " cosValue " + cosValue); Log.w("zhen", "dragStatus: " + mDragStatus + " 圓1:" + center1 + " 圓2:" + center2 + " 半徑: " + currentRadius); } int i = 0; private boolean inAnimation = false; private void doAnimation(int dragStatus, final PointF startPoint, final PointF endPoint) { if (inAnimation) return; inAnimation = true; final int step = 10; final float stepx = (endPoint.x - startPoint.x) / step; final float stepy = (endPoint.y - startPoint.y) / step; i = 1; mDragStatus = dragStatus; Log.d("zhen", "dragStatus: " + mDragStatus + " startPoint:" + startPoint + " endPoint:" + endPoint + " stepx: " + stepy + " stepx: " + stepy); new Thread(new Runnable() { @Override public void run() { while (i <= step) { startPoint.x += stepx; startPoint.y += stepy; postInvalidate(); i++; try { Thread.sleep(50); } catch (InterruptedException e) { e.printStackTrace(); } } mDragStatus = DragStatus.NORMAL; invalidate(); Log.e("zhen", "恢復為可拖動狀態"); } }).start(); } @IntDef({DragStatus.DRAG_MOVE, DragStatus.DRAG_TO, DragStatus.DRAG_BACK}) public @interface DragStatus { int NORMAL = 0; //拖動中 int DRAG_MOVE = 1; // int DRAG_TO = 2; //回彈 int DRAG_BACK = 3; } }
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援it145.com。
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45