Как получить NEW width/height корневого макета в onConfigurationChanged?
Один из наших представлений имеет ScrollView
в качестве корневого макета. Когда устройство повернуто и вызывается onConfigurationChanged()
, мы хотели бы получить новую ширину/высоту ScrollView
. Наш код выглядит следующим образом:
@Override
public void onConfigurationChanged(Configuration newConfig) {
Log.d(TAG, "Width: '" + findViewById(R.id.scrollview).getWidth() + "'");
Log.d(TAG, "Height: '" + findViewById(R.id.scrollview).getHeight() + "'");
super.onConfigurationChanged(newConfig);
Log.d(TAG, "Width: '" + findViewById(R.id.scrollview).getWidth() + "'");
Log.d(TAG, "Height: '" + findViewById(R.id.scrollview).getHeight() + "'");
}
Соответствующий раздел нашего AndroidManifest.xml выглядит следующим образом:
<activity android:name=".SomeActivity"
android:configChanges="keyboardHidden|orientation">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
</intent-filter>
</activity>
И, наконец, соответствующая часть нашего макета выглядит следующим образом:
<ScrollView xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/scrollview"
android:layout_height="fill_parent"
android:layout_width="fill_parent"
>
<LinearLayout android:id="@+id/container"
android:orientation="vertical"
android:layout_height="fill_parent"
android:minHeight="200dip"
android:layout_width="fill_parent"
>
На нашем Droid мы ожидали увидеть, что ширина ScrollView переходит на 854 при переключении в альбомную, а до 480 при переключении обратно в портрет (а высота - эквивалентного переключателя, минус строка меню). Однако мы видим обратное. Здесь наш LogCat:
// Switching to landscape:
03-26 11:26:16.490: DEBUG/ourtag(17245): Width: '480' // Before super
03-26 11:26:16.490: DEBUG/ourtag(17245): Height: '778' // Before super
03-26 11:26:16.529: DEBUG/ourtag(17245): Width: '480' // After super
03-26 11:26:16.536: DEBUG/ourtag(17245): Height: '778' // After super
// Switching to portrait:
03-26 11:26:28.724: DEBUG/ourtag(17245): Width: '854' // Before super
03-26 11:26:28.740: DEBUG/ourtag(17245): Height: '404' // Before super
03-26 11:26:28.740: DEBUG/ourtag(17245): Width: '854' // After super
03-26 11:26:28.740: DEBUG/ourtag(17245): Height: '404' // After super
Понятно, что мы приобретаем размеры портрета, когда переключаемся на пейзаж и пейзажные размеры, когда переключаемся на портрет. Что-то мы делаем неправильно? Мы могли бы взломать и решить эту проблему, но я чувствую, что там простое решение, которое нам не хватает.
Ответы
Ответ 1
getWidth() возвращает ширину, как показано на рисунке, что означает, что вам нужно подождать, пока она не будет нарисована на экране. onConfigurationChanged
вызывается перед перерисованием представления для новой конфигурации, поэтому я не думаю, что вы сможете получить новую ширину до конца.
Ответ 2
Для тех, кто ищет более подробное описание решения: вы можете использовать ViewTreeObserver
вашего представления и зарегистрировать OnGlobalLayoutListener
.
@Override
public void onConfigurationChanged(Configuration newConfiguration) {
super.onConfigurationChanged(newConfiguration);
final View view = findViewById(R.id.scrollview);
ViewTreeObserver observer = view.getViewTreeObserver();
observer.addOnGlobalLayoutListener(new OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
Log.v(TAG,
String.format("new width=%d; new height=%d", view.getWidth(),
view.getHeight()));
view.getViewTreeObserver().removeOnGlobalLayoutListener(this);
}
});
}
Ответ 3
Когда onGlobalLayout назвал это не уверенным, что представление было изменено в соответствии с новой ориентацией макета, так что для меня только подходящее решение работало правильно:
@Override
public void onConfigurationChanged(Configuration newConfig) {
final int oldHeight = mView.getHeight();
final int oldWidth = mView.getWidth();
mView.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
if (mView.getHeight() != oldHeight && mView.getWidth() != oldWidth) {
mView.getViewTreeObserver().removeOnGlobalLayoutListener(this);
//mView now has the correct dimensions, continue with your stuff
}
}
});
super.onConfigurationChanged(newConfig);}