Android 从 View 中获取 Activity 时遇到 TintContextWrapper cannot be cast to 的问题

问题描述:

如果一个 View 绘制于某个 Activity 的 ContentView 上, 那它的 Context 一定是和这个 Activity 相关联的. 因此我们想在 View 中直接用 Activity 方法时 (最常用的应该就是 Activity.startActivity() 方法了), 不必再向 View 中传递 Activity 对象.
一般在 View 中获取这个 Activity 对象都是简单的用下面代码就可以了:

Activity activity = (Activity) getContext();

但在 View 继承自 AppCompat 系的 View 时 (比如 AppCompatTextView, AppCompatImageView), 上面方法可能会得到下面异常:

**java.lang.ClassCastException: android.support.v7.widget.TintContextWrapper cannot be cast to ...Activity**

问题原因:

原因其实很简单. 上 AppCompatTextView 的构造函数代码:

public class AppCompatTextView extends TextView implements TintableBackgroundView {

    ......

    public AppCompatTextView(Context context) {
        this(context, null);
    }

    public AppCompatTextView(Context context, AttributeSet attrs) {
        this(context, attrs, android.R.attr.textViewStyle);
    }

    public AppCompatTextView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(TintContextWrapper.wrap(context), attrs, defStyleAttr);  // 注意这行

        mBackgroundTintHelper = new AppCompatBackgroundHelper(this);
        mBackgroundTintHelper.loadFromAttributes(attrs, defStyleAttr);

        mTextHelper = AppCompatTextHelper.create(this);
        mTextHelper.loadFromAttributes(attrs, defStyleAttr);
        mTextHelper.applyCompoundDrawablesTints();
    }

    ......
}

在构造函数中, context 被用 TintContextWrapper 包了一层. 所以这时 Activity 其实被保存在了 TintContextWrapper 中的 BaseContext 中了.
TintContextWrapper 不能直接强制转换为 Activity.

网上有说 23.3.0 的 v7 之后的包中会有这个问题, 之前的不会. 我手头只有 25.3.0 的源码, 没有验证其他版本的代码是什么样子. 不过这不重要, 因为下面的解决方案可以兼容各个情况.

问题解决:

知道原因就简单了. 可以简单的用 TintContextWrapper.getBaseContext() 得到这个 Activity.
但其实一层层的从 ContextWrapper 中把 Activity 剥出来更保险:

    /**
     * try get host activity from view.
     * views hosted on floating window like dialog and toast will sure return null.
     * @return host activity; or null if not available
     */
    public static Activity getActivityFromView(View view) {
        Context context = view.getContext();
        while (context instanceof ContextWrapper) {
            if (context instanceof Activity) {
                return (Activity) context;
            }
            context = ((ContextWrapper) context).getBaseContext();
        }
        return null;
    }

如上方法可以通用的解决从 View 中获取 Activity 的问题.

事实上, 谷歌 v7 包中的 android.support.v7.app.MediaRouteButton 就是这么干的.

参考:

Android get hosting Activity from a view


原创文章, 转载请注明出处: http://blog.csdn.net/liuxu0703/article/details/70145168

  • 4
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 4
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值