I'm managing the ThemeMode of my flutter application with Riverpod state Provider that works as expected up until I try to read Theme.of(context)
to get ThemeData's current values which causes rebuilding of the widget in excess (13~14 times in a row). So I decided create a provider for ThemeData following Riverpod's repository example but I'm still getting these unecessary rebuilds. How can I prevent these unnecessary riverpod rebuilds to get ThemeData? and why is it happening?
This code is available on github.
main app:
final themeProvider = Provider<ThemeData>(
(ref) => throw UnimplementedError(),
dependencies: const [],
);
void main() {
runApp(const ProviderScope(child: MainApp()));
}
class MainApp extends ConsumerWidget {
const MainApp({super.key});
@override
Widget build(BuildContext context, WidgetRef ref) {
final ThemeMode themeMode = ref.watch(themeModeStateProvider);
if (kDebugMode) {
print("building app");
}
return MaterialApp(
theme: FlexThemeData.light(scheme: FlexScheme.mandyRed),
darkTheme: FlexThemeData.dark(scheme: FlexScheme.mandyRed),
themeMode: themeMode,
builder: (context, child) {
final theme = Theme.of(context);
return ProviderScope(
overrides: [
themeProvider.overrideWithValue(theme),
],
child: child!,
);
},
home: const HomeScreen(),
);
}
}
ThemeMode Provider:
@riverpod
class ThemeModeState extends _$ThemeModeState {
@override
ThemeMode build() {
return ThemeMode.dark;
}
static ThemeMode getSystemTheme(BuildContext context) {
ThemeMode mode = ThemeMode.system;
if (mode == ThemeMode.system) {
if (MediaQuery.of(context).platformBrightness == Brightness.light) {
mode = ThemeMode.light;
} else {
mode = ThemeMode.dark;
}
}
return mode;
}
void toggleThemeMode() {
if (state == ThemeMode.dark) {
state = ThemeMode.light;
} else {
state = ThemeMode.dark;
}
}
}
homescreen:
class HomeScreen extends ConsumerWidget {
static String routeName = "home";
const HomeScreen({Key? key}) : super(key: key);
@override
Widget build(BuildContext context, WidgetRef ref) {
final ThemeData themeData = ref.watch(themeProvider);
final TextStyle headlineMedium = themeData.textTheme.headlineLarge!;
if (kDebugMode) {
print("building home");
}
return Scaffold(
body: Center(
child: Column(
children: [
Text(
"Hello World",
style: headlineMedium,
),
SwitchListTile(
contentPadding: EdgeInsets.zero,
title: const Text('Theme mode'),
value: ref.watch(themeModeStateProvider) == ThemeMode.light,
onChanged: (value) {
ref.watch(themeModeStateProvider.notifier).toggleThemeMode();
},
),
],
),
),
);
}
}
I am attaching a shorter code to reproduce this problem (without using generation):
By the way, don't use
ref.watch
in widget lifecycle management methods and callbacks. Useref.read
instead:Your problem lies in the
MainApp
widget, specifically in thebuilder
parameter. The short solution to the problem is to not useof(context)
insidebuilder
, and it looks like this:Now your rebuilds are optimized.
Speaking for the future, most likely your
ThemeData
should also have a full-fledgedNotifierProvider
and inside thebuild
method elegantlywatch
to the currentthemeModeStateProvider
. Then theProviderScope -> overrideWithValue
construct is not useful at all.Well, the long solution is to write an issue to the flutter repository.
The final version, taking into account
Localizations
, will look like this: