Android 5.0以下XML定义的drawable不识别?attr/属性的解决思路

最近写个小项目,之前一直使用5.0 和7.0系统的手机测试正常,换到4.4系统却报Resources$NotFoundException错误,很是疑惑,因为错误指向的drawable文件正常啊,把log复制到搜索引擎查了一下,也没什么收获,但查到有个类似的问题,是:TextView.setText()时,直接给了一个int型的参数,导致系统把这个参数当做Resource id去处理了。 根据这个思路,我打开我的drawable文件,试着把里面的"?attr/colorPrimary"换成“@color/颜色资源”,结果没有报错。网上查了一下,果然是5.0以下,在drawable中无法正确引用?attr/的值(我的想法是:系统本应该取到颜色值,可却把这个值做resource id处理了,然后根据id去寻找资源时,没有找到资源)。

可因为我在软件中加入了主题更换的功能,所以要获取实时的?attr/colorPrimary,而不是固定的color资源,于是想了些解决办法:

 1、新建drawable-21 文件夹,对于5.0和5.0 以下系统的drawable资源分别定义,这样可以解决,但是分别定义的drawable因为使用的资源的不同可能导致效果不同。

2、逻辑上避开在drawable中引用?attr/资源。比如我在某一个按钮上的效果:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <Button
        android:layout_width="96dp"
        android:layout_height="40dp"
        android:layout_centerInParent="true"
        android:background="@drawable/btn_pressed"
        android:text="按钮"
        android:textColor="#FFFFFF"/>

</RelativeLayout>

<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:state_pressed="false">
        <shape android:shape="rectangle">
            <solid android:color="?colorPrimary"/>
        </shape>
    </item>
    <item android:state_pressed="true">
        <shape android:shape="rectangle">
            <solid android:color="?colorPrimaryDark"/>
        </shape>
    </item>
</selector>
很简单,就不上图了,96*48的按钮,触摸时和不触摸时引用不同的颜色。

可在5.0 以下系统,肯定会报Resources$NotFoundException,怎么解决呢?

我们可以在这个按钮下面放一个同样大小的控件,任何设置该控件的background为?colorPrimary。然后在按钮的drawable文件中,将不触摸时的颜色设为全透明,触摸时设为一定透明度的黑色。如下:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <FrameLayout
        android:layout_width="96dp"
        android:layout_height="40dp"
        android:layout_centerInParent="true"
        android:background="?colorPrimary">
        <Button
            android:layout_width="96dp"
            android:layout_height="40dp"
            android:background="@drawable/btn_pressed"
            android:text="按钮"
            android:textColor="#FFFFFF"/>
    </FrameLayout>
</RelativeLayout>
<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:state_pressed="false">
        <shape android:shape="rectangle">
            <!--全透明色-->
            <solid android:color="#00000000"/>
        </shape>
    </item>
    <item android:state_pressed="true">
        <shape android:shape="rectangle">
            <!--透明度为12的黑色-->
            <solid android:color="#1E000000"/>
        </shape>
    </item>
</selector>
两次的效果是差不多的,通过调整透明色,可以达到一样的效果。这样,我们避开了在drawable中引用?attr的资源,而且到达预期的效果了。
3、在java代码中定义drawable。这也是最好的一种解决办法,我们可以在代码中获取到?attr/资源的值,然后在定义的drawable中引用就行了,

public class TestActivity extends AppCompatActivity{
    @RequiresApi(api = Build.VERSION_CODES.JELLY_BEAN)
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_test);
        Utils.context=this;
        Button button=(Button)findViewById(R.id.btn);
        button.setBackground(getStateListDrawable());
    }
//    对应drawable 中的selector
    private StateListDrawable getStateListDrawable(){
        StateListDrawable stateListDrawable=new StateListDrawable();
        stateListDrawable.addState(new int[]{-android.R.attr.state_pressed},getGradientDrawable(false));
        stateListDrawable.addState(new int[]{android.R.attr.state_pressed},getGradientDrawable(true));
        return stateListDrawable;
    }

//    对应drawable 中的 shape
    private GradientDrawable getGradientDrawable(boolean isPressed){
        GradientDrawable gradientDrawable=new GradientDrawable();
        gradientDrawable.setShape(GradientDrawable.RECTANGLE);
        // 获取颜色
        TypedValue primaryValue=new TypedValue();
        TypedValue primaryDarkValue=new TypedValue();
        this.getTheme().resolveAttribute(R.attr.colorPrimary,primaryValue,true);
        this.getTheme().resolveAttribute(R.attr.colorPrimaryDark,primaryDarkValue,true);
//        背景颜色
        if(isPressed){
            gradientDrawable.setColor(primaryDarkValue.data);
        } else {
            gradientDrawable.setColor(primaryValue.data);
        }
        gradientDrawable.setBounds(0,0,SizeUtils.dp2px(96),SizeUtils.dp2px(48));
        return gradientDrawable;
    }
}
代码很简单,可以对应在xml中定义drawable的方式来看,实现的效果和之前的一样。关于上面用到的
StateListDrawable  
GradientDrawable

 

两个类,大家可以去详细学习一下,我这里知识提一下解决思路,如果你有更好的方法,可以在下面评论告诉我。
还是上传个图片吧,上面实现的效果:



评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值