flutter使用Provider完成动态主题功能

介绍

动态切换主题功能,使用Provider状态管理完成

学习本章内容,必须掌握Provider状态管理,如果有不太理解的同学,请打开我的主页搜索 Provider 观看后再返回观看本博客

主题样式大全

factory ThemeData({
  Brightness brightness, // 应用整体主题的亮度。用于按钮之类的小部件,以确定在不使用主色或强调色时选择什么颜色。
  MaterialColor primarySwatch,// 定义一个单一的颜色以及十个色度的色块。
  Color primaryColor, // 应用程序主要部分的背景颜色(toolbars、tab bars 等)
  Brightness primaryColorBrightness, // primaryColor的亮度。用于确定文本的颜色和放置在主颜色之上的图标(例如工具栏文本)。
  Color primaryColorLight, // primaryColor的浅色版
  Color primaryColorDark, // primaryColor的深色版
  Color accentColor, // 小部件的前景色(旋钮、文本、覆盖边缘效果等)。
  Brightness accentColorBrightness, // accentColor的亮度。
  Color canvasColor, //  MaterialType.canvas 的默认颜色
  Color scaffoldBackgroundColor, // Scaffold的默认颜色。典型Material应用程序或应用程序内页面的背景颜色。
  Color bottomAppBarColor, // BottomAppBar的默认颜色
  Color cardColor, // Card的颜色
  Color dividerColor, // Divider和PopupMenuDivider的颜色,也用于ListTile之间、DataTable的行之间等。
  Color highlightColor, // 选中在泼墨动画期间使用的突出显示颜色,或用于指示菜单中的项。
  Color splashColor,  // 墨水飞溅的颜色。InkWell
  InteractiveInkFeatureFactory splashFactory, // 定义由InkWell和InkResponse反应产生的墨溅的外观。
  Color selectedRowColor, // 用于突出显示选定行的颜色。
  Color unselectedWidgetColor, // 用于处于非活动(但已启用)状态的小部件的颜色。例如,未选中的复选框。通常与accentColor形成对比。也看到disabledColor。
  Color disabledColor, // 禁用状态下部件的颜色,无论其当前状态如何。例如,一个禁用的复选框(可以选中或未选中)。
  Color buttonColor, // RaisedButton按钮中使用的Material 的默认填充颜色。
  ButtonThemeData buttonTheme, // 定义按钮部件的默认配置,如RaisedButton和FlatButton。
  Color secondaryHeaderColor, // 选定行时PaginatedDataTable标题的颜色。
  Color textSelectionColor, // 文本框中文本选择的颜色,如TextField
  Color cursorColor, // 文本框中光标的颜色,如TextField
  Color textSelectionHandleColor,  // 用于调整当前选定的文本部分的句柄的颜色。
  Color backgroundColor, // 与主色形成对比的颜色,例如用作进度条的剩余部分。
  Color dialogBackgroundColor, // Dialog 元素的背景颜色
  Color indicatorColor, // 选项卡中选定的选项卡指示器的颜色。
  Color hintColor, // 用于提示文本或占位符文本的颜色,例如在TextField中。
  Color errorColor, // 用于输入验证错误的颜色,例如在TextField中
  Color toggleableActiveColor, // 用于突出显示Switch、Radio和Checkbox等可切换小部件的活动状态的颜色。
  String fontFamily, // 文本字体
  TextTheme textTheme, // 文本的颜色与卡片和画布的颜色形成对比。
  TextTheme primaryTextTheme, // 与primaryColor形成对比的文本主题
  TextTheme accentTextTheme, // 与accentColor形成对比的文本主题。
  InputDecorationTheme inputDecorationTheme, // 基于这个主题的 InputDecorator、TextField和TextFormField的默认InputDecoration值。
  IconThemeData iconTheme, // 与卡片和画布颜色形成对比的图标主题
  IconThemeData primaryIconTheme, // 与primaryColor形成对比的图标主题
  IconThemeData accentIconTheme, // 与accentColor形成对比的图标主题。
  SliderThemeData sliderTheme,  // 用于呈现Slider的颜色和形状
  TabBarTheme tabBarTheme, // 用于自定义选项卡栏指示器的大小、形状和颜色的主题。
  CardTheme cardTheme, // Card的颜色和样式
  ChipThemeData chipTheme, // Chip的颜色和样式
  TargetPlatform platform, 
  MaterialTapTargetSize materialTapTargetSize, // 配置某些Material部件的命中测试大小
  PageTransitionsTheme pageTransitionsTheme, 
  AppBarTheme appBarTheme, // 用于自定义Appbar的颜色、高度、亮度、iconTheme和textTheme的主题。
  BottomAppBarTheme bottomAppBarTheme, // 自定义BottomAppBar的形状、高度和颜色的主题。
  ColorScheme colorScheme, // 拥有13种颜色,可用于配置大多数组件的颜色。
  DialogTheme dialogTheme, // 自定义Dialog的主题形状
  Typography typography, // 用于配置TextTheme、primaryTextTheme和accentTextTheme的颜色和几何TextTheme值。
  CupertinoThemeData cupertinoOverrideTheme 
})

