彻底搞定Android开发中软键盘的常见问题

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了彻底搞定Android开发中软键盘的常见问题相关的知识,希望对你有一定的参考价值。

  软键盘显示的原理

  软件盘的本质是什么?软键盘其实是一个Dialog。

  InputMethodService为我们的输入法创建了一个Dialog,并且将该Dialog的Window的某些参数(如Gravity)进行了设置,使之能够在底部或者全屏显示。当我们点击输入框时,系统对活动主窗口进行调整,从而为输入法腾出相应的空间,然后将该Dialog显示在底部,或者全屏显示。

  软键盘显示的调整

  android定义了一个属性,名字为windowSoftInputMode, 这个属性用于设置Activity主窗口与软键盘的交互模式,用于避免软键盘遮挡内容的问题。我们可以在AndroidManifet.xml中对Activity进行设置。如:android:windowSoftInputMode=”stateUnchanged|adjustPan”。

  该属性可选的值有两部分,一部分为软键盘的状态控制,控制软键盘是隐藏还是显示,另一部分是Activity窗口的调整,以便腾出空间展示软键盘。

  android:windowSoftInputMode的属性设置必须是下面中的一个值,或一个”state”值加一个”adjust”值的组合,各个值之间用 | 分开。

  stateUnspecified-未指定状态:当我们没有设置android:windowSoftInputMode属性的时候,软件默认采用的就是这种交互方式,系统会根据界面采取相应的软键盘的显示模式。

  stateUnchanged-不改变状态:当前界面的软键盘状态,取决于上一个界面的软键盘状态,无论是隐藏还是显示。

  stateHidden-隐藏状态:当设置该状态时,软键盘总是被隐藏,不管是否有输入的需求。

  stateAlwaysHidden-总是隐藏状态:当设置该状态时,软键盘总是被隐藏,和stateHidden不同的是,当我们跳转到下个界面,如果下个页面的软键盘是显示的,而我们再次回来的时候,软键盘就会隐藏起来。

  stateVisible-可见状态:当设置为这个状态时,软键盘总是可见的,即使在界面上没有输入框的情况下也可以强制弹出来出来。

  stateAlwaysVisible-总是显示状态:当设置为这个状态时,软键盘总是可见的,和stateVisible不同的是,当我们跳转到下个界面,如果下个页面软键盘是隐藏的,而我们再次回来的时候,软键盘就会显示出来。

  adjustUnspecified-未指定模式:设置软键盘与软件的显示内容之间的显示关系。当你跟我们没有设置这个值的时候,这个选项也是默认的设置模式。在这中情况下,系统会根据界面选择不同的模式。

  adjustResize-调整模式:该模式下窗口总是调整屏幕的大小用以保证软键盘的显示空间;这个选项不能和adjustPan同时使用,如果这两个属性都没有被设置,系统会根据窗口中的布局自动选择其中一个。

  adjustPan-默认模式:该模式下通过不会调整来保证软键盘的空间,而是采取了另外一种策略,系统会通过布局的移动,来保证用户要进行输入的输入框肯定在用户的视野范围里面,从而让用户可以看到自己输入的内容。

  案例

  没有滚动布局xml

  

  android:layout_width="match_parent"

  android:layout_height="match_parent"

  android:orientation="vertical">

  <edittext< p="">

  android:id="@+id/et1"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框1" />

  <button< p="">

  android:id="@+id/btn1"

  android:layout_width="match_parent"

  android:layout_height="wrap_content"

  android:layout_marginTop="5dp"

  android:text="猴子搬来的救兵按钮"

  android:textSize="15sp" />

  <edittext< p="">

  android:id="@+id/et2"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框2" />

  <edittext< p="">

  android:id="@+id/et3"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框3" />

  <edittext< p="">

  android:id="@+id/et4"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框4" />

  <edittext< p="">

  android:id="@+id/et5"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框5" />

  <edittext< p="">

  android:id="@+id/et6"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框6" />

  <edittext< p="">

  android:id="@+id/et7"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框7" />

  <edittext< p="">

  android:id="@+id/et8"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框8" />

  <edittext< p="">

  android:id="@+id/et9"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框9" />

  <edittext< p="">

  android:id="@+id/et10"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框10" />

  <edittext< p="">

  android:id="@+id/et11"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框11" />

  <edittext< p="">

  android:id="@+id/et12"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框12" />

  12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485861234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586

  对于没有滚动控件的布局来说,adjustPan就是默认的设置,比如我们案例应用中的文本输入8,上面的文本输入框123和按钮都会被顶上去,且页面布局不可以滚动。

  

技术分享

 

  有滚动布局xml

  

  android:layout_width="match_parent"

  android:layout_height="match_parent">

  <linearlayout< p="">

  android:layout_width="match_parent"

  android:layout_height="match_parent"

  android:orientation="vertical">

  <edittext< p="">

  android:id="@+id/et1"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框1" />

  <button< p="">

  android:id="@+id/btn1"

  android:layout_width="match_parent"

  android:layout_height="wrap_content"

  android:layout_marginTop="5dp"

  android:text="猴子搬来的救兵按钮"

  android:textSize="15sp" />

  <edittext< p="">

  android:id="@+id/et2"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框2" />

  <edittext< p="">

  android:id="@+id/et3"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框3" />

  <edittext< p="">

  android:id="@+id/et4"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框4" />

  <edittext< p="">

  android:id="@+id/et5"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框5" />

  <edittext< p="">

  android:id="@+id/et6"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框6" />

  <edittext< p="">

  android:id="@+id/et7"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框7" />

  <edittext< p="">

  android:id="@+id/et8"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框8" />

  <edittext< p="">

  android:id="@+id/et9"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框9" />

  <edittext< p="">

  android:id="@+id/et10"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框10" />

  <edittext< p="">

  android:id="@+id/et11"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框11" />

  <edittext< p="">

  android:id="@+id/et12"

  android:layout_width="match_parent"

  android:layout_height="50dp"

  android:text="文本输入框12" />

  

  1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909112345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091

  对于有滚动控件的布局,则是采用的adjustResize方式,比如我们案例应用中的文本输入8,上面的文本输入框123和按钮都会被顶上去,可以通过滚动来查看被顶上去的内容。

  

技术分享

 

  

技术分享

 

  根据这一原理,我们就可以把开发中遇到的软键盘遮挡页面的问题,利用ScrollView当做根布局,让系统采用adjustResize模式,很好地解决这一问题。

  自动弹出软键盘

  有时候需要一进入Activity后就自动弹出软键盘,可以通过设置一个时间函数来实现,具体写法如下:

  方法一:

  Timer timer=new Timer();

  timer.schedule(new TimerTask() {

  public void run() {

  InputMethodManager inputMethodManager=(InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);

  inputMethodManager.toggleSoftInput(0, InputMethodManager.HIDE_NOT_ALWAYS);

  }

  }, 1000); // 秒后自动弹出1234567812345678

  方法二:

  Timer timer = new Timer();

  timer.schedule(new TimerTask() {

  public void run() {

  InputMethodManager inputManager =

  (InputMethodManager) etInput.getContext().getSystemService(Context.INPUT_METHOD_SERVICE);

  inputManager.showSoftInput(etInput, 0);

  }

  },

  1000);// 1秒后自动弹出12345678910111234567891011

  不自动弹出软键盘

  有时进入Activity后不希望系统自动弹出软键盘,我们可以按照下面的方法来实现:

  方法一:

  在AndroidMainfest.xml中选择那个activity,设置windowSoftInputMode属性为adjustUnspecified|stateHidden

  

  Android:label="@string/app_name"

  Android:windowSoftInputMode="adjustUnspecified|stateHidden"

  Android:configChanges="orientation|keyboardHidden">

  

  

  

  

  123456789123456789

  方法二:

  让EditText失去焦点,使用EditText的clearFocus方法

  EditText edit=(EditText)findViewById(R.id.edit);

  edit.clearFocus();1212

  方法三:

  强制隐藏Android输入法窗口

  EditText edit=(EditText)findViewById(R.id.edit);

  InputMethodManager imm = (InputMethodManager)getSystemService(Context.INPUT_METHOD_SERVICE);

  imm.hideSoftInputFromWindow(edit.getWindowToken(),0);123123

  方法四:

  EditText始终不弹出软件键盘

  EditText edit=(EditText)findViewById(R.id.edit);

  edit.setInputType(InputType.TYPE_NULL);1212

  EditText设置ScrollView压缩背景图片解决办法

  在你的Activity里加上

  getWindow().setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_ADJUST_PAN);11

  动态关闭软键盘

  有时希望根据条件动态关闭软键盘,我们可以使用InputMethodManager类,按照下面的方法来实现:

  方法一:

  InputMethodManager imm = (InputMethodManager)getSystemService(Context.INPUT_METHOD_SERVICE); //得到InputMethodManager的实例

  if (imm.isActive()) {//如果开启

  imm.toggleSoftInput(InputMethodManager.SHOW_IMPLICIT,InputMethodManager.HIDE_NOT_ALWAYS);//关闭软键盘,开启方法相同,这个方法是切换开启与关闭状态的

  }12341234

  方法二:

  强制隐藏软键盘

  public void KeyBoardCancle() {

  View view = getWindow().peekDecorView();

  if (view != null) {

  InputMethodManager inputmanger = (InputMethodManager) getSystemService(ActivityBase.INPUT_METHOD_SERVICE);

  inputmanger.hideSoftInputFromWindow(view.getWindowToken(), 0);

  }

  }1234567812345678

  方法三:

  int flags = WindowManager.LayoutParams.FLAG_ALT_FOCUSABLE_IM;

  getWindow().addFlags(flags); 1212

  方法四:

  在onclick事件下.以下方法可行.(如果是EditText失去焦点/得到焦点,没有效果)

  InputMethodManager im = (InputMethodManager)getSystemService(Context.INPUT_METHOD_SERVICE);

  im.hideSoftInputFromWindow(getCurrentFocus().getApplicationWindowToken(), InputMethodManager.HIDE_NOT_ALWAYS);1212

  InputMethodManager的具体用法可以参考下面的链接:

  http://www.apihome.cn/api/android/InputMethodManager.html

  软键盘界面按钮功能设置方法

  使用android:imeOptinos可对Android自带的软键盘进行一些界面上的设置:

  <edittext< p="">

  android:id="@+id/text1"

  android:layout_width="150dip"

  android:layout_height="wrap_content"

  android:imeOptions="flagNoExtractUi"/> 1234512345

  android:imeOptions="flagNoExtractUi" //使软键盘不全屏显示,只占用一部分屏幕 11

  同时,这个属性还能控件软键盘右下角按键的显示内容,默认情况下为回车键

  android:imeOptions="actionNone" //输入框右侧不带任何提示

  android:imeOptions="actionGo" //右下角按键内容为‘开始‘

  android:imeOptions="actionSearch" //右下角按键为放大镜图片,搜索

  android:imeOptions="actionSend" //右下角按键内容为‘发送‘

  android:imeOptions="actionNext" //右下角按键内容为‘下一步‘

  android:imeOptions="actionDone" //右下角按键内容为‘完成‘ 123456123456

  

