Android侧滑粘稠效果的实现

*本篇文章已授权微信公众号 guolin_blog (郭霖)独家发布 *

之前在UI中国上看到一个侧滑效果,觉得还不错,于是就想实现一下。
UI效果是这样的:http://www.ui.cn/detail/198520.html

我自己实现的效果是这样的:


cehua.gif

效果并没有完全一致,UI效果里当两部分重合后有一种像水滴重合一样的效果,我并不知道用代码如何实现(如果大家知道请告诉我。。。),所以我这里只是简单的做了一个慢慢出现的效果。

好了,那我们就来看看这个效果是怎么实现的。

首先

这是一个自定义View。
效果分为两个部分,一个是背景的红色,一个是白色的叉叉。

背景的红色

当看到红色的效果的时候,我就想到了贝塞尔曲线。贝塞尔曲线是个神奇的东西,几乎所有神奇的曲线效果都可以用贝塞尔曲线来做。为了让这篇文章保持简洁,所以我就不介绍贝塞尔曲线了,如果有不了解的人,可以去这里看看。

到这里,我就假设你们都已经知道怎么画贝塞尔曲线啦。
我想你们也猜到了,红色效果其实就是用的三阶贝塞尔曲线。

红色效果.png

这个图画的就是红色效果的示意图(为了介绍所以将图画成了半圆,实际效果里,圆是扁扁的)。为了便于计算我将它设置为了一个正方形,图中的x2,x3,x5,x6都是中点。我们用Path.cubicTo()方法将x1到x7这七个点连起来之后就可以得一个半圆形了。
我们整个效果,其实就是控制这几个点而已。

背景红色的运动过程

在这里,我会直接给出代码来讲解。
为了便于计算,我们先将控件的宽高设为相等,让它变成一个正方形:

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec,widthMeasureSpec);
    }

然后就是绘制背景红色了,其实这个红色有上下两个,但是两个其实是一样的,所以我就只介绍上半部分。

        pathUp.reset();
        pathUp.moveTo(0,0);
        pathUp.cubicTo(x2,y26,0,y345,radius,y345);
        pathUp.cubicTo(radius*2,y345,x6,y26,radius*2,0);
        pathUp.lineTo(0,0);
        canvas.drawPath(pathUp,paint);

这里的过程就是从x1点开始连接一直连接到x7,最后在连到x1。
其中x2表示x2点的x坐标,y26表示x2点和x6点的y坐标,其余类同(取名字真的好累)。
radius表示圆的半径,也就是正方形边长的一半。
代码里,x3点和x5点的横坐标被我设置为和x1,x7一样,这样可以让圆扁一点。
我们需要变化的点,其实就是x2,x6,y26,y345这四个点而已。
y345,y26需要上下移动,来达到变小变大的效果。
至于x2,x6则是需要向外扩大,不然最后y345就算捅穿地表也不能把两边填满。
现在我们知道了需要变化的点的轨迹,那我们怎么控制它们呢?
我在View里写了一个方法:

public void controllAnimation(int progress,float max)

这个方法接收两个参数,第一个是当前值,第二个是最大的值。通过 当前值/最大值 我们就可以获取一个百分比的进度值。通过这个进度值我们就可以计算出当前点的位置:

    public void controllAnimation(int progress,float max){
        double fraction = progress/max;

        //这里根据进度改变,慢慢的变化
        y345 = (float) (radius*2*(fraction));

        if(y345<radius){
            x2 = 0;
            x6 = radius*2-x2;
        }else {
            x2 = (float) (-radius*2*((fraction-0.5)/fraction));
            x6 = radius*2-x2;
        }

        y26 = (float) (radius*fraction);
        invalidate();
    }

现在这个我用了一个seekBar,等到将其放进真的比如RecyclerView的侧滑删除时,只要将滑动距离作为progress参数传进去,滑动最大值作为max传进去就可以了。

白色叉叉

叉叉.png

叉叉就简单多了,图上的每个点都是中点。
叉叉也是分上半部和下半部,我们这里就只讲讲上半部分,下半部分一样的。

            //这里画那个叉叉
            //这是上半部分
            pathCha.reset();
            pathCha.moveTo((halfRadius*3)/2,radius);
            pathCha.lineTo(halfRadius,chaY1);
            pathCha.lineTo((halfRadius*3)/2,chaY2);
            pathCha.lineTo(radius,chaY1);
            pathCha.lineTo((radius*2)-((halfRadius*3)/2),chaY2);
            pathCha.lineTo(halfRadius*3,chaY1);
            pathCha.lineTo((radius*2)-((halfRadius*3)/2),radius);
            pathCha.lineTo((halfRadius*3)/2,radius);

这里radius表示半径,halfRadius表示半径的一半。
chaY1等于(radius*3)/4
chaY2等于 radius/2
chaY1表示x1,x3,x5点的y坐标,chaY2表示x2,x4点的y坐标。
我们需要改变的就是chaY1和chaY2的值,来达到一种叉叉慢慢出现的感觉。
代码如下:

        if(y345<radius){
            //一些代码
        }else {
            ...
            //慢慢的画出叉
            chaY1 = (float) (radius-(chaLength*2*((fraction-0.5)/fraction)));
            chaY2 = (float) (radius-(halfRadius*2*((fraction-0.5)/fraction)));
        }

