java进度条动画_Android自定义控件之圆形进度条动画

本文实例为大家分享了Android实现圆形进度条动画的具体代码,供大家参考,具体内容如下

首先贴上图片:

19a0d8c5db60d648c9ccde485c8b3fe9.png

额,感觉还行吧,就是进度条的颜色丑了点,不过咱是程序员,不是美工,配色这种问题当然不在考虑范围之内了。

下面说重点,如何来写一个这样的自定义控件。

首先,需要有一个灰色的底图,来作为未填充时的进度条;

然后,根据传入的当前进度值,绘制填充时的进度圆弧,这段圆弧所对应的圆心角,由当前进度与进度的最大值(一般是100)的比值计算得出;

其次,根据进度值绘制文字提示;

最后,重绘控件,加上动画,从而达到显示进度的效果。

代码如下:

1、attrs.xml

2、CircleProgressBar.java

package com.ctgu.circleprogressbar;

import android.animation.ValueAnimator;

import android.content.Context;

import android.content.res.TypedArray;

import android.graphics.Canvas;

import android.graphics.Color;

import android.graphics.LinearGradient;

import android.graphics.Paint;

import android.graphics.Paint.FontMetricsInt;

import android.graphics.Rect;

import android.graphics.RectF;

import android.graphics.Shader;

import android.util.AttributeSet;

import android.util.TypedValue;

import android.view.View;

import android.view.animation.OvershootInterpolator;

public class CircleProgressBar extends View

{

/**

* 进度条最大值,默认为100

*/

private int maxValue = 100;

/**

* 当前进度值

*/

private int currentValue = 0;

/**

* 每次扫过的角度,用来设置进度条圆弧所对应的圆心角,alphaAngle=(currentValue/maxValue)*360

*/

private float alphaAngle;

/**

* 底部圆弧的颜色,默认为Color.LTGRAY

*/

private int firstColor;

/**

* 进度条圆弧块的颜色

*/

private int secondColor;

/**

* 圆环的宽度

*/

private int circleWidth;

/**

* 画圆弧的画笔

*/

private Paint circlePaint;

/**

* 画文字的画笔

*/

private Paint textPaint;

/**

* 渐变圆周颜色数组

*/

private int[] colorArray = new int[] { Color.parseColor("#27B197"), Color.parseColor("#00A6D5") };//

/**

* 通过代码创建时才使用

*

* @param context

*/

public CircleProgressBar(Context context)

{

this(context, null);

}

/**

* 当从xml中加载view的时候,这个构造器才会被调用。其第二个参数中就包含自定义的属性。

*

* @param context

* 上下文

* @param attrs

* 自定义属性

*/

public CircleProgressBar(Context context, AttributeSet attrs)

{

this(context, attrs, 0);

}

/**

* 从xml加载时执行和应用一个特定的风格。这里有两种方式,一是从theme中获得,二是从style中获得。

* 第三个参数官方有这样的说明: defStyle - The default style to apply to this view. If 0,

* no style will be applied (beyond what is included in the theme). This may

* either be an attribute resource, whose value will be retrieved from the

* current theme, or an explicit style resource.

* 默认的风格会被应用到这个view上。如果是0,没有风格将会被应用

* (除了被包含在主题中)。这个也许是一个属性的资源,它的值是从当前的主题中检索,或者是一个明确的风格资源。

*

* @param context

* 上下文

* @param attrs

* 自定义的属性

* @param defStyleAttr

* 自定义风格

*/

public CircleProgressBar(Context context, AttributeSet attrs, int defStyleAttr)

{

super(context, attrs, defStyleAttr);

TypedArray ta = context.getTheme().obtainStyledAttributes(attrs, R.styleable.circleProgressBar,

defStyleAttr, 0);

int n = ta.getIndexCount();

for (int i = 0; i < n; i++)

{

int attr = ta.getIndex(i);

switch (attr)

{

case R.styleable.circleProgressBar_firstColor:

firstColor = ta.getColor(attr, Color.LTGRAY); // 默认底色为亮灰色

break;

case R.styleable.circleProgressBar_secondColor:

secondColor = ta.getColor(attr, Color.BLUE); // 默认进度条颜色为蓝色

break;

case R.styleable.circleProgressBar_circleWidth:

circleWidth = ta.getDimensionPixelSize(attr, (int) TypedValue.applyDimension(

TypedValue.COMPLEX_UNIT_DIP, 6, getResources().getDisplayMetrics())); // 默认圆弧宽度为6dp

break;

default:

break;

}

}

ta.recycle();

circlePaint = new Paint();

circlePaint.setAntiAlias(true); // 抗锯齿

circlePaint.setDither(true); // 防抖动

circlePaint.setStrokeWidth(circleWidth);

textPaint = new Paint();

textPaint.setAntiAlias(true);

textPaint.setDither(true);

}

@Override

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec)

