小白上手-自定义风扇型Loading控件

最近手头有些时间,就逛逛知乎看看有什么好玩的。结果发现一个帖子说是冷门网站的,就进去看了看,结果发现一个有趣的网站: codewars

个人感觉这是个挺有趣的网站,它里面有很多种语言,用户可以选择自己想要挑战的语言,纠错!然后里面也有互动社区,同一道题,往往会有多个解法,可以在社区里看到别人成熟简洁的解题方式和思路,这也是对个人编程能力的一种提升。

然后就说说为什么会想要做这么一个自定义View吧,在玩codewars的过程中,我发现它左上角有个有趣的图标,就是这个


codewars.png

当网页有刷新的时候,这个图标就像风扇一样转呀转,嘿嘿,还挺有趣,于是,自己也想着做一个出来。

然后我仿着做着一个出来,效果是这样的。


GIF.gif

啊~说到这个GIF我要吐槽一下了,真真啊,第一次录GIF,各种不顺利,不是效果出不来就不理想,最后调呀调,录啊录,最后才选择这一个上传的。好了,废话不多说,先说说我的实现思路吧。

实现思路

  • 先画外面的圆角矩形
  • 再画中间那个小圆圈
  • 最后画扇叶
  • 让扇叶转起来

view的测量部分就不介绍了,都是常规代码。我就说说里面扇叶的部分和控制扇叶转动以及转动速度这部分吧。

canvas.rotate(-rotateDegree, mWidth / 2, mHeight / 2);//控制风扇转速
for (int i = 0; i < fanCount; i++) { 
  canvas.drawArc(rectFan, -90, -180, true, bgPaint);    
  canvas.rotate(90, mWidth / 2, mHeight / 2);}
if (running)
  rotateHandler.sendEmptyMessageDelayed(1, 10);//控制风扇转动

一开始先旋转一下画布,是为了使扇叶有个旋转偏移量,已制造出旋转的效果,同时还能控制转速的效果。
for循环内部就是画出四片扇叶。
然后handler呢,主要是用来持续刷新view,已实现转动的效果。

