自定义view之可伸缩的圆弧与扇形

原文地址: 自定义view之可伸缩的圆弧与扇形

上一篇文章中讲解了如何自定义一个带有清除按钮的Edittext,这次讲解如何实现一个带有动画效果的圆弧及扇形图。先简单看一下效果:

简单看一下这两个图形:

  1. 弧形是根据输入的一个范围在0-360范围内的值,增加时会显示一个逐渐增加的动画,减少时也会有一个逐渐减少的动画,这个动画的插值器我设置的是先增速后减速。
  2. 扇形百分比动画是一个每次都会从开始的位置从新生成的动画,也可以做成类似于圆弧动画的效果。

自定义圆弧类

这个圆弧类我们直接继承自View,然后必然实现构造方法。

 private float value;//用户设置的值
    private Paint arcPaint;//要用到的画笔
    private RectF rectF;//绘制的范围
    private float oldValue;//过时的值

public ArcProgress(Context context) {
        super(context);
        init();
    }

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

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

在构造器中我们定义了一个init方法,这个方法主要是用来初始化一些东西,我只是初始化了画笔,注意一点,不能将rectF在这时候设置范围,因为我们是要根据用户设置的大小来填充rectF。

private void init() {
        arcPaint = new Paint();
        arcPaint.setAntiAlias(true);//抗锯齿
        arcPaint.setStyle(Paint.Style.STROKE);//只绘制圆弧边界
        arcPaint.setColor(Color.parseColor("#2c2c2c"));
        arcPaint.setStrokeWidth(50);//50px的圆弧宽度
    }

画笔大家应该用的都很熟练了,此处只说一点,如果不设置setStyle,默认是FILL,是填充效果。这个画笔是在ondraw方法中用来绘制圆弧的。

onsizechange方法

还记得上节内容中将的view的初始化顺序,

constructor->onmeasure->onDraw

现在引入一个新的方法,onSizeChanged(int w, int h, int oldw, int oldh)

这个方法是在控件的布局参数发生变化时调用的,oldvalue在第一次加载时是0。

这个方法是在onmeasure之后ondraw之前调用。调用顺序为:

constructor->onmeasure->onSizeChanged->onDraw

在这个方法中我们定义了rectF的范围

rectF = new RectF(50, 50, getMeasuredHeight() - 50, getMeasuredHeight() - 50);

我们设置了一个边界范围是50px,目的是看的更清楚,关于stroke的绘制是在宽度外还是内的问题此处不做详解。

ondraw方法

canvas.drawArc(rectF, 270, value, false, arcPaint);

方法只有一个最简单的绘制圆弧,注意第二个参数是绘制的起始角度,第三个参数是绘制的角度,为不是结束角度,结束角度是起始角度+绘制角度。第三个参数设置为false,这时候绘制的圆弧而不是扇形。

设置值的接口

public void setValue(final float v) {
        ValueAnimator animator = ValueAnimator.ofFloat(oldValue, v);
        oldValue = v;
        if (Math.abs(v - oldValue) > 180) {
            animator.setDuration(1000);
        } else {
            animator.setDuration(500);
        }
        animator.setInterpolator(new AccelerateDecelerateInterpolator());
        animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {
                value = (float) animation.getAnimatedValue();
                Log.d("change", "=" + animation.getAnimatedValue());
                invalidate();
            }
        });
        animator.start();
    }

这个接口是要暴漏出来的,设置为public。

此处我们定义了一个ValueAnimator,用oldv接收设置的v,此处为了用户体验,当变化角度小于180时设置持续时间为500ms,当变化角度大于180度时设置持续时间为500ms。

定义的插值器是一个先加速后减速的插值器。

重要的是为animator中添加UpdateListener,这个函数会持续返回给我们一个ValueAnimator对象,这个对象包含了我们在插值器中设定的不同的时间段对应的值,相当于一个时间函数。回调函数一直持续到动画结束。

此处我们通过ValueAnimator取出对应的数值,然后通过调用invalidate方法来刷新当前view,产生一个不断在动的效果。

这个invalid会刷新所有的可见view,但是必须工作在ui线程。

这样就完成了一个简单的view

button = (Button) view.findViewById(R.id.btn_change);
        circleProgress.setValue(300);
        progress.setValue(10, 10, 10);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                circleProgress.setValue(new Random().nextInt(360));
                progress.setValue(new Random().nextInt(20), new Random().nextInt(20),
                        new Random().nextInt(20));
            }
        });

这是调用代码,点击按钮生成随机的一个值。

同样的方式我们可以用来设置扇形。基本原理相似。
在ondraw方法中要设置为如下

canvas.drawArc(rectF, 0, 360 * percentOne, true, circlePaint1);
        canvas.drawArc(rectF, 360 * percentOne, 360 * percentTwo, true, circlePaint2);
        canvas.drawArc(rectF, 360 * (percentOne + percentTwo), 360 * percentThree, true, circlePaint3);

drawArc的第三个参数要用true,才能绘制扇形。

设置值的方法如下:

public void setValue(int value1, int value2, int value3) {
        int sum = value1 + value2 + value3;
        percentOne = (float) value1 / sum;
        newPercentOne = percentOne;
        percentTwo = (float) value2 / sum;
        newPercentTwo = percentTwo;
        percentThree = (float) value3 / sum;
        newPercentThree = percentThree;
        ValueAnimator animator1 = ValueAnimator.ofFloat(oldPercentOne, newPercentOne);
        ValueAnimator animator2 = ValueAnimator.ofFloat(oldPercentTwo, newPercentTwo);
        ValueAnimator animator3 = ValueAnimator.ofFloat(oldPercentThree, newPercentThree);
        animator1.setDuration(1000);
        animator2.setDuration(1000);
        animator3.setDuration(1000);
        animator1.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {
                percentOne = (float) animation.getAnimatedValue();

            }
        });
        animator2.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {
                percentTwo = (float) animation.getAnimatedValue();
            }
        });
        animator3.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {
                percentThree= (float) animation.getAnimatedValue();
                invalidate();
            }
        });
        AnimatorSet animatorSet=new AnimatorSet();
        animatorSet.playTogether(animator1,animator2,animator3);
        animatorSet.start();
        Log.d("rect :", "percentone=" + percentOne + ",percenttwo=" + percentTwo + ",percentthree=" + percentThree);

    }

我们用一个animatorset来包裹着三个动画同时发生,当然也可以按顺序发生,效果如下:


最后,关于动画的使用

安卓Property Animator动画详解(一)-官方文档

安卓Property Animator动画详解(二)-自定义属性

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

推荐阅读更多精彩内容