{// 分别获取期望的宽度和高度,并取其中较小的尺寸作为该控件的宽和高

int measureWidth = MeasureSpec.getSize(widthMeasureSpec);

int measureHeight = MeasureSpec.getSize(heightMeasureSpec);

setMeasuredDimension(Math.min(measureWidth, measureHeight), Math.min(measureWidth, measureHeight));

}

@Override

protected void onDraw(Canvas canvas)

{

int center = this.getWidth() / 2;

int radius = center - circleWidth / 2;

drawCircle(canvas, center, radius); // 绘制进度圆弧

drawText(canvas, center, radius);

}

/**

* 绘制进度圆弧

*

* @param canvas

* 画布对象

* @param center

* 圆心的x和y坐标

* @param radius

* 圆的半径

*/

private void drawCircle(Canvas canvas, int center, int radius)

{

circlePaint.setShader(null); // 清除上一次的shader

circlePaint.setColor(firstColor); // 设置底部圆环的颜色,这里使用第一种颜色

circlePaint.setStyle(Paint.Style.STROKE); // 设置绘制的圆为空心

canvas.drawCircle(center, center, radius, circlePaint); // 画底部的空心圆

RectF oval = new RectF(center - radius, center - radius, center + radius, center + radius); // 圆的外接正方形

// 绘制颜色渐变圆环

// shader类是Android在图形变换中非常重要的一个类。Shader在三维软件中我们称之为着色器,其作用是来给图像着色。

LinearGradient linearGradient = new LinearGradient(circleWidth, circleWidth, getMeasuredWidth()

- circleWidth, getMeasuredHeight() - circleWidth, colorArray, null, Shader.TileMode.MIRROR);

circlePaint.setShader(linearGradient);

circlePaint.setShadowLayer(10, 10, 10, Color.RED);

circlePaint.setColor(secondColor); // 设置圆弧的颜色

circlePaint.setStrokeCap(Paint.Cap.ROUND); // 把每段圆弧改成圆角的

alphaAngle = currentValue * 360.0f / maxValue * 1.0f; // 计算每次画圆弧时扫过的角度,这里计算要注意分母要转为float类型,否则alphaAngle永远为0

canvas.drawArc(oval, -90, alphaAngle, false, circlePaint);

}

/**

* 绘制文字

*

* @param canvas

* 画布对象

* @param center

* 圆心的x和y坐标

* @param radius

* 圆的半径

*/

private void drawText(Canvas canvas, int center, int radius)

{

float result = (currentValue * 100.0f / maxValue * 1.0f); // 计算进度

String percent = String.format("%.1f", result) + "%";

textPaint.setTextAlign(Paint.Align.CENTER); // 设置文字居中,文字的x坐标要注意

textPaint.setColor(Color.BLACK); // 设置文字颜色

textPaint.setTextSize(40); // 设置要绘制的文字大小

textPaint.setStrokeWidth(0); // 注意此处一定要重新设置宽度为0,否则绘制的文字会重叠

Rect bounds = new Rect(); // 文字边框

textPaint.getTextBounds(percent, 0, percent.length(), bounds); // 获得绘制文字的边界矩形

FontMetricsInt fontMetrics = textPaint.getFontMetricsInt(); // 获取绘制Text时的四条线

int baseline = center + (fontMetrics.bottom - fontMetrics.top) / 2 - fontMetrics.bottom; // 计算文字的基线,方法见http://blog.csdn.net/harvic880925/article/details/50423762

canvas.drawText(percent, center, baseline, textPaint); // 绘制表示进度的文字

}

