Android自由選擇TextView的文字

用過EditText的都知道,EditText有個特點,當在裏面長按的時候,會出現一個ContextMenu,提供了選擇文字,複製,剪切等功能。有時候,我們會想,如果不出現這個ContextMenu,直接就在view上選擇文字,那多美好啊。相信很多人抱有這樣的想法,很不幸,我也是。於是我就研究了一下EditText和TextView的代碼,然後將這個問題解決了。
      網上很多資料都說,要選擇一段文字,只需要用Selection.getSelectionStart()和Selection.getSelectionEnd()確定選擇的文字的頭和尾,然後加顏色就行。簡直是胡扯啊,我敢說這樣的代碼根本就沒有經過驗證,就發到網上了,然後一大堆人互相轉載,結果導致誤導了很多人,杯具 啊!! 
      好,我們來分析一下解決辦法。
      TextView是很多View的基類,如Button、EditText都是繼承自他,所以EditText裏面的代碼很少。我們看一下EditText的源碼,有一個Override的getDefaultEditable方法,看名字的意思是是否可編輯,這個方法直接返回true。還有一個getDefaultMovementMethod方法,它返回的是ArrowKeyMovementMethod.getInstance(),通過查看ArrowKeyMovementMethod的源碼,基本確定這個方法就是彈出ContextMenu和軌跡球監聽的“元兇”。
      下面,我們自己做一個view來打造自己的EditText。

      我取名TextPage,繼承EditText,在裏面覆蓋getDefaultEditable和getDefaultMovementMethod。

@Override  
public boolean getDefaultEditable() {  
    return false;  
}  
@Override  
protected MovementMethod getDefaultMovementMethod() {  
    return null;  
}  

現在測試一下,發現長按沒反應了,所料不錯,就是getDefaultMovementMethod方法控制了ContextMenu。
      看一下ArrowKeyMovementMethod的代碼,裏面提供了KeyEvent、軌跡球事件onTrackballEvent和touch事件onTouchEvent的處理。這些事件在何處調用的呢?我們看看TextView的onTouchEvent、onTrackballEvent和onKeyEvent方法裏面就明白了,在這些事件回調中調用了ArrowKeyMovementMethod裏面的這些方法。
      還有個問題,ContextMenu在哪裏觸發的?這個問題,用過ContextMenu的都知道,view裏面要使用ContextMenu,需要覆蓋一個onCreateContextMenu方法,然後在裏面創建ContextMenu的各個選項。在TextView裏面找onCreateContextMenu,果然有,裏面定義了選擇、複製、粘貼等選項。
      既然找到了這個,那麼我們就可以進一步分析選擇是如何做到的。
      onCreateContextMenu只是創建菜單,那麼菜單點擊之後,觸發了什麼呢?onCreateContextMenu裏面定義了一個MenuHandler對象,然後作爲參數傳遞給setOnMenuItemClickListener,找到MenuHandler,發現裏面的onMenuItemClick返回的是onTextContextMenuItem函數,找到onTextContextMenuItem,OMG,終於找到點擊menu觸發的函數了。但是裏面貌似沒有關鍵的東西,選擇的部分不在這裏。那麼,就應該在上面所說的那些事件裏面了。
      重點分析ArrowKeyMovementMethod的onTouchEvent方法。發現一個重要的方法getLayout(),然後獲取一個Layout對象,通過x和y座標知道當前字符串的offset位置。
      那麼,問題就可以完美的解決了。你可以點擊任何地方然後拖動,釋放之後,中間的文字就會被選中,so beautiful!

import android.content.Context;  
import android.graphics.Color;  
import android.text.Layout;  
import android.text.Selection;  
import android.view.ContextMenu;  
import android.view.Gravity;  
import android.view.MotionEvent;  
import android.widget.EditText;  
  
/** 
 * @author chroya 
 */  
public class TextPage extends EditText {  
    private int off; //字符串的偏移值  
  
    public TextPage(Context context) {  
        super(context);  
        initialize();  
    }  
  
    private void initialize() {  
        setGravity(Gravity.TOP);  
        setBackgroundColor(Color.WHITE);  
    }  
      
    @Override  
    protected void onCreateContextMenu(ContextMenu menu) {  
        //不做任何處理,爲了阻止長按的時候彈出上下文菜單  
    }  
      
    @Override  
    public boolean getDefaultEditable() {  
        return false;  
    }  
      
    @Override  
    public boolean onTouchEvent(MotionEvent event) {  
        int action = event.getAction();  
        Layout layout = getLayout();  
        int line = 0;  
        switch(action) {  
        case MotionEvent.ACTION_DOWN:  
            line = layout.getLineForVertical(getScrollY()+ (int)event.getY());          
            off = layout.getOffsetForHorizontal(line, (int)event.getX());  
            Selection.setSelection(getEditableText(), off);  
            break;  
        case MotionEvent.ACTION_MOVE:  
        case MotionEvent.ACTION_UP:  
            line = layout.getLineForVertical(getScrollY()+(int)event.getY());   
            int curOff = layout.getOffsetForHorizontal(line, (int)event.getX());              
            Selection.setSelection(getEditableText(), off, curOff);  
            break;  
        }  
        return true;  
    }  
}  

原文地址:http://chroya.iteye.com/blog/753634

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章