Android自定義控件實(shí)現(xiàn)時(shí)鐘效果
在學(xué)習(xí)安卓群英傳自定義控件章節(jié)的時(shí)候,有一個(gè)例子是繪制時(shí)鐘,在實(shí)現(xiàn)了書(shū)上的例子后就想看這個(gè)時(shí)鐘能不能動(dòng)起來(lái)。
這里選擇延遲一秒發(fā)送消息重繪view來(lái)實(shí)現(xiàn)的動(dòng)畫,對(duì)外提供了開(kāi)啟時(shí)鐘,關(guān)閉時(shí)鐘的方法,當(dāng)activity執(zhí)行onResume方法的時(shí)候,執(zhí)行startClock()方法,當(dāng)移除view或activity執(zhí)行onStop方法的時(shí)候可以執(zhí)行stopClock()方法。
首先根據(jù)view的寬高來(lái)確定圓心的位置,并畫出一個(gè)圓。再通過(guò)view高度的一半減去圓的半徑,確定刻度的起始位置,選擇刻度的長(zhǎng)度并繪制出來(lái)。然后再刻度下方繪制出數(shù)字。最終將畫布進(jìn)行旋轉(zhuǎn),時(shí)鐘總共有60個(gè)刻度,循環(huán)旋轉(zhuǎn),每次旋轉(zhuǎn)6度即可。
最后是繪制指針,通過(guò)計(jì)算算出指針對(duì)應(yīng)每個(gè)刻度的X,Y坐標(biāo)并繪制直線。
代碼實(shí)現(xiàn)
自定義控件的代碼:
public class ClockView extends View{ private Paint circlePaint,dialPaint,numberPaint; //view 的寬高 private float mWidth,mHeight; //圓的半徑 private float circleRadius; //圓心X,Y坐標(biāo) private float circleX,circleY; private int second,minute; private double hour; private Handler handler = new Handler(Looper.getMainLooper()){ @Override public void handleMessage(Message msg) { super.handleMessage(msg); if(msg.what==0){ invalidate(); } } }; public ClockView(Context context, AttributeSet attrs) { super(context, attrs); initPaint(); } private void initPaint(){ //刻盤圓,小時(shí)刻度,時(shí)針和分針的畫筆 circlePaint = new Paint(Paint.ANTI_ALIAS_FLAG); circlePaint.setColor(Color.BLACK); circlePaint.setStyle(Paint.Style.STROKE); circlePaint.setStrokeWidth(10); //分鐘刻度的畫筆 dialPaint = new Paint(Paint.ANTI_ALIAS_FLAG); dialPaint.setColor(Color.BLACK); dialPaint.setStrokeWidth(5); //數(shù)字的畫筆 numberPaint = new Paint(Paint.ANTI_ALIAS_FLAG); numberPaint.setColor(Color.BLACK); numberPaint.setStrokeWidth(5); numberPaint.setTextSize(30); } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(widthMeasureSpec, heightMeasureSpec); mWidth = getMeasuredWidth(); mHeight = getMeasuredHeight(); if(mWidth<mHeight){ //圓的半徑為view的寬度的一半再減9,防止貼邊 circleRadius = mWidth/2-9; circleX = mWidth/2; circleY = mHeight/2; } else{ circleRadius = mHeight/2-9; circleX = mWidth/2; circleY = mHeight/2; } } @Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); setTimes(); drawCirclePoint(canvas); drawCircle(canvas); drawDial(canvas); drawPointer(canvas); } /**圓心 * @param canvas */ private void drawCirclePoint(Canvas canvas){ canvas.drawCircle(circleX,circleY,5,circlePaint); } private void drawCircle(Canvas canvas){ canvas.drawCircle(circleX,circleY,circleRadius,circlePaint); } /**畫刻度及時(shí)間 * @param canvas */ private void drawDial(Canvas canvas){ //時(shí)鐘用長(zhǎng)一點(diǎn)的刻度,畫筆用畫圓的畫筆 Point hourStartPoint = new Point(circleX,circleY-circleRadius); Point hourEndPoint = new Point(circleX,circleY-circleRadius+40); //分鐘的刻度要稍微短一些,畫筆用畫圓的畫筆 Point startPoint2 = new Point(circleX,circleY-circleRadius); Point endPoint2 = new Point(circleX,circleY-circleRadius+10); //開(kāi)始畫刻度和數(shù)字,總共60個(gè)刻度,12個(gè)時(shí)鐘刻度,被5整除畫一個(gè)時(shí)鐘刻度,被其余的為分針刻度 String clockNumber; for(int i=0;i<60;i++){ if(i%5==0){ if(i==0){ clockNumber = "12"; } else{ clockNumber = String.valueOf(i/5); } //時(shí)針刻度 canvas.drawLine(hourStartPoint.getX(),hourStartPoint.getY(),hourEndPoint.getX(),hourEndPoint.getY(),circlePaint); //畫數(shù)字,需在時(shí)針刻度末端加30 canvas.drawText(clockNumber,circleX-numberPaint.measureText(clockNumber)/2,hourEndPoint.getY()+30,numberPaint); } else{ //畫分針刻度 canvas.drawLine(startPoint2.getX(),startPoint2.getY(),endPoint2.getX(),endPoint2.getY(),circlePaint); } //畫布旋轉(zhuǎn)6度 canvas.rotate(360/60,circleX,circleY); } } /**畫指針 * X點(diǎn)坐標(biāo) cos(弧度)*r * Y點(diǎn)坐標(biāo) sin(弧度)*r * toRadians將角度轉(zhuǎn)成弧度 * 安卓坐標(biāo)系與數(shù)學(xué)坐標(biāo)系不同的地方是X軸是相反的,所以為了調(diào)整方向,需要將角度+270度 * @param canvas */ private void drawPointer(Canvas canvas){ canvas.translate(circleX,circleY); float hourX = (float) Math.cos(Math.toRadians(hour*30+270))*circleRadius*0.5f; float hourY = (float) Math.sin(Math.toRadians(hour*30+270))*circleRadius*0.5f; float minuteX = (float) Math.cos(Math.toRadians(minute*6+270))*circleRadius*0.8f; float minuteY = (float) Math.sin(Math.toRadians(minute*6+270))*circleRadius*0.8f; float secondX = (float) Math.cos(Math.toRadians(second*6+270))*circleRadius*0.8f; float secondY = (float) Math.sin(Math.toRadians(second*6+270))*circleRadius*0.8f; canvas.drawLine(0,0,hourX,hourY,circlePaint); canvas.drawLine(0,0,minuteX,minuteY,circlePaint); canvas.drawLine(0,0,secondX,secondY,dialPaint); //一秒重繪一次 handler.sendEmptyMessageDelayed(0,1000); } public void startClock(){ setTimes(); invalidate(); } private void setTimes(){ Date date = new Date(); Calendar calendar = Calendar.getInstance(); calendar.setTime(date); second = getTimes(date,Calendar.SECOND); minute = getTimes(date,Calendar.MINUTE); hour = getTimes(date,Calendar.HOUR)+minute/12*0.2; } private int getTimes(Date date,int calendarField){ Calendar calendar = Calendar.getInstance(); calendar.setTime(date); return calendar.get(calendarField); } public void stopClock(){ handler.removeMessages(0); } }
public class Point {
private float x;
private float y;
public Point(float x, float y) { this.x = x; this.y = y; } public float getX() { return x; } public void setX(float x) { this.x = x; } public float getY() { return y; } public void setY(float y) { this.y = y; }
Acitivity:
public class ClockActivity extends Activity{ private ClockView clockView; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.clock_layout); clockView = (ClockView) findViewById(R.id.clock); } @Override protected void onResume() { super.onResume(); clockView.startClock(); } @Override protected void onStop() { super.onStop(); clockView.stopClock(); } }
xml布局:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="match_parent" android:gravity="center" android:layout_height="match_parent"> <com.example.customview.view.ClockView android:layout_width="match_parent" android:id="@+id/clock" android:layout_height="match_parent" /> </LinearLayout>
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android編程之菜單Menu的創(chuàng)建方法示例
這篇文章主要介紹了Android編程之菜單Menu的創(chuàng)建方法,結(jié)合實(shí)例形式分析了Android菜單Menu的布局、響應(yīng)及功能實(shí)現(xiàn)技巧,需要的朋友可以參考下2017-08-08Android判斷后臺(tái)服務(wù)是否開(kāi)啟的兩種方法實(shí)例詳解
這篇文章主要介紹了Android判斷后臺(tái)服務(wù)是否開(kāi)啟的方法的相關(guān)資料,這里提供了兩種方法及實(shí)例,需要的朋友可以參考下2017-07-07Android checkbox的listView(多選,全選,反選)具體實(shí)現(xiàn)方法
由于listview的一些特性,剛開(kāi)始寫這種需求的功能的時(shí)候都會(huì)碰到一些問(wèn)題,重點(diǎn)就是存儲(chǔ)每個(gè)checkbox的狀態(tài)值,在這里分享出了完美解決方法:2013-06-06Android開(kāi)發(fā)中Intent用法總結(jié)
這篇文章主要介紹了Android開(kāi)發(fā)中Intent用法,總結(jié)分析了Android中Intent的功能、使用方法與相關(guān)注意事項(xiàng),需要的朋友可以參考下2016-10-10Android SDK Manager無(wú)法更新問(wèn)題解決辦法
這篇文章主要介紹了Android SDK Manager無(wú)法更新問(wèn)題解決辦法的相關(guān)資料,需要的朋友可以參考下2017-04-04Android ScrollView顯示到底部或任意位置實(shí)現(xiàn)代碼
這篇文章主要介紹了 Android ScrollView顯示到底部或任意位置實(shí)現(xiàn)代碼的相關(guān)資料,需要的朋友可以參考下2017-02-02Android編程實(shí)現(xiàn)從字符串中查找電話號(hào)碼的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)從字符串中查找電話號(hào)碼的方法,涉及Android針對(duì)字符串的匹配與查找相關(guān)技巧,需要的朋友可以參考下2016-03-03Cocos2d-x 3.0中集成社交分享ShareSDK的詳細(xì)步驟和常見(jiàn)問(wèn)題解決
這篇文章主要介紹了Cocos2d-x 3.0中集成社交分享ShareSDK的詳細(xì)步驟和常見(jiàn)問(wèn)題的解決方法以及需要注意的問(wèn)題,需要的朋友可以參考下2014-04-04Android Studio添加第三方庫(kù)的注意事項(xiàng)
這篇文章給大家介紹的是Android Studio添加第三方庫(kù)遇到的一些坑,以及對(duì)應(yīng)的解決辦法,有需要的可以參考借鑒。2016-09-09