Android自定义view实现圆形进度条效果

作者:吐尔洪江Coding 时间:2021-09-12 00:30:03 

Android中实现进度条有很多种方式,自定义进度条一般是继承progressBar或继承view来实现,本篇中讲解的是第二种方式。

先上效果图:

Android自定义view实现圆形进度条效果

实现圆形进度条总体来说并不难,还是跟往常一样继承view,初始化画笔,按下面的步骤一步步来就好了。对初学者来说动画效果可能比较陌生,我们可以使用属性动画中的valueAnimator来实现动画效果。

实现步骤:

1、画出一个灰色的圆环作为背景。

2、画出上层的圆环覆盖下方的圆环。

3、加入动画效果

值得注意的是怎么设置圆环和文字的位置。

画出矩形只需要传入矩形对角线的坐标即可,如果不加以处理的话画出来的圆环的边缘是不完整的,刚开始接触自定义view的同学们一定要先好好看看Android坐标系相关内容,不然很难理解位置参数为什么这样设置。

完整代码:

package com.example.floatingwindow.widget;
?
import android.animation.ValueAnimator;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.RectF;
import android.util.AttributeSet;
import android.util.TypedValue;
import android.view.View;
import android.view.animation.DecelerateInterpolator;
?
import androidx.annotation.Nullable;
?
import com.example.floatingwindow.R;
?
public class ProgressBarView extends View {
?
? ? private Paint mPaintBack;
? ? private Paint mPaint;
? ? private Paint mPaintText;
? ? private float process;
? ? private int strokeWidth = 15;
? ? private int textSize = 20;
? ? private long duration = 3000;
? ? private float startDegree = 0;
? ? private float endDegree = 360;
? ? private String text = "完成";
? ? private String defaultText = "0%";
?
?
? ? public ProgressBarView(Context context) {
? ? ? ? super(context);
? ? ? ? init();
? ? }
?
? ? public ProgressBarView(Context context, @Nullable AttributeSet attrs) {
? ? ? ? super(context, attrs);
? ? ? ? init();
? ? }
?
? ? public ProgressBarView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
? ? ? ? super(context, attrs, defStyleAttr);
? ? ? ? init();
? ? }
?
? ? private void init() {
? ? ? ? mPaintBack = new Paint();
? ? ? ? mPaintBack.setColor(getResources().getColor(R.color.gray));
? ? ? ? mPaintBack.setStyle(Paint.Style.STROKE);
? ? ? ? mPaintBack.setAntiAlias(true);
? ? ? ? mPaintBack.setStrokeCap(Paint.Cap.ROUND);
? ? ? ? mPaintBack.setStrokeWidth(strokeWidth);
?
? ? ? ? mPaint = new Paint();
? ? ? ? mPaint.setColor(getResources().getColor(R.color.purple_200));
? ? ? ? mPaint.setStyle(Paint.Style.STROKE);
? ? ? ? mPaint.setAntiAlias(true);
? ? ? ? mPaint.setStrokeCap(Paint.Cap.ROUND);
? ? ? ? mPaint.setStrokeWidth(strokeWidth);
?
? ? ? ? mPaintText = new Paint();
? ? ? ? mPaintText.setAntiAlias(true);
? ? ? ? mPaintText.setStyle(Paint.Style.STROKE);
? ? ? ? mPaintText.setColor(Color.BLACK);
? ? ? ? mPaintBack.setStrokeCap(Paint.Cap.ROUND);
? ? ? ? mPaintText.setTextSize(sp2px(textSize));
? ? }
?
? ? public void setStrokeWidth(int width) {
? ? ? ? strokeWidth = width;
? ? }
?
? ? public void setTextSize(int textSize) {
? ? ? ? this.textSize = textSize;
? ? }
?
? ? public void setDuration(long duration) {
? ? ? ? this.duration = duration;
? ? }
?
? ? @Override
? ? protected void onDraw(Canvas canvas) {
? ? ? ? super.onDraw(canvas);
? ? ? ? //创建圆环矩形
? ? ? ? RectF rectF = new RectF(strokeWidth, strokeWidth, getWidth() - strokeWidth, getHeight() - strokeWidth);
? ? ? ? //画出灰色进度条作为背景
? ? ? ? canvas.drawArc(rectF, 0, 360, false, mPaintBack);
? ? ? ? //画进度条
? ? ? ? canvas.drawArc(rectF, 0, process, false, mPaint);
? ? ? ? //计算进度
? ? ? ? int percent = (int) (process / 360 * 100);
? ? ? ? //设置文字在canvas中的位置
? ? ? ? Paint.FontMetrics fm = mPaintText.getFontMetrics();
? ? ? ? int mTxtWidth = (int) mPaintText.measureText(text, 0, defaultText.length());
? ? ? ? int mTxtHeight = (int) Math.ceil(fm.descent - fm.ascent);
? ? ? ? int x = getWidth() / 2 - mTxtWidth / 2;
? ? ? ? int y = getHeight() / 2 + mTxtHeight / 4;
? ? ? ? if (percent < 100) {
? ? ? ? ? ? canvas.drawText(percent + "%", x, y, mPaintText);
? ? ? ? } else {
? ? ? ? ? ? canvas.drawText(text, x, y, mPaintText);
? ? ? ? }
? ? }
?
? ? /**
? ? ?* 设置动画效果
? ? ?*/
? ? public void start() {
? ? ? ? ValueAnimator valueAnimator = ValueAnimator.ofFloat(startDegree, endDegree);
? ? ? ? valueAnimator.setDuration(duration);
? ? ? ? valueAnimator.setInterpolator(new DecelerateInterpolator());
? ? ? ? valueAnimator.addUpdateListener(animation -> {
? ? ? ? ? ? process = (float) animation.getAnimatedValue();
? ? ? ? ? ? invalidate();
? ? ? ? });
? ? ? ? valueAnimator.start();
? ? }
?
? ? private int sp2px(int sp) {
? ? ? ? return (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_SP, sp,
? ? ? ? ? ? ? ? getResources().getDisplayMetrics());
? ? }
}

