Android: TextView 及其子类通过代码和 XML 设置字体大小的存在差异的分析
原因: 在代码中通过 setTextSize(float size) 设置,使用的是 sp 为默认单位。 而 XML 中使用了 px,所以需要使用先把做好 sp 和 px 的转换工作。
最近在做 app 内修改字体大小,同时在设置页面有个预览界面,这个时候需要通过代码设置字体大小了
1 tv.setTextSize(getResources().getDimension(R.dimen.XLargeTextSize));
结果发现比通过 XML 设置的字体大小要大不少,看到了 setTextSize() 的源码,发现了问题所在。
1. setTextSize(), 该方法调用的是 setTextSize(TypedValue.COMPLEX_UNIT_SP, size),看到了使用了一个尺寸单位 TypedValue.COMPLEX_UNIT_SP
1 @android.view.RemotableViewMethod 2 public void setTextSize(float size) { 3 setTextSize(TypedValue.COMPLEX_UNIT_SP, size); 4 }
跟进到 TypedValue 中,查看到常量,根据输入不同的单位,解析不同的 size,然后确定文字大小
查看到有三个常量:(px, dip(dp), sp)
1 /** {@link #TYPE_DIMENSION} complex unit: Value is raw pixels. */ 2 public static final int COMPLEX_UNIT_PX = 0; 3 /** {@link #TYPE_DIMENSION} complex unit: Value is Device Independent 4 * Pixels. */ 5 public static final int COMPLEX_UNIT_DIP = 1; 6 /** {@link #TYPE_DIMENSION} complex unit: Value is a scaled pixel. */ 7 public static final int COMPLEX_UNIT_SP = 2;
2. setTextSize(int unit, float size),在这个方法中,解析了 TypedValue 中的常量,解析方法为 applyDimension()
1 public void setTextSize(int unit, float size) { 2 Context c = getContext(); 3 Resources r; 4 5 if (c == null) 6 r = Resources.getSystem(); 7 else 8 r = c.getResources(); 9 10 setRawTextSize(TypedValue.applyDimension( 11 unit, size, r.getDisplayMetrics())); 12 }
3. TypedValue.applyDomension(),根据不同的单位,返回不同的 value。 1. 选择 px 为单位,直接返回输入的 value; 2. 选择 dp 为单位 ,返回为 value * density;3. 选择 sp 为单位,返回 value* scaledDensity
1 public static float applyDimension(int unit, float value, 2 DisplayMetrics metrics) 3 { 4 switch (unit) { 5 case COMPLEX_UNIT_PX: 6 return value; 7 case COMPLEX_UNIT_DIP: 8 return value * metrics.density; 9 case COMPLEX_UNIT_SP: 10 return value * metrics.scaledDensity; 11 case COMPLEX_UNIT_PT: 12 return value * metrics.xdpi * (1.0f/72); 13 case COMPLEX_UNIT_IN: 14 return value * metrics.xdpi; 15 case COMPLEX_UNIT_MM: 16 return value * metrics.xdpi * (1.0f/25.4f); 17 } 18 return 0; 19 }
4. setRawTextSize(),这个方法中,调用了 mTextPaint.setTextSize(), mTextPaint 是 TextPaint 的实例, TextPaint 继承 Paint,可以看到就是调用的 Paint 的 setTextSize() 方法。
1 private void setRawTextSize(float size) { 2 if (size != mTextPaint.getTextSize()) { 3 mTextPaint.setTextSize(size); 4 5 if (mLayout != null) { 6 nullLayouts(); 7 requestLayout(); 8 invalidate(); 9 } 10 } 11 }