上下文主题包装器不能强制转换为活动



在我的小应用程序中,我有一个PageAdapter->Fragment->ListView->ImageView。在onImageClick上,我想在另一个片段(片段3)中显示我的图像,但是当我在我的片段3中加载图像时,会出现异常

public class Fragment3 extends DialogFragment {
ImageLoader imageLoader;
...
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    view = (ViewGroup) inflater.inflate(R.layout.fragment3_layout, container, false);
    imageView = (ImageView) view.findViewById(R.id.imageViewFrag3);
    detail = (TextView) view.findViewById(R.id.textViewFrag3);
    imageUrl = getArguments().getString("URL");
    imageView.setTag(imageUrl); 
    activity=this;      
    imageLoader = new ImageLoader(activity.getActivity().getApplicationContext());      
    imageLoader.DisplayImage(imageUrl, (Activity) activity.getActivity(), imageView);       
    return view;
}
...
}

以及创建异常的代码。当我创建一个片段并添加 ListView 时,一切都通过了,但是当我创建另一个片段(子)并想要加载照片时,它失败了。

代码,我如何在没有页面适配器的情况下创建子片段

holder.imgViewImage = (ImageView) vi.findViewById(R.id.imageView01);
        holder.imgViewImage.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {                   
                android.support.v4.app.FragmentTransaction ft = activity.getFragmentManager().beginTransaction();                   
                Fragment3 newFragment = Fragment3.newInstance();                    
                Bundle bundle=new Bundle();
                bundle.putString("URL", holder.imgViewImage.getTag().toString());
                newFragment.setArguments(bundle);
                newFragment.show( ft, "dialog");
                Log.i("!!!", "Click on image");                 
            }
        })

这是照片加载线程,

class PhotosLoader extends Thread {
    public void run() {
        try {
            while(true)
            {
                if(photosQueue.photosToLoad.size()==0)
                    synchronized(photosQueue.photosToLoad){
                        photosQueue.photosToLoad.wait();
                    }
                if(photosQueue.photosToLoad.size()!=0)
                {
                    PhotoToLoad photoToLoad;
                    synchronized(photosQueue.photosToLoad){
                        photoToLoad=photosQueue.photosToLoad.pop();
                    }
                    Bitmap bmp=getBitmap(photoToLoad.url);
                    cache.put(photoToLoad.url, bmp);
                    if(((String)photoToLoad.imageView.getTag()).equals(photoToLoad.url)){
                        BitmapDisplayer bd=new BitmapDisplayer(bmp, photoToLoad.imageView);            
                        //Exception contextthemewrapper cannot be cast to activity !!!
                        Activity a= (Activity)(photoToLoad.imageView.getContext());
                        a.runOnUiThread(bd);
                    }
                }
                if(Thread.interrupted())
                    break;
            }
        } catch (InterruptedException e) {
        }
    }
}

和例外

02-09 08:22:57.510: E/AndroidRuntime(1449): FATAL EXCEPTION: Thread-114
02-09 08:22:57.510: E/AndroidRuntime(1449): java.lang.ClassCastException: android.view.ContextThemeWrapper cannot be cast to android.app.Activity
02-09 08:22:57.510: E/AndroidRuntime(1449):     at com.example.ids.ImageLoader$PhotosLoader.run(ImageLoader.java:171)

我也有类似的情况。可能是相关问题>从自定义按钮访问活动

用这个片段解决了我的情况。

private static Activity scanForActivity(Context cont) {
    if (cont == null)
        return null;
    else if (cont instanceof Activity)
        return (Activity)cont;
    else if (cont instanceof ContextWrapper)
        return scanForActivity(((ContextWrapper)cont).getBaseContext());
    return null;
}
Activity a= (Activity)(photoToLoad.imageView.getContext());

您不能假设Context可以强制转换为Activity。因此例外:您正在尝试投射一个实际上是ContextThemeWrapper ActivityContext

您可以替换

Activity a= (Activity)(photoToLoad.imageView.getContext());
a.runOnUiThread(bd);

例如

photoToLoad.imageView.post(bd);

Runnable发布到 UI 线程消息队列,类似于 Activity runOnUiThread()

我所知,如果一个视图显示在对话框中,它的上下文(从view.getContext()访问)实际上是一个ThemeContextWrapper实例,它可能不是一个活动。若要从对话框获取活动,可以使用 getOwnerActivity(),它返回对话框所属的活动。

除了

kikea 的答案之外的 Kotlin 解决方案。

fun Context.scanForActivity(): AppCompatActivity? {
   return when (this) {
       is AppCompatActivity -> this
       is ContextWrapper -> baseContext.scanForActivity()
       else -> null
    }
}

最新更新