注释已经比较详细了,废话就不多说了。贴代码了

时针分针秒钟都做上去了,采用的方法也很简单,仔细看一会就能看懂

自定义View类

package com.xh.mytime;

import java.util.Calendar;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.Rect;
import android.os.Handler;
import android.util.AttributeSet;
import android.view.View;

public class MyTimeTest extends View implements Runnable
{
private int srcId;
private int fensize;
private int shisize;
private int miaosize;
private Bitmap bitmap;
private float guimo;
private float wzhong;
private float hzhong;

private Handler handler = new Handler();

/*
* */
public MyTimeTest(Context context, AttributeSet attrs)
{
super(context, attrs);
// 读取相应的属性值
//在布局文件中定义了这样的一个属性,取得它的属性值,该属性值指向一张图片,通过bitmapFactory获得由它制定的图片bitmap
srcId = attrs.getAttributeResourceValue(null,
"Src", 0);
if (srcId > 0)
bitmap = BitmapFactory.decodeResource(getResources(),
srcId);

guimo = attrs.getAttributeFloatValue(null, "scale", 1);
//获取xml文件中的图片宽度中点,如果没有设置则说明图片是对称的。取默认返回参数
wzhong = attrs.getAttributeFloatValue(null,
"wzhong", bitmap.getWidth() / 2);
//获取xml文件中的图片高度中点,如果没有设置则说明图片是对称的。取默认返回参数
hzhong = attrs.getAttributeFloatValue(null,
"hzhong", bitmap.getHeight() / 2);
//获取到时针和分针的大小
fensize = (int) (attrs.getAttributeIntValue(null,
"fensize", 0));
shisize = (int) (attrs.getAttributeIntValue(null, "shisize",
0));
miaosize = (int)(attrs.getAttributeIntValue(null, "miaosize", 0));


}
//刷新,保证能接受到每一次view的变化.
@Override
public void run()
{
// 重新绘制View
invalidate();

handler.postDelayed(this, 1000);
}

//如果不覆盖此方法,默认的会将view铺满整个屏幕
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec)
{
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
// 根据图像的实际大小等比例设置View的大小
setMeasuredDimension((int) (bitmap.getWidth() * guimo), (int) (bitmap
.getHeight() * guimo));
}

@Override
protected void onDraw(Canvas canvas)
{
super.onDraw(canvas);
//笔
Paint paint = new Paint();
//矩形
Rect src = new Rect();
Rect target = new Rect();

src.left = 0;
src.top = 0;
src.right = bitmap.getWidth();
src.bottom = bitmap.getHeight();

target.left = 0;
target.top = 0;
target.bottom = (int) (src.bottom * guimo);
target.right = (int) (src.right * guimo);
// 画表盘图像
/*
* dai
* */
canvas.drawBitmap(bitmap, src, target, paint);
// 计算表盘中心点的横纵坐标
float centerX = bitmap.getWidth() * guimo * wzhong;
float centerY = bitmap.getHeight() * guimo * hzhong;
// 表表盘中心点画一个半径为5的实心圆圈
canvas.drawCircle(centerX, centerY, 5, paint);




Calendar calendar = Calendar.getInstance();

int currentMinute = calendar.get(Calendar.MINUTE);
int currentHour = calendar.get(Calendar.HOUR);
int currentsec = calendar.get(Calendar.SECOND);
// 计算分针和时间的角度

double hourRadian = Math.toRadians(
(360 - ((currentHour * 30) - 90))% 360 - (30 * currentMinute / 60)
);

double minuteRadian = Math.toRadians(
(360 - ((currentMinute * 6) - 90)) % 360
);
double secRadian = Math.toRadians(
(360 - ((currentsec * 6) - 90))% 360
);
// 在表盘上画分针

paint.setStrokeWidth(6);
// 在表盘上画分针
canvas.drawLine(centerX, centerY, (int) (centerX + shisize
* Math.cos(hourRadian)), (int) (centerY - shisize
* Math.sin(hourRadian)), paint);
paint.setStrokeWidth(5);

canvas.drawLine(centerX, centerY, (int) (centerX + fensize
* Math.cos(minuteRadian)), (int) (centerY - fensize
* Math.sin(minuteRadian)), paint);
// 设置实针为4个象素粗

paint.setStrokeWidth(4);
canvas.drawLine(centerX, centerY, (int) (centerX + miaosize
* Math.cos(secRadian)), (int) (centerY - miaosize
* Math.sin(secRadian)), paint);

handler.postDelayed(this, 1000);
}

}

布局

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context="com.xh.mytimetest.MainActivity"
android:orientation="vertical"
android:gravity="center">

<com.xh.mytime.MyTimeTest
android:id="@+id/myTimeTest2"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
Src="@drawable/mytime"
hzhong="0.512"
wzhong="0.505"
shisize="65"
fensize="80"
guimo="0.75"
miaosize="95"
android:clickable="true"/>

</LinearLayout>

源代码上传,另外包含一个数字时针,点击可跳转

http://files.cnblogs.com/files/1124hui/MyTime.zip

