// ignore_for_file: use_build_context_synchronously import 'package:dynamic_themes/dynamic_themes.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_i18n/widgets/I18nText.dart'; import 'package:revanced_manager/app/app.locator.dart'; import 'package:revanced_manager/services/manager_api.dart'; import 'package:revanced_manager/ui/widgets/settingsView/settings_section.dart'; import 'package:revanced_manager/ui/widgets/shared/haptics/haptic_radio_list_tile.dart'; import 'package:revanced_manager/ui/widgets/shared/haptics/haptic_switch_list_tile.dart'; class SUpdateThemeUI extends StatefulWidget { const SUpdateThemeUI({super.key}); @override State createState() => _SUpdateThemeUIState(); } class _SUpdateThemeUIState extends State { final ManagerAPI managerAPI = locator(); @override Widget build(BuildContext context) { return SettingsSection( title: 'settingsView.appearanceSectionTitle', children: [ ListTile( contentPadding: const EdgeInsets.symmetric(horizontal: 20.0), title: I18nText( 'settingsView.themeModeLabel', child: const Text( '', style: TextStyle( fontSize: 20, fontWeight: FontWeight.w500, ), ), ), trailing: FilledButton( onPressed: () => {showThemeDialog(context)}, child: getThemeModeName(), ), onTap: () => {showThemeDialog(context)}, ), if (managerAPI.isDynamicThemeAvailable) HapticSwitchListTile( contentPadding: const EdgeInsets.symmetric(horizontal: 20.0), title: I18nText( 'settingsView.dynamicThemeLabel', child: const Text( '', style: TextStyle( fontSize: 20, fontWeight: FontWeight.w500, ), ), ), subtitle: I18nText('settingsView.dynamicThemeHint'), value: getDynamicThemeStatus(), onChanged: (value) => { setUseDynamicTheme( context, value, ), }, ), ], ); } bool getDynamicThemeStatus() { return managerAPI.getUseDynamicTheme(); } Future setUseDynamicTheme(BuildContext context, bool value) async { await managerAPI.setUseDynamicTheme(value); final int currentTheme = (DynamicTheme.of(context)!.themeId ~/ 2) * 2; await DynamicTheme.of(context)!.setTheme(currentTheme + (value ? 1 : 0)); setState(() {}); } int getThemeMode() { return managerAPI.getThemeMode(); } Future setThemeMode(BuildContext context, int value) async { await managerAPI.setThemeMode(value); final bool isDynamicTheme = DynamicTheme.of(context)!.themeId.isEven; await DynamicTheme.of(context)! .setTheme(value * 2 + (isDynamicTheme ? 0 : 1)); final bool isLight = value != 2 && (value == 1 || DynamicTheme.of(context)!.theme.brightness == Brightness.light); SystemChrome.setSystemUIOverlayStyle( SystemUiOverlayStyle( systemNavigationBarIconBrightness: isLight ? Brightness.dark : Brightness.light, ), ); setState(() {}); } I18nText getThemeModeName() { switch (getThemeMode()) { case 0: return I18nText('settingsView.systemThemeLabel'); case 1: return I18nText('settingsView.lightThemeLabel'); case 2: return I18nText('settingsView.darkThemeLabel'); default: return I18nText('settingsView.systemThemeLabel'); } } Future showThemeDialog(BuildContext context) async { final ValueNotifier newTheme = ValueNotifier(getThemeMode()); return showDialog( context: context, builder: (context) => AlertDialog( title: I18nText('settingsView.themeModeLabel'), icon: const Icon(Icons.palette), contentPadding: const EdgeInsets.symmetric(vertical: 16), content: SingleChildScrollView( child: ValueListenableBuilder( valueListenable: newTheme, builder: (context, value, child) { return Column( mainAxisSize: MainAxisSize.min, crossAxisAlignment: CrossAxisAlignment.start, children: [ HapticRadioListTile( title: I18nText('settingsView.systemThemeLabel'), contentPadding: const EdgeInsets.symmetric(horizontal: 16), value: 0, groupValue: value, onChanged: (value) { newTheme.value = value!; }, ), HapticRadioListTile( title: I18nText('settingsView.lightThemeLabel'), contentPadding: const EdgeInsets.symmetric(horizontal: 16), value: 1, groupValue: value, onChanged: (value) { newTheme.value = value!; }, ), HapticRadioListTile( title: I18nText('settingsView.darkThemeLabel'), contentPadding: const EdgeInsets.symmetric(horizontal: 16), value: 2, groupValue: value, onChanged: (value) { newTheme.value = value!; }, ), ], ); }, ), ), actions: [ TextButton( onPressed: () { Navigator.of(context).pop(); }, child: I18nText('cancelButton'), ), FilledButton( onPressed: () { setThemeMode(context, newTheme.value); Navigator.of(context).pop(); }, child: I18nText('okButton'), ), ], ), ); } }