从多个小部件更新 AppBar 中的购物车徽章计数器

如何解决从多个小部件更新 AppBar 中的购物车徽章计数器

我正在使用具有标题和操作按钮的可重用 AppBar 小部件。

在应用栏操作中,有收藏夹图标按钮和购物车图标按钮,带有显示购物车中商品总数的徽章:

enter image description here

应用栏小部件:

import 'package:badges/badges.dart';
import 'package:Flutter/material.dart';

class CustomAppBar extends StatelessWidget implements PreferredSizeWidget {
  final BuildContext context;
  final String title;
  final bool showBackButton;
  final Widget widget;
  final bool showActions;

  CustomAppBar({
    @required this.context,@required this.title,this.showBackButton = true,this.widget,this.showActions = true,});

  @override
  Widget build(BuildContext context) {
    return AppBar(
      title: Text(title),leading: showBackButton
          ? new IconButton(
              icon: new Icon(
                Icons.arrow_back,),onpressed: () {
                if (widget != null) {
                  Navigator.push(
                    context,MaterialPageRoute(
                      builder: (context) => widget,);
                  return true;
                } else {
                  Navigator.pop(context);
                }
              },)
          : null,actions: !showActions ? null : <Widget>[
        IconButton(
          icon: const Icon(Icons.favorite_border),onpressed: () {
            Navigator.pushReplacement(
              context,MaterialPageRoute<void>(
                builder: (BuildContext context) {
                  return MainHome(
                    selectedindex: 1,);
                },);
          },Padding(
          padding: const EdgeInsets.symmetric(horizontal: 5),child: Badge(
            position: BadgePosition.topEnd(top: 3,end: 3),animationDuration: Duration(milliseconds: 300),animationType: BadgeAnimationType.slide,badgeColor: Colors.white,toAnimate: true,badgeContent: Text(
              '5',style: TextStyle(
                  fontSize: 8,color: Theme.of(context).primaryColor,fontWeight: FontWeight.bold),child: IconButton(
              icon: const Icon(Icons.shopping_cart_rounded),onpressed: () {
                Navigator.push(
                  context,MaterialPageRoute<void>(
                    builder: (BuildContext context) {
                      return MainHome(
                        selectedindex: 2,);
                    },);
              },],);
  }

  @override
  Size get preferredSize {
    return new Size.fromHeight(kToolbarHeight);
  }
}

应用栏用于所有屏幕,但具有不同的参数。

Products List Page

Search Widget

我想在用户添加到购物车、更新购物车、从购物车中删除时更新应用栏中的计数器。这可能发生在多个页面上: 产品列表页面、产品详细信息页面、购物车页面搜索建议小部件。所有这些页面都有购物车操作(添加/更新/删除)。

我不想在每个页面中管理这个计数器。我需要一种方法来在一个地方管理它并在购物车更新时收到通知以更新徽章

搜索了很多。有些使用 GLobalKey 来管理状态,但在我的情况下不起作用,因为 appbar 小部件是无状态的,不能是有状态的。

解决方法

我建议您使用任何类型的提供程序,例如 provider 包或 riverpod ,您需要名为 notifyListeners 的东西,以便在每次添加新商品时通知购物车徽章您的购物车,否则您的购物车项目将不会更新,除非在小部件重建时;像导航之类的。

您可以从 pub.dev 查看 riverpod 和 provider,确保完全理解文档,因为它可能很棘手!

希望这个回答对你有帮助!

,

您要做的是管理应用程序的状态。你的做法很艰难。

也许你应该看看一些状态管理解决方案。

这是一个很好的介绍的链接,它讨论了您想要实现的目标 intro to state managment

以下是一些流行的状态管理列表:

  • Provider(易于使用,Flutter 社区推荐)
  • Bloc(适用于非常大的项目)
  • GetX(简单易用)
  • Riverpod(它是提供者,但功能更强大)

没有完美的选择,只需使用您认为适合需要的东西即可。

,

按照 Fahmi Sawalha 和 Boris Kamtou 的建议,我使用了 provider 包来解决这个问题。

如果有人需要,我会分享代码。该代码可帮助您使用标题、上下文、showBackButton、showActions 等参数将自定义 AppBar 创建为无状态小部件。

而且代码还可以帮助您使用提供程序包来管理状态并从不同屏幕更新 ui。

首先在您的应用程序中的任何可访问位置创建 CartCounterNotifier 类。它应该扩展 ChangeNotifier 以便在值更改时通知侦听器,以便使用此提供程序的 ui 重建:

CartManager 是一个 Mixin,我在其中管理 DB 中的购物车数据。 getCartItemsCount() 函数获取购物车中商品数量的总和。

cart_counter.dart

import 'package:flutter/material.dart';

class CartCounterNotifier extends ChangeNotifier with CartManager {
  int _value = 0;

  int get value => _value;

  CartCounterNotifier() {
    getCartItemsCount().then((counter) {
      _value = counter ?? 0;
      notifyListeners();
    });
  }

  // You can send send parameters to update method. No need in my case. 
  //  Example:  void update(int newvalue) async { ...

  void update() async {
    int counter = await getCartItemsCount();

    _value = counter ?? 0;
    notifyListeners();
  }
}

该类有两个方法:

用于初始化计数器的构造函数和在值更改时调用的 update() 方法。

我创建了自定义应用栏以在所有页面上使用。它是一个实现 PreferedSizeWidget 的无状态小部件。

用 Consumer 将小部件包裹在您要显示计数器的位置。最好尽可能深入。就我而言,我没有用消费者包装整个 AppBar,而是包装了显示计数器的文本小部件

custom_app_bar.dart

import 'package:badges/badges.dart'; // add badges to pubspec in order to add badge for icons
import 'package:flutter/material.dart';
import 'package:order_flutter_package/services/cart_counter.dart';
import 'package:order_flutter_package/ui/views/home_view/home_view.dart';
import 'package:provider/provider.dart';

class CustomAppBar extends StatelessWidget implements PreferredSizeWidget {
  final BuildContext context;
  final String title;
  final bool showBackButton;
  final Widget widget;
  final bool showActions;

  CustomAppBar({
    @required this.context,@required this.title,this.showBackButton = true,this.widget,this.showActions = true,});

  @override
  Widget build(BuildContext context) {
    return PreferredSize(
      preferredSize: Size.fromHeight(50),child: AppBar(
        title: Text(title),leading: showBackButton
            ? new IconButton(
                icon: new Icon(
                  Icons.arrow_back,),onPressed: () {
                  if (widget != null) {
                    Navigator.push(
                      context,MaterialPageRoute(
                        builder: (context) => widget,);
                    return true;
                  } else {
                    Navigator.pop(context);
                  }
                },)
            : null,actions: !showActions
            ? null
            : <Widget>[
                IconButton(
                  icon: const Icon(Icons.favorite_border),onPressed: () {
                    Navigator.pushReplacement(
                      context,MaterialPageRoute<void>(
                        builder: (BuildContext context) {
                          return MainHome(
                            selectedIndex: 1,);
                        },);
                  },Padding(
                  padding: const EdgeInsets.symmetric(horizontal: 5),child: Badge(
                    position: BadgePosition.topEnd(top: 3,end: 3),animationDuration: Duration(milliseconds: 300),animationType: BadgeAnimationType.slide,badgeColor: Colors.white,toAnimate: true,badgeContent: Consumer<CartCounterNotifier>(
                        builder: (context,cartCounter,child) {
                      return Text(
                        cartCounter.value.toString(),style: TextStyle(
                            fontSize: 8,color: Theme.of(context).primaryColor,fontWeight: FontWeight.bold),);
                    }),child: IconButton(
                      icon: const Icon(Icons.shopping_cart_rounded),onPressed: () {
                        Navigator.push(
                          context,MaterialPageRoute<void>(
                            builder: (BuildContext context) {
                              return MainHome(
                                selectedIndex: 2,);
                            },);
                      },],);
  }

  @override
  Size get preferredSize {
    return new Size.fromHeight(kToolbarHeight);
  }
}

然后用 ChangeNotifierProvider 包装你的 main.dart 应用

import 'package:provider/provider.dart';
import 'package:order_flutter_package/services/cart_counter.dart';

ChangeNotifierProvider(
        create: (context) => CartCounterNotifier(),child: MyApp(),)

最后,为了更新计数器,在任何屏幕上,

// Very important to import provider else you got an error
import 'package:provider/provider.dart'; 

// On button pressed or any event
var cartCounter = context.read<CartCounterNotifier>();
cartCounter.update();

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其他元素将获得点击?
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。)
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbcDriver发生异常。为什么?
这是用Java进行XML解析的最佳库。
Java的PriorityQueue的内置迭代器不会以任何特定顺序遍历数据结构。为什么?
如何在Java中聆听按键时移动图像。
Java“Program to an interface”。这是什么意思?
Java在半透明框架/面板/组件上重新绘画。
Java“ Class.forName()”和“ Class.forName()。newInstance()”之间有什么区别?
在此环境中不提供编译器。也许是在JRE而不是JDK上运行?
Java用相同的方法在一个类中实现两个接口。哪种接口方法被覆盖?
Java 什么是Runtime.getRuntime()。totalMemory()和freeMemory()?
java.library.path中的java.lang.UnsatisfiedLinkError否*****。dll
JavaFX“位置是必需的。” 即使在同一包装中
Java 导入两个具有相同名称的类。怎么处理?
Java 是否应该在HttpServletResponse.getOutputStream()/。getWriter()上调用.close()?
Java RegEx元字符(。)和普通点?