WheelView转盘选择器

WheelView

转盘指示器,项目地址Github:https://github.com/FengJiaCheng/WheelView

演示

使用

  • 布局
<com.fjc.library.WheelView
       android:id="@+id/wheelView"
       android:layout_width="match_parent"
       android:layout_height="match_parent"
       app:centerTextSize="18sp"
       app:color="@color/red"
       app:menuTextSize="20sp"
       app:num="5"
       app:text="指示器" />
  • 属性

    • num: 选项数量
    • text: 指示器文本
    • color: 指示器颜色
    • menuColor: 选项背景色
    • centerTextColor: 指示器文本颜色
    • selectMenuColor: 选项按压时背景色
    • menuTextColor: 选项文本颜色
    • menuTextSize: 选项文本字体大小
    • centerTextSize: 指示器文本字体大小
  • 代码

    //设置选择器菜单内容
    String[] strs = {"菜单1", "菜单2", "菜单3", "菜单4"};
    int[] imgs = {R.drawable.menu1,R.drawable.menu2,R.drawable.menu3,R.drawable.menu4};
    wheelView.setMenu(strs, imgs);
    //选择监听器
    wheelView.setOnCheckListener(new WheelView.OnCheckListener() {
             @Override
             public void onCheck(int position) {
                // do something;
             }
         });
    //指示器从当前颜色变化到红色
    wheelView.startColorAnimation(Color.RED);
    //指示器从红色变化到绿色
    wheelView.startColorAnimation(Color.RED,Color.GREEN);
    //指示器按颜色数组顺序渐变
    int[] colors={Color.RED,Color.GREEN,Color.WHITE}
    wheelView.startColorAnimation(colors);
    

