Android自定义View基础开发之图片加载进度条
作者:happy_fsyy 发布时间:2022-05-01 05:52:14
标签:android,图片加载,进度条
学会了Paint,Canvas的基本用法之后,我们就可以动手开始实践了,先写个简单的图片加载进度条看看。
按照惯例,先看效果图,再决定要不要往下看:
既然看到这里了,应该是想了解这个图片加载进度条了,我们先看具体用法,再看自定义View的实现:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:custom="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent">
<ImageView
android:id="@+id/img"
android:layout_width="200dp"
android:layout_height="200dp"
android:scaleType="centerCrop"
android:layout_centerInParent="true"/>
<com.example.circleprogresstest.CircleProgressView
android:id="@+id/progressView"
android:layout_width="60dp"
android:layout_height="60dp"
android:layout_centerInParent="true"
custom:isShowProgress="true" />
</RelativeLayout>
ImageLoader.getInstance().displayImage(url, imageView, options,
new SimpleImageLoadingListener() ,
new ImageLoadingProgressListener() {
@Override
public void onProgressUpdate(String imageUri, View view, int current, int total) {
if(current==total){
progressView.setVisibility(View.GONE);
}else{
progressView.setSweepAngle((int)(360*current*1.0f/total));
progressView.postInvalidate();
}
}
}
);
可以看出,以上的用法,非常简单,在xml中添加我们自定义的View,和添加textview或者button完全相同,只是多了我们自己的自定义属性而已,可以设置圆的颜色,以及文字颜色,大小等等。之后,在MainActivity中使用的方法也是同样简单,只要在图片的进度更新的时候,同时更新我们进度条的进度就行了。
下面我们具体说下我们实现自定义进度条的过程,我们只需要重写onDraw()方法就够了,很明显,我们的进度条包括三部分,内圈圆,外圈圆弧,中间的文字,具体看代码:
protected void onDraw(Canvas canvas) {
mWidth=getMeasuredWidth();
mHeight=getMeasuredHeight();
radius=(float)(Math.min(mWidth,mHeight)*1.0/2)-strokeWidth/2;
//绘制内圈圆
mPaint.setColor(initColor);
mPaint.setStyle(Paint.Style.STROKE);
mPaint.setStrokeWidth(strokeWidth);
canvas.drawCircle(mWidth/2,mHeight/2,radius,mPaint);
//绘制覆盖的圆弧
mPaint.setColor(coverColor);
RectF rectF=new RectF(mWidth/2-radius,mHeight/2-radius,mWidth/2+radius,mHeight/2+radius);
canvas.drawArc(rectF,-90,sweepAngle,false,mPaint);
//绘制中间的文本
if(isShowProgress){
progressText=String.format(getResources().getString(R.string.progress_text),(int)(sweepAngle*100.0/360));
mPaint.setTextSize(textSize);
mPaint.setColor(textColor);
if(mBound==null){
mBound=new Rect();
}
mPaint.getTextBounds(progressText,0,progressText.length(),mBound);
mPaint.setStyle(Paint.Style.FILL);
canvas.drawText(progressText,mWidth/2-mBound.width()/2,mHeight/2+mBound.height()/2,mPaint);
}
}
当然,为了让我们可以自定义进度条的大小颜色,我们还采用了自定义属性,并且在构造器中,也需要加载xml中的各项属性:
<resources>
<declare-styleable name="CircleProgressView">
<attr name="initColor" format="color"/>
<attr name="coverColor" format="color"/>
<attr name="strokeWidth" format="dimension"/>
<attr name="progressTextSize" format="dimension"/>
<attr name="progressTextColor" format="color"/>
<attr name="isShowProgress" format="boolean"/>
</declare-styleable>
</resources>
private void initValues(Context context, AttributeSet attrs, int defStyleAttr){
TypedArray typedArray=context.getTheme().obtainStyledAttributes(attrs,R.styleable.CircleProgressView,defStyleAttr,0);
int num=typedArray.getIndexCount();
for(int i=0;i<num;i++){
int attr=typedArray.getIndex(i);
switch (attr){
case R.styleable.CircleProgressView_initColor:
initColor=typedArray.getColor(attr,Color.GRAY);
break;
case R.styleable.CircleProgressView_coverColor:
coverColor=typedArray.getColor(attr,Color.BLACK);
break;
case R.styleable.CircleProgressView_strokeWidth:
strokeWidth=typedArray.getDimensionPixelOffset(attr,5);
break;
case R.styleable.CircleProgressView_progressTextSize:
textSize=typedArray.getDimensionPixelSize(attr,30);
break;
case R.styleable.CircleProgressView_progressTextColor:
textColor=typedArray.getColor(attr,Color.BLACK);
break;
case R.styleable.CircleProgressView_isShowProgress:
isShowProgress=typedArray.getBoolean(attr,false);
break;
default:
break;
}
}
typedArray.recycle();
mPaint=new Paint();
mPaint.setAntiAlias(true);
}
源码下载
猜你喜欢
sidebarDepth: 3条件构造器说明以下出现的第一个入参boolean condition表示该条件是否加入最后生成的SQL中,例如
1.第一种方式采用System.Net.Dns的GetHostAddress的方式,具体请看代码:/// <summary> &
在编写脚本的过程中我们会遇到一些小问题比如一个的变量 为了在其他脚本中可以调用 我们需要写成public类型的这样的话在Inspector面
1.前言1.1.FastJson的介绍:JSON(javaScript Object Notation)是一种轻量级的数据交换格式。主要采用
Gson反序列化原理原理简述gson反序列化主要分为两个过程:根据TypeToken创建出对象根据json字符串解析数据,对对象属性赋值对象
在做 Android App 的时候,为了给用户省流量,为了不激起用户的愤怒,为了更好的用户体验,是需要根据用户当前网络情况来做一些调整的,
在使用ListView组件来显示列表数据时,有的时候我们需要改变列表中的数据,有以下方法:1、重新给ListView组件设置适配器这种方法重
JRebel 介绍IDEA上原生是不支持热部署的,一般更新了 Java 文件后要手动重启 Tomcat 服务器,才能生效,浪费不少生命啊。目
Purpose开发人员在合作的时候经常遇到以下场景:1.开发人员A在自己的本地数据库做了一些表结构的改动,并根据这些改动调整了DAO层的代码
使用@Async异步调用方法Async简介异步方法调用使用场景:处理日志、发送邮件、短信......spring中提供了@Async来实现异
Android 应用冷启动时,需要从Application开始启动,加载时间就会比较长,容易出现白色或黑色闪屏,观察一下手机上一些 其他AP
本文实例为大家分享了使用aop实现全局异常处理的具体代码,供大家参考,具体内容如下日常业务中存在的问题使用大量的try/catch来捕获异常
StopWatch是位于org.springframework.util包下的一个工具类,通过它可方便的对程序部分代码进行计时(ms级别),
本文主要介绍我为桌面和 Web 设计的一个超级秘密 Flutter 项目使用了画布和可拖动节点界面。本教程将展示我如何使用堆栈来使用小部件完
本文实例讲述了C#将制定目录文件名转换成大写的方法。分享给大家供大家参考。具体如下:using System;using System.IO
Android Material Design:PopupMenuAndroid Material Design 引入的PopupMenu类
前言从windows窗口的概念开始,通过对比去理解Android窗口体系,本文没有深入源码,重在理解概念代码都是抄来抄去,概念也是互相借鉴
本文以实例形式讲述了基于Java的图的广度优先遍历算法实现方法,具体方法如下:用邻接矩阵存储图方法:1.确定图的顶点个数和边的个数2.输入顶
1. 阐述对于Java中Map的遍历方式,很多文章都推荐使用entrySet,认为其比keySet的效率高很多。理由是:entrySet方法