CoordinatorLayout 和 AppBarLayout 高度
CoordinatorLayout and AppBarLayout elevation
我创建了这样的 AppBar 布局
<android.support.design.widget.AppBarLayout
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/appbar_layout"
android:layout_height="@dimen/app_bar_height"
android:layout_width="match_parent"
android:fitsSystemWindows="true"
android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
app:elevation="20dp">
<android.support.design.widget.CollapsingToolbarLayout...>
</android.support.design.widget.AppBarLayout>
有效并在 LinearLayout 中投射阴影:
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<include layout="@layout/app_bar_large" />
</LinearLayout>
然而当我把它放入 CoordinatorLayout 阴影就不见了:
<android.support.design.widget.CoordinatorLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<include layout="@layout/app_bar_large" />
</android.support.design.widget.CoordinatorLayout>
如何让appbar再次显示阴影?
这实际上是 CollapsingToolbarLayout
的实现细节,如 source code:
中所示
if (Math.abs(verticalOffset) == scrollRange) {
// If we have some pinned children, and we're offset to only show those views,
// we want to be elevate
ViewCompat.setElevation(layout, layout.getTargetElevation());
} else {
// Otherwise, we're inline with the content
ViewCompat.setElevation(layout, 0f);
}
当 CollapsingToolbarLayout
显示非固定元素时移除高度 - 默认情况下,它只有在只有固定子项可见时才会有高度。
原因如上,试试这个解决:
appBarLayout.addOnOffsetChangedListener(new AppBarLayout.OnOffsetChangedListener() {
@Override
public void onOffsetChanged(AppBarLayout appBarLayout, int verticalOffset) {
//some other code here
ViewCompat.setElevation(appBarLayout, The Elevation In Px);
}
});
解决方案是使用 app:elevation=0dp
删除默认高度并将 android:translationZ
设置为您想要的高度。
Note : The code below uses the latest AndroidX / Material libraries and might not work if you're using the old support library
<com.google.android.material.appbar.AppBarLayout
android:layout_height="wrap_content"
android:layout_width="match_parent"
android:translationZ="8dp"
app:elevation="0dp">
<!--
* `app:elevation=0dp` disables the default shadow that is automatically added on
scroll ; other values e.g. `6dp` are ignored despite what the official doc says
(see below)
* so instead we're using `android:translationZ` to add a shadow with a custom
elevation
-->
AppBarLayout # setTargetElevation() 的文档指出您可以使用 app:elevation
属性设置自定义海拔值,但对于大于 0dp
的值,它对我不起作用,所以我正在使用 translationZ
作为解决方法。
setTargetElevation()
现已弃用 AppBarLayout。
根据布局状态将自定义提升应用到 AppBarLayout 的新正确实现是使用 StateListAnimator。
material-components 使用这个,你可以看到 here
我在 this gist 中添加了始终显示 AppBarLayout 高度的示例实现。
您只需要 1. 在 /res/animator
下创建自定义状态列表动画器和 2. 像这样设置 AppBarLayout 的 StateListAnimator:
appBarLayout.stateListAnimator = AnimatorInflater.loadStateListAnimator(context, R.animator.appbar_always_elevated_state_list_animator)
我创建了这样的 AppBar 布局
<android.support.design.widget.AppBarLayout
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:android="http://schemas.android.com/apk/res/android"
android:id="@+id/appbar_layout"
android:layout_height="@dimen/app_bar_height"
android:layout_width="match_parent"
android:fitsSystemWindows="true"
android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
app:elevation="20dp">
<android.support.design.widget.CollapsingToolbarLayout...>
</android.support.design.widget.AppBarLayout>
有效并在 LinearLayout 中投射阴影:
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<include layout="@layout/app_bar_large" />
</LinearLayout>
然而当我把它放入 CoordinatorLayout 阴影就不见了:
<android.support.design.widget.CoordinatorLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<include layout="@layout/app_bar_large" />
</android.support.design.widget.CoordinatorLayout>
如何让appbar再次显示阴影?
这实际上是 CollapsingToolbarLayout
的实现细节,如 source code:
if (Math.abs(verticalOffset) == scrollRange) {
// If we have some pinned children, and we're offset to only show those views,
// we want to be elevate
ViewCompat.setElevation(layout, layout.getTargetElevation());
} else {
// Otherwise, we're inline with the content
ViewCompat.setElevation(layout, 0f);
}
当 CollapsingToolbarLayout
显示非固定元素时移除高度 - 默认情况下,它只有在只有固定子项可见时才会有高度。
原因如上,试试这个解决:
appBarLayout.addOnOffsetChangedListener(new AppBarLayout.OnOffsetChangedListener() {
@Override
public void onOffsetChanged(AppBarLayout appBarLayout, int verticalOffset) {
//some other code here
ViewCompat.setElevation(appBarLayout, The Elevation In Px);
}
});
解决方案是使用 app:elevation=0dp
删除默认高度并将 android:translationZ
设置为您想要的高度。
Note : The code below uses the latest AndroidX / Material libraries and might not work if you're using the old support library
<com.google.android.material.appbar.AppBarLayout
android:layout_height="wrap_content"
android:layout_width="match_parent"
android:translationZ="8dp"
app:elevation="0dp">
<!--
* `app:elevation=0dp` disables the default shadow that is automatically added on
scroll ; other values e.g. `6dp` are ignored despite what the official doc says
(see below)
* so instead we're using `android:translationZ` to add a shadow with a custom
elevation
-->
AppBarLayout # setTargetElevation() 的文档指出您可以使用 app:elevation
属性设置自定义海拔值,但对于大于 0dp
的值,它对我不起作用,所以我正在使用 translationZ
作为解决方法。
setTargetElevation()
现已弃用 AppBarLayout。
根据布局状态将自定义提升应用到 AppBarLayout 的新正确实现是使用 StateListAnimator。 material-components 使用这个,你可以看到 here
我在 this gist 中添加了始终显示 AppBarLayout 高度的示例实现。
您只需要 1. 在 /res/animator
下创建自定义状态列表动画器和 2. 像这样设置 AppBarLayout 的 StateListAnimator:
appBarLayout.stateListAnimator = AnimatorInflater.loadStateListAnimator(context, R.animator.appbar_always_elevated_state_list_animator)