android高级UI之PathMeasure<三>--Path测量实战(笑脸loading效果实现、划船效果实现)

接着上一次https://www.cnblogs.com/webor2006/p/15605936.html的PathMeasure学习继续,这里将对PathMeasure的学习进行收尾。

笑脸loading效果实现:

效果:

 

具体实现: 

1、新建View:

2、画左、右边眼睛:

由于左右眼睛就是两个实心圆,绘制比较简单:

package com.cexo.pathmeasurestudy;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.util.AttributeSet;
import android.view.View;

/**
 * Loading效果四:笑脸
 */
public class LoadingView4 extends View {

    //constants
    /**
     * 左眼距离左边的距离(控件宽度*EYE_PERCENT_W),
     * 右眼距离右边的距离(控件宽度*EYE_PERCENT_W)
     */
    private static final float EYE_PERCENT_W = 0.35F;
    /**
     * 眼睛距离top的距离(控件的高度*EYE_PERCENT_H)
     */
    private static final float EYE_PERCENT_H = 0.38F;

    //variables
    private Paint paint;
    private float eyesH = EYE_PERCENT_H;
    private float radius;

    public LoadingView4(Context context) {
        this(context, null);
    }

    public LoadingView4(Context context, AttributeSet attrs) {
        this(context, attrs, -1);
    }

    public LoadingView4(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        init();
    }

    private void init() {
        paint = new Paint();
        paint.setColor(Color.GRAY);
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        drawFace(canvas);
    }

    private void drawFace(Canvas canvas) {
        paint.setStyle(Paint.Style.FILL);
        //画左边的眼睛
        canvas.drawCircle(getWidth() * EYE_PERCENT_W, getHeight() * eyesH - radius, radius, paint);
        //画右边的眼睛
        canvas.drawCircle(getWidth() * (1 - EYE_PERCENT_W), getHeight() * eyesH - radius, radius, paint);
    }

    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        super.onSizeChanged(w, h, oldw, oldh);
        radius = getWidth() / 7F / 2;
    }
}

运行:

3、画嘴巴:

对于嘴巴是一条曲线,很明显需要使用到贝塞尔线来进行绘制,首先将path需要移动到眼睛的下面:

其中涉及到几个变量:

然后利用贝塞尔曲线来绘制一条曲线【关于这块如不熟,可以参考https://www.cnblogs.com/webor2006/p/12901271.html】:

其中又涉及到变量:

此时运行看一下效果:

嗯,嘴巴有了,不过感觉线条太细了,加粗一点:

再运行:

4、画大脑的轮廓:

接下来则来画大脑的轮廓了,这块就是绘制一个椭圆的路径,所以先来定义path:

其中构建椭圆的路径用的是addRoundRect api,对于第一个参数比较好理解,是一个path的左上右下的位置,而第二个和第三个参数:

也就是用这俩参数来控制圆角的大小的,下面运行看一下:

5、眼睛跟嘴巴动效实现:

要实现让嘴和眼睛来进行上下动,其实就是需要控制这三个值:

而如何来控制呢?由于是无限循环进行变动,所以这里用一个动画进行控制是最合适的,如下:

这种动画的用法有啥用呢?下面看一下日志打印就知道了:

 

等于是从0~1之间进行数值的变化的,那么,就可以用这个百分比来控制上下滚动的幅度啦,如下:

 

此时再运行你会发现有个bug:

原因是需要加这么一句话:

再运行就如最初看到的效果一样啦,但是你会发现貌似木有用到PathMeasure这个东东对吧,目前这效果确实没用上,下面划船的就用到啦。 

划船效果实现:

效果:

 

其实它已经在github上进行开源了,https://github.com/webor2006/UI2018

这个开源项目里面有挺多UI效果的,想学学UI相关的可以瞅瞅它,上面则跟着源码走一遍流程,既使是抄一遍其收获也是有的~~

具体实现: 

1、新建View:

