Как получить высоту 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);