欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Android編程基于自定義控件實(shí)現(xiàn)時(shí)鐘功能的方法

 更新時(shí)間:2018年03月13日 14:49:01   作者:lizonghuan  
這篇文章主要介紹了Android編程基于自定義控件實(shí)現(xiàn)時(shí)鐘功能的方法,結(jié)合實(shí)例形式詳細(xì)分析了Android自定義控件的定義及時(shí)鐘功能相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下

本文實(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ì)有所幫助。

相關(guān)文章

  • Android開發(fā)之Sqliteopenhelper用法實(shí)例分析

    Android開發(fā)之Sqliteopenhelper用法實(shí)例分析

    這篇文章主要介紹了Android開發(fā)之Sqliteopenhelper用法,實(shí)例分析了SQLiteOpenHelper類操作數(shù)據(jù)庫的相關(guān)技巧,需要的朋友可以參考下
    2015-05-05
  • Android ANR原理分析

    Android ANR原理分析

    ANR即Application Not Responding,顧名思義就是應(yīng)用程序無響應(yīng)。在Android中,一般情況下,四大組件均是工作在主線程中的,Android會(huì)隨時(shí)監(jiān)控應(yīng)用程序的響應(yīng)情況,如果因?yàn)橐恍┖臅r(shí)操作,那么系統(tǒng)就會(huì)顯示ANR對(duì)話框提示用戶對(duì)應(yīng)的應(yīng)用處于無響應(yīng)狀態(tài)
    2021-06-06
  • Android自定義View實(shí)現(xiàn)數(shù)獨(dú)游戲

    Android自定義View實(shí)現(xiàn)數(shù)獨(dú)游戲

    這篇文章主要為大家詳細(xì)介紹了Android自定義View實(shí)現(xiàn)數(shù)獨(dú)游戲,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-12-12
  • item高度不同時(shí)Recyclerview獲取滑動(dòng)距離的方法

    item高度不同時(shí)Recyclerview獲取滑動(dòng)距離的方法

    這篇文章主要介紹了item高度不同時(shí)Recyclerview獲取滑動(dòng)距離的方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-11-11
  • Flutter Widget開發(fā)Shortcuts快捷鍵實(shí)例

    Flutter Widget開發(fā)Shortcuts快捷鍵實(shí)例

    這篇文章主要為大家介紹了Flutter Widget開發(fā)Shortcuts快捷鍵實(shí)例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-12-12
  • kotlin實(shí)戰(zhàn)教程之lambda編程

    kotlin實(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-09
  • Android Oss上傳圖片的使用示例

    Android Oss上傳圖片的使用示例

    本篇文章主要介紹了Android Oss上傳圖片的使用示例,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-03-03
  • Android Studio如何快速導(dǎo)入jar和.so文件

    Android Studio如何快速導(dǎo)入jar和.so文件

    這篇文章主要介紹了Android Studio如何快速導(dǎo)入jar和.so文件的相關(guān)知識(shí),非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下
    2017-12-12
  • Android中Parcel用法詳解

    Android中Parcel用法詳解

    這篇文章主要介紹了Android中Parcel用法,結(jié)合實(shí)例形式較為詳細(xì)的分析了Parcel數(shù)據(jù)容器的原理與使用方法,需要的朋友可以參考下
    2016-06-06
  • Android簡潔的下拉放大刷新效果示例

    Android簡潔的下拉放大刷新效果示例

    本篇文章主要介紹了Android簡潔的下拉放大刷新效果示例,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-10-10

最新評(píng)論