然后搭建主框架:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <com.cexo.pathmeasurestudy.BoatWaveView
        android:id="@+id/boat_wave_view"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1" />

    <TextView
        style="@style/textview_button"
        android:onClick="start"
        android:text="开始" />

    <TextView
        style="@style/textview_button"
        android:onClick="stop"
        android:text="停止" />

</LinearLayout>

其中按钮的样式:

    <style name="textview_button">
        <item name="android:layout_width">match_parent</item>
        <item name="android:layout_height">45dp</item>
        <item name="android:background">@drawable/selector_blue_round_5dp</item>
        <item name="android:gravity">center</item>
        <item name="android:textColor">@android:color/white</item>
        <item name="android:textSize">14sp</item>
        <item name="android:layout_margin">5dp</item>
    </style>

    <style name="textview_title">
        <item name="android:layout_width">match_parent</item>
        <item name="android:layout_height">30dp</item>
        <item name="android:textColor">#35a8ee</item>
        <item name="android:textSize">14sp</item>
        <item name="android:gravity">center</item>
    </style>

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">

    <item android:state_pressed="true">
        <shape>
            <corners android:radius="5dp" />
            <solid android:color="#35a8ee" />
        </shape>
    </item>

    <item android:state_focused="true">
        <shape>
            <corners android:radius="5dp" />
            <solid android:color="#35a8ee" />
        </shape>
    </item>

    <item android:state_selected="true">
        <shape>
            <corners android:radius="5dp" />
            <solid android:color="#35a8ee" />
        </shape>
    </item>

    <item>
        <shape>
            <corners android:radius="5dp" />
            <solid android:color="#1296db" />
        </shape>
    </item>

</selector>

2、绘制坐标辅助线:

为了能看到绘制的坐标位置,首先来绘制一下背景的辅助线,也就是效果如下:

1、封装base:

对于坐标辅助线,可能其它View也可以使用,所以将其抽到Base当中来:

2、初始化paint:

很明显坐标网络是由三个样式组成:

所以先来初始化这三个paint:

package com.cexo.pathmeasurestudy;

import android.content.Context;
import android.content.res.Resources;
import android.graphics.Color;
import android.graphics.Paint;
import android.util.AttributeSet;
import android.view.View;

import androidx.annotation.Nullable;

public abstract class BaseView extends View {

    // 坐标画笔
    private Paint coordinatePaint;
    // 网格画笔
    private Paint gridPaint;
    // 写字画笔
    private Paint textPaint;
    // 坐标颜色
    private int coordinateColor;
    private int gridColor;
    // 坐标线宽度
    private final float coordinateLineWidth = 2.5f;
    // 网格宽度
    private final float gridLineWidth = 1f;
    // 字体大小
    private float textSize;

    public BaseView(Context context) {
        super(context, null);
    }

    public BaseView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs, -1);
    }

    public BaseView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        initCoordinate(context);
        init(context);
    }

    private void initCoordinate(Context context) {
        coordinateColor = Color.BLACK;
        gridColor = Color.LTGRAY;

        textSize = spToPx(10);

        coordinatePaint = new Paint();
        coordinatePaint.setAntiAlias(true);
        coordinatePaint.setColor(coordinateColor);
        coordinatePaint.setStrokeWidth(coordinateLineWidth);

        gridPaint = new Paint();
        gridPaint.setAntiAlias(true);
        gridPaint.setColor(gridColor);
        gridPaint.setStrokeWidth(gridLineWidth);

        textPaint = new Paint();
        textPaint.setAntiAlias(true);
        textPaint.setColor(coordinateColor);
        textPaint.setTextAlign(Paint.Align.CENTER);
        textPaint.setTextSize(textSize);
    }

    /**
     * 转换 sp 至 px
     */
    protected int spToPx(float spValue) {
        final float fontScale = Resources.getSystem().getDisplayMetrics().scaledDensity;
        return (int) (spValue * fontScale + 0.5f);
    }

    protected abstract void init(Context context);
}

3、画坐标和网格

由于它是一个通用的行为,很明显这个绘制应该是放在base当中,但是又并不是每个View都需要它,所以这里将绘制的逻辑只提取到base当中,而具体要不要用由子类来决定,如下:

那如何绘制呢?

1、画网格:

其中网络就是由横竖线组成的,具体绘制也不难,先将画布移到屏幕中心:

先来横着画竖线:

运行发现报错了:

空指针的原因是:

此时的效果是:

看着像竖着画对吧,其实是横着画指定高度的竖线,这里为了明白这点,可以只画一次循环,你会看到如下:

 

明白了吧,把代码还原,接下来再来竖着画横线:

此时的效果为:

4、画 x,y 轴:

此时运行,你会发现有问题:

 

这个其实原因也很简单,因为我们在绘网络时已经将画布的坐标点移到屏幕中心了:

此时再绘制坐标线时,应该将画布的中心坐标给还原,所以处理如下:

5、画刻度:

接下来则需要在x,y轴上进行刻度的绘制,跟绘制网络思路差不多,如下:

比较简单,直接看效果:

 

接下来则需要上面进行文字标注:

至此,坐标系效果就已经绘制完了。

3、绘制划船效果:

1、实现小船图片滑动效果:

接下来实现最核心的划船效果了,这里进行一个拆解,先来将小船图片绘制出来,然后再让它可以开始荡漾,如下:

1、首先将小船给绘制到屏幕上:

2、绘制小船行走的路径:

接下来则需要让小船进行水波荡漾的效果,此时是不是就需要改用这个api来进行绘制了?

这块如还不太清楚的可以参考https://www.cnblogs.com/webor2006/p/15605936.html,接下来则需要定义小船行走的轨迹,先来定义path:

接下来则就来构建一条浪的path,此时肯定得用到二阶贝塞尔曲线了,而二阶贝塞尔曲线在Android中已经有专门的API可供调用了,回忆一下:

下面先来用死的值构建一条曲线:

运行效果:

其中看出有辅助坐标系的作用了么?

有了坐标系,直接通过肉眼就可以知道你想要实现的效果,而关于贝塞尔曲线还有另一个API:

那它跟quadTo()有啥区别呢?先来看一下官网的解释:

关于它,我一直木有能理解透,好在搜到这么一篇大佬的文章https://blog.csdn.net/harvic880925/article/details/50995587才搞明白:

先来用它实现上面quadTo同样的效果:

其中可以算出:

控制点x坐标=上一个终点x坐标+控制点x位移=getWidth()/2-100+50 =getWidth()/2-50;

控制点y坐标=上一个终点y坐标+控制点y位移=getHeight()/2-50;

是不是就是图中的这个位置了?

而了解rQuadTo()这个API的原因是接下来绘制小船轨迹时就会用它来进行曲线的构建了,当一个扩展巩固,先来横向绘制满几个波浪:

 

运行效果:

对于上面这段代码是不是有点晕,这里就不详细说明了,说一下其绘制的思路,先将整个波浪的长度定为屏幕的1/3,也就是:

然后每次循环绘制一个浪,这里加一些日志你就明白其绘制的思路了:

运行日志输出:

2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: x:-360;y:901;width:1080;height:1802
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: i----:-360;x:-360;y:901
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: control1:(-270,881)
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: end1:(-180,901)
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: control2:(-90,921)
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: end1:(0,901)
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: i----:0;x:0;y:901
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: control1:(90,881)
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: end1:(180,901)
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: control2:(270,921)
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: end1:(360,901)
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: i----:360;x:360;y:901
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: control1:(450,881)
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: end1:(540,901)
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: control2:(630,921)
2022-02-21 06:33:17.425 21064-21064/com.cexo.pathmeasurestudy E/cexo: end1:(720,901)
2022-02-21 06:33:17.426 21064-21064/com.cexo.pathmeasurestudy E/cexo: i----:720;x:720;y:901
2022-02-21 06:33:17.426 21064-21064/com.cexo.pathmeasurestudy E/cexo: control1:(810,881)
2022-02-21 06:33:17.426 21064-21064/com.cexo.pathmeasurestudy E/cexo: end1:(900,901)
2022-02-21 06:33:17.426 21064-21064/com.cexo.pathmeasurestudy E/cexo: control2:(990,921)
2022-02-21 06:33:17.426 21064-21064/com.cexo.pathmeasurestudy E/cexo: end1:(1080,901)
2022-02-21 06:33:17.426 21064-21064/com.cexo.pathmeasurestudy E/cexo: i----:1080;x:1080;y:901
2022-02-21 06:33:17.426 21064-21064/com.cexo.pathmeasurestudy E/cexo: control1:(1170,881)
2022-02-21 06:33:17.426 21064-21064/com.cexo.pathmeasurestudy E/cexo: end1:(1260,901)
2022-02-21 06:33:17.426 21064-21064/com.cexo.pathmeasurestudy E/cexo: control2:(1350,921)
2022-02-21 06:33:17.426 21064-21064/com.cexo.pathmeasurestudy E/cexo: end1:(1440,901)

