Android实现带有指示器的进度条

背景

当我们看到UI给我们设计的效果的时候,我们习惯性的思路就是看看google有没有为我们提供相应的控件或者是能否在网上找到一些合适的轮子拿过来直接用。但是,有时候很不巧的是没有这样的轮子供我们直接使用,而且,UI以及产品就需要这样的效果的时候,我们就只能自己进行实现绘制。今天呢,我就带大家从简单的实现一个带有指示器的进度条,帮大家理解一下之定义View的绘制流程。

效果实现图

俗话说,没有效果图,就等于是耍那个啥,所以按照惯例,贴图:

开干

当我们决定要自己进行绘制的时候,我们首先需要对需求,效果进行观察、分析、拆解。就是所谓的大事化小,小事化无。

  1. 观察:效果图中包含 进度条、带三角箭头的圆角指示器、进度值以及进度动画
  2. 分析:通过观察我们可以将整个进度条分为 指示器,进度条、进度值、动画4部分
  3. 拆解:因此我们可以依次对上述部分的组成部分进行单独的绘制最后整体关联是实现上图的效果

自定义View的流程(按照上述部分进行)

大家应该都知道绘制流程包含三部分,自定义属性、测量 measure、draw、layout。但是,我们今天所讲的是进度条View,而不是ViewLayout,所以layout的部分今天暂且不讲。

自定义属性以及代码中获取

属性 说明
app:hpb_centerPadding="5dp" 指示器与进度条的间距
app:hpb_progressBarBackgroundColor="@color/colorPrimary" 进度条背景色
app:hpb_progressBarForegroundColor="@color/colorAccent" 进度条前景色
app:hpb_progressBarHeight="2dp" 指示器bottomPadding
app:hpb_textBottomPadding="5dp" 指示器与进度条的间距
app:hpb_textLeftPadding="5dp" 指示器leftPadding
app:hpb_textRightPadding="5dp" 指示器rightPadding
app:hpb_textTopPadding="5dp" 指示器topPadding
app:hpb_textSize="12sp" 指示器文字大小
app:hpb_textColor="#FFFFFF" 指示器文字颜色
app:hpb_progress="20" 进度值
    //获取自定义属性
    TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.HorizontalProgressBar);
    mTextSize = ta.getDimension(R.styleable.HorizontalProgressBar_hpb_textSize, Utils.sp2px(context, 12));
    mTextColor = ta.getColor(R.styleable.HorizontalProgressBar_hpb_textColor, Color.parseColor("#FFFFFF"));
    currentProgress = ta.getFloat(R.styleable.HorizontalProgressBar_hpb_progress, 0);
    mTextLeftPadding = (int) ta.getDimension(R.styleable.HorizontalProgressBar_hpb_textLeftPadding, Utils.sp2px(context, 5));
    mTextRightPadding = (int) ta.getDimension(R.styleable.HorizontalProgressBar_hpb_textRightPadding, Utils.sp2px(context, 5));
    mTextTopPadding = (int) ta.getDimension(R.styleable.HorizontalProgressBar_hpb_textTopPadding, Utils.sp2px(context, 5));
    mTextBottomPadding = (int) ta.getDimension(R.styleable.HorizontalProgressBar_hpb_textBottomPadding, Utils.sp2px(context, 5));
    mCenterPadding = (int) ta.getDimension(R.styleable.HorizontalProgressBar_hpb_centerPadding, Utils.sp2px(context, 5));
    mProgressHeight = (int) ta.getDimension(R.styleable.HorizontalProgressBar_hpb_progressBarHeight, Utils.sp2px(context, 2));
    mBackgroundColor = ta.getColor(R.styleable.HorizontalProgressBar_hpb_progressBarBackgroundColor, Color.parseColor("#E8E8E8"));
    mForegroundColor = ta.getColor(R.styleable.HorizontalProgressBar_hpb_progressBarForegroundColor, Color.parseColor("#912CEE"));
    ta.recycle();

    //进度条条背景色画笔初始化
    mBackgroundPaint = new Paint();
    mBackgroundPaint.setColor(mBackgroundColor);
    mBackgroundPaint.setStrokeCap(Paint.Cap.ROUND);
    mBackgroundPaint.setStrokeWidth(mProgressHeight);
    mBackgroundPaint.setAntiAlias(true);

    //指示器Path画笔初始化
    mPathPaint = new Paint();
    mPathPaint.setColor(mForegroundColor);
    mPathPaint.setStrokeCap(Paint.Cap.ROUND);
    mPathPaint.setPathEffect(new CornerPathEffect(Utils.dp2px(getContext(), 2)));
    mPathPaint.setAntiAlias(true);

    //进度条前景色画笔初始化
    mProgressPaint = new Paint();
    mProgressPaint.setColor(mForegroundColor);
    mProgressPaint.setStrokeWidth(mProgressHeight);
    mProgressPaint.setStrokeCap(Paint.Cap.ROUND);
    mProgressPaint.setAntiAlias(true);

    //进度值文案画笔初始化
    mTextPaint = new Paint();
    mTextPaint.setColor(mTextColor);
    mTextPaint.setTextSize(mTextSize);
    mTextPaint.setStyle(Paint.Style.FILL);

    //指示器Path初始化
    mPath = new Path();

