Android 自定义环形进度条

本文介绍如何在Android中创建自定义环形进度条,并详细解释了通过XML配置文件设置进度条属性的方法,包括进度条的颜色、宽度、背景颜色、文字颜色及大小等。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Android 自定义环形进度条

在这里插入图片描述

效果实现

attrs.xml

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <declare-styleable name="CircleProgress">
        <!--   进度条宽度     -->
        <attr name="progress_width" format="dimension" />
        <!--   进度条颜色     -->
        <attr name="progress_color" format="color" />
        <!--   进度条背景颜色     -->
        <attr name="progress_bg_color" format="color" />
        <!--   进度     -->
        <attr name="progress" format="integer" />
        <!--   文字大小     -->
        <attr name="circle_text_size" format="dimension" />
        <!--   文字颜色     -->
        <attr name="circle_text_color" format="color" />
        <!--   是否显示居中文字     -->
        <attr name="isVisibleText" format="boolean" />
        <!--   进度条起点位置     -->
        <attr name="location_start" format="enum">
            <enum name="left" value="1" />
            <enum name="top" value="2" />
            <enum name="right" value="3" />
            <enum name="bottom" value="4" />
        </attr>
    </declare-styleable>
</resources>

CircleProgress.java

public class CircleProgress extends View {
    public static final int DEFAULT_PROGRESS_WIDTH = 2;
    public static final int DEFAULT_PROGRESS_BG_COLOR = Color.GRAY;
    public static final int DEFAULT_PROGRESS_COLOR = Color.RED;
    public static final int DEFAULT_PROGRESS = 0;
    public static final int DEFAULT_TEXT_COLOR = Color.RED;
    public static final int DEFAULT_TEXT_SIZE = 30;
    public static final int DEFUALT_DURATION = 2000;

    private float startAngle = -90;
    private int locationStart;
    private int progress = 0;
    private int progressColor;
    private int progressBgColor;
    private int circleTextColor;
    private int progressWidth;
    private String circleText;
    private boolean isVisibleText;
    private int circleTextSize;
    private Paint mBgPaint;
    private Paint mProgressPaint;
    private Paint mTextPaint;
    private RectF rectF;
    private ObjectAnimator animator;

    public CircleProgress(Context context) {
        this(context, null);
    }

    public CircleProgress(Context context, @Nullable AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public CircleProgress(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init(context, attrs);
    }

    /**
     * 初始化操作
     */
    private void init(Context context, AttributeSet attrs) {
        TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.CircleProgress);
        progressWidth = a.getDimensionPixelOffset(R.styleable.CircleProgress_progress_width, DEFAULT_PROGRESS_WIDTH);
        progressBgColor = a.getColor(R.styleable.CircleProgress_progress_bg_color, DEFAULT_PROGRESS_BG_COLOR);
        progressColor = a.getColor(R.styleable.CircleProgress_progress_color, DEFAULT_PROGRESS_COLOR);
        progress = a.getInteger(R.styleable.CircleProgress_progress, DEFAULT_PROGRESS);
        circleTextColor = a.getColor(R.styleable.CircleProgress_circle_text_color, DEFAULT_TEXT_COLOR);
        circleTextSize = a.getDimensionPixelOffset(R.styleable.CircleProgress_circle_text_size, DEFAULT_TEXT_SIZE);
        isVisibleText = a.getBoolean(R.styleable.CircleProgress_isVisibleText, false);
        locationStart = a.getInt(R.styleable.CircleProgress_location_start, 2);
        a.recycle();

        mBgPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
        mBgPaint.setStyle(Paint.Style.STROKE);
        mBgPaint.setStrokeWidth(progressWidth);
        mBgPaint.setColor(progressBgColor);
        mBgPaint.setStrokeCap(Paint.Cap.ROUND);

        mProgressPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
        mProgressPaint.setStyle(Paint.Style.STROKE);
        mProgressPaint.setStrokeWidth(progressWidth);
        mProgressPaint.setColor(progressColor);
        mProgressPaint.setStrokeCap(Paint.Cap.ROUND);
        switch (locationStart) {
            case 1:
                startAngle = -180;
                break;
            case 2:
                startAngle = -90;
                break;
            case 3:
                startAngle = 0;
                break;
            case 4:
                startAngle = 90;
                break;
        }

        mTextPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
        mTextPaint.setTextSize(circleTextSize);
        mTextPaint.setColor(circleTextColor);
        mTextPaint.setTextAlign(Paint.Align.CENTER);
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        int widthSize = MeasureSpec.getSize(widthMeasureSpec);
        int heightSize = MeasureSpec.getSize(heightMeasureSpec);
        int size = widthSize < heightSize ? widthSize : heightSize;
        setMeasuredDimension(size, size);
    }

