问题 在Android中 - 如何使用ClickableSpan仅注册长按


我想在ClickableSpan包含的文本上注册点击次数,只要它们被点击超过1秒钟。有没有办法做到这一点?如果没有,捕获双击也没关系。

如果onClick方法捕获了一个包含有关点击的元数据的事件,那就太棒了 - 如果点击长度很短,我可以说忽略。

任何建议?

谢谢,维克多


5509
2018-01-02 15:33


起源



答案:


如果有人需要它,我发现它 这个地方

    package leeon.mobile.BBSBrowser;

    import android.text.Layout;
    import android.text.Selection;
    import android.text.Spannable;
    import android.text.method.LinkMovementMethod;
    import android.text.method.MovementMethod;
    import android.view.MotionEvent;
    import android.widget.TextView;

    public class LongClickLinkMovementMethod extends LinkMovementMethod {

      private Long lastClickTime = 0l;
      private int lastX = 0;
      private int lastY = 0;
      @Override
        public boolean onTouchEvent(TextView widget, Spannable buffer,
                                    MotionEvent event) {
            int action = event.getAction();

            if (action == MotionEvent.ACTION_UP ||
                action == MotionEvent.ACTION_DOWN) {
                int x = (int) event.getX();
                int y = (int) event.getY();
                lastX = x;
                lastY = y;
                int deltaX = Math.abs(x-lastX);
                int deltaY = Math.abs(y-lastY);

                x -= widget.getTotalPaddingLeft();
                y -= widget.getTotalPaddingTop();

                x += widget.getScrollX();
                y += widget.getScrollY();

                Layout layout = widget.getLayout();
                int line = layout.getLineForVertical(y);
                int off = layout.getOffsetForHorizontal(line, x);

                LongClickableSpan[] link = buffer.getSpans(off, off, LongClickableSpan.class);

                if (link.length != 0) {
                    if (action == MotionEvent.ACTION_UP) {
                      if (System.currentTimeMillis() - lastClickTime < 1000)
                        link[0].onClick(widget);
                      else if (deltaX < 10 && deltaY < 10)
                        link[0].onLongClick(widget);
                    } else if (action == MotionEvent.ACTION_DOWN) {
                        Selection.setSelection(buffer,
                                               buffer.getSpanStart(link[0]),
                                               buffer.getSpanEnd(link[0]));
                        lastClickTime = System.currentTimeMillis();
                    }
                    return true;
                }
            }

            return super.onTouchEvent(widget, buffer, event);
        }


        public static MovementMethod getInstance() {
            if (sInstance == null)
                sInstance = new LongClickLinkMovementMethod();

            return sInstance;
        }

        private static LongClickLinkMovementMethod sInstance;
    }


      [1]: http://java2s.com/Open-Source/Java/SSH/brick-leeon/leeon/mobile/BBSBrowser/LongClickLinkMovementMethod.java.htm

LongClickableSpan在同一个地方:

    package leeon.mobile.BBSBrowser;

    import android.text.style.ClickableSpan;
    import android.view.View;

    public abstract class LongClickableSpan extends ClickableSpan {

      abstract public void onLongClick(View view);

    }

10
2017-12-07 00:20



