這篇文章將為大家詳細(xì)講解有關(guān)Android中如何獲取軟鍵盤狀態(tài)和軟鍵盤高度,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。
網(wǎng)站建設(shè)哪家好,找成都創(chuàng)新互聯(lián)公司!專注于網(wǎng)頁設(shè)計、網(wǎng)站建設(shè)、微信開發(fā)、微信小程序定制開發(fā)、集團企業(yè)網(wǎng)站建設(shè)等服務(wù)項目。為回饋新老客戶創(chuàng)新互聯(lián)還提供了青原免費建站歡迎大家使用!
詳解Android中獲取軟鍵盤狀態(tài)和軟鍵盤高度
應(yīng)用場景
在Android應(yīng)用中有時會需要獲取軟鍵盤的狀態(tài)(即軟鍵盤是顯示還是隱藏)和軟鍵盤的高度。這里列舉了一些可能的應(yīng)用場景。
場景一
當(dāng)軟鍵盤顯示時,按下返回鍵應(yīng)當(dāng)是收起軟鍵盤,而不是回退到上一個界面,但部分機型在返回鍵處理上有bug,按下返回鍵后,雖然軟鍵盤會自動收起,但不會消費返回事件,導(dǎo)致Activity還會收到這次返回事件,執(zhí)行回退操作,這時就需要判斷,如果軟鍵盤剛剛由顯示變?yōu)殡[藏狀態(tài),就不執(zhí)行回退操作。
場景二
當(dāng)軟鍵盤彈出后,會將界面底部到中間的一大部分全部擋住,如果用戶要查看、操作被覆蓋的區(qū)域,必須先收起軟鍵盤,這會影響用戶交互。所以通常需要在軟鍵盤彈出后,將底部的一些View,例如Button,移到軟鍵盤的上方,方便用戶操作。
API的困境
Android SDK中沒有提供任何API來直接獲取軟鍵盤的狀態(tài)和軟鍵盤的高度,網(wǎng)上很多資料說InputMethodManager的isActive()方法可以獲取軟鍵盤狀態(tài),不過實際測試發(fā)現(xiàn),這個方法并沒有什么用,如果它返回false,可以判斷軟鍵盤一定是隱藏的,但如果它返回true,軟鍵盤既可能是顯示的,也可能是隱藏的。所以并不能通過isActive()方法來判斷軟鍵盤究竟是顯示還是隱藏的。要想獲取軟鍵盤的狀態(tài)和軟鍵盤的高度,只能通過間接方法實現(xiàn)。
注冊布局變化監(jiān)聽
在Android中當(dāng)軟鍵盤由隱藏變?yōu)轱@示,或由顯示變?yōu)殡[藏時,會觸發(fā)當(dāng)前布局中View的全局布局變化。通過監(jiān)聽全局布局的變化就可以得知軟鍵盤的狀態(tài)。
Android框架提供了一個ViewTreeObserver類,它是一個View視圖樹的觀察者類。ViewTreeObserver類中定義了一系列的公共接口(public interface)。當(dāng)一個View attach到一個窗口上時就會創(chuàng)建一個ViewTreeObserver對象,這樣當(dāng)一個View的視圖樹發(fā)生改變時,就會調(diào)用該對象的某個方法,將事件通知給每個注冊的監(jiān)聽者。
OnGlobalLayoutListener是ViewTreeObserver中定義的眾多接口中的一個,它用來監(jiān)聽一個視圖樹中全局布局的改變或者視圖樹中的某個視圖的可視狀態(tài)的改變。當(dāng)軟鍵盤由隱藏變?yōu)轱@示,或由顯示變?yōu)殡[藏時,都會調(diào)用當(dāng)前布局中所有存在的View中的ViewTreeObserver對象的dispatchOnGlobalLayout()方法,此方法中會遍歷所有已注冊的OnGlobalLayoutListener,執(zhí)行相應(yīng)的回調(diào)方法,將全局布局改變的消息通知給每個注冊的監(jiān)聽者。
向一個View中的ViewTreeObserver注冊O(shè)nGlobalLayoutListener的方法如下。
view.getViewTreeObserver().addOnGlobalLayoutListener(listener);
注冊O(shè)nGlobalLayoutListener時有一些需要注意的地方。
注冊的監(jiān)聽在不使用時需要調(diào)用removeOnGlobalLayoutListener或removeGlobalOnLayoutListener來移除監(jiān)聽,不然可能會導(dǎo)致內(nèi)存泄露。通常可以在Activity的onCreate()方法中注冊監(jiān)聽,在onDestory()方法中移除監(jiān)聽。
并不是只有顯示和隱藏軟鍵盤會觸發(fā)OnGlobalLayoutListener中的回調(diào),一個View在繪制完成,或者消失時都會觸發(fā)OnGlobalLayoutListener中的回調(diào)(由于在onCreate中無法獲取一個View的寬度和高度,很多時候就是通過注冊O(shè)nGlobalLayoutListener,在OnGlobalLayoutListener的回調(diào)中來獲取一個View的寬度和高度)。
為了在OnGlobalLayoutListener的回調(diào)中準(zhǔn)確的判斷是否是由于軟鍵盤狀態(tài)改變引起的,以及獲取軟鍵盤的高度,還需要另外一個接口。
獲取當(dāng)前窗口可見的顯示區(qū)域大小
在View中提供了一個方法getWindowVisibleDisplayFrame(),此方法會返回該view所附著的窗口的可見區(qū)域大小。當(dāng)軟鍵盤顯示時,窗口的可見區(qū)域大小會被壓縮,當(dāng)軟鍵盤隱藏時,窗口的可見區(qū)域大小會還原。不過并不是只有軟鍵盤的顯示和隱藏會影響窗口的可見區(qū)域大小,像大多數(shù)的平板和部分手機上有一排虛擬按鍵(虛擬的返回鍵,Home鍵等),虛擬按鍵的顯示和隱藏也會引起窗口可見區(qū)域的變化。不過好在除了軟鍵盤外,其他操作對窗口可見區(qū)域的影響占整個屏幕大小的比例都不是很大,通過設(shè)置一個合理的閾值,就可以較準(zhǔn)確的判斷出是否是軟鍵盤顯示和隱藏引起的布局變化。
此外,getWindowVisibleDisplayFrame()會返回窗口的可見區(qū)域高度,通過和屏幕高度相減,就可以得到軟鍵盤的高度了。
監(jiān)聽軟鍵盤的狀態(tài)變化
在獲取到軟鍵盤的狀態(tài)和高度后就可以執(zhí)行需要的操作了。如重新布局按鈕位置,設(shè)置變量,記錄當(dāng)前軟鍵盤狀態(tài)和上次軟鍵盤隱藏時間等。不過如果有多個類需要根據(jù)軟鍵盤狀態(tài)來執(zhí)行一些操作,如果每個類中都去這樣做一遍就很麻煩,而且也沒有必要。這時在可以自行定義一個接口,在主Activity中對軟鍵盤狀態(tài)變化進行監(jiān)聽,其他對軟鍵盤狀態(tài)感興趣的類,向主Activity中注冊軟鍵盤狀態(tài)變化監(jiān)聽。在主Activity中,當(dāng)軟鍵盤狀態(tài)發(fā)生改變時通知監(jiān)聽者。
完整示例代碼
完整的示例代碼如下。
public interface OnSoftKeyboardStateChangedListener { public void OnSoftKeyboardStateChanged(boolean isKeyBoardShow, int keyboardHeight); } //注冊軟鍵盤狀態(tài)變化監(jiān)聽 public void addSoftKeyboardChangedListener(OnSoftKeyboardStateChangedListener listener) { if (listener != null) { mKeyboardStateListeners.add(listener); } } //取消軟鍵盤狀態(tài)變化監(jiān)聽 public void removeSoftKeyboardChangedListener(OnSoftKeyboardStateChangedListener listener) { if (listener != null) { mKeyboardStateListeners.remove(listener); } } private ArrayListmKeyboardStateListeners; //軟鍵盤狀態(tài)監(jiān)聽列表 private OnGlobalLayoutListener mLayoutChangeListener; private boolean mIsSoftKeyboardShowing; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_login); mIsSoftKeyboardShowing = false; mKeyboardStateListeners = new ArrayList (); mLayoutChangeListener = new OnGlobalLayoutListener() { @Override public void onGlobalLayout() { //判斷窗口可見區(qū)域大小 Rect r = new Rect(); getWindow().getDecorView().getWindowVisibleDisplayFrame(r); //如果屏幕高度和Window可見區(qū)域高度差值大于整個屏幕高度的1/3,則表示軟鍵盤顯示中,否則軟鍵盤為隱藏狀態(tài)。 int heightDifference = screenHeight - (r.bottom - r.top); boolean isKeyboardShowing = heightDifference > screenHeight/3; //如果之前軟鍵盤狀態(tài)為顯示,現(xiàn)在為關(guān)閉,或者之前為關(guān)閉,現(xiàn)在為顯示,則表示軟鍵盤的狀態(tài)發(fā)生了改變 if ((mIsSoftKeyboardShowing && !isKeyboardShowing) || (!mIsSoftKeyboardShowing && isKeyboardShowing)) { mIsSoftKeyboardShowing = isKeyboardShowing; for (int i = 0; i < mKeyboardStateListeners.size(); i++) { OnSoftKeyboardStateChangedListener listener = mKeyboardStateListeners.get(i); listener.OnSoftKeyboardStateChanged(mIsSoftKeyboardShowing, heightDifference); } } } }; //注冊布局變化監(jiān)聽 getWindow().getDecorView().getViewTreeObserver().addOnGlobalLayoutListener(mLayoutChangeListener); } @SuppressWarnings("deprecation") @SuppressLint("NewApi") @Override protected void onDestroy() { //移除布局變化監(jiān)聽 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) { getWindow().getDecorView().getViewTreeObserver().removeOnGlobalLayoutListener(mLayoutChangeListener); } else { getWindow().getDecorView().getViewTreeObserver().removeGlobalOnLayoutListener(mLayoutChangeListener); } super.onDestroy(); };
其中screenHeight 是屏幕高度,關(guān)于屏幕高度的獲取方法,網(wǎng)上有很多,這里就不介紹了。
Android是一種基于Linux內(nèi)核的自由及開放源代碼的操作系統(tǒng),主要使用于移動設(shè)備,如智能手機和平板電腦,由美國Google公司和開放手機聯(lián)盟領(lǐng)導(dǎo)及開發(fā)。
關(guān)于“Android中如何獲取軟鍵盤狀態(tài)和軟鍵盤高度”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,使各位可以學(xué)到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。