测量

@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    int widthMode = MeasureSpec.getMode(widthMeasureSpec);
    int width = MeasureSpec.getSize(widthMeasureSpec);

    int heightMode = MeasureSpec.getMode(heightMeasureSpec);
    int height = MeasureSpec.getSize(heightMeasureSpec);

    setMeasuredDimension(measureWidth(widthMode, width), measureHeight(heightMode, height));
}

/**
 * 测量宽度
 *
 * @param mode
 * @param width
 * @return
 */
private int measureWidth(int mode, int width) {
    switch (mode) {
        case MeasureSpec.UNSPECIFIED:
        case MeasureSpec.AT_MOST:
        case MeasureSpec.EXACTLY:
            mWidth = width;
            break;
    }
    return mWidth;
}

/**
 * 测量高度
 *
 * @param mode
 * @param height
 * @return
 */
private int measureHeight(int mode, int height) {
    switch (mode) {
        case MeasureSpec.UNSPECIFIED:
        case MeasureSpec.AT_MOST:
        case MeasureSpec.EXACTLY:
            mHeight = height;
            break;
    }
    return mHeight;
}

//文字测量
private void measureText(String text) {
    Rect rect = new Rect();
    mTextPaint.getTextBounds(text, 0, text.length(), rect);
    mTextWidth = rect.width();
    mTextHeight = rect.height();
}

绘制

@Override
protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    mIndicatorWidth = mTextLeftPadding + mTextWidth + mTextRightPadding;
    mIndicatorHeight = mTextTopPadding + mTextHeight + mTextBottomPadding;

    float backgroundProgressBarStartX = (float) (1.0 * mIndicatorWidth / 2);
    float backgroundProgressBarEndX = mWidth - mIndicatorWidth / 2;

    float foregroundProgress = (float) (1.0 * (mWidth - mIndicatorWidth) * currentProgress / 100);

    float foregroundProgressBarStartX = (float) (1.0 * mIndicatorWidth / 2);
    float foregroundProgressBarEndX = foregroundProgressBarStartX + foregroundProgress;

    Log.e(TAG, "backgroundProgressBarStartX----" + backgroundProgressBarStartX
            + "----backgroundProgressBarEndX----" + backgroundProgressBarEndX
            + "----foregroundProgress----" + foregroundProgress
            + "----foregroundProgressBarStartX----" + foregroundProgressBarStartX
            + "----foregroundProgressBarEndX----" + foregroundProgressBarEndX
    );

    float progressX = foregroundProgress;
    float progressY = (float) (mCenterPadding + mIndicatorHeight * 1.5);

    //进度背景
    canvas.drawLine(backgroundProgressBarStartX, progressY, backgroundProgressBarEndX, progressY, mBackgroundPaint);

    //进度前景色
    canvas.drawLine(foregroundProgressBarStartX, progressY, foregroundProgressBarEndX, progressY, mProgressPaint);

    //指示器
    drawPath(canvas, progressX, 45);

    //文字
    drawText(canvas, mTextProgress, progressX);
}