可以看到,起始的坐标点都是已经超出屏幕了:

理解这个程序的核心一定是要知道此时画布的坐标点是在(0,0)这个位置,而非屏幕的中心哦,因为咱们在绘制完坐标系时已经将画布的平移给还原了:

也就是此时的绘制过程是从左到右进行的:

不过目前咱们这代码性能不太好,因为在onDraw()中会频繁的创建path:

所以这里将其放到onMeasure中只初始化一次:

@Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        if (!isInit) {
            isInit = true;
            width = getMeasuredWidth();
            height = getMeasuredHeight();
            waveLength = width / 3;

            //构建小船的路径
            boatPath = new Path();
            int x = -waveLength;
            int y = height / 2;
            Log.e("cexo", "x:" + x + ";y:" + y + ";width:" + width + ";height:" + height);
            boatPath.moveTo(x, y);
            int count = 0;
            for (int i = -waveLength; i < width * 1 + waveLength; i += waveLength) {
                Log.e("cexo", "i----:" + i + ";x:" + x + ";y:" + y);
                // rQuadTo 和 quadTo 区别在于
                // rQuadTo 是相对上一个点 而 quadTo是相对于画布
                int dx1 = waveLength / 4;
                int dy1 = -BOAT_WAVE_HEIGHT;
                int dx2 = waveLength / 2;
                int dy2 = 0;
                Log.e("cexo", "control1:(" + (x + dx1) + "," + (y + dy1) + ")");
                Log.e("cexo", "end1:(" + (x + dx2) + "," + (y + dy2) + ")");
                boatPath.rQuadTo(dx1, dy1, dx2, dy2);
                x = x + dx2;
                y = y + dy2;
                int dx11 = waveLength / 4;
                int dy11 = BOAT_WAVE_HEIGHT;
                int dx21 = waveLength / 2;
                int dy21 = 0;
                Log.e("cexo", "control2:(" + (x + dx11) + "," + (y + dy11) + ")");
                Log.e("cexo", "end1:(" + (x + dx21) + "," + (y + dy21) + ")");
                boatPath.rQuadTo(dx11, dy11, dx21, dy21);
                x = x + dx21;
                y = y + dy21;
            }
        }
    }

此时onDraw()中就只绘制path了:

好,有了path之后,接下来要想让小船图片随着这个path进行运行,此时PathMeasure就派上用场啦,需要对path进行测量如下:

接下来则就是绘制了,如下:

 

其中matrix.preTranslate()有啥作用呢? 可以参考https://blog.csdn.net/programchangesworld/article/details/49078387,接下来运行看一下效果:

其中你会发现小船跟波浪方向是一致的,只是船体并不是完全沿着波浪线来的,这也符合物理视觉。

3、让小船动起来:

接下来让小船进行动起来就比较简单了,我们只需要来控制这个值既可:

这里还是利用ValueAnimator来实现,如下:

 

 

运行:

2、绘制小船的浪:

现在船已经动起来了,不过貌似这波浪是空心的,没有大海的感觉,应该像这样才行:

下面则来实现这样的效果:

1、 将其变成实心的浪:

这里就需要先来构建一个新的path了,目前咱们绘制的path是小船的路径:

而这个path的构建其实跟小船的构建逻辑是一样的,也就是:

然后path的构建几乎一模一样:

@Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        if (!isInit) {
            isInit = true;
            width = getMeasuredWidth();
            height = getMeasuredHeight();
            waveLength = width / 3;

            //构建小船的路径
            boatPath = new Path();
            int x = -waveLength;
            int y = height / 2;
            Log.e("cexo", "x:" + x + ";y:" + y + ";width:" + width + ";height:" + height);
            boatPath.moveTo(x, y);
            for (int i = -waveLength; i < width * 1 + waveLength; i += waveLength) {
                Log.e("cexo", "i----:" + i + ";x:" + x + ";y:" + y);
                // rQuadTo 和 quadTo 区别在于
                // rQuadTo 是相对上一个点 而 quadTo是相对于画布
                int dx1 = waveLength / 4;
                int dy1 = -BOAT_WAVE_HEIGHT;
                int dx2 = waveLength / 2;
                int dy2 = 0;
                Log.e("cexo", "control1:(" + (x + dx1) + "," + (y + dy1) + ")");
                Log.e("cexo", "end1:(" + (x + dx2) + "," + (y + dy2) + ")");
                boatPath.rQuadTo(dx1, dy1, dx2, dy2);
                x = x + dx2;
                y = y + dy2;
                int dx11 = waveLength / 4;
                int dy11 = BOAT_WAVE_HEIGHT;
                int dx21 = waveLength / 2;
                int dy21 = 0;
                Log.e("cexo", "control2:(" + (x + dx11) + "," + (y + dy11) + ")");
                Log.e("cexo", "end1:(" + (x + dx21) + "," + (y + dy21) + ")");
                boatPath.rQuadTo(dx11, dy11, dx21, dy21);
                x = x + dx21;
                y = y + dy21;
            }

            //构建小船底下的浪的路径
            boatWavePath = new Path();
            x = -waveLength;
            y = height / 2;
            boatWavePath.moveTo(x, y);
            for (int i = -waveLength; i < width * 1 + waveLength; i += waveLength) {
                Log.e("cexo", "i----:" + i + ";x:" + x + ";y:" + y);
                // rQuadTo 和 quadTo 区别在于
                // rQuadTo 是相对上一个点 而 quadTo是相对于画布
                int dx1 = waveLength / 4;
                int dy1 = -BOAT_WAVE_HEIGHT;
                int dx2 = waveLength / 2;
                int dy2 = 0;
                Log.e("cexo", "control1:(" + (x + dx1) + "," + (y + dy1) + ")");
                Log.e("cexo", "end1:(" + (x + dx2) + "," + (y + dy2) + ")");
                boatWavePath.rQuadTo(dx1, dy1, dx2, dy2);
                x = x + dx2;
                y = y + dy2;
                int dx11 = waveLength / 4;
                int dy11 = BOAT_WAVE_HEIGHT;
                int dx21 = waveLength / 2;
                int dy21 = 0;
                Log.e("cexo", "control2:(" + (x + dx11) + "," + (y + dy11) + ")");
                Log.e("cexo", "end1:(" + (x + dx21) + "," + (y + dy21) + ")");
                boatWavePath.rQuadTo(dx11, dy11, dx21, dy21);
                x = x + dx21;
                y = y + dy21;
            }

            // 让 PathMeasure 与 Path 关联
            boatPathMeasure.setPath(boatPath, false);
        }
    }

然后绘制改一下path:

此时,运行,你会发现还是跟之前一样,不是实心的,其实是咱们的paint设置没改:

此时再运行看一下,还是不如预期:

而原因就得对path.close()有一定的了解了,这块的基础知识可以参考https://www.cnblogs.com/webor2006/p/15488224.html,这里直接给出代码了:

而对于path它有lineTo和rLineTo两个类似的api,此时就需要对这俩进行一个区别的了解,可以参考https://blog.csdn.net/wzping435/article/details/78583555,也就是构建这么一个区域可以达到闭环:

