Подтвердить что ты не робот

Определение того, какое слово щелкнуто в андроиде textview

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

В идеале я просто слушал событие 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, но мне удалось вырвать код из API Android 15 и немного изменить его, чтобы сделать то, что мне нужно. Спасибо 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);
}
4b9b3361

Ответ 1

Если вы настроили таргетинг на уровень API 14+, вы можете использовать getOffsetForPosition().

Для предыдущей версии Android посмотрите, можете ли вы скопировать код этого метода из Android-источника и расширить свой собственный TextView.