带有 span 的 TextView,我怎么知道点击了哪个?

TextView with spans , how can I know which one is clicked on?

我正在使用这个库,但是从他们拥有的 onClick 样式来看,它们都不能处理我需要的情况。 https://github.com/splitwise/TokenAutoComplete

所以我想自己做。所以我有一个ContactsCompletionView,这是一个TextView。然后我像这样重写 onTouchEvent

override fun onTouchEvent(event: MotionEvent): Boolean {
    val action = event.actionMasked
    val text = text
    var handled = super.onTouchEvent(event)
    if (isFocused && text != null && action == MotionEvent.ACTION_UP) {
        val offset = getOffsetForPosition(event.x, event.y)
        if (offset != -1) {
            var offseted = text.substring(offset, text.length)
            var indexLeft = offseted.indexOf("(") + 1
            var indexRight = offseted.indexOf(")")
            if (indexLeft > 0 && indexRight > indexLeft)
                Toast.makeText(context, offseted.substring(indexLeft, indexRight), Toast.LENGTH_SHORT).show()
        }
    }
    return handled
}

这是他们拥有的,但我不能使用 TokenImageSpan 因为它是受保护的 class:

@Override
public boolean onTouchEvent(@NonNull MotionEvent event) {
    int action = event.getActionMasked();
    Editable text = getText();
    boolean handled = false;

    if (tokenClickStyle == TokenClickStyle.None) {
        handled = super.onTouchEvent(event);
    }

    if (isFocused() && text != null && lastLayout != null && action == MotionEvent.ACTION_UP) {

        int offset = getOffsetForPosition(event.getX(), event.getY());

        if (offset != -1) {
            TokenImageSpan[] links = text.getSpans(offset, offset, TokenImageSpan.class);

            if (links.length > 0) {
                links[0].onClick();
                handled = true;
            } else {
                //We didn't click on a token, so if any are selected, we should clear that
                clearSelections();
            }
        }
    }

    if (!handled && tokenClickStyle != TokenClickStyle.None) {
        handled = super.onTouchEvent(event);
    }
    return handled;

}

我的代码有效,但我的问题是每当我在它的末尾按下标签时。它获取下一个对象。我认为这是因为我只是使用:

 val offset = getOffsetForPosition(event.x, event.y)
 if (offset != -1) {
     var offseted = text.substring(offset, text.length)
 }

当他们使用时:

  if (offset != -1) {
      TokenImageSpan[] links = text.getSpans(offset, offset, TokenImageSpan.class);
  }

TokenImageSpan 扩展了 ImageSpan,所以我可以那样使用它,但我不知道如何从 ImageSpan 中获取文本。请问我该如何解决这个问题?

您可以使用如下的可点击范围:

SpannableString ss = new SpannableString("your string comes 
here");
ClickableSpan clickableSpan = new ClickableSpan() {
@Override
public void onClick(View textView) {
    //do your stuff here on click
}
@Override
public void updateDrawState(TextPaint ds) {
    super.updateDrawState(ds);
    ds.setUnderlineText(false);
}
};

//set click range
ss.setSpan(clickableSpan, 8, 15, 
Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);

//your text view or edittext
textView.setText(ss);  
textView.setMovementMethod(LinkMovementMethod.getInstance());
textView.setHighlightColor(Color.TRANSPARENT);

这是我用的:

  override fun onTouchEvent(event: MotionEvent): Boolean {
    val action = event.actionMasked
    val text = text
    var handled = super.onTouchEvent(event)
    var offset = getOffsetForPosition(event.getX(), event.getY())
    if (isFocused && text != null && action == MotionEvent.ACTION_UP) {
        var cursor = this@ContactsCompletionView.selectionEnd
        if (cursor < 0)
            cursor = 0
        val links: Array<ViewSpan> = text.getSpans(0, cursor, ViewSpan::class.java)
        if (objects.size > 0 && objects.size >= links.size && links.size > 0 && offset < cursor)
            Snackbar().make(this, objects[links.size - 1].address, com.google.android.material.snackbar.Snackbar.LENGTH_SHORT).show()

    }

    return handled
}

如果将 ClickStyle 设置为 NONE,库会将光标移动到所选对象的末尾。所以我创建了它的一个子字符串,检查我有多少跨度,然后我用它作为我的对象的索引。为了显示光标左侧的第一个