Android animation - cute candle blowing animation

Android animation - cute candle blowing animation

1. Introduction

Recently I started writing some articles to record some of my previous small projects or View definitions. I still need to sort out the dusty things. This only completes part of the animation of burning and extinguishing, and there is no animation of why the flame is lit. I hope that students who are interested can continue to complete and share. Without further ado, let's take a look at these two cute little candles.

The candle held its breath and ignited the flame, but was blown out by the guy next to it 0^0. It still looks very cute. Looking at the picture, you should be able to think of how to implement it, custom View! By the way, how to do this process well? Let's follow the steps and take a look. The code is a little bit long, please be patient. Interested students can clone it from my GITHUB and read the code.

2. Process Implementation

Drawing and animation of candles

  • Based on object-oriented thinking, it is obvious that there are two candles here! In this case, we define a candle class with the basic properties of a candle.

  1. public abstract class ICandle {
  2.  
  3.  
  4. //Lower left coordinate of the candle bottom
  5.  
  6. protected int mCurX;
  7.  
  8. protected int mCurY;
  9.  
  10. //Candle width and height
  11.  
  12. protected int mCandleWidth;
  13.  
  14. protected int mCandleHeight;
  15.  
  16. //The left eye coordinates of the candle
  17.  
  18. protected Point mEyeLPoint;
  19.  
  20. //The right eye coordinates of the candle
  21.  
  22. protected Point mEyeRPoint;
  23.  
  24. //Candle eye radius
  25.  
  26. protected int mEyeRadius;
  27.  
  28. //Eye distance
  29.  
  30. protected int mEyeDevide;
  31.  
  32. //Body color
  33.  
  34. protected int mCandleColor;
  35.  
  36. //Stop the animation?
  37.  
  38. protected boolean mIsAnimStoping = false ;
  39.  
  40. //Candle wick coordinates
  41.  
  42. protected Point mCandlewickPoint;
  43.  
  44. public void initAnim(){
  45.  
  46. }
  47.  
  48. public void stopAnim(){
  49.  
  50. }
  51.  
  52. public void drawSelf(Canvas canvas){
  53.  
  54. }
  55.  
  56. }

There is also code corresponding to this candle, which is clear at a glance. The following methods may need to be explained: public void initAnim(), stopAnim() initializes the data needed to start and end the animation. The small candle will implement this method, drawSelf(Canvas canvas) passes in the canvas and then the candle draws itself.

Now it's time for us to take a look at the internal structure of the little candle, hiahiahiahia!

No, the candle is accompanied by flames! Let's look at the flames first, because the candle will burn itself later. +10086s

Flame

  • Let's take a look at our Fulham first

There doesn't seem to be anything wrong with it. First of all, the inner area is Flame, and the outer area is the light of humanity emitted by Mr. Flame burning himself and the scattered ashes (wipes tears manually).

Let's take a look at Flame's implementation. We'll analyze it step by step.

  1. private static   float CHANGE_FACTOR = 20;
  2.  
  3. private Paint mPaint;
  4.  
  5. private Path mPath;
  6.  
  7. //Lower left point coordinates
  8.  
  9. private int mCurX;
  10.  
  11. private int mCurY;
  12.  
  13. //flame width
  14.  
  15. private int mWidth;
  16.  
  17. //Flame height
  18.  
  19. private int mHeight;
  20.  
  21. //Record the initial height
  22.  
  23. private int mPreHeight;
  24.  
  25. //Record the initial width
  26.  
  27. private int mPreWidth;
  28.  
  29. //Change parameters of the Bezier curve control point at the top of the flame
  30.  
  31. private int mTopXFactor;
  32.  
  33. private int mTopYFactor;
  34.  
  35. //Used to realize the jitter of flame
  36.  
  37. private Random mRandom;
  38.  
  39. //Aura radius
  40.  
  41. private int mHaloRadius;
  42.  
  43. //Burning
  44.  
  45. private boolean mIsFiring;
  46.  
  47. //Whether to start and stop the animation
  48.  
  49. private boolean mIsStopAnim = false ;
  50.  
  51. private boolean mFlagStop = false ;
  52.  
  53. private LinearGradient mLinearGradient;
  54.  
  55. private RadialGradient mRadialGradient;
  56.  
  57. private ValueAnimator mFlameAnimator;
  58.  
  59. private ValueAnimator mHaloAnimator;

