Custom View-Imitate the reward button of the Hupu live broadcast game interface

Custom View-Imitate the reward button of the Hupu live broadcast game interface

As a senior basketball fan, I often use Hupu app to watch live games. Later, I noticed that there are two buttons in the lower right corner of the text live broadcast interface. You can send Hupu coins during the live broadcast to cheer for the team you support. The specific effect is shown in the figure below:

I personally think it's fun, so I decided to implement this button myself. Without further ado, let's take a look at the effect of the implementation:

This effect looks similar to popupwindow, but I use custom view to achieve it. Let me explain the process below.

First of all, from the effect of Hupu, you can see that its two buttons float above the entire interface, so it needs to be used in conjunction with FrameLayout. Therefore, I let its width follow the screen size and the height be fixed according to the dpi. Its actual size is like this:

In addition, when the view is initialized, we can see that it can be divided into three parts: the background circle, the text inside the circle, and the number above the circle. Therefore, under normal circumstances, you only need to draw these three parts in the onDraw method. First, prepare the custom attributes, brushes, and initialization data in the initialization method:

  1. private void init(Context context, AttributeSet attrs) {
  2. //Get custom properties
  3. TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.HoopView);
  4. mThemeColor = typedArray.getColor(R.styleable.HoopView_theme_color, Color.YELLOW);
  5. mText = typedArray.getString(R.styleable.HoopView_text);
  6. mCount = typedArray.getString(R.styleable.HoopView_count);
  7.  
  8. mBgPaint = new Paint();
  9. mBgPaint.setAntiAlias( true );
  10. mBgPaint.setColor(mThemeColor);
  11. mBgPaint.setAlpha(190);
  12. mBgPaint.setStyle(Paint.Style.FILL);
  13.  
  14. mPopPaint = new Paint();
  15. mPopPaint.setAntiAlias( true );
  16. mPopPaint.setColor(Color.LTGRAY);
  17. mPopPaint.setAlpha(190);
  18. mPopPaint.setStyle(Paint.Style.FILL_AND_STROKE);
  19.  
  20. mTextPaint = new TextPaint();
  21. mTextPaint.setAntiAlias( true );
  22. mTextPaint.setColor(mTextColor);
  23. mTextPaint.setTextSize(context.getResources().getDimension(R.dimen.hoop_text_size));
  24.  
  25. mCountTextPaint = new TextPaint();
  26. mCountTextPaint.setAntiAlias( true );
  27. mCountTextPaint.setColor(mThemeColor);
  28. mCountTextPaint.setTextSize(context.getResources().getDimension(R.dimen.hoop_count_text_size));
  29.  
  30. typedArray.recycle();
  31.  
  32. mBigRadius = context.getResources().getDimension(R.dimen.hoop_big_circle_radius);
  33. mSmallRadius = context.getResources().getDimension(R.dimen.hoop_small_circle_radius);
  34. margin = ( int ) context.getResources().getDimension(R.dimen.hoop_margin);
  35. mHeight = ( int ) context.getResources().getDimension(R.dimen.hoop_view_height);
  36. countMargin = ( int ) context.getResources().getDimension(R.dimen.hoop_count_margin);
  37.  
  38. mDatas = new String[] { "1" , "10" , "100" };
  39. // Calculate the length of the background frame change, the default is three buttons
  40. mChangeWidth = ( int ) (2 * mSmallRadius * 3 + 4 * margin);}

After measuring the width of the view in onMeasure, calculate the center coordinates of the background circle and some related data values ​​based on the width.

  1. @Override protected void onMeasure( int widthMeasureSpec, int heightMeasureSpec) {
  2.  
  3. int widthSize = MeasureSpec.getSize(widthMeasureSpec);
  4.  
  5. mWidth = getDefaultSize(widthSize, widthMeasureSpec);
  6.  
  7. setMeasuredDimension(mWidth, mHeight);
  8.  
  9.  
  10. // Only then is the mWidth value measured, and then the center coordinates and related values ​​are calculated
  11.  
  12. cx = mWidth - mBigRadius;
  13.  
  14. cy = mHeight - mBigRadius;
  15.  
  16. // Center of the big circle
  17.  
  18. circle = new PointF(cx, cy);
  19.  
  20. // The center of the three buttons
  21.  
  22. circleOne = new PointF(cx - mBigRadius - mSmallRadius - margin, cy);
  23.  
  24. circleTwo = new PointF(cx - mBigRadius - 3 * mSmallRadius - 2 * margin, cy);
  25.  
  26. circleThree = new PointF(cx - mBigRadius - 5 * mSmallRadius - 3 * margin, cy);
  27.  
  28. // The boundaries of the initial background frame are the four boundary points of the large circle
  29.  
  30. top = cy - mBigRadius;
  31.  
  32. bottom = cy + mBigRadius;
  33.  
  34. }

