Определение того, какое слово было нажато в текстовом представлении Android

По сути, я хочу отобразить отрывок текста (, потенциально довольно длинный текст ), и позволить пользователю щелкнуть любое слово. В этот момент я хочу определить, на какое слово они нажали. Я также хочу получить все предложение, в котором встречается слово (это довольно тривиально, предполагая, что я могу определить, в какой позиции находится слово в тексте ).

В идеале я бы просто прослушивал событие onTouch, получал X и Y и говорил что-то вроде textView.wordAt(event.x, event.y)или textView.cursorPositionNearest(event.x, event.y), но кажется, что это не так просто:-)

Мои текущие лучшие усилия включают использование TextView и создание одного ClickableSpan для каждого слова. Это работает, но не совсем элегантно, и я думаю, что оно начнет потреблять память, если я буду использовать его для более длинных текстов.

private final String text = "This is the text";
private TextView textView;

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_text_view);

    textView = (TextView) findViewById(R.id.text_view);

    SpannableString ss = new SpannableString(text);
    //  create spans for "this", "is", "the" and "text"
    ss.setSpan(new IndexedClickableSpan(0, 4), 0, 4, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
    ss.setSpan(new IndexedClickableSpan(5, 7), 5, 7, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
    ss.setSpan(new IndexedClickableSpan(8, 11), 8, 11, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
    ss.setSpan(new IndexedClickableSpan(12, 16), 12, 16, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);

    textView.setText(ss);
}

private final class IndexedClickableSpan extends ClickableSpan {

    int startIndex, endIndex;

    public IndexedClickableSpan(int startIndex, int endIndex) {
        this.startIndex = startIndex;
        this.endIndex = endIndex;
    }

    @Override
    public void onClick(View widget) {
        String word = TextViewActivity.this.text.substring(startIndex, endIndex);
        Toast.makeText(TextViewActivity.this, "You clicked on " + word, Toast.LENGTH_SHORT).show();
    }
}

Если у кого-то есть идея получше, я бы хотел ее услышать.

Заранее спасибо, Дэйв

Не уверен, как я действительно должен отвечать на вопросы о stackoverflow, но мне удалось вырвать некоторый код из Android 15 API и немного изменить его, чтобы делать то, что мне нужно. Спасибо Dheeraj за предложение.

Новый код позволяет мне получить позицию курсора на основе позиции события касания, оттуда я должен получить слово, которое было затронуто, и предложение, в котором оно появляется. Код прилагается:

public int getOffsetForPosition(TextView textView, float x, float y) {
    if (textView.getLayout() == null) {
        return -1;
    }
    final int line = getLineAtCoordinate(textView, y);
    final int offset = getOffsetAtCoordinate(textView, line, x);
    return offset;
}

private int getOffsetAtCoordinate(TextView textView2, int line, float x) {
    x = convertToLocalHorizontalCoordinate(textView2, x);
    return textView2.getLayout().getOffsetForHorizontal(line, x);
}

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

private int getLineAtCoordinate(TextView textView2, float y) {
    y -= textView2.getTotalPaddingTop();
    // Clamp the position to inside of the view.
    y = Math.max(0.0f, y);
    y = Math.min(textView2.getHeight() - textView2.getTotalPaddingBottom() - 1, y);
    y += textView2.getScrollY();
    return textView2.getLayout().getLineForVertical((int) y);
}
22
задан Dheeraj V.S. 23 July 2012 в 04:08
поделиться