Android踩坑日记:Android字体属性及测量(FontMetrics)

Posted tuke_tuke

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Android踩坑日记:Android字体属性及测量(FontMetrics)相关的知识,希望对你有一定的参考价值。

android字体属性及测量(FontMetrics)

  • 字体的几个参数,以Android API文档定义为尊,见下图

要点如下:

  1. 基准点是baseline
  2. Ascent是baseline之上至字符最高处的距离
  3. Descent是baseline之下至字符最低处的距离
  4. Leading文档说的很含糊,其实是上一行字符的descent到下一行的ascent之间的距离
  5. Top指的是指的是最高字符到baseline的值,即ascent的最大值
  6. bottom指的是最下字符到baseline的值,即descent的最大值
为了帮助理解,我特此搜索了不同的示意图。对照示意图,会很容易理解FontMetrics的参数。

图1

图2

图3

图4

图5

图6

  • 测试
    我们使用自定义的View和Textview里的字符串作为研究对象
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="#000000"
    android:orientation="vertical"
    >
    <!--PaintView画字体-->
   <video.ketu.com.fontmeasure.PaintView
       android:id="@+id/v_fontview"
       android:layout_width="match_parent"
       android:layout_height="1dp"
       android:layout_weight="1"
       />
   <!--Textview设置文字-->
   <TextView
       android:id="@+id/tv_fontview1"
       android:layout_width="match_parent"
       android:layout_height="wrap_content"
       android:text="中国话fgiqÃÇŸŒú"
       android:textColor="@android:color/white"
       android:textSize="80px"
       android:layout_weight="1"/>
</LinearLayout>

MainActivity.class

public class MainActivity extends AppCompatActivity 
    TextView textView;
    View paintView;
    @Override
    protected void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        paintView = findViewById(R.id.v_fontview);
        textView = (TextView) findViewById(R.id.tv_fontview1);
        //String text = "中国话fgiqÃÇŸŒúcqazweyghnhgd;lc,kjssnhjjomoomcod";
        String text = "中国话fgiqÃÇŸŒú";
        /*设置字体带下80px*/
        textView.setTextSize(TypedValue.COMPLEX_UNIT_PX,80);
        textView.setText(text);
        Paint.FontMetrics fontMetrics = textView.getPaint().getFontMetrics();
        Log.d("textView", "fontMetrics.top        is:" + fontMetrics.top);
        Log.d("textView", "fontMetrics.ascent     is:" + fontMetrics.ascent);
        Log.d("textView", "fontMetrics.descent    is:" + fontMetrics.descent);
        Log.d("textView", "fontMetrics.bottom     is:" + fontMetrics.bottom);
        Log.d("textView", "fontMetrics.leading    is:" + fontMetrics.leading);
    

PaintView.class

public class PaintView extends View 
    private Paint mPaint = new Paint();
    public PaintView(Context context) 
        super(context);
    
    public PaintView(Context context, AttributeSet attrs) 
        super(context, attrs);
    
    public PaintView(Context context, AttributeSet attrs, int defStyle) 
        super(context, attrs, defStyle);
    
    @Override
    protected void onDraw(Canvas canvas) 
        // TODO Auto-generated method stub
        mPaint.reset();
        mPaint.setColor(Color.WHITE);
        /*设置字体带下80px*/
        mPaint.setTextSize(80);
        //设置字体为斜体
        //mPaint.setTypeface(Typeface.create("", Typeface.ITALIC));
        // FontMetrics对象
        Paint.FontMetrics fontMetrics = mPaint.getFontMetrics();
        String text = "中国话fgiqÃÇŸŒú";
        // 计算每一个坐标
        float textWidth = mPaint.measureText(text);
        float baseX = 0;
        float baseY = 100;
        float topY = baseY + fontMetrics.top;
        float ascentY = baseY + fontMetrics.ascent;
        float descentY = baseY + fontMetrics.descent;
        float bottomY = baseY + fontMetrics.bottom;
        float leading = baseY + fontMetrics.leading;

        Log.d("paintview", "baseX    is:" + baseX);
        Log.d("paintview", "baseY    is:" + baseY);

        Log.d("paintview", "fontMetrics.top        is:" + fontMetrics.top);
        Log.d("paintview", "fontMetrics.ascent     is:" + fontMetrics.ascent);
        Log.d("paintview", "fontMetrics.descent    is:" + fontMetrics.descent);
        Log.d("paintview", "fontMetrics.bottom     is:" + fontMetrics.bottom);
        Log.d("paintview", "fontMetrics.leading    is:" + fontMetrics.leading);

        Log.d("paintview", "topY     is:" + topY);
        Log.d("paintview", "ascentY  is:" + ascentY);
        Log.d("paintview", "descentY is:" + descentY);
        Log.d("paintview", "bottomY  is:" + bottomY);

截面图


PaintView和TextView设置的字体大小都是80px,Log打印结果

PaintView结果

TextView结果


      Note 1:以上可见,字体属性类的FontMetrics类的top,ascent,descent,bottom,leading的值是正负数,是以基线baseline为0的相对值。当baseline是100时,各个参数的坐标就是正数。
所以对于文本框的文字的行高:fontMetrics.top-fontMetrics.bottom

      Note 2:以上的TextView的文本长度是单行,获得leading=0,那么如果TextView的文本是多行,并且设置行间距后,leading的变化

public class MainActivity extends AppCompatActivity 
    TextView textView;
    View paintView;
    @Override
    protected void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        paintView = findViewById(R.id.v_fontview);
        textView = (TextView) findViewById(R.id.tv_fontview1);
        String text = "中国话fgiqÃÇŸŒúcqazweyghnhgd;lc,kjssnhjjomoomcod";
        //String text = "中国话fgiqÃÇŸŒú";
        /*设置字体带下80px*/
        textView.setTextSize(TypedValue.COMPLEX_UNIT_PX,80);
        textView.setText(text);

        Paint.FontMetrics fontMetrics = textView.getPaint().getFontMetrics();

        Log.d("textView", "fontMetrics.top        is:" + fontMetrics.top);
        Log.d("textView", "fontMetrics.ascent     is:" + fontMetrics.ascent);
        Log.d("textView", "fontMetrics.descent    is:" + fontMetrics.descent);
        Log.d("textView", "fontMetrics.bottom     is:" + fontMetrics.bottom);
        Log.d("textView", "fontMetrics.leading    is:" + fontMetrics.leading);
    
<TextView
       android:id="@+id/tv_fontview1"
       android:layout_width="match_parent"
       android:layout_height="wrap_content"
       android:text="中国话fgiqÃÇŸŒúcqazweyghnhgd;lc,kjssnhjjomoomcod"
       android:textColor="@android:color/white"
       android:lineSpacingExtra="5px"
       android:textSize="80px"
       android:layout_weight="1"/>


截面图

结果Log

Note:显然,即使是多行,并且设置5px行距的情况下,仍不能获得leading。

以上是关于Android踩坑日记:Android字体属性及测量(FontMetrics)的主要内容,如果未能解决你的问题,请参考以下文章

Android 踩坑日记 - RecyclerView 布局问题

Android踩坑日记:FloatingActionButton的设置大小问题

Android踩坑日记:FloatingActionButton的设置大小问题

Android踩坑日记:android7.0动态相机权限

Android踩坑日记:android7.0动态相机权限

android踩坑日记1