序
在 Android 下使用自定义字体已经是一个比较常见的需求了,最近也做了个比较深入的研究。
那么按照惯例我又要出个一篇有关 Android 修改字体相关的文章,但是写下来发现内容还挺多的,所以我决定将它们拆分一下,分几篇来详细的讲解。主要会是一些常用的替换字体的方案,最后还会介绍一些全局替换的方案,当然也会包含最新的 『Fonts in XML』的方案。
期待你持续关注。
本篇是本系列的第六篇,之前已经发布的文章,有兴趣可以先看看。
一、前言
上一篇讲解了通过替换 AppCompatDelegate 来达到替换控件的目的,从而替换成我们需要的可设置自定义字体的控件,来达到替换字体的目的。
现在大多数人应该看出来了,到最后实现的目标就是如何快速、低入侵的替换全局控件,然后对这些控件进行重写,就可以达到我们很多的目的。换字体只是这其中的一种应用,还有其它的,例如:换肤、无痕埋点等等,都是有可借鉴的地方的。
本文再介绍一种方式,通过 LayoutInflaterCompat.setFactory()
替换掉 LayoutInflaterFactory 或者 LayoutInflater.Factory2,来达到我们替换控件的目的,从而实现全局字体的替换。
接下来开始介绍所有的技术细节。
二、setFactory()
2.1 setFactory() 的技术原理
对大家而言,LayoutInflater 应该是不陌生的,所有需要动态加载 layout-xml 中的 View 的地方都需要用到它的 inflater()
方法,例如:ListView、RecyclerView。
而本文需要用到的是它另外两个 Api 方法,setFactory()
和 setFactory2()
。它们的方法签名如下。
这两个方法分别接收 Factory 和 Factory2 ,它们两个都是 Interface。并且这两个方法的功能也是类似的。只是 setFactory2()
是在 Api Level 11 之后引入的,使用那个取决于项目的 minSdkVersion。
不过一般而言,我们也不需要直接使用它。我们需要只用 Support.v4 包中,为我们提供的 LayoutInflaterCompat 这个兼容类来做处理。和所有的兼容类一样,它其中会有一个 IMPL的变量,会根据不同的 Api Level 初始化不同的实例。
可以看到,这里只对 Api Level 21 作为一个分界,去处理逻辑,其中会有不同的实现,这里有兴趣可以一探究竟,有时间会单独出一篇文章来讲解,这里就不再深入了。
这里,我们需要用到 LayoutInflaterCompat.setFactory()
方法,它实际上已经被标记为 @Deprecated
了,一般推荐我们使用 LayoutInflaterCompat.setFactory2()
,但是它们的功能是一致的,这里就不纠结这些细节了。
可以看到,setFactory()
接收一个 LayoutInflaterFactory 的对象,它实际上是一个接口,需要我们实现其中的 onCreateView()
方法。
我们这里主要的功能,就在于实现 onCreateView()
方法,将我们需要的控件在这个方法中替换掉。
2.2 举个例子
对着源码说太干了。下面我举个实际的例子,相信就可以说明问题了。
首先我新建一个 Activity,在 super.onCreate()
之前,通过 LayoutInflaterCompat 重新设置 Factory,在关键地方打印好 Log。
再声明一个布局,让它去显示 layout-XML 布局,层级很简单,就是一个 LinearLayout 中间包含了一个 TextView。
然后,我们运行起来看看输出的 Log ,这里撇开了 DecorView 等这些布局的打印,只看关键部分。
从 Log 输出可以看出,实际上,你所有布局的控件,都会经过 LayoutInflaterFactory.onCreateView()
方法走一遍,去实现初始化的过程,在其中可以有效的分辨出是什么控件,以及它有什么属性。
并且 onCreateView()
方法的返回值,就是一个 View,如果要替换该 View,可以在此处将其初始化后返回回去即可。
三、利用 LayoutInflater 替换字体
既然原理都清楚了,那么我们接下来就开始实际操作一下,如何通过替换 LayoutInflaterFactory 来达到替换控件,从而达到替换字体的目的。
首先,定义一个 Activity 为基类,其中在 super..onCreate()
方法之前,调用 LayoutInflaterCompat.setFactory()
,然后将它的替换为 我们自己定义的 CustomFontCompatDelegate 类。
CustomFontCompatDelegate 的实现,也非常的简单,只需要在它的 onCreateView()
方法中,替换掉 TextView 就可以。
其实,所有替换字体的逻辑,都在 FontTextView 中,接下来我们再看看 FontTextView 的逻辑。
可以看到,在 FontTextView 中,直接完整的将字体替换成我们在 assets 目录下存放的 custom_font.ttf
字体文件。
到这里就完成了基本的功能,我们接下来看看如何使用它。
只需要使用一个 Activity ,继承我们刚才实现的 CustomFontActivity,然后写一个简单的布局,其中有三个 TextView。
最后,我们再来看看运行后的效果。
四、小结
到这里基本上就介绍清楚如何通过 LayoutInflaterCompat.setFactory()
去替换 Factory 这个接口,达到我们替换控件的目的,从而完美的替换全局的字体。
但是实际开发过程中,依然需要考虑所有可以显示文字的控件,例如:TextView、EditText、Button 等等,这些都是我们需要重写的控件。