【转载】Android绘图系列(五)——绘制文本

原文地址:Greathfs的Android绘图系列(五)——绘制文本

这个系列主要是介绍下Android自定义View和Android绘图机制,自己能力有限,如果在介绍过程中有什么错误,欢迎指正

绘制方法

首先,我们看下绘制文本相关方法

    // 第一种
    public void drawText (String text, float x, float y, Paint paint)
    public void drawText (String text, int start, int end, float x, float y, Paint paint)
    public void drawText (CharSequence text, int start, int end, float x, float y, Paint paint)
    public void drawText (char[] text, int index, int count, float x, float y, Paint paint)

    // 第二种
    public void drawPosText (String text, float[] pos, Paint paint)
    public void drawPosText (char[] text, int index, int count, float[] pos, Paint paint)

    // 第三种
    public void drawTextOnPath (String text, Path path, float hOffset, float vOffset, Paint paint)
    public void drawTextOnPath (char[] text, int index, int count, Path path, float hOffset, float vOffset, Paint paint)

这三种方式有什么区别吗?

第一种:基本的绘制方法,只能指定文本基线位置(基线x默认在字符串左侧,基线y默认在字符串下方)

第二种:可以分别指定每个文字的位置。

第三种:根据路径绘制文本

这里简单解释下基线

这里写图片描述

这是我们以前写英文的四线格,其中从上往下数第三条线就是基线!

这里再说下第一种绘制文本的参数问题:第一个参数是我们需要绘制的文本,第四个参数是我们的画笔,这两个不用多说,主要是第二和第三个参数的含义,这两个参数在不同的情况下的值还是不一样的,x默认是这个字符串的左边在屏幕的位置,如果设置了paint.setTextAlign(Paint.Align.CENTER);那就是字符的中心,y是指定这个字符baseline在屏幕上的位置,大家记住了,不要混淆,y不是这个字符中心在屏幕上的位置,而是baseline在屏幕上的位置!

绘制文本

Paint和第一种绘制方式

前面的文章简单介绍过Paint,这篇还是要再介绍点属性,因为我们再绘制文本的时候,Paint设置不同属性,那么绘制出的文本也不同

//基本设置 
paint.setStrokeWidth (2);//设置画笔宽度  
paint.setAntiAlias(true); //指定是否使用抗锯齿功能,如果使用,会使绘图速度变慢  
paint.setStyle(Paint.Style.FILL);//绘图样式,对于设文字和几何图形都有效  

//设置文字对齐方式,取值:align.CENTER、align.LEFT或align.RIGHT  
paint.setTextAlign(Align.CENTER);
//设置文字大小
paint.setTextSize(12);  

//样式设置  
paint.setFakeBoldText(true);//设置是否为粗体文字  
paint.setUnderlineText(true);//设置下划线  
paint.setTextSkewX((float) -0.25);//设置字体水平倾斜度,普通斜体字是-0.25  
paint.setStrikeThruText(true);//设置带有删除线效果  

//设置拉伸  
paint.setTextScaleX(2);//只会将水平方向拉伸,高度不会变 

OK,我们分别看下效果

  • 绘图样式不同效果

Paint一共有三种样式:填充,描边,填充和描边

         Paint paint=new Paint();
        paint.setColor(Color.BLUE);  //设置画笔颜色

        paint.setStrokeWidth (5);//设置画笔宽度
        paint.setAntiAlias(true); //指定是否使用抗锯齿功能,如果使用,会使绘图速度变慢
        paint.setTextSize(100);//设置文字大小

        //绘图样式设置为填充
        paint.setStyle(Paint.Style.FILL);
        canvas.drawText("这是一条测试数据", 20,200, paint);

        //绘图样式设置为描边
        paint.setStyle(Paint.Style.STROKE);
        canvas.drawText("这是一条测试数据", 20,400, paint);

        //绘图样式设置为填充且描边
        paint.setStyle(Paint.Style.FILL_AND_STROKE);
        canvas.drawText("这是一条测试数据", 20,600, paint);

效果图:


这里写图片描述
  • 文字样式不同效果
