Wie kann ich die benutzerdefinierte Schriftart programmgesteuert auf Spinner-Text einstellen?

93

Ich habe eine ttf-Schriftartdatei in meinem Assets-Ordner. Ich weiß, wie man es für Textansichten verwendet mit:

Typeface externalFont=Typeface.createFromAsset(getAssets(), "fonts/HelveticaNeueLTCom-Lt.ttf");
textview1.setTypeface(externalFont);

Ich habe die Suche nach meinem Spinner-Text in einer eigenen XML-Datei definiert (wie in Android üblich):

<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android" 
android:id="@+android:id/text1"
style="?android:attr/spinnerItemStyle"
android:singleLine="true"
android:textColor="#ffffff"
android:gravity="center" 
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:ellipsize="marquee" />

Ich kann diese Textansicht einfach nicht aus dem Code referenzieren, ich bekomme immer Nullzeigerausnahmen. ZB habe ich versucht:

TextView spinner_text=(TextView)findViewById(R.id.text1);
spinner_text.setTypeface(externalFont);

Ist es möglich, meine externe Schriftart auch für meinen Spinnertext auszuwählen, der in einer eigenen XML definiert ist?

Danke dir.

BEARBEITEN mit Antwort:

Das funktioniert:

String [] items = new String[2];
    items[0]="Something1";
    items[1]="Something2";

ArrayAdapter<String> adapter = new ArrayAdapter<String>(this,
                    R.layout.spinaca, items) {

         public View getView(int position, View convertView, ViewGroup parent) {
                 View v = super.getView(position, convertView, parent);

                 Typeface externalFont=Typeface.createFromAsset(getAssets(), "fonts/HelveticaNeueLTCom-Lt.ttf");
                 ((TextView) v).setTypeface(externalFont);

                 return v;
         }


         public View getDropDownView(int position,  View convertView,  ViewGroup parent) {
                  View v =super.getDropDownView(position, convertView, parent);

                 Typeface externalFont=Typeface.createFromAsset(getAssets(), "fonts/HelveticaNeueLTCom-Lt.ttf");
                 ((TextView) v).setTypeface(externalFont);
                 v.setBackgroundColor(Color.GREEN);

                 return v;
         }
 };


     adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);                                 
     spinner.setAdapter(adapter);

Es kann notwendig sein, hinzuzufügen

import android.view.ViewGroup;

Zu Ihrer Liste der Importe oben in Ihrer Datei. Aus irgendeinem Grund macht Eclipse diesen Vorschlag nicht, wenn die im Code enthaltene ViewGroup-Klasse nicht erkannt wird.

DixieFlatline
quelle
2
Vielen Dank Freund .. Nach einem langen Kampf habe ich dies gefunden. Dies rettete meinen Tag atlast
Andro Selva
Danke für die Antwort!
CommonSenseCode
Sehr schöne Fragen ...
Rahul Kushwaha

Antworten:

81

Dies hat bei mir funktioniert (unter Verwendung von Ideen aus den Antworten von CommonsWare und gsanllorente ):

private static class MySpinnerAdapter extends ArrayAdapter<String> {
    // Initialise custom font, for example:
    Typeface font = Typeface.createFromAsset(getContext().getAssets(),
                        "fonts/Blambot.otf");

    // (In reality I used a manager which caches the Typeface objects)
    // Typeface font = FontManager.getInstance().getFont(getContext(), BLAMBOT);

    private MySpinnerAdapter(Context context, int resource, List<String> items) {
        super(context, resource, items);
    }

    // Affects default (closed) state of the spinner
    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        TextView view = (TextView) super.getView(position, convertView, parent);
        view.setTypeface(font);
        return view;
    }

    // Affects opened state of the spinner
    @Override
    public View getDropDownView(int position, View convertView, ViewGroup parent) {
        TextView view = (TextView) super.getDropDownView(position, convertView, parent);
        view.setTypeface(font);
        return view;
    }
}

Wenn Sie, wie ich, den Spinner ursprünglich mit ArrayAdapter.createFromResource()und einer Array-Ressource (wie in der Spinner-Dokumentation ) gefüllt haben, würden Sie MySpinnerAdapter wie folgt verwenden:

