自定义view(二)
这里是自定义view(二),上一篇关于自定义view的一些基本知识,比如说自定义view的步骤、会涉及到哪些函数以及如何实现自定义属性,同时实现了一个很基础的自定义控件,一个自定义的计时器,需要看的人可以点击这个链接:http://www.cnblogs.com/YaoJianXun/p/5806926.html。
这次讲的是如何通过坐标系的变化实现一些更复杂的自定义view绘制,上一次博客我们实现了一个类似于计时器的环形控件,这次我们在那个基础上再做一次改动,通过坐标系的变动实现下面的效果:

这里想讲一下关于android坐标系的一些东西:
1、android 坐标系
在android中有两种常用坐标系:
1.1 android坐标系
android坐标系就是以屏幕左上方为原点,向右为x轴正方向,向下为y轴正方的坐标系,可以使用getLocationOnScreen(intlocation[])这个函数获取Android坐标系中点的位置,可就是视图左上角在android坐标系中的位置。
1.2 视图坐标系
这个坐标系描述的是子视图在父视图中的位置,就是子视图左上角在父视图总的位置,坐标系的原点在父视图的左上角,向右为x轴正方向,向下为y轴正方向。
1.3 坐标系常用函数
- view的方法:
getTop():获取到的是view自身顶边到期父布局顶边的距离。
getLeft():获取到的是view自身的左边到其父布局左边的距离。
getBottom(),getRight()也同上,获取的分别是view自身下边右边到其父布局下边右边的距离。
- MotionEvent的方法:
getX():获取点击事件距离控件左边的距离,视图坐标。
getY():获取点击事件距离控件上边的距离,视图坐标。
getRawX(),getRawY(),同上,只是他们获取的android坐标系中的距离,绝对坐标。
1.4 通过坐标系移动实现view绘制
这里介绍两个函数:
- translate():平移android坐标系
- rotate():旋转android坐标系
分析上面图中的效果如何实现:
- 外围的圆形
这个就不做讲述了,很容易实现的效果。
- 表盘上的刻度
这个就要分析一下了,首先画出12点的刻度很容,由于它是一条竖直的直线,只需要指定起止点就可以直接绘制出来了。那么其他的刻度呢,起止点就没有那么容易计算得到了,那么怎么办呢,这个时候其实只要将android坐标系以原点为中心旋转一定的角度,在绘制和12点方向一样的线,就可以了,只不过刻度不同,可能线的长度不同,这个需要设置条件语句进行具体判断,但是绘制的方向很容易就得出来了。下面是代码:
for (int i = 0; i < 60; i++) {
//通过旋转坐标系绘制刻度
String degree = String.valueOf(i / 5);
if (i % 15 == 0) {
mPaint2.setStrokeWidth(5);
mPaint2.setTextSize(30);
canvas.drawLine(mCircleXY, 20, mCircleXY, 80, mPaint2);
if (i == 0) {
canvas.drawText("12", mCircleXY - mPaint2.measureText(degree) / 2, 120, mPaint2);
} else {
canvas.drawText(degree, mCircleXY - mPaint2.measureText(degree) / 2, 120, mPaint2);
}
} else if (i % 5 == 0) {
mPaint2.setStrokeWidth(5);
mPaint2.setTextSize(30);
canvas.drawLine(mCircleXY, 20, mCircleXY,60, mPaint2);
canvas.drawText(degree, mCircleXY - mPaint2.measureText(degree) / 2, 100, mPaint2);
} else {
mPaint2.setStrokeWidth(5);
mPaint2.setTextSize(30);
canvas.drawLine(mCircleXY, 20, mCircleXY, 40, mPaint2);
}
canvas.rotate(6, mCircleXY, mCircleXY);
}
通过上面的代码也能看出,每次选择6度一共旋转60次,每次绘制线的时候判断一下,如果现在整点数,绘制的线长度长一点,如果是12点、3点、6点、9点绘制的线的长度要更长一点。
- 绘制时针分针
这个就更简单了,通过translate()函数移动android坐标系原点到表盘中间,在从中间画出两条长短不一的线就可以了。同时没过一秒,选择一定角度就可以了。下面是代码:
//平移坐标系,绘制时针
canvas.translate(mCircleXY, mCircleXY);
canvas.drawLine(0, 0, 100, 100, mPaint3); //旋转坐标系,绘制分针
canvas.rotate(6 * time, 0, 0);
canvas.drawLine(0, 0, 100, 200, mPaint4); time++;
postInvalidateDelayed(1000);
下面是完整的全部代码:
public class CustomView extends View {
private float mCircleXY, mLength, mRadius;
private Paint mPaint1 = new Paint(); //表圆弧
private Paint mPaint2 = new Paint(); //表刻度
private Paint mPaint3 = new Paint(); //表时针
private Paint mPaint4 = new Paint(); //表秒针
private WindowManager mWM;
private int time = 0;
private int hour = 0;
public CustomView(Context context) {
super(context);
mPaint1.setStyle(Paint.Style.STROKE);
mPaint1.setStrokeWidth(5);
mPaint1.setAntiAlias(false);
mPaint3.setStrokeWidth(20);
mPaint4.setStrokeWidth(10);
setValues();
}
public CustomView(Context context, AttributeSet attrs) {
super(context, attrs);
mPaint1.setStyle(Paint.Style.STROKE);
mPaint1.setStrokeWidth(5);
mPaint1.setAntiAlias(false);
mPaint3.setStrokeWidth(20);
mPaint4.setStrokeWidth(10);
setValues();
}
private void setValues() {
mWM = (WindowManager) getContext()
.getSystemService(Context.WINDOW_SERVICE);
mLength = mWM.getDefaultDisplay().getWidth();
mCircleXY = mLength / 2;
mRadius = (float) (mLength * 0.5);
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
//绘制钟表外圈圆环
canvas.drawCircle(mCircleXY, mCircleXY, mRadius-20, mPaint1);
//消除锯齿
canvas.setDrawFilter(new PaintFlagsDrawFilter(0, Paint.ANTI_ALIAS_FLAG
| Paint.FILTER_BITMAP_FLAG));
for (int i = 0; i < 60; i++) {
//通过旋转坐标系绘制刻度
String degree = String.valueOf(i / 5);
if (i % 15 == 0) {
mPaint2.setStrokeWidth(5);
mPaint2.setTextSize(30);
canvas.drawLine(mCircleXY, 20, mCircleXY, 80, mPaint2);
if (i == 0) {
canvas.drawText("12", mCircleXY - mPaint2.measureText(degree) / 2, 120, mPaint2);
} else {
canvas.drawText(degree, mCircleXY - mPaint2.measureText(degree) / 2, 120, mPaint2);
}
} else if (i % 5 == 0) {
mPaint2.setStrokeWidth(5);
mPaint2.setTextSize(30);
canvas.drawLine(mCircleXY, 20, mCircleXY,60, mPaint2);
canvas.drawText(degree, mCircleXY - mPaint2.measureText(degree) / 2, 100, mPaint2);
} else {
mPaint2.setStrokeWidth(5);
mPaint2.setTextSize(30);
canvas.drawLine(mCircleXY, 20, mCircleXY, 40, mPaint2);
}
canvas.rotate(6, mCircleXY, mCircleXY);
}
//平移坐标系,绘制时针
canvas.translate(mCircleXY, mCircleXY);
canvas.drawLine(0, 0, 100, 100, mPaint3);
//旋转坐标系,绘制分针
canvas.rotate(6 * time, 0, 0);
canvas.drawLine(0, 0, 100, 200, mPaint4);
time++;
postInvalidateDelayed(1000);
}
}
这样就可以绘制出图中所示的表盘,源码的链接:https://github.com/jiushi555/CustomView/tree/master/CustomTimepiece。
转载请标明出处。
不是闷骚的程序员算不上程序员。我的微信公众号“那点鼻事”,在这里周一到周五每天一篇文章,与技术无关,只哈牛逼。

