dart 不能将“Widget”类型的值赋给“PreferredSizeWidget”类型的变量

ekqde3dh  于 2023-03-15  发布在  其他
关注(0)|答案(3)|浏览(163)

此程序中有2个错误,已在上图中显示
主.dart文件

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

class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: 'Personal Expenses ',
      theme: ThemeData(
        primarySwatch: Colors.green,
        accentColor: Colors.amber,
        //errorColor: Colors.red[700],
        fontFamily: 'Quicksand',
        textTheme: ThemeData.light().textTheme.copyWith(
            title: TextStyle(
              fontFamily: 'OpenSans',
              fontWeight: FontWeight.bold,
              fontSize: 18,
            ),
            button: TextStyle(color: Colors.amber)),
        appBarTheme: AppBarTheme(
          textTheme: ThemeData.light().textTheme.copyWith(
                title: TextStyle(
                  fontFamily: 'OpenSans',
                  fontSize: 20,
                  fontWeight: FontWeight.bold,
                ),
              ),
        ),
      ),
      home: MyHomePage(),
    );
  }
}

class MyHomePage extends StatefulWidget {
 
  @override
  _MyHomePageState createState() => _MyHomePageState();
}

class _MyHomePageState extends State<MyHomePage> {
  final List<Transaction> _userTransactions = [
   
  ];
  bool _showChart = false;
  List<Transaction> get _recentTransactions {
    return _userTransactions.where((tx) {
      return tx.date.isAfter(
        DateTime.now().subtract(
          Duration(days: 7),
        ),
      );
    }).toList();
  }

  void _addNewTransaction(
      String txTitle, double txAmount, DateTime chosenDate) {
    final newTx = Transaction(
      title: txTitle,
      amount: txAmount,
      date: chosenDate,
      id: DateTime.now().toString(),
    );

    setState(() {
      _userTransactions.add(newTx);
    });
  }

  void _startAddNewTransaction(BuildContext ctx) {
    showModalBottomSheet(
      context: ctx,
      builder: (_) {
        return GestureDetector(
          onTap: () {},
          child: NewTransaction(_addNewTransaction),
          behavior: HitTestBehavior.opaque,
        );
      },
    );
  }

  void _deleteTransaction(String id) {
    setState(() {
      _userTransactions.removeWhere((tx) => tx.id == id);
    });
  }

  @override
  Widget build(BuildContext context) {
    final mediaQuery = MediaQuery.of(context);
    final isLandscape = mediaQuery.orientation == Orientation.landscape;
    final PreferredSizeWidget appbar = Platform.isIOS

“Widget”类型的值不能分配给“PreferredSizeWidget”类型的变量。

? CupertinoNavigationBar(
            middle: Text(
              'Personal Expenses ',
            ),
            trailing: Row(
              mainAxisSize: MainAxisSize.min,
              children: <Widget>[
                GestureDetector(
                  child: Icon(CupertinoIcons.add),
                  onTap: () => _startAddNewTransaction(context),
                )
              ],
            ),
          )
        : AppBar(
            title: Text(
              'Personal Expenses ',
              style: TextStyle(fontFamily: 'OpenSans'),
            ),
            actions: <Widget>[
              IconButton(
                icon: Icon(Icons.add),
                onPressed: () => _startAddNewTransaction(context),
              ),
            ],
          );
    final txListWidget = Container(
      height: (mediaQuery.size.height -
              appbar.preferredSize.height -
              mediaQuery.padding.top) *
          0.7,
      child: TransactionList(
        _userTransactions,
        _deleteTransaction,
      ),
    );
    final pageBody = SingleChildScrollView(
      child: Column(
        
        crossAxisAlignment: CrossAxisAlignment.stretch,
        children: <Widget>[
          if (isLandscape)
            Row(
              mainAxisAlignment: MainAxisAlignment.center,
              children: <Widget>[
                Text('Show Chart'),
                Switch.adaptive(
                  activeColor: Theme.of(context).accentColor,
                  value: _showChart,
                  onChanged: (val) {
                    setState(() {
                      _showChart = val;
                    });
                  },
                ),
              ],
            ),
         
        ],
      ),
    );

return平台.isIOS?库比蒂诺页面支架(子级:页面正文、导航栏:约巴,
无法将参数类型“PreferredSizeWidget”分配给参数类型“ObstructingPreferredSizeWidget?”。

)
        : Scaffold(
            .........,
                  ),
          );
  }
}
nbysray5

nbysray51#

你的代码可以在Flutter的非null安全版本中工作。我不知道为什么null安全版本现在会反对:

final PreferredSizeWidget appbar = Platform.isIOS ...

如果省略类型:

final appbar = Platform.isIOS ...

然后appbar被解释为Widget,并在appbar.preferredSize上给出错误。
您可以强制在运行时检查类型:

final dynamic appbar = Platform.isIOS ...

这对我用安卓设备测试很有效。不过,我还没有用苹果设备测试过。
我创建这个flutter issue是为了改变flutter的null-safe版本的行为,你可以订阅这个问题来获得更新通知。

**编辑:**Flutter团队提出了一个Dart issue,它解释了新的行为,并建议使用:

final PreferredSizeWidget appbar = (Platform.isIOS ? CupertinoNavigationBar() : AppBar()) 
                                   as PreferredSizeWidget;
js4nwp54

js4nwp542#

我遇到了同样的问题,这是由于Dart的零安全功能
溶液

final dynamic appBar = Platform.isIOS
    ? CupertinoNavigationBar() : AppBar()

或者你可以用

final PreferredSizedWidget appBar = Platform.isIOS
        ? CupertinoNavigationBar() as ObstructingPreferredSizedWidget : AppBar()
uxh89sit

uxh89sit3#

问题是,appBar接受实现PreferredSizewidget
和CupertinoPageScaffold接受实现障碍首选大小小部件的小部件
所以don't determine是appbar的数据类型
并且让它在运行时被确定。

就这样

final appBar = Platform.IOS ?  CupertinoNavigationBar() : AppBar()

并在Cupertino页面框架中添加Cupertino导航栏
并将AppBar添加到Scaffold

相关问题