如果看不太懂,可以debug把值给打印,然后把坐标点算出来,就容易明白了,这里稍加过一下:

其实是定位在左下角的位置:

3、让浪动起来:

好,接下来浪扭动起来,咋扭动呢?这里需要用到画布平移的api了:

用一个具体代码例子来理解:

            canvas.save();//锁画布(为了保存之前的画布状态)
            canvas.translate(10, 10);//把当前画布的原点移到(10,10),后面的操作都以(10,10)作为参照点,默认原点为(0,0)
            drawScene(canvas);
            canvas.restore();//把当前画布返回(调整)到上一个save()状态之前 

所以,要想让波浪动起来,咱们只需要来让画布进行移动既可,具体如下:

然后开始移动:

此时运行你就会看到运动的效果了:

呃,新问题来了,露底了。。问题在原因在于对于小船这个浪不够“长”,所以解决起来也比较简单,将长度加大就成了,如下:

此时再运行,就木有这种露底的现象了,这里就不演示了。

4、代码抽取:

好,现在的问题就暴露了,小船的轨迹和小船的浪轨迹这俩的生成规则几乎一模一样,那。。是不是有必要封装一下?是的,所以在继续往下实现之前先来干下这事,这里细节就不过多解释了,比较简单:

 

3、绘制海浪:

最后,再加一个海浪,目前只有一个显得有点单调,有了上面的抽取之后,再加浪就非常简单了,如下:

其中浪的颜色值为:

<color name="color_wave_blue">#503bff</color>

运行,你会发现有bug:

海浪断层了,其实这块也很容易解决,需要这样处理:

 

到此,整个效果完成,完整代码如下:

package com.cexo.pathmeasurestudy;

import android.animation.ValueAnimator;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Matrix;
import android.graphics.Paint;
import android.graphics.Path;
import android.graphics.PathMeasure;
import android.util.AttributeSet;

import androidx.annotation.Nullable;
import androidx.core.content.ContextCompat;

/**
 * 划船效果
 */
public class BoatWaveView extends BaseView {

    // 小船浪花的高度
    private static final int BOAT_WAVE_HEIGHT = 20;
    // 浪花每次的偏移量
    private final static int WAVE_OFFSET = 5;
    // 波浪高度
    private static final int WAVE_HEIGHT = 35;

    // 小船的图片
    private Bitmap boatBitmap;
    // 用于变换小船的
    private Matrix matrix;
    // 小船的路径
    public Path boatPath;
    // 小船的浪路径
    public Path boatWavePath;
    // 海浪的路径
    public Path wavePath;
    public Paint wavePaint;
    // 小船的浪色值
    private int boatBlue;
    // 浪花的色值
    private int waveBlue;
    private int width;
    private int height;
    // 浪花的宽度
    private int waveLength;
    private boolean isInit = false;
    private PathMeasure boatPathMeasure;
    private ValueAnimator animator;
    // 当前小船在path路径上的百分比位置
    float currentPosition;

    // 小船的浪花偏移量
    private int boatWaveOffset = 0;
    // 浪花当前的偏移量
    private int curWaveOffset = 0;

    public BoatWaveView(Context context) {
        super(context);
    }

