標籤:
Android上有很多IME應用,每種IME都有各自的特點,IME多數時候是和EditText配合使用,結合我自己的親身實踐分享一下使用EditText過程中遇到的一些問題及解決方案。
設定預設IME
有時候為了提高使用者體驗,在彈出IME時需要設定預設的輸入狀態,比如單詞應用彈出IME時,IME最好是在英文輸入狀態下。如果是字典應用,彈出IME時最好是在中文輸入狀態下,Android並沒有提供設定預設的輸入狀態的介面,但我們可以通過如下方法一樣能夠達到想要的效果:
預設中文:
mEditText.setInputType(EditorInfo.TYPE_CLASS_TEXT);
預設英文:
mEditText.setInputType(EditorInfo.TYPE_TEXT_VARIATION_URI);
開啟和關閉IME
手動控制IME的開關狀態也能提升使用者體驗,比如:
有的搜尋方塊會有一個清除按鈕,點擊清除按鈕時就應該彈出IME,因為使用者清除搜尋內容的目的多數時候是需要輸入新的內容;
執行搜尋時應該隱藏IME,因為顯示IME時會遮擋搜尋結果,使用者體驗不太好;
鬧鐘來時或者有其它window彈出時應該隱藏IME,因為IME也是window,如果不隱藏可能導致IME遮擋住了其它window等使用者體驗不太友好的問題。
開啟IME:
private void open(Context context, View editText){ InputMethodManager inputMethodManager = (InputMethodManager) context.getSystemService(Context.INPUT_METHOD_SERVICE); inputMethodManager.showSoftInput(editText, 0);}
關閉IME:
private void close(Context context, View editText){ InputMethodManager inputMethodManager = (InputMethodManager) context.getSystemService(Context.INPUT_METHOD_SERVICE); inputMethodManager.hideSoftInputFromWindow(editText.getWindowToken(), 0);}
監聽EditText的輸入狀態
上面這些都可以通過監聽EditText的輸入狀態來實現,具體實現方式如下:
mInputEditTxt.addTextChangedListener(new TextWatcher() { @Override public void beforeTextChanged(CharSequence s, int start, int count, int after) { } @Override public void onTextChanged(CharSequence s, int start, int before, int count) { System.out.println("監聽EditText輸入內容的變化,在這裡可以監聽輸入內容的長度。"); } @Override public void afterTextChanged(Editable s) { System.out.println("這裡可以實現所輸即所得,使用者輸入的同時可以立即在這雷根據輸入內容執行操作,顯示搜尋結果!"); } });
監聽IME中的斷行符號按鈕
比如搜狗IME的右下角有一個斷行符號按鈕,我們希望使用者點擊它時也執行確認功能,可以通過監聽EditText的按鍵點擊事件來實現:
/** * 監聽IME按鍵 * * */ mInputEditTxt.setOnKeyListener(new OnKeyListener() { @Override public boolean onKey(View v, int keyCode, KeyEvent event) { if (keyCode == KeyEvent.KEYCODE_ENTER && event.getAction() == KeyEvent.ACTION_UP) { System.out.println("手指彈起時執行確認功能"); return true; } return false; } });
改變IME中斷行符號按鈕的顯示內容
如果斷行符號按鈕是執行搜尋功能,則斷行符號按鈕上顯示"搜尋",如果是執行發送功能,則顯示"發送",如果是下一步,則顯示"下一步"。
實現這個功能需要調用EditText的setImeOptions方法:
/**** IME_ACTION_SEARCH 搜尋* IME_ACTION_SEND 發送* IME_ACTION_NEXT 下一步* IME_ACTION_DONE 完成*/mInputEditTxt.setImeOptions(EditorInfo.IME_ACTION_SEARCH);
限制輸入內容
有時候我們根本就不想使用者輸入一些雜七雜八的內容,因為這需要程式針對輸入的內容做各種處理,如果處理不當還會有好多不可預見的問題,索性在輸入內容時就禁止使用者輸入一些非法字元,這可以通過下面的方式實現,建立一個類InputTxtFilter:
public class InputTxtFilter{ public static final int INPUT_TYPE_EN = 0x01; public static final int INPUT_TYPE_CH = 0x02; private static final String[] SPELL = new String[]{ "a","b","c","d","e","f","g","h","i","j","k","l","m","n","o","p","q","r","s","t","u","v","w","x","y","z", "ā","á","ǎ","à","ō","ó","ǒ","ò","ē","é","ě","è","ī","í","ǐ","ì","ū","ú","ǔ","ù","ǖ","ǘ","ǚ","ǜ","ü" }; private static char[] chineseParam = new char[]{‘」‘,‘,‘,‘。‘,‘?‘,‘…‘,‘:‘,‘~‘,‘【‘,‘#‘,‘、‘,‘%‘,‘*‘,‘&‘,‘$‘,‘(‘,‘‘‘,‘’‘,‘“‘,‘”‘,‘『‘,‘〔‘,‘{‘,‘【‘ ,‘¥‘,‘£‘,‘‖‘,‘〖‘,‘《‘,‘「‘,‘》‘,‘〗‘,‘】‘,‘}‘,‘〕‘,‘』‘,‘”‘,‘)‘,‘!‘,‘;‘,‘—‘}; private InputTxtFilter( ){ } public static void inputFilter( final Context context, final EditText editText, final int type, final int inputLimit){ InputFilter[] filters = new InputFilter[1]; filters[0] = new InputFilter.LengthFilter(inputLimit){ public CharSequence filter(CharSequence source, int start, int end, Spanned dest, int dstart, int dend){ boolean isRightCharater = false; if(type == INPUT_TYPE_EN){ isRightCharater = isLetter(source.toString()); }else if(type == INPUT_TYPE_CH){ isRightCharater = isChineseWord(source.toString()); } if ( !isRightCharater|| dest.toString( ).length( )>=inputLimit ){ return ""; } return source; } }; editText.setFilters(filters); } /** * 檢測String是否全是中文 * */ public static boolean isChineseWord( String name ){ boolean res=true; char[] cTemp = name.toCharArray( ); for( int i = 0; i < name.length( ); i++ ){ if( !isChinese( cTemp[ i ] ) ){ res=false; break; } } return res; } /** * 是否為英文字母 * * */ public static boolean isLetter( String inputStr ){ char[] inputArray = inputStr.toCharArray( ); List<String> spellList = Arrays.asList( SPELL ); for( char input : inputArray ){ if( !spellList.contains( input + "" ) ){ return false; } } return true; } /** * 判定輸入漢字 * @param c */ public static boolean isChinese( char c ){ for( char param : chineseParam ){ if( param == c ){ return false; } } Character.UnicodeBlock ub = Character.UnicodeBlock.of( c ); if ( ub == Character.UnicodeBlock.CJK_UNIFIED_IDEOGRAPHS || ub == Character.UnicodeBlock.CJK_COMPATIBILITY_IDEOGRAPHS || ub == Character.UnicodeBlock.CJK_UNIFIED_IDEOGRAPHS_EXTENSION_A || ub == Character.UnicodeBlock.GENERAL_PUNCTUATION || ub == Character.UnicodeBlock.CJK_SYMBOLS_AND_PUNCTUATION || ub == Character.UnicodeBlock.HALFWIDTH_AND_FULLWIDTH_FORMS ){ return true; } return false; }}
在初始化EditText時,調用InputTxtFilter的inputFilter方法,傳入輸入長度限制、輸入內容的類型限制等即可,eg:
InputTxtFilter.inputFilter(this, mInputEditTxt, InputTxtFilter.INPUT_TYPE_EN, 5);
屏蔽EditText的複製、粘貼功能
在低版本的Android SDK中,如果對EditText的輸入長度有限制時,長按EditText並將選中的內容拖動到EditText輸入框中,如果這時候的長度超過了EditText的輸入長度限制,程式會直接崩潰掉,在高版本的Android SDK中這個問題已經改了,如果出現上面的情況會直接清空輸入框中的內容,為了避免這種討厭的問題,我們可以屏蔽EditText的複製和粘貼功能,只需要屏蔽EditText的長按響應即可:
/** * 屏蔽複製、粘貼功能 * * */ mInputEditTxt.setCustomSelectionActionModeCallback(new ActionMode.Callback() { public boolean onCreateActionMode(ActionMode actionMode, Menu menu) { return false; } public boolean onPrepareActionMode(ActionMode actionMode, Menu menu) { return false; } public boolean onActionItemClicked(ActionMode actionMode, MenuItem menuItem) { return false; } @Override public void onDestroyActionMode(ActionMode mode) { } }); mInputEditTxt.setLongClickable(false);
Android EditText的使用及值得注意的地方