Android实现光点模糊渐变的自旋转圆环特效

本文实例为大家分享了Android实现光点模糊渐变的自旋转圆环效果,供大家参考,具体内容如下

项目中需要实现的效果图如下:

可以这个表盘看到中间部分都是没有什么难点的,主要是周围圆环的三种效果:

1.渐变色

2.尖端的白点模糊效果

3.路径绘制

最终实现的效果图如下:

完美实现了三点要求。

实现思路:

1.首先是黑色底色圆环的绘制(黑色圈是固定不变的)。

2.在绘制好黑色底色圆环之后再绘制渐变色圆弧(蓝绿部分)。

3.最后绘制小星星部分,使用一张模糊图片得到bitmap,并通过PathMeasure进行路径绘制。

代码实现:

/**
 * Created by jiangzn on 17/2/3.
 */
public class RoundLightBarView extends ImageView {
  private int mTotalWidth, mTotalHeight;
  private int mCenterX, mCenterY;
  //底色画笔
  private Paint mCirclePaint;
  //进度条画笔
  private Paint mProgressPaint;
  //圆点画笔
  private Paint mbitmapPaint;
  private Matrix mMatrix; // 矩阵,用于对图片进行一些操作
  private float[] pos;   // 当前点的实际位置
  private float[] tan;   // 当前点的tangent值,用于计算图片所需旋转的角度

  private int mCircleR;

  private Context mContext;
  //距离外围的边距
  private float interval ;

  private int startAngle = 1;

  //球
  private Bitmap mLititleBitmap; // 圆点图片

