自定义View(五)--drawText

Canvas除了能绘制基础集合图形, 绘制图片以外, 还有一个非常重要, 同时也相对复杂一点的功能 – 绘制文字.

drawText 之 Canvas

canvas中关于绘制文字的方法

public void drawText(@NonNull String text, float x, float y, @NonNull Paint paint)
// start 和 end 用于控制绘制文字的个数
public void drawText(@NonNull String text, int start, int end, float x, float y, @NonNull Paint paint)
public void drawText(@NonNull CharSequence text, int start, int end, float x, float y, @NonNull Paint paint)
public void drawText(@NonNull char[] text, int index, int count, float x, float y, @NonNull Paint paint)
// 将文字绘制到路径上
// hOffset 和 vOffset 分别指文字相对于 Path 的水平偏移量 和 竖直偏移量
public void drawTextOnPath(@NonNull String text, @NonNull Path path, float hOffset, float vOffset, @NonNull Paint paint)
public void drawTextOnPath(@NonNull char[] text, int index, int count, @NonNull Path path, float hOffset, float vOffset, @NonNull Paint paint)

源码中关于 x, y 的解释

上面这些方法的功能, 还有其它参数的作用, 都比较容易理解. 其中 (x, y) 这两个参数需要重点关注. 根据以往的经验, 比如绘制点的时候, (x, y)就是点的坐标, 绘制矩形的时候, (x, y)表示矩形左上角的坐标. 那么这里的 (x, y) 是指所绘制文本所在矩形的左上角吗? 实际上并不是, 看看源码里面的注释:

/**
 * @param x The x-coordinate of the origin of the text being drawn    -> 所绘制文本原点的X坐标 
 * @param y The y-coordinate of the baseline of the text being drawn  -> 所绘制文本基线的Y坐标
 */
public void drawText(@NonNull String text, float x, float y, @NonNull Paint paint) {
    super.drawText(text, x, y, paint);
}

关于X还比较好理解, 就是文字原点的水平坐标, 那么基线是什么? 下面就用一个的小案例说明什么是基线:

绘制基线

定义一个View, 在 onDraw 方法中绘制一段文字, 和两条直线

public class TestTextView extends View {
    ...
    @Override
    protected void onDraw(Canvas canvas) {
        // 测试代码, 实际开发最好不要在这里创建对象
        Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG | Paint.DITHER_FLAG);
        paint.setColor(0xff000000);
        // 字体大小, 单位 px
        paint.setTextSize(45);
        float x = 540;
        float y = 500;
        // 绘制文字
        canvas.drawText("汉字 English 906", x, y, paint);
        paint.setColor(0xffff0000);
        // 绘制经过 (x, y) 的垂直线
        canvas.drawLine(x, 400, x, 520, paint);
        // 绘制经过 (x, y) 的水平线, 即上面这些文字的基线
        canvas.drawLine(80, y, 1000, y, paint);
    }
}

效果如下:

可以看出, x即表示文字原点的水平坐标. 仔细看会发现x离第一个文字中间还有一段缝隙, 这是为了美观, 所以文字所占的实际宽度要略大于显示出来的宽度;
而基线的位置, 用文字描绘起来可能不太容易. 可以回忆一下小时候的 拼音本

drawText 之 Paint(一) – 设置文字显示效果

Paint可以辅助我们进行文字的绘制

设置文字的显示效果

// 设置字体大小 单位 px
public void setTextSize(float textSize)
// 是否加粗
public void setFakeBoldText(boolean fakeBoldText)
// 设置文字删除线
public void setStrikeThruText(boolean strikeThruText)
// 设置文字下划线
public void setUnderlineText(boolean underlineText)
// skewX 一般取 (-1, 1)  负数右倾
public void setTextSkewX(float skewX)
// 设置文字水平拉伸
public void setTextScaleX(float scaleX)
// 设置单词间距
public void setWordSpacing(float wordSpacing)
// 设置字符间距. 默认值是 0  (实际上文字占用的宽度比显示出来的要宽一点, 表现形式为及时设置了0, 字符间还是有一定缝隙, 这是为了美观)
public void setLetterSpacing(float letterSpacing)
// 设置文字所在地区
public void setTextLocale(@NonNull Locale locale) 
public void setTextLocales(@NonNull @Size(min=1) LocaleList locales)
// 设置文字水平对齐方式:  Align.LEFT  Align.CETNER  Align.RIGHT
public void setTextAlign(Align align)
// 设置字体
public Typeface setTypeface(Typeface typeface)

文字水平对齐

通过 paint.setTextAlign(), 可以设置文字的对齐方式. 参数 Align 是一个枚举

