日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Android實現笑臉進度加載動畫

瀏覽:22日期:2022-09-19 08:13:21

最近看到豆瓣的笑臉loading很有意思,看一張效果圖:

Android實現笑臉進度加載動畫

下面分析一下如何實現這樣的效果:

1、默認狀態是一張笑臉的狀態(一個嘴巴,兩個眼睛,默認狀態)

2、開始旋轉,嘴巴追上眼睛(合并狀態)

3、追上以后自轉一周(自轉狀態)

4、然后逐漸釋放眼睛(分離狀態)

5、回到初始笑臉狀態(默認狀態)

一、默認狀態

首先需要確定好嘴巴和眼睛的初始位置,我這里的初始化嘴巴是一個半圓,在橫軸下方。眼睛分別與橫軸夾角60度,如下圖:

Android實現笑臉進度加載動畫

這兩部分可以使用pathMeasure,我這里使用最簡單的兩個api:canvas.drawArc()和canvas.drawPoint()。

1、畫嘴巴

//畫起始笑臉canvas.drawArc(-radius, -radius, radius, radius, startAngle, swipeAngle, false,facePaint);

這里的startAngle初始值為0,swiperAngle為180,半徑radius為40。

2、畫眼睛

(1)初始化眼睛坐標

/** * 初始化眼睛坐標 */ private void initEyes() {//默認兩個眼睛坐標位置 角度轉弧度leftEyeX = (float) (-radius * Math.cos(eyeStartAngle * Math.PI / 180));leftEyeY = (float) (-radius * Math.sin(eyeStartAngle * Math.PI / 180));rightEyeX = (float) (radius * Math.cos(eyeStartAngle * Math.PI / 180));rightEyeY = (float) (-radius * Math.sin(eyeStartAngle * Math.PI / 180)); }

注意:需要將角度轉弧度

(2)開始畫眼睛

//畫起始眼睛 canvas.drawPoint(leftEyeX, leftEyeY, eyePaint); canvas.drawPoint(rightEyeX, rightEyeY, eyePaint);二、合并狀態

這個狀態可以分為兩部分

嘴巴的旋轉 眼睛的旋轉1、嘴巴的旋轉

開啟動畫

faceLoadingAnimator = ValueAnimator.ofFloat(0, 1).setDuration(1000); faceLoadingAnimator.setInterpolator(new AccelerateDecelerateInterpolator()); faceLoadingAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { @Override public void onAnimationUpdate(ValueAnimator animation) {faceValue = (float) animation.getAnimatedValue();invalidate(); }});//動畫延遲500ms啟動faceLoadingAnimator.setStartDelay(200);faceLoadingAnimator.addListener(new Animator.AnimatorListener() { @Override public void onAnimationStart(Animator animation) { } @Override public void onAnimationEnd(Animator animation) {//恢復起始狀態currentStatus = smileStatus; } @Override public void onAnimationCancel(Animator animation) { } @Override public void onAnimationRepeat(Animator animation) { }});

動畫執行時間1s,記錄動畫當前執行進度值,存放在faceValue中。當動畫執行結束的時候,需要將狀態恢復到默認狀態,調用invalidate的時候,進入onDraw()方法,開始重新繪制嘴巴。

//記錄時刻的旋轉角度startAngle = faceValue * 360;//追上右邊眼睛if (startAngle >= 120 + startAngle / 2) { canvas.drawArc(-radius, -radius, radius, radius, startAngle, swipeAngle, false, facePaint); //開始自轉一圈 mHandler.sendEmptyMessage(2); //此時記錄自轉一圈起始的角度 circleStartAngle = 120 + startAngle / 2;} else { //追眼睛的過程 canvas.drawArc(-radius, -radius, radius, radius, startAngle, swipeAngle, false, facePaint);}

這里的每次旋轉角度為startAngle。當完全追趕上右側眼睛的時候,開始執行自轉一周,并停止當前動畫。

2、眼睛的旋轉

眼睛的開始旋轉速度明顯是慢于嘴巴的旋轉速度,所以每次的旋轉速度可以設置為嘴巴的一半

//畫左邊眼睛 ,旋轉的角度設置為笑臉旋轉角度的一半,這樣笑臉才能追上眼睛 leftEyeX = (float) (-radius * Math.cos((60 + startAngle / 2) * Math.PI / 180)); leftEyeY = (float) (-radius * Math.sin((60 + startAngle / 2) * Math.PI / 180)); canvas.drawPoint(leftEyeX, leftEyeY, eyePaint); //畫右邊眼睛 ,旋轉的角度設置為笑臉旋轉角度的一半,這樣笑臉才能追上眼睛 rightEyeX = (float) (radius * Math.cos((60 - startAngle / 2) * Math.PI / 180)); rightEyeY = (float) (-radius * Math.sin((60 - startAngle / 2) * Math.PI / 180)); canvas.drawPoint(rightEyeX, rightEyeY, eyePaint);三、自轉狀態1、開啟動畫

