【原创】自定义View——双色球彩票选号界面,模仿网易彩票

一、先看效果

ssqSelectNumber.gif

二、准备工作

典型的自定义view,把这个view当作一个对象来看,先撸撸这个view的属性,上面的动态图直观的可以看到以下几点:

  1. 上下左右的padding
  2. 球与球之间的space
  3. 每行球的个数(球列数需要吗?答:不需要,总球数除以行数就是列数了。)
  4. 是否显示遗漏(遗漏文字也是有颜色的,这个不是view的属性,可以单独设置在球上面)
  5. 球选中的颜色,未选中的颜色 (这个可以是小球的属性,但是由于一个选号区所有的球都是一样的,可以看成view的属性了)
  6. 球选中的背景,未选中的背景 (理由同上)

既然是选号,每一个球也可以看成对象,球的属性要考虑好,涉及后面的位置计算:

  1. 球的号码
  2. 球选中的颜色,未选中的颜色
  3. 球选中的背景,未选中的背景
  4. 遗漏值,遗漏值的颜色
  5. 球位置(left, top, right, bottom) (这里记录球所在的矩形区域的上下左右位置,为什么?留给大家思考)
  6. 遗漏位置 (同上)

当然一开始肯定考虑的不是这么全面,还有限制最多选多少球,已选的号码后面选球区不能重复等,就靠大家一步步完善了。

三、小球建模

public class Ball {
    private String number;                      // 号码
    private boolean isSelected;                 // 是否选中
    private int left;                           // 球左坐标
    private int top;                            // 球上坐标
    private int right;                          // 球右坐标
    private int bottom;                         // 球下坐标

    private float x;                            // 球圆心x
    private float y;                            // 球圆心y

    private int mLeft;                          // 遗漏左坐标
    private int mTop;                           // 遗漏上坐标
    private int mRight;                         // 遗漏右坐标
    private int mBottom;                        // 遗漏下坐标
    private String missValue = "8";             // 遗漏值
    private int missValueColor = Color.RED;     // 遗漏文字颜色
}

上面有一个圆心,但是如果利用圆心绘制文字的话,文字不能居中,如果要让它居中,还需要再绘制的时候计算,效率上就不是很高,所以舍弃了。同时要给球暴漏两个很重要的方法:

/**
     * 设置球的位置
     * @param left
     * @param top
     * @param right
     * @param bottom
     */
    public void setRect(int left, int top, int right, int bottom) {
        this.left = left;
        this.top = top;
        this.right = right;
        this.bottom = bottom;
    }

/**
     * 设置遗漏的位置
     * @param left
     * @param top
     * @param right
     * @param bottom
     */
    public void setMissRect(int left, int top, int right, int bottom) {
        this.mLeft = left;
        this.mTop = top;
        this.mRight = right;
        this.mBottom = bottom;
    }

这两个方法就是绘制很顺滑的关键,后面再细说。

四、绘制选号区

依据步骤二我们先在属性文件里列好自定义的属性:

<declare-styleable name="SelectBallsView">
        <attr name="numCount" format="integer"/>
        <attr name="start" format="integer"/>
        <attr name="end" format="integer"/>
        <attr name="hasZero" format="boolean"/>
        <attr name="ballColor" format="color"/>
        <attr name="txtSelectedColor" format="color"/>
        <attr name="txtUnselectedColor" format="color"/>
        <attr name="drawableSelected" format="reference|integer"/>
        <attr name="drawableUnselected" format="reference|integer"/>
    </declare-styleable>

初始化的时候拿到这些属性,在绘制之前,我们需要计算好每一个球的位置,那什么时候可以计算呢?答案就是onMeasure!!!一旦布局测量好宽度,我们每一个球的位置都是可以计算的:

@Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        width = getMeasuredWidth();
        missWidth = ballWidth = ballHeight = (width - (numCount - 1) * space - padding * 2) / numCount;
        missHeight = missWidth / 2;
        computeLocation(ballWidth, ballHeight);
        int rows = balls.size() / numCount + ((balls.size() % numCount > 0) ? 1 : 0);
        if (showMissValue) {
            height = rows * (ballHeight + missHeight) + (rows - 1) * space + padding * 2;
        } else {
            height = rows * ballHeight + (rows - 1) * space + padding * 2;
        }
        setMeasuredDimension(width, height);
    }