  public RoundLightBarView(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);

  }

  public RoundLightBarView(Context context, AttributeSet attrs) {
    super(context, attrs);
    mContext = context;
    interval = DensityUtils.px2dip(mContext, 50);
    //初始化画笔
    initPaint();
    //初始化bitmap
    initBitmap();
  }

  private void initBitmap() {
    mMatrix=new Matrix();
    pos = new float[2];
    tan = new float[2];
    mLititleBitmap= ((BitmapDrawable) getResources()
        .getDrawable(R.mipmap.white_round))
        .getBitmap();
  }

  private void initPaint() {
    //画黑底的深色圆画笔
    mCirclePaint = new Paint();
    //抗锯齿
    mCirclePaint.setAntiAlias(true);
    // 防抖动
    mCirclePaint.setDither(true);
    // 开启图像过滤,对位图进行滤波处理。
    mCirclePaint.setFilterBitmap(true);
    mCirclePaint.setColor(Color.BLACK);
    //空心圆
    mCirclePaint.setStyle(Paint.Style.STROKE);
    //圆半径
    mCircleR = DensityUtils.px2dip(mContext, 20);
    mCirclePaint.setStrokeWidth(mCircleR);

    //画彩色圆弧的画笔
    mProgressPaint = new Paint();
    //抗锯齿
    mProgressPaint.setAntiAlias(true);
    // 防抖动
    mProgressPaint.setDither(true);
    // 开启图像过滤,对位图进行滤波处理。
    mProgressPaint.setFilterBitmap(true);
    mProgressPaint.setColor(Color.BLUE);
    //空心圆
    mProgressPaint.setStyle(Paint.Style.STROKE);
    //设置笔刷样式为原型
    mProgressPaint.setStrokeCap(Paint.Cap.ROUND);
    //设置圆弧粗
    mProgressPaint.setStrokeWidth(mCircleR);
    //将绘制的内容显示在第一次绘制内容之上
    mProgressPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_ATOP));

    //圆点画笔
    mbitmapPaint = new Paint();
    mbitmapPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_ATOP));
    mbitmapPaint.setStyle(Paint.Style.FILL);
    mbitmapPaint.setAntiAlias(true);

  }

  @Override
  protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    //canvas去锯齿
    canvas.setDrawFilter(
        new PaintFlagsDrawFilter(0, Paint.ANTI_ALIAS_FLAG | Paint.FILTER_BITMAP_FLAG));
    //画底色圆
    canvas.drawCircle(mCenterX, mCenterY, mCenterX - mCircleR - interval, mCirclePaint);
    //画进度条
    int colorSweep[] = {Color.TRANSPARENT, Color.parseColor("#3bbaea"),Color.parseColor("#7ac9d3"),Color.parseColor("#7cc9d0")};

    //设置渐变色
    sweepGradient = new SweepGradient(mCenterX, mCenterY, colorSweep, null);
    //按照圆心旋转
    Matrix matrix = new Matrix();
    matrix.setRotate(startAngle, mCenterX, mCenterY);
    sweepGradient.setLocalMatrix(matrix);

    mProgressPaint.setShader(sweepGradient);

    canvas.drawArc(
        new RectF(0 + mCircleR + interval, 0 + mCircleR + interval,
            mTotalWidth - mCircleR - interval, mTotalHeight - mCircleR - interval),
        2 + startAngle, 350, false, mProgressPaint);

    startAngle++;
    if (startAngle == 360) {
      startAngle = 1;
    }

    //绘制白色小星星
    Path orbit = new Path();
    //通过Path类画一个90度(180—270)的内切圆弧路径
    orbit.addArc(
        new RectF(0 + mCircleR + interval, 0 + mCircleR + interval,
            mTotalWidth - mCircleR - interval, mTotalHeight - mCircleR - interval)
        , 2 + startAngle, 350);
    // 创建 PathMeasure
    PathMeasure measure = new PathMeasure(orbit, false);
    measure.getPosTan(measure.getLength() * 1, pos, tan);
    mMatrix.reset();
    mMatrix.postScale(2,2);
    mMatrix.postTranslate(pos[0] - mLititleBitmap.getWidth() , pos[1] - mLititleBitmap.getHeight() );  // 将图片绘制中心调整到与当前点重合
    canvas.drawBitmap(mLititleBitmap, mMatrix, mbitmapPaint);//绘制球
    mbitmapPaint.setColor(Color.WHITE);
    //绘制实心小圆圈
    canvas.drawCircle(pos[0], pos[1], 5, mbitmapPaint);

    //启动绘制
    postInvalidateDelayed(10);
  }

  SweepGradient sweepGradient;

  @Override
  protected void onSizeChanged(int w, int h, int oldw, int oldh) {
    super.onSizeChanged(w, h, oldw, oldh);
    mTotalWidth = w;
    mTotalHeight = h;
    mCenterX = mTotalWidth / 2;
    mCenterY = mTotalHeight / 2;

  }

}

总结:

总体实现难度并不大,复习了自定义View和canvas的知识点。

其中需要重视的点在绘图层需要注意给画笔添加覆盖模式:setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_ATOP)),将绘制的内容显示在第一次绘制的内容之上,还有一个比较难的点是PathMeasure进行对bitmap的路径收集和方向控制并绘制小星星的过程:

 // 创建 PathMeasure
    PathMeasure measure = new PathMeasure(orbit, false);
    measure.getPosTan(measure.getLength() * 1, pos, tan);
    mMatrix.reset();
    mMatrix.postScale(2,2);
    mMatrix.postTranslate(pos[0] - mLititleBitmap.getWidth() , pos[1] - mLititleBitmap.getHeight() );  // 将图片绘制中心调整到与当前点重合
    canvas.drawBitmap(mLititleBitmap, mMatrix, mbitmapPaint);//绘制球
    mbitmapPaint.setColor(Color.WHITE);
    //绘制实心小圆圈
    canvas.drawCircle(pos[0], pos[1], 5, mbitmapPaint);

源码下载:Android实现光点模糊渐变的自旋转圆环特效

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

(0)

