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)
自定義控件的代碼:
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);
}
}
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>
以上就是本文的全部內(nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持我們。
上一篇:android 震動(dòng)和提示音的實(shí)現(xiàn)代碼
欄 目:Android
下一篇:AccessibilityService實(shí)現(xiàn)微信發(fā)紅包功能
本文標(biāo)題:Android自定義控件實(shí)現(xiàn)時(shí)鐘效果
本文地址:http://www.jygsgssxh.com/a1/Android/8998.html
您可能感興趣的文章
- 01-10Android自定義View之繪制圓形頭像功能
- 01-10Android實(shí)現(xiàn)雙擊返回鍵退出應(yīng)用實(shí)現(xiàn)方法詳解
- 01-10android實(shí)現(xiàn)記住用戶名和密碼以及自動(dòng)登錄
- 01-10android實(shí)現(xiàn)簡單計(jì)算器功能
- 01-10Android 友盟第三方登錄與分享的實(shí)現(xiàn)代碼
- 01-10C++自定義API函數(shù)實(shí)現(xiàn)大數(shù)相乘算法
- 01-10android實(shí)現(xiàn)指紋識(shí)別功能
- 01-10Emoji表情在Android JNI中的兼容性問題詳解
- 01-10Android實(shí)現(xiàn)圓形漸變加載進(jìn)度條
- 01-10android開發(fā)環(huán)境中SDK文件夾下的所需內(nèi)容詳解


閱讀排行
本欄相關(guān)
- 01-10Android自定義View之繪制圓形頭像功能
- 01-10Android實(shí)現(xiàn)雙擊返回鍵退出應(yīng)用實(shí)現(xiàn)方
- 01-10android實(shí)現(xiàn)簡單計(jì)算器功能
- 01-10android實(shí)現(xiàn)記住用戶名和密碼以及自動(dòng)
- 01-10C++自定義API函數(shù)實(shí)現(xiàn)大數(shù)相乘算法
- 01-10Android 友盟第三方登錄與分享的實(shí)現(xiàn)代
- 01-10android實(shí)現(xiàn)指紋識(shí)別功能
- 01-10如何給Flutter界面切換實(shí)現(xiàn)點(diǎn)特效
- 01-10Android實(shí)現(xiàn)圓形漸變加載進(jìn)度條
- 01-10Emoji表情在Android JNI中的兼容性問題詳
隨機(jī)閱讀
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 08-05DEDE織夢data目錄下的sessions文件夾有什
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 04-02jquery與jsp,用jquery
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 08-05dedecms(織夢)副欄目數(shù)量限制代碼修改
- 01-10delphi制作wav文件的方法
- 08-05織夢dedecms什么時(shí)候用欄目交叉功能?
- 01-10C#中split用法實(shí)例總結(jié)