Paint paint=new Paint();
        paint.setColor(Color.BLUE);  //设置画笔颜色

        paint.setStrokeWidth (5);//设置画笔宽度
        paint.setAntiAlias(true); //指定是否使用抗锯齿功能,如果使用,会使绘图速度变慢
        paint.setTextSize(100);//设置文字大小
        paint.setStyle(Paint.Style.FILL);

        //样式设置
        //粗体
        paint.setFakeBoldText(true);
        //下划线
        paint.setUnderlineText(true);
        //删除线
        paint.setStrikeThruText(true);

        //不倾斜
        canvas.drawText("这是一条测试数据",20,200,paint);

        //向有倾斜
        paint.setTextSkewX((float) -0.25);
        canvas.drawText("这是一条测试数据",20,400,paint);

        //向左倾斜
        paint.setTextSkewX((float) 0.25);
        canvas.drawText("这是一条测试数据",20,600,paint);

效果图


这里写图片描述
  • 其它效果
 Paint paint=new Paint();
        paint.setColor(Color.BLUE);  //设置画笔颜色

        paint.setStrokeWidth (5);//设置画笔宽度
        paint.setAntiAlias(true); //指定是否使用抗锯齿功能,如果使用,会使绘图速度变慢
        paint.setTextSize(100);//设置文字大小
        paint.setStyle(Paint.Style.FILL);//绘图样式,设置为填充

        //变通样式字体
        canvas.drawText("这是一条测试数据", 20,200, paint);

        //水平方向拉伸两倍
        paint.setTextScaleX(2);//只会将水平方向拉伸,高度不会变
        canvas.drawText("这是一条测试数据", 20,400, paint);

        //写在同一位置,不同颜色,看下高度是否看的不变
        paint.setTextScaleX(1);//先还原拉伸效果
        canvas.drawText("这是一条测试数据", 20,600, paint);

        paint.setColor(Color.RED);
        paint.setTextScaleX(2);//重新设置拉伸效果
        canvas.drawText("这是一条测试数据", 20,600, paint);

效果图


这里写图片描述

OK,Paint关于绘制文本这部分的属性就介绍这些

第二种绘制方式

    public void drawPosText (String text, float[] pos, Paint paint)
    public void drawPosText (char[] text, int index, int count, float[] pos, Paint paint)

解释下参数: String text:要绘制的文字 char[] text:要绘制的文字数组 int index::第一个要绘制的文字的索引 int count:要绘制的文字的个数,用来算最后一个文字的位置,从第一个绘制的文字开始算起 float[] pos:每个字体的位置,同样两两一组,如{x1,y1,x2,y2,x3,y3……}

Paint paint=new Paint();
        paint.setColor(Color.BLUE);  //设置画笔颜色

        paint.setStrokeWidth (5);//设置画笔宽度
        paint.setAntiAlias(true); //指定是否使用抗锯齿功能,如果使用,会使绘图速度变慢
        paint.setTextSize(100);//设置文字大小
        //绘图样式设置为填充
        paint.setStyle(Paint.Style.FILL);

        canvas.drawPosText("Hello",new float[]{
                100,100,    // 第一个字符位置
                200,200,    // 第二个字符位置
                300,300,    // ...
                400,400,
                500,500
        },paint);

效果图


这里写图片描述

第三种绘制方式

    public void drawTextOnPath (String text, Path path, float hOffset, float vOffset, Paint paint)
    public void drawTextOnPath (char[] text, int index, int count, Path path, float hOffset, float vOffset, Paint paint)

解释下参数 float hOffset : 与路径起始点的水平偏移距离 float vOffset : 与路径中心的垂直偏移量
这种设计到Path(路径),这个后面会写,就先看看小例子了解下就行

Paint paint=new Paint();
        paint.setColor(Color.RED);  //设置画笔颜色

        paint.setStrokeWidth (5);//设置画笔宽度
        paint.setAntiAlias(true); //指定是否使用抗锯齿功能,如果使用,会使绘图速度变慢
        paint.setTextSize(45);//设置文字大小
        paint.setStyle(Paint.Style.STROKE);//绘图样式,设置为填充

        String string="Hello World";

        //先创建两个相同的圆形路径,并先画出两个路径原图
        Path circlePath=new Path();
        circlePath.addCircle(220,200, 180, Path.Direction.CCW);//逆向绘制
        canvas.drawPath(circlePath, paint);//绘制出路径原形

        Path circlePath2=new Path();
        circlePath2.addCircle(750,200, 180, Path.Direction.CCW);
        canvas.drawPath(circlePath2, paint);//绘制出路径原形

        paint.setColor(Color.GREEN);
        //hoffset、voffset参数值全部设为0,看原始状态是怎样的
        canvas.drawTextOnPath(string, circlePath, 0, 0, paint);
        //第二个路径,改变hoffset、voffset参数值
        canvas.drawTextOnPath(string, circlePath2, 80, 30, paint);

