Видалити підкреслення з посилань у TextView - Android


93

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

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

Чи можу я це зробити з XML або коду?

Відповіді:


220

Ви можете зробити це в коді, знайшовши та замінивши 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);
        }
    }

9
Це рішення передбачає, що текст інтервалу збігається з URL-адресою, що стосується базового посилання http: //. Однак Linkify розумний і перетворює номер телефону на зразок (212) 555-1212 в URL-адресу tel:2125551212. new URLSpanNoUnderlineВиклик повинен бути прийнятий , span.getURL()щоб зберегти цю інформацію, в іншому випадку ви створюєте погані посилання , які викликають виключення при натисканні. Я запропонував запропоноване рішення в черзі редагування вашої відповіді, оскільки сам не маю дозволів на редагування.
Mike Mueller

8
Не працює для мене. Це дає виняток класу класу в рядку Spannable s = (Spannable) textView.getText ();
Brijesh Thakur

5
Просто замініть цей рядок на Spannable s = new SpannableString (textView.getText ());
FOliveira

1
заміна на Spannable s = (Spannable) textView.getText()з Spannable s = new SpannableString(textView.getText());не працює для мене. Якщо я повернусь до кастингу, це спрацює, лише якщо TextView має android:textIsSelectable=true. Будь-яка ідея чому?
шок

2
Чи є спосіб зробити цю роботу з android:autoLinkатрибутом? В іншому випадку здається, що вам доведеться створити власні методи onclick для вбудованої функціональності.
Алкарін

32

Дано 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());

7
це не працює, якщо ви намагаєтесь отримати рядок із ресурсів, тому те, String content = getResources().getString(R.string.content);що містить посилання, більше не працює.
Rain Man

1
Як це може бути? Рядок не знає, звідки він походить. У змісті рядка має бути різниця.
Адріан Костер

не знаю, чому, але він не відображає URL-адресу, коли я викликаю рядок, з res/values/strings.xmlякого міститься саме те саме, що і приклад. Чи можете ви протестувати у своєму кінці?
Rain Man

Я точно не збираюся тестувати в своєму кінці (-:. Якщо ви хочете бути впевненими у вмісті рядка, увійдіть в журнал обох рядків разом з їх hashCode. Буде здивовано, якщо ви знайдете різні результати при використанні коду вище з точно такі ж струни.
Адріан Костер

1
Якщо ви хочете, щоб це працювало, отримуючи рядок із ресурсів, вам потрібно його закодувати. >= &gt;, <= &lt;тощо. Наприклад: <string name="link_to_google" >&lt;a href="https://www.google.com/"&gt;Google&lt;/a&gt;</string>Див. developer.android.com/guide/topics/resources/string-resource
Мікер

11

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

3

Я застосував рішення, яке, на мій погляд, є більш елегантним. Я зробив звичай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);
    }
}

3

Ось функція розширення 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()    

2

Щоразу, намагайтеся видалити підкреслення URL-адреси допомогою прокручуваного. Я пропоную лише такі речі:

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 ... !!!


Це найкраще рішення і єдине, що спрацювало для мене
GuilhE

Це рішення для тексту з однією URL-адресою.
CoolMind

1

Якщо ви використовуєте властивість Textview autolink і хочете видалити підкреслення, ви можете використовувати це:

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

public class NoUnderlineSpan extends UnderlineSpan {

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

По-друге, створіть та примірник NoUnderlineSpan, створіть Spannable з тексту рядка та встановіть діапазон на 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


Приємне рішення, але після повернення з програми Телефон (після натискання номера телефону) воно знову підкреслює TextView.
CoolMind

1
Так, якщо ваша активність переходить у фоновий режим, ви повинні використовувати цей код у своїй функції активності onResume, щоб підтримувати текстовий режим без стану підкреслення.
Francisco Moya

0

Якщо ви просто хочете текст і не турбуєтесь про посилання за 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://stackoverflow.com'>Stack Overflow</a> Rocks!";
textView.setText(stripLinks(content));

0

Ось Мій метод

 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 - це мій клас додатків, куди я поклав цей метод, щоб отримати доступ до нього з будь-якого місця


0

Для користувачів Xamarin, які знайшли цю публікацію, ось як я змусив її працювати:

  1. Створіть власний клас інтервалу для обробки підкреслення.
    class URLSpanNoUnderline : URLSpan {
            public URLSpanNoUnderline (string url) : base (url) {
            }

            public override void UpdateDrawState (TextPaint ds) {
                base.UpdateDrawState (ds);
                ds.UnderlineText = false;
            }
        }
  1. Створіть метод розширення, щоб знайти всі інтервали URL-адрес і замінити їх на наші власні інтервали.
public static void StripUnderlinesFromLinks (this TextView textView) {
                var spannable = new SpannableStringBuilder (textView.TextFormatted);
                var spans = spannable.GetSpans (0, spannable.Length (), Java.Lang.Class.FromType (typeof (URLSpan)));
                foreach (URLSpan span in spans) {
                    var start = spannable.GetSpanStart (span);
                    var end = spannable.GetSpanEnd (span);
                    spannable.RemoveSpan(span);
                    var newSpan = new URLSpanNoUnderline (span.URL);
                    spannable.SetSpan(newSpan, start, end, 0);
                }
                textView.TextFormatted = spannable;
            }

-1
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);
            }
        }
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.