Как получить высоту ActionBar?

Я пытаюсь получить высоту ActionBar (используя Шерлока) каждый раз, когда создается действие (особенно для обработки изменений конфигурации при вращении, когда высота ActionBar может измениться).

Для этого я использую метод ActionBar.getHeight() который работает только тогда, когда ActionBar Показано.

Когда первый вид деятельности создается впервые, я могу позвонить getHeight() в onCreateOptionsMenu Перезвоните. Но этот метод не вызывается после.

Поэтому мой вопрос: когда я могу вызвать getHeight() и быть уверенным, что он не вернет 0? Или, если это невозможно, как я могу установить высоту панели действий?

20 ответов

Решение

Хотя ответ @birdy является опцией, если вы хотите явно контролировать размер ActionBar, есть способ увеличить его без блокировки размера, который я нашел в документации поддержки. Это немного неловко, но это работает для меня. Вам понадобится контекст, этот пример будет действителен в Activity.

// Calculate ActionBar height
TypedValue tv = new TypedValue();
if (getTheme().resolveAttribute(android.R.attr.actionBarSize, tv, true))
{
    actionBarHeight = TypedValue.complexToDimensionPixelSize(tv.data,getResources().getDisplayMetrics());
}

В XML вы должны использовать этот атрибут:

android:paddingTop="?android:attr/actionBarSize"

Хорошо, я погуглил и попал на этот пост несколько раз, так что я думаю, что будет хорошо, если его описать не только для Шерлока, но и для appcompat:

Высота панели действий с помощью appCompat

Довольно похоже на описание @Anthony, вы можете найти его с помощью следующего кода (я только что изменил идентификатор ресурса):

TypedValue tv = new TypedValue();

if (getActivity().getTheme().resolveAttribute(R.attr.actionBarSize, tv, true))
{
    int actionBarHeight = TypedValue.complexToDimensionPixelSize(tv.data,getResources().getDisplayMetrics());
}

Предварительная проблема Sandwitch

Мне нужна была высота панели действий, потому что на устройствах до ICE_CREAM_SANDWITCH мой взгляд был перекрывающим панель действий. Я попытался установить android:layout_marginTop="? Android:attr/actionBarSize", android:layout_marginTop="? Attr / actionBarSize", установив перекрытие для view/actionbar и даже фиксированную высоту actionbar с пользовательской темой, но ничего не получалось. Вот как это выглядело:

проблема перекрытия

К сожалению, я обнаружил, что с помощью метода, описанного выше, я получаю только половину высоты (я предполагаю, что панель опций не занята), поэтому для полного исправления isue мне нужно удвоить высоту панели действий, и все выглядит нормально:

исправлено перекрытие

Если кто-то знает лучшее решение (чем удвоение actionBarHeight), я буду рад сообщить мне, так как я пришел из разработки для iOS, и я обнаружил, что большинство вещей в представлении Android довольно сбивают с толку:)

С Уважением,

hris.to

@ Энтони ответ работает для устройств, которые поддерживают ActionBar и для тех устройств, которые поддерживают только Sherlock Action Bar следующий метод должен быть использован

    TypedValue tv = new TypedValue();
    if (getTheme().resolveAttribute(android.R.attr.actionBarSize, tv, true))
        actionBarHeight = TypedValue.complexToDimensionPixelSize(tv.data,getResources().getDisplayMetrics());

    if(actionBarHeight ==0 && getTheme().resolveAttribute(com.actionbarsherlock.R.attr.actionBarSize, tv, true)){
            actionBarHeight = TypedValue.complexToDimensionPixelSize(tv.data,getResources().getDisplayMetrics());
    }

   //OR as stated by @Marina.Eariel
   TypedValue tv = new TypedValue();
   if(Build.VERSION.SDK_INT>=Build.VERSION_CODES.HONEYCOMB){
      if (getTheme().resolveAttribute(android.R.attr.actionBarSize, tv, true))
        actionBarHeight = TypedValue.complexToDimensionPixelSize(tv.data,getResources().getDisplayMetrics());
   }else if(getTheme().resolveAttribute(com.actionbarsherlock.R.attr.actionBarSize, tv, true){
        actionBarHeight = TypedValue.complexToDimensionPixelSize(tv.data,getResources().getDisplayMetrics());
   }

Я думаю, что самый безопасный способ будет:

private int getActionBarHeight() {
    int actionBarHeight = getSupportActionBar().getHeight();
    if (actionBarHeight != 0)
        return actionBarHeight;
    final TypedValue tv = new TypedValue();
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
        if (getTheme().resolveAttribute(android.R.attr.actionBarSize, tv, true))
            actionBarHeight = TypedValue.complexToDimensionPixelSize(tv.data, getResources().getDisplayMetrics());
    } else if (getTheme().resolveAttribute(com.actionbarsherlock.R.attr.actionBarSize, tv, true))
        actionBarHeight = TypedValue.complexToDimensionPixelSize(tv.data, getResources().getDisplayMetrics());
    return actionBarHeight;
}

Чтобы установить высоту ActionBar Вы можете создать новый Theme как этот:

<?xml version="1.0" encoding="utf-8"?>
<resources>   
    <style name="Theme.FixedSize" parent="Theme.Sherlock.Light.DarkActionBar">
        <item name="actionBarSize">48dip</item>
        <item name="android:actionBarSize">48dip</item> 
    </style> 
 </resources>

и установить это Theme на ваш Activity:

android:theme="@style/Theme.FixedSize"

Если вы используете панель инструментов в качестве панели действий,

Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);

затем просто используйте layout_height панели инструментов.

int actionBarHeight = toolbar.getLayoutParams().height;