/**

* 设置圆环的宽度

*

* @param width

*/

public void setCircleWidth(int width)

{

this.circleWidth = (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, width, getResources()

.getDisplayMetrics());

circlePaint.setStrokeWidth(circleWidth);

invalidate();

}

/**

* 设置圆环的底色,默认为亮灰色LTGRAY

*

* @param color

*/

public void setFirstColor(int color)

{

this.firstColor = color;

circlePaint.setColor(firstColor);

invalidate();

}

/**

* 设置进度条的颜色,默认为蓝色

*

* @param color

*/

public void setSecondColor(int color)

{

this.secondColor = color;

circlePaint.setColor(secondColor);

invalidate();

}

/**

* 设置进度条渐变色颜色数组

*

* @param colors

* 颜色数组,类型为int[]

*/

public void setColorArray(int[] colors)

{

this.colorArray = colors;

invalidate();

}

/**

* 按进度显示百分比

*

* @param progress

* 进度,值通常为0到100

*/

public void setProgress(int progress)

{

int percent = progress * maxValue / 100;

if (percent < 0)

{

percent = 0;

}

if (percent > 100)

{

percent = 100;

}

this.currentValue = percent;

invalidate();

}

/**

* 按进度显示百分比,可选择是否启用数字动画

*

* @param progress

* 进度,值通常为0到100

* @param useAnimation

* 是否启用动画,true为启用

*/

public void setProgress(int progress, boolean useAnimation)

{

int percent = progress * maxValue / 100;

if (percent < 0)

{

percent = 0;

}

if (percent > 100)

{

percent = 100;

}

if (useAnimation) // 使用动画

{

ValueAnimator animator = ValueAnimator.ofInt(0, percent);

animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener()

{

@Override

public void onAnimationUpdate(ValueAnimator animation)

{

currentValue = (int) animation.getAnimatedValue();

invalidate();

}

});

animator.setInterpolator(new OvershootInterpolator());

animator.setDuration(1000);

animator.start();

}

else

{

setProgress(progress);

}

}

}

3、activity_main.xml

xmlns:tools="http://schemas.android.com/tools"

xmlns:lh2="http://schemas.android.com/apk/res/com.ctgu.circleprogressbar"

android:layout_width="match_parent"

android:layout_height="match_parent" >

android:id="@+id/circleProgressBar"

android:layout_width="150dp"

android:layout_height="150dp"

android:layout_centerHorizontal="true"

android:layout_gravity="center"

android:layout_marginTop="20dp"

lh2:circleWidth="6dp"

lh2:firstColor="#d3d3d3"

lh2:secondColor="#3B95C8" />

android:id="@+id/seekbar"

android:layout_width="match_parent"

android:layout_height="wrap_content"

android:layout_alignParentBottom="true"

android:layout_marginBottom="40dp"

android:layout_marginLeft="10dp"

android:layout_marginRight="10dp"

android:background="#778899" />

4、MainActivity.java

package com.ctgu.circleprogressbar;

import android.app.Activity;

import android.graphics.Color;

import android.os.Bundle;

import android.os.Handler;

import android.widget.SeekBar;

public class MainActivity extends Activity

{

private CircleProgressBar circleProgressBar; // 自定义的进度条

private SeekBar seekbar; // 拖动条

private int[] colors = new int[] { Color.parseColor("#27B197"), Color.parseColor("#00A6D5") };

@Override

protected void onCreate(Bundle savedInstanceState)

{

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

circleProgressBar = (CircleProgressBar) findViewById(R.id.circleProgressBar);

// circleProgressBar.setFirstColor(Color.LTGRAY);

// circleProgressBar.setColorArray(colors); //觉得进度条颜色丑的,这里可以自行传入一个颜色渐变数组。

// circleProgressBar.setCircleWidth(6);

seekbar = (SeekBar) findViewById(R.id.seekbar);

seekbar.setMax(100);

seekbar.setOnSeekBarChangeListener(new SeekBar.OnSeekBarChangeListener()

{

@Override

public void onStopTrackingTouch(SeekBar seekBar)

{

}

@Override

public void onStartTrackingTouch(SeekBar seekBar)

{

}

@Override

public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser)

{

if (fromUser)

{

// circleProgressBar.setProgress(progress); //不使用动画

circleProgressBar.setProgress(progress, true); // 使用数字过渡动画

}

}

});

}

}

