Android修改字体样式的示例代码

在Android实际开发中根据UI的设计图,经常要去改变系统默认的字体样式

这样做会使apk变大很多啊

而且为什么android要使用ios的字体-_-#

单独设置字体样式

(1)Android系统提供了几种字体样式可供选择

通过设置typeface属性或者fontFamily属性设置

typeface属性:

  1. normal
  2. serif
  3. sans
  4. monospace

fontFamily属性:

  1. casual
  2. cursive
  3. serif
  4. monospace
  5. sans-serif
  6. sans-serif-condensed
  7. serif-monospace
  8. sans-serif-smallcaps

※typeface和fontFamily区别

android:typeface属性是增加API1

android:fontFamily在API16(4.1)中添加了属性

※当同时设置typeface和fontFamily时,只有fontFamily生效

查看一波TextView的源码

  private void setTypefaceFromAttrs(String familyName, int typefaceIndex, int styleIndex) {
    Typeface tf = null;
    if (familyName != null) {
      tf = Typeface.create(familyName, 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;
    }

    setTypeface(tf, styleIndex);
  }

从方法setTypefaceFromAttrs()看,如果你有set fontFamily属性,那么typefaceattribute将被忽略。

这边会发现这样设置typeface和fontFamily属性对中文不生效,这时候就需要引用外部的字体样式(这里谷歌设计规范推荐使用NOTO字体https://www.google.com/get/noto/

(2)使用字体样式文件设置(otf,ttf文件都可以)

在assets下新建一个fonts文件,把字体样式文件放进去

在代码中设置

AssetManager mgr = getAssets();
Typeface tf = Typeface.createFromAsset(mgr, "fonts/NotoSansCJKsc-Black.otf");
tv_1.setTypeface(tf);

批量设置字体样式

(1)自定义TextView

public class CustomTextView extends TextView
{

  public CustomTextView(Context context, AttributeSet attrs)
  {
    super(context, attrs);
  }

  //重写设置字体方法
  @Override
  public void setTypeface(Typeface tf)
  {
    tf = Typeface.createFromAsset(getContext().getAssets(), "fonts/NotoSansCJKsc-Light.otf");
    super.setTypeface(tf);
  }
}

之后在XML布局文件中使用CustomTextView代替TextView

  <com.test.fontfamily.CustomTextView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:padding="6dp"
    android:text="自定义字体"
    android:textSize="24dp"
    />

(2)更换整个App的字体

思路:遍历找到所有的TextView然后替换字体

百度了一下找到下面工具类

package com.test.fontfamily;

import android.app.Application;
import android.content.Context;
import android.graphics.Typeface;
import android.support.annotation.NonNull;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;

import java.lang.ref.SoftReference;
import java.lang.reflect.Field;
import java.util.HashMap;
import java.util.Map;

/**
 * Created by Administrator on 2017/10/24.
 */

public class FontUtils
{

  private static final String TAG = FontUtils.class.getSimpleName();
  private Map<String, SoftReference<Typeface>> mCache = new HashMap<>();
  private static FontUtils sSingleton = null;

  public static Typeface DEFAULT = Typeface.DEFAULT;

  // disable instantiate
  private FontUtils()
  {
  }

  public static FontUtils getInstance()
  {
    // double check
    if (sSingleton == null)
    {
      synchronized (FontUtils.class)
      {
        if (sSingleton == null)
        {
          sSingleton = new FontUtils();
        }
      }
    }
    return sSingleton;
  }

  /**
   * <p>Replace the font of specified view and it's children</p>
   *
   * @param root   The root view.
   * @param fontPath font file path relative to 'assets' directory.
   */
  public void replaceFontFromAsset(@NonNull View root, @NonNull String fontPath)
  {
    replaceFont(root, createTypefaceFromAsset(root.getContext(), fontPath));
  }

  /**
   * <p>Replace the font of specified view and it's children</p>
   *
   * @param root   The root view.
   * @param fontPath font file path relative to 'assets' directory.
   * @param style  One of {@link Typeface#NORMAL}, {@link Typeface#BOLD}, {@link Typeface#ITALIC}, {@link Typeface#BOLD_ITALIC}
   */
  public void replaceFontFromAsset(@NonNull View root, @NonNull String fontPath, int style)
  {
    replaceFont(root, createTypefaceFromAsset(root.getContext(), fontPath), style);
  }

  /**
   * <p>Replace the font of specified view and it's children</p>
   *
   * @param root   The root view.
   * @param fontPath The full path to the font data.
   */
  public void replaceFontFromFile(@NonNull View root, @NonNull String fontPath)
  {
    replaceFont(root, createTypefaceFromFile(fontPath));
  }

  /**
   * <p>Replace the font of specified view and it's children</p>
   *
   * @param root   The root view.
   * @param fontPath The full path to the font data.
   * @param style  One of {@link Typeface#NORMAL}, {@link Typeface#BOLD}, {@link Typeface#ITALIC}, {@link Typeface#BOLD_ITALIC}
   */
  public void replaceFontFromFile(@NonNull View root, @NonNull String fontPath, int style)
  {
    replaceFont(root, createTypefaceFromFile(fontPath), style);
  }

  /**
   * <p>Replace the font of specified view and it's children with specified typeface</p>
   */
  private void replaceFont(@NonNull View root, @NonNull Typeface typeface)
  {
    if (root == null || typeface == null)
    {
      return;
    }

    if (root instanceof TextView)
    { // If view is TextView or it's subclass, replace it's font
      TextView textView = (TextView) root;
      // Extract previous style of TextView
      int style = Typeface.NORMAL;
      if (textView.getTypeface() != null)
      {
        style = textView.getTypeface().getStyle();
      }
      textView.setTypeface(typeface, style);
    } else if (root instanceof ViewGroup)
    { // If view is ViewGroup, apply this method on it's child views
      ViewGroup viewGroup = (ViewGroup) root;
      for (int i = 0; i < viewGroup.getChildCount(); ++i)
      {
        replaceFont(viewGroup.getChildAt(i), typeface);
      }
    } // else return
  }

  /**
   * <p>Replace the font of specified view and it's children with specified typeface and text style</p>
   *
   * @param style One of {@link Typeface#NORMAL}, {@link Typeface#BOLD}, {@link Typeface#ITALIC}, {@link Typeface#BOLD_ITALIC}
   */
  private void replaceFont(@NonNull View root, @NonNull Typeface typeface, int style)
  {
    if (root == null || typeface == null)
    {
      return;
    }
    if (style < 0 || style > 3)
    {
      style = Typeface.NORMAL;
    }

    if (root instanceof TextView)
    { // If view is TextView or it's subclass, replace it's font
      TextView textView = (TextView) root;
      textView.setTypeface(typeface, style);
    } else if (root instanceof ViewGroup)
    { // If view is ViewGroup, apply this method on it's child views
      ViewGroup viewGroup = (ViewGroup) root;
      for (int i = 0; i < viewGroup.getChildCount(); ++i)
      {
        replaceFont(viewGroup.getChildAt(i), typeface, style);
      }
    } // else return
  }

  /**
   * <p>Create a Typeface instance with specified font file</p>
   *
   * @param fontPath font file path relative to 'assets' directory.
   * @return Return created typeface instance.
   */
  private Typeface createTypefaceFromAsset(Context context, String fontPath)
  {
    SoftReference<Typeface> typefaceRef = mCache.get(fontPath);
    Typeface typeface = null;
    if (typefaceRef == null || (typeface = typefaceRef.get()) == null)
    {
      typeface = Typeface.createFromAsset(context.getAssets(), fontPath);
      typefaceRef = new SoftReference<>(typeface);
      mCache.put(fontPath, typefaceRef);
    }
    return typeface;
  }

  private Typeface createTypefaceFromFile(String fontPath)
  {
    SoftReference<Typeface> typefaceRef = mCache.get(fontPath);
    Typeface typeface = null;
    if (typefaceRef == null || (typeface = typefaceRef.get()) == null)
    {
      typeface = Typeface.createFromFile(fontPath);
      typefaceRef = new SoftReference<>(typeface);
      mCache.put(fontPath, typefaceRef);
    }
    return typeface;
  }

  /**
   * <p>Replace system default font. <b>Note:</b>you should also add code below to your app theme in styles.xml. </p>
   * {@code <item name="android:typeface">monospace</item>}
   * <p>The best place to call this method is {@link Application#onCreate()}, it will affect
   * whole app font.If you call this method after view is visible, you need to invalid the view to make it effective.</p>
   *
   * @param context {@link Context Context}
   * @param fontPath font file path relative to 'assets' directory.
   */
  public void replaceSystemDefaultFontFromAsset(@NonNull Context context, @NonNull String fontPath)
  {
    replaceSystemDefaultFont(createTypefaceFromAsset(context, fontPath));
  }

  /**
   * <p>Replace system default font. <b>Note:</b>you should also add code below to your app theme in styles.xml. </p>
   * {@code <item name="android:typeface">monospace</item>}
   * <p>The best place to call this method is {@link Application#onCreate()}, it will affect
   * whole app font.If you call this method after view is visible, you need to invalid the view to make it effective.</p>
   *
   * @param context {@link Context Context}
   * @param fontPath The full path to the font data.
   */
  public void replaceSystemDefaultFontFromFile(@NonNull Context context, @NonNull String fontPath)
  {
    replaceSystemDefaultFont(createTypefaceFromFile(fontPath));
  }

  /**
   * <p>Replace system default font. <b>Note:</b>you should also add code below to your app theme in styles.xml. </p>
   * {@code <item name="android:typeface">monospace</item>}
   * <p>The best place to call this method is {@link Application#onCreate()}, it will affect
   * whole app font.If you call this method after view is visible, you need to invalid the view to make it effective.</p>
   */
  private void replaceSystemDefaultFont(@NonNull Typeface typeface)
  {
    modifyObjectField(null, "MONOSPACE", typeface);
  }

  private void modifyObjectField(Object obj, String fieldName, Object value)
  {
    try
    {
      Field defaultField = Typeface.class.getDeclaredField(fieldName);
      defaultField.setAccessible(true);
      defaultField.set(obj, value);

    } catch (NoSuchFieldException e)
    {
      e.printStackTrace();
    } catch (IllegalAccessException e)
    {
      e.printStackTrace();
    }
  }
}

阅读源码发现这里面主要方法有

replaceFont()

替换一个页面中的所有字体。用递归的方式去查找view是否是TextView或者TextView的子类,然后进行替换。不过这种方法效率不高

用法:在页面中

FontUtils.getInstance().replaceFontFromAsset(View root, String fontPath)

modifyObjectField()

替换App所有字体。利用反射替换系统默认字体

用法:

a.新建一个BaseApplication继承Application在onCreate方法中

FontUtils.getInstance().replaceSystemDefaultFontFromAsset(this,"fonts/NotoSansCJKsc-Thin.otf");

b.在AppTheme中添加

<item name="android:typeface">monospace</item>

c.清单文件中使用BaseApplication

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android修改自己程序字体的方法详解

    Android提供三种字体:"Sans","serif"和"monospace".1.在Android XML文件中设置字体可以采用android:typeface,例如android:typeface="monospace".在这里例子中我们在Activity中对android:text="Hello, World! 您好"分别进行了四种显示方式,依次为"Sans","ser

  • android 设置控件的颜色字体的方法

    1.用代码设置控件的颜色: 复制代码 代码如下: int b =  getResources().getColor(R.drawable.blue);//得到配置文件里的颜色    mButton.setTextColor(b); 2.设置空间的字体:方式一:mText.setTypeface(Typeface.createFromAsset(getAssets(),"fonts/HandmadeTypewriter.ttf"));//设置字体   注意:1.保证文件一定是ttf格式:

  • Android字体设置及Roboto字体使用方法

    本文实例讲述了Android字体设置及Roboto字体使用方法.分享给大家供大家参考.具体分析如下: 一.自定义字体 1.android Typeface使用TTF字体文件设置字体 我们可以在程序中放入ttf字体文件,在程序中使用Typeface设置字体. 第一步,在assets目录下新建fonts目录,把ttf字体文件放到这. 第二步,程序中调用: 复制代码 代码如下: AssetManager mgr=getAssets();//得到AssetManager Typeface tf=Type

  • Android TextView字体颜色设置方法小结

    本文实例总结了Android TextView字体颜色设置方法.分享给大家供大家参考,具体如下: 对于setTextView(int a)这里的a是传进去颜色的值.例如,红色0xff0000是指0xff0000如何直接传入R.color.red是没有办法设置颜色的,只有通过文章中的第三种方法先拿到资源的颜色值再传进去. tv.setTextColor(this.getResources().getColor(R.color.red)); 关键字: android textview color T

  • Android使用selector修改TextView中字体颜色和背景色的方法

    本文实例讲述了Android使用selector修改TextView中字体颜色和背景色的方法.分享给大家供大家参考,具体如下: android中的selector大家都很熟悉了,用它可以很方便的实现,控件在不同的动作中,颜色等值的变化.这里我说一下TextView中的一些应用. 我想大家都知道,Button按钮在源码上看是一种特殊的TextView,所以我们很多时候,按钮全是使用的TextView来完成,只要加一个android:clickable="true"就可以了. TextVi

  • android 字体颜色选择器(ColorPicker)介绍

    primary_text_yellow.xml 复制代码 代码如下: <?xml version="1.0" encoding="utf-8"?> <!-- Copyright (C) 2008 The Android Open Source Project Licensed under the Apache License, Version 2.0 (the "License"); you may not use this f

  • android根据分辨率自动调整字体大小的实例代码

    手机设备太多,分辨率也不一样,看到网上大部分的适应字体的方法是定义values320×480或value-hdpi方式去处理.采用第一种的就惨了,很多设备的分辨率是不一样的,难道要每种都定义吗?采用第二种的在平板电脑里没有效果. 最后还是代码的方式方便快捷... [java] 复制代码 代码如下: //遍历设置字体  public static void changeViewSize(ViewGroup viewGroup,int screenWidth,int screenHeight) {/

  • Android编程实现自动调整TextView字体大小以适应文字长度的方法

    本文实例讲述了Android编程实现自动调整TextView字体大小以适应文字长度的方法.分享给大家供大家参考,具体如下: package com.test.android.textview; import android.content.Context; import android.graphics.Paint; import android.util.AttributeSet; import android.widget.TextView; public class CustomTextV

  • Android修改字体样式的示例代码

    在Android实际开发中根据UI的设计图,经常要去改变系统默认的字体样式 这样做会使apk变大很多啊 而且为什么android要使用ios的字体-_-# 单独设置字体样式 (1)Android系统提供了几种字体样式可供选择 通过设置typeface属性或者fontFamily属性设置 typeface属性: normal serif sans monospace fontFamily属性: casual cursive serif monospace sans-serif sans-serif

  • Android 快速实现状态栏透明样式的示例代码

    在手机 app 开发过程中,经常会遇到一种需求,需要将 内容区域 顶到 状态栏 中去.这个时候,下面一段代码,就能很轻松解决问题了. 上代码之前先上效果图: 下面上一段代码: getWindow().requestFeature(Window.FEATURE_NO_TITLE); if(VERSION.SDK_INT >= VERSION_CODES.LOLLIPOP) { Window window = getWindow(); window.clearFlags(WindowManager.

  • Android修改Dialog样式的方法

    一.Dialog源码解析 1.1 new AlertDialog.Builder(this).create() protected AlertDialog(@NonNull Context context, @StyleRes int themeResId) { super(context, resolveDialogTheme(context, themeResId)); //创建AlertController,是Dialog布局相关代码 mAlert = new AlertControlle

  • Python设置Word全局样式和文本样式的示例代码

    目录 全局样式的定义 文本样式的定义 上一章节我们学习了如何生成 word 文档以及在文档行中添加各种内容,今天我们基于上一章节的内容进行添砖加瓦 —> 对内容进行各种样式的设置,让其能够看起来更加的美观. 全局样式的定义 通过全局样式的设置,可以使得 word 全文都可以继承这样的样式效果: 使用方法: style = document_obj.styles['Normal'] 通过 Document 对象调用 styles 对象集,通过中括号的方式选择全局样式,获得 样式对象 . for s

  • JS获取和修改元素样式的实例代码

    1.获取元素样式: 可以通过元素的style属性,获取元素行内样式.style属性是一个对象,包括一系列样式属性.例如:color, backgourdColor. 上面讲的通过style属性获取元素样式,不推荐使用. 下面的一段代码,可以获取元素运行时的样式,即全局的样式.这种方式可以动态获取元素的样式,例如元素大小. // node:将要获取其计算样式的元素节点 // attr: 样式属性名称 function getCurrentStyle(node, attr) { var style

  • Android按钮美化样式的实现代码

    话不多说,上运行效果图 在drawable文件夹下 新建button_drawable.xml <?xml version="1.0" encoding="utf-8" ?> <!--相当于做了一张圆角的图片,然后给button作为背景图片--> <shape xmlns:android="http://schemas.android.com/apk/res/android" android:shape="

  • Android实现微信登录的示例代码

    目录 一.布局界面 二.MainActivity.java 微信登录的实现与qq登录类似.不过微信登录比较麻烦,需要拿到开发者资质认证,花300块钱,然后应用的话还得有官网之类的,就是比较繁琐的前期准备工作,如果在公司里,这些应该都不是事,会有相关人提前准备好.在这里我们已经拿到了开发者认证,并且申请到了微信登录的授权. 现在直接介绍mob来实现微信登录的代码,并获取微信的相关数据,比较简单. 一.布局界面 布局界面只需要一个button来触发授权就可以 <Button android:id=&qu

  • Android绘制平移动画的示例代码

    目录 1.具体操作步骤 2.具体实施 创建ImageView 创建ObjectAnimator对象 3.具体实例 activity_main.xml MainActivity.java 1.具体操作步骤 创建ImageView对象 创建ObjectAnimator对象 通过ofFloat方法实现平移 2.具体实施 创建ImageView <ImageView android:id="@+id/car" android:layout_width="wrap_content

  • android开发权限询问的示例代码

    现在基于信息安全问题,特别是版本是23以上权限越严格. 特别是拍照,读,写权限 一般权限允许过,下次就不用询问了的,所以很多应用都喜欢在首页或者启动页直接询问,不允许的就用不了1.下面给出封装好的类,至于什么时候调看项目需要 public class EasyPermissions { private static final String TAG = "EasyPermissions"; public interface PermissionCallbacks extends Act

  • Android本地视频压缩方案的示例代码

    前言 本文讨论的不是类似秒拍的短视频录制,而是用户选择本地一个现有视频,压缩后上传.秒拍的实现其实是自定义视频录制功能,从而控制录制时长,分辨率,码率等,生成体积很小的视频再上传.而我们则没办法控制原视频的参数,可能是一个很大的视频需要压缩处理. 思路 利用ffmpeg对视频转码,通过设定参数生成分辨率和码率更小的视频,实现压缩.当然,ffmpeg的功能远不止如此,这是一个很大的专题. 用到的开源库:https://github.com/WritingMinds/ffmpeg-android-j

随机推荐