Android自定义view实现圆形进度条效果
作者:吐尔洪江Coding 发布时间:2021-09-12 00:30:03
标签:Android,view,进度条
Android中实现进度条有很多种方式,自定义进度条一般是继承progressBar或继承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


猜你喜欢
- mybatis-plus的代码生成器会在实体类中生成数据库所有字段,我们去用mapper接口查询时,会返回数据库所有的字段。但有些字段不是我
- 解析得到的代码能通过XHTML 1.0 STRICT验证;包含了标题,链接,字体,对齐,图片,引用,列表等方面的功能.&
- 本期文章源码:GitHub一文彻底搞懂《并查集》!概念并查集是一种树型的数据结构,用于处理一些不相交集合的合并及查询问题(即所谓的并、查)。
- 目录一、服务的概念二、Android的多线程编程2.1 线程的基本用法2.2 在子线程中更新UI更新方式一更新方式二2.3 解析异步消息处理
- 本文是精讲RestTemplate第6篇,前篇的blog访问地址如下:RestTemplate在Spring或非Spring环境下使用精讲R
- 自定义Starter命名规则注意artifactId的命名规则,Spring官方Starter通常命名为spring-boot-starte
- 先给大家展示下效果图,对第三方开源 android tickplusdrawable相关知识感兴趣的朋友一起学习吧。Android tick
- 文章描述可能我标题描述不太准确,所以还是要稍微解释下:横线样式就是将TextBox以一条底横线的形式展示在页面,占位提示就是Web的Plac
- Java基本概念JDK包含了不少Java开发相关命令。如,javac、java、javap、javaw、javadoc。虽然现在的Java开
- 随着JDK 14的发布(https://waylau.com/jdk-14-released/),各大Java IDE也开始支持JDK 14
- 用Dockerfile 构建一个java的编译环境,这里整理下实现步骤:1、包括以下软件包ubuntujdkmavensvn2、jdk、ma
- 用DataFormatString格式化GridView在 GridView里面显示数据,要显示的数据有好多位小数,就想让它只显示两位小数,
- 本文实例讲述了Android编程实现仿优酷旋转菜单效果。分享给大家供大家参考,具体如下:首先,看下效果:不好意思,不会制作动态图片,只好上传
- 这篇文章主要介绍了Java模拟多线程实现抢票,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考
- 问题发现今天发生了一件事,令我非常郁闷,就是我在使用一个SDK时,当我调用他的方法时,提示我方法中的参数var1, var2如下:// 方法
- 前言以下为本文要记录的大概内容:Java基础案例:两只老虎、三个和尚、考试奖励以下是本篇文章正文内容,仅供参考一、案例1:两只老虎1.题目:
- 最近遇到的一个场景,在一个被 @Transactional 注解的方法A中中调用了一个被 @Async 注解标记的方法B,由于方法B 在执行
- 首先给出一段代码:public class AslistMethod { public static void main(String[]
- 内存映射文件是利用虚拟内存把文件映射到进程的地址空间中去,在此之后进程操作文件,就像操作进程空间里的地址一样了,比如使用c语言的 memcp
- 在Flutter开发过程中,我门有时候需要对一些数据进行本地的持久化存储,使用sp文件形式虽然也能解决问题,但是有时数据量较大的时候,显然我