public enum Align {
    /**
     * The text is drawn to the right of the x,y origin  -> 文本绘制在(x,y)原点的右侧
     */
    LEFT    (0),
    /**
     * The text is drawn centered horizontally on the x,y origin -> 文本在(x,y)原点水平居中绘制
     */
    CENTER  (1),
    /**
     * The text is drawn to the left of the x,y origin -> 文本绘制在(x,y)原点的左侧
     */
    RIGHT   (2);

    private Align(int nativeInt) {
        this.nativeInt = nativeInt;
    }
    final int nativeInt;
}

分别看一下三种对齐方式的区别:

  • Align.LEFT
  • Align.CENTER
  • Align.RIGHT

此时, 可以重新描述一下 (x, y) :

(x, y) 是文字的原点坐标. 通过 paint.setTextAlign() 可以设置文字相对于原点的水平对齐方式. y 所在的水平线, 即文字的基线.

设置字体

通过 paint.setTypeface() 可以设置字体

  • 系统已经提供了几个字体常量
    Typeface.DEFAULT_BOLD    //
    Typeface.SANS_SERIF      // 
    Typeface.SERIF           // 
    Typeface.MONOSPACE       // 
    
  • 通过字体名称创建系统内置的字体
    // style 值字体风格, 取值范围如下
    // Typeface.NORMAL         普通风格
    // Typeface.BOLD           加粗
    // Typeface.ITALIC         倾斜
    // Typeface.BOLD_ITALIC    加粗 & 倾斜
    public static Typeface create(String familyName, @Style int style)
    // 案例
    Typeface typeface = Typeface.create("宋体", Typeface.NORMAL);
    
  • 自定义字体
    // 通过 assets 创建字体
    public static Typeface createFromAsset(AssetManager mgr, String path)
    // 通过字体文件路径创建爱你字体
    public static Typeface createFromFile(@Nullable String path)
    public static Typeface createFromFile(@Nullable File file)
    // 案例  (字体文件所在路径: assets/font/一腔诗意体.ttf)
    Typeface typeface = Typeface.createFromAsset(getContext().getAssets(), "fonts/一腔诗意体.ttf");
    
    效果如下:

drawText 之 Paint(二) – 测量文字尺寸

FontMetrics

FontMetrics 类用来描述文字的区域范围. Paint中定义了获取 FontMetrics 的方法

// 每次调用返回新的对象
public FontMetrics getFontMetrics()
// 计算结果填进传入的 FontMetrics, 频繁调用时性能好一点
public float getFontMetrics(FontMetrics metrics) 

FontMetrics 中包含以下几个变量:

public static class FontMetrics {
    /**
     * 文字可以绘制的最高高度 离 baselien 的垂直距离:  负值
     * top = top线的y坐标 - baseline线的y坐标   =>   top线Y坐标 = baseline线的y坐标 + fontMetric.top
     */
    public float   top;
    /**
     * 系统推荐的, 绘制文字时的最高高度 离 baselien 的垂直距离:  负值
     * ascent = ascent线的y坐标 - baseline线的y坐标   =>   ascent线Y坐标 = baseline线的y坐标 + fontMetric.ascent
     */
    public float   ascent;
    /**
     * 系统推荐的, 绘制文字时的最低高度 离 baselien 的垂直距离:  正值
     * descent = descent线的y坐标 - baseline线的y坐标  =>   descent线Y坐标 = baseline线的y坐标 + fontMetric.descent
     */
    public float   descent;
    /**
     * 文字可以绘制的最低高度 离 baselien 的垂直距离:  正值
     * bottom = bottom线的y坐标 - baseline线的y坐标  =>   bottom线Y坐标 = baseline线的y坐标 + fontMetric.bottom
     */
    public float   bottom;
    /**
     * 行的额外间距, 上一行的 bottom 和 下一行的 top 的间距
     */
    public float   leading;
}

top, ascent, descent, bottom 这四个变量所代表的水平线, 位置如下:

这四条线的作用: ascent ~ descent 之间的区域是系统推荐的安全范围, 我们在绘制文字时, 尽量在这个范围内完成, 这样不管是哪个地区的文字, 都能够正常的展示. top ~ bottom则表示绘制文字时理论上可以允许的最大范围, 超过这个范围的部分会被裁剪.

获取行距

// 获取两行文字, baselien的间距 (手动进行文字换行时, 可能会有用)
public float getFontSpacing()

获取文字的宽高

代码如下:

String text = "汉字 English 906";
Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG | Paint.DITHER_FLAG);
paint.setTextSize(100);
paint.setTextAlign(Paint.Align.CENTER);
// 原点
int x = 540, y = 500;
Rect mRect = new Rect();
// FontMetricsInt 和 FontMetrics 基本一样, 只是它里面的变量类型为 int
Paint.FontMetricsInt fontMetrics = paint.getFontMetricsInt();
int bottom = y + fontMetrics.bottom;
int top = y + fontMetrics.top;
// ② 文字实际所占高度即 bottom 和 top 之间的距离
int height = bottom - top;
// ③ 文字实际所占宽度通过 paint.measureText() 方法获取
int width = (int) paint.measureText(text);
// 因为设置了水平居中对齐
mRect.left = x - width / 2;
mRect.right = x + width / 2;
mRect.top = top;
mRect.bottom = bottom;
paint.setColor(Color.RED);
// 绘制文字实际所占的范围
canvas.drawRect(mRect, paint);
// ① 通过 paint.getTextBounds() 获取包裹文字的最小矩形范围
// 该矩形是以 (0,0)为原点计算出来的, 需要对齐做下平移, 才能找到文字真正所在的位置
paint.getTextBounds(text,0,text.length(),mRect);
// 通过平移, 找到真正的位置
int minWidth = mRect.width();
// 因为设置了文字水平居中对齐, 而 getTextBounds() 默认是按照 Align.LEFT 处理的
mRect.left = mRect.left + x - minWidth / 2;
mRect.right = mRect.left + minWidth;
mRect.top += y;
mRect.bottom += y;
paint.setColor(Color.YELLOW);
// 绘制最小矩形范围
canvas.drawRect(mRect, paint);
paint.setColor(Color.BLUE);
// 绘制文字
canvas.drawText(text, x, y, paint);

效果如下:

一些可能有用的方法

/**
 * 根据指定宽度截留文字
 * @param text  The text to measure. Cannot be null.
 * @param measureForwards:  文字的测量方向,true 表示由左往右测量
 * @param maxWidth:         允许的最大宽度
 * @param measuredWidth:    可选的。如果不为空,则返回测量的实际宽度
 * @return 返回值是截留的文字个数(如果宽度没有超限,则是文字的总个数)
 */
public int breakText(String text, boolean measureForwards, float maxWidth, float[] measuredWidth) 
/**
 * 检查字符串是否是一个单独的字形(单个字符, 或者unicode, 或者表情符号)
 */
public boolean hasGlyph(String string)

StaticLayout

drawText() 方法是不会换行的, 虽然可以配合 paint.breakText() 将字符串分割后分别绘制, 但这种方式效率并不高. StaticLayout 就是一个用于处理文字换行的工具类. 使用StaticLayout可以超出 宽度限制, 或者遇到 \n 时自动换行.

  • StaticLayout 的构造方法
    public StaticLayout(CharSequence source,     // 需要换行的字符串
                      TextPaint paint,         // paint
                      int width,               // 文字绘制区域的宽度, 超出这个宽度会自动换行
                      Alignment align,         // 文字对齐方式: ALIGN_NORMAL,  ALIGN_OPPOSITE,  ALIGN_CENTER
                      float spacingmult,       // 行间距的倍数,通常情况下填 1 就好;
                      float spacingadd,        // 行间距的额外增加值,通常情况下填 0 就好;
                      boolean includepad)      // 是否在文字上下添加额外的空间(留白),来避免某些过高的字符的绘制出现越界    
    public StaticLayout(CharSequence source,
                      int bufstart,            // source 的起始位置
                      int bufend,              // source 的结束位置
                      TextPaint paint,
                      int outerwidth,
                      Layout.Alignment align,
                      float spacingmult,
                      float spacingadd,
                      boolean includepad)
    public StaticLayout(CharSequence source,
                      int bufstart,
                      int bufend,
                      TextPaint paint,
                      int outerwidth,
                      Layout.Alignment align,
                      float spacingmult,
                      float spacingadd,
                      boolean includepad,
                      TextUtils.TruncateAt ellipsize,  // 文字无法全部显示时, 省略的方式
                      int ellipsizedWidth)             // 省略的宽度
    
  • StaticLayout 的用法
    // staticLayout 默认画在Canvas的(0,0) 坐标处. 只能通过 canvas.translate(x,y) 调整位置
    staticLayout.draw(canvas);
    

 上一篇
自定义View(六)--Paint 自定义View(六)--Paint
构造方法 直接构造方法// Create a new paint with default settings public Paint() // 使用指定的 flags 初始化 paint. 后续也可以通过 setFlags() 去改变这些
2019-03-21
下一篇 
自定义View(四)--Path 自定义View(四)--Path
Canvas 中除了一系列绘制点、线、基础集合图形、图片、文字的方法, 还有一个非常有用的方法 drawPath. 利用 Path, 除了能实现类似前面所说的这些功能, 还可以构建一些更加复杂的图形. drawPathpublic void
2019-03-17
  目录