android: windowSoftInputMode= "adjustResize" не имеет никакого значения?
у меня есть искусственный диалог, который использует этот макет:
<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_height="match_parent"
android:layout_width="match_parent"
android:id="@+id/containerPageConatiner">
<FrameLayout android:id="@+id/dialogHolder"
android:layout_height="wrap_content"
android:layout_width="wrap_content"
android:padding="15dp"
android:layout_gravity="center"
android:background="@drawable/panel_picture_frame_bg_focus_blue"/>
</FrameLayout>
я помещаю фрагмент внутри <FrameLayout>
в зависимости от открываемого диалога-действие, управляющее диалогом, выглядит следующим образом:
<activity
android:label="@string/app_name"
android:name=".activity.DialogActivity"
android:theme="@style/CustomTheme.Screen.Transparent"
android:windowSoftInputMode="adjustResize">
к сожалению, когда вы нажимаете на текст редактирования внутри диалогового окна, изменение размера не происходит. The windowSoftInputMode
буквально не имеет значения, поскольку функциональность такая же, как режим панорамирования.
документация говорит: "Это конечно работает только для приложений, которые имеют изменяемую область, которая может быть уменьшена, чтобы сделать достаточно места", но не говорит вам, что это означает под "изменяемой областью", и заставляет меня думать, что в некотором роде я не есть изменяемая площадь?
если кто-нибудь знает, что случилось, они могут мне помочь?
редактировать
окружение диалога так ничего не меняет:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/containerPageConatiner"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical" >
<View
android:layout_width="0dp"
android:layout_height="0dp"
android:layout_weight="1" />
<FrameLayout
android:id="@+id/dialogHolder"
android:layout_height="wrap_content"
android:layout_width="wrap_content"
android:padding="15dp"
android:layout_gravity="center"
android:background="@drawable/panel_picture_frame_bg_focus_blue"/>
<View
android:layout_width="0dp"
android:layout_height="0dp"
android:layout_weight="1" />
</LinearLayout>
EDIT2
Scrollview как родитель тоже не помогает:
<?xml version="1.0" encoding="utf-8"?>
<ScrollView xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/containerPageConatiner"
android:layout_width="match_parent"
android:layout_height="match_parent" >
<FrameLayout
android:id="@+id/dialogHolder"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:padding="15dp" />
</ScrollView>
10 ответов
Я создал новый проект, чтобы попытаться получить основные функции, работающие для изменения размера окна, и медленно переместил его к целевому peice моего проекта. Делая это, я отследил проблему до этого:
в моей иерархии тем у меня было это свойство:
<item name="android:windowFullscreen">true</item>
, который был похоронен на уровне Theme.Black.NoTitleBar.FullScreen
- предок моей пользовательской темы.
на документация предполагает, что это " флаг, указывающий, является ли это окно должен заполнить весь экран". Это звучит как хорошая вещь, если у вас есть приложение, которое занимает весь экран... За исключением того, что он по-прежнему занимает весь экран без флага.
на самом деле, как только вы взяли это, нет абсолютно никаких изменений в приложении вообще... кроме adjustResize
теперь отлично работает.
некоторое время назад у меня также была та же проблема в библиотеке, которую я создал. (MaterialDrawer)
насколько я вижу, все предоставленные ответы не решают основную проблему, они просто указывают на удаление полноэкранного флага (android:windowFullscreen
), что не является решением для многих.
вышеупомянутая "проблема" появляется только в версиях Android, начиная с уровня API 19 (KITKAT), потому что они изменили поведение. Чтобы быть правильным, это не проблема, она работает как предназначенный. См. комментарий сотрудника Android ( Android-Выпуск).
поэтому я начал копаться вокруг источника Android и пришел к следующему решению, используя ViewTreeObserver.OnGlobalLayoutListener
и реагирует, если клавиатура отображается / или скрыта. Если клавиатура отображается, я добавляю заполнение в нижнюю часть представления контейнера, которое затем эмулирует то же самое, что и adjustResize
будет делать.
решение
для упрощения использования я завернула все дело в простом вспомогательном классе KeyboardUtil.
/**
* Created by mikepenz on 14.03.15.
* This class implements a hack to change the layout padding on bottom if the keyboard is shown
* to allow long lists with editTextViews
* Basic idea for this solution found here: http://stackoverflow.com/a/9108219/325479
*/
public class KeyboardUtil {
private View decorView;
private View contentView;
public KeyboardUtil(Activity act, View contentView) {
this.decorView = act.getWindow().getDecorView();
this.contentView = contentView;
//only required on newer android versions. it was working on API level 19 (Build.VERSION_CODES.KITKAT)
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
decorView.getViewTreeObserver().addOnGlobalLayoutListener(onGlobalLayoutListener);
}
}
public void enable() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
decorView.getViewTreeObserver().addOnGlobalLayoutListener(onGlobalLayoutListener);
}
}
public void disable() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
decorView.getViewTreeObserver().removeOnGlobalLayoutListener(onGlobalLayoutListener);
}
}
//a small helper to allow showing the editText focus
ViewTreeObserver.OnGlobalLayoutListener onGlobalLayoutListener = new ViewTreeObserver.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
Rect r = new Rect();
//r will be populated with the coordinates of your view that area still visible.
decorView.getWindowVisibleDisplayFrame(r);
//get screen height and calculate the difference with the useable area from the r
int height = decorView.getContext().getResources().getDisplayMetrics().heightPixels;
int diff = height - r.bottom;
//if it could be a keyboard add the padding to the view
if (diff != 0) {
// if the use-able screen height differs from the total screen height we assume that it shows a keyboard now
//check if the padding is 0 (if yes set the padding for the keyboard)
if (contentView.getPaddingBottom() != diff) {
//set the padding of the contentView for the keyboard
contentView.setPadding(0, 0, 0, diff);
}
} else {
//check if the padding is != 0 (if yes reset the padding)
if (contentView.getPaddingBottom() != 0) {
//reset the padding of the contentView
contentView.setPadding(0, 0, 0, 0);
}
}
}
};
/**
* Helper to hide the keyboard
*
* @param act
*/
public static void hideKeyboard(Activity act) {
if (act != null && act.getCurrentFocus() != null) {
InputMethodManager inputMethodManager = (InputMethodManager) act.getSystemService(Activity.INPUT_METHOD_SERVICE);
inputMethodManager.hideSoftInputFromWindow(act.getCurrentFocus().getWindowToken(), 0);
}
}
}
затем вы можете использовать его в вашей деятельности или фрагмент следующим образом:
//initialize the KeyboardUtil (you can do this global)
KeyboardUtil keyboardUtil = new KeyboardUtil(activity, getContent().getChildAt(0));
//enable it
keyboardUtil.enable();
//disable it
keyboardUtil.disable();
весь класс util используется в вышеупомянутой библиотеке MaterialDrawer и можно найти здесь KeyboardUtil. Это всегда будет содержать последнюю версию. (если есть улучшения)
кажется, что проблема заключается в FrameLayout, поскольку он ведет себя таким образом, что каждый ребенок занимает видимое пространство этого кадра, поэтому нет необходимости изменять размер для детей. Попробуйте использовать RelativeLayout. Это должно сработать.
Я не знаю, почему, но если у вас есть <item name="android:windowTranslucentNavigation">true</item>
в вашей теме измените ее на false
. И это начнет работать. Действительно странный.
без использования ScrollView
как мой родитель я только что добавил android:fitsSystemWindows="true"
мой родительский вид (который был RelativeLayout
и adjustResize
к манифесту для деятельности, и это сработало.
мне пришлось установить
<item name="android:windowFullscreen">false</item>
несмотря на это, я никогда не устанавливал его в true, и приложение на самом деле не было полноэкранным.
вы можете adjustResize программно. Добавить это в onCreate() вашей деятельности.
getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_ADJUST_RESIZE);
как оригинальный плакат обнаружил, когда полноэкранный флаг присваивается деятельности Android: windowFullscreen атрибут не будет работать, и поэтому ваше окно не будет изменять размер, когда мягкая клавиатура видна, и она не будет прокручиваться.
простое удаление полноэкранного флага и не использование полноэкранной темы позволит прокручивать, когда видна мягкая клавиатура.