MySpinnerAdapter<String> adapter = new MySpinnerAdapter(
        getContext(),
        R.layout.view_spinner_item,
        Arrays.asList(getResources().getStringArray(R.array.my_array))
);
spinner.setAdapter(adapter);
Jonik
quelle
1
Toll! Ich ging noch weiter und erstellte eine assignAdapterWithOptions(Spinner spinner, int textArrayResId)Methode, um den Kontext zu ermitteln spinner.getContext()und den Adapter dem darin enthaltenen Spinner zuzuweisen (Spinner-Layouts sind für meine gesamte App
besser
Dies hilft mir sehr. Danke ... @ Jonik
Rahul Kushwaha
24

Sie würden die Schriftart über Ihre eigene Gewohnheit SpinnerAdapterin getView()und anwenden getDropDownView().

CommonsWare
quelle
Ich habe meine Frage mit meinem neuesten Problem bearbeitet. Können Sie mir bitte sagen, was ich falsch mache? Tnx
DixieFlatline
1
@DixieFlatline: Sie müssen einen Import für android.view.ViewGroupvermutlich
CommonsWare
15

Wenn Sie Ihren Adapter in einer anderen Datei implementieren, können Sie über den Konstruktor des Adapters auf die Funktion "getAssets ()" zugreifen, da Sie den Kontext als Parameter haben.

public class YourItemAdapter extends ArrayAdapter<String> {
int recurso;
Typeface tf;

public YourItemAdapter(Context _context, int _resource,
        List<String> _items) {

    super(_context, _resource, _items);
    recurso=_resource;
    tf=Typeface.createFromAsset(_context.getAssets(),"font/digital-7.ttf");
}

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    //You can use the new tf here.
    TextView spinner_text=(TextView)findViewById(R.id.text1);
    spinner_text.setTypeface(tf);
    }
}
gsanllorente
quelle
Dieser hilft wirklich, danke :)
Jashan PJ
Dies war irgendwie nützlich, obwohl es bei mir nicht funktionierte: Es findViewById(R.id.text1)schien die Textansicht nicht zu finden, obwohl die ID korrekt war. Einige andere Probleme in diesem Code: 1) fehlende return-Anweisung in getView(), 2) nicht verwendetes Feld recurso, 3) einige Stilprobleme wie das Benennen einer Variablen spinner_text(sollte sein spinnerText). Folgendes hat bei mir funktioniert .
Jonik
4

Versuchen Sie dies, erstellen Sie eine benutzerdefinierte custom_spinner.xml

<?xml version="1.0" encoding="utf-8"?>

<com.xxxx.xxxx.CheckedTextViewC

    xmlns:android="http://schemas.android.com/apk/res/android" 
    android:id="@android:id/text1"
    style="?android:attr/spinnerDropDownItemStyle"
    android:singleLine="true"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:ellipsize="marquee"
    android:textAlignment="center"
    android:paddingTop="5dp"
    android:paddingBottom="5dp"
    android:textSize="18sp"

    />

Erstellen Sie eine benutzerdefinierte CheckedtextView wie folgt

import android.content.Context;
import android.graphics.Typeface;
import android.util.AttributeSet;
import android.widget.CheckedTextView;

public class CheckedTextViewC extends CheckedTextView {

    public CheckedTextViewC(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        // TODO Auto-generated constructor stub
    }
    public CheckedTextViewC(Context context, AttributeSet attrs) {
        super(context, attrs);
        // TODO Auto-generated constructor stub
    }
    public CheckedTextViewC(Context context) {
        super(context);
        // TODO Auto-generated constructor stub
    }
    public void setTypeface(Typeface tf, int style) {
        if(!this.isInEditMode()){
        Typeface normalTypeface = Typeface.createFromAsset(getContext().getAssets(), "font/Roboto-Light.ttf");
        Typeface boldTypeface = Typeface.createFromAsset(getContext().getAssets(), "font/Roboto-Light.ttf");

        if (style == Typeface.BOLD) {
            super.setTypeface(boldTypeface/*, -1*/);
        } else {
            super.setTypeface(normalTypeface/*, -1*/);
        }
        }

    }
}

Implementieren Sie das neue Layout

adapter= new ArrayAdapter <String>(Menu.this,R.layout.custom_spinner, list);
joe1327
quelle
Ich denke, das wäre besser, weil Sie sich nicht die Hand am Adapter schmutzig machen müssen und Sie diese CustomTextview überall in Ihrer App verwenden können.
Milad
2

Dies ist die Fortsetzung meiner vorherigen Antwort: https://stackoverflow.com/a/51100507/787399

Aus Kompatibilitätsgründen können Sie die Stile und benutzerdefinierten Klassen für die Widgets in Android verwenden. Obwohl über Android Level 15, wurden neue /res/fontRessourcenordner eingeführt:

Schriftarten in Android

Schritt 1: Deklarieren Sie item_spinner.xml

<?xml version="1.0" encoding="utf-8"?>
<com.my_package.custom_views.FontTextView xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/tv_spinner"
    style="@style/App_TextViewStyleSmall"
    android:layout_gravity="start|bottom"
    android:layout_marginLeft="@dimen/dp_5"
    android:layout_marginStart="@dimen/dp_5"
    android:ellipsize="marquee"
    android:gravity="start|bottom"
    android:padding="@dimen/dp_10"
    android:singleLine="true"
    android:textAlignment="inherit" />
    <!--declared in layout: item_spinner.xml-->
    <!-- removed attributes:  android:layout_width="match_parent"
               android:layout_height="wrap_content"
               android:textColor="@color/text_grey_light"
               android:textSize="@dimen/sp_14" -->
    <!--style="?android:attr/spinnerItemStyle"-->

