亚洲免费在线-亚洲免费在线播放-亚洲免费在线观看-亚洲免费在线观看视频-亚洲免费在线看-亚洲免费在线视频

Android軟鍵盤的隱藏顯示研究

系統 1847 0
Android是一個針對觸摸屏專門設計的操作系統,當點擊編輯框,系統自動為用戶彈出軟鍵盤,以便用戶進行輸入。
那么,彈出軟鍵盤后必然會造成原有布局高度的減少,那么系統應該如何來處理布局的減少?我們能否在應用程序中進行自定義的控制?這些是本文要討論的重點。

Android軟鍵盤的隱藏顯示研究

一、軟鍵盤顯示的原理
軟件盤的本質是什么?軟鍵盤其實是一個Dialog!
InputMethodService為我們的輸入法創建了一個Dialog,并且將該Dialog的Window的某些參數(如Gravity)進行了設置,使之能夠在底部或者全屏顯示。當我們點擊輸入框時,系統對活動主窗口進行調整,從而為輸入法騰出相應的空間,然后將該Dialog顯示在底部,或者全屏顯示。
二、活動主窗口調整
android定義了一個屬性,名字為windowSoftInputMode, 用它可以讓程序可以控制活動主窗口調整的方式。我們可以在AndroidManifet.xml中對Activity進行設置。如:android:windowSoftInputMode="stateUnchanged|adjustPan"
該屬性可選的值有兩部分,一部分為軟鍵盤的狀態控制,另一部分是活動主窗口的調整。前一部分本文不做討論,請讀者自行查閱android文檔。
模式一,壓縮模式
windowSoftInputMode的值如果設置為adjustResize,那么該Activity主窗口總是被調整大小以便留出軟鍵盤的空間。
我們通過一段代碼來測試一下,當我們設置了該屬性后,彈出輸入法時,系統做了什么。
重寫Layout布局:

    
  1. public class ResizeLayout extends LinearLayout{
  2. private static int count = 0 ;
  3. public ResizeLayout(Context context, AttributeSet attrs) {
  4. super (context, attrs);
  5. }
  6. @Override
  7. protected void onSizeChanged( int w, int h, int oldw, int oldh) {
  8. super .onSizeChanged(w, h, oldw, oldh);
  9. Log.e( "onSizeChanged " + count++, "=>onResize called! w=" +w + ",h=" +h+ ",oldw=" +oldw+ ",oldh=" +oldh);
  10. }
  11. @Override
  12. protected void onLayout( boolean changed, int l, int t, int r, int b) {
  13. super .onLayout(changed, l, t, r, b);
  14. Log.e( "onLayout " + count++, "=>OnLayout called! l=" + l + ", t=" + t + ",r=" + r + ",b=" +b);
  15. }
  16. @Override
  17. protected void onMeasure( int widthMeasureSpec, int heightMeasureSpec) {
  18. super .onMeasure(widthMeasureSpec, heightMeasureSpec);
  19. Log.e( "onMeasure " + count++, "=>onMeasure called! widthMeasureSpec=" + widthMeasureSpec + ", heightMeasureSpec=" + heightMeasureSpec);
  20. }

我們的布局設置為:

    
  1. < com.winuxxan.inputMethodTest.ResizeLayout
  2. xmlns:android = "http://schemas.android.com/apk/res/android"
  3. android:id = "@+id/root_layout"
  4. android:layout_width = "fill_parent"
  5. android:layout_height = "fill_parent"
  6. android:orientation = "vertical"
  7. >
  8. < EditText
  9. android:layout_width = "fill_parent"
  10. android:layout_height = "wrap_content"
  11. />
  12. < LinearLayout
  13. android:id = "@+id/bottom_layout"
  14. android:layout_width = "fill_parent"
  15. android:layout_height = "fill_parent"
  16. android:orientation = "vertical"
  17. android:gravity = "bottom" > s
  18. < TextView
  19. android:layout_width = "fill_parent"
  20. android:layout_height = "wrap_content"
  21. android:text = "@string/hello"
  22. android:background = "#77777777"
  23. />
  24. </ LinearLayout >
  25. </ com.winuxxan.inputMethodTest.ResizeLayout >

AndroidManifest.xml的Activity設置屬性:android:windowSoftInputMode = "adjustResize"
運行程序,點擊文本框,查看調試信息:
E/onMeasure 6(7960): =>onMeasure called! widthMeasureSpec=1073742144, heightMeasureSpec = 1073742024
E/onMeasure 7(7960): =>onMeasure called! widthMeasureSpec=1073742144, heightMeasureSpec = 1073742025
E/onSizeChanged 8(7960): =>onSizeChanged called! w=320,h=201,oldw=320,oldh=377
E/onLayout 9(7960): =>OnLayout called! l=0, t=0,r=320,b=201
從調試結果我們可以看出,當我們點擊文本框后,根布局調用了onMeasure,onSizeChanged和onLayout。
實際上,當設置為adjustResize后,軟鍵盤彈出時,要對主窗口布局重新進行measure和layout,而在layout時,發現窗口的大小發生的變化,因此調用了onSizeChanged。
從下圖的運行結果我們也可以看出,原本在下方的TextView被頂到了輸入法的上方。

Android軟鍵盤的隱藏顯示研究 Android軟鍵盤的隱藏顯示研究

模式二,平移模式
windowSoftInputMode的值如果設置為adjustPan,那么該Activity主窗口并不調整屏幕的大小以便留出軟鍵盤的空間。相反,當前窗口的內容將自動移動以便當前焦點從不被鍵盤覆蓋和用戶能總是看到輸入內容的部分。這個通常是不期望比調整大小,因為用戶可能關閉軟鍵盤以便獲得與被覆蓋內容的交互操作。
上面的例子中,我們將AndroidManifest.xml的屬性進行更改:android: windowSoftInputMode = "adjustPan"

重新運行,并點擊文本框,查看調試信息:
E/onMeasure 6(8378): =>onMeasure called! widthMeasureSpec=1073742144, heightMeasureSpec=1073742200
E/onMeasure 7(8378): =>onMeasure called! widthMeasureSpec=1073742144, heightMeasureSpec=1073742201
E/onLayout 8(8378): =>OnLayout called! l=0, t=0,r=320,b=377
我們看到:系統也重新進行了measrue和layout,但是我們發現,layout過程中onSizeChanged并沒有調用,這說明輸入法彈出前后并沒有改變原有布局的大小。
從下圖的運行結果我們可以看到,下方的TextView并沒有被頂到輸入法上方。

Android軟鍵盤的隱藏顯示研究 Android軟鍵盤的隱藏顯示研究

事實上,當輸入框不會被遮擋時,該模式沒有對布局進行調整,然而當輸入框將要被遮擋時,窗口就會進行平移。也就是說,該模式始終是保持輸入框為可見。如下圖,整個窗口,包括標題欄均被上移,以保證文本框可見。

Android軟鍵盤的隱藏顯示研究 Android軟鍵盤的隱藏顯示研究

模式三 自動模式
當屬性windowSoftInputMode被設置為adjustUspecified時,它不被指定是否該Activity主窗口調整大小以便留出軟鍵盤的空間,或是否窗口上的內容得到屏幕上當前的焦點是可見的。系統將自動選擇這些模式中一種主要依賴于是否窗口的內容有任何布局視圖能夠滾動他們的內容。如果有這樣的一個視圖,這個窗口將調整大小,這樣的假設可以使滾動窗口的內容在一個較小的區域中可見的。這個是主窗口默認的行為設置。
也就是說,系統自動決定是采用平移模式還是壓縮模式,決定因素在于內容是否可以滾動。

三、偵聽軟鍵盤的顯示隱藏
有時候,借助系統本身的機制來實現主窗口的調整并非我們想要的結果,我們可能希望在軟鍵盤顯示隱藏的時候,手動的對布局進行修改,以便使軟鍵盤彈出時更加美觀。這時就需要對軟鍵盤的顯示隱藏進行偵聽。
直接對軟鍵盤的顯示隱藏偵聽的方法本人沒有找到,如果哪位找到的方法請務必告訴本人一聲。還有本方法針對壓縮模式,平移模式不一定有效。
我們可以借助軟鍵盤顯示和隱藏時,對主窗口進行了重新布局這個特性來進行偵聽。如果我們設置的模式為壓縮模式,那么我們可以對布局的onSizeChanged函數進行跟蹤,如果為平移模式,那么該函數可能不會被調用。
我們可以重寫根布局,因為根布局的高度一般情況下是不發生變化的。
假設跟布局為線性布局,模式為壓縮模式,我們寫一個例子,當輸入法彈出時隱藏某個view,輸入法隱藏時顯示某個view。

    
  1. public class ResizeLayout extends LinearLayout{
  2. private OnResizeListener mListener;
  3. public interface OnResizeListener {
  4. void OnResize( int w, int h, int oldw, int oldh);
  5. }
  6. public void setOnResizeListener(OnResizeListener l) {
  7. mListener = l;
  8. }
  9. public ResizeLayout(Context context, AttributeSet attrs) {
  10. super (context, attrs);
  11. }
  12. @Override
  13. protected void onSizeChanged( int w, int h, int oldw, int oldh) {
  14. super .onSizeChanged(w, h, oldw, oldh);
  15. if (mListener != null ) {
  16. mListener.OnResize(w, h, oldw, oldh);
  17. }
  18. }
  19. }

在我們的Activity中,通過如下方法調用:

    
  1. public class InputMethodTestActivity extends Activity {
  2. private static final int BIGGER = 1 ;
  3. private static final int SMALLER = 2 ;
  4. private static final int MSG_RESIZE = 1 ;
  5. private static final int HEIGHT_THREADHOLD = 30 ;
  6. class InputHandler extends Handler {
  7. @Override
  8. public void handleMessage(Message msg) {
  9. switch (msg.what) {
  10. case MSG_RESIZE: {
  11. if (msg.arg1 == BIGGER) {
  12. findViewById(R.id.bottom_layout).setVisibility(View.VISIBLE);
  13. } else {
  14. findViewById(R.id.bottom_layout).setVisibility(View.GONE);
  15. }
  16. }
  17. break ;
  18. default :
  19. break ;
  20. }
  21. super .handleMessage(msg);
  22. }
  23. }
  24. private InputHandler mHandler = new InputHandler();
  25. /** Called when the activity is first created. */
  26. @Override
  27. public void onCreate(Bundle savedInstanceState) {
  28. super .onCreate(savedInstanceState);
  29. setContentView(R.layout.main);
  30. ResizeLayout layout = (ResizeLayout) findViewById(R.id.root_layout);
  31. layout.setOnResizeListener( new ResizeLayout.OnResizeListener() {
  32. public void OnResize( int w, int h, int oldw, int oldh) {
  33. int change = BIGGER;
  34. if (h < oldh) {
  35. change = SMALLER;
  36. }
  37. Message msg = new Message();
  38. msg.what = 1 ;
  39. msg.arg1 = change;
  40. mHandler.sendMessage(msg);
  41. }
  42. });
  43. }
  44. }

這里特別需要注意的是,不能直接在OnResizeListener中對要改變的View進行更改,因為OnSizeChanged函數實際上是運行在View的layout方法中,如果直接在onSizeChange中改變view的顯示屬性,那么很可能需要重新調用layout方法才能顯示正確。然而我們的方法又是在layout中調用的,因此會出現錯誤。因此我們在例子中采用了Handler的方法。

Android軟鍵盤的隱藏顯示研究 Android軟鍵盤的隱藏顯示研究

Android軟鍵盤的隱藏顯示研究


更多文章、技術交流、商務合作、聯系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號聯系: 360901061

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對您有幫助就好】

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長會非常 感謝您的哦!!!

發表我的評論
最新評論 總共0條評論
主站蜘蛛池模板: 久久久久综合 | 伊人久久五月天 | 亚洲无线码一区在线观看 | 久热996re这里有精品 | 深夜日韩| 波多野结衣视频一区 | 免费观看成人碰视频公开 | 亚洲一一在线 | 啪啪婷婷| 在线观看日韩 | 久久91精品国产91久久 | 26uuu久久 | 亚洲永久精品一区二区三区 | 天天干天天操天天射 | 亚洲女人天堂a在线播放 | 高清国产精品久久久久 | 一级欧美日韩 | 神马我我不卡伦影视 | 国产成人香蕉在线视频网站 | 26uuu另类欧美亚洲日本 | 非常色的视频 | 四虎影视亚洲精品 | 亚洲日韩精品欧美一区二区 | 在线观看麻豆精品国产不卡 | 欧美综合图片区 | 九九国产精品九九 | 亚洲精品 国产 日韩 | 中文欧美日韩 | 国产精品欧美亚洲韩国日本不卡 | 狠狠综合久久久久尤物丿 | 日本精品欧洲www | 奇米在线视频777 | 国产91久久精品一区二区 | 婷婷 综合网站 | 日韩一区二区视频在线观看 | 伊人色综合网 | 99热国产在线观看 | 久久国产在线观看 | 国产成年人网站 | 立即播放免费毛片一级 | 爱爱小视频在线观看网站 |