4

I am using an EditText widget and would like to modify the context menu that is displayed when the user long presses the view. The problem that I am having is that I need to know the character position within the text of the long press so I can determine what I need to add to the context menu. The base class is doing this because one of the choices in the menu is 'Add “word_clicked_on” To Dictionary.' Setting ClickableSpans within the text does not appear to be a solution since it consumes the click event which makes it impossible to move the edit cursor within the spans.

Rodney Lambert
  • 707
  • 6
  • 7

1 Answers1

5

Here is the solution that I came up with and it does work so I wanted to share it:

First I concluded that I needed to extend the EditText class so that I could intercept the onTouchEvent, capture the ACTION_DOWN event, and save the position. Now that I have the position of the down point I can call getOffsetForPosition(downPointX, downPointY) and get the character position of the long-press. There is one big problem, getOffsetForPosition was not added until SDK 14! To make this solution work I had to back port the functionality of getOffsetForPosition and branch if the current SDK is earlier than SDK_INT 14. Here is the source code for the new class:

public class ScrapEditText extends EditText{

protected float LastDownPositionX, LastDownPositionY;

public ScrapEditText(Context context) 
{
    super(context);
}

public ScrapEditText(Context context, AttributeSet attrs)
{
    super(context, attrs);
}

@Override
public boolean onTouchEvent(MotionEvent event)
{
    final int action = event.getActionMasked();
    if(action == MotionEvent.ACTION_DOWN)
    {
        LastDownPositionX = event.getX();
        LastDownPositionY = event.getY();
    }
    return super.onTouchEvent(event);
}

public float GetLastDownPositionX()
{
    return LastDownPositionX;
}

public float GetLastDownPositionY()
{
    return LastDownPositionY;
}

public int GetOffsetForLastDownPosition()
{

    if(Build.VERSION.SDK_INT > 13)
    {
        // as of SDK 14 the getOffsetForPosition was added to TextView
        return getOffsetForPosition(LastDownPositionX, LastDownPositionY);
    }
    else
    {
        return GetOffsetForPositionOlderSdk();
    }
}

public int GetOffsetForPositionOlderSdk()
{
    if (getLayout() == null) return -1;
    final int line = GetLineAtCoordinateOlderSDK(LastDownPositionY);
    final int offset = GetOffsetAtCoordinateOlderSDK(line, LastDownPositionX);
    return offset;
}

public int GetLineAtCoordinateOlderSDK(float y)
{
    y -= getTotalPaddingTop();
    // Clamp the position to inside of the view.
    y = Math.max(0.0f, y);
    y = Math.min(getHeight() - getTotalPaddingBottom() - 1, y);
    y += getScrollY();
    return getLayout().getLineForVertical((int) y);     
}

protected int GetOffsetAtCoordinateOlderSDK(int line, float x) {
    x = ConvertToLocalHorizontalCoordinateOlderSDK(x);
    return getLayout().getOffsetForHorizontal(line, x);
}

protected float ConvertToLocalHorizontalCoordinateOlderSDK(float x) {
    x -= getTotalPaddingLeft();
    // Clamp the position to inside of the view.
    x = Math.max(0.0f, x);
    x = Math.min(getWidth() - getTotalPaddingRight() - 1, x);
    x += getScrollX();
    return x;
}

}

In your Activity derived class:

ScrapText = (ScrapEditText) findViewById(R.id.sample_text);  
ScrapText.setOnCreateContextMenuListener(new View.OnCreateContextMenuListener(){

@Override
public void onCreateContextMenu(ContextMenu menu, View v, ContextMenuInfo menuInfo) 
{
    int charOffset = FileText.GetOffsetForLastDownPosition();
}           
 });
Rodney Lambert
  • 707
  • 6
  • 7