Html.fromHtml устарел в Android N

Я использую Html.fromHtml для просмотра html в TextView.

Spanned result = Html.fromHtml(mNews.getTitle());
...
...
mNewsTitle.setText(result);

Но Html.fromHtml теперь устарел в Android N +

Что/Как найти новый способ сделать это?

Ответ 1

обновление: как отметил @Andy, Google создал HtmlCompat который можно использовать вместо метода, HtmlCompat ниже. Добавьте эту implementation 'androidx.core:core:1.0.1 зависимости implementation 'androidx.core:core:1.0.1 в файл build.gradle вашего приложения. Убедитесь, что вы используете последнюю версию androidx.core:core.

Это позволяет вам использовать:

HtmlCompat.fromHtml(html, HtmlCompat.FROM_HTML_MODE_LEGACY);

Вы можете прочитать больше о различных флагах в HtmlCompat-документации.

оригинальный ответ: в Android N они представили новый метод Html.fromHtml. Html.fromHtml теперь требует дополнительный параметр с именем flags. Этот флаг дает вам больше контроля над тем, как отображается ваш HTML.

На Android N и выше вы должны использовать этот новый метод. Более старый метод устарел и может быть удален в будущих версиях Android.

Вы можете создать свой собственный Util-метод, который будет использовать старый метод в более старых версиях и новый метод в Android N и выше. Если вы не добавите версию, проверьте, что ваше приложение будет работать на более низких версиях Android. Вы можете использовать этот метод в своем классе Util.

@SuppressWarnings("deprecation")
public static Spanned fromHtml(String html){
    if(html == null){
        // return an empty spannable if the html is null
        return new SpannableString("");
    }else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
        // FROM_HTML_MODE_LEGACY is the behaviour that was used for versions below android N
        // we are using this flag to give a consistent behaviour
        return Html.fromHtml(html, Html.FROM_HTML_MODE_LEGACY);
    } else {
        return Html.fromHtml(html);
    }
}

Вы можете преобразовать HTML.FROM_HTML_MODE_LEGACY в дополнительный параметр, если хотите. Это дает вам больше контроля над тем, какой флаг использовать.

Вы можете прочитать больше о различных флагах в документации класса Html.

Ответ 2

У меня было много этих предупреждений, и я всегда использую FROM_HTML_MODE_LEGACY, поэтому я создал вспомогательный класс под названием HtmlCompat, содержащий следующее:

   @SuppressWarnings("deprecation")
   public static Spanned fromHtml(String source) {
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            return Html.fromHtml(source, Html.FROM_HTML_MODE_LEGACY);
        } else {
            return Html.fromHtml(source);
        }
    }

Ответ 3

Сравнение флагов fromHtml().

<p style="color: blue;">This is a paragraph with a style</p>

<h4>Heading H4</h4>

<ul>
   <li style="color: yellow;">
      <font color=\'#FF8000\'>li orange element</font>
   </li>
   <li>li #2 element</li>
</ul>

<blockquote>This is a blockquote</blockquote>

Text after blockquote
Text before div

<div>This is a div</div>

Text after div

FROM_HTML FLAGS

Ответ 4

Или вы можете использовать androidx.core.text.HtmlCompat:

HtmlCompat.fromHtml("<b>HTML</b>", HtmlCompat.FROM_HTML_MODE_LEGACY)

HtmlCompat документы

Ответ 5

Если вам посчастливилось развиваться на Котлине, просто создайте функцию расширения:

fun String.toSpanned(): Spanned {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
        return Html.fromHtml(this, Html.FROM_HTML_MODE_LEGACY)
    } else {
        @Suppress("DEPRECATION")
        return Html.fromHtml(this)
    }
}

И тогда это так приятно использовать его везде:

yourTextView.text = anyString.toSpanned()

Ответ 6

fromHtml

Этот метод устарел на уровне API 24.

Вы должны использовать FROM_HTML_MODE_LEGACY

Отдельные элементы уровня блока с пустыми строками (два символа новой строки) между ними. Это унаследованное поведение до N.

Код

if (Build.VERSION.SDK_INT >= 24)
        {
            etOBJ.setText(Html.fromHtml("Intellij \n Amiyo",Html.FROM_HTML_MODE_LEGACY));

         }
 else
        {
           etOBJ.setText(Html.fromHtml("Intellij \n Amiyo"));
        }

Для Котлина

fun setTextHTML(html: String): Spanned
    {
        val result: Spanned = if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.N) {
            Html.fromHtml(html, Html.FROM_HTML_MODE_LEGACY)
        } else {
            Html.fromHtml(html)
        }
        return result
    }

Вызов

 txt_OBJ.text  = setTextHTML("IIT Amiyo")

Ответ 7

Из официального документа:

