android多个水波球,android球形水波百分比控件代码

本文主要介绍的是一个球形水波的百分比控件,市面上有各种形形色色的百分比控件,我一直觉得水波是最炫的,UI给了我这个机会,然而网上搜了一大堆,不是太复杂,代码太多(反正我是调不出效果来),就是有瑕疵的,所以只好自己写了,这里开源出来,方便大家。有什么问题或者建议大家留言指出。

先看效果,这里动态图不好截取,就贴张静态的

07cde25e6986f6eb77599bfe53f82a5b.png

对于水波百分比控件实现方法有如下几种

- 画好水波形状的bitmap,利用属性动画进行平移

- 利用曲线精确绘制目标水波

- 利用大范围曲线与容器做交集

第一种比较烦,网上有这种思路实现的,代码量比较庞大。bitmap移动时要注意的问题很多,一不小心就bug一堆了。第二种代码量小,但需要几何功底。很丢脸的说我算了好久。才算出公式(年代久远,都忘了),不过这种方法计算量大,绘制时遍历的点少。第三种方法,代码量极少,计算量几乎没有,遍历的点是第二种方法的两倍以上。考虑到遍历的消耗和计算的复杂度,选择第三种。

这里我们选择正弦曲线和圆做交集。

for (int i = left; i < length; i++) {

int x = i;

int y = (int) (Math.sin(Math.toRadians(x + mTranX) / 2) * mRadius / 4);

path2.lineTo(x, mH + y);

}

sin函数,x横坐标,y纵坐标,mTranX每次偏移量, 波形起伏mRadius / 4,

核心代码

利用圆的path与我们之前绘制的曲线做交集

Path pc = new Path();

pc.addCircle(mCentrePoint.x, mCentrePoint.y, mRadius, Path.Direction.CCW);

canvas.clipPath(pc, Region.Op.INTERSECT);

canvas.drawPath(path2, mWavePaint);

canvas.restore();

水位上升和水波起伏

while (isDraw) {

if (mWaterLevel > mNowHeight) {

mNowHeight = mNowHeight + mUpSpeed;

}

if (mStart) {

if (mTranX > mRadius) {

mTranX = 0;

}

mTranX = mTranX - mWaveSpeed;

}

drawUI();

}

这里由于动画效果比较细腻,更新UI界面比较平凡,所以我们采用surfaceView来实现(用view实现发现有卡顿,影响体验)

完整代码

就一个waveview类直接布局中引用

注释写的应该算比较清楚了。有什么疑问的可以留言

package com.aibaide.test;

import android.content.Context;

import android.graphics.Canvas;

import android.graphics.Color;

import android.graphics.Paint;

import android.graphics.Path;

import android.graphics.PixelFormat;

import android.graphics.Point;

import android.graphics.Region;

import android.util.AttributeSet;

import android.view.SurfaceHolder;

import android.view.SurfaceView;

/**

* gengqiquan

* 2016年6月2日16:16:48

* 水波显示百分比控件

*/