    @Override
    protected void onDraw(Canvas canvas) {
        float cx = getWidth() / 2F;
        float cy = getHeight() / 2F;
        float radius = getWidth() / 2 - progressWidth / 2;
        canvas.drawCircle(cx, cy, radius, mBgPaint);

        float sweepAngle = 360 / 100F * progress;
        rectF = new RectF(progressWidth / 2, progressWidth / 2, getWidth() - progressWidth / 2, getHeight() - progressWidth / 2);
        canvas.drawArc(rectF, startAngle, sweepAngle, false, mProgressPaint);

        if (isVisibleText && circleText != null) {
            Paint.FontMetrics fontMetrics = mTextPaint.getFontMetrics();
            canvas.drawText(circleText, getWidth() / 2, getHeight() / 2 - (fontMetrics.ascent + fontMetrics.descent) / 2, mTextPaint);
        }
    }

    /**
     * 动画执行进度
     *
     * @param endProgress 进度
     */
    public void startAnimProgress(@IntRange(from = 0, to = 100) int endProgress) {
        startAnimProgress(endProgress, DEFUALT_DURATION);
    }

    /**
     * 动画执行进度
     *
     * @param endProgress 进度
     * @param duration    动画持续时间
     */
    public void startAnimProgress(int endProgress, int duration) {
        animator = ObjectAnimator.ofInt(this, "progress", 0, endProgress);
        animator.setDuration(duration);
        animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {
                int progress = (int) animation.getAnimatedValue();
                setProgress(progress);
            }
        });
        animator.start();
    }

    /**
     * 获取当前进度
     */
    public int getProgress() {
        return progress;
    }

    /**
     * 设置进度
     */
    public void setProgress(@IntRange(from = 0, to = 100) int progress) {
        this.progress = progress;
        invalidate();
        if (mOnCircleProgressListener != null) {
            mOnCircleProgressListener.onProgressChange(progress);
        }
    }

    /**
     * 设置文本
     */
    public void setCircleText(String text) {
        circleText = text;
        invalidate();
    }

    public interface OnCircleProgressListener {
        void onProgressChange(int progress);
    }

    private OnCircleProgressListener mOnCircleProgressListener;

    /**
     * 监听进度
     */
    public void setOnCircleProgressListener(OnCircleProgressListener listener) {
        mOnCircleProgressListener = listener;
    }

    @Override
    protected void onDetachedFromWindow() {
        super.onDetachedFromWindow();
        mOnCircleProgressListener = null;
        if (animator != null) {
            animator.cancel();
            animator = null;
        }
    }
}

XML中使用

<com.example.mylibrary.CircleProgress
        android:id="@+id/circleProgress"
        android:layout_width="200dp"
        android:layout_height="200dp"
        android:layout_margin="15dp"
        app:circle_text_color="#f00"
        app:circle_text_size="30sp"
        app:isVisibleText="true"
        app:progress="0"
        app:progress_bg_color="#68000000"
        app:progress_color="#f00"
        app:progress_width="5dp" />

Java中使用

protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    circleProgress = findViewById(R.id.circleProgress);
    circleProgress.setOnCircleProgressListener(new CircleProgress.OnCircleProgressListener() {
        @Override
        public void onProgressChange(int progress) {
            circleProgress.setCircleText(String.valueOf(progress));
        }
    });
}

public void start(View view) {
    circleProgress.startAnimProgress(90);
}

public void progress(View view) {
    circleProgress.setProgress(50);
}

代码下载

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值