【问题标题】:Flutter : Prevent FutureBuilder always refresh every change screenFlutter:防止 FutureBuilder 总是刷新每个更改屏幕
【发布时间】:2020-05-25 08:56:02
【问题描述】:

我有 FutureBuilder 从 API 和代码中获取用户配置文件来获取这样的用户:

 Future<List<UserModel>> getUserByUsername({@required String username}) async {
    try {
      final response =
          await _client.get("$_baseUrl/getUserByUsername?username=$username");
      final Map<String, dynamic> responseJson = json.decode(response.body);
      if (responseJson["status"] == "ok") {
        List userList = responseJson['data'];
        final result = userList
            .map<UserModel>((json) => UserModel.fromJson(json))
            .toList();
        return result;
      } else {
        throw CustomError(responseJson['message']);
      }
    } catch (e) {
      return Future.error(e.toString());
    }
  }

如果您可以在上面的 GIF 中看到,My FutureBuilder 位于 BottomNavigationBar。每次我从 BottomNavigationBar 更改屏幕/页面并回到我的 FutureBuilder 总是刷新!

如何将其修复为仅刷新一次?

主屏幕

class _HomeScreenState extends State<HomeScreen> {
  @override
  Widget build(BuildContext context) {
    final username = Provider.of<SharedPreferencesFunction>(context).username;
    return SingleChildScrollView(
      child: Column(
        crossAxisAlignment: CrossAxisAlignment.stretch,
        mainAxisSize: MainAxisSize.min,
        children: <Widget>[
          CardTime(),
          FutureBuilder(
            future: userApi.getUserByUsername(username: username),
            builder: (BuildContext context,
                AsyncSnapshot<List<UserModel>> snapshot) {
              if (snapshot.connectionState == ConnectionState.done) {
                if (snapshot.hasError) {
                  return Center(
                    child: Text(
                      snapshot.error.toString(),
                    ),
                  );
                } else {
                  final user = snapshot.data[0];
                  return CardProfil(
                    imageUrl: "${userApi.baseImageUrl}/${user.fotoUser}",
                    detailProfil: [
                      Text(
                        user.namaUser,
                        style: TextStyle(fontWeight: FontWeight.bold),
                      ),
                      Text(user.idDevice),
                    ],
                  );
                }
              } else {
                return Center(
                  child: CircularProgressIndicator(),
                );
              }
            },
          ),
        ],
      ),
    );
  }
}

共享偏好功能

import 'package:flutter/cupertino.dart';
import 'package:shared_preferences/shared_preferences.dart';

class SharedPreferencesFunction extends ChangeNotifier {
  SharedPreferencesFunction() {
    initialSharedPreferences();
    getUsername();
  }
  String _username;
  String get username => _username;

  void initialSharedPreferences() {
    getUsername();
  }

  Future updateUsername(String username) async {
    SharedPreferences pref = await SharedPreferences.getInstance();
    await pref.setString("username", username);
    //! It's Important !!! After update / remove sharedpreferences  , must called getUsername() to updated the value.
    getUsername();
    notifyListeners();
  }

  Future removeUsername() async {
    SharedPreferences pref = await SharedPreferences.getInstance();
    final result = await pref.remove("username");
    //! It's Important !!! After update / remove sharedpreferences  , must called getUsername() to updated the value.
    getUsername();
    print(result);
    notifyListeners();
  }

  Future getUsername() async {
    SharedPreferences pref = await SharedPreferences.getInstance();
    final result = pref.getString("username");
    _username = result;
    notifyListeners();
  }
}

final sharedpref = SharedPreferencesFunction();

更新问题

我已经尝试过 Initialize FutureBuilder 并使用 initStatedidChangeDependencies 。但新问题是,如果我在 initState 内初始化,我的配置文件不会重建,因为 Provider listen=false。 如果我使用 didChangeDependencies 我的 FutureBuilder 仍然会在每次更改屏幕时刷新。 有什么问题吗?

