热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android编程使用自定义View实现水波进度效果示例

这篇文章主要介绍了Android编程使用自定义View实现水波进度效果,结合实例形式详细分析了Android水波动画效果的具体实现步骤与相关注意事项,需要的朋友可以参考下

本文实例讲述了Android编程使用自定义View实现水波进度效果。分享给大家供大家参考,具体如下:

首先上效果图:

简介:

1.自动适应屏幕大小;
2.水波自动横向滚动;
3.各种绘制参数可通过修改常量进行控制。

代码不多,注释也比较详细,全部贴上:

(一)自定义组件:

/**
 * 水波进度效果.
 */
public class WaterWaveView extends View {
  //边框宽度
  private int STROKE_WIDTH;
  //组件的宽,高
  private int width, height;
  /**
   * 进度条最大值和当前进度值
   */
  private float max, progress;
  /**
   * 绘制波浪的画笔
   */
  private Paint progressPaint;
  //波纹振幅与半径之比。(建议设置:<0.1)
  private static final float A = 0.05f;
  //绘制文字的画笔
  private Paint textPaint;
  //绘制边框的画笔
  private Paint circlePaint;
  /**
   * 圆弧圆心位置
   */
  private int centerX, centerY;
  //内圆所在的矩形
  private RectF circleRectF;
  public WaterWaveView(Context context) {
    super(context);
    init();
  }
  public WaterWaveView(Context context, AttributeSet attrs) {
    super(context, attrs);
    init();
  }
  public WaterWaveView(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    init();
  }
  //初始化
  private void init() {
    progressPaint = new Paint();
    progressPaint.setColor(Color.parseColor("#77cccc88"));
    progressPaint.setAntiAlias(true);
    textPaint = new Paint();
    textPaint.setColor(Color.WHITE);
    textPaint.setAntiAlias(true);
    circlePaint = new Paint();
    circlePaint.setStyle(Paint.Style.STROKE);
    circlePaint.setAntiAlias(true);
    circlePaint.setColor(Color.parseColor("#33333333"));
    autoRefresh();
  }
  @Override
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    if (width == 0 || height == 0) {
      width = getWidth();
      height = getHeight();
      //计算圆弧半径和圆心点
      int circleRadius = Math.min(width, height) >> 1;
      STROKE_WIDTH = circleRadius / 10;
      circlePaint.setStrokeWidth(STROKE_WIDTH);
      centerX = width / 2;
      centerY = height / 2;
      VALID_RADIUS = circleRadius - STROKE_WIDTH;
      RADIANS_PER_X = (float) (Math.PI / VALID_RADIUS);
      circleRectF = new RectF(centerX - VALID_RADIUS, centerY - VALID_RADIUS,
          centerX + VALID_RADIUS, centerY + VALID_RADIUS);
    }
  }
  private Rect textBounds = new Rect();
  //x方向偏移量
  private int xOffset;
  @Override
  protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    //绘制圆形边框
    canvas.drawCircle(centerX, centerY, VALID_RADIUS + (STROKE_WIDTH >> 1), circlePaint);
    //绘制水波曲线
    canvas.drawPath(getWavePath(xOffset), progressPaint);
    //绘制文字
    textPaint.setTextSize(VALID_RADIUS >> 1);
    String text1 = String.valueOf(progress);
    //测量文字长度
    float w1 = textPaint.measureText(text1);
    //测量文字高度
    textPaint.getTextBounds("8", 0, 1, textBounds);
    float h1 = textBounds.height();
    float extraW = textPaint.measureText("8") / 3;
    canvas.drawText(text1, centerX - w1 / 2 - extraW, centerY + h1 / 2, textPaint);
    textPaint.setTextSize(VALID_RADIUS / 6);
    textPaint.getTextBounds("M", 0, 1, textBounds);
    float h2 = textBounds.height();
    canvas.drawText("M", centerX + w1 / 2 - extraW + 5, centerY - (h1 / 2 - h2), textPaint);
    String text3 = "共" + String.valueOf(max) + "M";
    float w3 = textPaint.measureText(text3, 0, text3.length());
    textPaint.getTextBounds("M", 0, 1, textBounds);
    float h3 = textBounds.height();
    canvas.drawText(text3, centerX - w3 / 2, centerY + (VALID_RADIUS >> 1) + h3 / 2, textPaint);
    String text4 = "流量剩余";
    float w4 = textPaint.measureText(text4, 0, text4.length());
    textPaint.getTextBounds(text4, 0, text4.length(), textBounds);
    float h4 = textBounds.height();
    canvas.drawText(text4, centerX - w4 / 2, centerY - (VALID_RADIUS >> 1) + h4 / 2, textPaint);
  }
  //绘制水波的路径
  private Path wavePath;
  //每一个像素对应的弧度数
  private float RADIANS_PER_X;
  //去除边框后的半径(即内圆半径)
  private int VALID_RADIUS;
  /**
   * 获取水波曲线(包含圆弧部分)的Path.
   *
   * @param xOffset x方向像素偏移量.
   */
  private Path getWavePath(int xOffset) {
    if (wavePath == null) {
      wavePath = new Path();
    } else {
      wavePath.reset();
    }
    float[] startPoint = new float[2]; //波浪线起点
    float[] endPoint = new float[2]; //波浪线终点
    for (int i = 0; i <= VALID_RADIUS * 2; i += 2) {
      float x = centerX - VALID_RADIUS + i;
      float y = (float) (centerY + VALID_RADIUS * (1.0f + A) * 2 * (0.5f - progress / max)
          + VALID_RADIUS * A * Math.sin((xOffset + i) * RADIANS_PER_X));
      //只计算内圆内部的点,边框上的忽略
      if (calDistance(x, y, centerX, centerY) > VALID_RADIUS) {
        if (x = 0.5f) {
        //满格
        wavePath.moveTo(centerX, centerY - VALID_RADIUS);
        wavePath.addCircle(centerX, centerY, VALID_RADIUS, Path.Direction.CW);
      } else {
        //空格
        return wavePath;
      }
    } else {
      //添加圆弧部分
      float startDegree = calDegreeByPosition(startPoint[0], startPoint[1]); //0~180
      float endDegree = calDegreeByPosition(endPoint[0], endPoint[1]); //180~360
      wavePath.arcTo(circleRectF, endDegree - 360, startDegree - (endDegree - 360));
    }
    return wavePath;
  }
  private float calDistance(float x1, float y1, float x2, float y2) {
    return (float) Math.sqrt((x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2));
  }
  //根据当前位置,计算出进度条已经转过的角度。
  private float calDegreeByPosition(float currentX, float currentY) {
    float a1 = (float) (Math.atan(1.0f * (centerX - currentX) / (currentY - centerY)) / Math.PI * 180);
    if (currentY  centerY && currentX > centerX) {
      a1 += 360;
    }
    return a1 + 90;
  }
  public void setMax(int max) {
    this.max = max;
    invalidate();
  }
  //直接设置进度值(同步)
  public void setProgressSync(float progress) {
    this.progress = progress;
    invalidate();
  }
  /**
   * 自动刷新页面,创造水波效果。组件销毁后该线城将自动停止。
   */
  private void autoRefresh() {
    new Thread(new Runnable() {
      @Override
      public void run() {
        while (!detached) {
          xOffset += (VALID_RADIUS >> 4);
          SystemClock.sleep(100);
          postInvalidate();
        }
      }
    }).start();
  }
  //标记View是否已经销毁
  private boolean detached = false;
  @Override
  protected void onDetachedFromWindow() {
    super.onDetachedFromWindow();
    detached = true;
  }
}

