ScopedModel - как пройти более одной модели - PullRequest
0 голосов
/ 21 апреля 2020

В примере мы передаем одну модель, как эта

return ScopedModel<CounterModel>(
  model: CounterModel(), // only one model here
  child: MaterialApp(
    title: 'Scoped Model Demo',
    home: CounterHome('Scoped Model Demo'),
  ),
);

Как передать некоторые модели вместо одной модели? Чтобы потом их использовать таким образом

Widget build(BuildContext context) {
    final username =
      ScopedModel.of<UserModel>(context, rebuildOnChange: true).username;
    final counter =
      ScopedModel.of<CounterModel>(context, rebuildOnChange: true).counter;

    return Text(...);
}

1 Ответ

1 голос
/ 22 апреля 2020

Вы можете скопировать вставить и запустить полный код ниже. Вы можете сослаться на это https://newcodingera.com/scoped-model-in-flutter/ Вы можете передать свои три модели и обернуть их вложенными

Фрагмент кода

void main() {
  runApp(MyApp(
    counterModel: CounterModel(),
    userModel: UserModel('Brian'),
    dataModel: DataModel('this is test'),
  ));
}

class MyApp extends StatelessWidget {
  final CounterModel counterModel;
  final UserModel userModel;
  final DataModel dataModel;

  const MyApp({
    Key key,
    @required this.counterModel,
    @required this.userModel,
    @required this.dataModel,
  }) : super(key: key);

  @override
  Widget build(BuildContext context) {
    // At the top level of our app, we'll, create a ScopedModel Widget. This
    // will provide the CounterModel to all children in the app that request it
    // using a ScopedModelDescendant.
    return ScopedModel<DataModel>(
      model: dataModel,
      child: ScopedModel<UserModel>(
        model: userModel,
        child: ScopedModel<CounterModel>(
          model: counterModel,

Рабочая демонстрация

enter image description here

Полный код

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

void main() {
  runApp(MyApp(
    counterModel: CounterModel(),
    userModel: UserModel('Brian'),
    dataModel: DataModel('this is test'),
  ));
}

class MyApp extends StatelessWidget {
  final CounterModel counterModel;
  final UserModel userModel;
  final DataModel dataModel;

  const MyApp({
    Key key,
    @required this.counterModel,
    @required this.userModel,
    @required this.dataModel,
  }) : super(key: key);

  @override
  Widget build(BuildContext context) {
    // At the top level of our app, we'll, create a ScopedModel Widget. This
    // will provide the CounterModel to all children in the app that request it
    // using a ScopedModelDescendant.
    return ScopedModel<DataModel>(
      model: dataModel,
      child: ScopedModel<UserModel>(
        model: userModel,
        child: ScopedModel<CounterModel>(
          model: counterModel,
          child: MaterialApp(
            title: 'Scoped Model Demo',
            home: CounterHome('Scoped Model Demo'),
          ),
        ),
      ),
    );
  }
}

// Start by creating a class that has a counter and a method to increment it.
//
// Note: It must extend from Model.
class CounterModel extends Model {
  int _counter = 0;

  int get counter => _counter;

  void increment() {
    // First, increment the counter
    _counter++;

    // Then notify all the listeners.
    notifyListeners();
  }
}

class UserModel extends Model {
  String _username;

  UserModel(String username) : _username = username;

  String get username => _username;

  set username(String newName) {
    _username = newName;
    notifyListeners();
  }
}

class DataModel extends Model {
  String _data;

  DataModel(String data) : _data = data;

  String get data => _data;

  set data(String newData) {
    _data = newData;
    notifyListeners();
  }
}

class CounterHome extends StatelessWidget {
  final String title;

  CounterHome(this.title);

  @override
  Widget build(BuildContext context) {
    final counter =
        ScopedModel.of<CounterModel>(context, rebuildOnChange: true).counter;
    final userModel = ScopedModel.of<UserModel>(context, rebuildOnChange: true);
    final dataModel = ScopedModel.of<DataModel>(context, rebuildOnChange: true);

    return Scaffold(
      appBar: AppBar(
        title: Text(title),
      ),
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: <Widget>[
            Text('${userModel.username} pushed the button this many times:'),
            Text('${dataModel.data} From data model'),
            // Create a ScopedModelDescendant. This widget will get the
            // CounterModel from the nearest parent ScopedModel<CounterModel>.
            // It will hand that CounterModel to our builder method, and
            // rebuild any time the CounterModel changes (i.e. after we
            // `notifyListeners` in the Model).
            Text('$counter', style: Theme.of(context).textTheme.display1),
            Padding(
              padding: const EdgeInsets.all(8.0),
              child: RaisedButton(
                child: Text('Change Username'),
                onPressed: () {
                  userModel.username = 'Suzanne';
                },
              ),
            ),
            Padding(
              padding: const EdgeInsets.all(8.0),
              child: RaisedButton(
                child: Text('Change Data'),
                onPressed: () {
                  dataModel.data = 'data changed';
                },
              ),
            )
          ],
        ),
      ),
      // Use the ScopedModelDescendant again in order to use the increment
      // method from the CounterModel
      floatingActionButton: ScopedModelDescendant<CounterModel>(
        builder: (context, child, model) {
          return FloatingActionButton(
            onPressed: model.increment,
            tooltip: 'Increment',
            child: Icon(Icons.add),
          );
        },
      ),
    );
  }
}
...