Glide assert: java.lang.IllegalArgumentException: Вы должны вызвать этот метод в главном потоке

Кто-нибудь использовал Glide для извлечения изображений из фонового потока? Я продолжаю получать это утверждение:

java.lang.IllegalArgumentException: You must call this method on the main thread

но согласно этой теме, это должно работать:

https://github.com/bumptech/glide/issues/310

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

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

    Glide.get(mContext);
    loadUserImage(userImageUrl);

    // wait 5 seconds before trying again
    int imageLoadingTimeOut = mContext.getResources().getInteger(R.integer.image_loading_time_out);
    if (imageLoadingTimeOut > 0) {
        new Timer().schedule(new TimerTask() {
            @Override
            public void run() {
                if (!mUserImageLoaded) {
                    loadUserImage(userImageUrl);
                }
            }
        }, imageLoadingTimeOut);
    }
}

и loadUserImage:

private boolean mUserImageLoaded = false;

private void loadUserImage(String userImageUrl) {

    if (userImageUrl != null && !userImageUrl.isEmpty() && !mUserImageLoaded) {
        Glide.with(mContext).using(Cloudinary.getUrlLoader(mContext)).load(userImageUrl).crossFade().listener(new RequestListener<String, GlideDrawable>() {

            @Override
            public boolean onException(Exception e, String model, Target<GlideDrawable> target, boolean isFirstResource) {
                return false;
            }

            @Override
            public boolean onResourceReady(GlideDrawable resource, String model, Target<GlideDrawable> target, boolean isFromMemoryCache, boolean isFirstResource) {
                mImageMessageContent.invalidate();
                mUserImageLoaded = true;
                return false;
            }
        }).into(mImageMessageContent);

    } else {
        mImageMessageContent.setVisibility(View.GONE);
    }
}

а mContext - это просто указатель активности this.

В любом случае, могу ли я использовать Glide из потока, отличного от основного?

3 ответа

Решение

into(ImageView) метод Glide требует, чтобы вы вызывали его только в главном потоке, но когда вы передадите загрузку в Timer, он будет выполнен в background нить.

Что вы можете сделать, это получить растровое изображение, вызвав get() вместо into() а затем установить, что bitmap на ImageView позвонив setImageBitmap(),

Glide.with(getApplicationContext())
     .load("your url")
     .asBitmap()
     .into(new BitmapImageViewTarget(imgView) {
      @Override
      protected void setResource(Bitmap resource) {
       //Play with bitmap
        super.setResource(resource);
      }
    });

Вы также можете взглянуть на этот документ для получения дополнительной информации.

Обновить изображение в главной теме пользовательского интерфейса

runOnUiThread(new Runnable() {
                        @Override
                        public void run() {
                            Glide.with(MainActivity.this)
                                    .load("image URL")
                                    .into(imageView);
                        }
                    });

Размещение кода на тот случай, если это кому-нибудь поможет.

Bitmap myBitmap = Glide.with(applicationContext)
        .load(yourUrl)
        .asBitmap()
        .centerCrop()
        .into(Target.SIZE_ORIGINAL,Target.SIZE_ORIGINAL)
        .get()
imageView.setImageBitmap(myBitmap);

Вот решение Kotlin-way

Glide.with(context).asBitmap().load(photo_url).into(
      BitmapImageViewTarget(imgYourResourceID)
)

В моем случае я хочу показать уведомление от FirebaseMessagingService с изображением, которое будет загружено через GlideТо, что дало мне успех, это кусок кода:

try {
                    Bitmap bitmap= Glide.with(getApplicationContext())
                            .load(imageUrl)   // image url in string
                            .asBitmap().into(Target.SIZE_ORIGINAL, Target.SIZE_ORIGINAL).get();

                    // now i can pass bitmap to notificationBuilder like
                 notificationBuilder.setStyle(new NotificationCompat.BigPictureStyle().bigPicture(bitmap)); 

                } catch (InterruptedException e) {
                    e.printStackTrace();
                } catch (ExecutionException e) {
                    e.printStackTrace();
                }

Мне нужно расшифровать и отобразить большие изображения в recyclerView И поскольку это был тяжелый процесс, мне пришлось делать это в фоновом режиме, поэтому вот почему:

 public void decryptImage(Uri uri, ViewHolder holder) {

    Runnable runnable = new Runnable() {
        @Override
        public void run() {
            try {

                Glide.with(G.context)
                        .asBitmap()
                        .load(decrypt(G.getEncryptionKey(), G.getConstantKey(), uri.getPath()))
                        .into(new SimpleTarget<Bitmap>() {
                            @Override
                            public void onResourceReady(Bitmap resource, Transition<? super Bitmap> transition) {
                                Glide.with(G.context).load(resource).into(holder.mImageView);
                            }

                        });
    
        }
    };
    new Thread(runnable).start();


}
 private class AsyncTaskRunner extends AsyncTask<String, String, RequestBuilder>
{
    ProgressDialog progressDialog;
    @Override
    protected void onPreExecute() {
        progressDialog = ProgressDialog.show(MainActivity.this,
                "Please Wait",
                "Image is loading...");
    }

    @Override
    protected RequestBuilder<Drawable> doInBackground(String... strings) {
        URL url = null;
        try {
            url = new URL(strings[0]);
        } catch (MalformedURLException e) {
            e.printStackTrace();
        }
        RequestBuilder<Drawable> g= Glide.with(MainActivity.this).load(url);
        return g;
    }

    @Override
    protected void onPostExecute(RequestBuilder v) {
        v.into(imageView);
        imageView.setVisibility(View.VISIBLE);
        progressDialog.dismiss();
    }
}
Другие вопросы по тегам