700字范文,内容丰富有趣,生活中的好帮手!
700字范文 > Android自定义View之仿QQ运动步数进度效果

Android自定义View之仿QQ运动步数进度效果

时间:2022-11-20 07:06:32

相关推荐

Android自定义View之仿QQ运动步数进度效果

文章目录

前言先看效果图 ![在这里插入图片描述](https://img-/6e4ddec17933496ea4830fa08d8ffbe5.png?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzM1ODU3NDUx,size_16,color_FFFFFF,t_70#pic_center)一、实现思路二、实现步骤1、分析效果2、确定自定义属性,编写attrs.xml3、在布局中使用4、在自定义View中获取自定义属性5、onMeasure 确定大小6、 画外圆弧,内圆弧,文字6.1 画外圆弧(总进度)6.2. 画内圆弧(已走进度条)6.3 画文本 7、其他 设置方法7.1 设置最大值7.2 设置已走进度条7.1 设置已走进度条,实现动画效果 完整代码总结

前言

关于自定义View,仿QQ运动步数,


提示:以下是本篇文章正文内容,下面案例可供参考

先看效果图

一、实现思路

由图片可以看出,主要分为三个部分,总进度条,已走进度条,已走步数。但是又有许多可自行设置的属性,如线条宽度,总进度条颜色,已走进度条颜色,字体颜色等等。所以需要自定义属性Attrs.xml。下来可以慢慢实现效果。

二、实现步骤

1、分析效果

代码如下(示例):

2、确定自定义属性,编写attrs.xml

<declare-styleable name="QQStepView"><attr name="outerColor" format="color"/><attr name="innerColor" format="color"/><attr name="borderWidget" format="dimension"/><attr name="stepText" format="string"/><attr name="stepTextSize" format="dimension"/><attr name="stepTextColor" format="color"/></declare-styleable>

3、在布局中使用

<com.joekai.customview.views.QQStepViewandroid:id="@+id/step_view"android:layout_width="150dp"android:layout_height="150dp"app:outerColor="@color/colorPrimary"app:innerColor="@color/colorAccent"app:textColor="@color/colorPrimaryDark"app:borderWidget="5dp"app:textSize="18sp"android:background="@color/red"/>

4、在自定义View中获取自定义属性

//获取attrs.xml中设置的属性TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.QQStepView);mStepText = ta.getString(R.styleable.QQStepView_stepText);mOuterColor = ta.getColor(R.styleable.QQStepView_outerColor, mOuterColor);mInnerColor = ta.getColor(R.styleable.QQStepView_innerColor, mInnerColor);mBorderWidget = ta.getDimension(R.styleable.QQStepView_borderWidget, mBorderWidget);mTextSize = ta.getDimensionPixelSize(R.styleable.QQStepView_stepTextSize, sp2px(mTextSize));mTextColor = ta.getColor(R.styleable.QQStepView_stepTextColor, mTextColor);//回收ta.recycle();///在此处将所有画笔进行初始化//创建画笔mOuterPaint = new Paint();//取消锯齿mOuterPaint.setAntiAlias(true);//设置颜色mOuterPaint.setColor(mOuterColor);//设置圆弧宽度mOuterPaint.setStrokeWidth(mBorderWidget);//设置圆弧StylemOuterPaint.setStyle(Paint.Style.STROKE);//设置起始,结束圆角mOuterPaint.setStrokeCap(Paint.Cap.ROUND);mInnerPaint = new Paint();//取消锯齿mOuterPaint.setAntiAlias(true);//设置颜色mInnerPaint.setColor(mInnerColor);//设置圆弧宽度mInnerPaint.setStrokeWidth(mBorderWidget);//设置圆弧StylemInnerPaint.setStyle(Paint.Style.STROKE);//设置起始,结束圆角mInnerPaint.setStrokeCap(Paint.Cap.ROUND);mTextPaint = new Paint();//抗锯齿mTextPaint.setAntiAlias(true);mTextPaint.setTextSize(mTextSize);mTextPaint.setColor(mTextColor);

5、onMeasure 确定大小

@Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {super.onMeasure(widthMeasureSpec, heightMeasureSpec);//在布局文件中,可能是wrap_content ,也可能宽高不一致//宽高不一致时,取最小值,确保是个正方形//获取宽高的模式int widgetMode = MeasureSpec.getMode(widthMeasureSpec);int heightMode = MeasureSpec.getMode(heightMeasureSpec);//获取宽高的大小int widgetSize = MeasureSpec.getSize(widthMeasureSpec);int heightSize = MeasureSpec.getSize(heightMeasureSpec);switch (MeasureSpec.getMode(widthMeasureSpec)) {}setMeasuredDimension(widgetSize > heightSize ? heightSize : widgetSize, widgetSize > heightSize ? heightSize : widgetSize);}

6、 画外圆弧,内圆弧,文字

6.1 画外圆弧(总进度)

遇到的问题:边缘显示不全

原因:描边的时候设置了画笔宽度mBorderWidget,所以导致边缘超出限制。

设置画布区域:

//因为区域超出画布,所以将画笔起始点从新设置。RectF rectF = new RectF(mBorderWidget / 2, mBorderWidget / 2, getWidth() - mBorderWidget / 2, getHeight() - mBorderWidget / 2);//画圆弧 //1、画布区域,2、起始角度,3、结束角度 4、是否填充 5、画笔canvas.drawArc(rectF, mStartAngle, mEndAngle, false, mOuterPaint);

6.2. 画内圆弧(已走进度条)

//未设置最大值时,返回,不进行绘画。if (mMaxStep == 0) return;//计算百分比,用于绘画已走进度条;float sweepAngle = (float) mStep / mMaxStep;canvas.drawArc(rectF, mStartAngle, sweepAngle * mEndAngle, false, mInnerPaint);

6.3 画文本

mStepText = mStep + "";//计算的宽度 与 字体的长度有关 与字体的大小 用画笔来测量//获取文本的Rect//mTextPaint.getTextBounds(mStepText, 0, mStepText.length(), mTextBounds);//获取文本宽度//mTextBounds.width();//获取文本宽度float textWidget = mTextPaint.measureText(mStepText);//计算X起始位置int dx = (int) (getWidth() / 2 - textWidget / 2);//计算基线位置//dy代表的是:高度的一半到baseLine的距离//获取文字排版信息Paint.FontMetricsInt fontMetricsInt = mTextPaint.getFontMetricsInt();int dy = (fontMetricsInt.bottom - fontMetricsInt.top) / 2 - fontMetricsInt.bottom;//基线int baseLine = getHeight() / 2 + dy;canvas.drawText(mStepText, dx, baseLine, mTextPaint);

7、其他 设置方法

7.1 设置最大值

public synchronized void setMaxStep(int maxStep) {this.mMaxStep = maxStep;}

7.2 设置已走进度条

public synchronized void setStep(int step){this.mStep = step;//不断绘制invalidate();}

7.1 设置已走进度条,实现动画效果

public synchronized void setAnimatorStep(int step){//属性动画ValueAnimator valueAnimator = ObjectAnimator.ofFloat(0, step);valueAnimator.setDuration(1000);valueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {@Overridepublic void onAnimationUpdate(ValueAnimator valueAnimator) {float step = (float) valueAnimator.getAnimatedValue();setStep((int) step);}});valueAnimator.start();}

完整代码

/*** 仿QQ运动步数*/public class QQStepView extends View {private int mOuterColor = Color.RED;//总进度条颜色private int mInnerColor = Color.BLUE;//已走颜色private float mBorderWidget = 20;//圆弧宽度private int mTextSize = 20;//字体大小private String mStepText;private int mTextColor = Color.GREEN;//字体颜色private Paint mOuterPaint;//总画笔private Paint mInnerPaint;//已走画笔private Paint mTextPaint;//已走画笔private int mStartAngle = 135;private int mEndAngle = 270;private int mMaxStep = 2000;private int mStep = 1000;private Rect mTextBounds ;public QQStepView(Context context) {this(context, null);}public QQStepView(Context context, @Nullable AttributeSet attrs) {this(context, attrs, 0);}public QQStepView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);// 1 分析效果// 2 确定自定义属性,编写attrs.xml// 3 在布局中使用// 4 在自定义View中获取自定义属性// 7 其他TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.QQStepView);mStepText = ta.getString(R.styleable.QQStepView_stepText);mOuterColor = ta.getColor(R.styleable.QQStepView_outerColor, mOuterColor);mInnerColor = ta.getColor(R.styleable.QQStepView_innerColor, mInnerColor);mBorderWidget = ta.getDimension(R.styleable.QQStepView_borderWidget, mBorderWidget);mTextSize = ta.getDimensionPixelSize(R.styleable.QQStepView_stepTextSize, sp2px(mTextSize));mTextColor = ta.getColor(R.styleable.QQStepView_stepTextColor, mTextColor);//回收ta.recycle();//创建画笔mOuterPaint = new Paint();//取消锯齿mOuterPaint.setAntiAlias(true);//设置颜色mOuterPaint.setColor(mOuterColor);//设置圆弧宽度mOuterPaint.setStrokeWidth(mBorderWidget);//设置圆弧StylemOuterPaint.setStyle(Paint.Style.STROKE);//设置起始,结束圆角mOuterPaint.setStrokeCap(Paint.Cap.ROUND);mInnerPaint = new Paint();//取消锯齿mOuterPaint.setAntiAlias(true);//设置颜色mInnerPaint.setColor(mInnerColor);//设置圆弧宽度mInnerPaint.setStrokeWidth(mBorderWidget);//设置圆弧StylemInnerPaint.setStyle(Paint.Style.STROKE);//设置起始,结束圆角mInnerPaint.setStrokeCap(Paint.Cap.ROUND);mTextPaint = new Paint();//抗锯齿mTextPaint.setAntiAlias(true);mTextPaint.setTextSize(mTextSize);mTextPaint.setColor(mTextColor);mTextBounds = new Rect();}/*** 将sp值转换为px值,保证文字大小不变** @param spValue* @return*/public static int sp2px(int spValue) {final float fontScale = BaseApplication.getInstance().getResources().getDisplayMetrics().scaledDensity;return (int) (spValue * fontScale + 0.5f);}// 5 onMeasure@Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {super.onMeasure(widthMeasureSpec, heightMeasureSpec);//在布局文件中,可能是wrap_content ,也可能宽高不一致//高度不一致,取最大值,确保是个正方形//获取宽高的模式int widgetMode = MeasureSpec.getMode(widthMeasureSpec);int heightMode = MeasureSpec.getMode(heightMeasureSpec);//获取宽高的大小int widgetSize = MeasureSpec.getSize(widthMeasureSpec);int heightSize = MeasureSpec.getSize(heightMeasureSpec);switch (MeasureSpec.getMode(widthMeasureSpec)) {}setMeasuredDimension(widgetSize > heightSize ? heightSize : widgetSize, widgetSize > heightSize ? heightSize : widgetSize);}// 6 画外圆弧,内圆弧,文字@Overrideprotected void onDraw(Canvas canvas) {super.onDraw(canvas);//6.1 画外圆弧//问题:边缘显示不全 描边有宽度 mBorderWidget//获取圆点int center = getWidth() / 2;//获取半径int radius = (int) (center - mBorderWidget / 2);RectF rectF = new RectF(mBorderWidget / 2, mBorderWidget / 2, getWidth() - mBorderWidget / 2, getHeight() - mBorderWidget / 2);canvas.drawArc(rectF, mStartAngle, mEndAngle, false, mOuterPaint);//6.2 画内圆弧if (mMaxStep == 0) return;float sweepAngle = (float) mStep / mMaxStep;canvas.drawArc(rectF, mStartAngle, sweepAngle * mEndAngle, false, mInnerPaint);//6.3 画文本mStepText = mStep + "";//计算的宽度 与 字体的长度有关 与字体的大小 用画笔来测量//获取文本的Rect// mTextPaint.getTextBounds(mStepText, 0, mStepText.length(), mTextBounds);//获取文本宽度float textWidget = mTextPaint.measureText(mStepText);int dx = (int) (getWidth() / 2 - textWidget / 2);//dy代表的是:高度的一半到baseLine的距离//获取文字排版信息Paint.FontMetricsInt fontMetricsInt = mTextPaint.getFontMetricsInt();int dy = (fontMetricsInt.bottom - fontMetricsInt.top) / 2 - fontMetricsInt.bottom;//基线int baseLine = getHeight() / 2 + dy;canvas.drawText(mStepText, dx, baseLine, mTextPaint);}// 7 其他public synchronized void setMaxStep(int maxStep) {this.mMaxStep = maxStep;}public synchronized void setStep(int step){this.mStep = step;//不断绘制invalidate();}public synchronized void setAnimatorStep(int step){//属性动画ValueAnimator valueAnimator = ObjectAnimator.ofFloat(0, step);valueAnimator.setDuration(1000);valueAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {@Overridepublic void onAnimationUpdate(ValueAnimator valueAnimator) {float step = (float) valueAnimator.getAnimatedValue();setStep((int) step);}});valueAnimator.start();}}


总结

提示:这里对文章进行总结:

以上就是今天要讲的内容,本文仅仅简单介绍了自定义仿QQ运动步数,扩展效果可自己再去实现。

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。