嗨,谢谢你的这个例子。如果我的班级用于扩展ClickableSpan,我现在需要如何更改它才能使用你的课程? - bahar_p
我添加了longclickablespan,只需将textView的文本移动方法设置为LongClickMovementMethod.getInstance,并将ClickableSpans替换为LongClickableSpans,而不是现在有一个onLongClick方法!希望能帮助到你 ! - rupps
它只有在你松开触摸后才有效。如果你能以这样的方式修改它,即使有一段时间的触地得分,比如1或2秒,它会触发事件,这将使这个答案更加精彩。 - ashutiwari4
我现在不在Android上,会添加你的建议,应该很容易做到 widget.postDelayed(executeLongPressRunnable);  后 lastClickTime = System.currentTimeMillis();,哪里 executeLongPressRunnable 里面的东西是'Runnable' if (action == MotionEvent.ACTION_UP) {... 。在Action UP上你还应该设置一个标志,这样在触摸释放的情况下runnable不会重新点击。 - rupps


答案:


如果有人需要它,我发现它 这个地方

    package leeon.mobile.BBSBrowser;

    import android.text.Layout;
    import android.text.Selection;
    import android.text.Spannable;
    import android.text.method.LinkMovementMethod;
    import android.text.method.MovementMethod;
    import android.view.MotionEvent;
    import android.widget.TextView;

    public class LongClickLinkMovementMethod extends LinkMovementMethod {

      private Long lastClickTime = 0l;
      private int lastX = 0;
      private int lastY = 0;
      @Override
        public boolean onTouchEvent(TextView widget, Spannable buffer,
                                    MotionEvent event) {
            int action = event.getAction();

            if (action == MotionEvent.ACTION_UP ||
                action == MotionEvent.ACTION_DOWN) {
                int x = (int) event.getX();
                int y = (int) event.getY();
                lastX = x;
                lastY = y;
                int deltaX = Math.abs(x-lastX);
                int deltaY = Math.abs(y-lastY);

                x -= widget.getTotalPaddingLeft();
                y -= widget.getTotalPaddingTop();

                x += widget.getScrollX();
                y += widget.getScrollY();

                Layout layout = widget.getLayout();
                int line = layout.getLineForVertical(y);
                int off = layout.getOffsetForHorizontal(line, x);

                LongClickableSpan[] link = buffer.getSpans(off, off, LongClickableSpan.class);

                if (link.length != 0) {
                    if (action == MotionEvent.ACTION_UP) {
                      if (System.currentTimeMillis() - lastClickTime < 1000)
                        link[0].onClick(widget);
                      else if (deltaX < 10 && deltaY < 10)
                        link[0].onLongClick(widget);
                    } else if (action == MotionEvent.ACTION_DOWN) {
                        Selection.setSelection(buffer,
                                               buffer.getSpanStart(link[0]),
                                               buffer.getSpanEnd(link[0]));
                        lastClickTime = System.currentTimeMillis();
                    }
                    return true;
                }
            }

            return super.onTouchEvent(widget, buffer, event);
        }


        public static MovementMethod getInstance() {
            if (sInstance == null)
                sInstance = new LongClickLinkMovementMethod();

            return sInstance;
        }

        private static LongClickLinkMovementMethod sInstance;
    }


      [1]: http://java2s.com/Open-Source/Java/SSH/brick-leeon/leeon/mobile/BBSBrowser/LongClickLinkMovementMethod.java.htm

LongClickableSpan在同一个地方:

    package leeon.mobile.BBSBrowser;

    import android.text.style.ClickableSpan;
    import android.view.View;

    public abstract class LongClickableSpan extends ClickableSpan {

      abstract public void onLongClick(View view);

    }

10
2017-12-07 00:20



嗨,谢谢你的这个例子。如果我的班级用于扩展ClickableSpan,我现在需要如何更改它才能使用你的课程? - bahar_p
我添加了longclickablespan,只需将textView的文本移动方法设置为LongClickMovementMethod.getInstance,并将ClickableSpans替换为LongClickableSpans,而不是现在有一个onLongClick方法!希望能帮助到你 ! - rupps
它只有在你松开触摸后才有效。如果你能以这样的方式修改它,即使有一段时间的触地得分,比如1或2秒,它会触发事件,这将使这个答案更加精彩。 - ashutiwari4
我现在不在Android上,会添加你的建议,应该很容易做到 widget.postDelayed(executeLongPressRunnable);  后 lastClickTime = System.currentTimeMillis();,哪里 executeLongPressRunnable 里面的东西是'Runnable' if (action == MotionEvent.ACTION_UP) {... 。在Action UP上你还应该设置一个标志,这样在触摸释放的情况下runnable不会重新点击。 - rupps


我想做同样的事情,唯一想到的方法是自定义MovementMethod类替换LinkMovementMethod和ClickableSpan的替代品,它添加了一个长按抽象方法。它实际上很简单:

  1. 下载Android源代码并找到LinkMovementMethod和ClickableSpan类
  2. 从ClickableSpan创建MyCustomClickableSpan类并添加抽象方法onLongClick()
  3. 从LinkMovementMethod创建MyCustomLinkMovementMethod类,其中自定义onTouchEvent,以便在正确的位置调用MyCustomClickableSpan的onLongClick()
  4. 在代码中使用新类而不是LinkMovementMethod和ClickableSpan

编辑: 奇迹般有效。我犯了一些愚蠢的错误,例如从不同的Android版本下载LinkMovementMethod而不是在setSpan()中使用新的ClickableSpan类,但没有什么严重的。


1
2017-10-19 11:51



老问题..但我想知道你是否仍然有这个代码,可以在这里发布你的OnTouchEvent .. - rupps
请为OnTouchEvent提供一些代码 - do01