Android自由選擇TextView的文字 博客分類: Android AndroidUP

      用過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;
	}
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章