关于系统软键盘弹出隐藏监听的讨论
最近项目有一个比较苦B的需求,需要在用户输入内容之前,弹出推赞窗口,以让用户可以从中选择一些系统推荐的内容。这个需求通过拦截焦点,可以搞掂软键盘和推荐弹窗的冲突。但是更苦B的是,输入控件设计在屏幕的最下面(Bottom),这个导致了一个十分怪异的现象:
?
默认下输入控件是没有焦点的,当用户点击时,系统软键盘会自动优先弹出。因为有推荐功能,我们要先让用户看到推荐弹出窗。那就要把原来弹出的软件人为地隐藏,再弹出推荐窗,问题就出在这个环节上。
?
由于软件盘得隐藏有一定的时间,弹窗定苗就定在输入控件原来的位置(就是被软键盘推高的位置),软键盘隐藏后,弹窗就出现在屏幕的中部,但是输入控件已经回落到底部了!
?
?
解析问题之前,我们先了解一下,软键盘弹出的方式:
?
在AndroidManifest.xml文件中的Activity标签的属性中,可以找到android:windowSoftInputMode,其各个值的意义见下:
?
【A】stateUnspecified:软键盘的状态并没有指定,系统将选择一个合适的状态或依赖于主题的设置 【B】stateUnchanged:当这个activity出现时,软键盘将一直保持在上一个activity里的状态,无论是隐藏还是显示 【C】stateHidden:用户选择activity时,软键盘总是被隐藏 【D】stateAlwaysHidden:当该Activity主窗口获取焦点时,软键盘也总是被隐藏的 【E】stateVisible:软键盘通常是可见的 【F】stateAlwaysVisible:用户选择activity时,软键盘总是显示的状态 【G】adjustUnspecified:默认设置,通常由系统自行决定是隐藏还是显示 【H】adjustResize:该Activity总是调整屏幕的大小以便留出软键盘的空间 【I】adjustPan:当前窗口的内容将自动移动以便当前焦点从不被键盘覆盖和用户能总是看到输入内容的部分?
我设置的就是stateHidden|adjustResize,所以软键盘会托起activity的UI,隐藏的时候,UI重画,那当然就是耗费一点时间了。
?
既然知道问题,那解决方案就有了:监听Activity重画的状态,当Activity重画完才弹出推荐窗。
?
思路和简单,实现也不复杂,Activity重画,必定会回调某些内部的函数,我们只要重写那些函数,再发通知出来就可以了。
?
Activity的主要布局是RelativeLayout,由于属性设的是adjustResize,当软键盘隐藏时,会调用RelativeLayout的onSizeChanged方法,我们就从这里入手:
?
public class ResizeLayout extends RelativeLayout {public ResizeLayout(Context context) {super(context);// TODO Auto-generated constructor stub}public ResizeLayout(Context context, AttributeSet attrs, int defStyle) {super(context, attrs, defStyle);// TODO Auto-generated constructor stub}public ResizeLayout(Context context, AttributeSet attrs) {super(context, attrs);// TODO Auto-generated constructor stub}public void setOnResizeListener(OnResizeListener l) {mListener = l;}@Overrideprotected void onSizeChanged(int w, int h, int oldw, int oldh) {super.onSizeChanged(w, h, oldw, oldh);if (mListener != null) {mListener.OnResize(w, h, oldw, oldh);}}private OnResizeListener mListener;public interface OnResizeListener {void OnResize(int w, int h, int oldw, int oldh);};}?
我们重写onSizeChanged方法,并为自定义的layout定义一个回调接口,其实现的子类只要实现该接口,当使用此layout的Activity在软键盘隐藏时,就会通知到实现的子类了!
?
?
?
?