Подтвердить что ты не робот

Доступ к attrs в AttributeSet для пользовательских компонентов

У меня есть пользовательский компонент, содержащий два текстовых элемента, которые имеют настраиваемый метод настройки размера (два текстовых представления пропорциональны соотношению 1: 2). Поскольку это подкласс RelativeLayout, у него нет атрибута textSize, но мне интересно, можно ли еще установить атрибут android:textSize в XML-экземпляре для этого компонента, а затем захватить textSize атрибут из AttributeSet для использования с моим настраиваемым методом setSize() в конструкторе.

Я видел технику для этого с помощью настраиваемых атрибутов, но что, если я хочу захватить атрибут, уже находящийся в лексиконе Android?

4b9b3361

Ответ 1

да, это возможно;

предположим, что ваше объявление RelativeLayout (в xml) имеет textSize, определенный с 14sp:

android:textSize="14sp"

В конструкторе вашего пользовательского представления (тот, который принимает в AttributeSet) вы можете получить атрибуты из пространства имен Android как таковые:

String xmlProvidedSize = attrs.getAttributeValue("http://schemas.android.com/apk/res/android", "textSize");

Значение xmlProvidedSize будет чем-то вроде этого "14.0sp" и, возможно, с небольшим редактированием String вы можете просто извлечь числа.


Другой вариант объявления вашего собственного набора атрибутов будет небольшим, но это также возможно.

Итак, у вас есть собственный пользовательский вид, и ваши TextViews объявили что-то вроде этого:

public class MyCustomView extends RelativeLayout{

    private TextView myTextView1;
    private TextView myTextView2;

// rest of your class here

большой...

Теперь вам нужно также убедиться, что ваш пользовательский вид переопределяет конструктор, который принимает атрибутный набор следующим образом:

public MyCustomView(Context context, AttributeSet attrs){   
    super(context, attrs);
    init(attrs, context);  //nice, clean method to instantiate your TextViews// 
}

ok, посмотрим, что теперь метод init():

private void init(AttributeSet attrs, Context context){
    // do your other View related stuff here //


    TypedArray a = getContext().obtainStyledAttributes(attrs, R.styleable.MyCustomView);
    int xmlProvidedText1Size = a.int(R.styleable.MyCustomView_text1Size);
    int xmlProvidedText2Size = a.int(R.styleable.MyCustomView_text2Size);

    myTextView1.setTextSize(xmlProvidedText1Size);
    myTextView2.setTextSize(xmlProvidedText2Size);

    // and other stuff here //
}

Вероятно, вам интересно, откуда берутся R.styleable.MyCustomView, R.styleable.MyCustomView_text1Size и R.styleable.MyCustomView_text2Size; позвольте мне подробно остановиться на них.

Вы должны объявить имена атрибутов в файле attrs.xml(в каталоге значений), чтобы везде, где вы можете использовать свое пользовательское представление, значения, полученные из этих атрибутов, будут переданы в ваш конструктор.

Итак, посмотрим, как вы объявляете эти пользовательские атрибуты, как вы просили: Вот мой весь attrs.xml

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <declare-styleable name="MyCustomView">
        <attr name="text1Size" format="integer"/>
        <attr name="text2Size" format="integer"/>
    </declare-styleable>
</resources>

Теперь вы можете установить размер вашего TextViews в своем XML, но НЕ, не объявив пространство имен в своем макете, вот как:

<com.my.app.package.MyCustomView
    xmlns:josh="http://schemas.android.com/apk/res-auto"
    android:id="@+id/my_custom_view_id"
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    josh:text1Size="15"
    josh:text2Size="30"     
    />

Обратите внимание, как я объявлял пространство имен "josh" в качестве первой строки в вашем наборе атрибутов CustomView.

Надеюсь, это поможет Джошу,

Ответ 2

Принятый ответ немного длинный. Вот сжатая версия, на которую я надеюсь, будет легко следовать. Чтобы добавить к вашему пользовательскому представлению атрибут textSize (или все, что использует dp/sp), выполните следующие действия.

1. Создайте собственный атрибут

Создайте (или добавьте следующий раздел) в attrs.xml. Обратите внимание на формат dimension.

<resources>
    <declare-styleable name="CustomView">
        <attr name="textSize" format="dimension" />
    </declare-styleable>
</resources>

2. Задайте атрибут в макете xml.

Обратите внимание на пользовательское пространство имен app.

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
                xmlns:app="http://schemas.android.com/apk/res-auto"
                android:layout_width="match_parent"
                android:layout_height="match_parent">

    <com.example.myproject.CustomView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:textSize="20sp" /> 

</RelativeLayout>

3. Получите значение атрибута в своем пользовательском представлении

Обратите внимание на использование getDimensionPixelSize. В вашем пользовательском представлении просто работайте с пикселями. См. этот ответ, если вам нужно преобразовать в другое измерение.

public class CustomView extends View {

    private float mTextSize; // pixels

    public CustomView(Context context, AttributeSet attrs) {
        super(context, attrs);
        TypedArray a = context.getTheme().obtainStyledAttributes(
                attrs, R.styleable.CustomView, 0, 0);
        try {
            mTextSize = a.getDimensionPixelSize(R.styleable.CustomView_textSize, 0);
        } finally {
            a.recycle();
        }
    }

    /**
     * @param size in SP units
     */
    public void setTextSize(int size) {
        mTextSize = TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_SP,
                size, getResources().getDisplayMetrics());
        mTextPaint.setTextSize(mTextSize);
        invalidate();
        requestLayout();
    }

    // ...
}

Примечания