ti-enxame.com

android: windowSoftInputMode = "adjustResize" não faz diferença?

Eu tenho um diálogo falso que usa esse layout:

<?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> 

Eu coloco um fragmento dentro do <FrameLayout> dependendo do diálogo que está abrindo - A atividade que controla o Dialog se parece com isso:

<activity
    Android:label="@string/app_name"
    Android:name=".activity.DialogActivity"
    Android:theme="@style/CustomTheme.Screen.Transparent" 
    Android:windowSoftInputMode="adjustResize">

Infelizmente, quando você clica em um texto de edição dentro da caixa de diálogo, não ocorre redimensionamento. A windowSoftInputMode literalmente não faz diferença, pois a funcionalidade é a mesma do modo pan.

Documentação diz "Isto obviamente funciona apenas para aplicações que têm uma área redimensionável que pode ser reduzida para ganhar espaço suficiente" mas não lhe diz o que significa por "uma área redimensionável" e me faz pensar que em alguns como eu não tenho uma área redimensionável?

Se alguém souber o que está acontecendo, eles podem me ajudar?

EDIT

Cercar a caixa de diálogo assim não muda nada:

<?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 como pai não ajuda:

<?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>
26
Graeme

Criei um novo projeto para tentar obter os recursos básicos que funcionam para o redimensionamento de janelas e movê-lo lentamente para a parte de destino do meu projeto. Fazendo isso eu rastreei o problema até isso:

Na minha hierarquia de temas, eu tinha essa propriedade:

<item name="Android:windowFullscreen">true</item> 

que foi enterrado no nível de Theme.Black.NoTitleBar.FullScreen - Um antepassado do meu tema personalizado.

A documentação sugere que este é um "sinalizador indicando se esta janela deve preencher a tela inteira". Isso soa como uma coisa boa para ter se você tiver um aplicativo que ocupa a tela inteira ... Exceto que ainda ocupa toda a tela sem a bandeira.

Na verdade, uma vez que você tenha feito isso, não há absolutamente nenhuma mudança no aplicativo ... além de adjustResize agora funciona perfeitamente.

46
Graeme

Um tempo atrás eu também tive o mesmo problema em uma biblioteca que eu criei. ( MaterialDrawer )

Tanto quanto eu posso ver todas as respostas fornecidas não resolver o problema principal, eles apenas apontam para remover o sinalizador de tela cheia (Android:windowFullscreen), que não é solução para muitos por aí.

O "problema" mencionado acima só aparece nas versões do Android, começando com o nível 19 da API (KitKat), porque eles mudaram o comportamento. Para ser correto, não há problema, está funcionando como pretendido. Veja o comentário de um funcionário do Android ( Android-Issue ).


Então eu comecei a pesquisar a fonte Android e cheguei à seguinte solução usando o ViewTreeObserver.OnGlobalLayoutListener e reagindo se o teclado é mostrado/ou oculto. Se o teclado for exibido, adiciono o preenchimento à parte inferior da exibição do contêiner, que será emulado da mesma forma que o adjustResize faria.


Solução

Para simplificar o uso, envolvi tudo em uma simples classe auxiliar 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);
        }
    }
}

Você pode usá-lo em sua atividade ou fragmento fazendo o seguinte:

//initialize the KeyboardUtil (you can do this global)
KeyboardUtil keyboardUtil = new KeyboardUtil(activity, getContent().getChildAt(0));

//enable it
keyboardUtil.enable();

//disable it
keyboardUtil.disable();

Toda a classe util é usada na biblioteca acima mencionada MaterialDrawer e pode ser encontrada aqui KeyboardUtil . Isso sempre conterá a versão mais recente. (se houver melhorias)

44
mikepenz

Parece que o problema é com o FrameLayout, pois ele se comporta dessa forma, cada criança ocupando espaço visível daquele quadro, portanto não há necessidade de redimensionar para caber crianças. Tente usar o RelativeLayout. Deve funcionar.

2
ACM64

Você pode adjustResize programaticamente. Adicione isto em onCreate () da sua atividade.

getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_ADJUST_RESIZE);
0
Naimish Vinchhi

Eu não sei porque, mas se você tiver <item name="Android:windowTranslucentNavigation">true</item> no seu tema, mude para false. E vai começar a funcionar. Realmente estranho.

0
Nastromo

Sem usar um ScrollView como meu pai, eu adicionei Android:fitsSystemWindows="true" à minha view pai (que era uma RelativeLayout e adjustResize ao Manifest da atividade e funcionou.

0
Josh Laird

Tente colocar o seu LinearLayout em um ScrollView, que funcionou para mim uma vez ..

0
Cata

Eu tive que definir 

<item name="Android:windowFullscreen">false</item> 

Apesar disso, nunca defini a verdade e o aplicativo não era em tela cheia.

0
F0G