Android画图drawText入门(一)-字体结构

[本文参考并修改自博文:http://mikewang.blog.51cto.com/3826268/871765]。

最近在开发一个分段条时,需要使用的Canvas.drawText这个Api。在使用的过程中遇到不少困难,从中了解到自己对“绘制文本”的掌握十分不足,故决定写几篇博客来巩固一下对这一块的知识。

(一)Android的字体结构图

Android画图drawText入门(一)-字体结构_第1张图片

图中展现的属性较多,在本文我们需要关心的参数如下:

1. baseline           :基准线

2. Ascent              :baseline之上至字符最高处的距离

3. Descent            :baseline之下至字符最低处的距离

4. Origin               :字体坐标轴原点,Canvas.drawText中的“参数x”与“参数y”就是该原点的坐标。

5. Line-Height     :行高。


(二)Paint中的FontMetrics

FontMetric类持有上面“结构图”的部分参数,这些参数的取值具体视Paint的设置而定,例如典型的Paint.setTextSize。

我们可以通过Paint.getFontMetrics()来获取它。该类除了以上参数“Ascent”, “Descent”之外。还包含以下参数:

1. Leading:文档说的很含糊,其实是上一行字符的descent到下一行的ascent之间的距离。

2. Top       :指的是指的是最高字符到baseline的值,即ascent的最大值

3. Bottom :指的是最下字符到baseline的值,即descent的最大值

注: “Descent“与“ Bottom” 在baseline之下,所以取值范围为“零或者负数”。


(三)字体参数访问代码

public class LearnFont extends TextView{

	public LearnFont(Context context, AttributeSet attrs, int defStyle) {
		super(context, attrs, defStyle);
	}
	public LearnFont(Context context, AttributeSet attrs) {
		super(context, attrs);
	}
	public LearnFont(Context context) {
		super(context);
	}
	
	@Override
	protected void onDraw(Canvas canvas) {
		super.onDraw(canvas);
		
		Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
		paint.setTextSize(45);  // 字体大小
		paint.setTextScaleX(1); // 字体水平缩放比例
		paint.setTextSkewX(0);  // 字体倾斜角度
		
		System.out.println("---- 字体大小45时,的字体相关参数fontMetrics -----");
		FontMetrics fm45 = paint.getFontMetrics();
		System.out.println("ascent = " + fm45.ascent);
		System.out.println("descent = " + fm45.descent);
		System.out.println("top = " + fm45.top);
		System.out.println("bottom = " + fm45.bottom);
		System.out.println("行高LineHeight = " + (fm45.descent - fm45.top + 2));
		
		System.out.println("---- E字符显示矩形 getTextBounds -----");
		Rect textBounds = new Rect();
		paint.getTextBounds("jEh", 1, 1, textBounds);
		System.out.println("E.boundHeight = " + textBounds.height());
		System.out.println("E.boundWidth = " + textBounds.width());
		
		
		System.out.println("---- getTextWidths -----");
		float widths[] = new float[3];
		paint.getTextWidths("jEh", 0, 3, widths);
		System.out.println("jEh.textWidths = " + Arrays.toString(widths));
		
	}
}


(四)测试代码(引用自其它博客)

 1,测试的代码直接使用网上的代码,完整代码如下:

  
  
  
  
  1. public class FontMetricsDemoActivity extends Activity { 
  2.     private Canvas canvas; 
  3.  
  4.     /** Called when the activity is first created. */ 
  5.     @Override 
  6.     public void onCreate(Bundle savedInstanceState) { 
  7.         super.onCreate(savedInstanceState); 
  8.         setContentView(R.layout.main); 
  9.          
  10.         Paint textPaint = new Paint( Paint.ANTI_ALIAS_FLAG); 
  11.         textPaint.setTextSize( 55); 
  12.         textPaint.setColor( Color.WHITE); 
  13.  
  14.         // FontMetrics对象 
  15.         FontMetrics fontMetrics = textPaint.getFontMetrics(); 
  16.         String text = "abcdefghijklmnopqrstu"
  17.          
  18.          
  19.  
  20.         // 计算每一个坐标 
  21.         float baseX = 0
  22.         float baseY = 100
  23.         float topY = baseY + fontMetrics.top; 
  24.         float ascentY = baseY + fontMetrics.ascent; 
  25.         float descentY = baseY + fontMetrics.descent; 
  26.         float bottomY = baseY + fontMetrics.bottom; 
  27.         float leading = baseY + fontMetrics.leading; 
  28.          
  29.          
  30.         Log.d("fontMetrics""baseX    is:" + 0); 
  31.         Log.d("fontMetrics""baseY    is:" + 100); 
  32.         Log.d("fontMetrics""topY     is:" + topY); 
  33.         Log.d("fontMetrics""ascentY  is:" + ascentY); 
  34.         Log.d("fontMetrics""descentY is:" + descentY); 
  35.         Log.d("fontMetrics""bottomY  is:" + bottomY); 
  36.         Log.d("fontMetrics""leading  is:" + leading); 
  37.  
  38.          
  39.          
  40.         Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.fontmetrics); 
  41.         Bitmap mutableBitmap = bitmap.copy(Bitmap.Config.ARGB_8888, true); 
  42.          
  43.         canvas  = new Canvas(mutableBitmap); 
  44.          
  45.          
  46.  
  47.         // 绘制文本 
  48.         canvas.drawText(text, baseX, baseY, textPaint); 
  49.  
  50.         // BaseLine描画 
  51.         Paint baseLinePaint = new Paint( Paint.ANTI_ALIAS_FLAG); 
  52.          
  53.         baseLinePaint.setColor( Color.RED); 
  54.         canvas.drawLine(0, baseY, canvas.getWidth(), baseY, baseLinePaint); 
  55.  
  56.         // Base描画 
  57.         canvas.drawCircle( baseX, baseY, 5, baseLinePaint); 
  58.  
  59.         // TopLine描画 
  60.         Paint topLinePaint = new Paint( Paint.ANTI_ALIAS_FLAG); 
  61.         topLinePaint.setColor( Color.LTGRAY); 
  62.         canvas.drawLine(0, topY, canvas.getWidth(), topY, topLinePaint); 
  63.  
  64.         // AscentLine描画 
  65.         Paint ascentLinePaint = new Paint( Paint.ANTI_ALIAS_FLAG); 
  66.         ascentLinePaint.setColor( Color.GREEN); 
  67.         canvas.drawLine(0, ascentY, canvas.getWidth(), ascentY, ascentLinePaint); 
  68.  
  69.         // DescentLine描画 
  70.         Paint descentLinePaint = new Paint( Paint.ANTI_ALIAS_FLAG); 
  71.         descentLinePaint.setColor( Color.YELLOW); 
  72.         canvas.drawLine(0, descentY, canvas.getWidth(), descentY, descentLinePaint); 
  73.  
  74.         // ButtomLine描画 
  75.         Paint bottomLinePaint = new Paint( Paint.ANTI_ALIAS_FLAG); 
  76.         bottomLinePaint.setColor( Color.MAGENTA); 
  77.         canvas.drawLine(0, bottomY, canvas.getWidth(), bottomY, bottomLinePaint);  
  78.          
  79.         ImageView imageView = (ImageView) findViewById(R.id.imageView1); 
  80.         imageView.setImageBitmap(mutableBitmap); 
  81.          
  82.     } 

 log显示如下:

Android画图drawText入门(一)-字体结构_第2张图片

Note1:注意到各个数值都是整数,这是建立在baseY=100的情况下,去掉baseY,重新运行代码,log如下:

Android画图drawText入门(一)-字体结构_第3张图片

Note2: 参照线为baseline,即baseline=0的情况下,其他各线的数值。leading = 0,即行间距=0

2,以上是根据paint设置,获取相关的FontMetrics属性,并且只绘制了一行字符串,我们猜想,如果是多行,是否可以获得行间距leanding,代码如下:

  
  
  
  
  1. //test_multiply_lines 
  2.         TextView textView = (TextView) findViewById(R.id.textView1); 
  3.         String textMultiLines = "abcdefghijklmnopqrstuabcdefghijklmnopqrstuabcdefghijklmnopqrstuabcdefghijklmnopqrstuabcdefghijklmnopqrstu"
  4.         textView.setTextSize(55); 
  5.         textView.setText(textMultiLines); 
  6.          
  7.         FontMetrics fontMetrics2 = textView.getPaint().getFontMetrics(); 
  8.          
  9.          
  10.  
  11.         // 计算每一个坐标 
  12.  
  13.         float topY = fontMetrics2.top; 
  14.         float ascentY = fontMetrics2.ascent; 
  15.         float descentY =  fontMetrics2.descent; 
  16.         float bottomY = fontMetrics2.bottom; 
  17.         float leading =  fontMetrics2.leading; 
  18.          
  19.          
  20.  
  21.         Log.d("fontMetrics""topY     is:" + topY); 
  22.         Log.d("fontMetrics""ascentY  is:" + ascentY); 
  23.         Log.d("fontMetrics""descentY is:" + descentY); 
  24.         Log.d("fontMetrics""bottomY  is:" + bottomY); 
  25.         Log.d("fontMetrics""leading  is:" + leading); 

log如下:

Android画图drawText入门(一)-字体结构_第4张图片

Note:显然,即使是多行的情况下,仍不能获得leading。

3,如果text是单行,获得各个属性将会怎样,代码如下:

  
  
  
  
  1. String text = "abcdefghijklmnopqrstu"
  2.         TextView textView = (TextView) findViewById(R.id.textView1); 
  3.         textView.setTextSize(55); 
  4.         textView.setText(text); 
  5.          
  6.         FontMetrics fontMetrics = textView.getPaint().getFontMetrics(); 
  7.  
  8.         // 计算每一个坐标 
  9.         float baseX = 0
  10.         float baseY = 100
  11.         float topY = baseY + fontMetrics.top; 
  12.         float ascentY = baseY + fontMetrics.ascent; 
  13.         float descentY = baseY + fontMetrics.descent; 
  14.         float bottomY = baseY + fontMetrics.bottom; 
  15.         float leading =  fontMetrics.leading; 
  16.          
  17.          
  18.  
  19.         Log.d("fontMetrics""topY     is:" + fontMetrics.top); 
  20.         Log.d("fontMetrics""ascentY  is:" + fontMetrics.ascent); 
  21.         Log.d("fontMetrics""descentY is:" + fontMetrics.descent); 
  22.         Log.d("fontMetrics""bottomY  is:" + fontMetrics.bottom); 
  23.         Log.d("fontMetrics""leading  is:" + fontMetrics.leading); 

log如下图所示:

Android画图drawText入门(一)-字体结构_第5张图片

Note:与多行获得的属性都相同。

 

结论:

A:虽然paint和textView所设置的textSize均为55,且为相同的字符串,但是两个获得的FontMetrics属性值并不相同。但是,我们发现,做除法之后,均为1.5倍关系。做出猜测,即Paint下,为mdpi对应的size,而TextView的size已经关联到了显示屏幕本身的320dip。所以获得属性值均为整1.5倍数

B:各种情况下,均未获得leading值。

你可能感兴趣的:(Android画图drawText入门(一)-字体结构)