Because this involves the process of expanding and contracting by clicking a button, I have defined the following states, and certain operations can only be performed in specific states.

  1. private int mState = STATE_NORMAL; //Current expansion and contraction state
  2.  
  3. private boolean mIsRun = false ; //Whether it is expanding or shrinking
  4.  
  5.  
  6. //Normal state
  7.  
  8. public   static final int STATE_NORMAL = 0;
  9.  
  10. //Button expand
  11.  
  12. public   static final int STATE_EXPAND = 1;
  13.  
  14. //Button shrink
  15.  
  16. public   static final int STATE_SHRINK = 2;
  17.  
  18. //Unfolding
  19.  
  20. public   static final int STATE_EXPANDING = 3;
  21.  
  22. //Shrinking
  23.  
  24. public   static final int STATE_SHRINKING = 4;

Next, the onDraw method is executed. Let’s take a look at the code first:

  1. @Override protected void onDraw(Canvas canvas) {
  2.  
  3. switch (mState) {
  4.  
  5. case STATE_NORMAL:
  6.  
  7. drawCircle(canvas);
  8.  
  9. break;
  10.  
  11. case STATE_SHRINK:
  12.  
  13. case STATE_SHRINKING:
  14.  
  15. drawBackground(canvas);
  16.  
  17. break;
  18.  
  19. case STATE_EXPAND:
  20.  
  21. case STATE_EXPANDING:
  22.  
  23. drawBackground(canvas);
  24.  
  25. break;
  26.  
  27. }
  28.  
  29. drawCircleText(canvas);
  30.  
  31. drawCountText(canvas);
  32.  
  33. }

The numbers above the circle and the text inside the circle exist throughout the whole process, so I put these two operations outside the switch. In the normal state, the circle and the previous two parts of text are drawn. When clicking to expand, the background frame expansion process and text are drawn. In the expanded state, click again to draw the contraction process and text. Of course, in the method of drawing the background frame, it is also necessary to continuously draw the big circle, and the big circle also exists all the time.

