Android不让弹出键盘挡住View

The AndroidManifest.xml File

<activity android:windowSoftInputMode=["stateUnspecified",

"stateUnchanged", "stateHidden",

"stateAlwaysHidden", "stateVisible",

"stateAlwaysVisible", "adjustUnspecified",

"adjustResize", "adjustPan"] …… >

</activity>

attributes:

android:windowSoftInputMode

活动的主窗口如何与包含屏幕上的软键盘窗口交互。这个属性的设置将会影响两件事情:

1>软键盘的状态——是否它是隐藏或显示——当活动(Activity)成为用户关注的焦点。

2>活动的主窗口调整——是否减少活动主窗口大小以便腾出空间放软键盘或是否当活动窗口的部分被软键盘覆盖时它的内容的当前焦点是可见的。

它的设置必须是下面列表中的一个值,或一个”state…”值加一个”adjust…”值的组合。在任一组设置多个值——多个”state…”values,例如&mdash有未定义的结果。各个值之间用|分开。例如: <activity android:windowSoftInputMode="stateVisible|adjustResize" . . . >

在这设置的值(除"stateUnspecified"和"adjustUnspecified"以外)将覆盖在主题中设置的值

描述

"stateUnspecified"

软键盘的状态(是否它是隐藏或可见)没有被指定。系统将选择一个合适的状态或依赖于主题的设置。

这个是为了软件盘行为默认的设置。

"stateUnchanged"

软键盘被保持无论它上次是什么状态,是否可见或隐藏,当主窗口出现在前面时。

"stateHidden"

当用户选择该Activity时,软键盘被隐藏——也就是,当用户确定导航到该Activity时,而不是返回到它由于离开另一个Activity。

"stateAlwaysHidden"

软键盘总是被隐藏的,当该Activity主窗口获取焦点时。

"stateVisible"

软键盘是可见的,当那个是正常合适的时(当用户导航到Activity主窗口时)。

"stateAlwaysVisible"

当用户选择这个Activity时,软键盘是可见的——也就是,也就是,当用户确定导航到该Activity时,而不是返回到它由于离开另一个Activity。

"adjustUnspecified"

它不被指定是否该Activity主窗口调整大小以便留出软键盘的空间,或是否窗口上的内容得到屏幕上当前的焦点是可见的。系统将自动选择这些模式中一种主要依赖于是否窗口的内容有任何布局视图能够滚动他们的内容。如果有这样的一个视图,这个窗口将调整大小,这样的假设可以使滚动窗口的内容在一个较小的区域中可见的。这个是主窗口默认的行为设置。

"adjustResize"

该Activity主窗口总是被调整屏幕的大小以便留出软键盘的空间

"adjustPan"

该Activity主窗口并不调整屏幕的大小以便留出软键盘的空间。相反,当前窗口的内容将自动移动以便当前焦点从不被键盘覆盖和用户能总是看到输入内容的部分。这个通常是不期望比调整大小,因为用户可能关闭软键盘以便获得与被覆盖内容的交互操作。





