【问题标题】:NestedScrollView with SliverAppBar cause unexpected body padding带有 SliverAppBar 的 NestedScrollView 会导致意外的正文填充
【发布时间】:2021-10-18 16:44:27
【问题描述】:

当我运行此代码时,ListView 会有一个意外的顶部填充。 (见截图)为什么会这样,有没有办法避免这种情况?

我已经尝试过 SliverOverlapAbsorber、SafeArea 和 MediaQuery 来修复填充,但到目前为止似乎没有任何效果。

import 'package:flutter/material.dart';

void main() {
  runApp(MyApp());
}

class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      home: MyHomePage(),
    );
  }
}

class MyHomePage extends StatelessWidget {
  MyHomePage({Key? key}) : super(key: key);

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: NestedScrollView(
        headerSliverBuilder: (context, innerBoxIsScrolled) => [
          SliverAppBar(
            pinned: true,
            title: Text('Title'),
          ),
        ],
        body: ListView.builder(
          itemCount: 24,
          itemBuilder: (context, index) => Container(
            height: 40,
            alignment: Alignment.center,
            color: Colors.yellow,
            margin: EdgeInsets.only(top: index != 0 ? 8 : 0),
            child: Text('Body $index'),
          ),
        ),
      ),
    );
  }
}

Doctor summary (to see all details, run flutter doctor -v):
[✓] Flutter (Channel stable, 2.2.3, on macOS 11.5.1 20G80 darwin-x64, locale de-DE)
[!] Android toolchain - develop for Android devices (Android SDK version 30.0.2)
    ✗ Android license status unknown.
      Run `flutter doctor --android-licenses` to accept the SDK licenses.
      See https://flutter.dev/docs/get-started/install/macos#android-setup for more details.
[✓] Xcode - develop for iOS and macOS
[✓] Chrome - develop for the web
[✓] Android Studio (version 4.2)
[✓] IntelliJ IDEA Community Edition (version 2021.1.1)
[✓] VS Code (version 1.59.0)
[✓] Connected device (2 available)

【问题讨论】:

  • 我在 dartpad 中试过你的代码,它工作正常
  • 你是对的,它不会发生在 dartpad 上。但我可以在 ios 模拟器和实际的 ios 设备以及 android 模拟器上重现它。
  • 你用真机试过了吗
  • 是的,带有 ios 14.7.1(最新版本)和 ios 模拟器 (14.5) 的真正 iphone。还有安卓模拟器(11)

标签: flutter dart flutter-sliver nestedscrollview flutter-sliverappbar


【解决方案1】:

ListView 添加一些默认填充,check documentation

默认情况下,ListView 将自动填充列表的可滚动末端,以避免 MediaQuery 填充指示的部分障碍。为避免此行为,请使用零填充属性覆盖。

您可以通过在构建器中添加padding 配置来轻松摆脱它,就像这样(您可以添加0,但我不推荐它):

body: ListView.builder(
  padding: const EdgeInsets.only(top: 8), // <-- add this line
  itemCount: 24,
  itemBuilder: (context, index) => Container(
    height: 40,
    alignment: Alignment.center,
    color: Colors.yellow,
    margin: EdgeInsets.only(top: index != 0 ? 8 : 0),
    child: Text('Body $index'),
  ),
),

【讨论】:

  • 感谢您的回答,彼得。但是,我想避免直接设置填充。特别是因为 Dartpad 不显示填充,所以我不确定哪种行为实际上是正确的。顺便提一句。您的解决方案还将删除应保留的底部填充。所以我想出了一个在 ListView 周围使用 MediaQuery.removePadding(...) 的解决方案
  • 不客气,很高兴听到它有效。你说得对,removePadding 更好。
【解决方案2】:

如果您将 ListView 放在没有 appBar 的 Scaffold 内的某个位置,则 ListView 具有顶部填充。如果 appBar 存在,则没有填充。如果为 ListView 明确指定 EdgeInsets.zero 填充,则没有填充。

【讨论】:

  • 感谢阿比提供的信息。当 AppBar 丢失时,我不知道这种行为。我现在用 MediaQuery.removePadding(...) 包装 ListView 以避免这种情况。
【解决方案3】:

似乎没有办法避免手动删除填充。所以我的解决方案现在看起来像这样:

return Scaffold(
      body: DefaultTabController(
        length: 2,
        child: NestedScrollView(
          headerSliverBuilder: (context, innerBoxIsScrolled) => [
            SliverAppBar(
              pinned: true,
              title: Text('Title'),
              bottom: TabBar(tabs: [Tab(text: 'T 1'), Tab(text: 'T 2')]),
            ),
          ],
          body: MediaQuery.removePadding(
            removeTop: true,
            context: context,
            child: ListView.builder(
              itemCount: 24,
              itemBuilder: (context, index) => Container(
                height: 40,
                padding: EdgeInsets.zero,
                alignment: Alignment.center,
                color: Colors.yellow,
                margin: EdgeInsets.only(top: index != 0 ? 8 : 0),
                child: Text('Body $index'),
              ),
            ),
          ),
        ),
      ),
    );

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2022-07-21
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2019-04-11
    • 2014-07-19
    • 1970-01-01
    相关资源
    最近更新 更多