These are the parameters. The main thing is our animation implementation process, that is, our attribute animation ValueAnimator. There are two rendering classes here. I don’t know if you have used them. LinearGradient and RadialGradient. If you don’t know them, you can directly click on my blog to learn about them. LinearGradient draws flames, and RadialGradient draws divergent light.

I will not write the initialization process, you can look at this code. The main thing is how the small flame is drawn.

  1. mPaint.setStyle(Paint.Style.FILL);
  2.  
  3. mPaint.setShader(mLinearGradient);
  4.  
  5. mPath.reset();
  6.  
  7. mPath.moveTo(mCurX, mCurY);
  8.  
  9. mPath.quadTo(mCurX + mWidth / 2,
  10.  
  11. mCurY + mHeight / 3,
  12.  
  13. mCurX + mWidth, mCurY);
  14.  
  15. mPath.quadTo(mCurX + mWidth / 2 + ((1 - mRandom.nextFloat()) * CHANGE_FACTOR) + mTopXFactor,
  16.  
  17. mCurY - 2 * mHeight + mTopYFactor,
  18.  
  19. mCurX, mCurY);
  20.  
  21. canvas.drawPath(mPath, mPaint);

This is the drawing of flame. You can see that the quadratic Bezier curve is used here. Students who are not familiar with Bezier curves can also click this wave loading animation (Bezier curve) for a brief introduction. It was used in a water wave view. The drawing here is based on the rectangle in the previous picture. Let's take a look at this picture again (of course, it is an enhanced version of hiahia).

Then why is mRandom.nextFloat()) * CHANGE_FACTOR added to the x coordinate above? Think about it, the flame will swing left and right, so a random number is used to control the left and right swing.

  1. mFlameAnimator = ValueAnimator.ofFloat(0, 4).setDuration(4000);
  2.  
  3. mFlameAnimator.setRepeatCount(ValueAnimator.INFINITE);
  4.  
  5. mFlameAnimator.addUpdateListener(
  6.  
  7. new ValueAnimator.AnimatorUpdateListener() {
  8.  
  9. @Override
  10.  
  11. public void onAnimationUpdate(ValueAnimator animation) {
  12.  
  13. float zeroToOne = ( float ) animation.getAnimatedValue();
  14.  
  15. if (zeroToOne >= 1.0f && zeroToOne <= 1.2f) {
  16.  
  17. //The flame is burning
  18.  
  19. zeroToOne = 1.0f - 5 * (zeroToOne - 1.0f); //1-0
  20.  
  21. mHeight = ( int ) (mPreHeight * (1 - zeroToOne));
  22.  
  23. mIsFiring = true ;
  24.  
  25. } else if (zeroToOne >= 3.5f) {
  26.  
  27. if (mFlagStop) {
  28.  
  29. mFlameAnimator.cancel();
  30.  
  31. return ;
  32.  
  33. }
  34.  
  35. //The flame is blown out
  36.  
  37. zeroToOne = 2 * (zeroToOne - 3.5f);//0-2
  38.  
  39. mTopXFactor = ( int ) (-20 * zeroToOne);
  40.  
  41. mTopYFactor = ( int ) (160 * zeroToOne);
  42.  
  43. / mWidth = ( int ) (mPreWidth * (1 -zeroToOne));
  44.  
  45. mIsFiring = false ;
  46.  
  47. }
  48.  
  49. }
  50.  
  51. });

In 4 seconds, the flame has a series of activities. It moves up from the bottom with the wick, constantly changing the position of the flame. It is divided into two parts: the flame is lit and the flame is extinguished. From the code, we can see that when the flame is lit, mHeight slowly increases, and then there is a rising process. The other is when the flame is blown out. Because the height of the flame must remain the same when it is blown out, there is no need to change it. Instead, the two factors mTopXFactor and mTopYFactor are used to control the position of the flame. Well, now that the flame is there, the candle is burned to ashes and the tears are dry, and the light of life should also appear.

Drawing and animating the aperture is relatively simple

  1. mPaint.setStyle(Paint.Style.STROKE);
  2.  
  3. mPaint.setStrokeWidth(5);
  4.  
  5. mPaint.setShader(mRadialGradient);
  6.  
  7. canvas.drawCircle(mCurX + mWidth / 2,
  8.  
  9. mCurY - mHeight / 2, mHaloRadius, mPaint);
  10.  
  11. canvas.drawCircle(mCurX + mWidth / 2,
  12.  
  13. mCurY - mHeight / 2, mHaloRadius + 5, mPaint);
  14.  
  15. canvas.drawCircle(mCurX + mWidth / 2,
  16.  
  17. mCurY - mHeight / 2, mHaloRadius - 5, mPaint);

