如何调用无状态小部件的重建?

Posted

技术标签:

【中文标题】如何调用无状态小部件的重建?【英文标题】:How to invoke a rebuild of a stateless widget? 【发布时间】:2021-04-27 01:23:47 【问题描述】:

上下文

我有两个无状态小部件(页面):HomePageDetailsPage。显然应用程序启动并启动HomePage。用户可以按下一个按钮导航到DetailsPage,使用Navigator.pop() 按钮导航回HomePage

我知道DetailsPage 何时与.whenComplete() 方法一起使用。正是在这一点上,我想重建 HomePage 小部件。

代码

这是我行为的最小再现。

main.dart
import 'package:example/home.dart';
import 'package:flutter/material.dart';

void main() => runApp(MyApp());

class MyApp extends StatelessWidget 
  @override
  Widget build(BuildContext context) 
    return MaterialApp(home: HomePage());
  

home.dart
import 'package:example/details.dart';
import 'package:flutter/material.dart';

class HomePage extends StatelessWidget 
  static const name = 'Home Page';
  const HomePage() : super();

  @override
  Widget build(BuildContext context) 
    return Scaffold(
      body: Center(
        child: MaterialButton(
          color: Colors.blue,
          textColor: Colors.white,
          child: Text(name),
          onPressed: () 
            Navigator.push(
              context,
              MaterialPageRoute(builder: DetailsPage.builder),
            ).whenComplete(() => print('Rebuild now.'));
          ,
        ),
      ),
    );
  

details.dart
import 'package:flutter/material.dart';

class DetailsPage extends StatelessWidget 
  static const name = 'Details Page';
  static WidgetBuilder builder = (BuildContext _) => DetailsPage();
  const DetailsPage();

  @override
  Widget build(BuildContext context) 
    return Scaffold(
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: [
            Text(name),
            MaterialButton(
              color: Colors.blue,
              textColor: Colors.white,
              child: Text('Go Back'),
              onPressed: () => Navigator.pop(context),
            ),
          ],
        ),
      ),
    );
  

问题

如何在.whenComplete() 方法回调中调用此无状态小部件 (HomePage) 的重建?

【问题讨论】:

重建这个无状态小部件有什么意义? 你看过这个:***.com/questions/49804891/… 这个例子只是一个最小的例子。在我的实际应用程序中,我正在构建函数 (FutureBuilder) 中向 Web 服务器发出获取请求,并且我知道 Web 服务数据更新 之后 我离开第二页,所以我想要那个数据反映在第一页。 我认为这个例子行不通,坚果。它需要StatefulWidget,而我的实际实现没有要更新的状态。除非我能够在不实际更新状态值的情况下任意调用setState(可以吗?)。 我不知道为什么这个问题被否决了?它满足质量 Stack Overflow 问题的所有要求。 【参考方案1】:

您可以按如下方式强制重建小部件树:

class RebuildController   
  final GlobalKey rebuildKey = GlobalKey();
  
  void rebuild() 
    void rebuild(Element el) 
      el.markNeedsBuild();
      el.visitChildren(rebuild);
    
    (rebuildKey.currentContext as Element).visitChildren(rebuild);
  



class RebuildWrapper extends StatelessWidget  
  
  final RebuildController controller;
  final Widget child;

  const RebuildWrapper(Key? key, required this.controller, required this.child) : super(key: key);

  @override
  Widget build(BuildContext context) => Container(
    key: controller.rebuildKey,
    child: child,
  );


在你的情况下,

import 'package:flutter/material.dart';

void main() 
  runApp(MyApp());


class MyApp extends StatelessWidget 
  final RebuildController controller = RebuildController();

  MyApp(Key? key) : super(key: key);

  @override
  Widget build(BuildContext context) 
    return MaterialApp(
      title: 'Flutter Demo',
      theme: ThemeData(
        primarySwatch: Colors.blue,
      ),
      home: RebuildWrapper(
        controller: controller,
        child: HomePage(
          rebuildController: controller,
        ),
      ),
    );
  


class HomePage extends StatelessWidget 

  static const name = 'Home Page';
  final RebuildController rebuildController;

  const HomePage(Key? key, required this.rebuildController) : super(key: key);

  @override
  Widget build(BuildContext context) 
    print('Hello there!');
    return Scaffold(
      body: Center(
        child: MaterialButton(
          color: Colors.blue,
          textColor: Colors.white,
          child: const Text(name),
          onPressed: () 
            Navigator.push(
              context,
              MaterialPageRoute(builder: DetailsPage.builder),
            ).whenComplete(rebuildController.rebuild);
          ,
        ),
      ),
    );
  
  


class DetailsPage extends StatelessWidget 
  static const name = 'Details Page';
  static WidgetBuilder builder = (BuildContext _) => const DetailsPage();

  const DetailsPage(Key? key) : super(key: key);

  @override
  Widget build(BuildContext context) 
    return Scaffold(
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: [
            const Text(name),
            MaterialButton(
              color: Colors.blue,
              textColor: Colors.white,
              child: const Text('Go Back'),
              onPressed: () => Navigator.pop(context),
            ),
          ],
        ),
      ),
    );
  


class RebuildController   
  final GlobalKey rebuildKey = GlobalKey();
  
  void rebuild() 
    void rebuild(Element el) 
      el.markNeedsBuild();
      el.visitChildren(rebuild);
    
    (rebuildKey.currentContext as Element).visitChildren(rebuild);
  



class RebuildWrapper extends StatelessWidget  
  
  final RebuildController controller;
  final Widget child;

  const RebuildWrapper(Key? key, required this.controller, required this.child) : super(key: key);

  @override
  Widget build(BuildContext context) => Container(
    key: controller.rebuildKey,
    child: child,
  );


但是强制重建无状态小部件是不自然的,因为它们不应该被重建。您应该使用stateful widget 或其他state management 解决方案,以便您的主页仅在有意义状态更改时更新。

来源 - this answer

【讨论】:

这是有道理的。我同意你的看法。如果不使用真正的状态管理,这个问题确实没有解决方案。不过感谢黑客!这真的很有趣 我的荣幸!我在考虑 RepaintBoundary 但它不会重建你想要的行为的小部件。

以上是关于如何调用无状态小部件的重建?的主要内容,如果未能解决你的问题,请参考以下文章

在 Flutter 中使用 BLoC - 在有状态小部件与无状态小部件中的使用

从无状态小部件调用有状态小部件的 setState

如何在有状态或无状态小部件之外访问 BuildContext?

Flutter:了解无状态/有状态小部件如何布局其子级?

Flutter 状态管理(BloC):无状态与有状态小部件

为啥在软件开发中使用无状态的多于有状态的小部件? [关闭]