Android 畫一個太極圖實例代碼
今天練手一下,一起來畫個太極圖吧~
最終效果如下:
最終效果
一般都是先講原理,我就反其道而行,先講實現(xiàn)吧。
1.繼承實現(xiàn)初始化方法
繼承View,實現(xiàn)基本的構(gòu)造函數(shù):
public TestView(Context context) { this(context, null); } public TestView(Context context, AttributeSet attrs) { this(context, attrs, 0); } public TestView(Context context, AttributeSet attrs, int defStyleAttr) { this(context, attrs, defStyleAttr, 0); } @TargetApi(Build.VERSION_CODES.LOLLIPOP) public TestView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) { super(context, attrs, defStyleAttr, defStyleRes); init(); }
在init()方法中,進(jìn)行初始化操作,這里初始化一下畫筆就好。
private Paint mPaint; private void init() { initPaint(); } /** * 初始化畫筆 */ private void initPaint() { mPaint = new Paint(); //創(chuàng)建畫筆對象 mPaint.setColor(Color.BLACK); //設(shè)置畫筆顏色 mPaint.setStyle(Paint.Style.FILL); //設(shè)置畫筆模式為填充 mPaint.setStrokeWidth(10f); //設(shè)置畫筆寬度為10px mPaint.setAntiAlias(true); //設(shè)置抗鋸齒 mPaint.setAlpha(255); //設(shè)置畫筆透明度 }
在onSizeChanged()方法中獲取高寬,便于之后繪制計算。
private int mWidth; private int mHeight; @Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(w, h, oldw, oldh); mWidth = w; mHeight = h; }
創(chuàng)建兩個路徑,一下計算就在這兩個路徑中進(jìn)行。
private Path path0 = new Path(); private Path path1 = new Path();
然后到最關(guān)鍵的onDraw()方法了,這里會分幾步來演示。
1.移動布局到中間
@Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); //移動布局到中間 canvas.translate(mWidth / 2, mHeight / 2); }
ps:為了簡潔,之后的代碼都是在onDraw()中逐層增加的,之后就不寫onDraw()的外出括號了。
2.畫背景黃色
mPaint.setColor(0xffffff00); path0.addRect(-400, -400, 400, 400, Path.Direction.CW); canvas.drawPath(path0, mPaint);
第二步.png
3.畫白色圓背景,即太極圖的白魚部分。
mPaint.setColor(0xffffffff); path0.rewind(); path0.addCircle(0, 0, 200, Path.Direction.CW); canvas.drawPath(path0, mPaint);
4.畫黑色圓背景,即太極圖的黑魚部分,和白魚一樣大小位置,只是把白魚蓋住了,這里就需要用一些boolean運算進(jìn)行繪制了。
//白魚的背景 mPaint.setColor(0xffffffff); path0.rewind(); path0.addCircle(0, 0, 200, Path.Direction.CW); canvas.drawPath(path0, mPaint); //黑魚的背景 mPaint.setColor(0xff000000); path1.addCircle(0, 0, 200, Path.Direction.CW); canvas.drawPath(path0, mPaint);//這一段注意,之后要刪除
第四步.png
5.對黑魚(path1)進(jìn)行boolean計算,把不需要的部分去掉。這里就是要把圓的右半邊消除,這里就需要用到path.op()方法了。
mPaint.setColor(0xffffffff); path0.rewind(); path0.addCircle(0, 0, 200, Path.Direction.CW); canvas.drawPath(path0, mPaint); mPaint.setColor(0xff000000); path1.addCircle(0, 0, 200, Path.Direction.CW); path0.rewind(); path0.addRect(0, -200, 200, 200, Path.Direction.CW); path1.op(path0, Path.Op.DIFFERENCE); canvas.drawPath(path0, mPaint);//這一段注意,之后要刪除
第五步.png
6.這時候我們已經(jīng)把不需要的另一半黑色去掉了,但是黑魚應(yīng)該有個圓的頭,那么我們就拼接一個頭給它。
mPaint.setColor(0xffffffff); path0.rewind(); path0.addCircle(0, 0, 200, Path.Direction.CW); canvas.drawPath(path0, mPaint); mPaint.setColor(0xff000000); path1.addCircle(0, 0, 200, Path.Direction.CW); path0.rewind(); path0.addRect(0, -200, 200, 200, Path.Direction.CW); path1.op(path0, Path.Op.DIFFERENCE); path0.rewind(); path0.addCircle(0, -100, 100, Path.Direction.CW); path1.op(path0, Path.Op.UNION); canvas.drawPath(path1, mPaint);//這一段注意,之后要刪除
第六步.png
7.到這里,我們看到,只需要在繪制一個白魚的頭就可以了,那么也和第五步一樣,使用一個boolean運算把多余的黑色去掉即可。
mPaint.setColor(0xffffffff); path0.rewind(); path0.addCircle(0, 0, 200, Path.Direction.CW); canvas.drawPath(path0, mPaint); mPaint.setColor(0xff000000); path1.addCircle(0, 0, 200, Path.Direction.CW); path0.rewind(); path0.addRect(0, -200, 200, 200, Path.Direction.CW); path1.op(path0, Path.Op.DIFFERENCE); path0.rewind(); path0.addCircle(0, -100, 100, Path.Direction.CW); path1.op(path0, Path.Op.UNION); path0.rewind(); path0.addCircle(0, 100, 100, Path.Direction.CW); path1.op(path0, Path.Op.DIFFERENCE); canvas.drawPath(path1, mPaint);
第七步.png
8.至此,已經(jīng)繪制好了八卦圖的背景了,只需要在繪制魚的眼睛即可。
//畫黑色小圓 path0.rewind(); path0.addCircle(0, 100, 50, Path.Direction.CW); mPaint.setColor(0xff000000); canvas.drawPath(path0, mPaint); //畫白色小圓 path0.rewind(); path0.addCircle(0, -100, 50, Path.Direction.CW); mPaint.setColor(0xffffffff); canvas.drawPath(path0, mPaint);
第八步.png
完成,最后上完整的代碼。代碼寫得有點亂,不過也是練習(xí)而已,哈哈。至于其中的boolean運算什么的,之后在我的自定義View的筆記中在寫吧。
import android.annotation.TargetApi; import android.content.Context; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.Paint; import android.graphics.Path; import android.os.Build; import android.util.AttributeSet; import android.view.View; /** * Created by Whitelaning on 2016/6/28. * Email: whitelaning@qq.com */ public class TestView extends View { private Paint mPaint; private int mWidth; private int mHeight; public TestView(Context context) { this(context, null); } public TestView(Context context, AttributeSet attrs) { this(context, attrs, 0); } public TestView(Context context, AttributeSet attrs, int defStyleAttr) { this(context, attrs, defStyleAttr, 0); } @TargetApi(Build.VERSION_CODES.LOLLIPOP) public TestView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) { super(context, attrs, defStyleAttr, defStyleRes); init(); } private void init() { initPaint(); } /** * 初始化畫筆 */ private void initPaint() { mPaint = new Paint(); //創(chuàng)建畫筆對象 mPaint.setColor(Color.BLACK); //設(shè)置畫筆顏色 mPaint.setStyle(Paint.Style.FILL); //設(shè)置畫筆模式為填充 mPaint.setStrokeWidth(10f); //設(shè)置畫筆寬度為10px mPaint.setAntiAlias(true); //設(shè)置抗鋸齒 mPaint.setAlpha(255); //設(shè)置畫筆透明度 } @Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(w, h, oldw, oldh); mWidth = w; mHeight = h; } private Path path0 = new Path(); private Path path1 = new Path(); @Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); //移動布局到中間 canvas.translate(mWidth / 2, mHeight / 2); //畫大背景顏色 mPaint.setColor(0xffffff00); path0.addRect(-400, -400, 400, 400, Path.Direction.CW); canvas.drawPath(path0, mPaint); mPaint.setColor(0xffffffff); path0.rewind(); path0.addCircle(0, 0, 200, Path.Direction.CW); canvas.drawPath(path0, mPaint); mPaint.setColor(0xff000000); path1.addCircle(0, 0, 200, Path.Direction.CW); path0.rewind(); path0.addRect(0, -200, 200, 200, Path.Direction.CW); path1.op(path0, Path.Op.DIFFERENCE); path0.rewind(); path0.addCircle(0, -100, 100, Path.Direction.CW); path1.op(path0, Path.Op.UNION); path0.rewind(); path0.addCircle(0, 100, 100, Path.Direction.CW); path1.op(path0, Path.Op.DIFFERENCE); canvas.drawPath(path1, mPaint); //畫黑色小圓 path0.rewind(); path0.addCircle(0, 100, 50, Path.Direction.CW); mPaint.setColor(0xff000000); canvas.drawPath(path0, mPaint); //畫白色小圓 path0.rewind(); path0.addCircle(0, -100, 50, Path.Direction.CW); mPaint.setColor(0xffffffff); canvas.drawPath(path0, mPaint); } }
Whitelaning
It's very easy to be different but very difficult to be better
以上就是對Android 實現(xiàn)太極的實例代碼,有興趣朋友可以參考下,謝謝大家對本站的支持!
相關(guān)文章
Android UI控件RatingBar實現(xiàn)自定義星星評分效果
這篇文章主要為大家詳細(xì)介紹了Android UI控件RatingBar實現(xiàn)自定義星星評分效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-02-02Android基于reclyview實現(xiàn)列表回彈動畫效果
這篇文章主要為大家詳細(xì)介紹了Android基于reclyview實現(xiàn)列表回彈動畫效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2022-04-04Android實現(xiàn)后臺開啟服務(wù)默默拍照功能
這篇文章主要為大家詳細(xì)介紹了Android實現(xiàn)后臺開啟服務(wù)默默拍照功能,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-06-06Templates實戰(zhàn)之更優(yōu)雅實現(xiàn)自定義View構(gòu)造方法詳解
本篇文章介紹如何利用Android Studio提供的Live Templates更優(yōu)雅實現(xiàn)自定義View的構(gòu)造方法,說句人話就是:簡化自定義View構(gòu)造參數(shù)模板代碼的編寫,實現(xiàn)自動生成,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09Android View實現(xiàn)圓形進(jìn)度條
這篇文章主要為大家詳細(xì)介紹了Android View實現(xiàn)圓形進(jìn)度條,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-08-08