使用 initState

使用 didChangeDependencies

【问题讨论】:

标签: flutter dart


【解决方案1】:

initStatedidChangeDependencies 期间初始化Future

class _HomeScreenState extends State<HomeScreen> {
  Future<List<UserModel>> user;

  @override
  void initState() { 
    super.initState();

    // must use listen false here
    final username = Provider.of<SharedPreferencesFunction>(context, listen: false).username;
    user = userApi.getUserByUsername(username: username);
  }

  @override
  void didChangeDependencies() {
    super.didChangeDependencies();

    final username = Provider.of<SharedPreferencesFunction>(context).username;
    user = userApi.getUserByUsername(username: username);
  }

  @override
  Widget build(BuildContext context) {
    return SingleChildScrollView(
      child: Column(
        crossAxisAlignment: CrossAxisAlignment.stretch,
        mainAxisSize: MainAxisSize.min,
        children: <Widget>[
          FutureBuilder(
            future: user,
            builder: (context, snapshot) {
              // ...
            },
          ),
        ],
      ),
    );
  }
}

【讨论】:

  • 这不行,我已经在使用 initStatedidChangeDepedencies。如果我使用 initState,则找不到我的配置文件,因为 Provider 听 =false i.stack.imgur.com/Hf3p2.gif。如果我使用 didChangeDependencies 我的 FutureBuilder 仍然刷新每次我更改屏幕 i.stack.imgur.com/Hf3p2.gif 。我的代码有问题吗?
  • 两个链接上的 gif 相同。
  • 但是您可以检查一些事情。使用didChangeDependencies,您是在SharedPreferencesFunction 中为每个选项卡更改调用notifyListeners,还是在小部件树中调用setState?您可以尝试仅隔离 HomeScreen 并查看它是否仍会在 setState 等上刷新。
  • 我添加了我的共享首选项功能
  • 你需要等待getUsername,为什么不直接设置_username。您为setUsernameremoveUsername 调用了两次notifyListeners
【解决方案2】:

使用IndexedStack 作为标签栏的父级。

【讨论】:

    【解决方案3】:

    我遇到了类似的情况,并在每个视图/页面/标签栏视图/小部件/子项上使用AutomaticKeepAliveClientMixin,以保持每次我在标签栏来回切换时页面不刷新。

    class YourClass extends StatefulWidget {
      YourClass({
        Key key
      }): super(key key);
    
      @override
      _YourClassState createState() => _YourClassState();
    }
    
    // Must include AutomaticKeepAliveClientMixin
    class _YourClassState extends State<YourClass> with AutomaticKeepAliveClientMixin {
      Future resultGetData;
    
      void getData() {
        setState(() {
          resultGetData = getDataFromAPI();
        });
      }
    
      // Must include
      @override
      bool get wantKeepAlive => true;
    
      @override
      void initState() {
        getData();
        super.initState();
      }
    
      @override
      Widget build(BuildContext context) {
        super.build(context); // Must include
        return FutureBuilder(
          future: resultGetAllByUserIdMCId,
          builder: (context, snapshot) {
            // ...
            // Some Code
            // ...
          }
        );
      }
    
    }
    

    如果您想刷新数据,您可以使用运行getData() 函数的RefreshIndicator。将此代码放入FutureBuilderkey: PageStorageKey(widget.key) 会将卷轴保存在您离开的完全相同的位置。

    return RefreshIndicator(
      onRefresh: () async {
        getData();
      },
      child: ListView.separated(
        key: PageStorageKey(widget.key),
        itemCount: data.length,
        separatorBuilder: (BuildContext context, int index) {
          return Divider(height: 0);
        },
        itemBuilder: (context, index) {
          return ...;
        },
      ),
    );
    

    【讨论】:

      猜你喜欢
      • 2021-04-19
      • 2020-05-24
      • 2020-06-02
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多