Удалить подчеркивание из ссылок в TextView - Android

Я использую два textview чтобы отобразить ссылки из базы данных, мне удалось изменить цвета ссылок, но я хочу удалить подчеркивание

email.setText(c.getString(5));
    website.setText(c.getString(6));
    Linkify.addLinks(email, Linkify.ALL);
    Linkify.addLinks(website, Linkify.ALL);

Могу ли я сделать это из XML или кода?

14 ответов

Решение

Вы можете сделать это в коде, найдя и заменив URLSpan экземпляры с версиями, которые не подчеркиваются. После звонка Linkify.addLinks()вызовите функцию stripUnderlines() наклеены ниже на каждый из ваших TextViews:

    private void stripUnderlines(TextView textView) {
        Spannable s = new SpannableString(textView.getText());
        URLSpan[] spans = s.getSpans(0, s.length(), URLSpan.class);
        for (URLSpan span: spans) {
            int start = s.getSpanStart(span);
            int end = s.getSpanEnd(span);
            s.removeSpan(span);
            span = new URLSpanNoUnderline(span.getURL());
            s.setSpan(span, start, end, 0);
        }
        textView.setText(s);
    }

Для этого требуется настроенная версия URLSpan, которая не включает свойство подчеркивания TextPaint:

    private class URLSpanNoUnderline extends URLSpan {
        public URLSpanNoUnderline(String url) {
            super(url);
        }
        @Override public void updateDrawState(TextPaint ds) {
            super.updateDrawState(ds);
            ds.setUnderlineText(false);
        }
    }

Учитывая textView и содержание:

TextView textView = (TextView) findViewById(R.id.your_text_view_id);
String content = "your <a href='http://some.url'>html</a> content";

Вот краткий способ удалить подчеркивания из гиперссылок:

Spannable s = (Spannable) Html.fromHtml(content);
for (URLSpan u: s.getSpans(0, s.length(), URLSpan.class)) {
    s.setSpan(new UnderlineSpan() {
        public void updateDrawState(TextPaint tp) {
            tp.setUnderlineText(false);
        }
    }, s.getSpanStart(u), s.getSpanEnd(u), 0);
}
textView.setText(s);

Это основано на подходе, предложенном robUx4.

Чтобы сделать ссылки кликабельными, вам также необходимо позвонить:

textView.setMovementMethod(LinkMovementMethod.getInstance());

Вот функция расширения Kotlin:

fun TextView.removeLinksUnderline() {
    val spannable = SpannableString(text)
    for (u in spannable.getSpans(0, spannable.length, URLSpan::class.java)) {
        spannable.setSpan(object : URLSpan(u.url) {
            override fun updateDrawState(ds: TextPaint) {
                super.updateDrawState(ds)
                ds.isUnderlineText = false
            }
        }, spannable.getSpanStart(u), spannable.getSpanEnd(u), 0)
    }
    text = spannable
}

Применение:

txtView.removeLinksUnderline()    

UnderlineSpan уже существует, но может только установить подчеркивание.

Другое решение состоит в том, чтобы добавить не подчеркивающий интервал для каждого существующего URLSpan, Таким образом, состояние подчеркивания отключено непосредственно перед рисованием. Таким образом, вы сохраните URLSpan (возможно пользовательские) классы и все другие стили, установленные в другом месте.

public class NoUnderlineSpan extends UnderlineSpan {
    public NoUnderlineSpan() {}

    public NoUnderlineSpan(Parcel src) {}

    @Override
    public void updateDrawState(TextPaint ds) {
        ds.setUnderlineText(false);
    }
}

Вот как вы устанавливаете его, не удаляя существующий объект URLSpan:

URLSpan[] spans = s.getSpans(0, s.length(), URLSpan.class);
for (URLSpan span: spans) {
    int start = s.getSpanStart(span);
    int end = s.getSpanEnd(span);
    NoUnderlineSpan noUnderline = new NoUnderlineSpan();
    s.setSpan(noUnderline, start, end, 0);
}

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

package com.example.view;

import android.content.Context;
import android.support.v7.widget.AppCompatTextView;
import android.text.Spannable;
import android.text.SpannableString;
import android.text.style.URLSpan;
import android.util.AttributeSet;

