Android自定义控件实现时钟效果

在学习安卓群英传自定义控件章节的时候,有一个例子是绘制时钟,在实现了书上的例子后就想看这个时钟能不能动起来。

这里选择延迟一秒发送消息重绘view来实现的动画,对外提供了开启时钟,关闭时钟的方法,当activity执行onResume方法的时候,执行startClock()方法,当移除view或activity执行onStop方法的时候可以执行stopClock()方法。

首先根据view的宽高来确定圆心的位置,并画出一个圆。再通过view高度的一半减去圆的半径,确定刻度的起始位置,选择刻度的长度并绘制出来。然后再刻度下方绘制出数字。最终将画布进行旋转,时钟总共有60个刻度,循环旋转,每次旋转6度即可。

最后是绘制指针,通过计算算出指针对应每个刻度的X,Y坐标并绘制直线。

代码实现

自定义控件的代码:

public class ClockView extends View{
 private Paint circlePaint,dialPaint,numberPaint;
 //view 的宽高
 private float mWidth,mHeight;
 //圆的半径
 private float circleRadius;
 //圆心X,Y坐标
 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(){
  //刻盘圆,小时刻度,时针和分针的画笔
  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);

  //数字的画笔
  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);
 }

 /**画刻度及时间
  * @param canvas
  */
 private void drawDial(Canvas canvas){
  //时钟用长一点的刻度,画笔用画圆的画笔
  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);
  //开始画刻度和数字,总共60个刻度,12个时钟刻度,被5整除画一个时钟刻度,被其余的为分针刻度
  String clockNumber;
  for(int i=0;i<60;i++){
   if(i%5==0){
    if(i==0){
     clockNumber = "12";
    } else{
     clockNumber = String.valueOf(i/5);
    }
    //时针刻度
    canvas.drawLine(hourStartPoint.getX(),hourStartPoint.getY(),hourEndPoint.getX(),hourEndPoint.getY(),circlePaint);
    //画数字,需在时针刻度末端加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);
   }
   //画布旋转6度
   canvas.rotate(360/60,circleX,circleY);
  }
 }

 /**画指针
  * X点坐标 cos(弧度)*r
  * Y点坐标 sin(弧度)*r
  * toRadians将角度转成弧度
  * 安卓坐标系与数学坐标系不同的地方是X轴是相反的,所以为了调整方向,需要将角度+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>

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android获取设备CPU核数、时钟频率以及内存大小的方法

    本文实例讲述了Android获取设备CPU核数.时钟频率以及内存大小的方法.分享给大家供大家参考,具体如下: 因项目需要,分析了一下 Facebook 的开源项目 - Device Year Class. Device Year Class 的主要功能是根据 CPU核数.时钟频率 以及 内存大小 对设备进行分级.代码很简单,只包含两个类: DeviceInfo -> 获取设备参数, YearClass -> 根据参数进行分级. 下表是 Facebook 公司提供的分级标准,其中 Year 栏表

  • Android打造属于自己的时间钟表

    1.概述 本文主要讲解的是如何自定义一个时间钟表,通过简单的练习可以简单学习Android当中自定义view的一些常用绘图技巧,优化android绘图操作.言归正传,首先看下我们需要实现的效果: 当我们看到这个效果的时候脑子里应该有一定的思路了,我们应该把它分解成以下几个步骤: 1.仪表盘(圆) 2.刻度线(长 中 短) 3.刻度值(1-12) 4.指针(时  分  秒) 5.移动指针,计算指针位置 现在我们已经很清楚自己的思路了,那么我们一个一个来. 第一步:1.自定义View的属性,首先在r

  • Android实现时钟特效

    本文实例为大家分享了Android实现时钟特效的具体代码,供大家参考,具体内容如下 效果展示: 功能介绍: 如果您想换一张背景图,可以点击左下角按按钮切换背景图片. 如果您不想看见右上方的日期,可以点击它,他就会随即隐藏.如果你想 再次查看,请点击左下角切换壁纸按钮他就会被再次展示. Demo 下载地址: 点击此处跳转:AndroidClockDemo 部分代码展示: mainActivity部分: 实现切换,获得事件并显示等功能. public class MainActivity exten

  • android高仿小米时钟(使用Camera和Matrix实现3D效果)

    继续练习自定义View..毕竟熟才能生巧.一直觉得小米的时钟很精美,那这次就搞它~这次除了练习自定义View,还涉及到使用Camera和Matrix实现3D效果. 一个这样的效果,在绘制的时候最好选择一个方向一步一步的绘制,这里我选择由外到内.由深到浅的方向来绘制,代码步骤如下: 1.首先老一套~新建attrs.xml文件,编写自定义属性如时钟背景色.亮色(用于分针.秒针.渐变终止色).暗色(圆弧.刻度线.时针.渐变起始色),新建MiClockView继承View,重写构造方法,获取自定义属性值

  • android实现widget时钟示例分享

    一.在 AndroidManifest.xml文件中配置Widgets: 复制代码 代码如下: <manifest xmlns:android="http://schemas.android.com/apk/res/android"    package="com.example.widget"    android:versionCode="1"    android:versionName="1.0" >   

  • Android多功能时钟开发案例(基础篇)

    本文我们进入Android多功能时钟开发实战学习,具体的效果可以参考手机上的时钟,内容如下 首先我们来看一看布局文件layout_main.xml 整个布局: <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/container" android:layout_width="match_parent" androi

  • Android实现简单时钟View的方法

    通过Canvas的平移与旋转简化绘图逻辑是一个非常有用的技巧,下面的时钟view就是利用这个方法完成的,省去了使用三角函数计算坐标的麻烦. package com.example.swt369.simpleclock; import android.content.Context; import android.graphics.Canvas; import android.graphics.Paint; import android.support.annotation.Nullable; i

  • Android 仿日历翻页、仿htc时钟翻页、数字翻页切换效果

    废话不多说,效果图: 自定义控件找自网络,使用相对简单,具体还没有来得及深入研究,只是先用笨方法大概实现了想要的效果,后续有空会仔细研究再更新文章, 本demo切换方法是用的笨方法,也就是由新数字和旧数字相比较来切换数字变换的,大致使用方法如下: //获取输入框中的数字 int newNumber = Integer.parseInt(etInput.getText().toString()); //获取个.十.百位数字 int nbai = newNumber / 100; int nshi

  • Android自定义钟表特效

    最近该忙的都忙完了,自己自定义一直是个弱项,也一直想整个钟表玩玩,网上看了一圈,学习了不少,下面自己做做自定义 首先,制作钟表第一步,肯定是画个圆吧,这是最直接的思维了! 先创建自己的自定义类,继承View ,重写构造方法,在第一个和第二个构造中初始化画笔,设置颜色等 第一个构造器类似于咱们直接New对象,第二个就是在xml文件引用时用到的 public class Watch extends View { private Paint mPaint; private Context contex

  • Android多功能时钟开发案例(实战篇)

    上一篇为大家介绍的是Android多功能时钟开发基础内容,大家可以回顾一下,Android多功能时钟开发案例(基础篇) 接下来进入实战,快点来学习吧. 一.时钟 在布局文件中我们看到,界面上只有一个TextView,这个TextView的作用就是显示一个系统的当前时间,同时这个时间还是一秒一秒跳的,要实现一秒一秒的跳就需要我们每隔一秒就要刷新一下,同时我们这里还考虑了切换到另一个Tab的时候,这个时间就不跳动了,这样就会减少这个对系统的占用,考虑到了这点我们在这里用到了Handler,通过han

随机推荐