fromHtml(String) метод устарел на уровне API 24. Используйте fromHtml(String, int)вместо этого.

  • TO_HTML_PARAGRAPH_LINES_CONSECUTIVE Опция для toHtml(Spanned, int): Оберните последовательные строки текста, ограниченные '\n' внутри <p>элементы.

  • TO_HTML_PARAGRAPH_LINES_INDIVIDUAL Опция для toHtml(Spanned, int): оберните каждую строку текста, ограниченную '\n' внутри <p> или <li>элемент.

https://developer.android.com/reference/android/text/Html.html

Ответ 8

Чтобы увеличить ответ от @Rockney и @k2col, улучшенный код может выглядеть так:

@NonNull
public static Spanned fromHtml(@NonNull String html) {
    if (CompatUtils.isApiNonLowerThan(VERSION_CODES.N)) {
        return Html.fromHtml(html, Html.FROM_HTML_MODE_LEGACY);
    } else {
        //noinspection deprecation
        return Html.fromHtml(html);
    }
}

Где CompatUtils.isApiNonLowerThan:

public static boolean isApiNonLowerThan(int versionCode) {
    return Build.VERSION.SDK_INT >= versionCode;
}

Разница в том, что нет дополнительной локальной переменной, а устаревание - только в ветки else. Таким образом, это не будет подавлять все методы, кроме одной ветки.

Это может помочь, когда Google решит в некоторых будущих версиях Android отказаться даже от метода fromHtml(String source, int flags).

Ответ 9

Вы можете использовать

//noinspection deprecation
return Html.fromHtml(source);

чтобы отключить проверку только для одного оператора, но не для всего метода.

Ответ 10

Если вы используете Kotlin, я достиг этого, используя расширение Kotlin:

fun TextView.htmlText(text: String){
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
        setText(Html.fromHtml(text, Html.FROM_HTML_MODE_LEGACY))
    } else {
        setText(Html.fromHtml(text))
    }
}

Затем назовите его так:

textView.htmlText(yourHtmlText)

Ответ 11

Класс framework был изменен, чтобы потребовать, чтобы флаг информировал fromHtml() о том, как обрабатывать разрывы строк. Это было добавлено в Nougat, и затрагивает только проблему несовместимости этого класса в версиях Android.

Я опубликовал библиотеку совместимости для стандартизации и резервного копирования класса и включения большего числа обратных вызовов для элементов и стилей:

https://github.com/Pixplicity/HtmlCompat

В то время как он похож на класс Html framework, некоторые изменения подписи потребовались, чтобы разрешить больше обратных вызовов. Здесь образец с страницы GitHub:

Spanned fromHtml = HtmlCompat.fromHtml(context, source, 0);
// You may want to provide an ImageGetter, TagHandler and SpanCallback:
//Spanned fromHtml = HtmlCompat.fromHtml(context, source, 0,
//        imageGetter, tagHandler, spanCallback);
textView.setMovementMethod(LinkMovementMethod.getInstance());
textView.setText(fromHtml);

Ответ 12

Вот мое решение.

 if (Build.VERSION.SDK_INT >= 24) {
        holder.notificationTitle.setText(Html.fromHtml(notificationSucces.getMessage(), Html.FROM_HTML_MODE_LEGACY));
    } else {
        holder.notificationTitle.setText(Html.fromHtml(notificationSucces.getMessage()));

    }

Ответ 13

Или просто используйте класс HtmlCompat

Ответ 14

Попробуйте следующее для поддержки основных тегов HTML, включая теги ul ol li. Создайте обработчик тега, как показано ниже

import org.xml.sax.XMLReader;

import android.app.Activity;
import android.os.Bundle;
import android.text.Editable;
import android.text.Html;
import android.text.Html.TagHandler;
import android.util.Log;

public class MyTagHandler implements TagHandler {
    boolean first= true;
    String parent=null;
    int index=1;
    @Override
    public void handleTag(boolean opening, String tag, Editable output,
                          XMLReader xmlReader) {

        if(tag.equals("ul")) parent="ul";
        else if(tag.equals("ol")) parent="ol";
        if(tag.equals("li")){
            if(parent.equals("ul")){
                if(first){
                    output.append("\n\t•");
                    first= false;
                }else{
                    first = true;
                }
            }
            else{
                if(first){
                    output.append("\n\t"+index+". ");
                    first= false;
                    index++;
                }else{
                    first = true;
                }
            }
        }
    }
}

Установите текст на активность, как показано ниже

@SuppressWarnings("deprecation")
    public void init(){
        try {
            TextView help = (TextView) findViewById(R.id.help);
            if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.N) {
                help.setText(Html.fromHtml(getString(R.string.help_html),Html.FROM_HTML_MODE_LEGACY, null, new MyTagHandler()));
            } else {
                help.setText(Html.fromHtml(getString(R.string.help_html), null, new MyTagHandler()));
            }
        } catch (Exception e) {
            e.printStackTrace();
        }

    }

И HTML-текст в файлах строк ресурсов как

<! [CDATA [... необработанные HTML-данные...]]>