简体   繁体   English

使用ClickableSpan将TextView上的触摸事件重定向到父视图

[英]Redirect touch event on TextView with ClickableSpan to parent view

I have a RecyclerView with TextViews that could contain custom hashtags that should be clickable. 我有一个带有TextViews的RecyclerView,其中可能包含应该可单击的自定义主题标签。 So I have created subclass of TextView in which using Pattern I create ClickableSpan. 因此,我创建了TextView的子类,其中使用模式创建了ClickableSpan。 In order for ClickableSpan to be active, I have added 为了使ClickableSpan处于活动状态,我添加了

setMovementMethod(LinkMovementMethod.getInstance())

This method changes properties of TextView: 此方法更改TextView的属性:

setFocusable(true);
setClickable(true);
setLongClickable(true);

Click on links works, but it prevents ripple drawable to be shown on list item, and click on TextView outside of hashtags are ignored. 单击链接有效,但是它阻止了可绘制的涟漪图显示在列表项上,而单击井号之外的TextView将被忽略。

So I'm interested how can I redirect touches on TextView (all except on hashtag) to it's parent? 因此,我很感兴趣如何将对TextView的触摸(除hashtag之外的所有触摸)重定向到其父级?

So I haven't found solution on the internet so I ended up implementing it by myself. 所以我没有在互联网上找到解决方案,所以我最终自己实现了它。

So instead of using simple TextView with MovementMethod, I created a new class extending text view with onTouchEvent. 因此,我没有将简单的TextView与MovementMethod一起使用,而是创建了一个新类,该类使用onTouchEvent扩展文本视图。

public class MyTextView extends TextView {
@Override
  public boolean onTouchEvent(@NonNull MotionEvent event) {
    if (linksActive) {
      Layout layout = this.getLayout();
      if (layout != null) {
        int line = layout.getLineForVertical((int) event.getY());
        int offset = layout.getOffsetForHorizontal(line, event.getX());

        if (getText() != null && getText() instanceof Spanned) {
          Spanned spanned = (Spanned) getText();

          ClickableSpan[] links = spanned.getSpans(offset, offset, ClickableSpan.class);

          if (links.length > 0) {

            if (event.getAction() == MotionEvent.ACTION_DOWN) {
              return true;
            } else if (event.getAction() == MotionEvent.ACTION_UP) {
              links[0].onClick(this);
            } else {
              return super.onTouchEvent(event);
            }
          }
        }
      }
    }

    return super.onTouchEvent(event);
  }
}

Also I created a class that extends ClickableSpan which handles clicks: 我还创建了一个扩展ClickableSpan的类来处理点击:

public class URLSpan extends ClickableSpan {
  private String url;
  private int spanType;

  public URLSpan(String url, int spanType) {
    this.url = url;
    this.spanType = spanType;
  }

  public int getType() {
    return spanType;
  }

  @Override
  public void onClick(@NonNull View widget) {
    if (widget instanceof OnSpanClickListener) {
      ((OnSpanClickListener) widget).onSpanClicked(url, type);
    } else {
      // ignored
    }
  }

}

And I this case I have to manage spans by myself (in setText() method) 在这种情况下,我必须自己管理跨区(在setText()方法中)

public static void formatPhoneNumbers(@NonNull Spannable spannable, @ColorInt int color) {
    Matcher matcher = MY_Patterns.PHONE_NO.matcher(spannable);
    while (matcher.find()) {
      if (!Linkify.sPhoneNumberMatchFilter.acceptMatch(spannable, matcher.start(), matcher.end())) {
        continue;
      }

      removeSpans(spannable, SpanType.PHONE_NO, matcher.start(), matcher.end());

      if (hasClickableSpans(spannable, matcher.start(), matcher.end())) {
        // ignore different clickable spans overlap
        continue;
      }

      final ForegroundColorSpan span = new ForegroundColorSpan(color);
      final ClickableSpan urlSpan = new URLSpan(matcher.group(), SpanType.PHONE_NO);

      spannable.setSpan(urlSpan, matcher.start(), matcher.end(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
      spannable.setSpan(span, matcher.start(), matcher.end(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
    }
  }

  private static void removeSpans(@NonNull Spannable spannable, int type, int start, int end) {
    URLSpan[] arr = spannable.getSpans(start, end, URLSpan.class);
    for (URLSpan span : arr) {
      if (span.getType() == type) {
        spannable.removeSpan(span);
      }
    }
  }

  private static boolean hasClickableSpans(@NonNull Spannable spannable, int start, int end) {
    ClickableSpan[] arr = spannable.getSpans(start, end, ClickableSpan.class);
    return arr.length > 0;
  }

Thanks to Viktor -- 感谢Viktor-

I use this to allow touch events not on a clickable span to pass through to the parent 我用它来允许不在可点击范围内的触摸事件传递给父级

override fun dispatchTouchEvent(event: MotionEvent?): Boolean {

        if (hasOnClickListeners() == false && linksClickable == false) {
            // Avoid all touches
            return false
        }

        // Handle avoiding touch events, if not on a clickable span
        if (event != null && hasOnClickListeners() == false) {
            if (layout != null) {
                val line = layout.getLineForVertical(event.y.toInt())
                val offset = layout.getOffsetForHorizontal(line, event.x)

                if (text != null && text is Spanned) {
                    val spanned = text as Spanned

                    val spansAtPoint = spanned.getSpans(offset, offset, ClickableSpan::class.java)
                    if (spansAtPoint.size == 0) {
                        // No clickable span at event -- abort touch
                        return false
                    }
                }
            }
        }
        return super.dispatchTouchEvent(event)
    }

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM