[Android开发]仿带有粘性的圆形刷新控件(1)

实现效果:


实现过程:
首先是图形的绘制实现:
采用两个圆,一个是在原地不动的起始圆S,一个是被拉伸出去的圆E,并用两条贝塞尔曲线接合,然后填充。
具体的图形就像下图,为了跟好的切合,选取了图一的方案,两条贝塞尔曲线的控制点选取O和P


为了绘制贝塞尔曲线,我们需要获取A,B,D,C,O,P这6个点的坐标。而我们已知圆S和E的圆心坐标和半径
可根据两圆心的距离和圆心坐标求出角R2R1X的cos和sin值。然后再加上两个圆的半径就可以求出A、B、C、D的坐标。O和P的坐标可以根据上面四个起点的坐标加上圆心距离和cos和sin就可以求出。具体计算代码如下:

private boolean calculateBezierCurve(Circle circleStart, Circle circleEnd){
        float startRadius = circleStart.radius;
        float endRadius = circleEnd.radius;
        float startX = circleStart.centerPoint.x;
        float startY = circleStart.centerPoint.y;
        float endX= circleEnd.centerPoint.x;
        float endY = circleEnd.centerPoint.y;

        float mCircleDistance = getDistanceBetweenTwoPoints(startX,startY,endX,endY);
        //两个圆重合就无需要绘制连接曲线
        if(mCircleDistance == 0){
            return false;
        }

        float cos = (startX - endX)/mCircleDistance;
        float sin = (startY - endY)/mCircleDistance;

        float ax = startX - startRadius * sin;
        float ay = startY + startRadius * cos;
        pStartA.x = ax;
        pStartA.y = ay;

        float bx = startX + startRadius * sin;
        float by = startY - startRadius * cos;
        pStartB.x = bx;
        pStartB.y = by;

        float cx = endX - endRadius * sin;
        float cy = endY + endRadius * cos;
        pEndA.x = cx;
        pEndA.y = cy;

        float dx = endX + endRadius * sin;
        float dy = endY - endRadius * cos;
        pEndB.x = dx;
        pEndB.y = dy;

        float ox = cx + mCircleDistance /2 * cos;
        float oy = cy + mCircleDistance /2 * sin;
        pControlO.x = ox;
        pControlO.y = oy;

        float px = dx + mCircleDistance /2 * cos;
        float py = dy + mCircleDistance /2 * sin;
        pControlP.x = px;
        pControlP.y = py;

        return true;
    }

需要计算的还有两个圆的随手指移动,圆心坐标和半径的变化:downPoint和movePoint分别是手指第一次按下的点和随后滑动手指所在的点

private void calculateCircleSize(){
        float mMoveDistance = getDistanceBetweenTwoPoints(downPoint.x,downPoint.y,movePoint.x,movePoint.y);
        //两圆重合无需再计算
        if(mMoveDistance <= 0) return;
        mScale = mMoveDistance/MaxMoveDistance;
        //开始圆按比例缩小
        circleStart.radius = DEFAULT_RADIUS * (1- mScale);
        //拉出圆按比例放大
        circleEnd.radius = DEFAULT_RADIUS * mScale;

        //开始圆的位置不变,拉出圆的位置根据滑动的距离移动
        circleEnd.centerPoint.x = circleStart.centerPoint.x + movePoint.x - downPoint.x;
        circleEnd.centerPoint.y = circleStart.centerPoint.y + movePoint.y - downPoint.y;
    }

经过适当的计算后,就是绘制图形:

 protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);

        //关闭硬件加速,否则部分path的绘制不生效
        setLayerType(View.LAYER_TYPE_SOFTWARE,null);

        //根据按下的和滑动的点两个点的距离计算,开始圆和拉出圆的中心坐标以及半径
        calculateCircleSize();
        canvas.drawCircle(circleStart.centerPoint.x, circleStart.centerPoint.y, circleStart.radius, mBezierPaint);
        canvas.drawCircle(circleEnd.centerPoint.x, circleEnd.centerPoint.y, circleEnd.radius, mBezierPaint);

        if(calculateBezierCurve(circleStart,circleEnd)){
            drawBezierCurves(canvas);//绘制两圆间的贝塞尔曲线
        }
        
        if(loadAnimator.isRunning()){
            drawLoading(canvas);//绘制旋转时,中心的圆弧
        }else {
            drawLoadingNormal(canvas);//绘制中心的圆弧和箭头
        }
    }

然后就是中心圆弧的绘制,因为在加载时和在拖拉时图形不同,就区分开来绘制

在拖拉时,中心是一段圆弧加上一个小箭头。绘制原理大概是在初始化的时候预先创建了一段接近360度的圆圈,因为直接360度的时候后续用PathMeasure测量长度可能不准

        mLoadPath = new Path();
        float loadCircleRadius = DEFAULT_RADIUS - DEFAULT_PADDING;
        RectF circle = new RectF(-loadCircleRadius, -loadCircleRadius, loadCircleRadius, loadCircleRadius);
        mLoadPath.addArc(circle, 0, 359.9f);

用PathMeasure获取之前创建圆圈Path的长度,选取圆圈上开始的长度start为0,就是圆圈开始的地方,再选取截取的长度stop为3/4的圆长。并且截取这段圆弧。这样中心的圆弧就有了。