实战

引入包

由于用到了 Provider状态管理shared_preferences数据持久化
所以先引入一下,打开Flutter项目跟目录的pubspec.yaml


  shared_preferences: ^0.5.12+4
  provider: ^4.3.2+3

添加全局主题颜色

要设置主题,首先要知道自己的app需要多少种主题(颜色),这个主题我们要定义一个全局list出来

在项目跟目录 lib 下新建 global文件夹 在文件夹下新建 global_theme.dart文件 用来保存全局主题颜色,代码如下

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

final List themeList = [
  Colors.blue,
  Colors.black,
  Colors.red,
  Colors.purple,
  Colors.indigo,
  Colors.yellow,
  Colors.green,
];

我们就定义这几种颜色,大家如需要更多,可以自己定义

添加全局主题状态

由于我们的主题是动态的,所以改变后需要通知界面刷新,这里就要用到状态管理了,如果不熟悉 Provider状态管理 请先学习后在看

在项目跟目录 lib 下新建 provider文件夹 在文件夹下新建 theme_provider.dart文件 用来保存当前主题状态值,如果改变后,则会自动刷新用到此状态的ui,这就是状态管理的好处

import 'package:flutter/cupertino.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:zhong_mei_utils_flutter/global/Global.dart';

//类型随便取,继承ChangeNotifier
class ThemeProvider with ChangeNotifier {
  Color _color = themeList.first;//默认是我们设置的主题颜色列表第一个

  void setTheme(int index) {//给外部提供修改主题的方法
    print(index.toString());
    _color = themeList[index];
    notifyListeners();
  }

  Color get color => _color;//获取当前主题
}

主题状态类创建好了,添加到项目里,让项目管理此状态
修改main.dart

return runApp(MultiProvider(
          providers: [
            ChangeNotifierProvider(create: (context) => UserProvider()),
            ChangeNotifierProvider(create: (context) => ThemeProvider()),
          ],
          child: MyApp(),
        ));

主题状态设置完毕,接下来使用全局主题状态

一般来说,一个app只有一个MaterialApp,我们主题设置在MaterialApp,所以找到我们的MaterialApp,添加以下代码

