Android 自定义控件基础-抽奖转盘(SurfaceView)

  这次记录一下使用自定义的SurfaceView做一个简单的案例。
  SurfaceView相较于普通的View,它的绘制操作在一个子线程中进行的,而普通的View是在UI线  程中绘制的,基于这个原因,SurfaceView的优势就是:可以避免UI线程的阻塞。由于SurfaceView的这个特性,很多游戏都是基于它来完成。
  这里就有一个问题,那就是SurfaceView没有在UI线程进行绘制,也就是它没在OnDraw方法里面进行相关的绘制的操作,那SurfaceView的绘制在哪里进行的呢?
  其实SurfaceView中包含一个专门用于绘制的Surface,这个Surface中包含了一个Canvas。通常我们可以这样拿到Canvas:通过getHolder()方法来获得一个SurfaceHolder,在这个Holde里面就含有Canvas,同时Holder还管理着SurfaceView的生命周期
  SurfaceView的生命周期方法:

  1.surfaceCreated() ,通常我们在这个方法里面进行一些变量的初始化和将绘制图形的线程进行开启。

  2.surefaceChanged()

  3.surfaceDestroyed() 在这个方法里面,我们通常将绘制图形的线程进行关闭

1.SurfaceView的固定格式

  通常我们在使用SurfaceView的时候,都有一个固定的格式,如下:

package com.example.android_luckpan;

import android.content.Context;
import android.graphics.Canvas;
import android.util.AttributeSet;
import android.view.SurfaceHolder;
import android.view.SurfaceView;

/**
 * Created by 前世诀别的一纸书 on 2017/2/25.
 */

public class SurfaceViewTemplate extends SurfaceView implements SurfaceHolder.Callback , Runnable{
    private SurfaceHolder mHolder = null;
    private Canvas mCanvas = null;

    private boolean mIsRuning = false;
    private Thread t = null;


    public SurfaceViewTemplate(Context context) {
        super(context);
    }

    public SurfaceViewTemplate(Context context, AttributeSet attrs) {
        super(context, attrs);
        mHolder = getHolder();
        mHolder.addCallback(this);
    }

    @Override
    public void surfaceCreated(SurfaceHolder holder) {
        mIsRuning = true;
        t = new Thread(this);
        t.start();
    }

    @Override
    public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {

    }

    @Override
    public void surfaceDestroyed(SurfaceHolder holder) {
        mIsRuning =false;
    }

    @Override
    public void run() {
        while (mIsRuning)
        {
            draw();
        }
    }

    private void draw() {
        try {
            mCanvas = mHolder.lockCanvas();
            if(mCanvas != null)
            {
                //draw something

            }
        } catch (Exception e) {
        } finally {
            mHolder.unlockCanvasAndPost(mCanvas);
        }

    }

}

  我们在使用SurfaceView的时候,通常是在这个模板代码上来添加我们的东西。
  现在回到我们这个案例的主题上来。

2.定义变量

  这个的变量有很多,同时初始化的位置也不相同。

private SurfaceHolder mHolder = null;
    private Canvas mCanvas = null;

    //线程开关--用来控制线程是否启动
    private boolean mIsRuning = false;
    //用来判断是否点击了停止按钮
    private boolean mIsShouldEnd = false;


    private Thread t = null;
    
    //转盘上每一项上的图片
    private int mImgs[] = new int[]{R.mipmap.danfan, R.mipmap.ipad, R.mipmap.f040, R.mipmap.iphone, R.mipmap.meizi, R.mipmap.f015};
    //转盘上每一项的文字
    private String strs[] = new String[]{"单反相机", "IPAD", "恭喜发财", "IPHONE", "服装一套", "恭喜发财"};
    //转盘上每一项的颜色
    private int mColors[] = new int[]{Color.parseColor("#FF4500"), Color.parseColor("#FF69B4"), Color.parseColor("#FF4500"), Color.parseColor("#FF69B4"), Color.parseColor("#FF4500"), Color.parseColor("#FF69B4")};
    //将转盘上每一项的图片转换Bitmap类型
    private Bitmap mImgsBitmap[] = null;
    //转盘的背景图片
    private Bitmap mBgBitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.bg2);
    //绘制圆盘每一项的画笔
    private Paint mArcPaint = null;
    //绘制转盘上每一项上文字的画笔
    private Paint mTextPaint = null;
    private int mPadding = 0;
    private int mRadius = 0;
    private int mItemCount = 6;
    private int mCenter = 0;
    //转盘开始转动的初始角度
    private float mStartAngle = 0;
    //转盘转动的速度
    private volatile float mSpeed = 0;



    private RectF mRange = null;
    private int mTextColor = Color.parseColor("#FFF5EE");
    private int mTextSize = (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_SP, 20, getResources().getDisplayMetrics());

3.初始化变量和测量SurfaceView的大小

  在测量SurfaceView的大小的时候,一定要注意:我们要保证SurfaceView是正方形的,因为转盘是圆形,正方形才好控制圆形的位置和大小。

@Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        int width = Math.min(getMeasuredHeight(), getMeasuredWidth());
        mPadding = getPaddingLeft();
        mRadius = width - 2 * mPadding;
        mCenter = width / 2;

        setMeasuredDimension(width, width);
    }

  我们在surfaceCreated方法中初始化一些变量