Only one parameter is changed here, mHaloRadius, which is the radius of the aperture. But don't forget that other parameters are also changing at the same time, but they are placed in mFlameAnimator.

Well, that’s the end of the introduction to Flame. There is still a long way to go. I have written so much but it is not finished yet. It reminds me of something an ancient person said, but it is not me.

I'm dying of exhaustion before I die of old age

  • FireCandle

This name is a bit strange, FireCandle, awesome! ICandle has been introduced before, now let's take a look at its implementation class, FireCandle.

As usual, I won't talk about initialization, let's look at the variables that should be there.

  1. private Paint mPaint; //Center X coordinate
  2. private int mCenterX; //Record initial width
  3. private int mPreWidth; //Record initial height
  4. private int mPreHeight; //Candle wick rotation angle
  5. private int mCandlewickDegrees = 0; private Flame mFlame; private boolean mIsFire = false ; private boolean mIsStateOnStart = false ; private boolean mIsStateOnEnd = false ; private boolean mFlagStop = false ; private ValueAnimator mCandlesAnimator;

The naming is quite standard, and you should know what it does at a glance.

Let's focus on the coordination between drawing and attribute animation. I won't look at the drawing (slap in the face at the speed of light). Let's look at the animation.

  1. mCandlesAnimator = ValueAnimator.ofFloat(0, 4).setDuration(4000);
  2.  
  3. mCandlesAnimator.addUpdateListener(
  4.  
  5. new ValueAnimator.AnimatorUpdateListener() {
  6.  
  7. @Override
  8.  
  9. public void onAnimationUpdate(ValueAnimator animation) {
  10.  
  11. float zeroToOne = ( float ) animation.getAnimatedValue();
  12.  
  13. if (zeroToOne <= 1.0f) {
  14.  
  15. //Candle wick pulls down
  16.  
  17. mIsFire = true ;
  18.  
  19. mCandleWidth = mPreWidth + ( int ) (zeroToOne * 40);
  20.  
  21. mCandleHeight = mPreHeight - ( int ) (zeroToOne * 30);
  22.  
  23. mCandlewickDegrees = ( int ) (-60 + (180 + 60) * zeroToOne);
  24.  
  25. refreshEyePosition();
  26.  
  27. } else if (zeroToOne <= 2.0f) {
  28.  
  29. zeroToOne = zeroToOne - 1.0f;
  30.  
  31. //Candle wick swings upward
  32.  
  33. if (zeroToOne <= 0.2f) {
  34.  
  35. zeroToOne = 1.0f - 5 * zeroToOne;
  36.  
  37. mIsFire = false ;
  38.  
  39. mCandleWidth = mPreWidth + ( int ) (zeroToOne * 40);
  40.  
  41. mCandleHeight = mPreHeight - ( int ) (zeroToOne * 30);
  42.  
  43. mCandlewickDegrees = ( int ) (180 * zeroToOne);
  44.  
  45. } else {
  46.  
  47. if (mFlameStateListener != null && !mIsStateOnStart) {
  48.  
  49. mFlameStateListener.flameStart();
  50.  
  51. mIsStateOnStart = true ;
  52.  
  53. }
  54.  
  55. mCandleWidth = mPreWidth;
  56.  
  57. mCandleHeight = mPreHeight;
  58.  
  59. mCandlewickDegrees = 0;
  60.  
  61. if (mFlagStop) {
  62.  
  63. mCandlesAnimator.cancel();
  64.  
  65. }
  66.  
  67. }
  68.  
  69. refreshEyePosition();
  70.  
  71. } else if (zeroToOne >= 3.5f) {
  72.  
  73. //The candle wick is blown crooked
  74.  
  75. zeroToOne = 2 * (zeroToOne - 3.5f);//0-1
  76.  
  77. mCandlewickDegrees = ( int ) (-60 * zeroToOne);
  78.  
  79. if (mFlameStateListener != null && !mIsStateOnEnd) {
  80.  
  81. mFlameStateListener.flameEnd();
  82.  
  83. mIsStateOnEnd = true ;
  84.  
  85. }
  86.  
  87. }
  88.  
  89. }
  90.  
  91. });

This process is a bit long, but it is not complicated at all. First, let's look at the little candle in the animation. At the beginning, it squatted with a fat red face, so mCandleWidth became larger and mCandleHeight became smaller. The wick at the back rotated at a large angle as it squatted. You can see how the wick rotates. Just change the coordinate system and it will work.