Schritt 2: deklarieren Sie item_spinner_dropdown.xml:

<?xml version="1.0" encoding="utf-8"?>
<com.my_package.custom_views.FontTextView xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/tv_spinner"
    style="@style/App_TextViewStyleSmall"
    android:layout_gravity="start|bottom"
    android:layout_marginLeft="@dimen/dp_5"
    android:layout_marginStart="@dimen/dp_5"
    android:ellipsize="marquee"
    android:gravity="start|bottom"
    android:padding="@dimen/dp_10"
    android:singleLine="true" />
    <!--declared in layout: item_spinner_dropdown.xml -->
    <!--removed: ?android:attr/dropdownListPreferredItemHeight-->
    <!--style="?android:attr/spinnerDropDownItemStyle"-->

Schritt 3: Verwenden Sie den Spinner im Layout:

<LinearLayout
            android:id="@+id/ll_my_spinner"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_below="@+id/fet_bus_entity"
            android:layout_marginTop="@dimen/dp_12"
            android:orientation="horizontal">

            <com.my_package.custom_views.FontTextView
                style="@style/App_TextViewStyleSmall"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_gravity="start|bottom"
                android:gravity="start|bottom"
                android:text="@string/are_you_a" />

            <Spinner
                android:id="@+id/sp_my_spinner"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_marginLeft="@dimen/dp_5"
                android:layout_marginStart="@dimen/dp_5"
                android:layout_gravity="end|bottom"
                android:spinnerMode="dropdown" />
        </LinearLayout>

[Hinweis: Die ID von FontTextView ist in den Layouts, im Spinner-Element und im Dropdown-Element identisch.]

Schritt 4: Verwenden Sie es in der Aktivität / Fragment:

private void initSpinnerBusinessType(View rootView) {
        String[] ar_dd_bus_type = getResources().getStringArray(R.array.ar_dd_bus_type);
        List<String> lst_bus_type = Arrays.asList(ar_dd_bus_type);
        ArrayList<String> ar_bus_type = new ArrayList<>(lst_bus_type);
        //==

        ArrayAdapter<String> adapter = new ArrayAdapter<>(activity, R.layout.item_spinner, R.id.tv_spinner, ar_bus_type);
        adapter.setDropDownViewResource(R.layout
                .item_spinner_dropdown);
        //=========
        Spinner sp_my_spinner= rootView.findViewById(R.id.sp_my_spinner);
        sp_my_spinner.setAdapter(adapter);
    }

