Android全局替换字体

一、概述

  ps:上次说的LayoutInflaterCompat.setFactory2来全局替换字体的方案,在使用过程中发现存在缺陷。今天补充一个优化版方案

    LayoutInflaterCompat.setFactory2这个方案的好处是比较方便,全局替换,也可以做一些个性化的操作,如:修改字体填充、随意更改样式,可以就不同控件的字体单独区分设置,比较灵活。

    但是:其会导致布局的加载速度变慢。布局稍微复杂一点,这个加载速度是难以忍受的。

  今天另一种方案来替换全局字体:

    通过配置应用主题,全局替换默认字体

    步骤:

      1.在res文件夹下新建一个font文件夹,然后把你需要使用的字体库放进去

 

      2.在Application中的theme中应用你的主体

 <!-- Base application theme. -->
    <style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
        <!-- Customize your theme here. -->
        <item name="colorPrimary">@color/colorPrimary</item>
        <item name="colorPrimaryDark">@color/colorPrimaryDark</item>
        <item name="colorAccent">@color/colorAccent</item>
        <!-- 在这里设置全局字体 -->
        <item name="android:fontFamily">@font/source_han_sans_cn_bold</item>
        <item name="android:includeFontPadding">false</item>
    </style>

      3.搞定

 

 

  由于业务需要,各端之间统一字体(Android、IOS、PC、网页)。所以android也需要替换成特定的字体。

  以后有可能还会增加其他的字体。

  方案:

    使用LayoutInflaterCompat.setFactory2来全局替换字体。这样做的好处是可以一次性的替换大部分的字体。剩余的个性化字体再单独适配。

      这样效率最高,最省时间。

二、代码示例

  1.在BaseActivity中加入

//全局字体设置
LayoutInflaterCompat.setFactory2(getLayoutInflater(), getLayoutFactory2());

  2.自定义一个类继承LayoutInflater.Factory2接口

private LayoutInflater.Factory2 getLayoutFactory2() {
        return new TextViewFontProxy();
    }

    private class TextViewFontProxy implements LayoutInflater.Factory2 {

        @Nullable
        @Override
        public View onCreateView(@Nullable View parent, @NonNull String name, @NonNull Context context, @NonNull AttributeSet attrs) {
            return getProxyView(name, context, attrs);
        }

        @Nullable
        @Override
        public View onCreateView(@NonNull String name, @NonNull Context context, @NonNull AttributeSet attrs) {
            return getProxyView(name, context, attrs);
        }
    }

    private View getProxyView(@NonNull String name, @NonNull Context context, @NonNull AttributeSet attrs){
        switch(name){
            case "TextView":
                return new SiYuanBlackBodyTextView(context, attrs);
            case "EditText":
                return new SiYuanBlackBodyEditText(context, attrs);
        }
        return null;
    }

  3.自定义一个字体类

/**
 * 思源黑体,自定义字体
 */
class SiYuanBlackBodyTextView : AppCompatTextView {
    private val fontPath = "font/source_han_sans_cn_bold.otf"

    constructor(context: Context, attrs: AttributeSet?, defStyleAttr: Int) : super(
        context,
        attrs,
        defStyleAttr
    ) {
        setTypeFace()
    }

    constructor(context: Context, attrs: AttributeSet?) : super(context, attrs) {
        setTypeFace()
    }

    constructor(context: Context) : super(context) {
        setTypeFace()
    }

    private fun setTypeFace() {
        try {
            includeFontPadding = false
            val typeface = Typeface.createFromAsset(context.assets, fontPath)
            setTypeface(typeface)
        } catch (ignored: Throwable) {
        }
    }
}

 

三、问题补充

  这里需要注意一下字体的上下填充,思源合体默认上下填充是非常大的。如果不做适配直接回影响全局的布局。

  这里在设置字体的时候加入includeFontPadding = false即可。作用:去掉字体上下填充

posted on 2024-07-09 11:14  飘杨......  阅读(754)  评论(0)    收藏  举报