android输入法隐藏状态栏,android 输入法设置显示隐藏
Android是一個針對觸摸屏專門設(shè)計的操作系統(tǒng),當(dāng)點擊編輯框,系統(tǒng)自動為用戶彈出軟鍵盤,以便用戶進行輸入。
那么,彈出軟鍵盤后必然會造成原有布局高度的減少,那么系統(tǒng)應(yīng)該如何來處理布局的減少?我們能否在應(yīng)用程序中進行自定義的控制?這些是本文要討論的重點。
一、軟鍵盤顯示的原理
軟件盤的本質(zhì)是什么?軟鍵盤其實是一個Dialog!
InputMethodService為我們的輸入法創(chuàng)建了一個Dialog,并且將該Dialog的Window的某些參數(shù)(如Gravity)進行了設(shè)置,使之能夠在底部或者全屏顯示。當(dāng)我們點擊輸入框時,系統(tǒng)對活動主窗口進行調(diào)整,從而為輸入法騰出相應(yīng)的空間,然后將該Dialog顯示在底部,或者全屏顯示。
二、活動主窗口調(diào)整
android定義了一個屬性,名字為windowSoftInputMode, 用它可以讓程序可以控制活動主窗口調(diào)整的方式。我們可以在AndroidManifet.xml中對Activity進行設(shè)置。如:android:windowSoftInputMode="stateUnchanged|adjustPan"
該屬性可選的值有兩部分,一部分為軟鍵盤的狀態(tài)控制,另一部分是活動主窗口的調(diào)整。前一部分本文不做討論,請讀者自行查閱android文檔。
模式一,壓縮模式
windowSoftInputMode的值如果設(shè)置為adjustResize,那么該Activity主窗口總是被調(diào)整大小以便留出軟鍵盤的空間。
我們通過一段代碼來測試一下,當(dāng)我們設(shè)置了該屬性后,彈出輸入法時,系統(tǒng)做了什么。
重寫Layout布局:
1.publicclassResizeLayoutextendsLinearLayout{
2.privatestaticintcount?=0;
3.
4.publicResizeLayout(Context?context,?AttributeSet?attrs)?{
5.super(context,?attrs);
6.?????}
7.
8.@Override
9.protectedvoidonSizeChanged(intw,inth,intoldw,intoldh)?{
10.super.onSizeChanged(w,?h,?oldw,?oldh);
11.
12.?????????Log.e("onSizeChanged?"+?count++,"=>onResize?called!?w="+w?+",h="+h+",oldw="+oldw+",oldh="+oldh);
13.?????}
14.
15.@Override
16.protectedvoidonLayout(booleanchanged,intl,intt,intr,intb)?{
17.super.onLayout(changed,?l,?t,?r,?b);
18.?????????Log.e("onLayout?"+?count++,"=>OnLayout?called!?l="+?l?+",?t="+?t?+",r="+?r?+",b="+b);
19.?????}
20.
21.@Override
22.protectedvoidonMeasure(intwidthMeasureSpec,intheightMeasureSpec)?{
23.super.onMeasure(widthMeasureSpec,?heightMeasureSpec);
24.
25.?????????Log.e("onMeasure?"+?count++,"=>onMeasure?called!?widthMeasureSpec="+?widthMeasureSpec?+",?heightMeasureSpec="+?heightMeasureSpec);
26.?????}1. public class ResizeLayout extends LinearLayout{
2. private static int count = 0;
3.
4. public ResizeLayout(Context context, AttributeSet attrs) {
5. super(context, attrs);
6. }
7.
8. @Override
9. protected void onSizeChanged(int w, int h, int oldw, int oldh) {
10. super.onSizeChanged(w, h, oldw, oldh);
11.
12. Log.e("onSizeChanged " + count++, "=>onResize called! w="+w + ",h="+h+",oldw="+oldw+",oldh="+oldh);
13. }
14.
15. @Override
16. protected void onLayout(boolean changed, int l, int t, int r, int b) {
17. super.onLayout(changed, l, t, r, b);
18. Log.e("onLayout " + count++, "=>OnLayout called! l=" + l + ", t=" + t + ",r=" + r + ",b="+b);
19. }
20.
21. @Override
22. protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
23. super.onMeasure(widthMeasureSpec, heightMeasureSpec);
24.
25. Log.e("onMeasure " + count++, "=>onMeasure called! widthMeasureSpec=" + widthMeasureSpec + ", heightMeasureSpec=" + heightMeasureSpec);
26. }
我們的布局設(shè)置為:
1.
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.
9.
10.android:layout_width="fill_parent"
11.android:layout_height="wrap_content"
12./>
13.
14.
15.android:id="@+id/bottom_layout"
16.android:layout_width="fill_parent"
17.android:layout_height="fill_parent"
18.android:orientation="vertical"
19.android:gravity="bottom">s
20.
21.
22.android:layout_width="fill_parent"
23.android:layout_height="wrap_content"
24.android:text="@string/hello"
25.android:background="#77777777"
26./>
27.
28.
1.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.
9.
10. android:layout_width="fill_parent"
11. android:layout_height="wrap_content"
12. />
13.
14.
15. android:id="@+id/bottom_layout"
16. android:layout_width="fill_parent"
17. android:layout_height="fill_parent"
18. android:orientation="vertical"
19. android:gravity="bottom">s
20.
21.
22. android:layout_width="fill_parent"
23. android:layout_height="wrap_content"
24. android:text="@string/hello"
25. android:background="#77777777"
26. />
27.
28.
AndroidManifest.xml的Activity設(shè)置屬性:android:windowSoftInputMode = "adjustResize"
運行程序,點擊文本框,查看調(diào)試信息:
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
從調(diào)試結(jié)果我們可以看出,當(dāng)我們點擊文本框后,根布局調(diào)用了onMeasure,onSizeChanged和onLayout。
實際上,當(dāng)設(shè)置為adjustResize后,軟鍵盤彈出時,要對主窗口布局重新進行measure和layout,而在layout時,發(fā)現(xiàn)窗口的大小發(fā)生的變化,因此調(diào)用了onSizeChanged。
從下圖的運行結(jié)果我們也可以看出,原本在下方的TextView被頂?shù)搅溯斎敕ǖ纳戏健?/p>
模式二,平移模式
windowSoftInputMode的值如果設(shè)置為adjustPan,那么該Activity主窗口并不調(diào)整屏幕的大小以便留出軟鍵盤的空間。相反,當(dāng)前窗口的內(nèi)容將自動移動以便當(dāng)前焦點從不被鍵盤覆蓋和用戶能總是看到輸入內(nèi)容的部分。這個通常是不期望比調(diào)整大小,因為用戶可能關(guān)閉軟鍵盤以便獲得與被覆蓋內(nèi)容的交互操作。
上面的例子中,我們將AndroidManifest.xml的屬性進行更改:android: windowSoftInputMode = "adjustPan"
重新運行,并點擊文本框,查看調(diào)試信息:
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
我們看到:系統(tǒng)也重新進行了measrue和layout,但是我們發(fā)現(xiàn),layout過程中onSizeChanged并沒有調(diào)用,這說明輸入法彈出前后并沒有改變原有布局的大小。
從下圖的運行結(jié)果我們可以看到,下方的TextView并沒有被頂?shù)捷斎敕ㄉ戏健?/p>
事實上,當(dāng)輸入框不會被遮擋時,該模式?jīng)]有對布局進行調(diào)整,然而當(dāng)輸入框?qū)⒁徽趽鯐r,窗口就會進行平移。也就是說,該模式始終是保持輸入框為可見。如下圖,整個窗口,包括標(biāo)題欄均被上移,以保證文本框可見。
模式三 自動模式
當(dāng)屬性windowSoftInputMode被設(shè)置為adjustUspecified時,它不被指定是否該Activity主窗口調(diào)整大小以便留出軟鍵盤的空間,或是否窗口上的內(nèi)容得到屏幕上當(dāng)前的焦點是可見的。系統(tǒng)將自動選擇這些模式中一種主要依賴于是否窗口的內(nèi)容有任何布局視圖能夠滾動他們的內(nèi)容。如果有這樣的一個視圖,這個窗口將調(diào)整大小,這樣的假設(shè)可以使?jié)L動窗口的內(nèi)容在一個較小的區(qū)域中可見的。這個是主窗口默認(rèn)的行為設(shè)置。
也就是說,系統(tǒng)自動決定是采用平移模式還是壓縮模式,決定因素在于內(nèi)容是否可以滾動。
三、偵聽軟鍵盤的顯示隱藏
有時候,借助系統(tǒng)本身的機制來實現(xiàn)主窗口的調(diào)整并非我們想要的結(jié)果,我們可能希望在軟鍵盤顯示隱藏的時候,手動的對布局進行修改,以便使軟鍵盤彈出時更加美觀。這時就需要對軟鍵盤的顯示隱藏進行偵聽。
直接對軟鍵盤的顯示隱藏偵聽的方法本人沒有找到,如果哪位找到的方法請務(wù)必告訴本人一聲。還有本方法針對壓縮模式,平移模式不一定有效。
我們可以借助軟鍵盤顯示和隱藏時,對主窗口進行了重新布局這個特性來進行偵聽。如果我們設(shè)置的模式為壓縮模式,那么我們可以對布局的onSizeChanged函數(shù)進行跟蹤,如果為平移模式,那么該函數(shù)可能不會被調(diào)用。
我們可以重寫根布局,因為根布局的高度一般情況下是不發(fā)生變化的。
假設(shè)跟布局為線性布局,模式為壓縮模式,我們寫一個例子,當(dāng)輸入法彈出時隱藏某個view,輸入法隱藏時顯示某個view。
1.publicclassResizeLayoutextendsLinearLayout{
2.privateOnResizeListener?mListener;
3.
4.publicinterfaceOnResizeListener?{
5.voidOnResize(intw,inth,intoldw,intoldh);
6.?????}
7.
8.publicvoidsetOnResizeListener(OnResizeListener?l)?{
9.?????????mListener?=?l;
10.?????}
11.
12.publicResizeLayout(Context?context,?AttributeSet?attrs)?{
13.super(context,?attrs);
14.?????}
15.
16.@Override
17.protectedvoidonSizeChanged(intw,inth,intoldw,intoldh)?{
18.super.onSizeChanged(w,?h,?oldw,?oldh);
19.
20.if(mListener?!=null)?{
21.?????????????mListener.OnResize(w,?h,?oldw,?oldh);
22.?????????}
23.?????}
24.?}
1. public class ResizeLayout extends LinearLayout{
2. private OnResizeListener mListener;
3.
4. public interface OnResizeListener {
5. void OnResize(int w, int h, int oldw, int oldh);
6. }
7.
8. public void setOnResizeListener(OnResizeListener l) {
9. mListener = l;
10. }
11.
12. public ResizeLayout(Context context, AttributeSet attrs) {
13. super(context, attrs);
14. }
15.
16. @Override
17. protected void onSizeChanged(int w, int h, int oldw, int oldh) {
18. super.onSizeChanged(w, h, oldw, oldh);
19.
20. if (mListener != null) {
21. mListener.OnResize(w, h, oldw, oldh);
22. }
23. }
24. }
在我們的Activity中,通過如下方法調(diào)用:
1.publicclassInputMethodTestActivityextendsActivity?{
2.privatestaticfinalintBIGGER?=1;
3.privatestaticfinalintSMALLER?=2;
4.privatestaticfinalintMSG_RESIZE?=1;
5.
6.privatestaticfinalintHEIGHT_THREADHOLD?=30;
7.
8.classInputHandlerextendsHandler?{
9.@Override
10.publicvoidhandleMessage(Message?msg)?{
11.switch(msg.what)?{
12.caseMSG_RESIZE:?{
13.if(msg.arg1?==?BIGGER)?{
14.?????????????????????findViewById(R.id.bottom_layout).setVisibility(View.VISIBLE);
15.?????????????????}else{
16.?????????????????????findViewById(R.id.bottom_layout).setVisibility(View.GONE);
17.?????????????????}
18.?????????????}
19.break;
20.
21.default:
22.break;
23.?????????????}
24.super.handleMessage(msg);
25.?????????}
26.?????}
27.
28.privateInputHandler?mHandler?=newInputHandler();
29.
30./**?Called?when?the?activity?is?first?created.?*/
31.@Override
32.publicvoidonCreate(Bundle?savedInstanceState)?{
33.super.onCreate(savedInstanceState);
34.?????????setContentView(R.layout.main);
35.
36.?????????ResizeLayout?layout?=?(ResizeLayout)?findViewById(R.id.root_layout);
37.?????????layout.setOnResizeListener(newResizeLayout.OnResizeListener()?{
38.
39.publicvoidOnResize(intw,inth,intoldw,intoldh)?{
40.intchange?=?BIGGER;
41.if(h?
42.?????????????????????change?=?SMALLER;
43.?????????????????}
44.
45.?????????????????Message?msg?=newMessage();
46.?????????????????msg.what?=1;
47.?????????????????msg.arg1?=?change;
48.?????????????????mHandler.sendMessage(msg);
49.?????????????}
50.?????????});
51.?????}
52.?}
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.
6. private static final int HEIGHT_THREADHOLD = 30;
7.
8. class InputHandler extends Handler {
9. @Override
10. public void handleMessage(Message msg) {
11. switch (msg.what) {
12. case MSG_RESIZE: {
13. if (msg.arg1 == BIGGER) {
14. findViewById(R.id.bottom_layout).setVisibility(View.VISIBLE);
15. } else {
16. findViewById(R.id.bottom_layout).setVisibility(View.GONE);
17. }
18. }
19. break;
20.
21. default:
22. break;
23. }
24. super.handleMessage(msg);
25. }
26. }
27.
28. private InputHandler mHandler = new InputHandler();
29.
30. /** Called when the activity is first created. */
31. @Override
32. public void onCreate(Bundle savedInstanceState) {
33. super.onCreate(savedInstanceState);
34. setContentView(R.layout.main);
35.
36. ResizeLayout layout = (ResizeLayout) findViewById(R.id.root_layout);
37. layout.setOnResizeListener(new ResizeLayout.OnResizeListener() {
38.
39. public void OnResize(int w, int h, int oldw, int oldh) {
40. int change = BIGGER;
41. if (h < oldh) {
42. change = SMALLER;
43. }
44.
45. Message msg = new Message();
46. msg.what = 1;
47. msg.arg1 = change;
48. mHandler.sendMessage(msg);
49. }
50. });
51. }
52. }
這里特別需要注意的是,不能直接在OnResizeListener中對要改變的View進行更改,因為OnSizeChanged函數(shù)實際上是運行在View的layout方法中,如果直接在onSizeChange中改變view的顯示屬性,那么很可能需要重新調(diào)用layout方法才能顯示正確。然而我們的方法又是在layout中調(diào)用的,因此會出現(xiàn)錯誤。因此我們在例子中采用了Handler的方法。
總結(jié)
以上是生活随笔為你收集整理的android输入法隐藏状态栏,android 输入法设置显示隐藏的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: BLE HID简介
- 下一篇: Pycharm 报错Out of Mem