一:简述android
点击文本框EditText,系统会自动弹出软键盘(其本质是一个Dialog),这必然会引发当前Activity主窗口的大小调整ide
而Android提供了不一样的可选模式去调整活动窗口的大小,与之相关的属性为:android:windowSoftInputMode, 固然具体的实现是由系统完成的布局
能够在清单文件Manifest.xml中的Activity标签内设置测试
如:android:windowSoftInputMode="stateUnspecified|adjustPan"this
该属性可选的值有两部分,一部分为软键盘的状态控制,另外一部分是活动主窗口的调整。前一部分本文不作讨论,请读者自行查阅android文档。日志
一: 压缩模式xml
android:windowSoftInputMode="adjustResize", 那么无论活动主窗口压缩后文本框EditText是否可见(这将于下面一种模式造成对比),接口
当前Activity主窗口顶部保持不变,老是被从下向上压缩,压缩的距离等于软键盘的高度ci
测试代码(Android2.3.3SDK):文档
public class CustomRelativeLayout extends RelativeLayout{ public CustomRelativeLayout(Context context, AttributeSet attrs) { super(context, attrs); } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { Log.i("lanyan", "onMeasure"); super.onMeasure(widthMeasureSpec, heightMeasureSpec); } @Override protected void onLayout(boolean changed, int l, int t, int r, int b) { Log.i("lanyan", "onLayout"); super.onLayout(changed, l, t, r, b); } /** * 当前活动主窗口大小改变时调用 */ @Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { Log.i("lanyan", "onSizeChanged"); super.onSizeChanged(w, h, oldw, oldh); } }
<com.lanyan.drawable.widget.CustomRelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" > <TextView android:layout_width="fill_parent" android:layout_height="20dp" android:background="#9999CC" android:layout_alignParentTop="true" android:text="============= 我在顶部 ==============" android:textColor="#FFFFFF"/> <EditText android:layout_width="fill_parent" android:layout_height="wrap_content" android:layout_marginTop="220dp" /> <TextView android:layout_width="fill_parent" android:layout_height="20dp" android:background="#9999CC" android:layout_alignParentBottom="true" android:text="============= 我在底部 ==============" android:textColor="#FFFFFF"/>com.lanyan.drawable.widget.CustomRelativeLayout>
运行程序,点击文本框,日志顺序以下:
onMeasure
onSizeChanged
onLayout
实际上,当设置为adjustResize后,软键盘弹出时,要对主窗口布局从新进行measure和layout,而在layout时,发现窗口的大小发生的变化,所以调用了onSizeChanged
示意图1:EditText距离底部的距离 < 软键盘高度
能够看到, 顶部不变,底部被软键盘顶了上去,文本框被遮住
2,微调xml布局,使得EditText距离底部的距离 > 软键盘高度
二: 平移模式
android:windowSoftInputMode="adjustPan",此模式下,Activity主窗口大小始终保持不变,无论是否平移,
老是保证文本框不被软键盘覆盖且输入内容可见
上述代码的日志信息:
onMeasure
onLayout
能够看到,并不会调用onSizeChanged()方法
示意图1, EditText距离底部距离 < 软键盘高度
底部TextView并无被顶上去,而是活动主窗口总体向上平移(包括标题栏),具体平移的距离由系统measure,以便确保文本框可见
2,EditText距离底部的距离 > 软键盘高度
与上面相似,只是窗口并未平移,由于即便软键盘弹出,也不影响文本框是否可见,这种状况下,软键盘等因而"浮动"在Activity上面
三: 自动模式
android:windowSoftInputMode="adjustUnspecified"
系统自动决定是采用平移模式仍是压缩模式,决定因素在于内容是否能够滚动。
二:监听软键盘的显示隐藏
能够经过onSizeChanged()方法间接地对软键盘的显示隐藏进行监听(并未精确到软键盘显示隐藏以前/以后这种程度),从而能够在主窗口大小发生变化时,进行自定义的操做,如显示或隐藏某个view
因为View类并未提供相似setOnClickListener(....)这样方便的接口,因此仍是要重写根布局,而且加个回调接口便可
此方法仅当Activity为压缩模式是有效,平移模式窗口大小不变,系统不会调用onSizeChanged()方法
public class CustomRelativeLayout extends RelativeLayout{ private KeyboardChangeListener listener; public CustomRelativeLayout(Context context, AttributeSet attrs) { super(context, attrs); } @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { Log.i("lanyan", "onMeasure"); super.onMeasure(widthMeasureSpec, heightMeasureSpec); } @Override protected void onLayout(boolean changed, int l, int t, int r, int b) { Log.i("lanyan", "onLayout"); super.onLayout(changed, l, t, r, b); } /** * 当前活动主窗口大小改变时调用 */ @Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { Log.i("lanyan", "onSizeChanged"); super.onSizeChanged(w, h, oldw, oldh); if (null != listener) { listener.onKeyboardChange(w, h, oldw, oldh); } } public void setOnKeyboardChangeListener(KeyboardChangeListener listener) { this.listener = listener; } /** * Activity主窗口大小改变时的回调接口(本示例中,等价于软键盘显示隐藏时的回调接口) * @author mo * */ public interface KeyboardChangeListener { public void onKeyboardChange(int w, int h, int oldw, int oldh); } }
public class EditActivity extends Activity{ @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.edit); CustomRelativeLayout customRelativeLayout = (CustomRelativeLayout)findViewById(R.id.relativelayout1); customRelativeLayout.setOnKeyboardChangeListener(new CustomRelativeLayout.KeyboardChangeListener() { @Override public void onKeyboardChange(int w, int h, int oldw, int oldh) { //do your operation } }); } }
PS: 上述软键盘的弹出都是点击文本框,系统自动弹出的
也能够经过代码的方式手动控制软键盘的显示与隐藏(Android2.3.3SDK)
tv.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { InputMethodManager imm = (InputMethodManager)getSystemService(Context.INPUT_METHOD_SERVICE); //隐藏软键盘// imm.hideSoftInputFromWindow(tv.getWindowToken(), 0); //显示软键盘// imm.showSoftInputFromInputMethod(tv.getWindowToken(), 0); //切换软键盘的显示与隐藏 imm.toggleSoftInputFromWindow(tv.getWindowToken(), 0, InputMethodManager.HIDE_NOT_ALWAYS); //或者// imm.toggleSoftInput(0, InputMethodManager.HIDE_NOT_ALWAYS); } });