Android HTML ImageGetter как AsyncTask - PullRequest
       20

Android HTML ImageGetter как AsyncTask

32 голосов
/ 15 сентября 2011

Хорошо, я схожу с ума по этому поводу.У меня есть метод в моей программе, который анализирует HTML.Я хочу включить встроенные изображения, и у меня сложилось впечатление, что использование Html.fromHtml (string, Html.ImageGetter, Html.TagHandler) позволит этому произойти.

Поскольку Html.ImageGetter этого не делаетесть реализация, я должен написать.Однако, поскольку для разбора URL-адресов в Drawables требуется доступ к сети, я не могу сделать это в главном потоке, поэтому это должен быть AsyncTask.Я думаю.

Однако, когда вы передаете ImageGetter в качестве параметра Html.fromHtml, он использует метод getDrawable, который должен быть переопределен.Таким образом, нет способа вызвать всю сделку ImageGetter.execute, которая запускает метод doInBackground, и поэтому нет способа фактически сделать это асинхронным.

Я полностью ошибаюсь или, хуже того, это невозможно?Спасибо

Ответы [ 5 ]

100 голосов
/ 16 сентября 2011

Я сделал нечто очень похожее (я думаю) на то, что вы хотите сделать. Тогда мне нужно было разобрать HTML-код и установить его обратно в TextView, и мне также пришлось использовать Html.ImageGetter, и у меня возникла та же проблема при получении изображения в главном потоке.

Шаги, которые я сделал в основном:

  • Создайте свой собственный подкласс для Drawable для облегчения перерисовки, я назвал его URLDrawable
  • Возвращает URLDrawable in getDrawable метод Html.ImageGetter
  • После вызова onPostExecute я перерисовываю контейнер результата Spanned

Теперь код для URLDrawable выглядит следующим образом


public class URLDrawable extends BitmapDrawable {
    // the drawable that you need to set, you could set the initial drawing
    // with the loading image if you need to
    protected Drawable drawable;

    @Override
    public void draw(Canvas canvas) {
        // override the draw to facilitate refresh function later
        if(drawable != null) {
            drawable.draw(canvas);
        }
    }
}

Достаточно просто, я просто переопределяю draw, чтобы он выбрал Drawable, который я установил там после завершения AsyncTask.

Следующий класс представляет собой реализацию Html.ImageGetter и тот, который извлекает изображение из AsyncTask и обновляет изображение

public class URLImageParser implements ImageGetter {
    Context c;
    View container;

    /***
     * Construct the URLImageParser which will execute AsyncTask and refresh the container
     * @param t
     * @param c
     */
    public URLImageParser(View t, Context c) {
        this.c = c;
        this.container = t;
    }

    public Drawable getDrawable(String source) {
        URLDrawable urlDrawable = new URLDrawable();

        // get the actual source
        ImageGetterAsyncTask asyncTask = 
            new ImageGetterAsyncTask( urlDrawable);

        asyncTask.execute(source);

        // return reference to URLDrawable where I will change with actual image from
        // the src tag
        return urlDrawable;
    }

    public class ImageGetterAsyncTask extends AsyncTask<String, Void, Drawable>  {
        URLDrawable urlDrawable;

        public ImageGetterAsyncTask(URLDrawable d) {
            this.urlDrawable = d;
        }

        @Override
        protected Drawable doInBackground(String... params) {
            String source = params[0];
            return fetchDrawable(source);
        }

        @Override
        protected void onPostExecute(Drawable result) {
            // set the correct bound according to the result from HTTP call
            urlDrawable.setBounds(0, 0, 0 + result.getIntrinsicWidth(), 0 
                    + result.getIntrinsicHeight()); 

            // change the reference of the current drawable to the result
            // from the HTTP call
            urlDrawable.drawable = result;

            // redraw the image by invalidating the container
            URLImageParser.this.container.invalidate();
        }