[Weitere Hinweise finden Sie in meinem anderen Beitrag: https://stackoverflow.com/a/51077569/787399 und https://stackoverflow.com/a/22164007/787399 ]

Abhinav Saxena
quelle
0

Bitte folgen Sie den grundlegenden Anpassungen von FontTextView, FontEditView, FontRadioButton, FontCheckBox und FontButton.

[Die genaue Antwort finden Sie in diesem Handbuch unter: https://stackoverflow.com/a/51113022/787399 ]

Verwenden Sie benutzerdefiniertes FontTextView im ArrayAdapter-Elementlayout wie folgt:

public class FontEditText extends AppCompatEditText {

//    private String FONT = "fonts/roboto_regular.ttf";

    public FontEditText(Context context) {
        super(context, null);
//        setFontFromAsset(context, null, R.style.DefaultFontTextView);
//        FONT = getContext().getString(R.string.font_roboto_regular);
    }

    public FontEditText(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
        setFontFromAsset(context, attrs, R.attr.fetFontStyle);
    }

    public FontEditText(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        setFontFromAsset(context, attrs, defStyleAttr);
    }

    private void setFontFromAsset(Context context, AttributeSet attrs, int defStyle) {
        BaseActivity activity = (BaseActivity)((MyApplication) context.getApplicationContext()).getCurrentActivity();
        FontAndLocaleManager fontAndLocaleManager = activity.getFontAndLocaleManager();
        fontAndLocaleManager.setFontFromAsset(this, R.styleable.FontEditText, R.styleable.FontEditText_fetFontFace, attrs, defStyle);
    }
}

Verwenden Sie den Code:

public void setFontFromAsset(View view, int[] resViewStyleable, int resStyleableViewFontFace, AttributeSet attrs, int defStyle) {
        String strFont = null;
        Typeface tfFontFace = null;
        String strButton = FontButton.class.getCanonicalName(),
                strTextView = FontTextView.class.getCanonicalName(),
                strEditText = FontEditText.class.getCanonicalName(),
                strView = view.getClass().getCanonicalName();
        try {
            if (view.isInEditMode()) {
                return;
            }
            //R.string.font_roboto_regular
            strFont = context.getString(R.string.font_roboto_regular);
            tfFontFace = Typeface.createFromAsset(context.getAssets(), strFont);

            //AttributeSet set, int[] attrs, int defStyleAttr, int defStyleRes
            //R.styleable.FontButton
            TypedArray a = context.obtainStyledAttributes(attrs, resViewStyleable, defStyle, 0);
            //R.styleable.FontButton_btFontFace
            String derivedFont = a.getString(resStyleableViewFontFace);

            a.recycle();

            //==
            try {
                if (derivedFont != null) {
                    Typeface derivedFontFace = Typeface.createFromAsset(context.getAssets(), derivedFont);
                    if (strView.equals(strButton)) {
                        ((FontButton) view).setTypeface(derivedFontFace);
                    } else if (strView.equals(strTextView)) {
                        ((FontTextView) view).setTypeface(derivedFontFace);
                    } else if (strView.equals(strEditText)) {
                        ((FontEditText) view).setTypeface(derivedFontFace);
                    }
                    return;
                }

            } catch (Exception e) {
                e.printStackTrace();
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
        try {
            if (strFont != null && tfFontFace != null) {
                if (strView.equals(strButton)) {
                    ((FontButton) view).setTypeface(tfFontFace);
                } else if (strView.equals(strTextView)) {
                    ((FontTextView) view).setTypeface(tfFontFace);
                } else if (strView.equals(strEditText)) {
                    ((FontEditText) view).setTypeface(tfFontFace);
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }

    }

Beschreiben Sie Stil und Attribute in den jeweiligen xmls:

<!--FontTextView-->
    <declare-styleable name="FontTextViewStyle">
        <!-- Style of the FontTextView. -->
        <attr name="ftvFontStyle" format="reference"/>

    </declare-styleable>
    <declare-styleable name="FontTextView">
        <!-- Font face of FontTextView. -->
        <attr name="ftvFontFace" format="reference"/>
    </declare-styleable>

und

<!--FontTextView-->
<style name="StyledFontTextView" parent="@android:style/Theme.Light">
<item name="ftvFontStyle">@style/DefaultFontTextView</item>
</style>

<style name="DefaultFontTextView">
<item name="ftvFontFace">@string/font_roboto_regular</item>
</style>

Definieren Sie weitere Stile:

<style name="App_TextViewStyle" parent="@android:style/Widget.TextView">
        <item name="android:textColor">@color/text_grey</item>
        <item name="android:textSize">@dimen/sp_20</item>
        <item name="android:layout_width">match_parent</item>
        <item name="android:layout_height">wrap_content</item>
    </style>
    <style name="App_TextViewStyleMedium" parent="@android:style/Widget.TextView">
        <item name="android:textColor">@color/text_hint</item>
        <item name="android:textSize">@dimen/sp_18</item>
        <item name="android:layout_width">match_parent</item>
        <item name="android:layout_height">wrap_content</item>
    </style>
    <style name="App_TextViewStyleSmall" parent="@android:style/Widget.TextView">
        <item name="android:textColor">@color/text_grey_light</item>
        <item name="android:textSize">@dimen/sp_14</item>
        <item name="android:layout_width">match_parent</item>
        <item name="android:layout_height">wrap_content</item>
    </style>

Erwähnen Sie Schriftarten in Ihrer Datei strings.xml:

...
<string name="font_roboto_regular">fonts/roboto_regular.ttf</string>
...

und in den Layouts verwenden, um Code und Zeit zu sparen:

<com.mypackage.custom_views.FontTextView
                style="@style/App_TextViewStyleMedium"
                android:layout_gravity="start|bottom"
                android:gravity="start|bottom"
                app:fetFontFace="@string/font_roboto_regular"
                android:text="@string/are_you_a" />

Ab Android Level 16 wird dies alles vereinfacht, da Sie jetzt TTF und andere Schriftartenressourcen im /res/fontOrdner und nicht in Assets aufbewahren können. Dadurch werden die meisten benutzerdefinierten Klassen, Stile und Attribute entfernt, siehe:

Schriftarten in Android

Viel Spaß beim Codieren mit Stil !! :-)

Abhinav Saxena
quelle
Lange Antwort, aber einmalige Arbeit. Danach ist dies wiederverwendbar.
Abhinav Saxena
-1

Leute, ich habe eine großartige Lösung gefunden, ich wickle den Originaladapter per Helfer ein

Verwenden Sie diese Klasse SpinnerViewHelper und viel Spaß beim Programmieren mit Android

new SpinnerViewHelper((Spinner)view.findViewById(R.id.labelSurveyNumber),(parent, v, position, id) -> UrduFontHelper.set(v));

Der Lambda-Ausdruck wird verwendet.

Qamar4P
quelle