The drawing method above:

  1. /**
  2.  
  3. * Draw a large background circle
  4.  
  5. * @param canvas
  6.  
  7. */
  8.  
  9. private void drawCircle(Canvas canvas) {
  10.  
  11. left = cx - mBigRadius;
  12.  
  13. right = cx + mBigRadius;
  14.  
  15. canvas.drawCircle(cx, cy, mBigRadius, mBgPaint);
  16.  
  17. }
  18.  
  19.  
  20.  
  21. /**
  22.  
  23. * Draw a large circle with the text indicating the number of gold coins
  24.  
  25. * @param canvas
  26.  
  27. */
  28.  
  29. private void drawCountText(Canvas canvas) {
  30.  
  31. canvas.translate(0, -countMargin);
  32.  
  33. // Calculate the text width
  34.  
  35. float textWidth = mCountTextPaint.measureText(mCount, 0, mCount.length());
  36.  
  37. canvas.drawText(mCount, 0, mCount.length(), (2 * mBigRadius - textWidth - 35) / 2, 0.2f, mCountTextPaint);
  38.  
  39. }
  40.  
  41.  
  42.  
  43. /**
  44.  
  45. * Draw the text inside the big circle
  46.  
  47. * @param canvas
  48.  
  49. */
  50.  
  51. private void drawCircleText(Canvas canvas) {
  52.  
  53. StaticLayout layout = new StaticLayout(mText, mTextPaint, ( int ) (mBigRadius * Math.sqrt(2)), Layout.Alignment.ALIGN_CENTER, 1.0f, 0.0f, true );
  54.  
  55. canvas.translate(mWidth - mBigRadius * 1.707f, mHeight - mBigRadius * 1.707f);
  56.  
  57. layout.draw(canvas);
  58.  
  59. canvas.save();
  60.  
  61. }
  62.  
  63.  
  64.  
  65. /**
  66.  
  67. * Expand and shrink the background frame
  68.  
  69. * @param canvas
  70.  
  71. */
  72.  
  73. private void drawBackground(Canvas canvas) {
  74.  
  75. left = cx - mBigRadius - mChange;
  76.  
  77. right = cx + mBigRadius;
  78.  
  79. canvas.drawRoundRect( left , top , right , bottom, mBigRadius, mBigRadius, mPopPaint);
  80.  
  81. if ((mChange > 0) && (mChange <= 2 * mSmallRadius + margin)) {
  82.  
  83. // Draw the first button
  84.  
  85. canvas.drawCircle(cx - mChange, cy, mSmallRadius, mBgPaint);
  86.  
  87. // Draw the text inside the first button
  88.  
  89. canvas.drawText(mDatas[0], cx - (mBigRadius - mSmallRadius) - mChange, cy + 15, mTextPaint);
  90.  
  91. } else if ((mChange > 2 * mSmallRadius + margin) && (mChange <= 4 * mSmallRadius + 2 * margin)) {
  92.  
  93. // Draw the first button
  94.  
  95. canvas.drawCircle(cx - mBigRadius - mSmallRadius - margin, cy, mSmallRadius, mBgPaint);
  96.  
  97. // Draw the text inside the first button
  98.  
  99. canvas.drawText(mDatas[0], cx - mBigRadius - mSmallRadius - margin - 20, cy + 15, mTextPaint);
  100.  
  101.  
  102. // Draw the second button
  103.  
  104. canvas.drawCircle(cx - mChange, cy, mSmallRadius, mBgPaint);
  105.  
  106. // Draw the text inside the second button
  107.  
  108. canvas.drawText(mDatas[1], cx - mChange - 20, cy + 15, mTextPaint);
  109.  
  110. } else if ((mChange > 4 * mSmallRadius + 2 * margin) && (mChange <= 6 * mSmallRadius + 3 * margin)) {
  111.  
  112. // Draw the first button
  113.  
  114. canvas.drawCircle(cx - mBigRadius - mSmallRadius - margin, cy, mSmallRadius, mBgPaint);
  115.  
  116. // Draw the text inside the first button
  117.  
  118. canvas.drawText(mDatas[0], cx - mBigRadius - mSmallRadius - margin - 16, cy + 15, mTextPaint);
  119.  
  120.  
  121. // Draw the second button
  122.  
  123. canvas.drawCircle(cx - mBigRadius - 3 * mSmallRadius - 2 * margin, cy, mSmallRadius, mBgPaint);
  124.  
  125. // Draw the text inside the second button
  126.  
  127. canvas.drawText(mDatas[1], cx - mBigRadius - 3 * mSmallRadius - 2 * margin - 25, cy + 15, mTextPaint);
  128.  
  129.  
  130. // Draw the third button
  131.  
  132. canvas.drawCircle(cx - mChange, cy, mSmallRadius, mBgPaint);
  133.  
  134. // Draw the text inside the third button
  135.  
  136. canvas.drawText(mDatas[2], cx - mChange - 34, cy + 15, mTextPaint);
  137.  
  138. } else if (mChange > 6 * mSmallRadius + 3 * margin) {
  139.  
  140. // Draw the first button
  141.  
  142. canvas.drawCircle(cx - mBigRadius - mSmallRadius - margin, cy, mSmallRadius, mBgPaint);
  143.  
  144. // Draw the text inside the first button
  145.  
  146. canvas.drawText(mDatas[0], cx - mBigRadius - mSmallRadius - margin - 16, cy + 15, mTextPaint);
  147.  
  148.  
  149. // Draw the second button
  150.  
  151. canvas.drawCircle(cx - mBigRadius - 3 * mSmallRadius - 2 * margin, cy, mSmallRadius, mBgPaint);
  152.  
  153. // Draw the text inside the second button
  154.  
  155. canvas.drawText(mDatas[1], cx - mBigRadius - 3 * mSmallRadius - 2 * margin - 25, cy + 15, mTextPaint);
  156.  
  157.  
  158. // Draw the third button
  159.  
  160. canvas.drawCircle(cx - mBigRadius - 5 * mSmallRadius - 3 * margin, cy, mSmallRadius, mBgPaint);
  161.  
  162. // Draw the text inside the third button
  163.  
  164. canvas.drawText(mDatas[2], cx - mBigRadius - 5 * mSmallRadius - 3 * margin - 34, cy + 15, mTextPaint);
  165.  
  166. }
  167.  
  168. drawCircle(canvas);
  169.  
  170.  
  171. }

