`
chroya
  • 浏览: 655597 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

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;
	}
}
8
0
分享到:
评论
4 楼 night_iv 2016-01-27  
重写下面两个函数EditText完全作废,点击没有反应没有光标
@Override 
public boolean getDefaultEditable() { 
    return false; 

@Override 
protected MovementMethod getDefaultMovementMethod() { 
    return null; 
3 楼 kakafeika 2015-11-20  
[color=red][/color]
[size=x-small][/size]
2 楼 Coding.Ghost 2010-12-12  
liuborama 写道
有个问题请教一下:
这样写出来的程序是可以正常选择文字了,但是老是有光标闪动,有什么方法可以
把光标去掉呢?我尝试调用clearFocus方法,不管在Textpage初始化调用还是
在外面使用到它的实例处调用都不好用,请问有什么好方法吗?

貌似有个属性是设置光标不可见.我用过.但是记不清具体名字了.你可以查看下API文档
1 楼 liuborama 2010-09-16  
有个问题请教一下:
这样写出来的程序是可以正常选择文字了,但是老是有光标闪动,有什么方法可以
把光标去掉呢?我尝试调用clearFocus方法,不管在Textpage初始化调用还是
在外面使用到它的实例处调用都不好用,请问有什么好方法吗?

相关推荐

Global site tag (gtag.js) - Google Analytics