import com.example.utils.UrlSpanNoUnderline;

public class TextViewNoUnderline extends AppCompatTextView {
    public TextViewNoUnderline(Context context) {
        this(context, null);
    }

    public TextViewNoUnderline(Context context, AttributeSet attrs) {
        this(context, attrs, android.R.attr.textViewStyle);
    }

    public TextViewNoUnderline(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        setSpannableFactory(Factory.getInstance());
    }

    private static class Factory extends Spannable.Factory {
        private final static Factory sInstance = new Factory();

        public static Factory getInstance() {
            return sInstance;
        }

        @Override
        public Spannable newSpannable(CharSequence source) {
            return new SpannableNoUnderline(source);
        }
    }

    private static class SpannableNoUnderline extends SpannableString {
        public SpannableNoUnderline(CharSequence source) {
            super(source);
        }

        @Override
        public void setSpan(Object what, int start, int end, int flags) {
            if (what instanceof URLSpan) {
                what = new UrlSpanNoUnderline((URLSpan) what);
            }
            super.setSpan(what, start, end, flags);
        }
    }
}

И код для UrlSpanNoUnderline:

package com.jankstudios.smmagazine.utils;

import android.text.TextPaint;
import android.text.style.URLSpan;

public class UrlSpanNoUnderline extends URLSpan {
    public UrlSpanNoUnderline(URLSpan src) {
        super(src.getURL());
    }

    @Override
    public void updateDrawState(TextPaint ds) {
        super.updateDrawState(ds);
        ds.setUnderlineText(false);
    }
}

Если вы используете свойство автоматической ссылки Textview и хотите удалить подчеркивания, вы можете использовать его:

Сначала расширьте UnderlineSpan и удалите подчеркивание:

public class NoUnderlineSpan extends UnderlineSpan {

    @Override
    public void updateDrawState(TextPaint ds) {
        ds.setUnderlineText(false);
    }
}

Во-вторых, создайте и создайте экземпляр NoUnderlineSpan, создайте Spannable из текста String и установите span для spannable:

NoUnderlineSpan mNoUnderlineSpan = new NoUnderline();
if (yourTextView.getText() instanceof Spannable) {
    Spannable s = (Spannable) yourTextView.getText();
    s.setSpan(mNoUnderlineSpan, 0, s.length(), Spanned.SPAN_MARK_MARK);
}

Ссылка: http://prog3.com/sbdm/blog/maosidiaoxian/article/details/39156563

Вот функция расширения Kotlin как решение для удаления всех UrlSpan подчеркивает:

      private fun Spannable.removeAllUrlSpanUnderline() {
    for (urlSpan in getSpans(0, length, URLSpan::class.java)) {
        setSpan(object : UnderlineSpan() {
            override fun updateDrawState(tp: TextPaint) {
                tp.isUnderlineText = false
            }
        }, getSpanStart(urlSpan), getSpanEnd(urlSpan), 0)
    }
}

В моем случае я начал со строки с тегами href. fromHtml возвращает Spanned так что бросьте это в Spannableтак что это изменчиво. См. Пример использования ниже:

      val sampleHtmlString = "<a href=\"www.google.com\">first Link</a> and <a href=\"www.google.com\">second link</a>"

val sampleSpannable = HtmlCompat.fromHtml(caslString, HtmlCompat.FROM_HTML_MODE_LEGACY) as Spannable

val sampleSpannableWithoutUnderlines = sampleSpannable.removeAllUrlSpanUnderline()

Теперь вы можете установить sampleSpannableWithoutUnderlines на ваш TextView

Всякий раз, когда попробуйте удалить подчеркивание URL с помощью spannable, я предлагаю только эти вещи:

1> Создание пользовательского класса:

private class URLSpanNoUnderline extends URLSpan {
            public URLSpanNoUnderline(String url) {
                super(url);
            }
            @Override public void updateDrawState(TextPaint ds) {
                super.updateDrawState(ds);
                ds.setUnderlineText(false);
            }
        }

Для этого требуется настроенная версия URLSpan, которая не включает свойство подчеркивания TextPaint

2> setSpan с включаемым текстом:

spannableText.setSpan(new URLSpanNoUnderline(UrlText), 0, UrlText.length() , 0);

Здесь spannableText является объектом SpannableString...!!!

Вот мой метод

 public static void removeUnderlines(Spannable p_Text) {
            if (p_Text != null && p_Text.toString().length() > 0) {
                URLSpan[] spans = p_Text.getSpans(0, p_Text.length(), URLSpan.class);

                for (URLSpan span : spans) {
                    int start = p_Text.getSpanStart(span);
                    int end = p_Text.getSpanEnd(span);
                    p_Text.removeSpan(span);
                    span = new URLSpanNoUnderline(span.getURL());
                    p_Text.setSpan(span, start, end, 0);
                }
            }
        }

Назови это так

AppController.removeUnderlines((Spannable) eventEmail.getText());

Appcontroller - это мой класс приложения, в который я поместил этот метод, чтобы я мог получить к нему доступ из любого места

Для пользователей Xamarin, нашедших этот пост, вот как я заставил его работать:

  1. Создайте пользовательский класс span для обработки подчеркивания.

    class URLSpanNoUnderline: URLSpan {общедоступный URLSpanNoUnderline (строковый URL): базовый (URL) { }

        public override void UpdateDrawState (TextPaint ds) {
            base.UpdateDrawState (ds);
            ds.UnderlineText = false;
        }
    }
    
  2. Создайте метод расширения, чтобы найти все диапазоны URL-адресов и заменить их нашими пользовательскими интервалами.

    public static void StripUnderlinesFromLinks (это TextView textView) { var spannable = new SpannableStringBuilder (textView.TextFormatted); var spans = spannable.GetSpans (0, spannable.Length (), Java.Lang.Class.FromType (typeof (URLSpan))); foreach (диапазон URLSpan в промежутках) { var start = spannable.GetSpanStart (span); var end = spannable.GetSpanEnd (span); spannable.RemoveSpan(диапазон); var newSpan = new URLSpanNoUnderline (span.URL); spannable.SetSpan(newSpan, start, end, 0); } textView.TextFormatted = spannable; }

Если вы просто хотите текст и не беспокоиться о ссылке URL

Это будет ПОЛОЖИТЬ ссылку, но держать текст

private Spannable stripLinks(String content) {
    Spannable s = new SpannableString(content);
    URLSpan[] spans = s.getSpans(0, s.length(), URLSpan.class);
    for (URLSpan span : spans) {
        s.removeSpan(span);
    }

    return s;
}

дополнительные занятия не требуются

String content = "<a href='http://stackru.com'>Stack Overflow</a> Rocks!";
textView.setText(stripLinks(content));
      abstract class NoUnderlineClickableSpan : ClickableSpan() {        
    override fun updateDrawState(ds: TextPaint) {
        super.updateDrawState(ds)
        ds.isUnderlineText = false
    }    
}

Подчеркивание в Android: AutoLink можно удалить с помощью Android: TextAllCaps="true" и Android: TextIsSelectable="false"

<TextView
    android:autoLink="phone"
    android:text="070 300 1155"
    android:textAllCaps="true"
    android:textIsSelectable="false"
     />
public void setView()
{
TextView t=(TextView) findViewById(R.id.textView3);
        t.setText(Html.fromHtml("<a href=http://www.urdusms.net > UrduSMS "));
        t.setTextColor(Color.BLACK);
        t.setGravity(Gravity.CENTER);
        t.setMovementMethod(LinkMovementMethod.getInstance());

         Spannable s = (Spannable) t.getText();
            URLSpan[] spans = s.getSpans(0, s.length(), URLSpan.class);
            for (URLSpan span: spans) {
                int start = s.getSpanStart(span);
                int end = s.getSpanEnd(span);
                s.removeSpan(span);
                span = new URLSpanline_none(span.getURL());
                s.setSpan(span, start, end, 0);
            }
        t.setText(s);
 }
//inner class is    
private class URLSpanline_none extends URLSpan {
            public URLSpanline_none(String url) {
                super(url);
            }
            @Override public void updateDrawState(TextPaint ds) {
                super.updateDrawState(ds);
                ds.setUnderlineText(false);
            }
        }
Другие вопросы по тегам