Then comes the processing of click events. The expansion or contraction operation will only be triggered when the touch point is within the large circle. When the small circle is clicked, an interface is provided for external calls.

  1. @Override public boolean onTouchEvent(MotionEvent event) {
  2.  
  3. int   action = event.getAction();
  4.  
  5. switch ( action ) {
  6.  
  7. case MotionEvent.ACTION_DOWN:
  8.  
  9. //If the animation is in progress when clicking, do not process
  10.  
  11. if (mIsRun) return   true ;
  12.  
  13. PointF pointF = new PointF(event.getX(), event.getY());
  14.  
  15. if (isPointInCircle(pointF, circle, mBigRadius)) { //If the touch point is within the big circle, pop up or shrink the button according to the pop-up direction
  16.  
  17. if ((mState == STATE_SHRINK || mState == STATE_NORMAL) && !mIsRun) {
  18.  
  19. //Expand
  20.  
  21. mIsRun = true ;//This must be set to true first , because onAnimationStart is called after onAnimationUpdate
  22.  
  23. showPopMenu();
  24.  
  25. } else {
  26.  
  27. //shrink
  28.  
  29. mIsRun = true ;
  30.  
  31. hidePopMenu();
  32.  
  33. }
  34.  
  35. } else { //The touch point is not within the big circle
  36.  
  37. if (mState == STATE_EXPAND) { //If it is in the expanded state
  38.  
  39. if (isPointInCircle(pointF, circleOne, mSmallRadius)) {
  40.  
  41. listener.clickButton(this, Integer .parseInt(mDatas[0]));
  42.  
  43. } else if (isPointInCircle(pointF, circleTwo, mSmallRadius)) {
  44.  
  45. listener.clickButton(this, Integer .parseInt(mDatas[1]));
  46.  
  47. } else if (isPointInCircle(pointF, circleThree, mSmallRadius)) {
  48.  
  49. listener.clickButton(this, Integer .parseInt(mDatas[2]));
  50.  
  51. }
  52.  
  53. mIsRun = true ;
  54.  
  55. hidePopMenu();
  56.  
  57. }
  58.  
  59. }
  60.  
  61. break;
  62.  
  63. }
  64.  
  65. return super.onTouchEvent(event);
  66.  
  67. }

The expansion and contraction animation is to change the width attribute of the background frame, and listen to this attribute animation to redraw the entire view when the width value changes. Because I determined the radius of the big circle and the small circle and the distance between the small circle and the background frame at the beginning, the width of the background frame has been calculated during initialization:

  1. mChangeWidth = ( int ) (2 * mSmallRadius * 3 + 4 * margin);
  1. /**
  2.  
  3. * Pop-up background frame
  4.  
  5. */
  6.  
  7. private void showPopMenu() {
  8.  
  9. if (mState == STATE_SHRINK || mState == STATE_NORMAL) {
  10.  
  11. ValueAnimator animator = ValueAnimator.ofInt(0, mChangeWidth);
  12.  
  13. animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
  14.  
  15. @Override public void onAnimationUpdate(ValueAnimator animation) {
  16.  
  17. if (mIsRun) {
  18.  
  19. mChange = ( int ) animation.getAnimatedValue();
  20.  
  21. invalidate();
  22.  
  23. } else {
  24.  
  25. animation.cancel();
  26.  
  27. mState = STATE_NORMAL;
  28.  
  29. }
  30.  
  31. }
  32.  
  33. });
  34.  
  35. animator.addListener(new AnimatorListenerAdapter() {
  36.  
  37. @Override public void onAnimationStart(Animator animation) {
  38.  
  39. super.onAnimationStart(animation);
  40.  
  41. mIsRun = true ;
  42.  
  43. mState = STATE_EXPANDING;
  44.  
  45. }
  46.  
  47.  
  48.  
  49. @Override public void onAnimationCancel(Animator animation) {
  50.  
  51. super.onAnimationCancel(animation);
  52.  
  53. mIsRun = false ;
  54.  
  55. mState = STATE_NORMAL;
  56.  
  57. }
  58.  
  59.  
  60.  
  61. @Override public void onAnimationEnd(Animator animation) {
  62.  
  63. super.onAnimationEnd(animation);
  64.  
  65. mIsRun = false ;
  66.  
  67. //Set the state to expanded after the animation ends
  68.  
  69. mState = STATE_EXPAND;
  70.  
  71. }
  72.  
  73. });
  74.  
  75. animator.setDuration(500);
  76.  
  77. animator.start();
  78.  
  79. }
  80.  
  81. }
  1. /**
  2.  
  3. * Hide popup box
  4.  
  5. */
  6.  
  7. private void hidePopMenu() {
  8.  
  9. if (mState == STATE_EXPAND) {
  10.  
  11. ValueAnimator animator = ValueAnimator.ofInt(mChangeWidth, 0);
  12.  
  13. animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
  14.  
  15. @Override public void onAnimationUpdate(ValueAnimator animation) {
  16.  
  17. if (mIsRun) {
  18.  
  19. mChange = ( int ) animation.getAnimatedValue();
  20.  
  21. invalidate();
  22.  
  23. } else {
  24.  
  25. animation.cancel();
  26.  
  27. }
  28.  
  29. }
  30.  
  31. });
  32.  
  33. animator.addListener(new AnimatorListenerAdapter() {
  34.  
  35. @Override public void onAnimationStart(Animator animation) {
  36.  
  37. super.onAnimationStart(animation);
  38.  
  39. mIsRun = true ;
  40.  
  41. mState = STATE_SHRINKING;
  42.  
  43. }
  44.  
  45.  
  46.  
  47. @Override public void onAnimationCancel(Animator animation) {
  48.  
  49. super.onAnimationCancel(animation);
  50.  
  51. mIsRun = false ;
  52.  
  53. mState = STATE_EXPAND;
  54.  
  55. }
  56.  
  57.  
  58.  
  59. @Override public void onAnimationEnd(Animator animation) {
  60.  
  61. super.onAnimationEnd(animation);
  62.  
  63. mIsRun = false ;
  64.  
  65. //Set the state to shrink after the animation ends
  66.  
  67. mState = STATE_SHRINK;
  68.  
  69. }
  70.  
  71. });
  72.  
  73. animator.setDuration(500);
  74.  
  75. animator.start();
  76.  
  77. }
  78.  
  79. }