    public BoatWaveView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
    }

    public BoatWaveView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

    public void startAnim() {
        if (animator != null)
            animator.start();
    }

    public void stopAnim() {
        if (animator != null)
            animator.cancel();
    }

    @Override
    protected void init(Context context) {
        BitmapFactory.Options options = new BitmapFactory.Options();
        options.inSampleSize = 1;
        boatBitmap = BitmapFactory.decodeResource(getResources(), R.drawable.boat, options);

        matrix = new Matrix();

        boatBlue = ContextCompat.getColor(context, R.color.color_boat_blue);
        waveBlue = ContextCompat.getColor(context, R.color.color_wave_blue);

        wavePaint = new Paint();
        wavePaint.setAntiAlias(true);
        wavePaint.setColor(boatBlue);
//        wavePaint.setStrokeWidth(4);
//        wavePaint.setStyle(Paint.Style.STROKE);

        boatPath = new Path();
        boatWavePath = new Path();
        wavePath = new Path();

        boatPathMeasure = new PathMeasure();
        animator = ValueAnimator.ofFloat(0, 1f);
        animator.setDuration(4000);
        animator.setRepeatCount(ValueAnimator.INFINITE);
        animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {
                currentPosition = (float) animation.getAnimatedValue();
                boatWaveOffset = (boatWaveOffset + WAVE_OFFSET / 2) % width;//小船的浪走得慢一点
                curWaveOffset = (curWaveOffset + WAVE_OFFSET) % width;//浪走得快一点
                postInvalidate();
            }
        });
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        if (!isInit) {
            isInit = true;
            width = getMeasuredWidth();
            height = getMeasuredHeight();
            waveLength = width / 3;

            //构建小船的路径
            initPath(boatPath, waveLength, BOAT_WAVE_HEIGHT, false, 1);

            //构建小船底下的浪的路径
            initPath(boatWavePath, waveLength, BOAT_WAVE_HEIGHT, true, 2);

            // 初始化 浪的路径
            initPath(wavePath, waveLength, WAVE_HEIGHT, true, 2);

            // 让 PathMeasure 与 Path 关联
            boatPathMeasure.setPath(boatPath, false);
        }
    }

    /**
     * @param path       路径
     * @param length     浪花的宽度
     * @param waveHeight 浪花的高度
     * @param isClose    是否要闭合
     * @param lengthTime 浪花长的倍数
     */
    private void initPath(Path path, int length, int waveHeight, boolean isClose, float lengthTime) {
        // 初始化 小船的路径
        path.moveTo(-length, height / 2);
        for (int i = -length; i < width * lengthTime + length; i += length) {
            // rQuadTo 和 quadTo 区别在于
            // rQuadTo 是相对上一个点 而 quadTo是相对于画布
            path.rQuadTo(length / 4,
                    -waveHeight,
                    length / 2,
                    0);
            path.rQuadTo(length / 4,
                    waveHeight,
                    length / 2,
                    0);
        }

        if (isClose) {
            path.rLineTo(0, height / 2);
            path.rLineTo(-(width * 2 + 2 * length), 0);
            path.close();
        }
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        drawCoordinate(canvas);
        float length = boatPathMeasure.getLength();
        boatPathMeasure.getMatrix(length * currentPosition,
                matrix,
                PathMeasure.POSITION_MATRIX_FLAG | PathMeasure.TANGENT_MATRIX_FLAG);
        matrix.preTranslate(-boatBitmap.getWidth() / 2, -boatBitmap.getHeight() * 5 / 6);

        //根据轨迹来绘制小船
        canvas.drawBitmap(boatBitmap, matrix, null);

        // 画船的浪花
        canvas.save();
        canvas.translate(-boatWaveOffset, 0);
        wavePaint.setColor(boatBlue);
        canvas.drawPath(boatWavePath, wavePaint);
        canvas.restore();

        // 画浪花
        canvas.save();
        canvas.translate(-curWaveOffset, 0);
        wavePaint.setColor(waveBlue);
        canvas.drawPath(wavePath, wavePaint);
        canvas.restore();
    }
}

总结:

真是不容易,年后到现在就憋出了这么一篇,堕落啦【居然整个二月0篇】,另外有一个原因其实是由于年后公司组织架构调整了,到了一个全新的项目组,然后。。为了生存不得已需要耗用全部精力来熟悉新项目,所以学习计划就被搁置了,不过这也是给自己找借口,接下来还是得按照自己的学习计划前行,今年还是把去年落的计划一步一个脚印给补上,另外就是加强服务端java后端的学习【Java后台到全栈这门】,因为,年后被老大批了,说我们做app端的不思进取,把自己固守在自己的领域都不愿往后台搞一搞,好吧,算是逼自己换学习计划了,也挺好,接下来有时间就学它~~

posted on 2022-03-09 11:01  cexo  阅读(130)  评论(2编辑  收藏  举报

导航