Вложенный CustomScollView Flutter не может прокручивать дочерний scrollview

Когда для вложенной термоусадочной упаковки CustomScrollView установлено значение true, я не могу свернуть дочерние фрагменты.

Как прокрутить оба customScrollViews и сделать дочерний customScrollView сворачиваемым до минимальной высоты.

Вот живая демонстрация

Моя конечная цель - сделать вид списка стековых карточек с анимацией прокрутки.

Какие-нибудь другие решения? или есть способ решить эту проблему?

Посоветуйте мастера:)

class AnimationTest extends StatefulWidget {
  @override
  _AnimationTestState createState() => _AnimationTestState();
}

class _AnimationTestState extends State<AnimationTest>
    with WidgetsBindingObserver {
  @override
  void initState() {
    WidgetsBinding.instance.addObserver(this);
    super.initState();
  }

  @override
  void dispose() {
    WidgetsBinding.instance.removeObserver(this);
    super.dispose();
  }

  @override
  void didChangeAppLifecycleState(AppLifecycleState state) {
    print(state.toString());
  }

  @override
  Widget build(BuildContext context) {
    const double goldenRatio = 1.618;
    final Size displaySize = MediaQuery.of(context).size;
    final double minCardHeight = 24;
    final double maxCardHeight = 80;
    final List<Color> colors = Colors.primaries;
    List<Color> _colors = List();
    _colors.addAll(colors);
    _colors.addAll(colors);
    print(_colors.length);

    return Scaffold(
      body: Container(
        color: Colors.white,
        padding: EdgeInsets.only(top: 40),
        alignment: AlignmentDirectional.center,
        child: CustomScrollView(
          slivers: <Widget>[
            SliverList(
              delegate: SliverChildListDelegate([
                Column(
                  crossAxisAlignment: CrossAxisAlignment.center,
                  children: <Widget>[
                    CreditCard(),
                    Container(
                      width: displaySize.width * 0.85,
                      height: minCardHeight * _colors.length,
                      alignment: AlignmentDirectional.bottomCenter,
                      margin: EdgeInsets.all(0),
                      padding: EdgeInsets.all(0),
                      decoration: BoxDecoration(
                        borderRadius: BorderRadius.circular(8),
                        color: Colors.white,
                      ),
                      child: StackedList(
                        minCardHeight: minCardHeight,
                        maxCardHeight: maxCardHeight,
                        colors: _colors,
                      ),
                    )
                  ],
                ),
              ]),
            )
          ],
        ),
      ),
    );
  }
}

class StackedList extends StatefulWidget {
  final double minCardHeight;
  final double maxCardHeight;
  final List<Color> colors;
  StackedList({
    Key key,
    this.colors,
    this.maxCardHeight,
    this.minCardHeight,
  }) : super(key: key);
  @override
  _StackedListState createState() => _StackedListState();
}

class _StackedListState extends State<StackedList> {
  @override
  Widget build(BuildContext context) {
    return CustomScrollView(
      physics: AlwaysScrollableScrollPhysics(),
      shrinkWrap: true,
      slivers: <Widget>[
        ...widget.colors
            .map(
              (color) => StackedListChild(
                minHeight: widget.minCardHeight,
                maxHeight: widget.maxCardHeight,
                floating: false,
                pinned: true,
                child: getCard(
                  cardId: widget.colors.indexOf(color).toString(),
                  color: color,
                ),
              ),
            )
            .toList(),
        SliverToBoxAdapter(
            child: Container(
          height: MediaQuery.of(context).size.height,
          alignment: Alignment.center,
        ))
      ],
    );
  }

  Widget getCard({String cardId, Color color}) {
    const double cardWidth = 300;
    const double cardHeight = 370;
    return Hero(
      tag: cardId,
      child: Material(
        elevation: 3,
        color: Colors.white,
        child: InkWell(
          onTap: () => navigateToCardDetail(cardId: cardId, color: color),
          child: Container(
            width: cardWidth,
            height: cardHeight,
            color: color,
            // decoration: BoxDecoration(
            //   borderRadius: BorderRadius.only(
            //     topLeft: Radius.circular(8),
            //     topRight: Radius.circular(8),
            //   ),
            //   color: color,
            // ),
          ),
        ),
      ),
    );
  }

  navigateToCardDetail({String cardId, Color color}) {
    var route = PageRouteBuilder(
      pageBuilder: (context, animation, secondAnimation) =>
          SecondPage(cardId: cardId, color: color),
      barrierDismissible: true,
      transitionDuration: const Duration(milliseconds: 300),
      transitionsBuilder: (context, animation, secondaryAnimation, child) {
        // var begin = Offset(0.0, 1.0);
        // var end = Offset.zero;
        // var curve = Curves.fastOutSlowIn;

        // var tween =
        //     Tween(begin: begin, end: end).chain(CurveTween(curve: curve));

        return FadeTransition(
          //opacity: animation,
          //position: animation.drive(tween),
          opacity: animation,
          child: child,
        );
      },
    );
    Navigator.of(context).push(route);
  }
}

class StackedListChild extends StatelessWidget {
  final double minHeight;
  final double maxHeight;
  final bool pinned;
  final bool floating;
  final Widget child;

  SliverPersistentHeaderDelegate get _delegate => _StackedListDelegate(
      minHeight: minHeight, maxHeight: maxHeight, child: child);

  const StackedListChild({
    Key key,
    @required this.minHeight,
    @required this.maxHeight,
    @required this.child,
    this.pinned = false,
    this.floating = false,
  })  : assert(child != null),
        assert(minHeight != null),
        assert(maxHeight != null),
        assert(pinned != null),
        assert(floating != null),
        super(key: key);

  @override
  Widget build(BuildContext context) => SliverPersistentHeader(
      key: key, pinned: pinned, floating: floating, delegate: _delegate);
}

class _StackedListDelegate extends SliverPersistentHeaderDelegate {
  final double minHeight;
  final double maxHeight;
  final Widget child;

  _StackedListDelegate({
    @required this.minHeight,
    @required this.maxHeight,
    @required this.child,
  });

  @override
  double get minExtent => minHeight;

  @override
  double get maxExtent => math.max(maxHeight, minHeight);

  @override
  Widget build(
      BuildContext context, double shrinkOffset, bool overlapsContent) {
    return new SizedBox.expand(child: child);
  }

  @override
  bool shouldRebuild(_StackedListDelegate oldDelegate) {
    return maxHeight != oldDelegate.maxHeight ||
        minHeight != oldDelegate.minHeight ||
        child != oldDelegate.child;
  }
}

1 ответ

Помещение CustomScrollView внутрь другого CustomScrollView - это антипаттерн.

Вы можете использовать NestedScrollView, чтобы обернуть CustomScrollView для плавной прокрутки.

Другие вопросы по тегам