技术分享

 

  

技术分享

 

  同时,可能EditText添加相应的监听器,捕捉用户点击了软键盘右下角按钮的监听事件,以便进行处理。

  editText.setOnEditorActionListener(new OnEditorActionListener() {

  @Override

  public boolean onEditorAction(TextView v, int actionId, KeyEvent event) {

  Toast.makeText(MainActivity.this, "响应了配置后的按键", Toast.LENGTH_SHORT).show();

  return false;

  }

  }); 12345671234567

  踩过的坑

  一、软键盘无法顶起页面

  开发中有个需求是将页面底部的一个按钮顶起,但是开发时发现Android5.0以后的版本设置了adjustResize属性后无法成功顶起。纠结了好久,最后在stackoverflow找到解决方案,那就是在根布局上加上fitsSystemWindow=”true”即可。

  

  android:layout_width="match_parent"

  android:layout_height="match_parent"

  android:fitsSystemWindows="true"

  >1234512345

  

技术分享

 

  这里的fitsSystemWindow具体的作用就是你的contentview是否忽略actionbar,title,屏幕的底部虚拟按键,将整个屏幕当作可用的空间。

  正常情况,contentview可用的空间是去除了actionbar,title,底部按键的空间后剩余的可用区域;这个属性设置为true,则忽略,false则不忽略

  二、自定义软键盘按钮功能无效

  在edittext上加入Android:imeOptions=”actionSearch”这个属性没响应,最后发现在2.3及以上版本不起作用,解决方案:加上

  android:singleLine="true"11

  因为输入法键盘右下角默认的回车键本来就是换行用的,当设置单行后,回车换行就失去作用了,这样就可以设置为搜索、发送、go等等。

以上是关于彻底搞定Android开发中软键盘的常见问题的主要内容,如果未能解决你的问题,请参考以下文章

Android WebView中软键盘会遮挡输入框相关问题

android dialog中软键盘遮挡输入编辑框edittextt

在混合开发中软键盘引起的布局问题input与可编辑DIV兼容性问题汇总

android EditText彻底搞定输入框隐藏/显示背景 可编辑输入问题

H5如何对android和ios手机软键盘的监听

Android 手机软键盘的弹起和关闭的监听