Android自定义字体设置

Android自定义字体设置

1.typeface、fontFamily、textStyle介绍

1.typeface 字体

typeface 枚举类型,值如下

  • normal(默认字体)
  • sans (无衬线字体)
  • serif(有衬线字体)
  • monospace(等宽字体)

sans 和 serif 介绍:
在西方国家罗马字母阵营中,字体分为两大种类:Serif和Sans Serif。
serif(有衬线字体):在字的笔划开始及结束的地方有额外的装饰,而且笔划的粗细会因直横的不同而有不同。
sans serif(无衬线字体):没有额外的装饰,笔划粗细大致差不多

Sans-serif.jpg

monospace 介绍:
打字机体虽然也属于Sans Serif,但由于是等宽字体,所以另外独立出Monospace这一种类
monospace (等宽字体):指每个字符宽度都一样。优点容易对齐,经常用来显示代码。

2.fontFamily 字型家族

什么是字型家族?它和typeface 的区别?要想了解首先得了解下什么是 typeface、font
typeface:字体,是一个抽象的总体概念(它是一款“设计”),例如:宋体、楷体
font:是指特定尺寸、特定字重、字偶间距等信息的一种 Typeface 的具体实现
fontFamily 是font的一个集合

使用提示可知 fontFamily 值如下

  • sans-serif
  • sans-serif-condensed
  • sans-serif-smallcaps
  • serif
  • serif-monospace
  • monospace
  • casual
  • cursive

3.textStyle 字体样式

textStyle 标记类型,值如下

  • normal(默认字体)
  • bold (加粗)
  • italic(斜体)

4.三者关系

查看TextView源码(API 27)可知:TextView构造方法大致如下

public TextView(Context context,AttributeSet attrs,int defStyleAttr,int defStyleRes) {
    ...
    setTypefaceFromAttrs(fontTypeface, fontFamily, typefaceIndex, styleIndex);
    ...
}

setTypefaceFromAttrs方法

/**
 * 
 * @param fontTypeface 要设置的 Typeface
 * @param familyName 要设置的 fontFamily
 * @param typefaceIndex 要设置的 typeface
 * @param styleIndex 要设置的 style
 */
private void setTypefaceFromAttrs(Typeface fontTypeface,String familyName,
                                    int typefaceIndex,int styleIndex) {
    Typeface tf = fontTypeface;
    if (tf == null && familyName != null) {
        // 有fontFamily时,用fontFamily
        tf = Typeface.create(familyName, styleIndex);
    } else if (tf != null && tf.getStyle() != styleIndex) {
        tf = Typeface.create(tf, styleIndex);
    }
    if (tf != null) {
        setTypeface(tf);
        return;
    }
    switch (typefaceIndex) {
        case SANS:
            tf = Typeface.SANS_SERIF;
            break;
    
        case SERIF:
            tf = Typeface.SERIF;
            break;
    
        case MONOSPACE:
            tf = Typeface.MONOSPACE;
            break;
    }
    // 其它都用typeface
    setTypeface(tf, styleIndex);
 }

总结:

1.typeface、fontFamily:都是设置字体,都设置时优先使用 fontFamily。
2.textStyle:设置字体的样式

2.设置自定义字体

方式1:直接设置

1.把字体ttf文件放到assets/fonts目录下,没有此目录手动创建


type1.png

2.设置

Typeface typeface = Typeface.createFromAsset(getAssets(), "fonts/aa.ttf");
textView.setTypeface(typeface);

3.优化
使用优化:继承自TextView,重写getTypeface(),返回自定义typeface
typeface创建优化:可用map封装后进行获取,代码略

Override
public Typeface getTypeface() {
   return Typeface.createFromAsset(getContext().getAssets(), "fonts/aa.ttf");
}

总结:

适用于:指定TextView

方式2:反射设置

1.把系统的typeface替换为自定义的

Typeface typeFace = Typeface.createFromAsset(getAssets(), "fonts/aa.ttf");
try {
    // xml属性值与Typeface属性值对应
    // normal      Typeface.DEFAULT
    // sans        Typeface.SANS_SERIF
    // serif       Typeface.SERIF
    // monospace   Typeface.MONOSPACE
   Field field = Typeface.class.getDeclaredField("SERIF" );
   field.setAccessible( true);
   field.set( null, typeFace );
} catch (NoSuchFieldException e) {
   e.printStackTrace();
} catch (IllegalAccessException e) {
   e.printStackTrace();
}

