圆盘形菜单

项目中需要定义一个圆盘形选择菜单,效果如下图:


圆盘形菜单

1、自定义View

思路是

1、定义一个类存储每一个Item的信息,:
    class Item {
        //图片
        Bitmap bitmap;
        //每个Item旋转的角度
        float angle;
        //图片中心点,x坐标
        float x;
        //图片中心点,y坐标
        float y;
        String id;
        ...
    }
2、通过代码传入基本的信息,初始化Items,包括角度、X、Y坐标、bitmap和其他的信息
求Item中心点坐标

通过三角函数值,得到x、y坐标:

x=pointX + (float) (radius * Math.cos(item.angle * Math.PI / 180))

代码如下:

    private void setUpItems() {
        ...
        //第一个Item默认是0°
        int angle = 0;
        //每个Item间距相同的度数= 360 / itemCount
        degreeDelta = 360 / itemCount;
        //初始化每个Item
        for (int index = 0; index < itemCount; index++) {
            Item item = new Item();
            item.angle = angle;
            item.x = pointX + (float) (radius * Math.cos(item.angle * Math.PI / 180));
            item.y = pointY + (float) (radius * Math.sin(item.angle * Math.PI / 180));
            item.bitmap = resizeImage(ImageLoader.getInstance().loadImageSync(selectItems.get(index).getQue_img(), options));
            ...
            items.add(item);
            angle += degreeDelta;
        }
          
    }

x、y都在以radius 为半径的圆A上,其中圆A的中心点是pointX 、pointY (位于屏幕中心)

3、draw

移动画布,即坐标系到屏幕的中心,

    @Override
    public void onDraw(Canvas canvas) {
        ...
        canvas.translate(getMeasuredWidth()/2,getMeasuredHeight()/2);
        for (int index = 0; index < itemCount; index++) {
            ...
            canvas.drawBitmap(items.get(index).bitmap, items.get(index).x - bitmap.getWidth() / 2,
                    items.get(index).y - bitmap.getHeight() / 2, null);
        }
    }
4、为View添加动画

继承Animation,在动画执行的过程中,动态改变角度、x、y值


    class MyAnimation extends Animation {
        ...
        @Override
        protected void applyTransformation(float interpolatedTime, Transformation t) {
            super.applyTransformation(interpolatedTime, t);
            //每次执行动画,运行0.5°
            float angle = items.get(0).angle += 0.5;
            for (int index = 0; index < itemCount; index++) {
                Item item = items.get(index);
                item.angle = angle;
                item.x = pointX + (float) (radius * Math.cos(item.angle * Math.PI / 180));
                item.y = pointY + (float) (radius * Math.sin(item.angle * Math.PI / 180));
                angle += degreeDelta;
                angle = angle % 360;
            }
            postInvalidate();
        }
    }
5、在触摸事件ACTION_UP发生时,判断该点是否在Item内,如果在Item内,则产生点击事件

怎么判断一个点是否在一个圆内呢?
可以通过坐标差的平方根与半径进行对比,小于半径在圆内

判断 是否在Item内
    /**
     * 确定触摸事件(ACTION_UP)发生的位置是否在,item(小圆圈)内
     * @param x
     * @param y
     */
    private void confirmPointPosition(float x, float y) {
        ...
        for (int index = 0; index < itemCount; index++) {
            float imgCenterX = items.get(index).x;
            float imgCenterY = items.get(index).y;
            if (items.get(index).bitmap == null) {
                break;
            }
            float imgCircle = items.get(index).bitmap.getWidth();
            double r = Math.sqrt(Math.pow(x - imgCenterX, 2) + Math.pow(y - imgCenterY, 2));
            if (r <= imgCircle / 2) {
                ...
                listener.onClick(typeId, typeName, ageId);
            }
        }
    }

代码:RorateCircleDemo

2、自定义ViewGroup

自定义ViewGroup
  • 将菜单项添加到ViewGroup:初始化item view,绑定数据
  • 测量:先测量自身大小,再测量item view大小
  • 布局:计算每个item的left、top的位置

attrs.xml

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <attr name="defaultMenuLayout" format="reference"/>
    <declare-styleable name="CircleMenuLayout">
        <attr name="item_layout" format="reference"/>
    </declare-styleable>
</resources>

style.xml

<resources>
    <style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
        ...
        <item name="defaultMenuLayout">@style/defaultCircleMenuLayoutStyle</item>
    </style>
    <style name="defaultCircleMenuLayoutStyle">
        <item name="item_layout">@layout/default_layout_circle_item</item>
    </style>
</resources>
public class CircleMenuLayout extends ViewGroup {

    private int menuItemLayoutId= R.layout.default_layout_circle_item;

    private int[] items=new int[]{R.mipmap.circle_item_1,R.mipmap.circle_item_2,R.mipmap.circle_item_3,R.mipmap.circle_item_4
            ,R.mipmap.circle_item_5,R.mipmap.circle_item_6};

    private int itemCount;
    private double startAngle;
    private double swapAngle;
    private int radius;
    private float itemRadio=0.25f;
    private float paddingRadio=0.05f;

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

    public CircleMenuLayout(Context context, AttributeSet attrs) {
        this(context, attrs,R.attr.defaultMenuLayout);
    }

    public CircleMenuLayout(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        TypedArray typedArray=context.obtainStyledAttributes(attrs,R.styleable.CircleMenuLayout,defStyleAttr,0);
        menuItemLayoutId=typedArray.getResourceId(R.styleable.CircleMenuLayout_item_layout,R.layout.default_layout_circle_item);
        typedArray.recycle();
        setPadding(0,0,0,0);

        init();
    }

    private void init() {
        itemCount=items.length;
        startAngle=0;
        swapAngle=360/itemCount;
        buildMenuItems();
    }

    //将菜单项添加到ViewGroup
    private void buildMenuItems() {
        for (int i=0;i<itemCount;i++){
            View itemView=inflaterMenuView(i);
            initMenuView(itemView,i);
            addView(itemView);
        }
    }

    private void initMenuView(View itemView, final int position) {
        ImageView img= (ImageView) itemView.findViewById(R.id.img);
        img.setImageResource(items[position]);
    }

    private View inflaterMenuView(final int position) {
        LayoutInflater inflater=LayoutInflater.from(getContext());
        View view=inflater.inflate(menuItemLayoutId,this,false);
        view.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                Toast.makeText(getContext(),"item : "+position,Toast.LENGTH_SHORT).show();
            }
        });
        return view;
    }


    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        measureSelf(widthMeasureSpec,heightMeasureSpec);
        measureMenuItems();
    }

    /**
     * 调用measure方法测量每个子view
     */
    private void measureMenuItems() {
        radius=Math.max(getMeasuredWidth(),getMeasuredHeight());
        int childCount=getChildCount();
        int specMode=MeasureSpec.EXACTLY;
        int specSize= (int) (radius*itemRadio);
        for (int i=0;i<childCount;i++){
            View child=getChildAt(i);
            if (child.getVisibility()==GONE){
                continue;
            }
            int measureSpec=-1;
            measureSpec=MeasureSpec.makeMeasureSpec(specSize,specMode);
            child.measure(measureSpec,measureSpec);
        }

    }

    /**
     * 如果是精确模式,大小是宽高的最小值
     * 如果是最大模式,大小是背景或屏幕的宽
     */
    private void measureSelf(int widthMeasureSpec, int heightMeasureSpec) {
        int reqWidth=0;
        int reqHeight=0;
        int widthSize=MeasureSpec.getSize(widthMeasureSpec);
        int widthMode=MeasureSpec.getMode(widthMeasureSpec);
        int heightSize=MeasureSpec.getSize(heightMeasureSpec);
        int heightMode=MeasureSpec.getMode(heightMeasureSpec);

        if (widthMode!=MeasureSpec.EXACTLY || heightMode!=MeasureSpec.EXACTLY){
            reqWidth=getSuggestedMinimumWidth();
            reqWidth=reqWidth==0?getDefaultWidth():reqWidth;
            reqHeight=getSuggestedMinimumHeight();
            reqHeight=reqHeight==0?getDefaultWidth():reqHeight;
        }else {
            reqWidth=reqHeight=Math.min(widthSize,heightSize);
        }
        setMeasuredDimension(reqWidth,reqHeight);
    }

    private int getDefaultWidth() {
        return getResources().getDisplayMetrics().widthPixels;
    }

    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
        int childCount=getChildCount();
        int padding= (int) (radius*paddingRadio);
        int left=0;
        int top=0;
        for (int i=0;i<childCount;i++){
            View child=getChildAt(i);
            if (child.getVisibility()==GONE){
                continue;
            }
            int itemSize=Math.max(child.getMeasuredWidth(),child.getMeasuredHeight());
            startAngle%=360;

            left= (int) (radius/2+(radius/2-padding-itemSize/2)*Math.cos(Math.toRadians(startAngle)))-itemSize/2;
            top= (int) (radius/2+(radius/2-padding-itemSize/2)*Math.sin(Math.toRadians(startAngle)))-itemSize/2;
            child.layout(left,top,left+itemSize,top+itemSize);
            startAngle+=swapAngle;
        }
    }
}

使用适配器,将变化隔离出去
因为每个菜单项都是一个view,可以将加载菜单项的布局、始化菜单项、绑定数据的工作通过adapter分离出去;在CircleMenuLayout 中,仅仅实现测量和布局就行。


public class CircleMenuLayout extends ViewGroup {

    private ListAdapter mAdapter;
    AdapterDataSetObserver mDataSetObserver;

    ...

    public void setAdapter(ListAdapter adapter){
        if (mAdapter != null && mDataSetObserver != null) {
            mAdapter.unregisterDataSetObserver(mDataSetObserver);
        }
        if (adapter!= null){
            this.mAdapter=adapter;
            mDataSetObserver = new AdapterDataSetObserver();
            mAdapter.registerDataSetObserver(mDataSetObserver);
            buildMenuItems();
            requestLayout();
        }
    }

    @Override
    protected void onAttachedToWindow() {
        if (mAdapter!=null){
            buildMenuItems();
        }
        super.onAttachedToWindow();
    }

    private void buildMenuItems() {

        int itemCount=mAdapter.getCount();
        startAngle=0;
        if (itemCount>0){
            swapAngle=360/itemCount;
        }
        for (int i=0;i<itemCount;i++){
            View itemView=mAdapter.getView(i,null,null);
            addView(itemView);
        }
    }

    ...

    private class AdapterDataSetObserver extends DataSetObserver {
        @Override
        public void onChanged() {
            buildMenuItems();
            requestLayout();
        }
        @Override
        public void onInvalidated() {
            buildMenuItems();
            requestLayout();
        }
    }
}

参考:android 源码设计模式

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

推荐阅读更多精彩内容