Как я могу инициализировать провайдера?

В моем приложении реализован поставщик управления состоянием. Теперь мне нужно добавить некоторые данные в класс после загрузки экрана.

Как мне этого добиться?

stepInfo.addToList = new VaccStep (); // Нужно вызвать его один раз после загрузки экрана.

Я пытался вызвать этот метод из initState, но он дает ошибку!

    class AdminAddVaccination extends StatefulWidget {
      @override
      State createState() => new AdminAddVaccinationState();
    }

    class AdminAddVaccinationState extends State<AdminAddVaccination> {

      @override
      void initState() {
        super.initState();
        var stepInfo = Provider.of<StepInfo>(context); // ERROR!!
        stepInfo.addToList = new VaccStep(); // ERROR!!
      }

      Widget build(BuildContext context) {
        return new ChangeNotifierProvider(
          builder: (context) => StepInfo(),
          child: ScreenBody(),
        );
      }
    }

    class ScreenBody extends StatelessWidget {


      @override
      Widget build(BuildContext context) {
        var stepInfo = Provider.of<StepInfo>(context);

        return new Scaffold(
            resizeToAvoidBottomPadding: false,
            key: stepInfo.scaffoldKey,
            body: new GestureDetector(
                onTap: () {
                  FocusScope.of(context).requestFocus(new FocusNode());
                },
                child: new SafeArea(
                  top: true,
                  bottom: false,
                  child: new Stack(children: <Widget>[
                    new Opacity(
                      opacity: 0.04,
                      child: new Image.asset(
                        "assets/userProfile/heartBeat.png",
                        fit: BoxFit.cover,
                        height: 250.0,
                      ),
                    ),
                    new Column(
                      mainAxisAlignment: MainAxisAlignment.start,
                      crossAxisAlignment: CrossAxisAlignment.center,
                      children: <Widget>[
                        new Container(
                          color: primaryGreen,
                          width: double.infinity,
                          height: 65.0,
                          child: new Stack(
                            children: <Widget>[
                              new Align(
                                  alignment: Alignment.center,
                                  child: stepInfo.loading
                                      ? JumpingText('......')
                                      : new Container()),
                              new Align(
                                alignment: Alignment.centerLeft,
                                child: new Padding(
                                  padding: EdgeInsets.only(top: 5.0, left: 20.0),
                                  child: new InkWell(
                                    child: new Container(
                                      child: Icon(
                                        Icons.arrow_back,
                                        color: Colors.white,
                                        size: 30.0,
                                      ),
                                    ),
                                    onTap: () {
                                      Navigator.pop(context);
                                    },
                                  ),
                                ),
                              ),
                            ],
                          ),
                        ),
                        new Padding(
                          padding: EdgeInsets.only(top: 0.0),
                          child: new Material(
                            elevation: 1.0,
                            color: Colors.transparent,
                            child: new Container(
                              color: borderColor,
                              width: double.infinity,
                              height: 5.0,
                            ),
                          ),
                        ),
                        VaccName(),
                      ],
                    ),
                    ItemListing(),
                    AddStep(),
                  ]),
                )));
      }
    }

    Error!!
    flutter: The following ProviderNotFoundError was thrown building 
    Builder:
    flutter: Error: Could not find the correct Provider<StepInfo> above this AdminAddVaccination Widget
    flutter:
    flutter: To fix, please:
    flutter:
    flutter:   * Ensure the Provider<StepInfo> is an ancestor to this AdminAddVaccination Widget
    flutter:   * Provide types to Provider<StepInfo>
    flutter:   * Provide types to Consumer<StepInfo>
    flutter:   * Provide types to Provider.of<StepInfo>()
    flutter:   * Always use package imports. Ex: `import 'package:my_app/my_code.dart';

4 ответа

Просто добавьте конструктор в свой провайдер:

class StepInfo extends ChangeNotifier {

   StepInfo() {
      this.addToList = new VaccStep();
   }

   [...]

}

Вы должны установить listen:false и некоторую задержку на initstate

Provider.of<StepInfo>(context, listen: false);

Future.delayed(Duration(milliseconds: 100)).then((_) {
stepInfo.addToList = new VaccStep();
});

то же самое в этом случае или в этом

addPostFrameCallback()

Решением этой проблемы является реализация addPostFrameCallback, планирующая обратный вызов в конце кадра. Он выполняет метод, когда контекст доступен.

      @override
void initState() {
  super.initState();
  WidgetsBinding.instance.addPostFrameCallback((_) {
      var stepInfo = Provider.of<StepInfo>(context,listen:false);
      stepInfo.addToList = new VaccStep(); 
  });
}

Измените метод initState() и build() в классе AdminAddVaccination, как показано ниже:

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

  @override
  Widget build(BuildContext context) {
    var stepInfo = new StepInfo();
    stepInfo.addToList = new VaccStep();
    return new ChangeNotifierProvider<StepInfo>(
      builder: (context) => stepInfo,
      child: ScreenBody(),
    );
  }
Другие вопросы по тегам