(二)使用方法:

在xml布局中引入上述组件,然后在activity或fragment中设置属性:

WaterWaveView bar = (WaterWaveView) getActivity().findViewById(R.id.water_wave_view);
    bar.setMax(500);
    bar.setProgressSync(361.8f);

更多关于Android相关内容感兴趣的读者可查看本站专题:《Android开发动画技巧汇总》、《Android编程之activity操作技巧总结》、《Android视图View技巧总结》、《Android布局layout技巧总结》、《Android开发入门与进阶教程》、《Android资源操作技巧汇总》及《Android控件用法总结》

希望本文所述对大家Android程序设计有所帮助。


推荐阅读
  • 本文介绍了ADB(Android Debug Bridge)的基本概念、安装方法、环境配置、连接真机步骤以及常用命令和高级技巧。ADB是一个强大的工具,适用于Android设备的开发和调试。 ... [详细]
  • 目录介绍01.CoordinatorLayout滑动抖动问题描述02.滑动抖动问题分析03.自定义AppBarLayout.Behavior说明04.CoordinatorLayo ... [详细]
  • 本文介绍了两种有效的方法来检查Android应用是否拥有特定权限,如媒体读写权限(media_rw)。通过这些方法,开发者和安全人员可以更好地了解应用的行为,确保其不会滥用权限。 ... [详细]
  • 本文详细介绍了ejabberd中的验证码服务、接收器以及服务器间通信的监督者和工作进程,包括它们的启动方式和主要功能。 ... [详细]
  • 本文详细介绍了如何从SVN中获取项目,并在本地环境中进行有效的构建和开发,包括具体的步骤和配置方法。 ... [详细]
  • 利用Java与Tesseract-OCR实现数字识别
    本文深入探讨了如何利用Java语言结合Tesseract-OCR技术来实现图像中的数字识别功能,旨在为开发者提供详细的指导和实践案例。 ... [详细]
  • 本文详细介绍了如何通过修改 Jenkins 的配置文件来解决因权限设置不当导致的登录后页面为空的问题,包括多种权限配置策略的选择与应用。 ... [详细]
  • 使用Solr从MySQL导入数据构建全量索引
    为了更好地掌握Solr的各项功能,本文档将在本地Windows环境中演示如何从MySQL数据库中导入数据至Solr,并构建全量索引。这将有助于开发者熟悉Solr的数据处理流程,尤其是在无法直接在生产服务器上进行实践的情况下。 ... [详细]
  • Struts2(六) 用Struts完成客户列表显示
    Struts完成客户列表显示所用的基础知识在之前的随笔中已经讲过。这篇是介绍如何使用Struts完成客户列表显示。下面是完成的代码执行逻辑图:抽取项目部分代码相信大家 ... [详细]
  • 在尝试将SpringBoot与MyBatis框架进行集成时,遇到了一个常见的问题:org.apache.ibatis.builder.BuilderException。此错误通常指示XML配置文件中存在语法或结构上的问题。本文将探讨具体原因及解决方案。 ... [详细]
  • 本文深入探讨了在Java编程语言中,如何使用`org.apache.polygene.api.association.AssociationDescriptor.qualifiedName()`方法,并提供了多个实际应用的代码示例。这些示例源自GitHub、StackOverflow和Maven等知名平台,旨在帮助开发者更好地理解和应用这一方法。 ... [详细]
  • 本实验利用xmlspy2013和firefox工具,探讨如何编写和验证基于DTD的XML文档。实验包括分析XML实例以创建相应的DTD文档,并通过编写有效的XML文档来验证DTD的有效性。 ... [详细]
  • 本文详细介绍了MyBatis中的延迟加载功能,包括其基本概念、实现方式以及如何在实际开发中应用。通过具体的代码示例,帮助读者更好地理解和掌握这一优化数据库查询性能的重要技术。 ... [详细]
  • 在日常开发中,经常需要通过Web服务进行数据交互。然而,在参数传递时会遇到一些限制,例如自定义实体类和Hashtable等复杂数据类型无法直接传递。本文将详细介绍如何在ASP.NET环境中有效地传递Hashtable数据。 ... [详细]
  • 本文详细介绍了DOM(文档对象模型)的基本概念、结构及操作方法。DOM作为一种API,允许开发者以编程方式访问HTML和XML文档的结构,实现页面内容的动态修改。 ... [详细]
author-avatar
手机用户2702938061
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有