1. 程式人生 > >android軟鍵盤的顯示後隱藏

android軟鍵盤的顯示後隱藏

轉載:http://www.apkbus.com/android-59367-1-1.html

一:簡述

點選文字框EditText,系統會自動彈出軟鍵盤(其本質是一個Dialog),這必然會引起當前Activity主視窗的大小調整
而Android提供了不同的可選模式去調整活動視窗的大小,與之相關的屬性為:android:windowSoftInputMode, 當然具體的實現是由系統完成的
可以在清單檔案Manifest.xml中的Activity標籤內設定
如:android:windowSoftInputMode="stateUnspecified|adjustPan"
該屬性可選的值有兩部分,一部分為軟鍵盤的狀態控制,另一部分是活動主視窗的調整。前一部分本文不做討論,請讀者自行查閱android文件。


一: 壓縮模式
android:windowSoftInputMode="adjustResize", 那麼不管活動主視窗壓縮後文本框EditText是否可見(這將於下面一種模式形成對比),
當前Activity主視窗頂部保持不變,總是被從下向上壓縮,壓縮的距離等於軟鍵盤的高度
測試程式碼(Android2.3.3SDK):
  1. public class CustomRelativeLayout extends RelativeLayout{
  2.     public CustomRelativeLayout(Context context, AttributeSet attrs) {
  3.         super(context, attrs);
  4.     }
  5.     @Override
  6.     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
  7.         Log.i("lanyan", "onMeasure");
  8.         super.onMeasure(widthMeasureSpec, heightMeasureSpec);
  9.     }
  10.     @Override
  11.     protected void onLayout(boolean changed, int l, int t, int r, int b) {
  12.         Log.i("lanyan", "onLayout");
  13.         super.onLayout(changed, l, t, r, b);
  14.     }
  15.     /**
  16.      * 當前活動主視窗大小改變時呼叫
  17.      */
  18.     @Override
  19.     protected void onSizeChanged(int w, int h, int oldw, int oldh) {
  20.         Log.i("lanyan", "onSizeChanged");
  21.         super.onSizeChanged(w, h, oldw, oldh);
  22.     }
  23. }
複製程式碼
  1. <com.lanyan.drawable.widget.customrelativelayout xmlns:android="http://schemas.android.com/apk/res/android" 
  2.     android:layout_width="fill_parent"
  3.     android:layout_height="fill_parent"
  4.     android:orientation="vertical"
  5.     >
  6.     <textview android:layout_width="fill_parent" 
  7.     android:layout_height="20dp" android:background="#9999CC"
  8.     android:layout_alignParentTop="true" android:text="============= 我在頂部 =============="
  9.     android:textColor="#FFFFFF"/>    
  10.     <edittext android:layout_width="fill_parent" 
  11.     android:layout_height="wrap_content" 
  12.     android:layout_marginTop="220dp" />
  13.     <textview android:layout_width="fill_parent" 
  14.     android:layout_height="20dp" android:background="#9999CC"
  15.     android:layout_alignParentBottom="true" android:text="============= 我在底部 =============="
  16.     android:textColor="#FFFFFF"/>
複製程式碼 執行程式,點選文字框,日誌順序如下:
onMeasure
onSizeChanged
onLayout
實際上,當設定為adjustResize後,軟鍵盤彈出時,要對主窗口布局重新進行measure和layout,而在layout時,發現視窗的大小發生的變化,因此呼叫了onSizeChanged
示意圖1:EditText距離底部的距離 < 軟鍵盤高度
2012071500281924.png2012071500283045.png
可以看到, 頂部不變,底部被軟鍵盤頂了上去,文字框被遮住
2,微調xml佈局,使得EditText距離底部的距離 > 軟鍵盤高度
2012071500403320.png2012071500405871.png

二: 平移模式
android:windowSoftInputMode="adjustPan",此模式下,Activity主視窗大小始終保持不變,不管是否平移,
總是保證文字框不被軟鍵盤覆蓋且輸入內容可見
上述程式碼的日誌資訊:
onMeasure
onLayout
可以看到,並不會呼叫onSizeChanged()方法
示意圖1, EditText距離底部距離 < 軟鍵盤高度
2012071500443369.png2012071500452269.png
底部TextView並沒有被頂上去,而是活動主視窗整體向上平移(包括標題欄),具體平移的距離由系統measure,以便確保文字框可見
2,EditText距離底部的距離 > 軟鍵盤高度

