三角函数之美-水波纹载入LoadingView
一、前言
学习是要总结的。近期几天学习了画图相关的,可是使用的机会较少,如今又快要遗忘了,这次看了水波纹的绘制。认为十分有意思,还是 把实现的方法记录下来。技术无他,为手熟尔。还是要多练习,空淡误国,实干兴邦,让我们看看今天的三角函数之美吧。
二、概述
肯定大家对中学学习的三角函数都不陌生吧。只是学习的sin、cos是超越函数一类函数。是初等函数的一种,借用维基百科的一张图:
一个完整的正弦函数应该是这种:>y=Asin(ωx+φ)+h,A决定峰值,ω决定周期。φ表示初相位,h表示y轴的位置。在Android我们使用通常是Math.sin(30*Math.PI/180)这种形式,我原来搞不懂这样写的含义。后来才明确Math.PI就是π。π即180度。所以上述式子相当于sin(π/6)=1/2,在中学我们通常是sin(30°)=1/2,把sin(x)中的x赋值为角度,而在计算机语言x通常是弧度。
这个流量界面就是通过sin函数曲线不断改变x位置绘制而来,实现的方式肯定也还有其他的方法,通过绘制贝塞尔曲线也有实现的案例。我们将通过样例逐渐对水波纹loadingview绘制,好了不多说了,手把手带你一步一步实现美丽炫酷的loading水波纹效果。
三、波浪控件的实现
首先我们先实现波浪控件的绘制:
先计算坐标点:
//这里我们以view的总宽度为周期,y = a * sin(2π) + b
for (int i = 0; i < mTotalWidth; i++){
mPointY[i] = (float) (20 * (Math.sin( 2 * Math.PI * i / w)));
}
绘制曲线
for (int i = 0;i < mTotalWidth; i++){
canvas.drawLine(i,mTotalHeight - mDaymicPointY[i] - 300,i,mTotalHeight,mPaint);
}
这里减去300仅仅是为了控制在Y轴上的位置,我们能够动态设置一个数值。在view上下移动。从而达到流量界面。我们先一步一步来。
控制移动
//改变两条波纹的移动点
mXoffset += X_SPEED;
//假设已经移动到末尾处,则到头又一次移动
if(mXoffset > mTotalWidth){
mXoffset = 0;
}
// 超出屏幕的挪到前面,mXoffset表示第一条水波纹要移动的距离
int yIntelrval = mPointY.length - mXoffset;
//使用System.arraycopy方式又一次填充第一条波纹的数据
System.arraycopy(mPointY, 0, mDaymicPointY, mXoffset, yIntelrval);
System.arraycopy(mPointY, yIntelrval, mDaymicPointY, 0, mXoffset);
如今你会发现一个简单的水波纹效果出来了,例如以下图所看到的:
你会发现这个水波纹跟我们的效果还是有些差距。别急,我们还有这个类PorterDuffXfermode,能够实现与所绘制的图像按一定规则进行混合,形成新的像素值,那么我们就能够先画一个水波纹效果,然后在画一个圆,使用以下这幅神图进行混合,你会发现奇迹的,PorterDuffXfermode的相关使用能够參考这边博客:
Android中Canvas画图之PorterDuffXfermode使用及工作原理具体解释
使用哪种规则呢,自己找找呗,都是进行某种规则进行混合的,我找到了,使用的是SrcIn,就能够进行混合实现效果了
如今就是这种效果,我们在加一个seekbar。拖动显示效果。基本上就能够出现这个波浪loading效果了
int layerId = canvas.saveLayer(0, 0, canvasWidth, canvasHeight, null, Canvas.ALL_SAVE_FLAG);
canvas.drawCircle(mTotalWidth / 2, mTotalHeight / 2, mTotalWidth / 2, mCriclePaint);
//设置颜色混合模式
mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_IN));
//高减去宽除以2使水波纹底部在圆底部。动态改变percent值,在Y轴上变化
for (int i = 0; i < mTotalWidth; i++) {
canvas.drawLine(i, mTotalHeight - mDaymicPointY[i] - (mTotalHeight - mTotalWidth) / 2 - percent * mTotalWidth / 100, i, mTotalHeight - (mTotalHeight - mTotalWidth) / 2, mPaint);
}
//最后将画笔去除Xfermode
mPaint.setXfermode(null);
canvas.restoreToCount(layerId);
效果例如以下图所看到的
咦,说好的明明是水波纹绿颜色呢,怎么变得浅了很多,硬件加速会对效果有影响的, Android从3.0(API Level 11)開始。在绘制View的时候支持硬件加速,充分利用GPU的特性,使得绘制更加平滑。可是会多消耗一些内存。
提供了4个方面打开或者关闭硬件加速,这里也提一下:
1.Application级别:
<applicationandroid:hardwareAccelerated="true" ...>
2.Activity级别:
<activity android:hardwareAccelerated="false" ...>
3.Window级别:
getWindow().setFlags(
WindowManager.LayoutParams.FLAG_HARDWARE_ACCELERATED,
WindowManager.LayoutParams.FLAG_HARDWARE_ACCELERATED);
注:Android还不支持在Window级别关闭硬件加速。
4.View级别:
myView.setLayerType(View.LAYER_TYPE_HARDWARE, null);
注:Android还不支持在View级别开启硬件加速。
我们直接在application以下全局关闭硬件加速就可以,以下的效果就是这种:
我贴下所有实现过程的代码:
public class WaveView extends View {
private Paint mPaint, mCriclePaint,mTextPaint;
// 倾斜或旋转、高速变化,当在屏幕上画一条直线时, 横竖不会出现锯齿,
// 可是当斜着画时, 就会出现锯齿的效果,所以须要设置抗锯齿
private DrawFilter mDrawFilter;
private int mTotalHeight, mTotalWidth;
private int mXoffset = 0;
private float[] mPointY;
private float[] mDaymicPointY;
//波浪线移动速度
private static final int X_SPEED = 20;
private int percent;
public void setPercent(int percent) {
this.percent = percent;
}
public WaveView(Context context) {
super(context);
init();
}
public WaveView(Context context, AttributeSet attrs) {
super(context, attrs);
init();
}
public WaveView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init();
}
private void init() {
//图片线条(通用)的抗锯齿须要另外设置
mDrawFilter = new PaintFlagsDrawFilter(0, Paint.ANTI_ALIAS_FLAG | Paint.FILTER_BITMAP_FLAG);
//实例化一个画笔
mPaint = new Paint();
//去除画笔锯齿
mPaint.setAntiAlias(true);
//设置画笔风格为实线
mPaint.setStyle(Paint.Style.FILL);
//设置画笔颜色
mPaint.setColor(Color.GREEN);
//实例化圆的画笔
mCriclePaint = new Paint(mPaint);
mCriclePaint.setColor(Color.parseColor("#88dddddd"));
mCriclePaint.setAlpha(255);
//实例化文字画笔
mTextPaint = new Paint();
mTextPaint.setAntiAlias(true);
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
//去除锯齿
canvas.setDrawFilter(mDrawFilter);
runWave();
int canvasWidth = canvas.getWidth();
int canvasHeight = canvas.getHeight();
int layerId = canvas.saveLayer(0, 0, canvasWidth, canvasHeight, null, Canvas.ALL_SAVE_FLAG);
canvas.drawCircle(mTotalWidth / 2, mTotalHeight / 2, mTotalWidth / 2, mCriclePaint);
//设置颜色混合模式
mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_IN));
//高减去宽除以2使水波纹底部在圆底部,动态改变percent值,在Y轴上变化
for (int i = 0; i < mTotalWidth; i++) {
canvas.drawLine(i, mTotalHeight - mDaymicPointY[i] - (mTotalHeight - mTotalWidth) / 2 - percent * mTotalWidth / 100, i, mTotalHeight - (mTotalHeight - mTotalWidth) / 2, mPaint);
}
//最后将画笔去除Xfermode
mPaint.setXfermode(null);
canvas.restoreToCount(layerId);
//改变两条波纹的移动点
mXoffset += X_SPEED;
//假设已经移动到末尾处。则到头又一次移动
if (mXoffset > mTotalWidth) {
mXoffset = 0;
}
String text = percent + "%";
mTextPaint.setTextSize(80);
float textLength = mTextPaint.measureText(text);
canvas.drawText(text,(mTotalWidth - textLength) / 2,mTotalHeight / 2 - 20,mTextPaint);
//引起view重绘
postInvalidateDelayed(300);
}
@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
super.onSizeChanged(w, h, oldw, oldh);
mTotalHeight = h;
mTotalWidth = w;
//数组的长度为view的宽度
mPointY = new float[w];
mDaymicPointY = new float[w];
//这里我们以view的总宽度为周期,y = a * sin(2π) + b
for (int i = 0; i < mTotalWidth; i++) {
mPointY[i] = (float) (20 * (Math.sin(2 * Math.PI * i / w)));
}
}
private void runWave() {
// 超出屏幕的挪到前面。mXoffset表示第一条水波纹要移动的距离
int yIntelrval = mPointY.length - mXoffset;
//使用System.arraycopy方式又一次填充第一条波纹的数据
System.arraycopy(mPointY, 0, mDaymicPointY, mXoffset, yIntelrval);
System.arraycopy(mPointY, yIntelrval, mDaymicPointY, 0, mXoffset);
}
}
上面就是实现的所有代码了,都有凝视。相信大家能看懂,事实上实现这些效果还是非常easy的,可是我们还有很多扩展的地方,做成一个网络载入的等待动画,或者做成一个下载的的进度,等待大家自己去实现。
源代码:下载地址猛戳
三角函数之美-水波纹载入LoadingView的更多相关文章
- 手把手教你画一个 逼格满满圆形水波纹loadingview Android
才没有完结呢o( ̄︶ ̄)n .大家好,这里是番外篇. 拜读了爱哥的博客,又学到不少东西.爱哥曾经说过: 要站在巨人的丁丁上. 那么今天,我们就站在爱哥的丁丁上来学习制作一款自定义view(开个玩笑,爱 ...
- OC语言编写:为视图添加丝滑的水波纹
先看一下最终效果图: 首先我们可以把如此丝滑的水波纹拆分一下下: 一条规律的曲线. 曲线匀速向右移动. 曲线下方的位置用颜色填充. 于是先来一条曲线吧. 对于需要产生波动如此规律的曲线,我们首先想到的 ...
- android 5.0 默认水波纹背景属性,可设置不论什么View
actionBarItemBackground 5.0以上超出边界圆形水波纹 selectableItemBackground 5.0以上边界内圆形水波纹 这两个属性在5.0下面是默认的灰色效果 ...
- jquery ripples水波纹效果( 涟漪效果)
这个效果是我从bootstrap-material-design上面分离下来的,bootstrap-material-design的一些组件样式我不太不喜欢,但是非常喜欢这个水波纹效果,所以就有了这篇 ...
- 如何使用 HTML5 Canvas 制作水波纹效果
今天,我们继续分享 JavaScript 实现的效果例子,这篇文章会介绍使用 JavaScript 实现水波纹效果.水波效果以图片为背景,点击图片任意位置都会触发.有时候,我们使用普通的 Javasc ...
- 兼容Android的水波纹效果
Android的水波纹效果只有高版本才有,我们希望自己的应用在低版本用低版本的阴影,高版本用水波纹,这怎么做呢?其实,只要分drawable和drawablev21两个文件夹就好了. 普通情况下的se ...
- android自定义控件(4)-自定义水波纹效果
一.实现单击出现水波纹单圈效果: 照例来说,还是一个自定义控件,观察这个效果,发现应该需要重写onTouchEvent和onDraw方法,通过在onTouchEvent中获取触摸的坐标,然后以这个坐标 ...
- android 5.0 水波纹 实现
1. 定义一个普通圆角背景的xml; rounded_corners.xml <?xml version="1.0" encoding="utf-8"?& ...
- VC_MFC水波纹控件,开源
代码和效果图: https://github.com/wjx0912/MfcWaterEffect 集成以下5个文件即可: watereffect\DIB.hwatereffect\DIB.cppwa ...
随机推荐
- RabbitMQ学习总结(4)——分发任务在多个工作者之间实例教程
一.Work Queues(using the Java Client) 走起 在第上一个教程中我们写程序从一个命名队列发送和接收消息.在这一次我们将创建一个工作队列,将用于分发耗时的任务在多个工 ...
- (转)Epoll模型详解
1. 内核中提高I/O性能的新方法epoll epoll是什么?按照man手册的说法:是为处理大批量句柄而作了改进的poll.要使用epoll只需要这三个系统调 用:epoll_create(2), ...
- ASP.NET-ActionFilter过滤器用法实例
ActionFilter可以对每一个传过来的action请求进行过滤,非常有用,但是如果在这里判断过多,那么网站的性能和速度会不会变慢,这个问题值得思考,现在先放在这里. public class A ...
- JStorm中的并行( parallelismction )介绍
JStorm中的并行( parallelismction )介绍 JStrom中.一个计算任务通过多台机器使得计算分解为多个独立并行执行在集群内执行的任务(tasks).从而得到水平扩展. JStor ...
- C/S和B/S交互
近期一直在做C/S的项目,每天都超忙,抽个时间写篇博客,之前一直做C/S项目就是各种窗口.各种控件,拖来拖去,然后点进去写方法,做BS的时候呢,由于一直使用的是mvc,所以就是常常手写代码.或者拖引用 ...
- 移动端fixed后 横竖屏切换时上部或下部出现空隙问题
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- Servlet简单计算器 2.0
jsp 输入界面: <%@ page language="java" contentType="text/html; charset=UTF-8" pag ...
- Java Servlet 3.0 新特性
Servlet 3.0 新特性概述 Servlet 3.0 作为 Java EE 6 规范体系中一员,随着 Java EE 6 规范一起发布.该版本在前一版本(Servlet 2.5)的基础上提供了若 ...
- 51nod 1102 面积最大的矩形 && 新疆大学OJ 1387: B.HUAWEI's billboard 【单调栈】+【拼凑段】(o(n) 或 o(nlog(n))
题面1:  题面2:  两道题除了数据范围不同,没有任何差异,两道题都可以o(n)(单调栈),o(nlog(n))(我自己的做法)解决. 解题思路1:(单调栈) 对于每个点找到右边第一个比它小的位 ...
- NodeJS学习笔记 (9)网络服务-https(ok)
模块概览 这个模块的重要性,基本不用强调了.在网络安全问题日益严峻的今天,网站采用HTTPS是个必然的趋势. 在nodejs中,提供了 https 这个模块来完成 HTTPS 相关功能.从官方文档来看 ...