自定義SurfaceView實(shí)現(xiàn)跳動(dòng)的泡泡

先看下效果圖

QQ20170410-214631-HD.gif

不記得從哪里看到的這樣的效果了,今天想起來(lái)了,覺得挺好玩就隨便實(shí)現(xiàn)了一下权埠。

大致的步驟如下:
1、繼承SurfaceView 并實(shí)現(xiàn)SurfaceHolder.Callback,和Runnable接口煎谍。至于為什么使用SurfaceView而不用View攘蔽,大家可以讀一下下面這個(gè)文章:http://blog.csdn.net/tianfeng701/article/details/7601627

2、畫四個(gè)球球(很簡(jiǎn)單吧~)透明度隨喜好調(diào)一下呐粘。

3满俗、四個(gè)球,那么以圓心為起點(diǎn)事哭,定義四個(gè)軌跡線漫雷。(軌跡線方向隨意)

4、通過子線程不斷移動(dòng)小球(移動(dòng)球球圓心)

代碼是最好的老師鳍咱,我就不啰嗦了降盹。

public class BreatheView extends SurfaceView implements SurfaceHolder.Callback, Runnable {

    /**
     * 是否處于繪制狀態(tài)
     */
    private boolean mIsDrawing;
    /**
     * 幫助類
     */
    private SurfaceHolder mHolder;
    /**
     * 畫布
     */
    private Canvas mCanvas;
    /**
     * 畫筆
     */
    private Paint mPaint;
    private Paint mPaintTwo;
    private Paint mPaintThree;
    private Paint mPaintFour;
    /**
     * 寬高
     */
    private float width;
    private float height;

    private final int ALPHA = 50;
    private final int SLEEP_TIME = 100;
    /**
     * 坐標(biāo)
     */
    PointF oneStartPoint;
    PointF twoStartPoint;
    PointF threeStartPoint;
    PointF fourStartPoint;
    /**
     * 判斷是否繪制
     */
    boolean isDraw = true;
    boolean isDrawTwo = true;
    boolean isDrawThree = true;
    boolean isDrawFour = true;