This process looks like popping up or shrinking. In fact, every time the width value changes a little, all components are redrawn. However, the size and position of the text, large circle and other contents do not change. Only the width value of the background box changes, so this effect occurs.

Usage in xml:

  1. <LinearLayout
  2.  
  3. android:layout_width= "match_parent"  
  4.  
  5. android:layout_height= "wrap_content"  
  6.  
  7. android:layout_alignParentBottom= "true"  
  8.  
  9. android:layout_marginBottom= "20dp"  
  10.  
  11. android:layout_alignParentRight= "true"  
  12.  
  13. android:orientation= "vertical" >
  14.  
  15.  
  16. <com.xx.hoopcustomview.HoopView
  17.  
  18. android:id= "@+id/hoopview1"  
  19.  
  20. android:layout_width= "match_parent"  
  21.  
  22. android:layout_height= "wrap_content"  
  23.  
  24. android:layout_marginRight= "10dp"  
  25.  
  26. app:text= "Support Rocket"  
  27.  
  28. app: count = "1358"  
  29.  
  30. app:theme_color= "#31A129" />
  31.  
  32.  
  33. <com.xx.hoopcustomview.HoopView
  34.  
  35. android:id= "@+id/hoopview2"  
  36.  
  37. android:layout_width= "match_parent"  
  38.  
  39. android:layout_height= "wrap_content"  
  40.  
  41. android:layout_marginRight= "10dp"  
  42.  
  43. app:text= "The Heat are invincible"  
  44.  
  45. app: count = "251"  
  46.  
  47. app:theme_color= "#F49C11" />
  48.  
  49. </LinearLayout>

Use in activity:

  1. hoopview1 = (HoopView) findViewById(R.id.hoopview1);
  2.  
  3. hoopview1.setOnClickButtonListener(new HoopView.OnClickButtonListener() {
  4.  
  5. @Override public void clickButton( View   view , int num) {
  6.  
  7. Toast.makeText(MainActivity.this, "hoopview1 increased" + num, Toast.LENGTH_SHORT).show();
  8.  
  9. }
  10.  
  11. });

This is roughly the implementation process. It is still a little different from the original effect. Mine still has many flaws, such as the centering problem of the text. I have not implemented the rotation animation of the text in the small circle when it pops up or shrinks.

<<:  Miscellaneous: MVC/MVP/MVVM (Part 1)

>>:  A widget dancing on a needle with shackles on its legs

Recommend

From now on, we have one less open source mobile phone system

Firefox OS, a bold attempt by traditional browser...

Form costs reduced by 30%, home decoration advertising case!

With the further deepening of regulation and cont...

How long will “live streaming with goods” remain popular?

This article will focus on interpreting and analy...

How does Kuaishou attract traffic? Kuaishou’s e-commerce traffic strategy

1. Why is the Kuaishou platform gradually becomin...

Li Xiaolai: See you seven years later-2022

Li Xiaolai: See You Seven Years Later-2022Resourc...

How can we encourage users to actively share and attract traffic?

01 Since the Internet traffic dividend began to g...

How to use the Golden Circle Rule to write a marketing promotion plan?

Double 11, Double 12, Christmas Eve and other fes...

A guide to Metaverse advertising bonuses!

Since the concept of the metaverse was proposed, ...