2.设置typeface值为 sans、 serif、 monospace其中一种,可在TextView、Activity、Application里面的xml样式里面设置

android:typeface="serif"

原理介绍:
1.xml设置完typeface为指定值后,最终xml会解析为代码执行(例如:设置android:typeface="serif"
最终会解析调用setTypeface(Typeface.SERIF))
2.预先把Typeface的某个系统字体通过反射强制替换自定义的Typeface(例如:把Typeface.SERIF强制替换为自定义的字体)
3.xml中设置typeface为已替换的字体即可(例如:android:typeface="serif")
举例说明:例如:xml设置字体android:typeface="serif",代码设置的字体是Typeface.SERIF,可是Typeface.SERIF里面的内容是强制替换后的自定义的字体
总结:

适用于:App字体样式不多的情况,因为你能替换的就SANS_SERIF、SERIF、MONOSPACE这几个值

方式3:fontFamily设置

1.把字体ttf文件放到res/fonts目录下,没有此目录手动创建或使用Android studio创建


type3.png

2.设置

1.TextView设置
android:fontFamily="@font/aa"
2.activity、application样式里设置
<item name="android:fontFamily">@font/aa</item>

3.兼容
因为 fontFamily API16 新增,所以要使用低版本的兼容库 com.android.support:appcompat-v7:26.0.0 以上

1.TextView设置
app:fontFamily="@font/aa"
2.activity、application样式里设置
<item name="fontFamily">@font/aa</item>

总结:

适用于:各种情况

Demo github地址: devblog androidFont

3.设置下载字体(了解)

为什么要下载字体:最主要的原因就是可以减少apk的体积,因为字体放到apk内会增大apk体积,通过下载就不会。
使用范围:此功能是Android 8.0(API26)新增,但是兼容库已兼容到API14,所以可放心使用
原理:所有的app都可以从字体提供app(例如:Google Play Services)里获取字体,字体提供app会把字体缓存到本地,所以如果多个app用相同的字体,它们则会共用同一个字体文件,优点:减少手机内存和磁盘空间,提高整体系统的运行状况。
使用前提:手机上必须有一个字体提供app,目前了解的国内没有,国外的只有Google Play Services(版本11及以上),由于目前国内很少安装Google Play Services,所以国内此功能目前没有用

方式1:Android studio创建可下载字体

Android studio 3.0及以上可下载,以下以Android studio 3.1.3介绍

1.选择more Fonts

more.png

2.选择创建可下载字体

select_font.png

3.点击ok,则会在font目录下创建对应的字体xml配置文件,内容如下:

xmlfont.png

创建证书文件font_certs.xml

font_certs.png

参数介绍:
fontProviderAuthority 字体提供app的权限
fontProviderPackage 字体提供app的包名,用于获取是哪个字体提供app
fontProviderQuery 字体提供app的字体名称,用于获取是哪个字体
fontProviderCerts 字体提供app的证书
总结:Android studio 生成的字体使用了兼容库,已兼容低版本;证书的生成由Android studio 自动生成,不用管理

4.第3步点击ok后,也会在AndroidManifest的application节点下生成如下内容:

preload.png

创建预加载字体文件preloaded_fonts.xml

preload_fonts.png

介绍:以上是声明使用预加载字体,如果没有使用预加载字体,则会在第一次设置可下载字体的时候,会先下载字体然后设置,这样会增加布局绘制时间,所以在AndroidManifest下声明要预加载的字体,则会在app启动时就加载即可解决这个问题,如果下载出现超时或无网等失败则会使用默认字体

方式2:代码创建可下载字体

以下使用了兼容库,参数和上面一致

// 创建请求
FontRequest request = new FontRequest(
        "com.google.android.gms.fonts",
        "com.google.android.gms",
        query,
        R.array.com_google_android_gms_fonts_certs);
        
// 请求回调
FontsContractCompat.FontRequestCallback callback = new FontsContractCompat.FontRequestCallback() {
    @Override
    public void onTypefaceRetrieved(Typeface typeface) {
        // 成功
        mDownloadableFontTextView.setTypeface(typeface);
    }

    @Override
    public void onTypefaceRequestFailed(int reason) {

    }
};
// 请求
FontsContractCompat.requestFont(MainActivity.this, request, callback, getHandlerThreadHandler());

Demo github地址: Downloadable Fonts sample app

总结:

适用于:不适用国内

个人博客地址: devbolg

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容