1. 指示器绘制

通过Path进行绘制,描绘出矩形以及下三角

/**
 * 绘制指示器
 *
 * @param canvas
 * @param progressX
 */
private void drawPath(Canvas canvas, float progressX, float rotateAngle) {
    mPath.reset();

    mPath.moveTo(progressX, 0); //1
    mPath.lineTo(progressX + mIndicatorWidth, 0); //2
    mPath.lineTo(progressX + mIndicatorWidth, mIndicatorHeight); //3
    mPath.lineTo((float) (progressX + mIndicatorWidth * 0.75), mIndicatorHeight); //4
    mPath.lineTo((float) (progressX + mIndicatorWidth * 0.5), (float) (mIndicatorHeight * 1.5)); //5
    mPath.lineTo((float) (progressX + mIndicatorWidth * 0.25), mIndicatorHeight); //6
    mPath.lineTo(progressX, mIndicatorHeight);//7
    mPath.close();

    canvas.drawPath(mPath, mPathPaint);
}

2. 文字的绘制

文字的绘制涉及到测量上边测量中已经提及,但是在绘制中有涉及到基线获取如下所示

/**
 * 计算绘制文字时的基线到中轴线的距离
 *
 * @param p
 * @return 基线和centerY的距离
 */
public float getBaseline(Paint p) {
    Paint.FontMetrics fontMetrics = p.getFontMetrics();
    return (fontMetrics.descent - fontMetrics.ascent) / 2 - fontMetrics.descent;
}

/**
 * 绘制文字
 *
 * @param canvas
 * @param text
 * @param progressX
 */
private void drawText(Canvas canvas, String text, float progressX) {
    float baseline = getBaseline(mTextPaint);
    float textY = mTextTopPadding + mTextHeight / 2 + baseline;
    canvas.drawText(text, progressX + mTextTopPadding, textY, mTextPaint);
}

总结

以上的完整代码已经上传至github,github地址,如有需要可以上去看看,也可以下载直接使用或者是通过jcenter引入,由于jcenter现状已经停止维护,所以后续会迁入maven

以上就是Android实现带有指示器的进度条的详细内容,更多关于Android 指示器进度条的资料请关注我们其它相关文章!

(0)

