用過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 ; } }
原文作者:homebei2
原文連結:http://blog.csdn.net/homebei2/archive/2011/01/12/6130417.aspx