带有动态列表的Flutter自定义滚动

时间:2019-02-15 15:38:36

标签: flutter flutter-layout

对颤振的CustomScrollView

有疑问

我现在拥有的是:

@override
Widget build(BuildContext context) {
  return Container(
    child: CustomScrollView(
      shrinkWrap: true,
      slivers: <Widget>[
        SliverPadding(
          padding: EdgeInsets.all(0),
          sliver: SliverList(
            delegate: SliverChildListDelegate(
              <Widget>[
                Card(),
                Card(),
                Container(
                  child: ListView.builder(
                    physics: const NeverScrollableScrollPhysics(),
                  ),
                )
              ],
            ),
          ),
        ),
      ],
    ),
  );
}

*代码已作为示例进行了简化。

您可以看到里面有2个Cards()和一个Container的{​​{1}}。两张卡都有固定的高度,带有列表的容器具有动态的高度。 现在的工作方式:随着ListView的长大,我们可以滚动到列表的底部。因此通常两个ListView小部件都变得不可见,因为Card的元素占据了整个屏幕。

该想法是强制将第二个List固定到屏幕顶部。所以基本上它应该在它自己的位置上,但是随着越来越多的滚动,它应该固定在屏幕顶部,而我越来越多地滚动列表。 问题是我该怎么做?

谢谢!

1 个答案:

答案 0 :(得分:1)

您可以使用SliverPersistentHeader,我为您准备了此示例:

      @override
      Widget build(BuildContext context) {
        return Scaffold(
          body: SafeArea(
            child: CustomScrollView(
              slivers: <Widget>[
                SliverPadding(
                  padding: EdgeInsets.all(0),
                  sliver: SliverList(
                      delegate: SliverChildBuilderDelegate(
                    (context, index) {
                      return ListTile(
                        title: Text("index: $index"),
                      );
                    },
                    childCount: 3,
                  )),
                ),
                SliverPersistentHeader(
                  pinned: true,
                  delegate: PersistentHeader("Card 1"),
                ),
                SliverPersistentHeader(
                  pinned: true,
                  delegate: PersistentHeader("Card 2"),
                ),
                SliverPadding(
                  padding: EdgeInsets.all(0),
                  sliver: SliverList(
                      delegate: SliverChildBuilderDelegate((context, index) {
                    return ListTile(
                      title: Text("index: $index"),
                    );
                  })),
                ),
              ],
            ),
          ),
        );
      }
    }

    class PersistentHeader extends SliverPersistentHeaderDelegate {
      final String title;

      PersistentHeader(this.title);

      @override
      Widget build(
          BuildContext context, double shrinkOffset, bool overlapsContent) {
        return Container(
          color: Colors.white,
          child: Card(
            color: Colors.white,
            elevation: 7.0,
            child: SizedBox(height: 100.0, child: Center(child: Text(title))),
          ),
        );
      }

      @override
      double get maxExtent => 100.0;

      @override
      double get minExtent => 100.0;

      @override
      bool shouldRebuild(SliverPersistentHeaderDelegate oldDelegate) {
        return false;
      }
    }

更多信息:https://docs.flutter.io/flutter/widgets/SliverPersistentHeader-class.html