引言
最近,在做公司一個design折疊效果的時候遇到個問題朴译,就是我們本身app的方法數(shù)太多了井佑,dex分包技術(shù)還沒搞定。不得不盡量縮減一些不必要的包眠寿、類躬翁。當(dāng)我們引入RecyclerView的時候,恰好是壓死駱駝的最后一根稻草盯拱,故不得不采用其他方案來代替RecyclerView 和 CollapsingToolbarLayout實現(xiàn)的折疊效果盒发。本文試著采用 NestedScrollView 嵌套 ListView的方法來實現(xiàn)折疊效果。具體結(jié)果如圖所示:
布局文件
如下所示:
<android.support.design.widget.CoordinatorLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent">
<android.support.design.widget.AppBarLayout
android:layout_width="match_parent"
android:layout_height="270dp"
android:fitsSystemWindows="true">
<android.support.design.widget.CollapsingToolbarLayout
android:id="@+id/collapsing_toolbar_layout"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="#F03843"
app:contentScrim="#F03843"
app:layout_scrollFlags="scroll|exitUntilCollapsed">
<TextView
android:layout_width="match_parent"
android:layout_height="match_parent"
app:layout_collapseMode="parallax"
app:layout_collapseParallaxMultiplier="0.7"/>
<android.support.v7.widget.Toolbar
android:id="@+id/toolbar"
android:layout_width="match_parent"
android:layout_height="?attr/actionBarSize"
android:minHeight="?attr/actionBarSize"
app:contentInsetEnd="0dp"
app:contentInsetStart="0dp"
app:layout_collapseMode="pin">
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="match_parent">
<ImageButton
android:id="@+id/toolbar_import_navigation"
android:layout_width="?attr/actionBarSize"
android:layout_height="?attr/actionBarSize"
android:layout_alignParentLeft="true"
android:layout_alignParentStart="true"
android:layout_centerVertical="true"
android:src="@drawable/icon_toolbar_navigation"/>
<TextView
android:id="@+id/toolbar_title"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:gravity="center"
android:lineSpacingMultiplier="1.5"
android:text="@string/select_category"
android:textColor="@color/dark_gray"
android:textSize="@dimen/text_40px"/>
<TextView
android:id="@+id/toolbar_import"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentEnd="true"
android:layout_alignParentRight="true"
android:layout_centerVertical="true"
android:layout_marginRight="16dp"
android:lineSpacingMultiplier="1.5"
android:text="@string/vocabulary_import"
android:textColor="@color/dark_gray"
android:textSize="@dimen/text_32px"/>
</RelativeLayout>
</android.support.v7.widget.Toolbar>
<android.support.design.widget.TabLayout
android:id="@+id/tabs"
android:layout_width="match_parent"
android:layout_height="wrap_content"
app:tabGravity="fill" />
</android.support.design.widget.CollapsingToolbarLayout>
</android.support.design.widget.AppBarLayout>
<android.support.v4.widget.NestedScrollView
android:layout_width="match_parent"
android:layout_height="match_parent"
android:fillViewport="true"
android:layout_marginTop="2dp"
android:id="@+id/vocabulary_nested_scroll"
app:layout_behavior="@string/appbar_scrolling_view_behavior">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/parent_list_view"
android:orientation="vertical">
<NoScrollListView
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/list" />
</LinearLayout>
</android.support.v4.widget.NestedScrollView>
</android.support.design.widget.CoordinatorLayout>
上述布局需要注意的是:
- 必須采用CoordinatorLayout作為外層包裹狡逢,至于原因是由于使用Behavior需要宁舰。
- AppBarLayout做一個兼容。
- android:fitsSystemWindows奢浑,經(jīng)本人代碼測試蛮艰,表示CollapsingToolbarLayout的上邊界是否擴展到statusbar,這里如果想使用透明的statusbar殷费,這里更新下應(yīng)該該布局的根布局的該屬性為true印荔,不過這里兼容性還是存在問題的低葫,特別是android 4.4版本的折疊布局的透明statusbar,不知道改為有沒更好的方式仍律。
- app:contentScrim 表示CollapsingToolbarLayout折疊以后嘿悬,toolbar的顏色。
- app:layout_scrollFlags="scroll|exitUntilCollapsed" 表示CoordinatorLayout的依賴元素滾動的時候水泉,進行折疊善涨。
scroll - 想滾動就必須設(shè)置這個。 enterAlways - 實現(xiàn)quick return效果, 當(dāng)向下移動時草则,立即顯示View(比如Toolbar). exitUntilCollapsed - 向上滾動時收縮View钢拧,但可以固定Toolbar一直在上面。 enterAlwaysCollapsed - 當(dāng)你的View已經(jīng)設(shè)置minHeight屬性又使用此標志時炕横,你的View只能以最小高度進入源内,只有當(dāng)滾動視圖到達頂部時才擴大到完整高度。(參見: http://android.jobbole.com/82193/) - layout_collapseMode份殿,pin表示不動膜钓,parallax視差效果
- 將 app:layout_behavior="@string/appbar_scrolling_view_behavior"指定給NestedScrollView,即當(dāng)該控件滑動的時候卿嘲,其他CollapsingToolbarLayout內(nèi)的子view做相應(yīng)的改變?nèi)纾ㄒ暡睿┗騪in(不變)颂斜。
解決NestedScrollView嵌套listView問題。
貌似網(wǎng)上一抓一大把拾枣,本文采用其中之一方案沃疮,復(fù)寫listView。
public class NoScrollListView extends ListView {
public NoScrollListView(Context context) {
super(context);
}
public NoScrollListView(Context context, AttributeSet attrs) {
super(context, attrs);
}
public NoScrollListView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
@Override
public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST);
super.onMeasure(widthMeasureSpec, expandSpec);
}
}
解決NestedScrollView嵌套的listView滾動時無響應(yīng)bug梅肤。
private void adjustParentView() { //兼容NestedscrollView
int actionBarHeight = 0;
TypedValue tv = new TypedValue();
if (getTheme().resolveAttribute(android.R.attr.actionBarSize, tv, true)) {
actionBarHeight = TypedValue.complexToDimensionPixelSize(tv.data,getResources().getDisplayMetrics());
}
Rect outRect = new Rect(); //狀態(tài)欄高度
getWindow().getDecorView().getWindowVisibleDisplayFrame(outRect);
mParentView.setMinimumHeight(DeviceUtils.screenHeight() - actionBarHeight - outRect.top);
}
注:直接使用 RecyclerView更省事省力司蔬。
題外話:直接使用RecyclerView折疊動畫不平滑問題
在stackoverflow上找到如下解決方案,大致是由于google官方留的Behavior坑姨蝴。
public final class FlingBehavior extends AppBarLayout.Behavior {
private static final int TOP_CHILD_FLING_THRESHOLD = 3;
private boolean isPositive;
public FlingBehavior() {
}
public FlingBehavior(Context context, AttributeSet attrs) {
super(context, attrs);
}
@Override
public boolean onNestedFling(CoordinatorLayout coordinatorLayout, AppBarLayout child, View target, float velocityX, float velocityY, boolean consumed) {
if (velocityY > 0 && !isPositive || velocityY < 0 && isPositive) {
velocityY = velocityY * -1;
}
if (target instanceof RecyclerView && velocityY < 0) {
final RecyclerView recyclerView = (RecyclerView) target;
final View firstChild = recyclerView.getChildAt(0);
final int childAdapterPosition = recyclerView.getChildAdapterPosition(firstChild);
consumed = childAdapterPosition > TOP_CHILD_FLING_THRESHOLD;
}
return super.onNestedFling(coordinatorLayout, child, target, velocityX, velocityY, consumed);
}
@Override
public void onNestedPreScroll(CoordinatorLayout coordinatorLayout, AppBarLayout child, View target, int dx, int dy, int[] consumed) {
super.onNestedPreScroll(coordinatorLayout, child, target, dx, dy, consumed);
isPositive = dy > 0;
}
}
最后在AppBarLayout如下添加代碼:
<android.support.design.widget.AppBarLayout
android:layout_width="match_parent"
android:layout_height="210dp"
android:theme="@style/ThemeOverlay.AppCompat.Dark.ActionBar"
app:layout_behavior="com.youdao.vocabulary.widget.FlingBehavior">
題外話2
java.lang.IllegalStateException: View can not be anchored to the the parent CoordinatorLayout
當(dāng)我們給一個view設(shè)置app:layout_anchor時葱她,在最新的23.2.0包會出現(xiàn)問題,解決辦法是用android:layout_gravity="bottom|end" 替換似扔。
題外話3
23.2.0中在CoordinatorLayout中使用Toolbar 吨些,toolbar無法與頂部對齊,即頂部存在垂直間距炒辉。解決方法在AppBarLayout添加 android:fitsSystemWindows="true"