将<include>标记与?attr / myAttr一起使用

时间:2015-05-15 12:21:32

标签: android android-layout android-styles android-attributes

我正在尝试在视图中包含不同的布局,具体取决于父Theme

遵循这个想法:

attrs.xml

<attr name="themeLayout" format="reference" />

styles.xml

<style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
    <item name="themeLayout">@layout/layout_a</item>
</style>

<style name="AppThemeSecond" parent="Theme.AppCompat.Light.DarkActionBar">
    <item name="themeLayout">@layout/layout_b</item>
</style>

activity.xml

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <include layout="?attr/themeLayout" />

</RelativeLayout>

当我运行上面的代码时,我会得到以下例外:

java.lang.RuntimeException: Unable to start activity ComponentInfo{com.my.package/com.my.package.MainActivity}: android.view.InflateException: You must specifiy a valid layout reference. The layout ID ?attr/themeLayout is not valid.
            at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2325)
            at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2387)
            at android.app.ActivityThread.access$800(ActivityThread.java:151)
            at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1303)
            at android.os.Handler.dispatchMessage(Handler.java:102)
            at android.os.Looper.loop(Looper.java:135)
            at android.app.ActivityThread.main(ActivityThread.java:5254)
            at java.lang.reflect.Method.invoke(Native Method)
            at java.lang.reflect.Method.invoke(Method.java:372)
            at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:903)
            at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:698)
     Caused by: android.view.InflateException: You must specifiy a valid layout reference. The layout ID ?attr/themeLayout is not valid.
            at android.view.LayoutInflater.parseInclude(LayoutInflater.java:866)

我做错了什么?是否可以这样做?

注意#1:我将Theme的{​​{1}}设置为MainActivity

注意#2:在布局编辑器的设计视图中,一切都按预期工作。如果我切换主题,则包含会正确更新 请参阅以下屏幕截图:

preview design tab android studio

3 个答案:

答案 0 :(得分:4)

不幸的是,这是不可能的,但我真的很喜欢这个想法。我跟踪了LayoutInflater的流量,它将layout属性设为TypedValue.TYPE_REFERENCE,这意味着?attr不被允许。他们甚至在方法中留下了评论来解释。

public int getAttributeResourceValue(int idx, int defaultValue) {
    int t = nativeGetAttributeDataType(mParseState, idx);
    // Note: don't attempt to convert any other types, because
    // we want to count on aapt doing the conversion for us.
    if (t == TypedValue.TYPE_REFERENCE) {
        return nativeGetAttributeData(mParseState, idx);
    }
    return defaultValue;
}
     

android.content.res.XmlBlock.java:385

基本上你没有做错任何事 - 预览中的通货膨胀工作方式不同导致了混乱。

答案 1 :(得分:4)

由于Lamorak在他的"in-depth" answer中详细解释了所有内容,我想除了说没有针对您的问题的<include />标记的任何解决方案。 因此,遵循在我的测试项目中使用的不同方法:

#1 - 将 <include /> 替换为 <ViewStub />

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".MainActivity">

    <ViewStub
        android:id="@+id/myStub"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:inflatedId="@+id/myInflatedStub"
        android:layout="?attr/themeLayout" />

</RelativeLayout>

#2 - 给ViewStub充气

ViewStub myStub = (ViewStub) findViewById(R.id.myStub);
myStub.inflate();

这种方法有什么缺点吗?那么你不会在Design-Tab中看到ViewStub-Layout,并且使用<merge />标签将不起作用。

答案 2 :(得分:1)

经过一些研究,我发现了这个问题 - &#34; 已在Android23中修复。但风箱,异常将被抛出。