Android -- 一个滑动旋转的弧形菜单

效果图

效果图.gif

这是一个自定义的弧形菜单控件,手指滑动可以对其进行旋转,点击图标可以做一些操作,功能就是这样,下面介绍是如何实现的。

功能实现

自定义属性

要实现这样一个控件,首先要知道这个圆弧的半径mRadius,以及初始可见的图标个数mVisiableItemCount(这里是5个)。我们来设置两个自定义属性,在attrs.xml中添加如下代码:

<declare-styleable name="ArcDragMenu">
    <attr name="mradius" format="dimension" />
    <attr name="visibleitemcount" format="integer" />
</declare-styleable>

这样我们就可以在布局文件中设置自定义的属性。

<com.example.arcmenu.view.ArcDragMenu
        android:id="@+id/arcdragmenu"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        app:mradius="360dp"
        app:visibleitemcount="5"/>

在ArcDragMenu的构造方法中获取自定义属性的值。

public ArcDragMenu(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        // 获取自定义属性的值
        TypedArray a = context.getTheme().obtainStyledAttributes(attrs,
                R.styleable.ArcDragMenu, defStyleAttr, 0);
        mRadius = (int) a.getDimension(R.styleable.ArcDragMenu_mradius, TypedValue
                .applyDimension(TypedValue.COMPLEX_UNIT_DIP, 360,
                        getResources().getDisplayMetrics()));
        mVisiableItemCount = (int) a.getInteger(R.styleable.ArcDragMenu_visibleitemcount, 5);
        a.recycle();
}

计算角度和位置

角度.PNG

如图,我们把整个圆弧的角度分成mVisiableItemCount份(这里是5份),那么图中蓝∠占1份,黄∠占2份,黑∠占2.5份。黑∠的对边为VIew宽度的一半,斜边为圆弧半径mRadius,由此可得:

黑∠ = Math.asin((getMeasuredWidth()/2.0)/mRadius);

蓝∠的角度的大小angleDelay为:

angleDelay = Math.asin((getMeasuredWidth()/2.0)/mRadius)*2/ mVisiableItemCount;

第一个图标初始角度mInitialAngle的值(即黄∠):

//这里加负号表示位于中心轴的左边
mInitialAngle = angleDelay *(-(mVisiableItemCount /2.0 - 0.5));

第二个图标的角度为mInitialAngle+angleDelay ,其他以此类推。
知道了角度,计算位置就很简单了,这里就不一一计算了,直接看代码。

    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
        angleDelay = Math.asin((getMeasuredWidth()/2.0)/mRadius)*2/ mVisiableItemCount;
        mInitialAngle = angleDelay *(-(mVisiableItemCount /2.0-0.5));
        if(mCurrAngle ==0){
            mCurrAngle = mInitialAngle;
        }
        double angle = mCurrAngle;
        int count = getChildCount();
        for (int i = 0; i < count; i++){
            View child = getChildAt(i);
            //子View的左上角坐标(cl,ct)
            int cl = (int) (mRadius * Math.sin(angle)) + getMeasuredWidth()/2 - child.getMeasuredWidth()/2;
            int ct = (int) (mRadius * Math.cos(angle)) ;
            //测量的子View的宽,高
            int cWidth = child.getMeasuredWidth();
            int cHeight = child.getMeasuredHeight();
            //设置子view的位置
            child.layout(cl, ct, cl + cWidth, ct + cHeight);
            angle += angleDelay;
        }
    }

滑动

由上面的代码可以看出,图标的位置是由当前角度mCurrAngle来计算的,所以我们只需改变mCurrAngle的值即可滑动控件。我们要计算出手指按下的角度,手指移动过程中角度,从而计算出移动了多少角度,然后加到mCurrAngle上。部分代码如下:

    @Override
    public boolean dispatchTouchEvent(MotionEvent ev) {
        float x = ev.getRawX();
        float y = ev.getRawY();
        switch (ev.getAction()) {
            case MotionEvent.ACTION_DOWN:
                mLastX = x;
                mLastY = y;
                break;
            case MotionEvent.ACTION_MOVE:
                /**
                 * 获得开始的角度
                 */
                float start = getAngle(mLastX, mLastY);
                /**
                 * 获得当前的角度
                 */
                float end = getAngle(x, y);
                float dr = end - start;
                //防止超出范围,左滑到最后一个,右滑到第一个就不能再滑了
                if(mCurrAngle + dr <= mInitialAngle && mCurrAngle + dr >= mInitialAngle - (mMenuItemCount- mVisiableItemCount)*angleDelay){
                    mCurrAngle += dr;
                }
                // 重新布局
                requestLayout();

                mLastX = x;
                mLastY = y;
                break;
            case MotionEvent.ACTION_UP:
                
                break;

            default:
                break;
        }
        return super.dispatchTouchEvent(ev);
    }

    private float getAngle(float xTouch, float yTouch) {
        double x = xTouch - getMeasuredWidth()/2;
        double y = yTouch;
        return (float) (Math.asin(x / Math.hypot(x, y)));//其中Math.hypot(x, y)为sqrt(x2 +y2)
    }