解析

  • 绘制

    @Override
     protected void onDraw(Canvas canvas) {
         super.onDraw(canvas);
         int r = witdh > height ? height / 2 : witdh / 2;
         // 扇形半径
         int r1 = (int) (r * r1_xishu);
         // 指示器半径
         int r2 = (int) (r * r2_xishu);
         Paint paint = new Paint();
         paint.setStrokeWidth(3);
         paint.setAntiAlias(true);
         paint.setStyle(Paint.Style.FILL);
         // 画扇形
         for (int i = 0; i < num; i++) {
             // 默认颜色
             paint.setColor(menuColor);
             // 按下的menu背景变色
             if (touchId == i) {
                 paint.setColor(selectedMenuColor);
             }
             // 根据id计算弧度
             double pencent = (Math.PI * i / num * 2);
             // 扇形圆心与控件中心的偏移量
             int deltaX = (int) (delta * Math.cos(pencent));
             int deltaY = (int) (delta * Math.sin(pencent));
    
             RectF oval = new RectF();                     //RectF对象
             oval.left = witdh / 2 - r1 + deltaX;
             oval.top = height / 2 - r1 + deltaY;
             oval.right = witdh / 2 + r1 + deltaX;
             oval.bottom = height / 2 + r1 + deltaY;
             canvas.drawArc(oval, (float) ((i - 0.5) / (float) num * 360), 1 / (float) num * 360, true, paint);
         }
         paint.setStyle(Paint.Style.FILL);//设置实心
         // 设置为指示器颜色
         paint.setColor(color);
         // 画圆
         canvas.drawCircle(witdh / 2, height / 2, (float) (r2), paint);
         // 画三角形
         canvas.drawPath(getTrianglePath(angle, (int) (r2 / 0.8)), paint);
     }
    
    1. 绘制四个扇形,扇形半径为r1,但是为了制造出缝隙,需要将扇形圆心在控件中心基础上加一个偏移量(这个偏移量不等于缝隙的宽度)
    2. drawCircle绘制指示器的圆
    3. drawPath绘制指示器尖角,r2 / 0.8是因为尖角所在圆的半径比指示器的圆的半径略大
        private Path getTrianglePath(float i, int r) {
    
            //角度转弧度
            double angel = Math.PI * i / 180;
            int pointX = witdh / 2;
            int pointY = height / 2;
    
            //第一个点,指针尖角
            double x1 = Math.cos(angel) * r + pointX;
            double y1 = Math.sin(angel) * r + pointY;
    
            //默认指针尖角60度
            double a1 = Math.PI * (i + 180 + 30) / 180;
            double a2 = Math.PI * (i + 180 - 30) / 180;
    
            //三角形边长,边长不能太小,否则不能被圆覆盖住两角
            double aa = 0.3 * r;
    
            //第二个点
            double x2 = Math.cos(a1) * aa + x1;
            double y2 = Math.sin(a1) * aa + y1;
    
            //第三个点
            double x3 = Math.cos(a2) * aa + x1;
            double y3 = Math.sin(a2) * aa + y1;
    
            Path path = new Path();
            path.moveTo((float) x1, (float) y1);
            path.lineTo((float) x2, (float) y2);
            path.lineTo((float) x3, (float) y3);
            path.close();
    
            return path;
        }
    

    尖角画了一个等边三角形,其中尖角为60度,其中a1,a2是其他两个角以尖角(x1,y1)为原点时的角度,以此计算sin和cos,sin、cos乘以边长aa就可以得到相对于(x1,y1)的坐标偏移值,以此确定两点坐标。

  • 触摸事件

    @Override
    public boolean onTouchEvent(MotionEvent event) {
    
        float x = event.getX();
        float y = event.getY();
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                touchId = getCheckId(x, y);
                break;
            case MotionEvent.ACTION_MOVE:
                // 手指移出扇形时取消
                if (getCheckId(x, y) != touchId) {
                    touchId = -1;
                }
                break;
            case MotionEvent.ACTION_UP:
                if (getCheckId(x, y) == touchId && touchId != -1) {
                    check(touchId);
                }
                touchId = -1;
                break;
        }
        //重新绘制
        invalidate();
        return true;
    }
    

    利用getCheckId方法获取当前手指触摸的选项

    private int getCheckId(float x, float y) {
    
        int index = -1;
        // 中心点坐标
        int potX = witdh / 2;
        int potY = height / 2;
    
        int r = potX > potY ? potY : potX;
        // r*r1_xishu是扇形的半径,r1是扇形拼成的大圆的半径,两者之间有delta偏移量
        int r1 = (int) (r * r1_xishu + delta);
        // 指示器半径
        int r2 = (int) (r * r2_xishu);
    
        if (Math.pow(x - potX, 2) + Math.pow(y - potY, 2) < Math.pow(r1, 2)
                && Math.pow(x - potX, 2) + Math.pow(y - potY, 2) > Math.pow(r2, 2)) {
            double delta_x = x - potX;
            double delta_y = y - potY;
            double r3 = Math.sqrt(Math.pow(delta_x, 2) + Math.pow(delta_y, 2));
            double sin = delta_y / r3;
            double cos = delta_x / r3;
            //求arcsin反三角 0-2pi 会有两值
            double angel1 = (Math.asin(sin) + 2 * Math.PI) % (2 * Math.PI);
            double angel2 = (3 * Math.PI - angel1) % (2 * Math.PI);
            double selectAngel = 0;
            // 利用cos确认 正确的角度值
            if (Math.cos(angel2) * cos > 0) {
                selectAngel = angel2;
            } else if (Math.cos(angel1) * cos > 0) {
                selectAngel = angel1;
            }
            // 弧度转角度
            double selectAngel2 = (selectAngel * 180 / Math.PI);
            // 根据角度算id
            index = (int) (selectAngel2 / (360 / (double) num) + 0.5) % num;
    
        }
    
        return index;
    }
    
    1. 先判断是不是在大圆和小圆之间。此时大圆的半径不是扇形半径(比扇形半径略大,需要加上偏移值)
    2. delta_x``delta_y是按压点相对于中心点的偏移值
    3. double r3 = Math.sqrt(Math.pow(delta_x, 2) + Math.pow(delta_y, 2));求出按压点距离控件中心的距离
    4. 求出sin,cos
    5. arcsin反三角求弧度,弧度范围在[-PI/2,PI/2]之间,所以在一个2PI周期里面有两个值,即代码中angel1,angel2
    6. 比较cos值来确定正确的弧度,其中angel1,angel2的cos符号相反。
    7. index = (int) (selectAngel2 / (360 / (double) num) + 0.5) % num;根据角度算出当前的id,为什么有0.5因为绘图时扇形角度整体偏移1/2。第一个扇形是关于x轴对称的,角度范围是(-360/num/2,360/num/2)。
  • 选择选项
    private void check(final int index) {
         if (colorAnimator != null) {
             colorAnimator.cancel();
         }
         AnimatorSet set = new AnimatorSet();
    
         // 计算目标角度
         float angle2 = index / (float) num * 360;
         // 使得每次旋转选取最近方向
         if (Math.abs(angle2 - angle) > 180) {
             if (angle2 > angle) {
                 angle += 360;
    
             } else {
                 angle2 += 360;
             }
         }
         // 旋转动画
         ObjectAnimator animator1 = ObjectAnimator.ofFloat(WheelView.this, "angle", angle, angle2);
         set.play(animator1);
         set.setDuration(rotateDuration);
         set.start();
         checkId = index;
         set.addListener(new Animator.AnimatorListener() {
             @Override
             public void onAnimationStart(Animator animator) {
    
             }
    
             @Override
             public void onAnimationEnd(Animator animator) {
                 if (listener != null) {
                     listener.onCheck(index);
                 }
             }
    
             @Override
             public void onAnimationCancel(Animator animator) {
    
             }
    
             @Override
             public void onAnimationRepeat(Animator animator) {
             }
         });
     }
    
    1. 计算目标角度float angle2 = index / (float) num * 360;
    2. 为了使转盘总是往最近方向转,需要给目标角度和当前角度中较小的角度加上360

    比如 从30度转到 270度,从优弧转即30-270,从劣弧转即390-270
    3. 转动完毕执行回调事件

  • 指示器颜色渐变
    public void startColorAnimation(int[] colors) {
         if (colorAnimator != null) {
             colorAnimator.cancel();
         }
         colorAnimator =
                 ObjectAnimator.ofInt(WheelView.this, "color", colors);
         colorAnimator.setEvaluator(new ArgbEvaluator());
         colorAnimator.setDuration(colorDuration);
         colorAnimator.start();
     }
    

    startColorAnimationz最终都是调用startColorAnimation(int[] colors) 这个方法。关于这个指示器颜色动画的使用,用户可以灵活选择执行的时间,执行的时长,执行的颜色。

详细请看源码,项目地址Github:https://github.com/FengJiaCheng/WheelView

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

推荐阅读更多精彩内容