Android-自定义控件之时针-霞辉的更多相关文章

  1. Android getAttributeIntValue()详解-霞辉

    经常使用getAttributeIntValue()方法,但是大多使用的形式是attrs.getAttributeFloatValue(null, "xxx", 0);只是在中间传 ...

  2. Android自定义控件之自定义ViewGroup实现标签云

    前言: 前面几篇讲了自定义控件绘制原理Android自定义控件之基本原理(一),自定义属性Android自定义控件之自定义属性(二),自定义组合控件Android自定义控件之自定义组合控件(三),常言 ...

  3. Android自定义控件之自定义组合控件

    前言: 前两篇介绍了自定义控件的基础原理Android自定义控件之基本原理(一).自定义属性Android自定义控件之自定义属性(二).今天重点介绍一下如何通过自定义组合控件来提高布局的复用,降低开发 ...

  4. Android自定义控件之自定义属性

    前言: 上篇介绍了自定义控件的基本要求以及绘制的基本原理,本篇文章主要介绍如何给自定义控件自定义一些属性.本篇文章将继续以上篇文章自定义圆形百分比为例进行讲解.有关原理知识请参考Android自定义控 ...

  5. Android自定义控件之基本原理

    前言: 在日常的Android开发中会经常和控件打交道,有时Android提供的控件未必能满足业务的需求,这个时候就需要我们实现自定义一些控件,今天先大致了解一下自定义控件的要求和实现的基本原理. 自 ...

  6. Android自定义控件1

    概述 Android已经为我们提供了大量的View供我们使用,但是可能有时候这些组件不能满足我们的需求,这时候就需要自定义控件了.自定义控件对于初学者总是感觉是一种复杂的技术.因为里面涉及到的知识点会 ...

  7. 一起来学习Android自定义控件1

    概述 Android已经为我们提供了大量的View供我们使用,但是可能有时候这些组件不能满足我们的需求,这时候就需要自定义控件了.自定义控件对于初学者总是感觉是一种复杂的技术.因为里面涉及到的知识点会 ...

  8. [Xamarin.Android] 自定义控件

    [Xamarin.Android] 自定义控件 前言 软件项目开发的过程中,免不了遇到一些无法使用内建控件就能满足的客户需求,例如:时速表.折线图...等等.这时开发人员可以透过自定义控件的方式,为项 ...

  9. android自定义控件实现TextView按下后字体颜色改变

    今天跟大家分享一下Android自定义控件入门,先介绍一个简单的效果TextView,按下改变字体颜色,后期慢慢扩展更强大的功能 直接看图片             第一张是按下后截的图,功能很简单, ...

随机推荐

  1. JAVA版A星算法实现

    import java.util.ArrayDeque; import java.util.ArrayList; import java.util.Collections; import java.u ...

  2. spring MVC cors跨域实现源码解析

    # spring MVC cors跨域实现源码解析 > 名词解释:跨域资源共享(Cross-Origin Resource Sharing) 简单说就是只要协议.IP.http方法任意一个不同就 ...

  3. Bootstrap WPF Style(二)--Glyphicons 字体图标

    介绍 关于Glyphicons字体图标,首先给出友情链接 Glyphicons 这个项目是在Bootstrap WPF Style项目基础上做的,详见http://www.cnblogs.com/ts ...

  4. nginx : server_name localhost 和 chrome : Provisional headers are shown

    问题相关问题现象:解决思路解决方案总结 问题相关 nginx : server_name localhost chrome : Provisional headers are shown 问题现象: ...

  5. Linux编程之ICMP洪水攻击

    我的上一篇文章<Linux编程之PING的实现>里使用ICMP协议实现了PING的程序,ICMP除了实现这么一个PING程序,还有哪些不为人知或者好玩的用途?这里我将介绍ICMP另一个很有 ...

  6. Python学习--21 电子邮件

    发送邮件 SMTP是发送邮件的协议,Python内置对SMTP的支持,可以发送纯文本邮件.HTML邮件以及带附件的邮件. Python对SMTP支持有smtplib和email两个模块,email负责 ...

  7. MySQL日志系统

    body { font-family: Helvetica, arial, sans-serif; font-size: 14px; line-height: 1.6; padding-top: 10 ...

  8. OSS最新进度汇报(2.25)

    OSS系列最新进度情况如下:一. OSS.Social进度 1. 客服接口升级至新客服接口 2. BaseRecMsg中RecMsg字符串类型修改为xml类型 3. 添加Redis缓存注册实现,代码见 ...

  9. PC端Web项目开发流程

    从前一直再做前端,突然想到如果有一天领导让自己独立承担一个web 项目的话是否有足够的能力去接这个任务,要学会自己去搭建一些基础的工具信息.所有的这一切在心里都要有个大致的流程,不然真正做的时候难免会 ...

  10. JS完成页面跳转并传参的方法|附加:循环遍历对象

    此方法只能传递较少参数 方法如下: <a href='page/index.html'>跳转</a> 以上是正常写法,如果要传参按一下写法: <!--参数写在?后面,多个 ...