一:簡述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); } });