1

我正在开发一个 Flutter 应用程序,它充当客户端,通过 API 连接到服务器。应用程序发出请求,并根据响应进展状态。

我的问题如下:我可以提出请求,然后根据响应更新 UI 或打开新页面吗?

我用过FutureBuilder如下图。问题是FutureBuilder需要返回 UI。就我而言,如果响应正常,我想打开一个新页面(参见 //todo 行)。我尝试使用Navigator.pushReplacement,但它并没有真正起作用。有任何想法吗?

import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart';
import 'dart:async';
import 'package:flutter/rendering.dart';
import 'model.dart';

class Start extends StatefulWidget {
  final String title;

  Start({Key key, @required this.title}) : super(key: key);

  @override
  State<StatefulWidget> createState() => new StartState();
}

class StartState extends State<Start> {

  Future<StartReply> _startReply;

  _makeRequest() {
    setState(() {
      _startReply = ...; // actual API request
    });
  }

  @override
  Widget build(BuildContext context) {
    return MaterialApp(
        title: widget.title,
        home: Scaffold(
            appBar: AppBar(
              title: Text(widget.title),
              leading: IconButton(
                  icon: Icon(Icons.arrow_back),
                  onPressed: () => Navigator.of(context).pop(false)
              ),
            ),
            body: Center(
                child: FutureBuilder(
                  future: _startReply,
                  builder: (context, snapshot) {
                    if(snapshot.connectionState == ConnectionState.none) {
                      return ElevatedButton(
                          onPressed: _makeRequest,
                          child: Text("Make request")
                      );
                    } else if(snapshot.connectionState == ConnectionState.done && snapshot.hasData) {
                      // todo open page here
                      return Text('Started!', style: TextStyle(color: Colors.green, fontStyle: FontStyle.italic));
                    } else if(snapshot.hasError) {
                      debugPrint('StartReply: ${snapshot.data}');
                      return Text('Error (${snapshot.error})', style: TextStyle(color: Colors.red, fontStyle: FontStyle.italic));
                    } else {
                      return CircularProgressIndicator();
                    }
                  }
                )
            )
        )
    );
  }
}
4

1 回答 1

2

是的,如果您想要根据异步任务更改 UI 之外的任何其他操作,则不应使用 FutureBuilder。您应该管理自己的异步。这里有一些代码可以帮助您入门:


class MyWidget extends StatefulWidget {
  @override
  _MyWidgetState createState() => _MyWidgetState();
}

class _MyWidgetState extends State<MyWidget> {
  
  bool loaded;
  
  @override
  void initState() {
    super.initState();
    
    asyncInit();
  }
  
  Future<void> asyncInit() async {
    final response =
    await doTheNetworkRequest() //imagine that this was an http request
    if (yes) {
      setState(() {
        loaded = true;
      });
    } else {
      Navigator.of(context).push(...);
    }
  }
  
  @override
  Widget build(BuildContext context) {
    return loaded == true ? Text('Loaded') : Text('Loading');
  }
}
于 2021-05-02T09:56:45.707 回答