一、软键盘显示的原理
软件盘的本质是什么?软键盘其实是一个Dialog!
InputMethodService为我们的输入法创建了一个Dialog,并且将该Dialog的Window的某些参数(如Gravity)进行了设置,使之能够在底部或者全屏显示。当我们点击输入框时,系统对活动主窗口进行调整,从而为输入法腾出相应的空间,然后将该Dialog显示在底部,或者全屏显示。
二、活动主窗口调整
android定义了一个属性,名字为windowSoftInputMode, 用它可以让程序可以控制活动主窗口调整的方式。我们可以在AndroidManifet.xml中对Activity进行设置。如:android:windowSoftInputMode="stateUnchanged|adjustPan"
该属性可选的值有两部分,一部分为软键盘的状态控制,另一部分是活动主窗口的调整。前一部分本文不做讨论,请读者自行查阅android文档。
模式一,压缩模式
windowSoftInputMode的值如果设置为adjustResize,那么该Activity主窗口总是被调整大小以便留出软键盘的空间。
我们通过一段代码来测试一下,当我们设置了该属性后,弹出输入法时,系统做了什么。
重写Layout布局:

 
  1. publicclassResizeLayoutextendsLinearLayout{
  2. privatestaticintcount=0;
  3. publicResizeLayout(Contextcontext,AttributeSetattrs){
  4. super(context,attrs);
  5. }
  6. @Override
  7. protectedvoidonSizeChanged(intw,inth,intoldw,intoldh){
  8. super.onSizeChanged(w,h,oldw,oldh);
  9. Log.e("onSizeChanged"+count++,"=>onResizecalled!w="+w+",h="+h+",oldw="+oldw+",oldh="+oldh);
  10. }
  11. @Override
  12. protectedvoidonLayout(booleanchanged,intl,intt,intr,intb){
  13. super.onLayout(changed,l,t,r,b);
  14. Log.e("onLayout"+count++,"=>OnLayoutcalled!l="+l+",t="+t+",r="+r+",b="+b);
  15. }
  16. @Override
  17. protectedvoidonMeasure(intwidthMeasureSpec,intheightMeasureSpec){
  18. super.onMeasure(widthMeasureSpec,heightMeasureSpec);
  19. Log.e("onMeasure"+count++,"=>onMeasurecalled!widthMeasureSpec="+widthMeasureSpec+",heightMeasureSpec="+heightMeasureSpec);
  20. }

我们的布局设置为:

 
  1. <com.winuxxan.inputMethodTest.ResizeLayout
  2. xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:id="@+id/root_layout"
  4. android:layout_width="fill_parent"
  5. android:layout_height="fill_parent"
  6. android:orientation="vertical"
  7. >
  8. <EditText
  9. android:layout_width="fill_parent"
  10. android:layout_height="wrap_content"
  11. />
  12. <LinearLayout
  13. android:id="@+id/bottom_layout"
  14. android:layout_width="fill_parent"
  15. android:layout_height="fill_parent"
  16. android:orientation="vertical"
  17. android:gravity="bottom">s
  18. <TextView
  19. android:layout_width="fill_parent"
  20. android:layout_height="wrap_content"
  21. android:text="@string/hello"
  22. android:background="#77777777"
  23. />
  24. </LinearLayout>
  25. </com.winuxxan.inputMethodTest.ResizeLayout>

AndroidManifest.xml的Activity设置属性:android:windowSoftInputMode = "adjustResize"
运行程序,点击文本框,查看调试信息:
E/onMeasure 6(7960): =>onMeasure called! widthMeasureSpec=1073742144, heightMeasureSpec = 1073742024
E/onMeasure 7(7960): =>onMeasure called! widthMeasureSpec=1073742144, heightMeasureSpec = 1073742025
E/onSizeChanged 8(7960): =>onSizeChanged called! w=320,h=201,oldw=320,oldh=377
E/onLayout 9(7960): =>OnLayout called! l=0, t=0,r=320,b=201
从调试结果我们可以看出,当我们点击文本框后,根布局调用了onMeasure,onSizeChanged和onLayout。
实际上,当设置为adjustResize后,软键盘弹出时,要对主窗口布局重新进行measure和layout,而在layout时,发现窗口的大小发生的变化,因此调用了onSizeChanged。
从下图的运行结果我们也可以看出,原本在下方的TextView被顶到了输入法的上方。

模式二,平移模式
windowSoftInputMode的值如果设置为adjustPan,那么该Activity主窗口并不调整屏幕的大小以便留出软键盘的空间。相反,当前窗口的内容将自动移动以便当前焦点从不被键盘覆盖和用户能总是看到输入内容的部分。这个通常是不期望比调整大小,因为用户可能关闭软键盘以便获得与被覆盖内容的交互操作。
上面的例子中,我们将AndroidManifest.xml的属性进行更改:android: windowSoftInputMode = "adjustPan"

重新运行,并点击文本框,查看调试信息:
E/onMeasure 6(8378): =>onMeasure called! widthMeasureSpec=1073742144, heightMeasureSpec=1073742200
E/onMeasure 7(8378): =>onMeasure called! widthMeasureSpec=1073742144, heightMeasureSpec=1073742201
E/onLayout 8(8378): =>OnLayout called! l=0, t=0,r=320,b=377
我们看到:系统也重新进行了measrue和layout,但是我们发现,layout过程中onSizeChanged并没有调用,这说明输入法弹出前后并没有改变原有布局的大小。
从下图的运行结果我们可以看到,下方的TextView并没有被顶到输入法上方。

事实上,当输入框不会被遮挡时,该模式没有对布局进行调整,然而当输入框将要被遮挡时,窗口就会进行平移。也就是说,该模式始终是保持输入框为可见。如下图,整个窗口,包括标题栏均被上移,以保证文本框可见。

模式三 自动模式
当属性windowSoftInputMode被设置为adjustUspecified时,它不被指定是否该Activity主窗口调整大小以便留出软键盘的空间,或是否窗口上的内容得到屏幕上当前的焦点是可见的。系统将自动选择这些模式中一种主要依赖于是否窗口的内容有任何布局视图能够滚动他们的内容。如果有这样的一个视图,这个窗口将调整大小,这样的假设可以使滚动窗口的内容在一个较小的区域中可见的。这个是主窗口默认的行为设置。
也就是说,系统自动决定是采用平移模式还是压缩模式,决定因素在于内容是否可以滚动。

三、侦听软键盘的显示隐藏
有时候,借助系统本身的机制来实现主窗口的调整并非我们想要的结果,我们可能希望在软键盘显示隐藏的时候,手动的对布局进行修改,以便使软键盘弹出时更加美观。这时就需要对软键盘的显示隐藏进行侦听。
直接对软键盘的显示隐藏侦听的方法本人没有找到,如果哪位找到的方法请务必告诉本人一声。还有本方法针对压缩模式,平移模式不一定有效。
我们可以借助软键盘显示和隐藏时,对主窗口进行了重新布局这个特性来进行侦听。如果我们设置的模式为压缩模式,那么我们可以对布局的onSizeChanged函数进行跟踪,如果为平移模式,那么该函数可能不会被调用。
我们可以重写根布局,因为根布局的高度一般情况下是不发生变化的。
假设跟布局为线性布局,模式为压缩模式,我们写一个例子,当输入法弹出时隐藏某个view,输入法隐藏时显示某个view。

 
  1. publicclassResizeLayoutextendsLinearLayout{
  2. privateOnResizeListenermListener;
  3. publicinterfaceOnResizeListener{
  4. voidOnResize(intw,inth,intoldw,intoldh);
  5. }
  6. publicvoidsetOnResizeListener(OnResizeListenerl){
  7. mListener=l;
  8. }
  9. publicResizeLayout(Contextcontext,AttributeSetattrs){
  10. super(context,attrs);
  11. }
  12. @Override
  13. protectedvoidonSizeChanged(intw,inth,intoldw,intoldh){
  14. super.onSizeChanged(w,h,oldw,oldh);
  15. if(mListener!=null){
  16. mListener.OnResize(w,h,oldw,oldh);
  17. }
  18. }
  19. }

在我们的Activity中,通过如下方法调用:

 
  1. publicclassInputMethodTestActivityextendsActivity{
  2. privatestaticfinalintBIGGER=1;
  3. privatestaticfinalintSMALLER=2;
  4. privatestaticfinalintMSG_RESIZE=1;
  5. privatestaticfinalintHEIGHT_THREADHOLD=30;
  6. classInputHandlerextendsHandler{
  7. @Override
  8. publicvoidhandleMessage(Messagemsg){
  9. switch(msg.what){
  10. caseMSG_RESIZE:{
  11. if(msg.arg1==BIGGER){
  12. findViewById(R.id.bottom_layout).setVisibility(View.VISIBLE);
  13. }else{
  14. findViewById(R.id.bottom_layout).setVisibility(View.GONE);
  15. }
  16. }
  17. break;
  18. default:
  19. break;
  20. }
  21. super.handleMessage(msg);
  22. }
  23. }
  24. privateInputHandlermHandler=newInputHandler();
  25. /**Calledwhentheactivityisfirstcreated.*/
  26. @Override
  27. publicvoidonCreate(BundlesavedInstanceState){
  28. super.onCreate(savedInstanceState);
  29. setContentView(R.layout.main);
  30. ResizeLayoutlayout=(ResizeLayout)findViewById(R.id.root_layout);
  31. layout.setOnResizeListener(newResizeLayout.OnResizeListener(){
  32. publicvoidOnResize(intw,inth,intoldw,intoldh){
  33. intchange=BIGGER;
  34. if(h<oldh){
  35. change=SMALLER;
  36. }
  37. Messagemsg=newMessage();
  38. msg.what=1;
  39. msg.arg1=change;
  40. mHandler.sendMessage(msg);
  41. }
  42. });
  43. }
  44. }

这里特别需要注意的是,不能直接在OnResizeListener中对要改变的View进行更改,因为OnSizeChanged函数实际上是运行在View的layout方法中,如果直接在onSizeChange中改变view的显示属性,那么很可能需要重新调用layout方法才能显示正确。然而我们的方法又是在layout中调用的,因此会出现错误。因此我们在例子中采用了Handler的方法。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
可以通过监听软键盘的状态来实现监听键盘弹出。 首先,在你的 AndroidManifest.xml 文件中,给当前的 activity 设置 `android:windowSoftInputMode` 属性为 `adjustResize` 或者 `adjustPan`,以便在键盘弹出时调整界面布局。例如: ```xml <activity android:name=".MainActivity" android:windowSoftInputMode="adjustResize"> ... </activity> ``` 然后,在当前 activity 的代码中,使用 `ViewTreeObserver` 监听布局变化事件,以便在键盘弹出或隐藏时做出相应的处理。具体步骤如下: 1. 获取当前 activity 的根视图(例如,使用 `findViewById(android.R.id.content)` 方法)。 2. 给根视图的 `ViewTreeObserver` 添加一个布局变化监听器(例如,使用 `addOnGlobalLayoutListener()` 方法)。 3. 在监听器的回调方法中,获取当前根视图的高度,并计算出当前键盘的高度。 4. 根据键盘的高度和当前根视图的高度,判断键盘是否弹出,并做出相应的处理(例如,调整界面布局或显示/隐藏某些控件)。 下面是一个示例代码: ```java View rootView = findViewById(android.R.id.content); rootView.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() { @Override public void onGlobalLayout() { Rect r = new Rect(); rootView.getWindowVisibleDisplayFrame(r); int screenHeight = rootView.getRootView().getHeight(); int keyboardHeight = screenHeight - r.bottom; // 判断键盘是否弹出,并做出相应的处理 if (keyboardHeight > screenHeight * 0.15) { // 键盘弹出 // TODO: 处理键盘弹出事件 } else { // 键盘隐藏 // TODO: 处理键盘隐藏事件 } } }); ``` 需要注意的是,由于 Android 版本的不同,键盘弹出时的界面布局调整可能会有所不同。在某些情况下,使用 `adjustPan` 属性可能会更适合,具体可以根据实际情况进行选择。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值