circleAnimator = ValueAnimator.ofFloat(0, 1).setDuration(1000);circleAnimator.setInterpolator(new LinearInterpolator());circleAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { @Override public void onAnimationUpdate(ValueAnimator animation) {circleValue = (float) animation.getAnimatedValue();invalidate(); }});circleAnimator.addListener(new Animator.AnimatorListener() { @Override public void onAnimationStart(Animator animation) { } @Override public void onAnimationEnd(Animator animation) {mHandler.sendEmptyMessage(3); } @Override public void onAnimationCancel(Animator animation) { } @Override public void onAnimationRepeat(Animator animation) { }});2、重新繪制

canvas.drawArc(-radius, -radius, radius, radius,circleStartAngle + circleValue * 360,swipeAngle, false, facePaint);四、分離狀態

主要的注意點就是眼睛的旋轉角度設置為嘴巴旋轉角度的2倍,這樣才會達到眼睛超過嘴巴的效果,主要的旋轉代碼如下:

startAngle = faceValue * 360;//判斷當前笑臉的起點是否已經走過260度 (吐出眼睛的角度,角度可以任意設置)if (startAngle >= splitAngle) { //畫左邊眼睛 ,旋轉的角度設置為笑臉旋轉角度的2倍,這樣眼睛才能快于笑臉旋轉速度 leftEyeX = (float) (-radius * Math.cos((eyeStartAngle + startAngle * 2) * Math.PI / 180)); leftEyeY = (float) (-radius * Math.sin((eyeStartAngle + startAngle * 2) * Math.PI / 180)); canvas.drawPoint(leftEyeX, leftEyeY, eyePaint); //畫右邊眼睛 ,旋轉的角度設置為笑臉旋轉角度的2倍,這樣眼睛才能快于笑臉旋轉速度 rightEyeX = (float) (radius * Math.cos((eyeStartAngle - startAngle * 2) * Math.PI / 180)); rightEyeY = (float) (-radius * Math.sin((eyeStartAngle - startAngle * 2) * Math.PI / 180)); canvas.drawPoint(rightEyeX, rightEyeY, eyePaint);}//畫笑臉canvas.drawArc(-radius, -radius, radius, radius, startAngle, swipeAngle,false, facePaint);最后附上完整代碼