相关推荐

  • android自定义等级评分圆形进度条

    本文实例为大家分享了android评分圆形进度条的具体代码,供大家参考,具体内容如下 一.测试截图 二.实现原理 package com.freedomanlib; import java.util.Timer; import java.util.TimerTask; import android.annotation.SuppressLint; import android.content.Context; import android.graphics.Canvas; import andr

  • Android自定义圆形进度条效果

    本文实例为大家分享了Android自定义圆形进度条效果的具体代码,供大家参考,具体内容如下 1 控件 RoundProgress package listview.tianhetbm.p2p.ui; import android.content.Context; import android.content.res.TypedArray; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import a

  • Android progressbar实现带底部指示器和文字的进度条

    本文实例为大家分享了Android实现带指示器和文字的进度条,供大家参考,具体内容如下 根据项目要求需要实现以下效果: 列出源码: public class TextProgressBar extends LinearLayout { String text; Paint mPaint; private Rect textRect; private Bitmap bitmap; private ProgressBar progressBar; int progress; int proWidth

  • Android 进度条自动前进效果的实现代码

    今天给大家分享进度条自动前进功能的实现,先给大家分享实现效果图,感觉不错可以参考实现代码. 效果如下图: 首先布局要设置进度条最大值: <ProgressBar android:id="@+id/pro1" style="@android:style/Widget.ProgressBar.Horizontal" android:layout_width="400dp" android:layout_centerHorizontal=&quo

  • Android自定义控件之圆形进度条动画

    本文实例为大家分享了Android实现圆形进度条动画的具体代码,供大家参考,具体内容如下 首先贴上图片: 额,感觉还行吧,就是进度条的颜色丑了点,不过咱是程序员,不是美工,配色这种问题当然不在考虑范围之内了. 下面说重点,如何来写一个这样的自定义控件. 首先,需要有一个灰色的底图,来作为未填充时的进度条: 然后,根据传入的当前进度值,绘制填充时的进度圆弧,这段圆弧所对应的圆心角,由当前进度与进度的最大值(一般是100)的比值计算得出: 其次,根据进度值绘制文字提示: 最后,重绘控件,加上动画,从

  • Android seekbar实现可拖动进度条

    本文实例为大家分享了Android seekbar实现可拖动进度条的具体代码,供大家参考,具体内容如下 SeekBar通过滑块的位置来标识数值 允许用户通过拖动滑块来改变进度值的大小 控件:SeekBar            两个TextView 显示状态 实现SeekBar.OnSeekBarChangeListener接口 对事件进行监听 xml文件: <?xml version="1.0" encoding="utf-8"?> <Linea

  • Android Studio实现进度条效果

    本文实例为大家分享了Android Studio实现进度条效果的具体代码,供大家参考,具体内容如下 实验作业 要求一个进度条,进度随机 效果图 xml代码 <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://s

  • Android自定义View实现圆形进度条

    本文实例为大家分享了Android自定义View实现圆形进度条的具体代码,供大家参考,具体内容如下 效果如下: 主要代码 CircularProgressView.java public class CircularProgressView extends View { private Paint mBackPaint, mProgPaint; // 绘制画笔 private RectF mRectF; // 绘制区域 private int[] mColorArray; // 圆环渐变色 pr

  • Android实现进度条(ProgressBar)的功能与用法

    进度条(ProgressBar)的功能与用法,供大家参考,具体内容如下 进度条是UI界面中一种实用的UI组件,用于显示一个耗时操作显示出来的百分比,进度条可以动态的显示进度,避免是用户觉得系统长时间未反应,提高用户的体验. 下面程序简单示范了进度条的用法,界面布局文件如下: 在layout下的activity_main中: <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:andr

  • Android自定义圆弧进度条加数字动态变化

    本文实例为大家分享了Android自定义圆弧进度条数字变化的具体代码,供大家参考,具体内容如下 效果如下: 思路:一个内环圆弧和一个外环圆弧,因为有一个圆圈是在圆弧上做圆周运动,所以在画圆的时候必须要得到圆弧上的各个点的坐标,这里其实就用到了PathMeasure这个类,可以帮我们拿到这些点,在画圆弧的时候也理所应当的要使用path,然后根据外界动态的传值进行重绘就能达到动态的效果 代码如下: public class ProgressPathRainbow extends View { pri

  • Android自定义分段式进度条

    安卓自定义分段式的进度条,供大家参考,具体内容如下 前一段时间公司新项目接到一个新需求,其中界面需要用到一个分段式的进度条,找了半天没有发现类似的控件,于是决定自己写一个,话不多说,上代码 package com.djt.aienglish.widget; import android.content.Context; import android.content.res.TypedArray; import android.graphics.Canvas; import android.gra

  • Android ProgressBar 模拟进度条效果的实现

    进度条的使用 圆形进度条 <ProgressBar android:id="@+id/pb" android:layout_width="50dp" android:layout_height="50dp" /> 如图 长条不显示进度进度条 <ProgressBar android:id="@+id/pb2" android:layout_width="300dp" android:lay

随机推荐