EDYCJA : Więc minęło trochę czasu i chciałbym dodać to, co uważam za najlepszy sposób, aby to zrobić, a nie mniej za pomocą XML!
Najpierw będziesz chciał utworzyć nową klasę, która zastąpi dowolny widok, który chcesz dostosować. (np. chcesz mieć przycisk z niestandardowym krojem czcionki? Rozszerz Button
). Zróbmy przykład:
public class CustomButton extends Button {
private final static int ROBOTO = 0;
private final static int ROBOTO_CONDENSED = 1;
public CustomButton(Context context) {
super(context);
}
public CustomButton(Context context, AttributeSet attrs) {
super(context, attrs);
parseAttributes(context, attrs); //I'll explain this method later
}
public CustomButton(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
parseAttributes(context, attrs);
}
}
Teraz, jeśli go nie masz, dodaj dokument XML pod res/values/attrs.xml
i dodaj:
<resources>
<!-- Define the values for the attribute -->
<attr name="typeface" format="enum">
<enum name="roboto" value="0"/>
<enum name="robotoCondensed" value="1"/>
</attr>
<!-- Tell Android that the class "CustomButton" can be styled,
and which attributes it supports -->
<declare-styleable name="CustomButton">
<attr name="typeface"/>
</declare-styleable>
</resources>
Okej, więc pomijając to, wróćmy do parseAttributes()
metody z wcześniejszej:
private void parseAttributes(Context context, AttributeSet attrs) {
TypedArray values = context.obtainStyledAttributes(attrs, R.styleable.CustomButton);
//The value 0 is a default, but shouldn't ever be used since the attr is an enum
int typeface = values.getInt(R.styleable.CustomButton_typeface, 0);
switch(typeface) {
case ROBOTO: default:
//You can instantiate your typeface anywhere, I would suggest as a
//singleton somewhere to avoid unnecessary copies
setTypeface(roboto);
break;
case ROBOTO_CONDENSED:
setTypeface(robotoCondensed);
break;
}
values.recycle();
}
Teraz wszystko gotowe. Możesz dodać więcej atrybutów do wszystkiego (możesz dodać kolejny dla stylu kroju - pogrubienie, kursywa, itp.), Ale teraz zobaczmy, jak go użyć:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:custom="http://schemas.android.com/apk/res/com.yourpackage.name"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<com.yourpackage.name.CustomButton
android:id="@+id/button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Click Me!"
custom:typeface="roboto" />
</LinearLayout>
xmlns:custom
Linia może być naprawdę coś, ale konwencja jest co pokazano powyżej. Liczy się to, że jest unikalny i dlatego używana jest nazwa pakietu. Teraz wystarczy użyć custom:
przedrostka dla swoich atrybutów i android:
przedrostka dla atrybutów Androida.
Ostatnia sprawa: jeśli chcesz to wykorzystać w stylu ( res/values/styles.xml
), należy nie dodać xmlns:custom
linię. Po prostu odwołaj się do nazwy atrybutu bez przedrostka:
<style name="MyStyle>
<item name="typeface">roboto</item>
</style>
(PREVIOUS ANSWER)
Używanie niestandardowego kroju pisma w systemie Android
To powinno pomóc. Zasadniczo nie ma sposobu, aby to zrobić w XML i, o ile wiem, nie ma prostszego sposobu na zrobienie tego w kodzie. Zawsze możesz mieć metodę setLayoutFont (), która tworzy krój pisma raz, a następnie uruchamia setTypeface () dla każdego. Wystarczy, że zaktualizujesz go za każdym razem, gdy dodasz nowy element do układu. Coś jak poniżej:
public void setLayoutFont() {
Typeface tf = Typeface.createFromAsset(
getBaseContext().getAssets(), "fonts/BPreplay.otf");
TextView tv1 = (TextView)findViewById(R.id.tv1);
tv1.setTypeface(tf);
TextView tv2 = (TextView)findViewById(R.id.tv2);
tv2.setTypeface(tf);
TextView tv3 = (TextView)findViewById(R.id.tv3);
tv3.setTypeface(tf);
}
EDYCJA : Więc po prostu zabrałem się za zaimplementowanie czegoś takiego samodzielnie, a jak skończyłem, zrobiłem taką funkcję:
public static void setLayoutFont(Typeface tf, TextView...params) {
for (TextView tv : params) {
tv.setTypeface(tf);
}
}
Następnie użyj tej metody z onCreate () i przekaż wszystkie TextViews, które chcesz zaktualizować:
Typeface tf = Typeface.createFromAsset(getAssets(), "fonts/BPreplay.otf");
//find views by id...
setLayoutFont(tf, tv1, tv2, tv3, tv4, tv5);
EDYCJA 9/5/12:
Ponieważ nadal uzyskuje się wyświetlenia i głosy, chciałbym dodać znacznie lepszą i bardziej kompletną metodę:
Typeface mFont = Typeface.createFromAsset(getAssets(), "fonts/BPreplay.otf");
ViewGroup root = (ViewGroup)findViewById(R.id.myrootlayout);
setFont(root, mFont);
/*
* Sets the font on all TextViews in the ViewGroup. Searches
* recursively for all inner ViewGroups as well. Just add a
* check for any other views you want to set as well (EditText,
* etc.)
*/
public void setFont(ViewGroup group, Typeface font) {
int count = group.getChildCount();
View v;
for(int i = 0; i < count; i++) {
v = group.getChildAt(i);
if(v instanceof TextView || v instanceof Button /*etc.*/)
((TextView)v).setTypeface(font);
else if(v instanceof ViewGroup)
setFont((ViewGroup)v, font);
}
}
Jeśli przekażesz mu katalog główny twojego układu, będzie on rekurencyjnie sprawdzał widoki TextView
lub Button
(lub inne, które dodasz do tej instrukcji if) w tym układzie i ustawi czcionkę bez konieczności określania ich przez identyfikator. To oczywiście zakłada, że chcesz ustawić czcionkę dla każdego widoku.