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

Я использую Html.fromHtml просмотреть HTML в TextView,

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

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

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

15 ответов

Решение

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

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

Параметры флага:

public static final int FROM_HTML_MODE_COMPACT = 63;
public static final int FROM_HTML_MODE_LEGACY = 0;
public static final int FROM_HTML_OPTION_USE_CSS_COLORS = 256;
public static final int FROM_HTML_SEPARATOR_LINE_BREAK_BLOCKQUOTE = 32;
public static final int FROM_HTML_SEPARATOR_LINE_BREAK_DIV = 16;
public static final int FROM_HTML_SEPARATOR_LINE_BREAK_HEADING = 2;
public static final int FROM_HTML_SEPARATOR_LINE_BREAK_LIST = 8;
public static final int FROM_HTML_SEPARATOR_LINE_BREAK_LIST_ITEM = 4;
public static final int FROM_HTML_SEPARATOR_LINE_BREAK_PARAGRAPH = 1;
public static final int TO_HTML_PARAGRAPH_LINES_CONSECUTIVE = 0;
public static final int TO_HTML_PARAGRAPH_LINES_INDIVIDUAL = 1;

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

У меня было много этих предупреждений, и я всегда использую 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);
        }
    }

Сравните флаги 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 ФЛАГИ

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

HtmlCompat.fromHtml(textWithHtmlTags, HtmlCompat.FROM_HTML_MODE_LEGACY)

HtmlCompat документы

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

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

fromHtml

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

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

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

Код

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

Если вы используете 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)

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

fromHtml(String) На уровне API 24 метод устарел. fromHtml(String, int) вместо.

  1. TO_HTML_PARAGRAPH_LINES_CONSECUTIVE Вариант для toHtml(Spanned, int): Перенос последовательных строк текста, разделенных '\n' внутри <p> элементы.

  2. TO_HTML_PARAGRAPH_LINES_INDIVIDUAL Вариант для toHtml(Spanned, int): Перенос каждой строки текста, разделенной '\n' внутри <p> или <li> элемент.

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

Для пользователей Kotlin мы можем использоватьString.parseAsHtml()функция расширения, которая использует HtmlCompat , которая, в свою очередь, имеет проверки совместимости.

Это доступно из расширений ядра Android kotlin, доступных изandroidx.core:core-ktx

Чтобы расширить ответ от @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) метод.

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

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

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

Просто сделайте функцию:

public Spanned fromHtml(String str){
  return Build.VERSION.SDK_INT >= 24 ? Html.fromHtml(str, Html.FROM_HTML_MODE_LEGACY) : Html.fromHtml(str);
}

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

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

    }

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

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

https://github.com/Pixplicity/HtmlCompat

Хотя он похож на класс Html фреймворка, потребовалось внести некоторые изменения в сигнатуру, чтобы разрешить больше обратных вызовов. Вот пример со страницы 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);

Попробуйте следующее для поддержки основных тегов 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-текст в файлах строк ресурсов как

Другие вопросы по тегам