FanLoading.java
package com.kenny.fcmpushtest.UI.customView;

    import android.content.Context;
    import android.content.res.TypedArray;
    import android.graphics.Canvas;
    import android.graphics.Color;
    import android.graphics.Paint;
    import android.graphics.RectF;
    import android.os.Handler;
    import android.os.Message;
    import android.util.AttributeSet;
    import android.view.View;

    import com.kenny.fcmpushtest.R;

    /**
     * Created by Kenny on 2016/10/19 10:33.
     * Desc:
     */
    public class FanLoading extends View {
        private static final String TAG = FanLoading.class.getSimpleName();
        private int mWidth;
        private int mHeight;
        private int defaultWidth = 200;
        private int defaultHeight = 200;
        private int withinRadius;//内圆半径
        private int fanRadius;//画风扇叶的半径
        private int fanCount = 4;//扇叶数

        //画图相关
        private Paint bgPaint;//画底色和画风扇
        private int bgColor;
        private Paint circlePaint;//内圆画笔
        private int circleColor;
        private RectF rect;//外部圆角矩形
        private RectF rectFan;//扇叶画图范围

        private int rotateDegree = 0;//旋转角度,用于旋转画布,实现动态旋转效果
        private int speed = 10;//旋转速率
        private boolean running;

        private Handler rotateHandler = new Handler() {
            @Override
            public void handleMessage(Message msg) {
                super.handleMessage(msg);
                if (!running)
                    return;
                rotateDegree += speed;
                if (rotateDegree == Integer.MAX_VALUE)
                    rotateDegree = 0;
                postInvalidate();
            }
        };

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

        public FanLoading(Context context, AttributeSet attrs) {
            this(context, attrs, 0);
        }

        public FanLoading(Context context, AttributeSet attrs, int defStyleAttr) {
            super(context, attrs, defStyleAttr);
            TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.FanLoading);
            bgColor = ta.getColor(R.styleable.FanLoading_rectBackgroundColor, Color.parseColor("#982C22"));
            circleColor = ta.getColor(R.styleable.FanLoading_circleColor, Color.parseColor("#ff0000"));
            speed = ta.getInt(R.styleable.FanLoading_fanSpeed, 10);
            running = ta.getBoolean(R.styleable.FanLoading_autoStart, false);
            ta.recycle();
        }

        @Override
        protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
            mWidth = getMyDefaultSize(defaultWidth, widthMeasureSpec);
            mHeight = getMyDefaultSize(defaultHeight, heightMeasureSpec);
            withinRadius = mWidth / 2 - 20;
            fanRadius = withinRadius - 10;
            setMeasuredDimension(mWidth, mHeight);
            init();
        }

        private void init() {
            bgPaint = new Paint();
            bgPaint.setColor(bgColor);
            bgPaint.setAntiAlias(true);
            bgPaint.setStyle(Paint.Style.FILL_AND_STROKE);

            circlePaint = new Paint();
            circlePaint.setColor(circleColor);
            circlePaint.setAntiAlias(true);
            circlePaint.setStyle(Paint.Style.FILL_AND_STROKE);

            rect = new RectF(0, 0, mWidth, mHeight);
            float left = mWidth / 2 - fanRadius / 2;
            float top = mHeight / 2 - fanRadius;
            float right = mWidth / 2 + fanRadius / 2;
            float bottom = mHeight / 2;
            rectFan = new RectF(left, top, right, bottom);
        }

        private int getMyDefaultSize(int size, int measureSpec) {
            int result = size;
            //获得测量模式
            int specMode = View.MeasureSpec.getMode(measureSpec);
            //获得测量大小
            int specSize = View.MeasureSpec.getSize(measureSpec);
            //判断模式是否是 EXACTLY
            if (specMode == View.MeasureSpec.EXACTLY) {
                //如果模式是 EXACTLY 则直接使用specSize的测量大小
                result = specSize;
            } else {
                //如果是其他两个模式,先设置一个默认大小值 200
                //如果是 AT_MOST 也就是 wrap_content 的话,就取默认值 200 和 specSize 中小的一个为准。
                if (specMode == View.MeasureSpec.AT_MOST) {
                    result = Math.min(result, specSize);
                }
            }
            return result;
        }

        @Override
        protected void onDraw(Canvas canvas) {
            super.onDraw(canvas);
            canvas.drawRoundRect(rect, 20, 20, bgPaint);
            canvas.drawCircle(mWidth / 2, mHeight / 2, (mHeight / 2) - 20, circlePaint);
            canvas.rotate(-rotateDegree, mWidth / 2, mHeight / 2);
            for (int i = 0; i < fanCount; i++) {
                canvas.drawArc(rectFan, -90, -180, true, bgPaint);
                canvas.rotate(90, mWidth / 2, mHeight / 2);
            }
            if (running)
                rotateHandler.sendEmptyMessageDelayed(1, 10);
        }

        @Override
        protected void onDetachedFromWindow() {
            super.onDetachedFromWindow();
            running = false;
        }

        public void start() {
            running = true;
            postInvalidate();
        }

        public void stop() {
            running = false;
            rotateHandler.removeMessages(1);
        }

        public void setRectBackgroundColor(int color) {
            bgColor = color;
            bgPaint.setColor(bgColor);
            postInvalidate();
        }

        public void setCircleColor(int color) {
            circleColor = color;
            circlePaint.setColor(circleColor);
            postInvalidate();
        }

        public void setSpeed(int s) {
            speed = s;
        }

        public boolean isRunning() {
            return running == true;
        }
    }

自定义属性部分

attr.xml
<?xml version="1.0" encoding="utf-8"?>
    <resources>

        <declare-styleable name="FanLoading">
            <attr name="rectBackgroundColor" format="color" />
            <attr name="circleColor" format="color" />
            <attr name="fanSpeed" format="integer" />
            <attr name="autoStart" format="boolean" />
        </declare-styleable>

    </resources>

恩,然后就大概这么多吧。

总结

总的来说,这个自定义view差不多已经是实现codewars的那个图标效果了,但是还是有一些没能实现,比如它中间并不是一个圆,而是像花瓣一样的。
整个过程花了差不多一天的时间,代码也有许多需要优化的地方,希望大家多多指教。

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,291评论 25 707
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,120评论 4 61
  • 文/啸谷 01 又有一个亲戚的亲戚倒下了,27岁,企业白领,送到医院时双肾衰竭,并发多处脏器感染,生命垂危,孩子仅...
    啸谷阅读 632评论 0 10
  • 养猫的感觉就像谈恋爱,养猫这个“爱情故事”要从一年前谈起了,去年元旦家里领导带我出去逛街,偶然的机会看到了柱子,从...
    柱先森阅读 645评论 0 2
  • 如果我们注定要在一起。我何不提早行使我的权利。何以琛是这么说的吧
    猪猫比巴卜阅读 137评论 0 0