效果图


这里写图片描述

drawText()中原点坐标(x,y)

我们知道,我们在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);  

(1)setTextAlign(Paint.Align.LEFT)

         @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        Paint paint=new Paint();
        //设置基本属性
        paint.setAntiAlias(true);
        paint.setColor(Color.RED);
        paint.setStyle(Paint.Style.FILL);
        paint.setTextSize(120);

        paint.setTextAlign(Paint.Align.LEFT);
//        paint.setTextAlign(Paint.Align.RIGHT);
//        paint.setTextAlign(Paint.Align.CENTER);
        canvas.drawText("abcdefghijk", 200, 200, paint);
        //绘制坐标系
        paint.setColor(Color.BLUE);
        canvas.drawLine(0, 200, getWidth(), 200, paint);
        canvas.drawLine(200, 0, 200, getHeight(), paint);
    }
这里写图片描述

看完效果图,我们应该知道了原点(x,y)在矩形的左侧,即矩形从(x,y)的点开始绘制

(2)setTextAlign(Paint.Align.RIGHT)

把上面的代码改一改

//画基线
        //设置基本属性
        paint.setAntiAlias(true);
        paint.setColor(Color.RED);
        paint.setStyle(Paint.Style.FILL);
        paint.setTextSize(120);

//        paint.setTextAlign(Paint.Align.LEFT);
        paint.setTextAlign(Paint.Align.RIGHT);
//        paint.setTextAlign(Paint.Align.CENTER);
        canvas.drawText("abcdefghijk", 200, 200, paint);
        //绘制坐标系
        paint.setColor(Color.BLUE);
        canvas.drawLine(0, 200, getWidth(), 200, paint);
        canvas.drawLine(200, 0, 200, getHeight(), paint);
这里写图片描述

原点(x,y)应在所要绘制矩形的右侧,也就是说正个矩形都在(0,200)的左侧,所以我们看到的是什么都没有。

(3)setTextAlign(Paint.Align.RIGHT)

 Paint paint=new Paint();
        //设置基本属性
        paint.setAntiAlias(true);
        paint.setColor(Color.RED);
        paint.setStyle(Paint.Style.FILL);
        paint.setTextSize(120);

//        paint.setTextAlign(Paint.Align.LEFT);
//        paint.setTextAlign(Paint.Align.RIGHT);
        paint.setTextAlign(Paint.Align.CENTER);
        canvas.drawText("abcdefghijk", 200, 200, paint);
        //绘制坐标系
        paint.setColor(Color.BLUE);
        canvas.drawLine(0, 200, getWidth(), 200, paint);
这里写图片描述

原点(x,y)就在所要绘制文字所在矩形区域的正中间,换句话说,系统会根据(x,y)的位置和文字所在矩形大小,会计算出当前开始绘制的点,以使原点(x,y)正好在所要绘制的矩形的正中间。

FontMetrics

绘制text的其他线

除了基线,系统在绘制Text时,还是有其它线的,如下图

这里写图片描述

从图中可以知道,除了基线,还有另外的四条线,它们分别是 top,ascent,descent和bottom,它们的含义分别为:

  • top:可绘制的最高高度所在线
  • bottom:可绘制的最低高度所在线
  • ascent :系统建议的,绘制单个字符时,字符应当的最高高度所在线
  • descent:系统建议的,绘制单个字符时,字符应当的最低高度所在线

FontMetrics介绍

既然还有四条线,那这些线的位置怎么计算呢?
Android提供了一个类:FontMetrics,这个类里面对应有成员变量

这里写图片描述

它们的计算方法如下

ascent = ascent线的y坐标 - baseline线的y坐标;//负数

descent = descent线的y坐标 - baseline线的y坐标;//正数

top = top线的y坐标 - baseline线的y坐标;//负数

bottom = bottom线的y坐标 - baseline线的y坐标;//正数

leading = top线的y坐标 - ascent线的y坐标;//负数

FontMetrics的这几个变量的值都是以baseLine为基准的,对于ascent来说,baseline线在ascent线之下,所以必然baseline的y值要大于ascent线的y值,所以ascent变量的值是负的。其他几个同理。