return MaterialApp(
      theme: ThemeData.light().copyWith(
        primaryColor: Provider.of(context).color,
        buttonTheme: ButtonThemeData(
          buttonColor: Provider.of(context).color,
          textTheme: ButtonTextTheme.normal,
        ),
      ),

上段代码设置了app主题颜色跟按钮颜色使用状态管理里的颜色,而按钮文本颜色则是高亮,如果按钮背景色是黑色,那么按钮文字颜色自动白色

主题修改界面

lib 下添加 settings_theme.dart.dart

import 'package:flutter/material.dart';
import 'package:provider/provider.dart';
import 'package:shared_preferences/shared_preferences.dart';
import 'package:weui/icon/index.dart';
import 'package:zhong_mei_utils_flutter/global/Global.dart';
import 'package:zhong_mei_utils_flutter/provider/ThemeProvider.dart';

class SettingsView extends StatefulWidget {
  @override
  _SettingsViewState createState() => _SettingsViewState();
}

class _SettingsViewState extends State {
  int _index;//我们当前主题设置的是 全局主题列表中的第几个?
  @override
  void initState() {
    super.initState();
    loadData();//查询当前持久化数据
  }

  void loadData() async {
    SharedPreferences sp = await SharedPreferences.getInstance();//获取持久化操作对象
    setState(() {
      _index = sp.getInt("theme") ?? 0;//查询持久化框架中保存的theme字段,如果是null则默认是0
    });
  }

  Widget _itemBuilder(BuildContext context, int index) {
    return GestureDetector(
      child: Container(
        width: double.infinity,
        height: 50,
        margin: EdgeInsets.only(top: 10, bottom: 10),
        decoration: BoxDecoration(
          color: themeList[index],
          borderRadius: BorderRadius.all(Radius.circular(20)),
        ),
        child: _index != index
            ? Text("")//如果没选中则无东西
            : Row(
                mainAxisAlignment: MainAxisAlignment.end,
                children: [
                  Icon(
                    WeIcons.hook,//如果选中了则给一个图标,这个图标是一个对勾,大家可以自己找
                    color: Colors.white,
                  ),
                  SizedBox(width: 16),
                ],
              ),
      ),
      onTap: () async {
        SharedPreferences sp = await SharedPreferences.getInstance();
        sp.setInt("theme", index);//点击后,修改持久化框架里的theme数据库
        Provider.of(context, listen: false).setTheme(index);//修改全局状态为选中的值
        setState(() {
          _index = index;//设置当前界面选种值,刷新对勾
        });
      },
    );
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text("设置主题"),
        centerTitle: true,
        elevation: 10,
      ),
      body: Padding(
        padding: EdgeInsets.all(20),
        child: Scrollbar(
          child: ListView.builder(
            itemBuilder: _itemBuilder,
            itemCount: themeList.length,
          ),
        ),
      ),
    );
  }
}

这时候已经完成了,但是有一个问题,就是我们正常使用都没有问题,但是退出app再进来,我们没有读取持久化保存的主题,所以又变默认第一个颜色

这时候我们想读取数据看中的主题,先了解一个概念
app启动时,都有白屏,一般开发者会做一个启动图,来掩盖白屏,我们这里需要知道,白屏是因为app在加载一些东西,还没有渲染界面,那么我们加载主题也应该在app启动前加载

否则会出现,界面渲染了,但是主题没读取出来,造成主题先显示默认颜色,才显示我们设置的主题颜色

那么flutter项目的main.dart中有一段代码

return runApp(MultiProvider(
          providers: [
            ChangeNotifierProvider(create: (context) => UserProvider()),
            ChangeNotifierProvider(create: (context) => ThemeProvider()),
          ],
          child: MyApp(),
        ));

可以看到 child: MyApp() 之前是没有ui渲染操作的,所以我们要在 return runApp之前读取到主题,并且生成ThemeProvider对象,代码改动如下

int theme;//添加一个变量,接收数据库读取返回

void main() async {//由于读取数据库需要异步,所以加上async
  await loadData();//读取数据库保存的主题
      ThemeProvider themeProvider = ThemeProvider();//new一个主题状态对象
      themeProvider.setTheme(theme ?? 0);//给对象设置成我们读取的主题
      return runFxApp(
        MultiProvider(
          providers: [
            ChangeNotifierProvider(create: (context) => UserProvider()),
            ChangeNotifierProvider(create: (context) => themeProvider),//这里ThemeProvider()  改成 themeProvider对象
          ],
          child: MyApp(),
        ),
        // onEnsureInitialized: (info) {},
        enableLog: false,
        uiBlueprints: uiSize,
      );
}

void loadData() async {
  SharedPreferences sp = await SharedPreferences.getInstance();
  theme = sp.getInt("theme") ?? 0;//如果读取是空则返回0
}

经过以上改动,我们完成了读取持久化保存的主题,并且在app启动时预先加载了主题

实现效果演示

s103xavuim.gif

可以看到,启动项目到登陆页面的时候主题也是最新的了,这里我们用到了持久化操作

你可能感兴趣的:(flutter使用Provider完成动态主题功能)