Android: 绘制的文本如何排版

项目需求,在一个特定的矩形内绘制一段文本,并且要让文本处于矩形的正中间。


原来的方法:

先设置paint颜色,字体,字体大小等等

paint.setColor(Color.WHITE);
paint.setTextSize(fontSize);


Path textPath = new Path();

textPath.moveTo(rect_x1,textPontY);
textPath.cubicTo(rect_x1,textPontY, rect_x1,textPontY, rect_x1+rect_width,textPontY);
paint.setTextAlign(Paint.Align.CENTER);
canvas.drawTextOnPath(mailBox.getBoxNumber(), textPath, 0, 0, paint);


关键在于这个 textPontY的值如何确定。如果定位不准,达到的效果就是在水平方向上它是居中的(3点一线),在垂直方向并不是


后来想到,通过path来设置,让path沿着一个封闭的矩形,如下:

Path textPath = new Path();
textPath.moveTo(rect_x1,rect_y1);
textPath.lineTo(rect_x1 + rect_width, rect_y1);
textPath.lineTo(rect_x1 + rect_width, rect_y1+rect_height);
textPath.lineTo(rect_x1 , rect_y1+rect_height);
textPath.lineTo(rect_x1,rect_y1);
textPath.close();

然后设置 paint.setTextAlign(Paint.Align.CENTER); ,然后再 canvas.drawTextOnPath(mailBox.getBoxNumber(), textPath, 0, 0, paint);

结果证明这个想法是错误的,它并不能智能定位到矩形正中间的位置



如何 textPontY的值如何确定 ?涉及到 “android字体大小对应的高度”


下文来自: http://www.iteye.com/topic/474526

Canvas 作为绘制文本时,使用FontMetrics对象,计算位置的坐标

它的思路和java.awt.FontMetrics的基本相同。

FontMetrics对象

它以四个基本坐标为基准,分别为:

・FontMetrics.top
・FontMetrics.ascent
・FontMetrics.descent
・FontMetrics.bottom

该图片将如下

Java代码
  1. PainttextPaint=newPaint(Paint.ANTI_ALIAS_FLAG);
  2. textPaint.setTextSize(35);
  3. textPaint.setColor(Color.WHITE);
  4. //FontMetrics对象
  5. FontMetricsfontMetrics=textPaint.getFontMetrics();
  6. Stringtext="abcdefghijklmnopqrstu";
  7. //计算每一个坐标
  8. floatbaseX=0;
  9. floatbaseY=100;
  10. floattopY=baseY+fontMetrics.top;
  11. floatascentY=baseY+fontMetrics.ascent;
  12. floatdescentY=baseY+fontMetrics.descent;
  13. floatbottomY=baseY+fontMetrics.bottom;
  14. //绘制文本
  15. canvas.drawText(text,baseX,baseY,textPaint);
  16. //BaseLine描画
  17. PaintbaseLinePaint=newPaint(Paint.ANTI_ALIAS_FLAG);>
  18. baseLinePaint.setColor(Color.RED);
  19. canvas.drawLine(0,baseY,getWidth(),baseY,baseLinePaint);
  20. //Base描画
  21. canvas.drawCircle(baseX,baseY,5,baseLinePaint);
  22. //TopLine描画
  23. PainttopLinePaint=newPaint(Paint.ANTI_ALIAS_FLAG);
  24. topLinePaint.setColor(Color.LTGRAY);
  25. canvas.drawLine(0,topY,getWidth(),topY,topLinePaint);
  26. //AscentLine描画
  27. PaintascentLinePaint=newPaint(Paint.ANTI_ALIAS_FLAG);
  28. ascentLinePaint.setColor(Color.GREEN);
  29. canvas.drawLine(0,ascentY,getWidth(),ascentY,ascentLinePaint);
  30. //DescentLine描画
  31. PaintdescentLinePaint=newPaint(Paint.ANTI_ALIAS_FLAG);
  32. descentLinePaint.setColor(Color.YELLOW);
  33. canvas.drawLine(0,descentY,getWidth(),descentY,descentLinePaint);
  34. //ButtomLine描画
  35. PaintbottomLinePaint=newPaint(Paint.ANTI_ALIAS_FLAG);
  36. bottomLinePaint.setColor(Color.MAGENTA);
  37. canvas.drawLine(0,bottomY,getWidth(),bottomY,bottomLinePaint);
-------------------------------------
由上可得出一个常识:drawText绘制文本时,是从baseLine这个位置开始的,并不简单认为是从文本的左下角开始绘制。

由上图可以分析出:

float temp = Math.abs(fontMetrics.top - fontMetrics.bottom);
float textPontY = rect_y1 + (rect_height - temp)/2.0f + temp - fontMetrics.bottom;


把 textPointY设置成这样的值,就对了!


更多相关文章

  1. Android中Gravity中的一些值都是些什么意思
  2. Android(安卓)JetPack Compose 入门
  3. 【Android】How Android(安卓)Draws Views
  4. android实现emoji输入
  5. android控件的属性
  6. android中ListView点击和ListView的item里面的button或ImageView
  7. Android图形框架简介
  8. Android实现图表绘制和展示
  9. Android(安卓)自定义View——自定义View控件

随机推荐

  1. Android(安卓)进程间通信-Intent、Messen
  2. Android(安卓)动画AlphaAnimation类方法
  3. React Native兼容iOS Android的TabBar
  4. Android(安卓)rom开发:webview崩溃问题Bin
  5. Android(安卓)GridView的使用
  6. android Data Backup(上)
  7. Android(安卓)进程间通信AIDL学习
  8. android actionbar+drawerLayout相关联
  9. Android(安卓)模拟器上的127.0.0.1 local
  10. android 手机内存SD卡和cpu等信息的获取