(一)Flutter 实现 Android CollapsingToolbarLayout折叠布局结果

手机软件开发 2024-9-10 01:42:46 184 0 来自 中国
作为一名Flutter 浩繁码海 中的一名Android 转门生,最近开辟中遇到一个功能,要实现一个类似Android  CollapsingToolbarLayout 折叠布局的结果,在Android 开辟中我们通过 CoordinatorLayout + AppBarLayout +CollapsingToolbarLayout  来实现这个结果,但是在Flutter 中,则是通过 SliverAppBar  ,借助它的属性 flexibleSpace 来实现。
咱们可以先来看看Android 实当代码:

<?xml version="1.0" encoding="utf-8"?><androidx.coordinatorlayout.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"    tools:context=".chat.activitys.IntimacyActivity">    <com.google.android.material.appbar.AppBarLayout        android:id="@+id/app_bar_intimacy"        android:layout_width="match_parent"        android:layout_height="wrap_content">            <com.google.android.material.appbar.CollapsingToolbarLayout                android:layout_width="match_parent"                android:layout_height="wrap_content"                app:layout_scrollFlags="scroll|exitUntilCollapsed|snap"                app:statusBarScrim="@android:color/transparent">                <!-- 须要折叠的布局-->                <RelativeLayout                    android:layout_width="match_parent"                    android:layout_height="@dimen/dp_310">                    <ImageView                        android:layout_width="match_parent"                        android:layout_height="wrap_content"                        android:background="@drawable/intimacy_bg" />                    <TextView                        android:id="@+id/tv_familiar_count"                        android:layout_width="wrap_content"                        android:layout_height="wrap_content"                        android:layout_marginStart="@dimen/dp_26"                        android:layout_marginTop="@dimen/dp_74"                        android:background="@drawable/familiar_count_bg"                        android:gravity="center"                        android:text="相识天数\n1000天"                        android:textColor="@color/white"                        android:textSize="@dimen/sp_11"                        android:textStyle="bold" />                    <ImageView                        android:layout_width="wrap_content"                        android:layout_height="wrap_content"                        android:layout_alignTop="@id/iv_my_head"                        android:layout_marginStart="@dimen/dp_30"                        android:layout_marginTop="@dimen/dp_18"                        android:layout_toRightOf="@id/tv_familiar_count"                        android:background="@drawable/intimacy_lighting" />                    <com.donews.renrenplay.android.views.CircleImageView                        android:id="@+id/iv_my_head"                        android:layout_width="@dimen/dp_72"                        android:layout_height="@dimen/dp_72"                        android:layout_alignTop="@id/tv_familiar_count"                        android:layout_marginTop="@dimen/dp_31"                        android:layout_toRightOf="@id/tv_familiar_count"                        android:src="@drawable/intimacy_head_bg"                        app:borderColor="@color/white"                        app:borderWidth="@dimen/dp_1" />                    <com.donews.renrenplay.android.views.CircleImageView                        android:id="@+id/iv_other_head"                        android:layout_width="@dimen/dp_72"                        android:layout_height="@dimen/dp_72"                        android:layout_alignTop="@id/iv_my_head"                        android:layout_marginStart="@dimen/dp_60"                        android:layout_toRightOf="@id/iv_my_head"                        android:src="@drawable/intimacy_head_bg"                        app:borderColor="@color/white"                        app:borderWidth="@dimen/dp_1" />                    <RelativeLayout                        android:layout_width="match_parent"                        android:layout_height="@dimen/dp_51"                        android:layout_below="@id/iv_my_head"                        android:layout_marginStart="@dimen/dp_51"                        android:layout_marginTop="@dimen/dp_34"                        android:layout_marginEnd="@dimen/dp_51"                        android:layout_marginBottom="@dimen/dp_11"                      android:background="@drawable/shape_14_gradient_ffffff_f7e8ff">                        <TextView                            android:id="@+id/tv_intimacy_tip"                            android:layout_width="wrap_content"                            android:layout_height="wrap_content"                            android:layout_marginStart="@dimen/dp_14"                            android:layout_marginTop="@dimen/dp_8"                      android:background="@drawable/shape_4_gradient_c3a5ff_dfcdff"                            android:paddingStart="@dimen/dp_4"                            android:paddingEnd="@dimen/dp_4"                            android:text="密切度"                            android:textColor="@color/white"                            android:textSize="@dimen/sp_12" />                        <TextView                            android:id="@+id/tv_intimacy_percent_value"                            android:layout_width="wrap_content"                            android:layout_height="wrap_content"                            android:layout_alignParentRight="true"                            android:layout_marginTop="@dimen/dp_6"                            android:layout_marginEnd="@dimen/dp_15"                            android:includeFontPadding="false"                            android:text="0/0"                            android:textColor="@color/c_8B5AED"                            android:textSize="@dimen/sp_9" />                        <rogressBar                            android:id="@+id/pb_intimacy_progress"                            style="@android:style/Widget.ProgressBar.Horizontal"                            android:layout_width="match_parent"                            android:layout_height="@dimen/dp_6"                            android:layout_alignBaseline="@id/tv_intimacy_tip"                            android:layout_marginStart="@dimen/dp_7"                            android:layout_marginEnd="@dimen/dp_14"                            android:layout_toRightOf="@id/tv_intimacy_tip"                            android:max="100"  android:progressDrawable="@drawable/layer_intimacy_pb_drawable" />                        <TextView                            android:id="@+id/tv_intimacy_authentication"                            android:layout_width="wrap_content"                            android:layout_height="match_parent"                            android:layout_below="@id/pb_intimacy_progress"                            android:layout_marginStart="@dimen/dp_17"                            android:layout_marginTop="@dimen/dp_4"                            android:text="密切度1000可认证密切关系,快去提升密切度吧~"                            android:textColor="@color/c_C4A7FF"                            android:textSize="@dimen/sp_10" />                    </RelativeLayout>                </RelativeLayout>                <androidx.appcompat.widget.Toolbar                    android:id="@+id/tb_intimacy"                    android:layout_width="match_parent"                    android:layout_height="@dimen/dp_80"                    android:background="@color/white"                    app:contentInsetStart="0dp"                    app:layout_collapseMode="pin">                    <com.donews.renrenplay.android.views.TitleLayout                        android:id="@+id/tl_intimacy"                        android:layout_width="match_parent"                        android:layout_height="wrap_content"                        android:paddingTop="@dimen/dp_20"                        app:showTitleStr="你和朋侪的密切关系" />                </androidx.appcompat.widget.Toolbar>            </com.google.android.material.appbar.CollapsingToolbarLayout>            <LinearLayout                android:layout_width="match_parent"                android:layout_height="wrap_content"                android:layout_alignParentBottom="true"                android:layout_marginTop="-20dp"                android:background="@drawable/shape_22_top_ffffff"                androidrientation="vertical">                <com.donews.renrenplay.android.views.SimpleViewpagerIndicator                    android:id="@+id/indicator_intimacy"                    android:layout_width="wrap_content"                    android:layout_height="@dimen/dp_58"                    android:layout_gravity="center"                    android:tag="overScroll"                    app:isshow_underline="true"                    app:selected_textcolor="@color/c_333333"                    app:selected_textsize="@dimen/sp_16"                    app:unselected_textcolor="@color/c_999999"                    app:unselected_textsize="@dimen/sp_16" />                <View                    android:layout_width="match_parent"                    android:layout_height="@dimen/dp_1"                    android:background="@color/c_F5F5F5" />            </LinearLayout>    </com.google.android.material.appbar.AppBarLayout>    <androidx.viewpager.widget.ViewPager        android:id="@+id/vp_intimacy"        android:layout_width="match_parent"        android:layout_height="match_parent"        android:background="@color/white"        app:layout_behavior="@string/appbar_scrolling_view_behavior" /></androidx.coordinatorlayout.widget.CoordinatorLayout>这个是实现的结果就是 折叠完布局之后,有个tab 指示器,下面是viewpage 页。
我们在实现这个结果之前先来看看 SliverAppbar 的一些比力紧张的属性,

title :SliverAppbar 的标题 title;
centerTitle :标题居中;
pinned :true时 SliverAppBar 会固定在页面顶部;false时,SliverAppBar 会
随着滑动向上滑动;
floating :当值为true时 SliverAppBar设置的title会随着上滑动潜伏
然后设置的bottom会体现在原AppBar的位置
当值为false时 SliverAppBar设置的title会不会潜伏
然后设置的bottom会体现在原AppBar设置的title下面;
snap: 当snap设置为true时,向下滑动页面,SliverAppBar(以及此中设置的
flexibleSpace内容)会立即体现出来,
反之当snap设置为false时,向下滑动时,只有当ListView的数据滑动到
顶部时,SliverAppBar才会下拉体现出来;
expandedHeight:完全睁开的高度;
elevation:阴影高度;
flexibleSpaceappbar 折叠的内容地区
bottomappbar 底部地区
1、SliverAppbar 紧张有三部门,
第一部门:标题, 通过 title 属性设置;
第二部门:折叠的内容部门,通过 flexibleSpace 属性设置的
FlexibleSpaceBar 中设置 background;
第三部门:tabbar 标签栏, 通过bottom属性设置,在这里团结
PreferredSize 来使用。
2、 接下来我们将这三部门举行设置,我们如今initState 函数中 设置tabController 属性,以及tabs ,tabviews,此中HomePage(),ProfilePage() 就是两个单独拎出来的widget,vsync: this 中的这个this ,得让state 继续SingleTickerProviderStateMixin。
  TabController? tabController; //tabs  List<Tab> tabs = [    Tab(text: 'Home'),    Tab(text: 'Profile'),  ];  //tabviews  List<Widget> tabViews = [HomePage(), ProfilePage()];  @override  void initState() {    super.initState();    this.tabController = TabController(length: 2, vsync: this);  }3、接下来就是 build 函数,设置sliverAppbar ,TabBar, TabView 。
@override  Widget build(BuildContext context) {    return Scaffold(        body: ExtendedNestedScrollView(      onlyOneScrollInBody: true,      headerSliverBuilder: (BuildContext context, bool innerBoxIsScrolled) {        return [          // sliverappbar 有三部门, 第一部门 标题,通过title属性设置;          // 第二部门就是用来折叠部门的轮播图,通过 flexibleSpace 属性          // 设置的FlexibleSpaceBar中设置;          //第三部门就是通过 bottom 设置的 TabBar 标签栏,在这里团结 PreferredSize 来使用;          SliverAppBar(            //SliverAppbar 的标题 title            title: buildHeader(),            //标题居中            centerTitle: true,            //true时 SliverAppBar 会固定在页面顶部;false时,SliverAppBar 会随着滑动向上滑动            pinned: true,            //当值为true时 SliverAppBar设置的title会随着上滑动潜伏            //然后设置的bottom会体现在原AppBar的位置            //当值为false时 SliverAppBar设置的title会不会潜伏            //然后设置的bottom会体现在原AppBar设置的title下面             floating: false,            // //当snap设置为true时,向下滑动页面,SliverAppBar(以及此中设置的flexibleSpace内容)会立即体现出来,            // //反之当snap设置为false时,向下滑动时,只有当ListView的数据滑动到顶部时,SliverAppBar才会下拉体现出来。             snap: false,            // automaticallyImplyLeading: true,            //睁开的高度            expandedHeight: 200,            elevation: 0,            //appbr 下的内容地区           flexibleSpace: FlexibleSpaceBar(              //配景              //设置的是一个widget也就是说在这里可以使用恣意的              //Widget组合 在这里直接使用的是一个图片              background: buildFlexibleSpaceWidget(),        ),            bottom: buildFlexibleTooBarWidget(), //appbar 底部地区          ),        ];      },      body: TabBarView(        controller: this.tabController,        children: tabViews,      ),    ));  }

可以看到在这里我们用到了一个 插件 ExtendedNestedScrollView,为啥用这个呢?而不是CustomScrollview 呢? 可以看个结果图, 1.jpg ,使用CustomScrollview 时,当我们将列表滑动上去之后,会出现这种环境,item1,2 在上面固定拉不下来,除非当我们将折叠内容睁开之后,列表才气拉下来正常展示。以是我们这里用到了ExtendedNestedScrollView插件。
3、在设置SliverAppbar 中用到的方法函数
buildFlexibleSpaceWidget() {    return Stack(      children: [        Image.asset('assets/images/integral_center_bg.png', width: MediaQuery.of(context).size.width,fit: BoxFit.fill,),        Container(            margin: EdgeInsets.only(top: 100),            child: Column(              children: [                Text('我是小米'),                SizedBox(                  height: 20,                ),                Row(                  crossAxisAlignment: CrossAxisAlignment.center,                  mainAxisAlignment: MainAxisAlignment.center,                  children: [                    Text('我的性别女'),                    SizedBox(                      width: 20,                    ),                    Text('我的年事19'),                  ],                )              ],            )),      ],    );  }  //构建SliverAppBar的标题title  buildHeader() {    //透明组件    return Container(      width: double.infinity,      padding: EdgeInsets.only(left: 10),      height: 38,      decoration: BoxDecoration(        color: Colors.white,        border: Border.all(color: Colors.white),        borderRadius: BorderRadius.circular(30),      ),      child: Row(        mainAxisAlignment: MainAxisAlignment.center,        children: [          Icon(            Icons.search_rounded,            size: 18,          ),          SizedBox(            width: 4,          ),          Text(            "搜刮",            style: TextStyle(              fontSize: 14,            ),          ),        ],      ),    );  }  //[SliverAppBar]的bottom属性配制  buildFlexibleTooBarWidget() {    //[PreferredSize]用于设置在AppBar大概是SliverAppBar    //的bottom中 实现 PreferredSizeWidget    return PreferredSize(      //界说大小      preferredSize: Size(MediaQuery.of(context).size.width, 44),      //设置恣意的子Widget      child: Container(        alignment: Alignment.center,        child: Container(          // color: Colors.grey,          //随着向上滑动,TabBar的宽度渐渐增大          //父布局Container束缚为 center对齐          //以是程现出来的是中央x轴放大的结果          width: MediaQuery.of(context).size.width,          child: TabBar(            controller: tabController,            tabs: tabs,          ),        ),      ),    );  }看下终极结果:(包容下?,现在我的级别还不能插入视频,先上图片看看吧)



4.jpg 这时你会发现一个征象, tabbar 如今是在sliverAppbar 底部,有些人大概不太想要这个结果,就想要将tabbar 放在SliverAppbar 的下面,实在也是可以的哦~~

我们可以将SliverAppbar 原来的底子大将bottom属性去掉,然后 再追加个SliverPersistentHeader 组件,我把整个修改后的build函数中的代码展示下,这样方便和上面的代码做对比。
  @override  Widget build(BuildContext context) {    return Scaffold(        body: ExtendedNestedScrollView(      onlyOneScrollInBody: true,      headerSliverBuilder: (BuildContext context, bool innerBoxIsScrolled) {        return [          // sliverappbar 有三部门, 第一部门 标题,通过title属性设置;          // 第二部门就是用来折叠部门的轮播图,通过 flexibleSpace 属性          // 设置的FlexibleSpaceBar中设置;          //第三部门就是通过 bottom 设置的 TabBar 标签栏,在这里团结 PreferredSize 来使用;          SliverAppBar(            //SliverAppbar 的标题 title            title: buildHeader(),            //标题居中            centerTitle: true,            //true时 SliverAppBar 会固定在页面顶部;false时,SliverAppBar 会随着滑动向上滑动            pinned: true,            //当值为true时 SliverAppBar设置的title会随着上滑动潜伏            //然后设置的bottom会体现在原AppBar的位置            //当值为false时 SliverAppBar设置的title会不会潜伏            //然后设置的bottom会体现在原AppBar设置的title下面            // floating: false,            //            // //当snap设置为true时,向下滑动页面,SliverAppBar(以及此中设置的flexibleSpace内容)会立即体现出来,            // //反之当snap设置为false时,向下滑动时,只有当ListView的数据滑动到顶部时,SliverAppBar才会下拉体现出来。            // snap: false,            // automaticallyImplyLeading: true,            //睁开的高度            expandedHeight: 200,            elevation: 0,            //appbr 下的内容地区            flexibleSpace: FlexibleSpaceBar(              //配景              //设置的是一个widget也就是说在这里可以使用恣意的              //Widget组合 在这里直接使用的是一个图片              background: buildFlexibleSpaceWidget(),            ),            // bottom: buildFlexibleTooBarWidget(), //appbar 底部地区          ),          SliverPersistentHeader(            pinned: true,            delegate: StickyTabBarDelegate(              child: TabBar(                labelColor: Colors.black,                controller: this.tabController,                tabs: tabs,              ),            ),          ),        ];      },      body: TabBarView(        controller: this.tabController,        children: tabViews,      ),    ));  }此中我们会发现,SliverPersistentHeader 中的delegate 是我们创建的一个class类StickyTabBarDelegate,
class StickyTabBarDelegate extends SliverPersistentHeaderDelegate {  final TabBar child;  StickyTabBarDelegate({required this.child});  @override  Widget build(      BuildContext context, double shrinkOffset, bool overlapsContent) {    return Container(      color: Colors.green,      child: this.child,    );  }  @override  double get maxExtent => this.child.preferredSize.height;  @override  double get minExtent => this.child.preferredSize.height;  @override  bool shouldRebuild(SliverPersistentHeaderDelegate oldDelegate) {    return true;  }}在build 函数中我们可以设置tabbar 的配景致哟~~~
看下结果图,各人可以对比上面的结果图,其他滑动结果和上面是一样的哈!

5.jpg
您需要登录后才可以回帖 登录 | 立即注册

Powered by CangBaoKu v1.0 小黑屋藏宝库It社区( 冀ICP备14008649号 )

GMT+8, 2025-4-4 14:53, Processed in 0.098056 second(s), 36 queries.© 2003-2025 cbk Team.

快速回复 返回顶部 返回列表