Flutter Provider's state error

一个应该很常见的场景,一个接口请求list,初期是空的,请求成功后,需要更新list,然后UI会重新build,但是这个时候,如果你想在请求的时候,手动设置 state 为 loading,表示正在请求,这个时候就会报错。
使用的技术栈:

  • riverpodhooks
  • riverpod_annotation
  • freezed
  • flutter_riverpod

Tried to modify a provider while the widget tree was building.
If you are encountering this error, chances are you tried to modify a provider
以后会尽量用日语来写,为了练习日语。

EN

This error happens when you try to change a provider’s state inside a widget’s life cycle (for example, inside build, initState, dispose, didUpdateWidget, or didChangeDependencies). Riverpod (or Flutter’s provider package) requires that you not directly modify state during these phases, because it can lead to inconsistent or unintended UI states.


Why this happens

In Flutter, widgets can rebuild multiple times for all kinds of reasons (e.g., a parent changed, you rotated the device, etc.). If you modify the state in the middle of a build:

  • You risk triggering more rebuilds before the UI has finished its current update.
  • You could cause inconsistent states if multiple widgets depend on the same provider but rebuild at slightly different times.

As a result, Riverpod/Provider actively prevents you from modifying a provider during these critical moments and throws the error you’re seeing.


How to fix

1. Move provider modifications outside widget life cycles

The simplest and often best solution is to initiate the provider update in response to a user action or an asynchronous event (like an API call), rather than in build or initState. For example, call:

ElevatedButton(
  onPressed: () {
    // This is allowed because it’s in a callback
    ref.read(yourProvider.notifier).updateSomething();
  },
  child: Text('Update Provider'),
);

This way, the provider is updated after the build, not during it.

2. Use a post-build callback to delay the update

If you truly need to modify a provider as soon as the widget builds (for example, you need an initial fetch), you can schedule the update after the first build completes by using a callback such as Future(() { ... }) or WidgetsBinding.instance.addPostFrameCallback. For instance:


void initState() {
  super.initState();

  // Schedules the provider update to run AFTER the current frame
  WidgetsBinding.instance.addPostFrameCallback((_) {
    ref.read(yourProvider.notifier).updateSomething();
  });
}

This tells Flutter to wait until the current widget tree build is finished before modifying the provider’s state, which avoids the error.


Recap

  1. Never call ref.read(...) or context.read(...) to change a provider directly inside build, initState, or other life-cycle methods.
  2. Instead, use an external event like a button press, or schedule the update to happen after the current build (via a future or addPostFrameCallback).

Following these guidelines ensures your app state remains consistent and helps avoid tricky rebuild and state management issues.

JP

Flutter()Riverpod()のProviderを使用(しよう)して開発(かいはつ)している(さい)に、
「Tried to modify a provider while the widget tree was building…」という
Error(エラー)()場合(ばあい)があります。
これはWidget(ウィジェット)ライフサイクル()(Life-cycle)――
たとえばbuildinitStatedisposedidUpdateWidgetdidChangeDependenciesなどのタイミング()(Timing)で
Providerの状態(じょうたい)(State)を直接(ちょくせつ)変更(へんこう)してしまうと発生(はっせい)する問題(もんだい)です。


なぜこのErrorが()きるのか

Widget(ウィジェット)描画(びょうが)build処理(しょり)頻繁(ひんぱん)()ばれます。
その途中(とちゅう)Provider()更新(こうしん)すると、
ほかのWidget(ウィジェット)との整合(せいごう)(くず)れ、
画面(がめん)UI()(User Interface))が不正(ふせい)状態(じょうたい)になる(おそ)れがあります。
そのため、内部(ないぶ)再描画(さいびょうが)制御(せいぎょ)(おこな)っている
Riverpod()Provider()は、
こうした誤用(ごよう)()きた(さい)
警告(けいこく)(Warning)としてこのErrorを投げ()てくれます。


対策(たいさく)

  1. 推奨(すいしょう)ライフサイクル()内(ない)での変更(へんこう)を避(さ)ける

    • たとえばonPressedなど、ユーザー()(User)の操作(そうさ)
      非同期(ひどうき)(Asynchronous)のイベント()(Event)が完了(かんりょう)した(あと)
      Providerを更新(こうしん)するように設計(せっけい)しましょう。
    // ButtonのonPressed内でProviderを更新
    ElevatedButton(
      onPressed: () {
        ref.read(yourProvider.notifier).updateSomething();
      },
      child: Text('Update Provider'),
    );
  2. どうしても初期化(しょきか)で設定(せってい)が必要(ひつよう)な場合(ばあい)

    • initStatedidChangeDependenciesなどですぐにProviderを更新(こうしん)したい場合(ばあい)は、
      ビルド()(Build)が完了(かんりょう)してから実行(じっこう)するよう
      WidgetsBinding.instance.addPostFrameCallbackなどを使(つか)い、
      更新(こうしん)タイミング()(おく)らせる方法(ほうほう)があります。
    
    void initState() {
      super.initState();
    
      WidgetsBinding.instance.addPostFrameCallback((_) {
        // ビルド完了後にProviderを更新
        ref.read(yourProvider.notifier).updateSomething();
      });
    }

