如果通过代码(在特定设备上)创建TextView,为什么字体不适用

时间:2019-09-04 10:05:09

标签: android xml

我在项目中更改了自定义字体。在模拟器和许多设备上都可以正常工作,但是在某些设备上,在某些情况下(例如,当我从代码创建TextView时)不起作用。

我有一个演示此问题的示例项目。

这是我的样式文件

<resources>
    <style name="TextStyle.Caption" parent="TextAppearance.AppCompat.Caption">
        <item name="android:textSize">12sp</item>
        <item name="android:fontFamily">@font/eesti_pro_display_bold</item>
        <item name="android:letterSpacing">0.03</item>
    </style>
</resources>

这是我的xml。在这里,我有带有textAppearance的TextView

<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Hello World!"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        android:textAppearance="@style/TextStyle.Caption"/>

</androidx.constraintlayout.widget.ConstraintLayout>

我在其中创建另一个TextView的活动

class MainActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        val layout: ConstraintLayout = findViewById(R.id.container)
        createItemTitleTextView(this.baseContext).apply {
            layout.addView(this)
            text = "Hello World from code!"
        }
    }
}

private fun createItemTitleTextView(context: Context) = TextView(context).apply {
    TextViewCompat.setTextAppearance(this, R.style.TextStyle_Caption)
}

模拟器上的结果(预期结果):

Emulator result

在Samsung Galaxy J5上的结果。斜体字体是Galaxy系统字体。 请注意,我的字体在xml中有效,但在代码中无效

enter image description here

编辑

我知道我可以使用setTypeface,并且一切都按预期工作,但是我想知道为什么setTextAppearance不起作用。因为我只想更改样式文件,而不是整个项目,而是将setTypeface从代码添加到每个TextView中

2 个答案:

答案 0 :(得分:1)

在Android 8.0之前,fontFamily仅支持默认值:

android:fontFamily="sans-serif"           // roboto regular  
android:fontFamily="sans-serif-light"     // roboto light  
android:fontFamily="sans-serif-condensed" // roboto condensed  
android:fontFamily="sans-serif-thin"      // roboto thin (android 4.2)  
android:fontFamily="sans-serif-medium"    // roboto medium (android 5.0)

https://developer.android.com/guide/topics/ui/look-and-feel/fonts-in-xml

对于自定义字体,您需要下载ttf文件并按以下方式使用它:

  Typeface typeface = Utils.getTypeface(context, textStyle);
    if (typeface != null) setTypeface(typeface);

,Utils文件中的功能为:

public static Typeface getTypeface(Context context, int textStyle) {
    try {
        String font = "fonts/font-file-name.ttf"
        Typeface myTypeface = Typeface.createFromAsset(context.getAssets(), font);
        return myTypeface;
    } catch (Exception e) {
        Log.d(Constants.TAG, "OpenSansTextView init: ");
        e.printStackTrace();
    }
    return null;
}

我的ttf文件的位置:

app\src\main\assets\fonts\font-file-name.ttf

编辑(替代解决方案)

如果您已经有太多的TextView,您无法在Java的任何地方添加“ setTypeface”, 然后,只需按Ctrl + Shift + R并将<TextView替换为<com.your.package.app.MyTextView。  您可以按以下方式创建自定义TextView:

package com.your.package.app;

import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Typeface;
import android.util.AttributeSet;
import android.util.Log;
import android.view.Gravity;


public class MyTextView extends android.support.v7.widget.AppCompatTextView {
    Context context;
    Integer textStyle = 0;
    String textOS;

    public MyTextView(Context context) {
        super(context);
        this.context = context;
        init(null);
    }

    public MyTextView(Context context, AttributeSet attrs) {
        super(context, attrs);
        this.context = context;
        init(attrs);
    }

    public MyTextView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        this.context = context;
        init(attrs);
    }

    private void init(AttributeSet attrs) {
        try {
             String font = "fonts/font-file-name.ttf"
             Typeface myTypeface = Typeface.createFromAsset(context.getAssets(), font);
             if (typeface != null) setTypeface(typeface);
        } catch (Exception e) {
            Log.d(Constants.TAG, "MyTextView init: ");
            e.printStackTrace();
        }
    }

}

答案 1 :(得分:0)

TextViewCompat.setTextAppearance(tv, R.style.TextStyle_Caption)不起作用,因为我创建了TextView。但是,我需要创建 AppCompatTextView ,一切都会按预期运行