/**
     * 计算球位置  文字位置
     *
     * @param ballWidth
     * @param ballHeight
     */
    private void computeLocation(int ballWidth, int ballHeight) {
        int size = balls.size();
        for (int i = 0; i < size; i++) {
            Ball cb = balls.get(i);
            if (showMissValue) {
                cb.setRect((i % numCount) * ballWidth + i % numCount * space + padding, 
                        (i / numCount) * ballHeight + i / numCount * (space + missHeight) + padding,
                        (i % numCount + 1) * ballWidth + i % numCount * space + padding, 
                        (i / numCount + 1) * ballHeight + i / numCount * (space + missHeight) + padding);
                cb.setMissRect((i % numCount) * ballWidth + i % numCount * space + padding, 
                        (i / numCount) * ballHeight + i / numCount * (space + missHeight) + padding + ballHeight,
                        (i % numCount + 1) * ballWidth + i % numCount * space + padding, 
                        (i / numCount + 1) * ballHeight + i / numCount * (space + missHeight) + padding + missHeight);
            } else {
                cb.setRect((i % numCount) * ballWidth + i % numCount * space + padding, 
                        (i / numCount) * ballHeight + i / numCount * space + padding,
                        (i % numCount + 1) * ballWidth + i % numCount * space + padding, 
                        (i / numCount + 1) * ballHeight + i / numCount * space + padding);
            }
        }
    }

这里就是计算并保存球的上下左右坐标信息,一来避免onDraw的时候计算,二来方便绘制文字的位置。
有了小球的信息,接下来就是简单的绘制了:

@Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        int size = balls.size();
        for (int i = 0; i < size; i++) {
            Ball cb = balls.get(i);
            RectF rectF = new RectF(cb.getLeft(), cb.getTop(), cb.getRight(), cb.getBottom());
            if (bitmapSelected != null || bitmapUnselected != null) {
                //绘制图片
                canvas.drawBitmap(cb.isSelected() ? bitmapSelected : bitmapUnselected, null,
                        new Rect(cb.getLeft(), cb.getTop(), cb.getRight(), cb.getBottom()), ballPaint);
            } else {
                //绘制小球边框和小球
                canvas.drawArc(new RectF(cb.getLeft() - strokeWidth, cb.getTop() - strokeWidth,
                        cb.getRight() + strokeWidth, cb.getBottom() + strokeWidth), 0, 360, false, circlePaint);
                ballPaint.setColor(cb.isSelected() ? ballColor : Color.WHITE);
                canvas.drawOval(rectF, ballPaint);
            }
            //绘制文字
            txtPaint.setColor(cb.isSelected() ? txtSelectedColor : txtUnselectedColor);
            canvas.drawText(cb.getNumber(), rectF.centerX(), rectF.centerY() - txtMidValue, txtPaint);
            //绘制遗漏
            if (showMissValue) {
                msPaint.setColor(cb.getMissValueColor());
                RectF missRectF = new RectF(cb.getmLeft(), cb.getmTop(), cb.getmRight(), cb.getmBottom());
                canvas.drawText(cb.getMissValue(), missRectF.centerX(), missRectF.centerY() - missMidValue, msPaint);
            }
        }
    }

onDraw里面就是遍历所有的球,首先看有没有设置选号区的小球背景,没有就绘制球,有的话直接绘制bitmap,着重看下绘制文字:

RectF rectF = new RectF(cb.getLeft(), cb.getTop(), cb.getRight(), cb.getBottom());
//绘制文字
txtPaint.setColor(cb.isSelected() ? txtSelectedColor : txtUnselectedColor);
canvas.drawText(cb.getNumber(), rectF.centerX(), rectF.centerY() - txtMidValue, txtPaint);

先定义一个绘制的RectF对象,就是当前小球的区域,文字的位置是rectF.centerX()和rectF.centerY() - txtMidValue,这个txtMidValue是怎么计算的呢?

        txtPaint = new Paint();
        txtPaint.setAntiAlias(true);
        txtPaint.setTextSize(sp2px(mContext, 14));
        txtPaint.setTextAlign(Paint.Align.CENTER);
        Paint.FontMetrics fontMetrics = txtPaint.getFontMetrics();
        txtMidValue = (fontMetrics.top + fontMetrics.bottom) / 2;