自定义view(二)的更多相关文章
- 自定义view(二)
1.View 的绘制 通过继承View 并重写它的onDraw()来完成绘制. onDraw()有一个参数,就是Canvas对象.使用这个Canvas就可以绘制图像了,Canvas canvas = ...
- Android自定义View (二) 进阶
转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/24300125 继续自定义View之旅,前面已经介绍过一个自定义View的基础的例 ...
- 自定义View(二)--继承自ViewGroup
自定义View包括很多种,上一次随笔中的那一种是完全继承自View,这次写的这个小Demo是继承自ViewGroup的,主要是将自定义View继承自ViewGroup的这个流程来梳理一下,这次的Dem ...
- Android 自定义View (二) 进阶
转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/24300125 继续自定义View之旅,前面已经介绍过一个自定义View的基础的例 ...
- 自定义View(二),强大的Canvas
本文转自:http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2012/1212/703.html Android中使用图形处理引擎,2D部分是 ...
- Android 自定义View二(深入了解自定义属性attrs.xml)
1.为什么要自定义属性 要使用属性,首先这个属性应该存在,所以如果我们要使用自己的属性,必须要先把他定义出来才能使用.但我们平时在写布局文件的时候好像没有自己定义属性,但我们照样可以用很多属性,这是为 ...
- Android绘图机制(二)——自定义View绘制形, 圆形, 三角形, 扇形, 椭圆, 曲线,文字和图片的坐标讲解
Android绘图机制(二)--自定义View绘制形, 圆形, 三角形, 扇形, 椭圆, 曲线,文字和图片的坐标讲解 我们要想画好一些炫酷的View,首先我们得知道怎么去画一些基础的图案,比如矩形,圆 ...
- Android自定义View(二)
前言 魅族手机的闹钟应用中有个倒计时,这个控件还是蛮有趣的.左边是魅族闹钟,右边是我们最终实现的效果,虽然有些细节还需优化,不过基本上已经达到了想要的效果,我们先来就来看看如何实现吧. 分析 确定宽高 ...
- 【朝花夕拾】Android自定义View篇之(二)Canvas常用功能
前言 转在请申明,转自[https://www.cnblogs.com/andy-songwei/p/10960012.html],谢谢! 上一篇讲View的绘制流程中讲到过,最后一步是draw流程, ...
随机推荐
- iOS workspace 依次编译多个工程
目的:当我封装一个framework的时候,需要检验当前的改动,但是又不想编译完framework,又要编译测试工程. 步骤: 1. 选择测试工程 2. Edit Scheme 3. 选中Build- ...
- OSG世界坐标转屏幕坐标(转载)
OSG世界坐标转屏幕坐标 #define M(row,col) m[col * 4 + row] void Transform_Point(double out[4], const double m[ ...
- JS判断doctype文档模式-document.compatMode
IE对盒模型的渲染在 Standards Mode和Quirks Mode是有很大差别的,在Standards Mode下对于盒模型的解释和其他的标准浏览器是一样,但在Quirks Mode模式下则有 ...
- 10种处理PHP字符串的措施
PHP有一个海量字符串操作库,提供了大约100个分割.连接.解析和搜索文本的功能.在这一点上,PHP的性能是如此地强大以至于在处理字符串相关问题时,判断哪一个是最好的方法会比较困难.本文中我讲了10个 ...
- javascript中类的属性访问权限研究(1)
本篇文章主要针对javascript的属性进行分析,由于javascript是一种基于对象的语言,本身没有类的概念,所以对于javascript的类的定义有很多名字,例于原型对象,构造函数等,它们都是 ...
- document.form.command.value
问题:在一个JSP页面中需要多个提交按钮,每个按钮点击后需要把同一个form提交到不同的页面进行处理 解决:用JS. <html><head><title>一个表单 ...
- 按钮(Buton)组价的功能和用法
Button继承了TextView,它主要是在UI界面上生成一个按钮,该按钮可以供用户单机,当用户单击按钮时,按钮会触发一个onClick事件. 按钮使用起来比较容易,可以通过为按钮指定android ...
- js数组快速排序
<script type="text/javascript"> var arr = [1, 2, 3, 54, 22, 1, 2, 3]; function quick ...
- HTML5 & CSS3初学者指南(3) – HTML5新特性
介绍 本文介绍了 HTML5 的一些新特性.主要包含以下几个方面: Web 存储 地理位置 拖放 服务器发送事件 Web存储 HTML5 Web 存储的设计与构想是一个更好的机制来存储客户端的网络数据 ...
- Ubuntu16.04安装GTK3主题:OSX-Arc
Ubuntu16.04安装GTK3主题:OSX-Arc GTK3主题:OSX-Arc描述: 前几个月,Gnome3.20升3.22的时候,出现了大量主题崩溃的现象,其中包括Arc.Flatabulou ...