與上面類似,只是視窗並未平移,因為即使軟鍵盤彈出,也不影響文字框是否可見,這種情況下,軟鍵盤等於是"浮動"在Activity上面

三: 自動模式
android:windowSoftInputMode="adjustUnspecified"
系統自動決定是採用平移模式還是壓縮模式,決定因素在於內容是否可以滾動。

二:監聽軟鍵盤的顯示隱藏
可以通過onSizeChanged()方法間接地對軟鍵盤的顯示隱藏進行監聽(並未精確到軟鍵盤顯示隱藏之前/之後這種程度),從而可以在主視窗大小發生變化時,進行自定義的操作,如顯示或隱藏某個view

由於View類並未提供類似setOnClickListener(....)這樣方便的介面,所以還是要重寫根佈局,並且加個回撥介面即可
此方法僅當Activity為壓縮模式是有效,平移模式視窗大小不變,系統不會呼叫onSizeChanged()方法
  1. public class CustomRelativeLayout extends RelativeLayout{
  2.     private KeyboardChangeListener listener;
  3.     public CustomRelativeLayout(Context context, AttributeSet attrs) {
  4.         super(context, attrs);
  5.     }
  6.     @Override
  7.     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
  8.         Log.i("lanyan", "onMeasure");
  9.         super.onMeasure(widthMeasureSpec, heightMeasureSpec);
  10.     }
  11.     @Override
  12.     protected void onLayout(boolean changed, int l, int t, int r, int b) {
  13.         Log.i("lanyan", "onLayout");
  14.         super.onLayout(changed, l, t, r, b);
  15.     }
  16.     /**
  17.      * 當前活動主視窗大小改變時呼叫
  18.      */
  19.     @Override
  20.     protected void onSizeChanged(int w, int h, int oldw, int oldh) {
  21.         Log.i("lanyan", "onSizeChanged");
  22.         super.onSizeChanged(w, h, oldw, oldh);
  23.         if (null != listener) {
  24.             listener.onKeyboardChange(w, h, oldw, oldh);
  25.         }
  26.     }
  27.     public void setOnKeyboardChangeListener(KeyboardChangeListener listener) {
  28.         this.listener = listener;
  29.     }
  30.     /**
  31.      * Activity主視窗大小改變時的回撥介面(本示例中,等價於軟鍵盤顯示隱藏時的回撥介面)
  32.      * @author mo
  33.      *
  34.      */
  35.     public interface KeyboardChangeListener {
  36.         public void onKeyboardChange(int w, int h, int oldw, int oldh);
  37.     }
  38. }
複製程式碼
  1. public class EditActivity extends Activity{
  2.     @Override
  3.     protected void onCreate(Bundle savedInstanceState) {
  4.         super.onCreate(savedInstanceState);
  5.         setContentView(R.layout.edit);
  6.         CustomRelativeLayout customRelativeLayout = (CustomRelativeLayout)findViewById(R.id.relativelayout1);
  7.         customRelativeLayout.setOnKeyboardChangeListener(new CustomRelativeLayout.KeyboardChangeListener() {
  8.             @Override
  9.             public void onKeyboardChange(int w, int h, int oldw, int oldh) {
  10.                 //do your operation
  11.             }
  12.         });
  13.     }
  14. }
複製程式碼 PS: 上述軟鍵盤的彈出都是點選文字框,系統自動彈出的
也可以通過程式碼的方式手動控制軟鍵盤的顯示與隱藏(Android2.3.3SDK)
  1. tv.setOnClickListener(new OnClickListener() {
  2.             @Override
  3.             public void onClick(View v) {
  4.                 InputMethodManager imm = (InputMethodManager)getSystemService(Context.INPUT_METHOD_SERVICE);
  5.                 //隱藏軟鍵盤
  6. //                imm.hideSoftInputFromWindow(tv.getWindowToken(), 0);
  7.                 //顯示軟鍵盤
  8. //                imm.showSoftInputFromInputMethod(tv.getWindowToken(), 0);
  9.                 //切換軟鍵盤的顯示與隱藏
  10.                 imm.toggleSoftInputFromWindow(tv.getWindowToken(), 0, InputMethodManager.HIDE_NOT_ALWAYS);
  11.                 //或者
  12. //                imm.toggleSoftInput(0, InputMethodManager.HIDE_NOT_ALWAYS);
  13.             }
  14.         });