public void surfaceCreated(SurfaceHolder holder) {
        Log.i("main", "1");
        mArcPaint = new Paint();
        mArcPaint.setAntiAlias(true);
        mArcPaint.setDither(true);


        mTextPaint = new Paint();
        mTextPaint.setAntiAlias(true);
        mTextPaint.setDither(true);
        mTextPaint.setTextSize(mTextSize);
        mTextPaint.setColor(mTextColor);

        mRange = new RectF(mPadding, mPadding, mPadding + mRadius, mPadding + mRadius);

        mImgsBitmap = new Bitmap[mItemCount];
        for (int i = 0; i < mItemCount; i++) {
            mImgsBitmap[i] = BitmapFactory.decodeResource(getContext().getResources(), mImgs[i]);
        }

        Log.i("main", "3");
        mIsRuning = true;
        t = new Thread(this);
        t.start();
    }

3.绘制背景图片

 &emsp本质上我们应该在子线程的run方法里面绘制,我们将绘制的代码提取成了一个方法。

@Override
    public void run() {
        Log.i("main", "2");
        while (mIsRuning) {
            long start = System.currentTimeMillis();
            draw();
            long end = System.currentTimeMillis();
            if (end - start < 50) {
                try {
                    Thread.sleep(50 - (end - start));
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }

  在run方法我们要注意的是,这里的draw方法绘制所需的事件不足50毫秒的,会让子线程睡眠,也就是说我们每一次进行绘制的操作,至少会花费50毫秒的时间。同时要注意的是:如果这里不睡眠的,程序容易闪退或者卡顿,我经过测试的。
  draw方面里面才是我们真正绘制的操作。

private void draw() {
        mCanvas = mHolder.lockCanvas();
        try {

            if (mCanvas != null) {
                //绘制背景图片
                drawBg();
                //mStartAngle = 0;
                int tempAngle = 360 / mItemCount;
                for(int i = 0; i < mItemCount; i++)
                {
                    //绘制圆块
                    mArcPaint.setColor(mColors[i]);
                    mCanvas.drawArc(mRange, mStartAngle, tempAngle, true, mArcPaint);
                    //绘制文字
                    drawText((int) mStartAngle, tempAngle, strs[i]);
                    //绘制图片
                    drawIcon((int) mStartAngle, tempAngle, mImgsBitmap[i]);
                    mStartAngle += tempAngle;
                }
                mStartAngle += mSpeed;
                if(mIsShouldEnd)
                {
                    mSpeed--;
                }
                if(mSpeed <= 0)
                {
                    mSpeed = 0;
                    mIsShouldEnd = false;
                }
            }
        } catch (Exception e) {
        } finally {
            mHolder.unlockCanvasAndPost(mCanvas);
        }

    }

  绘制背景图片

 private void drawBg() {
        mCanvas.drawColor(0xffffffff);
        mCanvas.drawBitmap(mBgBitmap, null, new Rect(mPadding / 2, mPadding / 2, getMeasuredWidth() - mPadding / 2, getMeasuredWidth() - mPadding / 2), null);
 }

5.绘制文字

private void drawText(int mStartAngle, int tempAngle, String str) {
        Path path = new Path();
        path.addArc(mRange, mStartAngle, tempAngle);
        /**
         * hOffset -- 水平偏移量,沿着圆弧的偏移量
         * vOffset -- 垂直偏移量, 向圆心的偏移量
         */
        int textWidth = (int) mTextPaint.measureText(str);
        int hOffset = (int) (mRadius * Math.PI / mItemCount / 2 - textWidth / 2);
        int vOffset = mRadius / 2 / 6;
        mCanvas.drawTextOnPath(str, path, hOffset,vOffset,mTextPaint);
    }

6.绘制图片

private void drawIcon(int mStartAngle, int tempAngle, Bitmap bitmap) {
        int imgWidth = mRadius / 2 / 4;
        float angle = (float) ((mStartAngle + tempAngle / 2) * (Math.PI / 180));

        int x = (int) (mCenter + mRadius / 2/ 2 * Math.cos(angle));
        int y = (int) (mCenter + mRadius / 2 / 2 * Math.sin(angle));
        Rect rect = new Rect(x - imgWidth / 2, y - imgWidth / 2, x + imgWidth / 2, y + imgWidth / 2);
        mCanvas.drawBitmap(bitmap, null, rect, null);


    }

  在这里我们可以直接为mSpeed设置值,然后它就可以转动起来。同时这里还有一个问题,那就是怎么控制每次转盘转到指定的位置上呢?这里就要用到很多的数学知识了。


public void startPan(int index)
    {
        float angle = 360 * 1.0f / mItemCount;

        float from = 270 - (index + 1) * angle;
        float end = from + angle;

        float targetFrom = 10 * 360 + from;
        float targetEnd = 10 * 360 + end;

        /**
         * v1--
         * (v1 + 0) * (v1 + 1) / 2 = targetFrom
         *v1 * v1 +v1 - 2 * targetFrom = 0;
         *
         *
         */
        float v1 = (float)((-1 + Math.sqrt(1 + 8 * targetFrom)) / 2);
        float v2 = (float) ((-1 + Math.sqrt(1 + 8 * targetEnd) ) /2);

        mSpeed = (float) (v1 + Math.random() * (v2 - v1));

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

推荐阅读更多精彩内容