我们的叉叉是在上下两个红色接触在一起的时候才绘制的,也就是进度为一半的时候。

这样就把这个View说完了,View的代码也不是很长,我就直接贴出来吧:

public class CeHuaView extends View {

    Paint paint,paintCha;
    Path pathUp,pathDown,pathCha,pathCha2;

    int width;

    float radius;
    float halfRadius;

    //这里取名字有点随便,因为不知道怎么取,可以看我博客里的图,应该就能知道意思了
    //这里y表示上半部分,yy表示下半部分,x同理
    float y345,yy345;
    float y26,yy26;
    float x2,x6;

    float chaLength;
    float chaY1,chaY2;

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

    public CeHuaView(Context context, @Nullable AttributeSet attrs) {
        this(context, attrs,0);
    }

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

    private void init() {
        pathUp = new Path();
        pathDown = new Path();
        pathCha = new Path();
        pathCha2 = new Path();

        paint = new Paint();
        paint.setColor(Color.parseColor("#ff5777"));
        paint.setAntiAlias(true);
        paint.setStyle(Paint.Style.FILL);

        paintCha = new Paint();
        paintCha.setColor(Color.parseColor("#ffffff"));
        paintCha.setAntiAlias(true);
        paintCha.setStyle(Paint.Style.FILL);
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec,widthMeasureSpec);
        width = MeasureSpec.getSize(widthMeasureSpec);

        radius = width/2;
        halfRadius = width/4;
        y345 = radius;
        yy345 = radius;

        y26 = halfRadius;
        yy26 = halfRadius*3;

        x2 = 0;
        x6 = radius*2;

        chaLength = radius/4;
        chaY1 = chaLength*3;
        chaY2 = chaLength*2;
    }

    public void controllAnimation(int progress,float max){
        double fraction = progress/max;

        //这里根据进度改变,慢慢的变化
        y345 = (float) (radius*2*(fraction));
        yy345 = radius*2-y345;

        if(y345<radius){
            x2 = 0;
            x6 = radius*2-x2;
        }else {
            x2 = (float) (-radius*2*((fraction-0.5)/fraction));
            x6 = radius*2-x2;

            //慢慢的画出叉
            chaY1 = (float) (radius-(chaLength*2*((fraction-0.5)/fraction)));
            chaY2 = (float) (radius-(halfRadius*2*((fraction-0.5)/fraction)));
        }

        y26 = (float) (radius*fraction);
        yy26 = (float) ((radius*2)-(radius*fraction));
        invalidate();
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        //这里画粘稠的效果
        //这是上半部分
        pathUp.reset();
        pathUp.moveTo(0,0);
        pathUp.cubicTo(x2,y26,0,y345,radius,y345);
        pathUp.cubicTo(radius*2,y345,x6,y26,radius*2,0);
        pathUp.lineTo(0,0);
        canvas.drawPath(pathUp,paint);

        //这里画粘稠的效果
        //这是下半部分
        pathDown.reset();
        pathDown.moveTo(0,radius*2);
        pathDown.cubicTo(x2,yy26,0,yy345,radius,yy345);
        pathDown.cubicTo(radius*2,yy345,x6,yy26,radius*2,radius*2);
        pathDown.lineTo(0,radius*2);
        canvas.drawPath(pathDown,paint);

        if(y345>radius){
            //这里画那个叉叉
            //这是上半部分
            pathCha.reset();
            pathCha.moveTo((halfRadius*3)/2,radius);
            pathCha.lineTo(halfRadius,chaY1);
            pathCha.lineTo((halfRadius*3)/2,chaY2);
            pathCha.lineTo(radius,chaY1);
            pathCha.lineTo((radius*2)-((halfRadius*3)/2),chaY2);
            pathCha.lineTo(halfRadius*3,chaY1);
            pathCha.lineTo((radius*2)-((halfRadius*3)/2),radius);
            pathCha.lineTo((halfRadius*3)/2,radius);

            //这是下半部分
            pathCha.lineTo((radius*2)-((halfRadius*3)/2),radius);
            pathCha.lineTo(halfRadius*3,radius*2-chaY1);
            pathCha.lineTo((radius*2)-((halfRadius*3)/2),radius*2-chaY2);
            pathCha.lineTo(radius,radius*2-chaY1);
            pathCha.lineTo((halfRadius*3)/2,radius*2-chaY2);
            pathCha.lineTo(halfRadius,radius*2-chaY1);
            canvas.drawPath(pathCha,paintCha);
        }
    }
}

在RecyclerView里应用

recycler.gif

因为这篇文章主要也不是讲RecyclerView的侧滑实现的,所以这方面的知识大家可以去这里看看。
在RecyclerView应用的代码也参考自这里。

好了,本篇文章结束了。
还有很多不完善的地方,也和UI中国的效果有出入,需要调整。
才疏学浅,如有错误,欢迎大家批评指正。

最后

本篇文章的代码:CeHuaView

最后的最后

感谢我可爱的女朋友。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,179评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,229评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,032评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,533评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,531评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,539评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,916评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,813评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,568评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,654评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,354评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,937评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,918评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,152评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,852评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,378评论 2 342

推荐阅读更多精彩内容