这样图标就可以随着手指一起滑动了,但是你可能会觉得太生硬了,手指松开就立刻停了,如果快速滑动时让它Fling一会就好了。

Fling

当手指抬起时,我们计算一下移动的角的速度。

// 计算每秒移动的角度
float anglePerSecond = mTmpAngle * 1000 / (System.currentTimeMillis() - mDownTime);

我们开一个任务去慢慢递减anglePerSecond 的值,同时去改变mCurrAngle的值,这样手指抬起后还能继续滑动,代码如下:

   /**
    * 记录上一次的x,y坐标
    */
    private float mLastX;
    private float mLastY;

    @Override
    public boolean dispatchTouchEvent(MotionEvent ev) {
        float x = ev.getRawX();
        float y = ev.getRawY();
        switch (ev.getAction()) {
            case MotionEvent.ACTION_DOWN:
                mLastX = x;
                mLastY = y;
                mDownTime = System.currentTimeMillis();
                mTmpAngle = 0;
                // 如果当前已经在快速滚动
                if (isFling){
                    // 移除快速滚动的回调
                    removeCallbacks(mFlingRunnable);
                    isFling = false;
                    return true;
                }
                break;
            case MotionEvent.ACTION_MOVE:
                /**
                 * 获得开始的角度
                 */
                float start = getAngle(mLastX, mLastY);
                /**
                 * 获得当前的角度
                 */
                float end = getAngle(x, y);
                float dr = end - start;
                //防止超出范围,左滑到最后一个,右滑到第一个就不能再滑了
                if(mCurrAngle + dr <= mInitialAngle && mCurrAngle + dr >= mInitialAngle - (mMenuItemCount- mVisiableItemCount)*angleDelay){
                    mCurrAngle += dr;
                }

                mTmpAngle += end - start;
                // 重新布局
                requestLayout();

                mLastX = x;
                mLastY = y;
                break;
            case MotionEvent.ACTION_UP:
                // 计算每秒移动的角度
                float anglePerSecond = mTmpAngle * 1000
                        / (System.currentTimeMillis() - mDownTime);
                // 如果达到该值认为是快速移动
                if (Math.abs(anglePerSecond) > FLINGABLE_VALUE && !isFling) {
                    // post一个任务,去自动滚动
                    post(mFlingRunnable = new AutoFlingRunnable(anglePerSecond));

                    return true;
                }

                // 如果当前旋转角度超过NOCLICK_VALUE屏蔽点击
                if (Math.abs(mTmpAngle) > NOCLICK_VALUE || System.currentTimeMillis()-mDownTime >500) {
                    return true;
                }
                break;

            default:
                break;
        }
        return super.dispatchTouchEvent(ev);
    }

    private float getAngle(float xTouch, float yTouch) {
        double x = xTouch - getMeasuredWidth()/2;
        double y = yTouch;
        return (float) (Math.asin(x / Math.hypot(x, y)));//其中Math.hypot(x, y)为sqrt(x2 +y2)
    }

    /**
     * 自动滚动的任务
     */
    private class AutoFlingRunnable implements Runnable{

        private float angelPerSecond;

        public AutoFlingRunnable(float velocity)
        {
            this.angelPerSecond = velocity;
        }

        public void run(){
            // 如果小于0.1,则停止
            if (Math.abs(angelPerSecond) < 0.1f){
                isFling = false;
                return;
            }
            isFling = true;
            // 不断改变mCurrAngle ,让其滚动,/60为了避免滚动太快
            float dr = (angelPerSecond / 60);
            if(mCurrAngle + dr <= mInitialAngle && mCurrAngle + dr >= mInitialAngle - (mMenuItemCount- mVisiableItemCount)*angleDelay){
                mCurrAngle += dr;
            }else if(mCurrAngle + dr <= mInitialAngle){
                mCurrAngle = mInitialAngle - (mMenuItemCount- mVisiableItemCount)*angleDelay;
            }else if(mCurrAngle + dr >= mInitialAngle - (mMenuItemCount- mVisiableItemCount)*angleDelay){
                mCurrAngle = mInitialAngle;
            }
            // 逐渐减小这个值
            angelPerSecond /= 1.066f;
            postDelayed(this, 10);
            // 重新布局
            requestLayout();
        }
    }

到此已经全部结束了,有哪些做的不对的地方,希望大家多多指点。
源码

欢迎关注.jpg

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,019评论 25 707
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,089评论 4 62
  • 1.浮动元素有什么特征?对父容器、其他浮动元素、普通元素、文字分别有什么影响? 浮动元素不在文档的普通流中,它可以...
    饥人谷_米弥轮阅读 347评论 0 0
  • 先讲一个小故事。 寒假前夕,一名广告专业的学生小袁来电告诉我,她已获得一份寒假实习生职位——位于北京的蓝色光标总部...
    珞狮南路南阅读 441评论 0 0
  • 是前世注定的缘分么? 不经意间听见你轻柔的呼唤 等你, 在桥头于是我来了,义无返顾地 没有迟疑 ,没有惶惑 踏着落...
    水木宁阅读 1,170评论 26 36