State management

宣言的とは

  • flutterは宣言的でUI=f(state)という形で成り立っている
    • buildのメソッドに状態を渡してUIができる
  • 状態が変わったら自動的にUIが再描画される

ephemeral stateとapp state

  • ユーザが管理すべきなのは「任意の時点でUIを再構築するために必要なあらゆるデータ」
  • このような状態はephemeral stateとapp stateに分けられる

ephemeral state

  • UI stateやlocal stateとも呼ばれる
  • 1つのWidgetに納められる状態のこと
    • PageViewでの現在のページや複雑なanimationの現在の進行状況、BottomNavigationBarの現在選択されたtabなど
  • このような場合は状態管理テクニックを使わず、StatefiuWidgetを使えば良い

App State

  • 以下のようなapp全体で共有したい状態のこと
    • User設定
    • Login情報

明確な境界はない?

  • StatefulWidgetを使えば全ての状態を管理できるので
  • また判断によってephemeral stateapp stateとなることもある
  • というわけでケースバイケースなので、明確な区切りはない

Simple app state management

どんなアプリか

  • ここではproviderパッケージを使いシンプルなアプリの状態管理をする
  • 以下のようなアプリ
    • catalogとcartの2つのscreenを持つ
    • catalogはcustom app barを持ち、scroll viewも持つ
  • 以下がWidget Tree my image

stateの引き上げ

  • flutterでは状態を使用するWidgetの上に配置する
  • これはUIの変更は再構築となるので、MyCart.updateWith(new)ということはできない
  • つまりWidgetのメソッドを外部から呼び出して操作することはできない
// BAD: DO NOT DO THIS
// CartWidgetがitem一覧を持っていて、そこに更新したいitemを渡す感じ
void myTapHandler() {
  var cartWidget = somehowGetMyCartWidget();
  cartWidget.updateWith(item);
} 
  • 仮に動くようにできたとしても、以下のようなものにも対応しないといけない
// BAD: DO NOT DO THIS
Widget build(BuildContext context) {
  return SomeWidget(
    // The initial state of the cart.
  );
}

void updateWith(Item item) {
  // Somehow you need to change the UI from here.
}
  • UIの現状を考慮しデータ更新していくのはバグを生みかねない
  • flutterではcontentが変わったタイミングでWidgetが更新される
  • というわけで以下のようにすべき
// GOOD
// まず親Widgetでcartの状態管理をする(cartModelを更新する)
void myTapHandler(BuildContext context) {
  var cartModel = somehowGetMyCartModel(context);
  cartModel.add(item);
} 

// GOOD
// その上でCartWidgetないではそのモデルを参照する
Widget build(BuildContext context) {
  var cartModel = somehowGetMyCartModel(context);
  return SomeWidget(
    // Just construct the UI once, using the current state of the cart.
    // ···
  );
}

stateへのアクセス

  • catalogのitemをユーザがタップしたらcartに加わりますが、cartはMyListItemの上にいるのにどうやってアクセスするのでしょうか?
  • 以下の感じでcallbackで上から指定することができる
@override
Widget build(BuildContext context) {
  return SomeWidget(
    // Construct the widget, passing it a reference to the method above.
    MyListItem(myTapCallback),
  );
}

void myTapCallback(Item item) {
  print('user tapped on $item');
} 
  • 上記でも動くがいろいろな箇所で指定しないといけない
  • flutterにはInheritedWidgetなどで孫に渡したりできるが、より使いやすいproviderを使うことにする

ChangeNotifier

  • ChangeNotifierはそのlistenerに変更を通知する
  • providerを使ったCartModelは以下の感じ
class CartModel extends ChangeNotifier {
  /// Internal, private state of the cart.
  final List<Item> _items = [];

  /// An unmodifiable view of the items in the cart.
  UnmodifiableListView<Item> get items => UnmodifiableListView(_items);

  /// The current total price of all items (assuming all items cost $42).
  int get totalPrice => _items.length * 42;

  /// Adds [item] to cart. This and [removeAll] are the only ways to modify the
  /// cart from the outside.
  void add(Item item) {
    _items.add(item);
    // This call tells the widgets that are listening to this model to rebuild.
    notifyListeners();
  }

  /// Removes all items from the cart.
  void removeAll() {
    _items.clear();
    // This call tells the widgets that are listening to this model to rebuild.
    notifyListeners();
  }
} 
  • 基本的にビジネスロジックにnotifyListenersをつければいい
  • 以下のようにテストもできる
test('adding item increases total cost', () {
  final cart = CartModel();
  final startingPrice = cart.totalPrice;
  var i = 0;
  cart.addListener(() {
    expect(cart.totalPrice, greaterThan(startingPrice));
    i++;
  });
  cart.add(Item('Dash'));
  expect(i, 1);
});

ChangeNotifierProvider

  • ChangeNotifierをその子孫に提供する
  • CartModelの場合、MyCartMyCatalogの両方の上のどこかに置く必要がある
  • というわけでMyAppを以下の感じにする
void main() {
  runApp(
    ChangeNotifierProvider(
      create: (context) => CartModel(),
      child: const MyApp(),
    ),
  );
} 
  • ここでCartModelをインスタンス化していて、lifecycleの管理はproviderがします
  • 複数個渡したいときは以下のようにMultiProviderを使います
void main() {
  runApp(
    MultiProvider(
      providers: [
        ChangeNotifierProvider(create: (context) => CartModel()),
        Provider(create: (context) => SomeOtherClass()),
      ],
      child: const MyApp(),
    ),
  );
} 

Consumer

  • CartModelを使用する側
return Consumer<CartModel>(
  builder: (context, cart, child) {
    return Text('Total price: ${cart.totalPrice}');
  },
); 
  • まずアクセスしたいmodelの型指定が必要(今回はCartModel)
  • ConsumerのbuilderはChangeNotifiernotifyListenersが呼ばれたタイミングで呼ばれる
  • builderの引数は以下
    • context…いつもの
    • cart…ChangeNotifierのインスタンス
    • child…最適化のためにあるもので、Consumer下に重いけどモデルが変わっても変わらないWidgetがいる場合に使う
return Consumer<CartModel>(
  builder: (context, cart, child) => Stack(
    children: [
      // Use SomeExpensiveWidget here, without rebuilding every time.
      if (child != null) child,
      Text('Total price: ${cart.totalPrice}'),
    ],
  ),
  // Build the expensive widget here.
  child: const SomeExpensiveWidget(),
); 
  • ベスプラとしてはConsumerはできる限り奥におきたい

Provider.of

  • cart全削除ボタンのように、アクセスはするけど変更通知が不要な場合は以下のようにできる
Provider.of<CartModel>(context, listen: false).removeAll(); 

使ってみた

まとめ

いろいろなアプローチ

概要

それぞれ