public class WaveView extends SurfaceView implements SurfaceHolder.Callback {

Point mCentrePoint;

int mNowHeight = 0;//当前水位

int mRadius = 0;

boolean mStart = false;//是否开始

float mTextSise = 60;//文字大小

Context mContext;

int mTranX = 0;//水波平移量

private Paint mCirclePaint;

private Paint mOutCirclePaint;

private Paint mWavePaint;

private Paint mTextPaint;

private SurfaceHolder holder;

private RenderThread renderThread;

private boolean isDraw = false;// 控制绘制的开关

private int mCircleColor = Color.parseColor("#ff6600");//背景内圆颜色

private int mOutCircleColor = Color.parseColor("#f5e6dc");//背景外圆颜色

private int mWaveColor = Color.parseColor("#ff944d");//水波颜色

private int mWaterLevel;// 水目标高度

private int flowNum = 60;//水目标占百分比这里是整数。

private int mWaveSpeed = 5;//水波起伏速度

private int mUpSpeed = 2;//水面上升速度

/**

* @param context

*/

public WaveView(Context context) {

super(context);

// TODO Auto-generated constructor stub

mContext = context;

init(mContext);

}

/**

* @param context

* @param attrs

*/

public WaveView(Context context, AttributeSet attrs) {

super(context, attrs);

// TODO Auto-generated constructor stub

mContext = context;

init(mContext);

}

/**

* @param context

* @param attrs

* @param defStyleAttr

*/

public WaveView(Context context, AttributeSet attrs, int defStyleAttr) {

super(context, attrs, defStyleAttr);

// TODO Auto-generated constructor stub

mContext = context;

init(mContext);

}

private void init(Context context) {

mContext = context;

setZOrderOnTop(true);

holder = this.getHolder();

holder.addCallback(this);

holder.setFormat(PixelFormat.TRANSLUCENT);

renderThread = new RenderThread();

mCirclePaint = new Paint();

mCirclePaint.setColor(mCircleColor);

mCirclePaint.setStyle(Paint.Style.FILL);

mCirclePaint.setAntiAlias(true);

mOutCirclePaint = new Paint();

mOutCirclePaint.setColor(mOutCircleColor);

mOutCirclePaint.setStyle(Paint.Style.FILL);

mOutCirclePaint.setAntiAlias(true);

mWavePaint = new Paint();

mWavePaint.setStrokeWidth(1.0F);

mWavePaint.setColor(mWaveColor);

mWavePaint.setStyle(Paint.Style.FILL);

mWavePaint.setAntiAlias(true);

mTextPaint = new Paint();

mTextPaint.setStrokeWidth(1.0F);

mTextPaint.setColor(Color.WHITE);

mTextPaint.setTextSize(mTextSise);

mTextPaint.setTextAlign(Paint.Align.CENTER);

mTextPaint.setStyle(Paint.Style.FILL);

mTextPaint.setAntiAlias(true);

}

@Override

public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {

mRadius = (int) (0.5 * width * 0.92);

mCentrePoint = new Point(width / 2, height / 2);

mWaterLevel = (int) (2 * mRadius * flowNum / 100f);//算出目标水位高度

}

@Override

public void surfaceCreated(SurfaceHolder holder) {

isDraw = true;

if (renderThread != null && !renderThread.isAlive())

renderThread.start();

}

@Override

public void surfaceDestroyed(SurfaceHolder holder) {

isDraw = false;

}

/**

* 绘制界面的线程

*

* @author Administrator

*/

private class RenderThread extends Thread {

@Override

public void run() {

// 不停绘制界面,这里是异步绘制,不采用外部通知开启绘制的方式,水波根据数据更新才会开始增长

while (isDraw) {

if (mWaterLevel > mNowHeight) {

mNowHeight = mNowHeight + mUpSpeed;

}

if (mStart) {

if (mTranX > mRadius) {

mTranX = 0;

}

mTranX = mTranX - mWaveSpeed;

}

drawUI();

}

super.run();

}

}

/**

* 界面绘制

*/

public void drawUI() {

Canvas canvas = holder.lockCanvas();

try {

drawCanvas(canvas);

} catch (Exception e) {

e.printStackTrace();

} finally {

if (canvas != null)

holder.unlockCanvasAndPost(canvas);

}

}

private void drawCanvas(Canvas canvas) {

//画背景圆圈

canvas.drawCircle(mCentrePoint.x, mCentrePoint.y, mRadius / 0.92f, mOutCirclePaint);

canvas.drawCircle(mCentrePoint.x, mCentrePoint.y, mRadius, mCirclePaint);

if (mStart) {

//计算正弦曲线的路径

int mH = mCentrePoint.y + mRadius - mNowHeight;

int left = - mRadius / 2;

int length = 4 * mRadius;

Path path2 = new Path();

path2.moveTo(left, mH);

for (int i = left; i < length; i++) {

int x = i;

int y = (int) (Math.sin(Math.toRadians(x + mTranX) / 2) * mRadius / 4);

path2.lineTo(x, mH + y);

}

path2.lineTo(length, mH);

path2.lineTo(length, mCentrePoint.y + mRadius);

path2.lineTo(0, mCentrePoint.y + mRadius);

path2.lineTo(0, mH);

canvas.save();

//这里与圆形取交集,除去正弦曲线多画的部分

Path pc = new Path();

pc.addCircle(mCentrePoint.x, mCentrePoint.y, mRadius, Path.Direction.CCW);

canvas.clipPath(pc, Region.Op.INTERSECT);

canvas.drawPath(path2, mWavePaint);

canvas.restore();

//绘制文字

canvas.drawText(flowNum + "%", mCentrePoint.x, mCentrePoint.y, mTextPaint);

}

}

public void setFlowNum(int num) {

flowNum = num;

mStart = true;

}

public void setTextSise(float s) {

mTextSise = s;

mTextPaint.setTextSize(s);

}

//设置水波起伏速度

public void setWaveSpeed(int speed) {

mWaveSpeed = speed;

}

//设置水面上升速度

public void setUpSpeed(int speed) {

mUpSpeed = speed;

}

public void setColor(int waveColor, int circleColor, int outcircleColor) {

mWaveColor = waveColor;

mCircleColor = circleColor;

mOutCircleColor = outcircleColor;

mWavePaint.setColor(mWaveColor);

mCirclePaint.setColor(mCircleColor);

mOutCirclePaint.setColor(mOutCircleColor);

}

//精确算法,每次正弦曲线从曲线与圆的交集处开始

// private int getX(double h) {

// int x = 0;

// int R = mRadius;

// if (h < R) {

// double t = 2 * R * h - h * h;

// x = (int) (R - Math.abs(Math.sqrt(t)));

// } else {

// double t = -2 * R * h + h * h;

// x = (int) (R - Math.abs(Math.sqrt(t)));

// }

// return x;

// }

}

最后奉上本文的源码:源码下载

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/502736.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

c++ 二维数组_【技术篇】C指针与二维数组深度辨析

一、源代码/*第01行*/ CLion 2019.3.4 x64中编写的源代码截图二、深度辨析CLion 2019.3.4 x64中编译的结果截图zippo[4][2]是一个四行二列的int型二维数组①不论一维数组还是二维数组&#xff0c;数组名就是指向数组首个元素的指针&#xff0c;也即数组名就是数组首个元素的…

shiro启动之后页面访问不了_java:shiro入门——4

【7】测试【7.1】启动点击apply然后点击OK【7.2】登录过滤访问http://localhost:8080/platform/home的时候&#xff0c;会被【7.3】角色过滤使用“admin”用户登录&#xff0c;密码&#xff1a;123根据SecurityServiceImpl我们可以知道使用admin账号登录成功之后&#xff1a;此…

oreo另一个意思_记一次有意思的统计(部分大宗商品价格指数相关性统计)

最近闲来无事&#xff0c;对部分大宗商品近十年的价格涨幅做了个统计&#xff0c;发现有些有意思的现象&#xff1a;大多资产价格走势如果放到一个足够长得时间维度里&#xff0c;那其实整体走势是比较一致的&#xff1b;有些资产价格走势高度相关&#xff0c;但是在某个时间段…

android左右耳机声音大小不一样,AirPods左右两边声音大小不同怎么办 单侧无声和两侧音量不同解决方法...

AirPods连接 iPhone 后如果出现了左右两边声音大小不一样&#xff0c;或者单侧无声的问题&#xff0c;可能是软件导致的暂时性故障&#xff0c;也有可能是硬件问题。当耳机音量出现异常时&#xff0c;可以通过以下几种方式尝试恢复。单侧无声和两侧音量不同解决方法&#xff1a…

laravel 任务队列_Laravel5.5之事件监听、任务调度、队列

流程&#xff1a;1.1 创建eventphp artisan make:event UserLoginLoginController.php/*** The user has been authenticated.** param IlluminateHttpRequest $request* param mixed $user* return mixed*/protected function authenticated(Request $request, $user){eve…

pytorch微调bert_小版BERT也能出奇迹:最火的预训练语言库探索小巧之路

选自Medium作者&#xff1a;Victor Sanh机器之心编译参与&#xff1a;魔王过去一段时间&#xff0c;大模型层出不穷。在大家纷纷感叹「大力出奇迹」的时候&#xff0c;作为调用预训练语言模型最流行的库&#xff0c;HuggingFace 尝试用更少的参数量、更少的训练资源实现同等的性…

if test 多条件_秒懂Python编程中的if __name__ == #39;main#39; 作用和原理

在大多数编排得好一点的脚本或者程序里面都有这段if __name__ main:1 这段代码的功能一个python的文件有两种使用的方法&#xff1a;第一是直接作为脚本执行&#xff0c;第二是import到其他的python脚本中被调用(模块重用)执行。因此if __name__ main: 的作用就是控制这两种情…

python背景颜色怎么随机_Python中的随机颜色

我同意TigerhawkT3&#xff08;1&#xff09;你教授对pick_color()的实现是垃圾。但我不认为random.choice()&#xff0c;或者你教授滥用random.shuffle()的方式是最好的选择。两者的问题是&#xff0c;在连续调用时可以获得相同的颜色&#xff0c;这是在正方形内绘制正方形时不…

python 解决手机拍的书籍图片发灰的问题

老师给发的作业经常是手机拍的&#xff0c;而不是扫描&#xff0c;背景发灰&#xff0c;如果二次打印就没有看了&#xff0c;象这样&#xff1a; 如果使用photoshop 处理&#xff0c;有些地方还是扣不干净&#xff0c;不如python 做的好&#xff0c;处理后如下&#xff1a; 具体…

2016年cypher资源_2021-2027年中国鱿鱼行业市场供需规模及未来前景分析报告