同样我们可以推算出:

ascent线Y坐标 = baseline线的y坐标 + fontMetric.ascent;

descent线Y坐标 = baseline线的y坐标 + fontMetric.descent;

top线Y坐标 = baseline线的y坐标 + fontMetric.top;

bottom线Y坐标 = baseline线的y坐标 + fontMetric.bottom;

绘制ascent,descent,top,bottom线

获取FontMetrics实例

        Paint.FontMetrics fontMetrics=paint.getFontMetrics();
        Paint.FontMetricsInt fontMetricsInt = paint.getFontMetricsInt();

区别:得到对象的成员变量的值一个为float类型,一个为int类型。

绘制

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("abcdefghijk", 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.BLACK);
        canvas.drawLine(baseLineX, descent, 3000, descent, paint);

        //画bottom
        paint.setColor(Color.RED);
        canvas.drawLine(baseLineX, bottom, 3000, bottom, paint);
这里写图片描述

获取文字宽度、高度和最小矩形

获取文字宽度

 String text="abcdefghijk";
 //文字宽度
 float width = mPaint.measureText(text);

获取文字高度

        float top = fontMetrics.top + baseLineY;
        float bottom = fontMetrics.bottom + baseLineY;
        //文字高度
        float height= bottom - top; //注意top为负数
        //文字中点y坐标
        float center = (bottom - top) / 2;

获取文字最小矩形

/** 
 * 获取指定字符串所对应的最小矩形,以(0,0)点所在位置为基线 
 * @param text  要测量最小矩形的字符串 
 * @param start 要测量起始字符在字符串中的索引 
 * @param end   所要测量的字符的长度 
 * @param bounds 接收测量结果 
 */  
public void getTextBounds(String text, int start, int end, Rect bounds); 

例子:

String text = "abcdefghijk";  
Paint paint = new Paint();  
//设置paint  
paint.setTextSize(120); //以px为单位  

Rect minRect = new Rect();  
paint.getTextBounds(text,0,text.length(),minRect); 

drawText实现中文垂直居中

drawText里的origin是以baseline为基准的,直接以目标矩形的bottom传进drawText,字符位置会偏下。

 Rect rect = new Rect(50, 50, 1000, 200);
        Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
        paint.setStrokeWidth(3);
        paint.setTextSize(80);
        String testString = "abcdefghijk";
        paint.setColor(Color.BLUE);
        canvas.drawRect(rect, paint);
        paint.setColor(Color.RED);
        canvas.drawText(testString, rect.left, rect.bottom, paint);
这里写图片描述

所以我们需要自己计算下基线的位置

这里写图片描述

这是上面的一张图,我们知道,我们绘制的Text在FontMetrics.top和FontMetrics.bottom线之间,FontMetrics.top的数值是个负数,其绝对值就是字体绘制边界到baseline的距离。所以如果是把文字画在 FontMetrics高度的矩形中, drawText就应该传入 -FontMetrics.top。要画在targetRect的居中位置,baseline的计算公式就是:

targetRect.centerY() - (FontMetrics.bottom - FontMetrics.top) / 2 - FontMetrics.top

优化后即:

(targetRect.bottom + targetRect.top - fontMetrics.bottom - fontMetrics.top) / 2

 Rect targetRect = new Rect(50, 50, 1000, 200);
        Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
        paint.setStrokeWidth(3);
        paint.setTextSize(80);
        String testString = "abcdefghijk";
        paint.setColor(Color.BLUE);
        canvas.drawRect(targetRect, paint);
        paint.setColor(Color.RED);
        Paint.FontMetricsInt fontMetrics = paint.getFontMetricsInt();
        int baseline = (targetRect.bottom + targetRect.top - fontMetrics.bottom - fontMetrics.top) / 2;
        paint.setTextAlign(Paint.Align.CENTER);
        canvas.drawText(testString, targetRect.centerX(), baseline, paint);
这里写图片描述

结尾

Github地址

参考文章:

http://blog.csdn.net/harvic880925/article/details/50423762

http://blog.csdn.net/hursing/article/details/18703599

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,463评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,868评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,213评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,666评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,759评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,725评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,716评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,484评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,928评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,233评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,393评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,073评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,718评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,308评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,538评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,338评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,260评论 2 352

推荐阅读更多精彩内容