最后就是动画效果了,使用valueanimator,传入开始和结束的进度以及执行时间。然后每次进度发生变化时做UI刷新。

xml布局:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
? ? xmlns:app="http://schemas.android.com/apk/res-auto"
? ? xmlns:tools="http://schemas.android.com/tools"
? ? android:layout_width="match_parent"
? ? android:layout_height="match_parent"
? ? tools:context=".MainActivity">
?
? ? <com.example.floatingwindow.widget.ProgressBarView
? ? ? ? android:id="@+id/progressBar"
? ? ? ? android:layout_width="150dp"
? ? ? ? android:layout_height="150dp"
? ? ? ? app:layout_constraintTop_toTopOf="parent"
? ? ? ? app:layout_constraintBottom_toBottomOf="parent"
? ? ? ? app:layout_constraintStart_toStartOf="parent"
? ? ? ? app:layout_constraintEnd_toEndOf="parent">
? ? </com.example.floatingwindow.widget.ProgressBarView>
?
</androidx.constraintlayout.widget.ConstraintLayout>

MainActivity:

class MainActivity : AppCompatActivity() {
? ? override fun onCreate(savedInstanceState: Bundle?) {
? ? ? ? super.onCreate(savedInstanceState)
? ? ? ? setContentView(R.layout.activity_main)
?
? ? ? ? progressBar.start()
?
? ? ? ? progressBar.setOnClickListener { progressBar.start()}
? ? }

来源:https://blog.csdn.net/daydayup05/article/details/122150161

标签:Android,view,进度条
0
投稿

猜你喜欢

  • java的主要特性学习总结

    2021-11-16 05:46:19
  • Java中i++的一些问题总结

    2022-04-11 18:07:34
  • Java线程同步Lock同步锁代码示例

    2023-11-17 13:23:54
  • Spring AOP面向切面编程实现原理方法详解

    2021-07-22 00:26:07
  • java和c/c++ 数据类型长度的比较

    2021-06-18 00:07:45
  • Android使用phonegap从相册里面获取照片(代码分享)

    2023-07-24 18:53:03
  • Java面向对象程序设计:继承,多态用法实例分析

    2021-08-03 06:54:04
  • 深入浅析Java反射机制

    2023-11-25 07:02:03
  • Android APK使用Debug签名重新打包 Eclipse更改默认Debug签名

    2022-03-07 13:27:12
  • 使用C++程序获取新浪行情数据的方法

    2022-06-16 13:14:35
  • Spring Boot Admin实践详解

    2023-08-25 06:57:53
  • IDEA 设置显示内存的使用情况和内存回收的方法

    2023-10-08 15:51:25
  • java编程常用技术(推荐)

    2023-11-06 09:03:44
  • Java Structs框架原理案例详解

    2023-12-11 16:46:22
  • Java超详细讲解抽象类的原理与用法

    2022-10-31 20:51:42
  • SpringMVC路径匹配中使用通配符问题

    2023-07-18 20:44:03
  • Java Lambda 表达式源码解析

    2021-06-30 08:12:48
  • 详解Android中点击事件的几种实现方式

    2022-07-03 09:05:09
  • C#自定义签名章实现方法

    2022-08-02 05:22:37
  • docker网络配置过程详解介绍

    2023-02-07 23:04:15
  • asp之家 软件编程 m.aspxhome.com