如何解决从多个小部件更新 AppBar 中的购物车徽章计数器
我正在使用具有标题和操作按钮的可重用 AppBar 小部件。
在应用栏操作中,有收藏夹图标按钮和购物车图标按钮,带有显示购物车中商品总数的徽章:
应用栏小部件:
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);
}
}
我想在用户添加到购物车、更新购物车、从购物车中删除时更新应用栏中的计数器。这可能发生在多个页面上: 产品列表页面、产品详细信息页面、购物车页面、搜索建议小部件。所有这些页面都有购物车操作(添加/更新/删除)。
我不想在每个页面中管理这个计数器。我需要一种方法来在一个地方管理它并在购物车更新时收到通知以更新徽章
我搜索了很多。有些使用 GLobalKey 来管理状态,但在我的情况下不起作用,因为 appbar 小部件是无状态的,不能是有状态的。
解决方法
我建议您使用任何类型的提供程序,例如 provider
包或 riverpod
,您需要名为 notifyListeners
的东西,以便在每次添加新商品时通知购物车徽章您的购物车,否则您的购物车项目将不会更新,除非在小部件重建时;像导航之类的。
您可以从 pub.dev 查看 riverpod 和 provider,确保完全理解文档,因为它可能很棘手!
希望这个回答对你有帮助!
,您要做的是管理应用程序的状态。你的做法很艰难。
也许你应该看看一些状态管理解决方案。
这是一个很好的介绍的链接,它讨论了您想要实现的目标 intro to state managment
以下是一些流行的状态管理列表:
没有完美的选择,只需使用您认为适合需要的东西即可。
,按照 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 举报,一经查实,本站将立刻删除。