likes
comments
collection
share

【Flutter】Sliver的各种修饰你不会不知道吧Sliver 大家肯定都不陌生了,它是 Flutter 开发中非常

作者站长头像
站长
· 阅读数 16

Sliver 的各种装饰器

前言

最近项目确实很多,工作也很忙,更新都不给力了,(主要是晚上也要打奥杜尔啊) 大家见谅。

事情是这么回事,一朋友在开发中遇到问题请教到我这里,一顿扒扒之后我感觉他还是对 Silver 以及它对应的一些装饰器不是很熟悉导致走错了方向,遂有此文,一个是自己复习下,一方面也是给大家科普一下当做是查漏补缺了。

正文:

Sliver 大家肯定都不陌生了,它是 Flutter 开发中非常常见的控件,Sliver 通常在 CustomScrollView 中使用,允许开发者灵活地组合不同类型的 Sliver 组件。

我们一般都会用到如 SliverList:用于创建一个可滚动的列表。SliverGrid:用于创建一个可滚动的网格。SliverAppBar:通常用于创建一个具有可扩展和收缩功能的应用栏。这些 Sliver ,而其中最最常见的又是 SliverList 这个控件。

相对比 ListView ,我个人更喜欢使用 SliverList ,说起 ListView 其实本质就是对 Sliver 的一种封装。ListView 提供了一个简单易用的接口,用于创建垂直滚动的列表,同时隐藏了许多底层实现的复杂性,适合大多数基本列表场景。简单理解就是场景化封装,熟悉谷歌的都知道谷歌最爱干这事了。

而我们直接使用 SliverList 就会更加的灵活,可以添加不同的组合滚动,可以设置不同的装饰器,从而到达我们想要的效果。

下面就一起看看常用的装饰器有哪些,并且实现朋友说的那个场景。

一、常见的装饰器

1.1 SliverToBoxAdapter

SliverToBoxAdapter 允许您在 Sliver 中插入一个普通的非 Sliver 组件。它可以用于在 CustomScrollView 中混合 Sliver 和普通 Widget。

SliverToBoxAdapter(
  child: Container(
    height: 100,
    color: Colors.orange,
    child: Center(child: Text('This is a normal widget')),
  ),
)

这个是最常见,最基础的装饰了,必须要把普通的 Widget 转换为 Sliver 的 Widgert,否则直接报错。

1.2 SliverFillRemaining
SliverFillRemaining(
  child: Container(
    color: Colors.green,
    child: Center(child: Text('Filling remaining space')),
  ),
)

这个也是很常见的一个包装器,可以允许您填充剩余的可滚动空间。有点权重的那种意思。各种场景都比较实用,比如一些 Loading 布局,错误布局,自定义的一些效果也很常用。

1.3 SliverPadding

SliverPadding(
  padding: EdgeInsets.all(16.0),
  sliver: SliverList(
    delegate: SliverChildBuilderDelegate(
      (BuildContext context, int index) {
        return ListTile(title: Text('Item $index'));
      },
      childCount: 100,
    ),
  ),
)

SliverPadding 是一个用于在 Sliver 组件周围添加填充的装饰器。它允许您为 Sliver 的内容设置内边距,从而控制 Sliver 与其父容器或其他 Sliver 之间的距离。

我发现很多人不知道这个装饰,比如不同的 Sliver 包装 SliverPadding ,那么它是对不同的 Sliver 生效的,不是对内部的子 Item 生效,不管是包装的一个Silver 还是一组 Sliver 都是这样。

例如一个 CustomScrollView 中我们设置不同的 Sliver 组

                SliverPadding(
                      padding: const EdgeInsets.all(10),
                      sliver: SliverList(
                        delegate: SliverChildListDelegate(
                          [
                            Container(
                              height: 100,
                              color: Colors.red,
                              child: Center(child: Text('Custom Item 1')),
                            ),
                            Container(
                              height: 100,
                              color: Colors.green,
                              child: Center(child: Text('Custom Item 2')),
                            ),
                            Container(
                              height: 100,
                              color: Colors.blue,
                              child: Center(child: Text('Custom Item 3')),
                            ),
                            // 添加更多的子项
                          ],
                        ),
                      ),
                    ),


                    SliverPadding(
                      padding: const EdgeInsets.all(5),
                      sliver: SliverList(
                        delegate: SliverChildListDelegate(
                          [
                            ListTile(title: Text('Item 1')),
                            ListTile(title: Text('Item 2')),
                            ListTile(title: Text('Item 3')),
                            // 可以添加更多的 ListTile 或其他 Widget
                          ],
                        ),
                      ),
                    )