public class FaceView2 extends View { //圓弧半徑 private int radius = 40; //圓弧畫筆寬度 private float paintWidth = 15; //笑臉狀態(一個臉,兩個眼睛) private final int smileStatus = 0; //加載狀態 合并眼睛,旋轉 private final int loadingStatus = 1; //合并完成 轉一圈 private final int circleStatus = 2; //轉圈完成 吐出眼睛 private final int splitStatus = 3; //當前狀態 private int currentStatus = smileStatus; //笑臉畫筆 private Paint facePaint; //眼睛畫筆 private Paint eyePaint; //笑臉開始角度 private float startAngle; //笑臉弧度 private float swipeAngle; //左側眼睛起點x軸坐標 private float leftEyeX = 0; //左側眼睛起點y軸坐標 private float leftEyeY = 0; //右側眼睛起點x軸坐標 private float rightEyeX; //右側眼睛起點y軸坐標 private float rightEyeY; //一開始默認狀態笑臉轉圈動畫 private ValueAnimator faceLoadingAnimator; //吞并完成后,自轉一圈動畫 private ValueAnimator circleAnimator; //faceLoadingAnimator動畫進度值 private float faceValue; //circleAnimator動畫進度值 private float circleValue; //記錄開始自轉一圈的起始角度 private float circleStartAngle; //吐出眼睛的角度 private float splitAngle; private float initStartAngle; //眼睛起始角度 private float eyeStartAngle = 60; public FaceView2(Context context) {this(context, null); } public FaceView2(Context context, AttributeSet attrs) {this(context, attrs, 0); } public FaceView2(Context context, AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);//自定義屬性TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.FaceView2,defStyleAttr, 0);initStartAngle = typedArray.getFloat(R.styleable.FaceView2_startAngle, 0);swipeAngle = typedArray.getFloat(R.styleable.FaceView2_swipeAngle, 180);splitAngle = typedArray.getFloat(R.styleable.FaceView2_splitAngle, 260);typedArray.recycle();startAngle = initStartAngle;eyeStartAngle += startAngle;initEyes();initPaint();//開始默認動畫initAnimator(); } /** * 初始化畫筆 */ private void initPaint() {//初始化畫筆facePaint = new Paint();facePaint.setStrokeWidth(paintWidth);facePaint.setColor(Color.RED);facePaint.setAntiAlias(true);facePaint.setStyle(Paint.Style.STROKE);facePaint.setStrokeCap(Paint.Cap.ROUND);eyePaint = new Paint();eyePaint.setStrokeWidth(paintWidth);eyePaint.setColor(Color.RED);eyePaint.setAntiAlias(true);eyePaint.setStyle(Paint.Style.STROKE);eyePaint.setStrokeCap(Paint.Cap.ROUND); } /** * 初始化眼睛坐標 */ private void initEyes() {//默認兩個眼睛坐標位置 角度轉弧度leftEyeX = (float) (-radius * Math.cos(eyeStartAngle * Math.PI / 180));leftEyeY = (float) (-radius * Math.sin(eyeStartAngle * Math.PI / 180));rightEyeX = (float) (radius * Math.cos(eyeStartAngle * Math.PI / 180));rightEyeY = (float) (-radius * Math.sin(eyeStartAngle * Math.PI / 180)); } private Handler mHandler = new Handler(new Handler.Callback() {@RequiresApi(api = Build.VERSION_CODES.KITKAT)@Overridepublic boolean handleMessage(Message msg) { switch (msg.what) {case 1: //啟動一開始笑臉轉圈動畫,并且開始合并眼睛 currentStatus = loadingStatus; faceLoadingAnimator.start(); break;case 2: //暫停眼睛和笑臉動畫 currentStatus = circleStatus; faceLoadingAnimator.pause(); //啟動笑臉自轉一圈動畫 circleAnimator.start(); break;case 3: //恢復笑臉轉圈動畫,并且開始分離眼睛 currentStatus = splitStatus; circleAnimator.cancel(); faceLoadingAnimator.resume(); invalidate(); break; } return false;} }); @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP) @Override protected void onDraw(Canvas canvas) {super.onDraw(canvas);//畫布移到中間canvas.translate(getWidth() / 2, getHeight() / 2);switch (currentStatus) { //起始狀態 case smileStatus://起始角度為0startAngle = initStartAngle;//畫起始笑臉canvas.drawArc(-radius, -radius, radius, radius, startAngle, swipeAngle, false,facePaint);//重置起始眼睛坐標initEyes();//畫起始眼睛canvas.drawPoint(leftEyeX, leftEyeY, eyePaint);canvas.drawPoint(rightEyeX, rightEyeY, eyePaint);//更改狀態,進行笑臉合并眼睛mHandler.sendEmptyMessage(1);break; //合并狀態 case loadingStatus://記錄時刻的旋轉角度startAngle = faceValue * 360;//追上右邊眼睛if (startAngle >= 120 + startAngle / 2) { canvas.drawArc(-radius, -radius, radius, radius, startAngle, swipeAngle, false, facePaint); //開始自轉一圈 mHandler.sendEmptyMessage(2); //此時記錄自轉一圈起始的角度 circleStartAngle = 120 + startAngle / 2;} else { //追眼睛的過程 canvas.drawArc(-radius, -radius, radius, radius, startAngle, swipeAngle, false, facePaint);}//畫左邊眼睛 ,旋轉的角度設置為笑臉旋轉角度的一半,這樣笑臉才能追上眼睛leftEyeX = (float) (-radius * Math.cos((60 + startAngle / 2) * Math.PI / 180));leftEyeY = (float) (-radius * Math.sin((60 + startAngle / 2) * Math.PI / 180));canvas.drawPoint(leftEyeX, leftEyeY, eyePaint);//畫右邊眼睛 ,旋轉的角度設置為笑臉旋轉角度的一半,這樣笑臉才能追上眼睛rightEyeX = (float) (radius * Math.cos((60 - startAngle / 2) * Math.PI / 180));rightEyeY = (float) (-radius * Math.sin((60 - startAngle / 2) * Math.PI / 180));canvas.drawPoint(rightEyeX, rightEyeY, eyePaint);break; //自轉一圈狀態 circleValue * 360 為旋轉角度 case circleStatus:canvas.drawArc(-radius, -radius, radius, radius,circleStartAngle + circleValue * 360,swipeAngle, false, facePaint);break; //笑臉眼睛分離狀態 case splitStatus:startAngle = faceValue * 360;//判斷當前笑臉的起點是否已經走過260度 (吐出眼睛的角度,角度可以任意設置)if (startAngle >= splitAngle) { //畫左邊眼睛 ,旋轉的角度設置為笑臉旋轉角度的2倍,這樣眼睛才能快于笑臉旋轉速度 leftEyeX = (float) (-radius * Math.cos((eyeStartAngle + startAngle * 2) * Math.PI / 180)); leftEyeY = (float) (-radius * Math.sin((eyeStartAngle + startAngle * 2) * Math.PI / 180)); canvas.drawPoint(leftEyeX, leftEyeY, eyePaint); //畫右邊眼睛 ,旋轉的角度設置為笑臉旋轉角度的2倍,這樣眼睛才能快于笑臉旋轉速度 rightEyeX = (float) (radius * Math.cos((eyeStartAngle - startAngle * 2) * Math.PI / 180)); rightEyeY = (float) (-radius * Math.sin((eyeStartAngle - startAngle * 2) * Math.PI / 180)); canvas.drawPoint(rightEyeX, rightEyeY, eyePaint);}//畫笑臉canvas.drawArc(-radius, -radius, radius, radius, startAngle, swipeAngle,false, facePaint);break;} } /** * 初始化動畫 */ private void initAnimator() {faceLoadingAnimator = ValueAnimator.ofFloat(0, 1).setDuration(1000);faceLoadingAnimator.setInterpolator(new AccelerateDecelerateInterpolator());faceLoadingAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { @Override public void onAnimationUpdate(ValueAnimator animation) {faceValue = (float) animation.getAnimatedValue();invalidate(); }});//動畫延遲500ms啟動faceLoadingAnimator.setStartDelay(200);faceLoadingAnimator.addListener(new Animator.AnimatorListener() { @Override public void onAnimationStart(Animator animation) { } @Override public void onAnimationEnd(Animator animation) {//恢復起始狀態currentStatus = smileStatus; } @Override public void onAnimationCancel(Animator animation) { } @Override public void onAnimationRepeat(Animator animation) { }});circleAnimator = ValueAnimator.ofFloat(0, 1).setDuration(1000);circleAnimator.setInterpolator(new LinearInterpolator());circleAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() { @Override public void onAnimationUpdate(ValueAnimator animation) {circleValue = (float) animation.getAnimatedValue();invalidate(); }});circleAnimator.addListener(new Animator.AnimatorListener() { @Override public void onAnimationStart(Animator animation) { } @Override public void onAnimationEnd(Animator animation) {mHandler.sendEmptyMessage(3); } @Override public void onAnimationCancel(Animator animation) { } @Override public void onAnimationRepeat(Animator animation) { }}); }}

自定義屬性

<declare-styleable name='FaceView2'><attr name='startAngle' format='dimension' /><attr name='swipeAngle' format='dimension' /><attr name='splitAngle' format='dimension' /></declare-styleable>

布局文件中使用

<com.example.viewdemo.FaceView2 android:layout_width='match_parent' android:layout_height='match_parent'/>

完整代碼都在上面啦.

到這里就結束啦.

以上就是Android實現笑臉進度加載動畫的詳細內容,更多關于Android 笑臉進度加載的資料請關注好吧啦網其它相關文章!

標簽: Android
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
麻豆精品99| 青青草国产成人99久久| 国产毛片久久久| 日韩精品免费观看视频| 亚洲制服一区| 亚洲激情久久| 亚洲一区欧美激情| 麻豆成人在线| 亚洲日韩中文字幕一区| 亚洲欧美网站在线观看| 日本91福利区| 国产欧美一区二区三区国产幕精品| 日本一区福利在线| 欧美在线91| 麻豆一区二区三区| 日本在线啊啊| 久久精品国产www456c0m| 欧美午夜不卡| 婷婷亚洲精品| 精品国内亚洲2022精品成人| 岛国av在线播放| 久久中文亚洲字幕| 蜜桃av一区| 18国产精品| 日韩伦理福利| 国产精品日本| 欧美日一区二区在线观看| 免费看久久久| 999国产精品| 亚洲最新av| 麻豆久久久久久| 日韩电影二区| 亚洲欧美日韩国产综合精品二区| 久久久久久久久成人| 色天使综合视频| 丝袜亚洲精品中文字幕一区| 国产精品天堂蜜av在线播放| 日韩中文字幕高清在线观看| 久久国产99| 国产精品伦一区二区| 女生影院久久| 免费视频久久| 久久女人天堂| 最新日韩av| 国产欧美日韩影院| 三上悠亚国产精品一区二区三区 | 欧美一区二区三区久久| 福利一区二区| 亚洲欧美网站| 精品欧美日韩精品| 亚洲一区二区三区高清| 国产精品一国产精品k频道56| 夜鲁夜鲁夜鲁视频在线播放| 久久亚洲图片| 国产在视频一区二区三区吞精| 999在线观看精品免费不卡网站| 国产欧美日韩一级| 久久中文字幕av一区二区不卡| 日本精品久久| 久久久精品日韩| 久久亚洲图片| 日韩综合一区| 激情欧美亚洲| 麻豆精品视频在线观看视频| 久久99伊人| 狠狠躁少妇一区二区三区| 亚洲不卡视频| 国内精品福利| 欧美国产不卡| 日av在线不卡| 亚洲不卡系列| 欧美国产日韩电影| 蜜桃伊人久久| 久久精品观看| 精品国产中文字幕第一页| 免费日韩一区二区| 久久国产日韩| 国产成人精品一区二区三区视频| 亚洲香蕉视频| 久久麻豆精品| 国产成人久久精品麻豆二区| 欧美日韩18| 免费精品视频最新在线| 99久久久久久中文字幕一区| 久久精品免费看| 欧美日韩午夜| 综合激情一区| 午夜国产一区二区| 麻豆视频在线观看免费网站黄 | 婷婷中文字幕一区| 国产精品99久久精品| 91成人在线网站| 男人的天堂久久精品| 久久香蕉精品| 伊人久久大香线蕉av超碰演员| 中文字幕在线官网| 欧美91在线| 国产探花在线精品| 日韩专区视频网站| 蜜臀av免费一区二区三区| 国产h片在线观看| 国产黄色精品| 日韩欧乱色一区二区三区在线| 一本一道久久a久久精品蜜桃| 在线亚洲人成| 麻豆网站免费在线观看| 国产成人精品福利| 久久精品亚洲一区二区| 国产精品手机在线播放| 欧美精品国产一区| 日本免费新一区视频| 中文字幕一区二区三区在线视频| 99国产精品久久久久久久成人热 | 国产精品一二| 欧美日韩亚洲三区| 国产日韩精品视频一区二区三区| 午夜天堂精品久久久久| 中文字幕一区二区三区四区久久| 热久久国产精品| 免费在线视频一区| 蜜桃久久精品一区二区| 视频一区二区中文字幕| 丝袜美腿高跟呻吟高潮一区| 老鸭窝亚洲一区二区三区| 午夜一区在线| 中文无码久久精品| 色综合视频一区二区三区日韩 | 国产伦精品一区二区三区在线播放| 日韩欧美精品一区二区综合视频| 深夜福利一区| 日韩1区2区日韩1区2区| 日韩精选在线| 日本中文字幕不卡| 日韩精品国产欧美| 久久激情综合网| 国产精品九九| 荡女精品导航| 亚洲成人不卡| 制服诱惑一区二区| 亚洲一区欧美二区| 色狠狠一区二区三区| 欧美日韩亚洲一区三区| 美女精品视频在线| 天堂√中文最新版在线| 亚洲激情五月| 日韩精品欧美成人高清一区二区| 国产精品一区二区三区av| 久久亚洲资源中文字| 樱桃视频成人在线观看| 一区二区视频欧美| 日韩精品免费视频一区二区三区| 欧美极品中文字幕| 亚洲伦乱视频| 亚洲一区激情| 欧美一区91| 久久精品系列| 激情六月综合| 视频精品一区| 精品中文字幕一区二区三区 | 国产成人精品一区二区三区视频 | 日韩专区一卡二卡| 日韩在线黄色| 麻豆91小视频| 99热精品久久| 亚洲精品免费观看| 美女毛片一区二区三区四区最新中文字幕亚洲| 岛国精品一区| 欧美特黄a级高清免费大片a级| 日本高清久久| 91综合网人人| 久久午夜精品| 久久精品九色| 午夜日本精品| 国产精品视频一区视频二区| 国产精品原创| 在线看片日韩| 另类小说一区二区三区| 狠狠色综合网| 美女免费视频一区| 国产视频欧美| 国产精品7m凸凹视频分类| 日本欧美在线| 欧美日韩中文字幕一区二区三区| 日韩精彩视频在线观看| 久久夜夜操妹子| 欧美日韩中出| 五月天久久网站| 久久精品二区亚洲w码| 噜噜噜躁狠狠躁狠狠精品视频| 久久中文在线| 欧美在线资源| 精品日产乱码久久久久久仙踪林| 亚洲成人免费| 精品一区二区三区中文字幕视频| 伊人影院久久| 精品国产18久久久久久二百| 三级欧美在线一区| 伊人久久高清| 国产精品观看| 美女久久一区|