同时,用PathMeasure获取截点stop的坐标以及正切角,用新建的path画一个小箭头,箭头的顶点在stop的坐标上。再根据正切角获取箭头需要旋转的角度。具体代码如下:

 private void drawLoadingNormal(Canvas canvas){
        //这里包含对画布坐标系的转换,快照一下,防止影响后续绘制
        canvas.save();
        //将画布中心移到开始圆的中心
        canvas.translate(circleStart.centerPoint.x,circleStart.centerPoint.y);
        //根据移动的距离比例,对画布缩小和旋转
        canvas.scale(1 - mScale,1 - mScale);
        canvas.rotate(360 * mScale);

        pathMeasure.setPath(mLoadPath,false);//将中心圆圈的path和pathMeasure关联
        float[] pos = new float[2];
        float[] tan = new float[2];
        float stop = pathMeasure.getLength() * 0.75f;
        float start = 0;
        pathMeasure.getPosTan(stop,pos,tan);//获取截取圆弧的结束点的坐标和方向趋势
        //根据tan获取旋转的角度,用于旋转后面绘制的箭头
        float degrees =(float)(Math.atan2(tan[1],tan[0])*180/Math.PI);

        Matrix matrix = new Matrix();
        Path triangle = new Path();
        //绘制箭头,此时的箭头的顶点坐标还在原点
        triangle.moveTo(pos[0] - 5, pos[1] + 5);
        triangle.lineTo(pos[0],pos[1]);
        triangle.lineTo(pos[0] + 5, pos[1] + 5);
        triangle.close();
        //将箭头移动到圆弧结束点的位置并旋转
        matrix.setRotate(degrees+90, pos[0],pos[1]);

        Path showPath = new Path();
        //前面的箭头添加将要绘制的路径里面
        showPath.addPath(triangle,matrix);
        //截取圆圈从起始点到结束的圆弧并添加到要绘制的path中,true代表不将截取的圆弧的起点移动到之前path的最后一个点上
        pathMeasure.getSegment(start,stop,showPath,true);

        canvas.drawPath(showPath, mLoadPaint);
        canvas.restore();
    }

绘制加载时候的圆弧同理,只是少画了箭头,同时start和stop的位置根据animator给与的value来选取,这里的value的值由0慢慢变化到1

private void drawLoading(Canvas canvas){
        //基本和绘制一般状态的时候一样,除了截取的起点和终点需要动态的计算
        canvas.save();
        canvas.translate(circleStart.centerPoint.x, circleStart.centerPoint.y);
        canvas.scale(1 - mScale,1 - mScale);
        pathMeasure.setPath(mLoadPath,false);
        Path newPath = new Path();
        float stop = pathMeasure.getLength() * mLoadAnimatorValue;
        float start = (float)(stop - (0.5 - Math.abs(mLoadAnimatorValue - 0.5)) * 200f);
        pathMeasure.getSegment(start,stop,newPath,true);
        canvas.drawPath(newPath, mLoadPaint);
        canvas.restore();
    }

手指状态获取的代码如下:

 @Override
    public boolean onTouchEvent(MotionEvent event) {
        float x = event.getX();
        float y = event.getY();
        //动画执行时,无需改变两点的坐标
        switch (event.getAction()){
            case MotionEvent.ACTION_DOWN:
                if(!stickyAnimator.isRunning() && !loadAnimator.isRunning()){
                    downPoint.x = x;
                    downPoint.y = y;
                    movePoint.set(downPoint);
                    resetLoadAnimator();
                }
                break;
            case MotionEvent.ACTION_MOVE:
                if(!stickyAnimator.isRunning() && !loadAnimator.isRunning() && !loading){
                    movePoint.x = x;
                    movePoint.y = y;
                    float distanceMove = getDistanceBetweenTwoPoints(downPoint.x,downPoint.y,movePoint.x,movePoint.y);
                    //滑动距离在动作范围内,则开始执行回滚动画和loading动画
                    if(inLoadArea(distanceMove)){
                        loading = true;
                        executeAnimator(distanceMove);
                    }
                    invalidate();
                }
                break;
            case MotionEvent.ACTION_UP:
            case MotionEvent.ACTION_CANCEL:
                if(!stickyAnimator.isRunning() && !loadAnimator.isRunning() && !loading){
                    movePoint.x = x;
                    movePoint.y = y;
                    float distanceUp = getDistanceBetweenTwoPoints(downPoint.x,downPoint.y,movePoint.x,movePoint.y);
                    //滑动距离在动作范围内,则开始执行回滚动画和loading动画,否则只开始回滚动画
                    if(inLoadArea(distanceUp)){
                       loading = true;
                    }
                    executeAnimator(distanceUp);
                }
                break;
        }
        return true;
    }

动画的内容在下一篇讲
http://www.jianshu.com/p/5d35e37ef02a

项目地址:https://github.com/clam314/StickyCircleView

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

推荐阅读更多精彩内容

  • 实现效果: 上一篇写了关于图形相关绘制:http://www.jianshu.com/p/25aa4789b3fd...
    clam314阅读 404评论 0 0
  • 《机械制图》10%(50+30=80) 单项选择题 Q-B1-E-001 L 基本幅面不能满足需要而采用加长幅面时...
    开源时代阅读 3,709评论 1 1
  • 转载:http://www.jianshu.com/p/32fcadd12108 每个UIView有一个伙伴称为l...
    F麦子阅读 6,121评论 0 13
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 11,947评论 4 60
  • 我真的曾为你流泪 你那么小 那么小 我看见你的疼痛 像生冷的盐水流进我的眼睛里 在这个世界上 你是那样孤独的孩子 ...
    宓牛阅读 152评论 0 0