如何使TextView中的文本链接可点击

3
Android Studio 2.3.1

我正在尝试创建一些不是网页或HTML,而只是普通文本的内容,我希望它看起来像一个可点击的网页链接。这段文本是:包含3个评论,我想让它看起来像一个可点击的网页链接。
private void setupTextViewAsLinkClickable() {
    if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
        mTvReviews.setMovementMethod(LinkMovementMethod.getInstance());
        mTvReviews.setText(Html.fromHtml("Contains 3 reviews", Html.FROM_HTML_MODE_LEGACY));
    }
    else {
        mTvReviews.setMovementMethod(LinkMovementMethod.getInstance());
        mTvReviews.setText(Html.fromHtml("Contains 3 reviews"));
    }
}

我也试过对我的XML文件进行以下操作:
        <TextView
            android:id="@+id/tvReviews"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:fontFamily="sans-serif-condensed"
            android:autoLink="all"
            android:linksClickable="true"
            android:clickable="true"
            android:textSize="@dimen/runtime_textsize"
            android:text="Contains 3 reviews" />
2个回答

8

尝试使用这段代码,它是我项目中的可工作代码。

SpannableString ss = new SpannableString("Android is a Software stack");
ClickableSpan clickableSpan = new ClickableSpan() {
    @Override
    public void onClick(View textView) {
        startActivity(new Intent(MyActivity.this, NextActivity.class));
    }
    @Override
    public void updateDrawState(TextPaint ds) {
            super.updateDrawState(ds);
            ds.setUnderlineText(false);
        }
};
ss.setSpan(clickableSpan, 22, 27, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);

TextView textView = (TextView) findViewById(R.id.hello);
textView.setText(ss);
textView.setMovementMethod(LinkMovementMethod.getInstance());
textView.setHighlightColor(Color.TRANSPARENT);

如何设置文本视图中的某一部分可点击


2
你可以使用这个类来让TextView看起来像网页链接,甚至可以点击。
public final class LinkUtils {
    public static final Pattern URL_PATTERN =
            Pattern.compile("((https?|ftp)(:\\/\\/[-_.!~*\\'()a-zA-Z0-9;\\/?:\\@&=+\\$,%#]+))");

    public interface OnClickListener {
        void onLinkClicked(final String link);

        void onClicked();
    }

    static class SensibleUrlSpan extends URLSpan {
        /**
         * Pattern to match.
         */
        private Pattern mPattern;

        public SensibleUrlSpan(String url, Pattern pattern) {
            super(url);
            mPattern = pattern;
        }

        public boolean onClickSpan(View widget) {
            boolean matched = mPattern.matcher(getURL()).matches();
            if (matched) {
                super.onClick(widget);
            }
            return matched;
        }
    }

    static class SensibleLinkMovementMethod extends LinkMovementMethod {

        private boolean mLinkClicked;

        private String mClickedLink;

        @Override
        public boolean onTouchEvent(TextView widget, Spannable buffer, MotionEvent event) {
            int action = event.getAction();

            if (action == MotionEvent.ACTION_UP) {
                mLinkClicked = false;
                mClickedLink = null;
                int x = (int) event.getX();
                int y = (int) event.getY();

                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);

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

                if (link.length != 0) {
                    SensibleUrlSpan span = (SensibleUrlSpan) link[0];
                    mLinkClicked = span.onClickSpan(widget);
                    mClickedLink = span.getURL();
                    return mLinkClicked;
                }
            }
            super.onTouchEvent(widget, buffer, event);

            return false;
        }

        public boolean isLinkClicked() {
            return mLinkClicked;
        }

        public String getClickedLink() {
            return mClickedLink;
        }

    }

    public static void autoLink(final TextView view, final OnClickListener listener) {
        autoLink(view, listener, null);
    }

    public static void autoLink(final TextView view, final OnClickListener listener,
                                final String patternStr) {
        String text = view.getText().toString();
        if (TextUtils.isEmpty(text)) {
            return;
        }
        Spannable spannable = new SpannableString(text);

        Pattern pattern;
        if (TextUtils.isEmpty(patternStr)) {
            pattern = URL_PATTERN;
        } else {
            pattern = Pattern.compile(patternStr);
        }
        Matcher matcher = pattern.matcher(text);
        while (matcher.find()) {
            SensibleUrlSpan urlSpan = new SensibleUrlSpan(matcher.group(1), pattern);
            spannable.setSpan(urlSpan, matcher.start(1), matcher.end(1),
                    Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
        }
        view.setText(spannable, TextView.BufferType.SPANNABLE);

        final SensibleLinkMovementMethod method = new SensibleLinkMovementMethod();
        view.setMovementMethod(method);
        if (listener != null) {
            view.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    if (method.isLinkClicked()) {
                        listener.onLinkClicked(method.getClickedLink());
                    } else {
                        listener.onClicked();
                    }
                }
            });
        }
    }
}

在活动调用中

String testStr = "Text 。http://www.yahoo.com , Text ";

textView1.setTextSize(20);
textView1.setText(testStr);

LinkUtils.autoLink(textView1, new LinkUtils.OnClickListener() {
    @Override
    public void onLinkClicked(final String link) {
                    Log.i("Log", "Log"+link);
    }

    @Override
    public void onClicked() {
        Log.i("Log", "Log");
    }
});

我尝试了第二个。文本看起来很正常。 - ant2009
当您想要点击TextView并执行操作时,这将是解决方案。如果这正是您要寻找的,请告诉我。 - Moien.Dev
是的,我想点击TextView并打开一个DialogFragment。 - ant2009

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接