那么这里的 Padding 是体现在两个 SliverList 之间的间距。

1.4 SliverOpacity

理解了 SliverPadding 之后,再看类似的包装器就是很简单了

SliverOpacity(
  opacity: 0.5,
  sliver: SliverList(
    delegate: SliverChildBuilderDelegate(
      (BuildContext context, int index) {
        return ListTile(title: Text('Item $index'));
      },
      childCount: 100,
    ),
  ),
)

SliverOpacity 组件允许您为 Sliver 设置透明度。它可以控制整个 Sliver 的可见性,对实现渐变效果非常有用。

1.5 DecoratedSliver

DecoratedSliver(
  decoration: BoxDecoration(
    color: Colors.blue.withOpacity(0.2),
    borderRadius: BorderRadius.circular(10),
  ),
  sliver: SliverList(
    delegate: SliverChildBuilderDelegate(
      (BuildContext context, int index) {
        return ListTile(title: Text('Item $index'));
      },
      childCount: 100,
    ),
  ),
)

DecoratedSliver 是一个用于给 Sliver 添加装饰效果的组件。它可以通过 BoxDecoration 来设置背景色、边框、阴影以及圆角等。

1.6 SliverVisibility 与 SliverOffstage

居然透明度都可以有 Sliver 类型,那么显示隐藏的功能没道理没有啊。