Джава:

    int actionBarHeight;
    int[] abSzAttr;
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
        abSzAttr = new int[] { android.R.attr.actionBarSize };
    } else {
        abSzAttr = new int[] { R.attr.actionBarSize };
    }
    TypedArray a = obtainStyledAttributes(abSzAttr);
    actionBarHeight = a.getDimensionPixelSize(0, -1);

XML:

    ?attr/actionBarSize

Вот обновленная версия с Kotlin, которую я использовал, предполагая, что телефон выше, чем Honeycomb:

val actionBarHeight = with(TypedValue().also {context.theme.resolveAttribute(android.R.attr.actionBarSize, it, true)}) {
            TypedValue.complexToDimensionPixelSize(this.data, resources.displayMetrics)
        }

Если вы используете Xamarin/C#, это код, который вы ищете:

var styledAttributes = Context.Theme.ObtainStyledAttributes(new[] { Android.Resource.Attribute.ActionBarSize });
var actionBarSize = (int)styledAttributes.GetDimension(0, 0);
styledAttributes.Recycle();

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

public int getActionBarHeight() {
    final TypedArray ta = getContext().getTheme().obtainStyledAttributes(
            new int[] {android.R.attr.actionBarSize});
    int actionBarHeight = (int) ta.getDimension(0, 0);
    return actionBarHeight;
}

Готовый способ выполнить самый популярный ответ:

public static int getActionBarHeight(
  Activity activity) {

  int actionBarHeight = 0;
  TypedValue typedValue = new TypedValue();

  try {

    if (activity
      .getTheme()
      .resolveAttribute(
        android.R.attr.actionBarSize,
        typedValue,
        true)) {

      actionBarHeight =
        TypedValue.complexToDimensionPixelSize(
          typedValue.data,
          activity
            .getResources()
            .getDisplayMetrics());
    }

  } catch (Exception ignore) {
  }

  return actionBarHeight;
}

Вы просто должны добавить это.

public int getActionBarHeight() {
        int height;

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
            height = getActivity().getActionBar().getHeight();
        } else {
            height = ((ActionBarActivity) getActivity()).getSupportActionBar().getHeight();

        }
        return height;
    }

Я нашел более общий способ обнаружить это:

  int[] location = new int[2];
  mainView.getLocationOnScreen(location);
  int toolbarHeight = location[1];

где ' mainView ' - это корневой вид вашего макета.

Идея в основном состоит в том, чтобы получить Y-позицию главного окна, так как оно установлено прямо под панелью действий (или панелью инструментов).

Этот вспомогательный метод должен пригодиться кому-то. Пример: int actionBarSize = getThemeAttributeDimensionSize(this, android.R.attr.actionBarSize);

public static int getThemeAttributeDimensionSize(Context context, int attr)
{
    TypedArray a = null;
    try{
        a = context.getTheme().obtainStyledAttributes(new int[] { attr });
        return a.getDimensionPixelSize(0, 0);
    }finally{
        if(a != null){
            a.recycle();
        }
    }
}

В xml вы можете использовать? Attr/actionBarSize, но если вам нужен доступ к этому значению в Java, вам нужно использовать следующий код:

public int getActionBarHeight() {
        int actionBarHeight = 0;
        TypedValue tv = new TypedValue();
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
            if (getTheme().resolveAttribute(android.R.attr.actionBarSize, tv,
                    true))
                actionBarHeight = TypedValue.complexToDimensionPixelSize(
                        tv.data, getResources().getDisplayMetrics());
        } else {
            actionBarHeight = TypedValue.complexToDimensionPixelSize(tv.data,
                    getResources().getDisplayMetrics());
        }
        return actionBarHeight;
    }

Высота панели действий зависит от применяемой темы. Если вы используете AppCompatActivity, высота будет отличаться в большинстве случаев от обычной активности.

Если вы используете AppCompatActivity, вы должны использовать R.attr.actionBarSize вместо android.R.attr.actionBarSize

public static int getActionBarHeight(Activity activity) {
        TypedValue typedValue = new TypedValue();

        int attributeResourceId = android.R.attr.actionBarSize;
        if (activity instanceof AppCompatActivity) {
            attributeResourceId = R.attr.actionBarSize;
        }

        if (activity.getTheme().resolveAttribute(attributeResourceId, typedValue, true)) {
            return TypedValue.complexToDimensionPixelSize(typedValue.data, activity.getResources().getDisplayMetrics());
        }

        return (int) Math.floor(activity.getResources()
                .getDimension(R.dimen.my_default_value));
    }
The action bar now enhanced to app bar.So you have to add conditional check for getting height of action bar.

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
   height = getActivity().getActionBar().getHeight();
 } else {
  height = ((ActionBarActivity) getActivity()).getSupportActionBar().getHeight();
}

В javafx (используя Gluon) высота устанавливается во время выполнения или что-то в этом роде. В то время как возможность получить ширину, как обычно,...

double width = appBar.getWidth();

Вы должны создать слушателя:

GluonApplication application = this;

application.getAppBar().heightProperty().addListener(new ChangeListener(){
    @Override
    public void changed(ObservableValue observable, Object oldValue, Object newValue) {
        // Take most recent value given here
        application.getAppBar.heightProperty.get()
    }
});

... И принять самое последнее значение, на которое оно изменилось. Чтобы получить высоту.

Простой способ найти высоту панели действий - из Activity onPrepareOptionMenu метод.

@Override
public boolean onPrepareOptionsMenu(Menu menu) {
           ....
           int actionBarHeight = getActionBar().getHeight());
           .....
    }

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

Проверено только в API 25 и 24

C#

Resources.GetDimensionPixelSize(Resource.Dimension.abc_action_bar_default_height_material); 

Джава

getResources().getDimensionPixelSize(R.dimen.abc_action_bar_default_height_material);
Другие вопросы по тегам