1. 程式人生 > >android EditText完美解決禁止輸入法表情

android EditText完美解決禁止輸入法表情

         最近有個需求,需要評論的時候禁止輸入法的表情輸入。然後在網上各種找,發現有些人啊,也是服了,同樣的東西反覆寫,並且還沒解決問題,我就想問問,這個你們自己真的動手執行的了,效果實現了嗎,還是說,效果實現了,不想分享出來呢。

        作為一個有分享精神的人,我把自己做的分享一下,希望能夠幫助有需要的人,同時自己也做做筆記。

        在網上搜索的發現有三種方案,我全部拿過來。

       第一種方案:檢測輸入的內容中是否含有emoji    

       第一步:先在xml佈局裡新增一個自定義的edittext

 <com.beijingmeifu.studentclient.activity.examination.ContainsEmojiEditText
           

Android:id="@+id/et_answer"
            android:layout_width="fill_parent"
            android:layout_height="fill_parent" />

      第二步:實現這個自定義的edittext

public class ContainsEmojiEditText extends EditText {
    //輸入表情前的游標位置
    private int cursorPos;
    //輸入表情前EditText中的文字
    private String inputAfterText;
    //是否重置了EditText的內容
    private boolean resetText;

    private Context mContext;

    public ContainsEmojiEditText(Context context) {
        super(context);
        this.mContext = context;
        initEditText();
    }


    public ContainsEmojiEditText(Context context, AttributeSet attrs) {
        super(context, attrs);
        this.mContext = context;
        initEditText();
    }


    public ContainsEmojiEditText(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        this.mContext = context;
        initEditText();
    }


    // 初始化edittext 控制元件
    private void initEditText() {
        addTextChangedListener(new TextWatcher() {
            @Override
            public void beforeTextChanged(CharSequence s, int start, int before, int count) {
                if (!resetText) {
                    cursorPos = getSelectionEnd();
                    // 這裡用s.toString()而不直接用s是因為如果用s,
                    // 那麼,inputAfterText和s在記憶體中指向的是同一個地址,s改變了,
                    // inputAfterText也就改變了,那麼表情過濾就失敗了
                    inputAfterText= s.toString();
                }


            }


            @Override
            public void onTextChanged(CharSequence s, int start, int before, int count) {
                if (!resetText) {
                    if (count >= 2) {//表情符號的字元長度最小為2
                        CharSequence input = s.subSequence(cursorPos, cursorPos + count);
                        if (containsEmoji(input.toString())) {
                            resetText = true;
                            Toast.makeText(mContext, "不支援輸入Emoji表情符號", Toast.LENGTH_SHORT).show();
                            //是表情符號就將文字還原為輸入表情符號之前的內容
                            setText(inputAfterText);
                            CharSequence text = getText();
                            if (text instanceof Spannable) {
                                Spannable spanText = (Spannable) text;
                                Selection.setSelection(spanText, text.length());
                            }
                        }
                    }
                } else {
                    resetText = false;
                }
            }


            @Override
            public void afterTextChanged(Editable editable) {


            }
        });
    }




    /**
     * 檢測是否有emoji表情
     *
     * @param source
     * @return
     */
    public static boolean containsEmoji(String source) {
        int len = source.length();
        for (int i = 0; i < len; i++) {
            char codePoint = source.charAt(i);
            if (!isEmojiCharacter(codePoint)) { //如果不能匹配,則該字元是Emoji表情
                return true;
            }
        }
        return false;
    }
    /**
     * 判斷是否是Emoji
     *
     * @param codePoint 比較的單個字元
     * @return
     */
    private static boolean isEmojiCharacter(char codePoint) {
        return (codePoint == 0x0) || (codePoint == 0x9) || (codePoint == 0xA) ||
                (codePoint == 0xD) || ((codePoint >= 0x20) && (codePoint <= 0xD7FF)) ||
                ((codePoint >= 0xE000) && (codePoint <= 0xFFFD)) || ((codePoint >= 0x10000)
                && (codePoint <= 0x10FFFF));
    }

}

這是第一種方案,但是第一種方案,有個缺陷,就是有些表情禁止不了,l例如這個表情:



就是這個表情,折磨死我了,我在Emoji官網上查了好久,也用各種轉換工具實驗就是不行,如果那位大神通過這個方法屛避掉這個表情,到時候希望給我留言

      第二種方案:利用正則表示式遮蔽表情

     下面貼出我在網上找的一個正則表達:
     private static final String exPressionReg = "^([a-z]|[A-Z]|[0-9]|[\u2E80-\u9FFF]){3,}|
    @(?:\w?\.)+\w?
  |[.net]{4}|[.org]{4}|[http://]{7}|[ftp://]{6}$";

    接下來就是編寫TextWatcher過濾監聽器了。
    基本思路:

   1,在輸入表情前,先儲存EditText已經輸入的文字;
    2,輸入表情後,利用正則表示式對剛輸入的東東進行匹配,
    如果是表情,則保留擷取先前輸入的文字,View重畫。

    有了思路,就看下面程式碼的實現:

   mEditRoom.addTextChangedListener(new TextWatcher() {
   @Override
    public void beforeTextChanged(CharSequence s, int start, int count, int after) {
   if(!resetText){
   //輸入前,獲取游標的位置(文字結尾處)
   cursorPos = mEditRoom.getSelectionEnd();
   temp = s.toString();
       }
    }
   @Override
   public void onTextChanged(CharSequence s, int start, int before, int count) {
   if(!resetText){
    if(count >= 2){
    CharSequence content = s.subSequence(cursorPos,cursorPos+count);
   boolean isExpression = PatternUtils.checkExpression(content.toString());
   if(isExpression){
   resetText = true;
   mEditRoom.setText(temp);
   mEditRoom.setSelection(temp.length());
   mEditRoom.invalidate();
   CommonToast.showToast(mContext,getString(R.string.device_location_cannot_input_expression));
     }
   }
   }else {
   resetText = false;
  }
  }
   @Override
   public void afterTextChanged(Editable s) {
     }
  })}

 但是這種方法同方案一一樣,還是有個別表情遮蔽不了,我想的是應該是可以的,對照Emoji官網,然後轉換工具,可以知道轉換字元,然後在匹配,但是太麻煩了有興趣的大神可以試著嘗試一下。

    第三種方案:給EditText設定過濾器

   我也是通過在各種方案解決問題的,上程式碼:

public class MainActivity extends Activity {
    public EditText contentEd;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        contentEd =(EditText) findViewById(R.id.contentEdit);
        InputFilter[] emojiFilters = {emojiFilter};
        contentEd.setFilters(new InputFilter[]{emojiFilter});
    }
    InputFilter emojiFilter = new InputFilter() {
        Pattern emoji = Pattern.compile("[\ud83c\udc00-\ud83c\udfff]|[\ud83d\udc00-\ud83d\udfff]|[\u2600-\u27ff]",
                Pattern.UNICODE_CASE | Pattern.CASE_INSENSITIVE);

        @Override
        public CharSequence filter(CharSequence source, int start, int end, Spanned dest, int dstart, int dend) {
            Matcher emojiMatcher = emoji.matcher(source);
            if (emojiMatcher.find()) {
                Toast.makeText(MainActivity.this,"不支援輸入表情", 0).show();
                return "";
            }
            return null;
        }
    };
    
    
}

這種方法沒什麼太多說的,就是過濾器,過濾掉特殊字元,就行了。

三種方案我都會上原始碼

原始碼:

如果不能直接跳轉的話,請複製網址下載。