报告类型&#xff1a;产业研究报告格式&#xff1a;电子版、纸介版、电子纸介出品单位&#xff1a;智研咨询官网链接&#xff1a;中国产业信息网 - 产业前景投资趋势门户-智研旗下产业信息咨询平台​www.chyxx.com报告链接&#xff1a;2021-2027年中国鱿鱼行业市场供需规模及未…

地面控制点的定义与作用_什么是地面塌陷

地面塌陷2020年1月13日&#xff0c;青海西宁市城中区一公交车站附近地面突然塌陷&#xff0c;一辆搭载乘客的公交车掉入坑中&#xff0c;致使9人遇难。2019年12月12日&#xff0c;厦门吕厝路口地铁1号线和2号线外的配套物业开发项目施工现场发生约500平方米地面塌陷&#xff0c…

animate动画案例_animate动画案例——小小购物狂

如今各平台小动画层出不穷&#xff0c;大部分这种二维动画都是animate或者flash做的&#xff0c;例如下面这种效果animate既可以将各种内容做成动画。既可以设计适合游戏、电视节目和 Web 的交互式动画。让卡通和横幅广告栩栩如生。也可以用来创作动画涂鸦和头像。并向电子学习…

男孩子不上学了学计算机要学历吗,十三岁男孩不上学,能学什么手艺?

十三岁男孩不上学,能学什么手艺?十三岁时的孩子,有些学校要求我们先上过义务教育再去学习,有些学校是允许十三岁就直接接受教育的,有些学校是对十三岁还在上半学的学生进行补习一下的。那么,十三岁男孩不上学,可以学什么手艺?其实,有很多孩子对自己在学校学习时未能掌握的知识…

numpy 拼接_数据分析-numpy的拼接与交换

1.数组的拼接import numpy as npt1np.arange(24).reshape((4,6))t2np.arange(100,124).reshape((4,6))print(t1)print("*"*50)print(t2)print("*"*50)#竖直拼接t3np.vstack((t1,t2))print(t3)print("*"*50)#水平拼接t4np.hstack((t1,t2))print(t…

iptables 指定网卡_LINUX系统下的IPTABLES防火墙系统讲解(二)实战操作

iptables数据流方向iptables操作命令:#iptables --helpUsage: iptables -[AD] chain rule-specification [options]iptables -[RI] chain rulenum rule-specification [options]iptables -D chain rulenum [options]iptables -[LFZ] [chain] [options]iptables -[NX] chainipta…

java接口文档生成工具_接口文档生成

一、为什么要写接口文档&#xff1f;1.正规的团队合作或者是项目对接&#xff0c;接口文档是非常重要的&#xff0c;一般接口文档都是通过开发人员写的。一个工整的文档显得是非重要。2.项目开发过程中前后端工程师有一个统一的文件进行沟通交流开发&#xff0c;项目维护中或者…

联想计算机如何设置用户名和密码忘了,联想(Lenovo)路由器无线wifi密码忘记了怎么办啊?...

联想(Lenovo)路由器无线wifi密码忘记了怎么办&#xff1f;忘记wifi密码这个问题&#xff0c;很多用户都会遇到。因为手机、笔记本、平板电脑在首次连接wifi信号后&#xff0c;会自动保存该wifi信号密码&#xff0c;以后会自动进行连接&#xff0c;无需用户手动输入wifi密码&…

mysql binlog查看_MySQL--17 配置binlog-server 及中间件

配置binlog-server修改mha配置文件[rootmysql-db03 ~]# vim /etc/mha/app1.cnf[binlog1]no_master1hostname10.0.0.53master_binlog_dir/data/mysql/binlog/备份binlog#创建备份binlog目录[rootmysql-db03 ~]# mkdir -p /data/mysql/binlog/#进入该目录[rootmysql-db03 ~]# cd …

桥梁在线计算机监测系统,桥梁在线监测系统

原标题&#xff1a;桥梁在线监测系统监测背景我国是个桥梁大国&#xff0c;据最新数据统计&#xff0c;我国超过100万座公路桥梁(不含市政桥梁)。影响桥梁的因素居多&#xff0c;人为因素、车辆长期超载、材料自身退化等&#xff0c;缺乏及时到位的管理养护导致结构各部分在远没…

idea黑色好还是白色好_白色牛仔裤,好看又好搭

白色是属于夏天的颜色&#xff0c;也是最纯洁、最惹人注目的颜色。无论时尚如何轮回迭代&#xff0c;白色给我们的代名词永远是优雅、高贵、纯洁、干净、高贵、永恒等这些美好的词汇。白色是时光、流动、轻巧的颜色&#xff0c;它代表着东方的安静和中庸&#xff0c;也是留白含…