it-swarm.com.ru

Тема/Стиль не применяется, когда инфлятор используется с ApplicationContext

У меня есть тема, которая определяет textColor для TextView как красный.

Я использую LayoutInflater для создания экземпляра TextView. Проблема в том, что стили не применяются к TextView, когда инфлятор создается с помощью ApplicationContext - цвет не красный. Все отлично работает, когда LayoutInflater создан с использованием активности.

Почему это происходит и как это можно исправить?

/res/values/styles.xml:

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <style name="MyTheme">
        <item name="Android:textViewStyle">@style/MyTextView</item>
    </style>

    <style name="MyTextView" parent="@Android:style/Widget.TextView">
        <item name="Android:textColor">#f00</item>
    </style>
</resources>

AndroidManifest.xml:

<application 
    Android:icon="@drawable/icon" 
    Android:label="@string/app_name"
    Android:theme="@style/MyTheme"
    >

Код:

public class A extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.test_a);

        final LayoutInflater goodInflater = getInflater((Activity)this); 
        final LayoutInflater badInflater = getInflater(getApplicationContext());
        final LinearLayout container = (LinearLayout)findViewById(R.id.container);

        findViewById(R.id.add_with_appContext).setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                add(container, badInflater); // Creates gray TextView
            }            
        });

        findViewById(R.id.add_with_activity).setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                add(container, goodInflater); // Creates red TextView
            }            
        });
    }

    private LayoutInflater getInflater(Context context) {
        return (LayoutInflater)context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    }

    private void add(LinearLayout container, LayoutInflater inflater) {
        inflater.inflate(R.layout.my_template, container, true);
    }
}

/res/layout/test_a.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:Android="http://schemas.Android.com/apk/res/Android"
    Android:layout_width="fill_parent"
    Android:layout_height="wrap_content"
    Android:orientation="vertical">

    <Button 
        Android:text="Add with AppContext" 
        Android:id="@+id/add_with_appContext" 
        Android:layout_width="wrap_content" 
        Android:layout_height="wrap_content" 
        />

    <Button 
        Android:text="Add with Activity" 
        Android:id="@+id/add_with_activity" 
        Android:layout_width="wrap_content" 
        Android:layout_height="wrap_content" 
        />

    <LinearLayout
        Android:id="@+id/container"
        Android:layout_width="fill_parent"
        Android:layout_height="wrap_content"
        Android:orientation="vertical"  
        />

</LinearLayout>

/res/layout/my_template.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout 
    xmlns:Android="http://schemas.Android.com/apk/res/Android"
    Android:layout_width="fill_parent" 
    Android:layout_height="wrap_content"
    >

    <TextView
        Android:id="@+id/text"
        Android:text="Some text..."
        Android:layout_width="fill_parent" 
        Android:layout_height="wrap_content"
    />

</LinearLayout>
43
alex2k8

Решение № 1

Метод inflate принимает необязательный аргумент «ViewGroup root»:

public View inflate (int resource, ViewGroup root, boolean attachToRoot)

Если у нас есть значение для передачи в качестве «корневого» параметра, то, следовательно, мы можем использовать его для получения «контекста активности», откуда мы можем получить правильный LayoutInflater:

ViewGroup root > activity context > LayoutInflater

Так что мой код может быть:

private void add(LinearLayout container) {
    LayoutInflater inflater = getInflater(container.getContext());
    inflater.inflate(R.layout.my_template, container, true);
}

Решение № 2

Просто попытался установить тему Application Context программно, и она работает :

getApplicationContext().setTheme(R.style.MyTheme);

Я думаю, что было бы логично ожидать эту разметку:

<application 
    Android:icon="@drawable/icon" 
    Android:label="@string/app_name"
    Android:theme="@style/MyTheme"
    >

установить его автоматически, но это не так.

50
alex2k8

Никогда не используйте контекст приложения для раздувания представлений, потому что стилизация не работает с этим контекстом. Всегда используйте контекст действия при игре с представлениями. Единственное исключение - когда вам нужно создать RemoteViews из Сервиса.

Более подробную информацию о различных типах контекстов и их возможностях можно найти в этой превосходной статье .

33
BladeCoder

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

public class CustomView extends ViewGroup{

public CustomView (Context context) {
    super(context);
    init(context);
}

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

public CustomView (Context context, AttributeSet attrs, int defStyle) {
    super(context, attrs, defStyle);
    init(context);
}

@TargetApi(21)
public CustomView (Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
    super(context, attrs, defStyleAttr, defStyleRes);
    init(context);
}

private void init(Context context) {
    LayoutInflater  mInflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    View v = mInflater.inflate(R.layout.review_list_item, this, true);
    //rest of view initialization       
}   
}
0
abedfar