SliverVisibility(
    visible: true, // 根据条件设置为 true 或 false
    sliver: SliverList(
        delegate: SliverChildBuilderDelegate(
        (BuildContext context, int index) {
            return ListTile(title: Text('Item $index'));
        },
        childCount: 10,
        ),
    ),
 SliverOffstage(
    offstage: !_isVisible,
    sliver: SliverList(
        delegate: SliverChildBuilderDelegate(
        (BuildContext context, int index) {
            return ListTile(title: Text('Item $index'));
        },
        childCount: 10,
        ),
    ),

占位与不占位的显示隐藏也是非常方便。

1.7 SliverMainAxisGroup 与 SliverCrossAxisGroup

     SliverMainAxisGroup(
            children: <Widget>[
              SliverToBoxAdapter(child: Container(height: 100, color: Colors.red)),
              SliverToBoxAdapter(child: Container(height: 100, color: Colors.green)),
              SliverToBoxAdapter(child: Container(height: 100, color: Colors.blue)),
            ],
          ),

老版本中我们如果想要对列表分组我们可能需要导入一些第三方库,如 sticky_headers 之类的进行列表的分组,现在我们可以直接操作了。

我们在 SliverMainAxisGroup 中配合 SliverPersistentHeader 固定住分组的头部就可以实现分组的效果。

   CustomScrollView(
        slivers: [
          // 第一个组的头部
          SliverPersistentHeader(
            pinned: true,
            delegate: _SliverAppBarDelegate(
              child: Container(
                color: Colors.blue,
                height: 60.0,
                alignment: Alignment.center,
                child: Text(
                  'Group 1',
                  style: TextStyle(color: Colors.white, fontSize: 20),
                ),
              ),
            ),
          ),
          // 第一个组的内容
          SliverMainAxisGroup(
            children: <Widget>[
              SliverToBoxAdapter(child: Container(height: 100, color: Colors.red)),
              SliverToBoxAdapter(child: Container(height: 100, color: Colors.green)),
              SliverToBoxAdapter(child: Container(height: 100, color: Colors.blue)),
            ],
          ),
          // 第二个组的头部
          SliverPersistentHeader(
            pinned: true,
            delegate: _SliverAppBarDelegate(
              child: Container(
                color: Colors.green,
                height: 60.0,
                alignment: Alignment.center,
                child: Text(
                  'Group 2',
                  style: TextStyle(color: Colors.white, fontSize: 20),
                ),
              ),
            ),
          ),
          // 第二个组的内容
          SliverMainAxisGroup(
            children: <Widget>[
              SliverToBoxAdapter(child: Container(height: 100, color: Colors.yellow)),
              SliverToBoxAdapter(child: Container(height: 100, color: Colors.orange)),
              SliverToBoxAdapter(child: Container(height: 100, color: Colors.purple)),
            ],
          ),
        ],
      ),


class _SliverAppBarDelegate extends SliverPersistentHeaderDelegate {
  final Widget child;

  _SliverAppBarDelegate({required this.child});

  @override
  double get minExtent => child != null ? 60.0 : 0.0;

  @override
  double get maxExtent => child != null ? 60.0 : 0.0;

  @override
  Widget build(BuildContext context, double shrinkOffset, bool overlapsContent) {
    return child;
  }

  @override
  bool shouldRebuild(_SliverAppBarDelegate oldDelegate) {
    return child != oldDelegate.child;
  }
}   

1.8 SliverAppbar 相关

     SliverAppBar(
            expandedHeight: 200.0, // 展开的高度
            floating: false, // 是否在滚动时保持在可见区域
            pinned: true, // 是否固定在顶部
            flexibleSpace: FlexibleSpaceBar(
              title: Text('SliverAppBar Example'),
              background: Image.network(
                'xxxxxx', // 替换为你的图片链接
                fit: BoxFit.cover,
              ),
            ),
          ),

SliverAppBar 是一个非常强大的组件,用于创建可滚动的应用栏。它可以与其他 Sliver 组件(如 SliverList、SliverGrid 等)结合使用,提供灵活的滚动效果,有点类似 Android 中的灵活布局的效果。

SliverAppBar 的主要属性

expandedHeight: 指定 SliverAppBar 展开时的高度。

pinned: 设置为 true 时,AppBar 将在滚动到顶部时固定在屏幕顶部。

floating: 设置为 true 时,当向下滚动时,AppBar 会重新出现。

flexibleSpace: 用于定义可伸缩的空间,通常使用 FlexibleSpaceBar。

backgroundColor: 设置 AppBar 的背景色。

actions: 一个包含 IconButton 或其他小部件的列表,通常用于显示操作按钮。

SliverAppBar 虽然可以配合 SliverPersistentHeader 实现炫酷的效果,但是多半也是停留在 Demo 中,由于谷歌位太浓了,反倒是没有太多使用的机会。

在更新了 Flutter 3.24.0 之后,加入了全新版本 SliverFloatingHeader 、PinnedHeaderSliver、SliverResizingHeader 会更加的好用,出场机会更多了。

因为他们是对 SliverAppBar 的各种行为的有一次场景化封装,更加简化的使用了。并且并不局限于 SliverAppBar 这种只能是文本、图片的布局,可以自定义布局更方便的实现特定的效果了。后期会更加期待他们的出场。

1.9 其他

除了上述的常用 Sliver 和对应的包装之外,当然还有非常多的 Sliver 和包装器,我没有穷举出来,只是这些常用的我们要了解,下面给出一个实战中的典型的示例。

二、场景复现

效果图是这样的:

【Flutter】Sliver的各种修饰你不会不知道吧Sliver 大家肯定都不陌生了,它是 Flutter 开发中非常

乍一看,这还不简单?

其实在我们的 LoadView 中加载的 Silver 就会有问题,因为 LoadView 是需要确定父布局的宽高的,如果没有圆角的装饰器,我们直接使用一个 expanded 的 LoadView 直接往下排就行了。

但是如果我们想要把 LoadView 中的 Sliver 真正的内容包裹起来使用装饰器去做就不好做。

最想到的是使用 Container 包裹,设置装饰器:

  Container(
      width: double.infinity,
      margin: EdgeInsets.only(left: 15, right: 15, bottom: 15),
      padding: EdgeInsets.only(top: ScreenUtil.getStatusBarH(context)),
      decoration: BoxDecoration(
        color: const Color(0xFF4DCFF6).withOpacity(0.2), // 设置背景颜色和不透明度
        borderRadius: BorderRadius.circular(5), // 设置圆角
      ),
      child: LoadStateLayout(
        state: controller.loadingState,
        errorMessage: controller.errorMessage,
        errorRetry: () {
          controller.retryRequest();
        },
        successSliverWidget: [
          SliverList(
            delegate: SliverChildBuilderDelegate((context, index) {
              return ReportStaffRequestItem(item: state.datas[index]);
            }, childCount: state.datas.length),
          ),
        ],
      ),
    ).expanded(),

我们对 Container 设置 expanded 之后它的高度是确定的,那么 LoadView 有明确的宽高是没问题的,但是此时就是剩余的高度全部都被装饰了,没有达到我们动态列表的包裹内容装饰的效果。

那么我们外层包裹一个 Column 不就行了吗,内部就是高度自适应。

Column(
  children: [
    Container(
      width: double.infinity,
      margin: EdgeInsets.only(left: 15, right: 15, bottom: 15),
      padding: EdgeInsets.only(top: ScreenUtil.getStatusBarH(context)),
      decoration: BoxDecoration(
        color: const Color(0xFF4DCFF6).withOpacity(0.2), // 设置背景颜色和不透明度
        borderRadius: BorderRadius.circular(5), // 设置圆角
      ),
      child: LoadStateLayout(
        state: controller.loadingState,
        errorMessage: controller.errorMessage,
        errorRetry: () {
          controller.retryRequest();
        },
        successSliverWidget: [
          SliverList(
            delegate: SliverChildBuilderDelegate((context, index) {
              return ReportStaffRequestItem(item: state.datas[index]);
            }, childCount: state.datas.length),
          ),
        ],
      ),
    ),
  ],
)

结果大家肯定是知道的,报错了,问题就出在高度自适应的问题上,因为 LoadStateLayout 需要明确的宽高,因为内部有错误和空布局的处理。

那怎么办?修改 LoadStateLayout 的代码吗?

其实聪明的你肯定发现了,在前文中我们介绍了 Sliver 与装饰器的介绍,我们可以直接通过装饰器来实现:

     LoadStateLayout(
            state: controller.loadingState,
            errorMessage: controller.errorMessage,
            errorRetry: () {
                    controller.retryRequest();
                  },
            successSliverWidget: [
                    
            SliverPadding(
                padding: const EdgeInsets.only(left: 15,right: 15,bottom: 15),
                sliver: DecoratedSliver(
                decoration: BoxDecoration(
                    color: const Color(0xFF4DCFF6).withOpacity(0.2), // 设置背景颜色和不透明度
                    borderRadius: BorderRadius.circular(5), // 设置圆角
                ),
                sliver: SliverList(
                    delegate: SliverChildBuilderDelegate((context, index) {
                        return ReportStaffRequestItem(item: state.datas[index]);
                    },
                    childCount: state.datas.length,
                    ),
                ),
                ),
            ),
                    
            ],
        ).expanded(),

效果:

【Flutter】Sliver的各种修饰你不会不知道吧Sliver 大家肯定都不陌生了,它是 Flutter 开发中非常

故意这么写就是为了清楚,这里的装饰器是给 Sliver 层级的装饰,而不是内部的 Item 子布局,如果要达到效果图的样子,我们只需要改内部的布局间距即可。

最终效果:

【Flutter】Sliver的各种修饰你不会不知道吧Sliver 大家肯定都不陌生了,它是 Flutter 开发中非常

总结

本文我们介绍了常见的 Sliver 以及对应的装饰器,后面我们介绍了如何进行实战的开发。

在我们进行开发的过程中,牢记每一个 Sliver,不管是单个 Sliver 控件还是 Sliver 控件组,我们都可以分别进行装饰,不管是动画效果,渐变显隐,还是边框装饰,间距设置都是可以更方便的实现的。

本文对应的 Fultter 版本是 3.19.2 以及 3.24.0 ,由于 iOS 权限的问题,我相信大家的 Flutter 版本至少都已经升级到 3.19.+ 了吧,所以这些 Sliver 都是可以用的,其实我们看到谷歌也是一直在更新 Sliver 的控件与包装器,越来越方便了呢。

那本文的代码比较简单,相对比较基础,并且全部的代码也已经在文中展出,这里就不放出项目链接了。

今天的分享就到这里啦,当然如果你有其他的更多的更好的实现方式,也希望大家能评论区交流一起学习进步。如果我的文章有错别字,不通顺的,或者代码、注释、有错漏的地方,同学们都可以指出修正。

如果感觉本文对你有一点的启发和帮助,还望你能点赞支持一下,你的支持对我真的很重要。

Ok,这一期就此完结。

【Flutter】Sliver的各种修饰你不会不知道吧Sliver 大家肯定都不陌生了,它是 Flutter 开发中非常

转载自:https://juejin.cn/post/7402797060980490274
评论
请登录