    public BreatheView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        initView();
    }

    public BreatheView(Context context, AttributeSet attrs) {
        super(context, attrs);
        initView();
    }

    public BreatheView(Context context) {
        super(context);
        initView();
    }

    private void initView() {
        mHolder = getHolder();
        mHolder.addCallback(this);
        setFocusable(true);
        setFocusableInTouchMode(true);
        this.setKeepScreenOn(true);

        mPaint = new Paint();
        mPaint.setAntiAlias(true);
        mPaint.setColor(Color.CYAN);
        mPaint.setStyle(Paint.Style.FILL);
        mPaint.setAlpha(ALPHA);

        mPaintTwo = new Paint();
        mPaintTwo.setAntiAlias(true);
        mPaintTwo.setColor(Color.BLUE);
        mPaintTwo.setStyle(Paint.Style.FILL);
        mPaintTwo.setAlpha(ALPHA);

        mPaintThree = new Paint(mPaint);
        mPaintThree.setAntiAlias(true);
        mPaintThree.setColor(Color.RED);
        mPaintThree.setStyle(Paint.Style.FILL);
        mPaintThree.setAlpha(ALPHA);

        mPaintFour = new Paint();
        mPaintFour.setAntiAlias(true);
        mPaintFour.setColor(Color.GREEN);
        mPaintFour.setStyle(Paint.Style.FILL);
        mPaintFour.setAlpha(ALPHA);

        oneStartPoint = new PointF(300, 220);
        twoStartPoint = new PointF(20, 250);
        threeStartPoint = new PointF(500, 90);
        fourStartPoint = new PointF(0, 0);
    }

    private Canvas getCanvas(){
        return mCanvas;
    }

    private void setCanvas(Canvas aCanvas){
        this.mCanvas = aCanvas;
    }

    @Override
    public void run() {
        long start = System.currentTimeMillis();
        while (mIsDrawing) {
            try {
                Thread.sleep(SLEEP_TIME);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            draw();
        }
        long end = System.currentTimeMillis();
        if (end - start < SLEEP_TIME) {
            try {
                Thread.sleep(SLEEP_TIME - (end - start));
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }

    @Override
    public void surfaceChanged(SurfaceHolder arg0, int arg1, int arg2, int arg3) {
    }

    @Override
    public void surfaceCreated(SurfaceHolder arg0) {
        mIsDrawing = true;
        new Thread(this).start();
    }

    @Override
    public void surfaceDestroyed(SurfaceHolder arg0) {
        mIsDrawing = false;
    }

    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        super.onSizeChanged(w, h, oldw, oldh);
        width = w;
        height = h;
    }

    private synchronized void draw() {
        try {
            lockCanvas();
            drawOne();
            drawTwo();
            drawThree();
            drawFour();
            unlockCanvas();
        } catch (Exception e) {
        }
    }

    private void drawOne() {
        getCanvas().drawColor(Color.WHITE);
        if (oneStartPoint.x == 300 || oneStartPoint.y == 220) {
            isDraw = true;
        } else if (oneStartPoint.x == 330 || oneStartPoint.y == 250) {
            isDraw = false;
        }
        if (isDraw) {
            getCanvas().drawCircle(oneStartPoint.x++, oneStartPoint.y++, 180, mPaint);
        } else {
            getCanvas().drawCircle(oneStartPoint.x--, oneStartPoint.y--, 180, mPaint);
        }
    }

    private void lockCanvas() {
        setCanvas(mHolder.lockCanvas());
    }

    private void unlockCanvas() {
        if (getCanvas() != null) {
            mHolder.unlockCanvasAndPost(getCanvas());
        }
    }

    private void drawFour() {
        //第四個(gè)泡泡
        if (fourStartPoint.x == 0 || fourStartPoint.y == 0) {
            isDrawFour = true;
        } else if (fourStartPoint.x == 20 || fourStartPoint.y == 20) {
            isDrawFour = false;
        }
        if (isDrawFour) {
            getCanvas().drawCircle(fourStartPoint.x++, fourStartPoint.y++, 220, mPaintFour);
        } else {
            getCanvas().drawCircle(fourStartPoint.x--, fourStartPoint.y--, 220, mPaintFour);
        }
    }

    private void drawThree() {
        //第三個(gè)泡泡
        if (threeStartPoint.x == 500 || threeStartPoint.y == 200) {
            isDrawThree = true;
        } else if (threeStartPoint.x == 520 || threeStartPoint.y == 520) {
            isDrawThree = false;
        }
        if (isDrawThree) {
            getCanvas().drawCircle(threeStartPoint.x++, threeStartPoint.y++, 220, mPaintThree);
        } else {
            getCanvas().drawCircle(threeStartPoint.x--, threeStartPoint.y--, 220, mPaintThree);
        }
    }

    private void drawTwo() {
        //第二個(gè)泡泡
        if (twoStartPoint.x == 20 || twoStartPoint.y == 250) {
            isDrawTwo = true;
        } else if (twoStartPoint.x == 35 || twoStartPoint.y == 265) {
            isDrawTwo = false;
        }
        if (isDrawTwo) {
            getCanvas().drawCircle(twoStartPoint.x++, twoStartPoint.y++, 240, mPaintTwo);
        } else {
            getCanvas().drawCircle(twoStartPoint.x--, twoStartPoint.y--, 240, mPaintTwo);
        }
    }
}

github地址:https://github.com/519401502/-SurfaceView-

筆者能力有限,不足之處歡迎指出谤辜!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末蓄坏,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子丑念,更是在濱河造成了極大的恐慌涡戳,老刑警劉巖,帶你破解...
    沈念sama閱讀 223,126評(píng)論 6 520
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件脯倚,死亡現(xiàn)場(chǎng)離奇詭異渔彰,居然都是意外死亡鸟顺,警方通過查閱死者的電腦和手機(jī)纠屋,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,421評(píng)論 3 400
  • 文/潘曉璐 我一進(jìn)店門疲吸,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)指蚁,“玉大人憎妙,你說(shuō)我怎么就攤上這事吮螺∠赐遥” “怎么了斩跌?”我有些...
    開封第一講書人閱讀 169,941評(píng)論 0 366
  • 文/不壞的土叔 我叫張陵尊残,是天一觀的道長(zhǎng)炒瘸。 經(jīng)常有香客問我,道長(zhǎng)寝衫,這世上最難降的妖魔是什么顷扩? 我笑而不...
    開封第一講書人閱讀 60,294評(píng)論 1 300
  • 正文 為了忘掉前任,我火速辦了婚禮慰毅,結(jié)果婚禮上隘截,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好技俐,可當(dāng)我...
    茶點(diǎn)故事閱讀 69,295評(píng)論 6 398
  • 文/花漫 我一把揭開白布乘陪。 她就那樣靜靜地躺著,像睡著了一般雕擂。 火紅的嫁衣襯著肌膚如雪啡邑。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,874評(píng)論 1 314
  • 那天井赌,我揣著相機(jī)與錄音谤逼,去河邊找鬼。 笑死仇穗,一個(gè)胖子當(dāng)著我的面吹牛流部,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播纹坐,決...
    沈念sama閱讀 41,285評(píng)論 3 424
  • 文/蒼蘭香墨 我猛地睜開眼枝冀,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了耘子?” 一聲冷哼從身側(cè)響起果漾,我...
    開封第一講書人閱讀 40,249評(píng)論 0 277
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎谷誓,沒想到半個(gè)月后绒障,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,760評(píng)論 1 321
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡捍歪,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,840評(píng)論 3 343
  • 正文 我和宋清朗相戀三年户辱,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片糙臼。...
    茶點(diǎn)故事閱讀 40,973評(píng)論 1 354
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡庐镐,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出弓摘,到底是詐尸還是另有隱情焚鹊,我是刑警寧澤痕届,帶...
    沈念sama閱讀 36,631評(píng)論 5 351
  • 正文 年R本政府宣布韧献,位于F島的核電站,受9級(jí)特大地震影響研叫,放射性物質(zhì)發(fā)生泄漏锤窑。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,315評(píng)論 3 336
  • 文/蒙蒙 一嚷炉、第九天 我趴在偏房一處隱蔽的房頂上張望渊啰。 院中可真熱鬧,春花似錦、人聲如沸绘证。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,797評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)嚷那。三九已至胞枕,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間魏宽,已是汗流浹背腐泻。 一陣腳步聲響...
    開封第一講書人閱讀 33,926評(píng)論 1 275
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留队询,地道東北人派桩。 一個(gè)月前我還...
    沈念sama閱讀 49,431評(píng)論 3 379
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像蚌斩,于是被迫代替她去往敵國(guó)和親铆惑。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,982評(píng)論 2 361

推薦閱讀更多精彩內(nèi)容