前言:但行好事,莫问前程。只需努力每一天。
相关文章:
1、《android Graphics(一):概述及基本几何图形绘制》
2、《android Graphics(二):路径及文字》
3、《android Graphics(三):区域(Range)》
4、《android Graphics(四):canvas变换与操作》
5、《android Graphics( 五):drawText()详解》
long long ago,有讲过有关Canvas绘图的系列知识,当时着重在基本绘图要领和canvas的保存与回退上,对drawText没有细讲,但DrawText这个函数,确是相当复杂,这里开一篇重新讲解一下。
那么问题来了,在canvas在利用drawText绘制文字时,也是有规则的,这个规则就是基线!
我们先来看一下什么是基线:
可见基线就是四线格中的第三条线!
也就是说,只要基线的位置定了,那文字的位置必然是定了的!
/** * text:要绘制的文字 * x:绘制原点x坐标 * y:绘制原点y坐标 * paint:用来做画的画笔 */ public void drawText(String text, float x, float y, Paint paint)上面这个构造函数是最常用的drawText方法,传进去一个String对象就能画出对应的文字。
在(x,y)中最让人捉急的是y坐标,一般而言,(x,y)所代表的位置是所画图形对应的矩形的左上角点。但在drawText中是非常例外的,y所代表的是基线的位置!
public class MyView extends View{ public MyView(Context context, AttributeSet attrs) { super(context, attrs); } }2、重写onDraw函数:
protected void onDraw(Canvas canvas) { super.onDraw(canvas); int baseLineX = 0 ; int baseLineY = 200; //画基线 Paint paint = new Paint(); paint.setColor(Color.RED); canvas.drawLine(baseLineX, baseLineY, 3000, baseLineY, paint); //写文字 paint.setColor(Color.GREEN); paint.setTextSize(120); //以px为单位 canvas.drawText("harvic\'s blog", baseLineX, baseLineY, paint); }在这里,先定义drawText原点的位置:(0,200)
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <com.example.blogDrawText.MyView android:layout_width="fill_parent" android:layout_height="fill_parent" /> </LinearLayout>效果图如下:
结论:
我们知道,我们在drawText(text, x, y, paint)中传进去的源点坐标(x,y);其中,y表示的基线的位置。那x代表什么呢?从上面的例子运行结果来看,应当是文字开始绘制的地方。
并不是!x代表所要绘制文字所在矩形的相对位置。相对位置就是指指定点(x,y)在在所要绘制矩形的位置。我们知道所绘制矩形的纵坐标是由Y值来确定的,而相对x坐标的位置,只有左、中、右三个位置了。也就是所绘制矩形可能是在x坐标的左侧绘制,也有可能在x坐标的中间,也有可能在x坐标的右侧。而定义在x坐标在所绘制矩形相对位置的函数是:
/** * 其中Align的取值为:Panit.Align.LEFT,Paint.Align.CENTER,Paint.Align.RIGHT */ Paint::setTextAlign(Align align);我们再来重新看一下上面的例子,当我们设置为不同的值时,绘制结果是怎样的。
protected void onDraw(Canvas canvas) { super.onDraw(canvas); int baseLineY = 200; int baseLineX = 0 ; //画基线 Paint paint = new Paint(); paint.setColor(Color.RED); canvas.drawLine(baseLineX, baseLineY, 3000, baseLineY, paint); //写文字 paint.setColor(Color.GREEN); paint.setTextSize(120); //以px为单位 paint.setTextAlign(Paint.Align.LEFT); canvas.drawText("harvic\'s blog", baseLineX, baseLineY, paint); }结果如下:
可见,原点(x,y)在矩形的左侧,即矩形从(x,y)的点开始绘制
(2)、setTextAlign(Paint.Align.CENTER)
同样的代码,把相对位置设置为:setTextAlign(Paint.Align.CENTER)
protected void onDraw(Canvas canvas) { super.onDraw(canvas); int baseLineY = 200; int baseLineX = 0 ; //画基线 Paint paint = new Paint(); paint.setColor(Color.RED); canvas.drawLine(baseLineX, baseLineY, 3000, baseLineY, paint); //写文字 paint.setColor(Color.GREEN); paint.setTextSize(120); //以px为单位 paint.setTextAlign(Paint.Align.CENTER); canvas.drawText("harvic\'s blog", baseLineX, baseLineY, paint); }结果为:
所以原点(x,y)就在所要绘制文字所在矩形区域的正中间,换句话说,系统会根据(x,y)的位置和文字所在矩形大小,会计算出当前开始绘制的点。以使原点(x,y)正好在所要绘制的矩形的正中间。
(3)、setTextAlign(Paint.Align.RIGHT)
将相对位置设置为:setTextAlign(Paint.Align.RIGHT)
protected void onDraw(Canvas canvas) { super.onDraw(canvas); int baseLineY = 200; int baseLineX = 0 ; //画基线 Paint paint = new Paint(); paint.setColor(Color.RED); canvas.drawLine(baseLineX, baseLineY, 3000, baseLineY, paint); //写文字 paint.setColor(Color.GREEN); paint.setTextSize(120); //以px为单位 paint.setTextAlign(Paint.Align.RIGHT); canvas.drawText("harvic\'s blog", baseLineX, baseLineY, paint); }结果为:
所以原点(x,y)应当在所要绘制矩形的右侧,在上面的代码中,也就是说正个矩形都在(0,200)的左侧,所以我们看到的是什么都没有。
(4)、注意
下面,我们再看一个例子:
我们只写一个大写字母A,然后将其相对位置设置为paint.setTextAlign(Paint.Align.CENTER)
protected void onDraw(Canvas canvas) { super.onDraw(canvas); int baseLineY = 200; int baseLineX = 0 ; //画基线 Paint paint = new Paint(); paint.setColor(Color.RED); canvas.drawLine(baseLineX, baseLineY, 3000, baseLineY, paint); //写文字 paint.setColor(Color.GREEN); paint.setTextSize(120); //以px为单位 paint.setTextAlign(Paint.Align.CENTER); canvas.drawText("A", baseLineX, baseLineY, paint); }结果如下:
我们可以看到字母A在原点(0,200)的正中间。
这也就是我们要强调的:相对位置是根据所要绘制文字所在矩形来计算的。
除了基线以外,如上图所示,另外还有四条线,分别是ascent,descent,top,bottom,他们的意义分别是:
如上图所示,黑色部分表示电视屏幕,红色框就表示安全区域框。
这个安全框是用来干嘛的?这个安全框就是系统推荐给我们的显示区域,虽然说我们可以讲电视屏幕是每个区域都是可以显示图像的,但由于制式的不同,每个国家的屏幕大小并不一定我们这里的屏幕大小一致,当遇到不一致时,就会裁剪。但系统给我们推荐的显示区域是无论哪种制式都是可以完整显示出来的,所以我们在制作视频时,尽量要把要显示的图像放在所推荐的显示区域内。
同样,在这里,我们在绘制文字时,ascent是推荐的绘制文字的最高高度,就表示在绘制文字时,尽力要在这个最高高度以下绘制文字。descent是推荐的绘制文字的最底高度线,同样表示是在绘制文字时尽量在这个descent线以上来绘制文字。而top线则指该文字可以绘制的最高高度线,bottom则是表示该文字可以绘制的最低高度线。ascent,descent是系统建议上的绘制高度,而top,bottom则是物理上屏幕最高,最低可以画的高度值。他们的差别与我们上面说的视频处理的安全框和屏幕的道理是一样的。
FontMetrics::ascent; FontMetrics::descent; FontMetrics::top; FontMetrics::bottom;他们的意义与值的计算方法分别如下:
从这个图中,我们先说明两点,然后再回过头来看上面的公式:
1、X轴,Y轴的正方向走向是X轴向右是正方向,Y轴向下是正方向,所以越往下Y坐标越大!
2、大家千万不要将FontMetrics中的ascent,descent,top,bottom与现实中的ascent,descent,top,bottom所在线混淆!这几条线是真实存在的,而FontMetrics中的ascent,descent,top,bottom这个变量的值就是用来计算这几条线的位置的。下面我们将看到如何利用这几个变量来计算这几条线的位置。
看完这个图,我们再重新说说这几个公式,我们拿一个来说吧,其它都是相同的道理。
ascent = ascent线的y坐标 - baseline线的y坐标;FontMetrics的这几个变量的值都是以baseline为基准的,对于ascent来说,baseline线在ascent线之下,所以必然baseline的y值要大于ascent线的y值,所以ascent变量的值是负的。
descent = descent线的y坐标 - baseline线的y坐标;descent线在baseline线之下,所以必然descent线的y坐标要大于baseline线的y坐标,所以descent变量的值必然是正数。
ascent线Y坐标 = baseline线Y坐标 + fontMetric.ascent;推算过程如下:
(3)、获取FontMetrics对象
获取FontMetrics对象是根据paint对象来获取的:
Paint paint = new Paint(); Paint.FontMetrics fm = paint.getFontMetrics(); Paint.FontMetricsInt fmInt = paint.getFontMetricsInt();从这里可以看到,通过paint.getFontMetrics()得到对应的FontMetrics对象。这里还有另外一个FontMetrics同样的类叫做FontMetricsInt,它的意义与FontMetrics完全相同,只是得到的值的类型不一样而已,FontMetricsInt中的四个成员变量的值都是Int类型,而FontMetrics得到的四个成员变量的值则都是float类型的。
在这个例子中,我们先写一行字,然后画出这行字中的top线,ascent线,baseline线,descent线和bottom线。
我们直接上完整代码:
protected void onDraw(Canvas canvas) { super.onDraw(canvas); int baseLineY = 200; int baseLineX = 0 ; Paint paint = new Paint(); //写文字 paint.setColor(Color.GREEN); paint.setTextSize(120); //以px为单位 paint.setTextAlign(Paint.Align.LEFT); canvas.drawText("harvic\'s blog", baseLineX, baseLineY, paint); //计算各线在位置 Paint.FontMetrics fm = paint.getFontMetrics(); float ascent = baseLineY + fm.ascent; float descent = baseLineY + fm.descent; float top = baseLineY + fm.top; float bottom = baseLineY + fm.bottom; //画基线 paint.setColor(Color.RED); canvas.drawLine(baseLineX, baseLineY, 3000, baseLineY, paint); //画top paint.setColor(Color.BLUE); canvas.drawLine(baseLineX, top, 3000, top, paint); //画ascent paint.setColor(Color.GREEN); canvas.drawLine(baseLineX, ascent, 3000, ascent, paint); //画descent paint.setColor(Color.YELLOW); canvas.drawLine(baseLineX, descent, 3000, descent, paint); //画bottom paint.setColor(Color.RED); canvas.drawLine(baseLineX, bottom, 3000, bottom, paint); }
这段代码中,总共分为三部分,写文字、计算各线所在位置、画出各条线;我们逐段来讲
先看写文字:
int baseLineY = 200; int baseLineX = 0 ; Paint paint = new Paint(); //写文字 paint.setColor(Color.GREEN); paint.setTextSize(120); //以px为单位 paint.setTextAlign(Paint.Align.LEFT); canvas.drawText("harvic\'s blog", baseLineX, baseLineY, paint);有关drawText的问题我们已经讲过,在这段代码中,我们需要注意的是两点:
//计算各线在位置 Paint.FontMetrics fm = paint.getFontMetrics(); float ascent = baseLineY + fm.ascent; float descent = baseLineY + fm.descent; float top = baseLineY + fm.top; float bottom = baseLineY + fm.bottom;首先,利用 paint.getFontMetrics()得到FontMetrics的实例,然后利用我们上面的公式即可得到各条线的y坐标。
这部分,我们将讲解如何获取所绘制字符串所占区域的高度、宽度和仅包裹字符串的最小矩形。我们来看张图来讲述下他们的意义:
在这张图中,文字底部的绿色框就是所绘制字符串所占据的大小。我们要求的宽度和高度也就是这个绿色框的宽度和高度。
从图中也可以看到,红色框部分,它的宽和高紧紧包围着字符串,所以红色框就是我们要求的最小矩形。即能包裹字符串的最小矩形。
Paint.FontMetricsInt fm = paint.getFontMetricsInt(); int top = baseLineY + fm.top; int bottom = baseLineY + fm.bottom; //所占高度 int height = bottom - top;
int width = paint.measureText(String text);使用示例如下:
Paint paint = new Paint(); //设置paint paint.setTextSize(120); //以px为单位 //获取宽度 int width = (int)paint.measureText("harvic\'s blog");
/** * 获取指定字符串所对应的最小矩形,以(0,0)点所在位置为基线 * @param text 要测量最小矩形的字符串 * @param start 要测量起始字符在字符串中的索引 * @param end 所要测量的字符的长度 * @param bounds 接收测量结果 */ public void getTextBounds(String text, int start, int end, Rect bounds);我们简单展示下使用代码及结果:
String text = "harvic\'s blog"; Paint paint = new Paint(); //设置paint paint.setTextSize(120); //以px为单位 Rect minRect = new Rect(); paint.getTextBounds(text,0,text.length(),minRect); Log.e("qijian",minRect.toShortString());在这段代码中,首先设置字体大小,然后利用paint.getTextBounds()得到最小矩形,最后,我将其打印出来
可以看到这个矩形的左上角位置为(8,-90),右下角的位置为(634,26);
大家可能会有疑问,为什么左上角的Y坐标是个负数?从代码中,我们也可以看到,我们并没有给getTextBounds()传递基线位置。那它就是以(0,0)为基线来得到这个最小矩形的!所以这个最小矩形的位置就是以(0,0)为基线的结果!
2、得到最小矩形的实际位置
我们先来看一个原理:
在上面这个图中,我们将黑色矩形平行下移距离Y(黄色线依照的是基线的位置),那么平移后的左上角点的y坐标就是 y2 = y1 + Y;
同样的道理,由于paint.getTextBounds()得到最小矩形的基线是y = 0;那我们直接将这个矩形移动baseline的距离就可以得到这个矩形实际应当在的位置了。
所以矩形应当所在实际位置的坐标是:
Rect minRect = new Rect(); paint.getTextBounds(text,0,text.length(),minRect); //最小矩形,实际top位置 int minTop = bounds.top + baselineY; //最小矩形,实际bottom位置 int minBottom = bounds.bottom + baselineY;3、实例
我们先看一下完整的代码,然后再细讲
protected void onDraw(Canvas canvas) { super.onDraw(canvas); String text = "harvic\'s blog"; int baseLineY = 200; int baseLineX = 0 ; //设置paint Paint paint = new Paint(); paint.setTextSize(120); //以px为单位 paint.setTextAlign(Paint.Align.LEFT); //画text所占的区域 Paint.FontMetricsInt fm = paint.getFontMetricsInt(); int top = baseLineY + fm.top; int bottom = baseLineY + fm.bottom; int width = (int)paint.measureText(text); Rect rect = new Rect(baseLineX,top,baseLineX+width,bottom); paint.setColor(Color.GREEN); canvas.drawRect(rect,paint); //画最小矩形 Rect minRect = new Rect(); paint.getTextBounds(text,0,text.length(),minRect); minRect.top = baseLineY + minRect.top; minRect.bottom = baseLineY + minRect.bottom; paint.setColor(Color.RED); canvas.drawRect(minRect,paint); //写文字 paint.setColor(Color.BLACK); canvas.drawText(text, baseLineX, baseLineY, paint); }这段代码看起来比较吓人,但只是看起来,我们分段来看看,这段代码总共分为四部分:设置paint,画字符串所占据矩形,画最小矩形,画文字
String text = "harvic\'s blog"; int baseLineY = 200; int baseLineX = 0 ; //设置paint Paint paint = new Paint(); paint.setTextSize(120); //以px为单位 paint.setTextAlign(Paint.Align.LEFT);设置paint这部分,主要是设置字体的大小,因为我们在字体所占的区域大小跟字体的大小是有直接关系的,如果不设置,那么在获取所占区域大小时,将利用系统默认的大小来测量了,当然是不行的。
Paint.FontMetricsInt fm = paint.getFontMetricsInt(); int top = baseLineY + fm.top; int bottom = baseLineY + fm.bottom; int width = (int)paint.measureText(text); Rect rect = new Rect(baseLineX,top,baseLineX+width,bottom); paint.setColor(Color.GREEN); canvas.drawRect(rect,paint);这里就是利用我们前面我们讲过的获取top线和bottom线的方法,获取宽度时就是利用paint.measureText(text);
Rect minRect = new Rect(); paint.getTextBounds(text,0,text.length(),minRect); minRect.top = baseLineY + minRect.top; minRect.bottom = baseLineY + minRect.bottom; paint.setColor(Color.RED); canvas.drawRect(minRect,paint);这部分也就没什么难度了,首先根据paint.getTextBounds()得到基线为y=0的最小矩形的各点坐标,然后根据基线得到其实际的top和bottom坐标;然后将其画出来即可
在这个图中,我们给定左上角的位置,即(left,top);我们知道要画文字,drawText()中传进去的Y坐标是基线的位置,所以我们就必须根据top的位置计算出baseline的位置。
我们来看一个公式:
FontMetrics.top = top - baseline;
所以baseline = FontMetrics.top - top;
因为FontMetrics.top是可以得到的,又因为我们的top坐标是给定的,所以通过这个公式就能得到baseline的位置了。
下面举个例子来说明一下根据矩形左上项点绘制文字的过程:
先看下效果图:
在这个效果图中,因为我们会给定矩形左上角顶点(left,top),所以们先画出top线的位置,然后计算出baseline的位置,并画出来。最后根据baseline把文字写出来。
代码如下:
protected void onDraw(Canvas canvas) { super.onDraw(canvas); String text = "harvic\'s blog"; int top = 200; int baseLineX = 0 ; //设置paint Paint paint = new Paint(); paint.setTextSize(120); //以px为单位 paint.setTextAlign(Paint.Align.LEFT); //画top线 paint.setColor(Color.YELLOW); canvas.drawLine(baseLineX, top, 3000, top, paint); //计算出baseLine位置 Paint.FontMetricsInt fm = paint.getFontMetricsInt(); int baseLineY = top - fm.top; //画基线 paint.setColor(Color.RED); canvas.drawLine(baseLineX, baseLineY, 3000, baseLineY, paint); //写文字 paint.setColor(Color.GREEN); canvas.drawText(text, baseLineX, baseLineY, paint); }这段代码,比较简单,首先是我们给定top给的位置int top = 200;然后根据top线位置计算出baseline所在位置,并画出来:
//计算出baseLine位置 Paint.FontMetricsInt fm = paint.getFontMetricsInt(); int baseLineY = top - fm.top; //画基线 paint.setColor(Color.RED); canvas.drawLine(baseLineX, baseLineY, 3000, baseLineY, paint);最后就是写出文字了,没什么难度就不讲了。
在这个图中,总共有四条线:top线,bottom线,baseline和center线;
图中center线正是在top线和bottom线的正中间。
为了方便推导公式,我另外标了三个距离A,B,C;
很显然,距离A和距离C是相等的,都等于文字所在矩形高度以的一半,即:
A = C = (bottom - top)/2;
又因为bottom = baseline + FontMetrics.bottom;
top = baseline+FontMetrics.top;
所以,将它们两个代入上面的公式,就可得到:
A = C = (FontMetrics.bottom - FontMetrics.top)/2;
而距离B,则表示Center线到baseline的距离。
很显然距离B = C - (bottom - baseline);
又因为
FontMetrics.bottom = bottom-baseline;
C = A;
所以,B = A - FontMetrics.bottom;
所以baseline = center + B = center + A - FontMetrics.bottom = center + (FontMetrics.bottom - FontMetrics.top)/2 - FontMetrics.bottom;
根据上面的推导过程,我们最终可知,当给定中间线center位置以后,baseline的位置为:
baseline = center + (FontMetrics.bottom - FontMetrics.top)/2 - FontMetrics.bottom;
下面我们举个例子来说明下这个公式的用法。
效果图如下:
代码如下:
protected void onDraw(Canvas canvas) { super.onDraw(canvas); String text = "harvic\'s blog"; int center = 200; int baseLineX = 0 ; //设置paint Paint paint = new Paint(); paint.setTextSize(120); //以px为单位 paint.setTextAlign(Paint.Align.LEFT); //画center线 paint.setColor(Color.YELLOW); canvas.drawLine(baseLineX, center, 3000, center, paint); //计算出baseLine位置 Paint.FontMetricsInt fm = paint.getFontMetricsInt(); int baseLineY = center + (fm.bottom - fm.top)/2 - fm.bottom; //画基线 paint.setColor(Color.RED); canvas.drawLine(baseLineX, baseLineY, 3000, baseLineY, paint); //写文字 paint.setColor(Color.GREEN); canvas.drawText(text, baseLineX, baseLineY, paint); }这段代码也没什么难度,给定中间线的位置为200,然后计算出baseline的位置,然后把文字在baseline的基础上画出来。
如果本文有帮到你,记得加关注哦
源码下载地址:http://download.csdn.net/detail/harvic880925/9383971
请大家尊重原创者版权,转载请标明出处:http://blog.csdn.net/harvic880925/article/details/50423762 谢谢