        /***
         * Get the Drawable from URL
         * @param urlString
         * @return
         */
        public Drawable fetchDrawable(String urlString) {
            try {
                InputStream is = fetch(urlString);
                Drawable drawable = Drawable.createFromStream(is, "src");
                drawable.setBounds(0, 0, 0 + drawable.getIntrinsicWidth(), 0 
                        + drawable.getIntrinsicHeight()); 
                return drawable;
            } catch (Exception e) {
                return null;
            } 
        }

        private InputStream fetch(String urlString) throws MalformedURLException, IOException {
            DefaultHttpClient httpClient = new DefaultHttpClient();
            HttpGet request = new HttpGet(urlString);
            HttpResponse response = httpClient.execute(request);
            return response.getEntity().getContent();
        }
    }
}

Наконец, ниже приведен пример программы, демонстрирующей, как все работает:

String html = "Hello " +
"<img src='http://www.gravatar.com/avatar/" + 
"f9dd8b16d54f483f22c0b7a7e3d840f9?s=32&d=identicon&r=PG'/>" +
" This is a test " +
"<img src='http://www.gravatar.com/avatar/a9317e7f0a78bb10a980cadd9dd035c9?s=32&d=identicon&r=PG'/>";

this.textView = (TextView)this.findViewById(R.id.textview);
URLImageParser p = new URLImageParser(textView, this);
Spanned htmlSpan = Html.fromHtml(html, p, null);
textView.setText(htmlSpan);
5 голосов
/ 02 мая 2016

Довольно мило.Однако тип DefaultHttpClient устарел.Попробуйте этот метод извлечения:

private InputStream fetch(String urlString) throws MalformedURLException, IOException {

        URL url = new URL(urlString);
        HttpURLConnection urlConnection = (HttpURLConnection) url.openConnection();
        InputStream stream = urlConnection.getInputStream();

        return stream;

    }
1 голос
/ 10 августа 2016

если вы используете Picasso, измените часть кода @momo на

/***
         * Get the Drawable from URL
         * @param urlString
         * @return
         */
        public Drawable fetchDrawable(String urlString) {
            try {
                Drawable drawable = fetch(urlString);
                drawable.setBounds(0, 0, 0 + drawable.getIntrinsicWidth(), 0
                        + drawable.getIntrinsicHeight());
                return drawable;
            } catch (Exception e) {
                return null;
            }
        }

        private Drawable fetch(String urlString) throws MalformedURLException, IOException {
            return new BitmapDrawable(c.getResources(), Picasso.with(c).load(urlString).get());
        }
1 голос
/ 15 сентября 2011

Я немного запутался, HTML-код, который вы хотите сделать статичным, просто для форматирования, или он динамический и поступает из Интернета? Если вы хотите последнее, то есть визуализировать HTML-код и получить изображения, то это будет немного больно (совет - просто используйте WebView?).

В любом случае, вам сначала нужно запустить AsyncTask для получения исходного HTML. Затем вы передадите эти результаты в Html.fromHtml() с пользовательской реализацией для класса Html.ImageGetter. Затем в этой реализации вам нужно будет запустить отдельный AsyncTask для получения каждого из изображений (вы, вероятно, захотите реализовать некоторое кэширование).

Однако, прочитав документацию (и я думаю, что я видел несколько примеров), мне показалось, что это не то, для чего они подразумевали Html.ImageGetter. Я думаю, что он предназначен для жестко закодированного HTML со ссылками на внутренние объекты рисования, но это только мое мнение.

0 голосов
/ 04 февраля 2016

AsyncTask task = new AsyncTask () {

                @Override
                protected String doInBackground(Integer... params) {
                    span = Html.fromHtml(noticeList.get(0)
                            .getContent(), imgGetter, null);
                    return null;
                }
                @Override
                protected void onPostExecute(String result) {
                    super.onPostExecute(result);
                    text.setMovementMethod(ScrollingMovementMethod
                            .getInstance());
                    if(span != null){
                        text.setText(span);
                    }
                }
            };

            task.execute();
...