用過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; } }