Android編程基于自定義控件實(shí)現(xiàn)時(shí)鐘功能的方法
本文實(shí)例講述了Android編程基于自定義控件實(shí)現(xiàn)時(shí)鐘功能的方法。分享給大家供大家參考,具體如下:
在學(xué)習(xí)安卓群英傳自定義控件章節(jié)的時(shí)候,有一個(gè)例子是繪制時(shí)鐘,在實(shí)現(xiàn)了書上的例子后就想看這個(gè)時(shí)鐘能不能動(dòng)起來。
這里選擇延遲一秒發(fā)送消息重繪view來實(shí)現(xiàn)的動(dòng)畫,對(duì)外提供了開啟時(shí)鐘,關(guān)閉時(shí)鐘的方法,當(dāng)activity執(zhí)行onResume
方法的時(shí)候,執(zhí)行startClock()
方法,當(dāng)移除view或activity執(zhí)行onStop
方法的時(shí)候可以執(zhí)行stopClock()
方法。
首先根據(jù)view的寬高來確定圓心的位置,并畫出一個(gè)圓。再通過view高度的一半減去圓的半徑,確定刻度的起始位置,選擇刻度的長度并繪制出來。然后再刻度下方繪制出數(shù)字。最終將畫布進(jìn)行旋轉(zhuǎn),時(shí)鐘總共有60個(gè)刻度,循環(huán)旋轉(zhuǎn),每次旋轉(zhuǎn)6度即可。
最后是繪制指針,通過計(jì)算算出指針對(duì)應(yīng)每個(gè)刻度的X,Y坐標(biāo)并繪制直線。
代碼實(shí)現(xiàn)
自定義控件的代碼(ClockView.java):
package com.example.clock; import java.util.Calendar; import java.util.Date; import android.content.Context; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.Paint; import android.os.Handler; import android.os.Looper; import android.os.Message; import android.util.AttributeSet; import android.view.View; 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í)鐘用長一點(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); // 開始畫刻度和數(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); } }
Point.java:
package com.example.clock; 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(ClockActivity.java):
package com.example.clock; import android.app.Activity; import android.os.Bundle; public class ClockActivity extends Activity { /** Called when the activity is first created. */ private ClockView clockView; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); clockView = (ClockView) findViewById(R.id.clock); } @Override protected void onResume() { super.onResume(); clockView.startClock(); } @Override protected void onStop() { super.onStop(); clockView.stopClock(); } }
xml布局(main.xml):
<?xml version="1.0" encoding="utf-8"?> <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>
更多關(guān)于Android相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《Android開發(fā)入門與進(jìn)階教程》、《Android調(diào)試技巧與常見問題解決方法匯總》、《Android基本組件用法總結(jié)》、《Android視圖View技巧總結(jié)》、《Android布局layout技巧總結(jié)》及《Android控件用法總結(jié)》
希望本文所述對(duì)大家Android程序設(shè)計(jì)有所幫助。
- Android獲取設(shè)備CPU核數(shù)、時(shí)鐘頻率以及內(nèi)存大小的方法
- Android多功能時(shí)鐘開發(fā)案例(實(shí)戰(zhàn)篇)
- android實(shí)現(xiàn)widget時(shí)鐘示例分享
- Android 仿日歷翻頁、仿htc時(shí)鐘翻頁、數(shù)字翻頁切換效果
- Android多功能時(shí)鐘開發(fā)案例(基礎(chǔ)篇)
- android高仿小米時(shí)鐘(使用Camera和Matrix實(shí)現(xiàn)3D效果)
- Android實(shí)現(xiàn)簡單時(shí)鐘View的方法
- Android自定義動(dòng)態(tài)壁紙開發(fā)(時(shí)鐘)
- Android仿小米時(shí)鐘效果
- Android自定義View實(shí)現(xiàn)時(shí)鐘功能
相關(guān)文章
Android開發(fā)之Sqliteopenhelper用法實(shí)例分析
這篇文章主要介紹了Android開發(fā)之Sqliteopenhelper用法,實(shí)例分析了SQLiteOpenHelper類操作數(shù)據(jù)庫的相關(guān)技巧,需要的朋友可以參考下2015-05-05Android自定義View實(shí)現(xiàn)數(shù)獨(dú)游戲
這篇文章主要為大家詳細(xì)介紹了Android自定義View實(shí)現(xiàn)數(shù)獨(dú)游戲,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12item高度不同時(shí)Recyclerview獲取滑動(dòng)距離的方法
這篇文章主要介紹了item高度不同時(shí)Recyclerview獲取滑動(dòng)距離的方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-11-11Flutter Widget開發(fā)Shortcuts快捷鍵實(shí)例
這篇文章主要為大家介紹了Flutter Widget開發(fā)Shortcuts快捷鍵實(shí)例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12kotlin實(shí)戰(zhàn)教程之lambda編程
這篇文章主要給大家介紹了關(guān)于kotlin實(shí)戰(zhàn)教程之lambda編程的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用kotlin具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-09-09Android Studio如何快速導(dǎo)入jar和.so文件
這篇文章主要介紹了Android Studio如何快速導(dǎo)入jar和.so文件的相關(guān)知識(shí),非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2017-12-12