我正在尝试在视图中包含不同的布局,具体取决于父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:在布局编辑器的设计视图中,一切都按预期工作。如果我切换主题,则包含会正确更新 请参阅以下屏幕截图:
答案 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中修复。但风箱,异常将被抛出。