代码注释很详细了,基本上了解自定义控件的都看得懂。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/497520.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

人脸识别必读的N篇文章

来源&#xff1a;机器视觉概要&#xff1a;人脸检测/跟踪的目的是在图像/视频中找到各个人脸所在的位置和大小&#xff1b;对于跟踪而言&#xff0c;还需要确定帧间不同人脸间的对应关系。一 、人脸检测/跟踪人脸检测/跟踪的目的是在图像/视频中找到各个人脸所在的位置和大小&a…

所谓高情商就是会说话--总结

思维导图&#xff1a;https://www.processon.com/view/link/60d6fe741e08532a43bea811

美国12大科技公司如何参与自动驾驶?

来源&#xff1a;腾讯科技概要&#xff1a;无人驾驶汽车的潜力已经逐步显现&#xff0c;这一点&#xff0c;从本周在拉斯维加斯举行的国际消费电子展&#xff08;CES&#xff09;就可以看出端倪。无人驾驶汽车的潜力已经逐步显现&#xff0c;这一点&#xff0c;从本周在拉斯维加…

java web响应式框架_Web开发的十佳HTML5响应式框架

HTML5框架是一类有助于快速轻松创建响应式网站的程序包。这些HTML5框架有着能减轻编程任务和重复代码负担的神奇功能。关于HTML5的框架种类繁多&#xff0c;并且很瘦欢迎&#xff0c;因为它能允许开发人员花费更少的时间和精力去创建一个令人惊艳的网站。在本文中&#xff0c;我…

开放-封闭原则(OCP)

开放-封闭原则&#xff08;The Open-Close Principle&#xff09; 软件实体&#xff08;类、模块、函数等&#xff09;应该是可以扩展的&#xff0c;但是不可以修改的。 两个特征 1、对于扩展是开放的&#xff08;Open for extension&#xff09;&#xff1b; 2、对于更改是…

风向2018 | AI 突围

来源&#xff1a;36氪概要&#xff1a;回到战场&#xff0c;考验各家公司CEO和谋士的时刻&#xff0c;已经到来。问&#xff1a;2017年AI行业最大的进展是什么&#xff1f;答&#xff1a;创业公司的估值。高估值&#xff0c;曾将AI初创公司带上高光的创投舞台&#xff0c;也正将…

php return 值_php return的用法是什么

php return的用法&#xff1a;1、通过“return expression”语法返回一个表达式结果&#xff1b;2、使用“return(expr)”语法返回函数表达式&#xff1b;3、直接使用return返回值即可。本文操作环境&#xff1a;Windows7系统、PHP7.1、Dell G3电脑。基本用法&#xff1a;a)、r…

谷歌放出AI平民化大招: 李飞飞宣布推出AutoML云平台,让普通企业也能用上深度学习

作者&#xff1a;杨晓凡概要&#xff1a;谷歌云机器学习平台&#xff08;Google Cloud AI&#xff09;自从上线以来就以预训练的、可以直接调用的高效机器学习模型吸引了许多企业级用户在其上构建简单的机器学习应用。谷歌云机器学习平台&#xff08;Google Cloud AI&#xff0…

AI博弈论:DeepMind让智能体在非对称博弈中找纳什均衡

Root 林鳞 编译自 DeepMind官方博客量子位 出品 | 公众号 QbitAI随着人工智能系统在现实世界中扮演越来越重要的角色&#xff0c;理解不同的系统如何相互作用至关重要。刚刚&#xff0c;DeepMind发表了一篇名为Symmetric Decomposition of Asymmetric Games的论文。在这篇论文中…

php ie 下载 乱码,php ie下载文件名乱码怎么办

php ie下载文件名乱码的解决办法&#xff1a;1、通过header方法解决乱码&#xff1b;2、通过“function remote_filesize($uri,$user,$pw) {...}”等方法解决乱码。php文件下载IE文件名乱码问题一直用chrome浏览器&#xff0c;没发现问题。今天用ie6&#xff0c;发现文件下载时…

无人驾驶技术排名:百度居中游,苹果特斯拉垫底 | 行业