まとめ

  • Providerの状態(じょうたい)直接(ちょくせつ)変(か)える処理(しょり)は、
    buildinitStatedisposeなど
    Widget(ウィジェット)ライフサイクル()内で(おこな)うと
    UI()不安定(ふあんてい)になる(おそ)れがあり、Errorが発生(はっせい)します。
  • 基本(きほん)はユーザー操作(そうさ)非同期(ひどうき)処理(しょり)の完了(かんりょう)後(あと)など、
    ライフサイクル()(Life-cycle)の(そと)でProviderを更新(こうしん)するよう注意(ちゅうい)しましょう。
  • 初期化(しょきか)時やどうしても必要(ひつよう)なタイミングでは、
    WidgetsBinding.instance.addPostFrameCallbackFuture(() { … })使(つか)い、
    一度(いちど)ビルドが(かん)了してから更新(こうしん)することで
    安全(あんぜん)状態(じょうたい)(たも)つことができます。

こうした対策(たいさく)()ることで、
「Tried to modify a provider while the widget tree was building…」という
Errorを回避(かいひ)でき、
安定(あんてい)したUI()(たも)つことができます。

JP 2

Riverpod()Hook(フック)(Hook)を使用(しよう)して開発(かいはつ)している場合(ばあい)でも、
基本(きほん)前述(ぜんじゅつ)注意点(ちゅういてん)(Point)と(おな)じです。
つまり、Widget(ウィジェット)build()(ちゅう)ライフサイクル()(Life-cycle)の処理(しょり)途中(とちゅう)で、
Provider()直接(ちょくせつ)更新(こうしん)しないようにすることが大切(たいせつ)です。


HookConsumerWidgetの場合

Riverpod()Flutter()のHookを()合わせ()(Combine)て使(つか)(さい)は、
画面(がめん)(Screen/Page)を定義(ていぎ)する(とき)HookConsumerWidget継承(けいしょう)(Inherit)すると便利(べんり)です。
以下(いか)(れい)では、
useEffect利用(りよう)(Use)して、初回(しょかい)ビルド完了(かんりょう)(Complete)(あと)
Provider()状態(じょうたい)変更(へんこう)(Update)している(れい)です。

import 'package:flutter/material.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:flutter_hooks/flutter_hooks.dart';

// なにかしらのProviderを定義(StateNotifierProviderなど)
final yourProvider = StateNotifierProvider<YourNotifier, YourState>((ref) {
  return YourNotifier();
});

class YourNotifier extends StateNotifier<YourState> {
  YourNotifier() : super(YourState());

  void updateSomething() {
    // 状態を更新する処理
  }
}

class YourState {
  // 状態管理のためのプロパティなど
}

class ExamplePage extends HookConsumerWidget {
  const ExamplePage({Key? key}) : super(key: key);

  
  Widget build(BuildContext context, WidgetRef ref) {
    // 状態を監視
    final state = ref.watch(yourProvider);

    // build完了後に一度だけProviderを更新したい場合などにuseEffectを使用
    useEffect(() {
      Future.microtask(() {
        ref.read(yourProvider.notifier).updateSomething();
      });
      // dispose時の処理が必要なければnull
      return null;
    }, []);

    return Scaffold(
      appBar: AppBar(
        title: Text('HookConsumerWidget Example'),
      ),
      body: Center(
        child: Text('State: $state'),
      ),
    );
  }
}

ポイント

  1. build()(なか)直接(ちょくせつ)Providerを更新しない

    • かわりにuseEffectFuture(() { … })WidgetsBinding.instance.addPostFrameCallbackなどを使(つか)って
      ビルド()(Build)が()わった(あと)にProviderの状態(じょうたい)変更(へんこう)します。
  2. ユーザー()(User)の操作(そうさ)イベント()(Event)に(おう)じて更新する

    • onPressedなどのコールバック()(Callback)の(なか)更新(こうしん)する(ぶん)には問題(もんだい)ありません。

まとめ

  • HookConsumerWidget()使(つか)った場合(ばあい)も、
    ライフサイクル()build()など)の途中(とちゅう)
    Providerを変更(へんこう)しないように注意(ちゅうい)することが重要(じゅうよう)です。
  • useEffect使(つか)うことで初回(しょかい)ビルド完了(かんりょう)後(あと)にProvider()更新(こうしん)でき、
    不要(ふよう)タイミング()(Timing)での変更(へんこう)()けられます。
  • ユーザー()(User)の操作(そうさ)(おう)じて状態(じょうたい)変更(へんこう)する処理(しょり)同様(どうよう)で、
    onPressedなどビルド()(そと)タイミング()更新(こうしん)すれば安全(あんぜん)状態(じょうたい)(たも)てます。

これらを徹底(てってい)することで、
「Tried to modify a provider while the widget tree was building…」というError(エラー)発生(はっせい)(ふせ)ぎ、
安定(あんてい)したUI()(User Interface)を構築(こうちく)(Build)できます。