canvas.rotate(mCandlewickDegrees, mCenterX, mCurY - mCandleHeight); This method. The process of swinging up is the same, so I won't go into details. refreshEyePosition(); This method is used to change the eye position. It is used in two places, so it is slightly independent. Pay attention to the variable mIsFire. When there is no flame, do other drawing, such as red eyes, etc. Okay, okay, this is the end of the introduction to the small candle.

SecCandle

Big candles, handsome candles are the centerpieces. The actual drawing process is similar to that of small candles, so I won’t explain it here.

Drawing View and Controller Together

AnimControler

The function of this class is very simple. It draws the floor and assigns the calculated height and width to the two candles, and then controls the two candles to start animation separately.

  1. mFirCandle = new FirCandle(mRelativeX + mWidth / 6, mRelativeY + mHeight);
  2.  
  3. mFirCandle.initCandle(mFirCandleWidth, mFirCandleHeight);
  4.  
  5. mFirCandle.initAnim();
  6.  
  7. mSecCandle = new SecCandle(mRelativeX + mWidth / 2, mRelativeY + mHeight);
  8.  
  9. mSecCandle.initCandle(mSecCandleWidth, mSecCandleHeight - 80);
  10.  
  11. mSecCandle.initAnim();

******, is our View

  • CandlesAnimView
  1. //16ms refresh Canvas
  2.  
  3. mInvalidateAnimator = ValueAnimator.ofInt(0, 1).setDuration(16);
  4.  
  5. mInvalidateAnimator.setRepeatCount(ValueAnimator.INFINITE);
  6.  
  7. mInvalidateAnimator.addListener(new AnimatorListenerAdapter() {
  8.  
  9. @Override
  10.  
  11. public void onAnimationRepeat(Animator animation) {
  12.  
  13. invalidate();
  14.  
  15. }
  16.  
  17. });
  18.  
  19. mInvalidateAnimator.start();

The task of this attribute animation is to quickly refresh the interface so that the Candle animation can be displayed in time.

  1. @Override
  2. protected void onMeasure( int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(widthMeasureSpec, heightMeasureSpec); int width = measureDimension(WIDTH_DEFAULT * mDensity,
  3. widthMeasureSpec); int height = measureDimension(HEIGHT_DEFAULT *mDensity,
  4. heightMeasureSpec);
  5. setMeasuredDimension(width, height);
  6. } public   int measureDimension( int defaultSize, int measureSpec) { int result; int specMode = MeasureSpec.getMode(measureSpec); int specSize = MeasureSpec.getSize(measureSpec); if (specMode == MeasureSpec.EXACTLY) {
  7. result = specSize;
  8. } else {
  9. result = defaultSize; if (specMode == MeasureSpec.AT_MOST) {
  10. result = Math. min (result, specSize);
  11. }
  12. } return result;
  13. } @Override
  14. protected void onDraw(Canvas canvas) { if (!mIsInit) {
  15. initConfig();
  16. mIsInit = true ;
  17. }
  18. mAnimControler.drawMyView(canvas);
  19. }

You can see that *** called our controller in the view and passed the cavas.

***Tip: Have you noticed that the duration of each animation is the same?

three,***

Well, I have said so much about a simple view customization. This is my first time writing on Jianshu, I hope you can support me.

I hope you can put forward any suggestions and opinions.

<<:  Introduction to Android hardware acceleration principle and implementation

>>:  Summary of the use of global variables and local variables in Android

Recommend

Why can't the iPhone be made in the U.S.? Jobs gave the answer

If Jobs' attempt to manufacture Apple compute...

Is it necessary to refer to keywords to guide bidding in bidding promotion?

Now the Baidu backend has its own keyword bidding...

Marketing promotion hot spots calendar in May

As we enter May, the tenderness of late spring an...

Baidu Map Marketing allows customers to find you at a glance!

Baidu map marketing uses Baidu maps as a display ...

4 tips to enhance article resonance and create popular articles!

If you want an article on a public account to go ...

How to quickly attract “new customers” and “new fans”?

Time is money, so relying solely on long-termism ...

Why do startup products lack highly active users?

This article attempts to explain it from the thre...

What does Meilishuo do? Case analysis of SEO optimization of Meilishuo website!

Dongguan SEO Feng Chao came into contact with the...

Five major marketing tactics behind Alipay’s lucky draw, a must-read for SEM!

The screen was flooded with messages! I believe e...

The secret to increasing information flow conversion rate from 3% to 13% lies here!

The effect is poor. What went wrong? First, let’s...