原理的话,参考百度词条,“canvas绘制文字如何居中”。在此不罗嗦了。绘制完之后,需要响应点击事件,最好的办法是重写onTouchEvent事件:

@Override
    public boolean onTouchEvent(MotionEvent event) {
        float x = event.getX();
        float y = event.getY();
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:

                break;
            case MotionEvent.ACTION_UP:
                int size = balls.size();
                for (int i = 0; i < size; i++) {
                    Ball tempBall = balls.get(i);
                    if (pointAtBall(tempBall, x, y)) {
                        tempBall.setSelected(!tempBall.isSelected());
                        requestLayout();
                        invalidate();
                        break;
                    }
                }
                break;
        }
        return true;
    }

    /**
     * 这个点落在球上
     *
     * @param tempBall
     * @param x
     * @param y
     * @return
     */
    private boolean pointAtBall(Ball tempBall, float x, float y) {
        Rect rect = new Rect(tempBall.getLeft(), tempBall.getTop(), tempBall.getRight(), tempBall.getBottom());
        return rect.contains((int) x, (int) y);
    }

手指抬起的时候,判断一下点是不是落在当前小球上,是的话就跳出循环,调用一requestLayout()和invalidate(),刷新一下布局。

五、示例

新建布局:

<ScrollView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        >
        <LinearLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:orientation="vertical"
            >
            <Button
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="显示/隐藏遗漏"
                android:onClick="showMiss"
                />

            <com.taovo.rjp.ssqselectnumber.SelectBallsView
                android:id="@+id/view_1"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                app:numCount="10"
                app:start="0"
                app:end="9"
                app:hasZero="false"
                app:ballColor="#ee00ff"
                app:txtSelectedColor="#ffffff"
                app:txtUnselectedColor="#ee00ff"
                />

            <com.taovo.rjp.ssqselectnumber.SelectBallsView
                android:id="@+id/view_2"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                app:numCount="9"
                app:start="1"
                app:end="33"
                app:hasZero="true"
                app:ballColor="#ff0066"
                app:txtSelectedColor="#ffffff"
                app:txtUnselectedColor="#ff0066"
                />

            <com.taovo.rjp.ssqselectnumber.SelectBallsView
                android:id="@+id/view_3"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                app:numCount="8"
                app:start="1"
                app:end="16"
                app:hasZero="true"
                app:ballColor="#0088ff"
                app:txtSelectedColor="#ffffff"
                app:txtUnselectedColor="#0088ff"
                />

            <com.taovo.rjp.ssqselectnumber.SelectBallsView
                android:id="@+id/view_4"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                app:numCount="7"
                app:start="1"
                app:end="6"
                app:hasZero="false"
                app:ballColor="#00ff26"
                app:txtSelectedColor="#ffffff"
                app:txtUnselectedColor="#00ff26"
                />

            <com.taovo.rjp.ssqselectnumber.SelectBallsView
                android:id="@+id/view_5"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                app:numCount="6"
                app:start="1"
                app:end="6"
                app:hasZero="false"
                app:ballColor="#00ff26"
                app:txtSelectedColor="#ffffff"
                app:txtUnselectedColor="#00ff26"
                app:drawableSelected="@mipmap/basketball"
                app:drawableUnselected="@mipmap/football"
                />

            <Button
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:text="确定"
                android:onClick="confirm"
                />
        </LinearLayout>
    </ScrollView>

mainActivity两个响应方法:

public void showMiss(View view){
        view1.setShowMissValue(!view1.getShowMissValue());
        view2.setShowMissValue(!view2.getShowMissValue());
        view3.setShowMissValue(!view3.getShowMissValue());
        view4.setShowMissValue(!view4.getShowMissValue());
        view5.setShowMissValue(!view5.getShowMissValue());
    }

    public void confirm(View view){
        String str1 = view1.getSelectBallsString();
        String str2 = view2.getSelectBallsString();
        String str3 = view3.getSelectBallsString();
        String str4 = view4.getSelectBallsString();
        String str5 = view5.getSelectBallsString();
        Toast.makeText(this, "选中的号码是:\n" + str1 + "\n" + str2 + "\n" + str3 + "\n" + str4 + "\n" + str5, Toast.LENGTH_SHORT).show();
    }

完事。效果就是一开始的示例效果了。有兴趣可以下载源码下来跑一跑,附上链接
GayHub

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

推荐阅读更多精彩内容