来源&#xff1a;网易科技概要&#xff1a;其调查研究显示&#xff0c;无人驾驶汽车行业的现状已经发生了天翻地覆的变化&#xff0c;而特斯拉、苹果公司在今年的排名垫底。1月17日消息&#xff0c;据CNET网站报道&#xff0c;日前&#xff0c;美国市场研究机构Navigant Resear…

h5 换脸 php,【部分原创】python实现视频内的face swap(换脸)

1.准备工作&#xff0c;按博主的环境为准Python 3.5Opencv 3Tensorflow 1.3.1Keras 2cudnn和CUDA&#xff0c;如果你的GPU足够厉害并且支持的话&#xff0c;可以选择安装那就先安装起来&#xff0c;有兴趣的朋友给我个暗示&#xff0c;好让我有动力写下去&#xff0c;想实现整套…

接口隔离原则(ISP)

接口隔离原则&#xff08;The Interface Segregation Interface&#xff09; 这个原则用来处理“胖&#xff08;fat&#xff09;”接口&#xff08;类的接口不是内聚的&#xff09;所具有的缺点。“胖”接口可以分解成多组方法。 考虑一个安全系统&#xff0c;有一些Door对象&a…

马歇尔·赫伯特:人工智能的前沿技术与实例分析

来源&#xff1a;中国人工智能学会2017年12月11日&#xff0c;国际知名机器人专家、美国卡耐基梅隆大学机器人研究所所长马歇尔赫伯特(Martial Hebert)教授和首席科学家大卫伯恩(David Bourne)教授访问了中国科学技术大学参观中科大机器人实验室并作演讲。演讲人简介&#xff1…

ActiveMQ消费者平滑关闭

平滑关闭的思路就是让正在执行的任务线程正常执行完毕&#xff0c;然后再关闭JVM。在JVM关闭之前触发一个shutdown hook&#xff0c;jvm自带这个hook&#xff0c;在java启动时候就可以注册这样的hook。 ##1、简述JVM关闭钩子&#xff08;shutdown hook&#xff09; 首先JVM的关…

二叉堆时间复杂度 php,二叉堆(Binary Heap)

二叉堆这个数据结构有点意思&#xff0c;自己做了个总结&#xff0c;内容结构如下&#xff1a;二叉堆性质二叉堆操作应用二叉堆性质&#xff1a;堆(Heap)是一个可以被看成近似完全二叉树的结构&#xff0c;具有完全二叉树的特性&#xff1a;缺少的叶子节点总是位于右子节点n个节…

产业丨一文读懂人工智能产业链,未来10年2000亿美元市场

来源&#xff1a;国防科技信息网概要&#xff1a;针对人工智能产业链&#xff0c;主要有三个核心&#xff1a;基础技术、人工智能技术及人工智能应用&#xff0c;本文将从主要从这三个方面进行梳理。人工智能&#xff08;Artificial Intelligence&#xff09;&#xff0c;英文缩…

2017年高性能计算领域的成功与失败

来源&#xff1a;中科院信息科技战略情报概要&#xff1a;2017年&#xff0c;机器学习和各种人工智能应用在高性能计算领域持续发挥着重要影响力。2017年12月9日&#xff0c;TOP500官网刊文总结了高性能计算在2017年取得的新进展&#xff0c;以及未来发展趋势&#xff0c;主要内…

Science封面:谁动了我的DNA?原来是你拿去做机器人了

来源&#xff1a;机器人大讲堂概要&#xff1a;慕尼黑工业大学的Friedrich C. Simmel团队使用DNA分子&#xff0c;组装出了一个可以远程控制的纳米机械臂&#xff0c;并用它成功推动了一个纳米金颗粒。本周《Science》期刊的封面故事&#xff0c;介绍了一款德国制造的灵活DNA手…

中国倒数第五!毕马威全球自动驾驶报告|附下载

来源&#xff1a;智东西概要&#xff1a;指向高效、安全的自动驾驶被认为是未来汽车形态&#xff0c;渐渐融入到各国汽车发展战略中去&#xff0c;科技公司和传统车企纷纷入局。指向高效、安全的自动驾驶被认为是未来汽车形态&#xff0c;渐渐融入到各国汽车发展战略中去&#…