相关推荐

  • Android自定义View实现圆环交替效果

    下面请先看效果图: 看上去是不很炫的样子,它的实现上也不是很复杂,重点在与onDraw()方法的绘制. 首先是我们的attrs文件: <?xml version="1.0" encoding="utf-8"?> <resources> <attr name="firstColor" format="color"/> <attr name="secondColor"

  • android自定义View实现圆环颜色选择器

    最近工作需要,自定了一个颜色选择器,效果图如下: 颜色种类是固定的,圆环上有个指示器,指示选中的颜色,这个定义起来应该是很简单了,直接上代码. public class MyColorPicker extends View { private int mThumbHeight; private int mThumbWidth; private String[] colors ; private int sections; //每个小块的度数 private int sectionAngle; p

  • Android中自定义View实现圆环等待及相关的音量调节效果

    圆环交替.等待效果 效果就这样,分析了一下,大概有这几个属性,两个颜色,一个速度,一个圆环的宽度. 自定View的几个步骤: 1.自定义View的属性 2.在View的构造方法中获得我们自定义的属性 3.重写onMesure  4.重写onDraw 1.自定义属性: <?xml version="1.0" encoding="utf-8"?> <resources> <attr name="firstColor" f

  • Android开发笔记之:在ImageView上绘制圆环的实现方法

    绘制圆环其实很简单,有大概以下三种思路. 这里先说网上提到的一种方法.思路是先绘制内圆,然后绘制圆环(圆环的宽度就是paint设置的paint.setStrokeWidth的宽度),最后绘制外圆.请看核心源码: 复制代码 代码如下: <SPAN xmlns="http://www.w3.org/1999/xhtml">package yan.guoqi.rectphoto;import android.content.Context;import android.graph

  • 利用Android模仿微信摄像圆环进度效果实例

    前言 大家在平时的生活上遇到新奇的事情,都要立即打开微信视频录下来发给朋友看看.这个录制进度条看起来还不错哦,就仿着写了一个,不是样式完全的高仿,是功能的仿制.下面话不多说了,来一起看看详细的介绍吧. 微信效果: 源码下载: github代码直通车 本地下载 自制效果: 实现过程: 1.自定义圆半径和圆环颜色属性: <declare-styleable name="CiclePercentView"> <attr name="radius" for

  • Android自定义View之酷炫数字圆环

    先看下最终的效果 一.开始实现 新建一个DoughnutView继承View public class DoughnutView extends View { } 先重写onMeasure方法. /** * 当布局为wrap_content时设置默认长宽 * * @param widthMeasureSpec * @param heightMeasureSpec */ @Override protected void onMeasure(int widthMeasureSpec, int hei

  • Android自定义view绘制圆环占比动画

    一.实现效果图 二.核心代码 1.自定义MyProgressView.java package com.czhappy.effectdemo.view; import android.content.Context; import android.content.res.TypedArray; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.graphics.Canvas;

  • Android实现动态圆环的图片头像控件

    先看效果图: 现在大部分的app上难免会使用到圆形头像,所以今天我给大家分享一个单独使用的,并且周围带有圆环动画的花哨圆形头像控件,本控件是在圆形头像控件基础上实现的,只是在其周围再画一些不同大小的圆而已,就可以实现如图的效果. 圆形头像的基本原理是将设置的资源文件转化成Bitmap,然后通过BitmapShader类将Bitmap成为Paint的渲染器,然后在onDraw()中通过canvas.drawCircle(rx,ry,radius,paint);画布上画圆,而这个圆就是形成了圆形头像

  • Android自定义带动画的半圆环型进度效果

    本文实例为大家分享了Android半圆环型进度效果的具体代码,供大家参考,具体内容如下 package com.newair.ondrawtext; import android.animation.ValueAnimator; import android.annotation.TargetApi; import android.content.Context; import android.content.res.TypedArray; import android.graphics.Can

  • Android自定义View之酷炫圆环(二)

    先看下最终的效果 静态: 动态: 一.开始实现 新建一个DoughnutProgress继承View public class DoughnutProgress extends View { } 先给出一些常量.变量以及公共方法的代码,方便理解后面的代码 private static final int DEFAULT_MIN_WIDTH = 400; //View默认最小宽度 private static final int RED = 230, GREEN = 85, BLUE = 35;

随机推荐