将ShapeDrawable设置为其图标时,不显示ActionBar项目

时间:2014-06-27 12:55:40

标签: android android-actionbar android-support-library shapedrawable

我正在开发一个简单的项目,我希望尽可能避免使用PNG图像。我需要一个' +' (添加)按钮,我使用带有给定代码的ShapeDrawable创建它。

RES /抽拉/ plus_icon_drawable.xml     

<layer-list xmlns:android="http://schemas.android.com/apk/res/android">
    <item>
        <shape android:shape="line">
            <stroke android:color="#000"
                android:width="3dp"/>
        </shape>
    </item>
    <item>
        <rotate
            android:fromDegrees="90"
            android:toDegrees="90">
            <shape android:shape="line">
                <stroke android:color="#000"
                    android:width="3dp"/>
            </shape>
        </rotate>
    </item>
</layer-list>

当我将它添加为Button或ImageButton的背景时,此代码可以正常工作,但当我将其添加为ActionBar项目的图标时,该项目不会显示。

当我将PNG图像设置为图标时,它可以正常工作。 ShapeDrawables是否有任何限制阻止它在ActionBar上呈现?

这是我的ActiobBar代码

RES /菜单/ action_my_actions.xml     

<menu xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:id="@+id/action_some_id"
        android:icon="@drawable/plus_icon_drawable"
        android:title="My Action"
        android:showAsAction="always"/>
</menu>

更新(2014年7月8日):使用Native和AppCompat进行测试。实际上该项目存在,但ShapeDrawable图像没有得到渲染。操作项响应点击次数。

我在这里缺少什么?

1 个答案:

答案 0 :(得分:3)

  

ShapeDrawables是否有任何限制阻止它在ActionBar上呈现?

没有那样的。事情就是如此。这就是为什么你会看到这个:

Layer/ShapeDrawables没有默认尺寸。换句话说,ShapeDrawable的宽度和高度均为零。 stroke-width参数也没有指示任何内容。

  

当我将它添加为Button或ImageButton的背景时,此代码可以正常工作,但当我将其添加为ActionBar项目的图标时,该项目不会显示。

是的,当然。 ImageButtons/Buttons具有填充/最小宽度/最小高度的概念。这些允许ShapeDrawable有一些呼吸的空间。例如,使用Widget.Holo.Button设置样式的Button会设置minWidthminHeight

<item name="android:minHeight">48dip</item>
<item name="android:minWidth">64dip</item>

这里的有效测试是:

按如下方式定义ImageView

<RelativeLayout
    android:layout_width="match_parent"
    android:layout_height="match_parent" >

    <ImageView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:src="@drawable/add" />

</RelativeLayout>

ImageView在上面的定义中没有规模调整指南 - 我们正在使用wrap_content。什么都不会显示。另一方面,如果你使用png drawable,它本身就定义了大小。在上面的示例中,使用显式大小(例如50dp)将使ShapeDrawable visible

这是ActionMenuItemView设置drawable的方式:

public void setIcon(Drawable icon) {
    mIcon = icon;
    if (icon != null) {

        // Zero in case you don't define the <size /> 
        int width = icon.getIntrinsicWidth();
        int height = icon.getIntrinsicHeight();
        if (width > mMaxIconSize) {
            final float scale = (float) mMaxIconSize / width;
            width = mMaxIconSize;
            height *= scale;
        }
        if (height > mMaxIconSize) {
            final float scale = (float) mMaxIconSize / height;
            height = mMaxIconSize;
            width *= scale;
        }
        icon.setBounds(0, 0, width, height);
    }
    setCompoundDrawables(icon, null, null, null);

    updateTextButtonVisibility();
}

ShapeDrawable#inflate(...)方法的摘录:

 setIntrinsicWidth((int)
            a.getDimension(com.android.internal.R.styleable.ShapeDrawable_width, 0f));
 setIntrinsicHeight((int)
            a.getDimension(com.android.internal.R.styleable.ShapeDrawable_height, 0f));

如果未定义ShapeDrawable_widthShapeDrawable_height,则固有宽度和高度将设置为零。这意味着,语句icon.setBounds(0, 0, width, height);(来自setIcon(Drawable)方法)实际上是=&gt; icon.setBounds(0, 0, 0, 0);

所以,我们似乎确实在处理那些没有尺码信息的ShapeDrawables

可能的解决办法:

<shape android:shape="line">
    <size android:width="?android:attr/actionBarSize"
          android:height="?android:attr/actionBarSize"/>
    <stroke android:color="#000"
            android:width="3dp"/>
</shape>

但这不会起作用 - 由于某种原因,?android:attr/actionBarSize不被视为一个维度。这很奇怪,因为?android:attr/actionBarSize在布局时提供给android:layout_width时工作正常。

解决方法是定义一个新维度,比如actionBarSizeDp并设置它:

res/values/dimens.xml

<dimen name="actionBarSizeDp">48dp</dimen>

res/values-land/dimens.xml

<dimen name="actionBarSizeDp">40dp</dimen>

这些值是android自己的res / values-XX / dimens.xml中定义的默认操作栏大小值。

最后,在actionBarSizeDp中使用plus_icon_drawable.xml

<layer-list xmlns:android="http://schemas.android.com/apk/res/android">
    <item>
        <shape android:shape="line">
            <size android:width="@dimen/actionBarSizeDp"
                  android:height="@dimen/actionBarSizeDp"/>
            <stroke android:color="#000"
                    android:width="3dp"/>
        </shape>
    </item>
    <item>
        <rotate
            android:fromDegrees="90"
            android:toDegrees="90">
            <size android:width="@dimen/actionBarSizeDp"
                  android:height="@dimen/actionBarSizeDp"/>
            <shape android:shape="line">
                <stroke android:color="#000"
                        android:width="3dp"/>
            </shape>
        </rotate>
    </item>
</layer-list>

事实上,在其中一个size上定义LayerDrawables就足够了。其他(s)将相应地绘制。