Прозрачная строка состояния и панель действий Android

97

Я провел несколько исследований по этой теме и не смог найти полного решения, поэтому, шаг за шагом, методом проб и ошибок, я наконец выяснил, как мы можем достичь этих результатов: имея прозрачные или цветные Actionbarи Statusbar. Смотрите мой ответ ниже.

GuilhE
источник

Ответы:

238

Я разрабатываю приложение, которое должно выглядеть одинаково на всех устройствах с> = API14, когда дело доходит до настройки панели действий и строки состояния. Я наконец нашел решение, и, поскольку мне потребовалось немного времени, я поделюсь им, чтобы спасти некоторые из ваших. Начнем с использования зависимости appcompat-21.

Прозрачная панель действий :

values ​​/ styles.xml :

<style name="AppTheme" parent="Theme.AppCompat.Light">
...
</style>

<style name="AppTheme.ActionBar.Transparent" parent="AppTheme">
    <item name="android:windowContentOverlay">@null</item>
    <item name="windowActionBarOverlay">true</item>
    <item name="colorPrimary">@android:color/transparent</item>
</style>

<style name="AppTheme.ActionBar" parent="AppTheme">
    <item name="windowActionBarOverlay">false</item>
    <item name="colorPrimary">@color/default_yellow</item>
</style>


значения-v21 / styles.xml :

<style name="AppTheme" parent="Theme.AppCompat.Light">
    ...
</style>

<style name="AppTheme.ActionBar.Transparent" parent="AppTheme">
    <item name="colorPrimary">@android:color/transparent</item>
</style>

<style name="AppTheme.ActionBar" parent="AppTheme">
    <item name="colorPrimaryDark">@color/bg_colorPrimaryDark</item>
    <item name="colorPrimary">@color/default_yellow</item>
</style>

Теперь вы можете использовать эти темы в своей, AndroidManifest.xmlчтобы указать, какие действия будут прозрачными или цветными ActionBar:

    <activity
            android:name=".MyTransparentActionbarActivity"
            android:theme="@style/AppTheme.ActionBar.Transparent"/>

    <activity
            android:name=".MyColoredActionbarActivity"
            android:theme="@style/AppTheme.ActionBar"/>

введите описание изображения здесь введите описание изображения здесь введите описание изображения здесь введите описание изображения здесь введите описание изображения здесь

Примечание: в API> = 21, чтобы получить Actionbarпрозрачность, вам нужно также получить Statusbarпрозрачную, иначе ваши цветовые стили не будут соблюдаться и останутся светло-серыми.



Прозрачная строка состояния (работает только с API> = 19) :
это довольно просто, просто используйте следующий код:

protected void setStatusBarTranslucent(boolean makeTranslucent) {
        if (makeTranslucent) {
            getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
        } else {
            getWindow().clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
        }
    }

Но вы заметите неприятный результат: введите описание изображения здесь

Это происходит потому, что, когда Statusbarмакет прозрачный, макет будет использовать его высоту. Чтобы этого не произошло, нам просто необходимо:

РЕШЕНИЕ ПЕРВОЕ:
добавьте эту строку android:fitsSystemWindows="true"в контейнер представления макета того, что вы хотите разместить под панелью действий:

    ...
        <LinearLayout
                android:fitsSystemWindows="true"
                android:layout_width="match_parent"
                android:layout_height="match_parent">
            ...
        </LinearLayout>
    ...

РЕШЕНИЕ ВТОРОЕ:
Добавьте несколько строк к нашему предыдущему методу:

protected void setStatusBarTranslucent(boolean makeTranslucent) {
        View v = findViewById(R.id.bellow_actionbar);
        if (v != null) {
            int paddingTop = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT ? MyScreenUtils.getStatusBarHeight(this) : 0;
            TypedValue tv = new TypedValue();
            getTheme().resolveAttribute(android.support.v7.appcompat.R.attr.actionBarSize, tv, true);
            paddingTop += TypedValue.complexToDimensionPixelSize(tv.data, getResources().getDisplayMetrics());
            v.setPadding(0, makeTranslucent ? paddingTop : 0, 0, 0);
        }

        if (makeTranslucent) {
            getWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
        } else {
            getWindow().clearFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
        }
    }

Где R.id.bellow_actionbarбудет идентификатор представления контейнера макета того, что мы хотим разместить под Actionbar:

...
    <LinearLayout
            android:id="@+id/bellow_actionbar"
            android:layout_width="match_parent"
            android:layout_height="match_parent">
        ...
    </LinearLayout>
...

введите описание изображения здесь

Так вот, думаю, я что-то не забываю. В этом примере я не использовал, Toolbarно думаю, будет тот же результат. Вот как я настраиваю свои Actionbar:

@Override
    protected void onCreate(Bundle savedInstanceState) {
        View vg = getActionBarView();
        getWindow().requestFeature(vg != null ? Window.FEATURE_ACTION_BAR : Window.FEATURE_NO_TITLE);

        super.onCreate(savedInstanceState);
        setContentView(getContentView());

        if (vg != null) {
            getSupportActionBar().setCustomView(vg, new ActionBar.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT));
            getSupportActionBar().setDisplayShowCustomEnabled(true);
            getSupportActionBar().setDisplayShowHomeEnabled(false);
            getSupportActionBar().setDisplayShowTitleEnabled(false);
            getSupportActionBar().setDisplayUseLogoEnabled(false);
        }
        setStatusBarTranslucent(true);
    }

Примечание: это abstract classрасширяет ActionBarActivity
надежду, это помогает!

GuilhE
источник
9
Вам действительно следует использовать android: fitsSystemWindows, если вы хотите, чтобы ваш View не находился под прозрачной строкой состояния.
ianhanniballake
1
Это совершенно правильно, я просматривал свой код и тоже использую его. Не знаю, я пропустил это в этом примере. Обновлю, спасибо!
GuilhE
4
Вам не хватает MyScreenUtils.getStatusBarHeight(this) здесь ответа stackoverflow.com/a/3410200/1307690
Роман Назаревич
1
есть ли способ сделать его полупрозрачным, но не полностью прозрачным? Я на api 23
aks
6
Чтобы получить Transparent StatusBar Works, мне нужно было добавить FLAG_LAYOUT_NO_LIMITS Flag, поместив код getWindow().addFlags(WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS);после этогоgetWindow().addFlags(WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS);
Eduardo ER
3

Он поддерживает после KITKAT. Просто добавьте следующий код внутри метода onCreate вашей Activity. Никаких модификаций файла манифеста не требуется.

 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
                Window w = getWindow(); // in Activity's onCreate() for instance
                w.setFlags(WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS, WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS);
            }
Индика Руван Сенавиратне
источник
9
Это делает панель навигации прозрачной
NOlivNeto
0

Просто добавьте эти строки кода в свой java-файл активности / фрагмента:

getWindow().setFlags(
    WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS,
    WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS
);
Фейсал Шейх
источник