- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试使用 Hydrated Bloc 在我的应用程序中保留状态。我发现的教程都使用以前版本的 Hydrated Blocs 并使用
BlocSupervisor
,它在 v.5 的
bloc
包的 Dart 版本中被删除(
https://pub.dev/packages/bloc/changelog) .
flutter_bloc
和
hydrated_bloc
在更新到
bloc
v.5(
https://pub.dev/packages/flutter_bloc/changelog 和
https://pub.dev/packages/hydrated_bloc/changelog)时删除了它。文档说它应该替换为
BlocObserver
。
hydred_bloc
和
flutter_bloc
没有列出替代品。到目前为止,我还没有找到使用
BlocSupervisor
和
BlocDelegate
以外的任何东西的
HydratedBloc 教程;只有
flutter_bloc
教程。
如何创建与
BlocObserver
等效的 HydratedBloc 以保持状态?
编辑:好吧,多亏了你的榜样,我想我现在走在了正确的轨道上。
这是我的代码的相关部分,类已重命名:
class ClassABLoC
extends HydratedBloc<ClassAEvent, ClassAState> {
//This does initialize it with some values, but not the ones from storage. Trying to call fromJson() there gives an error.
ClassABLoC() : super(ClassAState.dataNotReceived());
@override
Stream<classAState> mapEventToState(
classAEvent event) async* {
if (event is classAInitialize){
print("Initializing");
yield fromJson(json.decode(
HydratedBloc.storage.read("vehicleNumber") as String,
) as Map<String, dynamic>,
);
}
if (event is classAValidate) {
yield classAState.validated(
theEvent: event,
numberValidated: validateTheInput(event.numberVal, event.vehicleNumber),
distanceValidated:
validateTheInput(event.distanceVal, event.vehicleDistanceTraveled),
yearValidated: validateTheInput(event.yearVal, event.vehicleYear),
vinValidated: validateTheInput(event.vinVal, event.vin),
licensePlateValidated:
validateTheInput(event.licensePlateVal, event.vehicleLicensePlate),
revsPerDistValidated:
validateTheInput(event.revsPerDistVal, event.vehicleRevsPerDist),
fuelTypeValidated: validateTheInput(event.fuelTypeVal, event.fuelType),
fuelCapacityValidated:
validateTheInput(event.fuelCapacityVal, event.fuelCapacity),
siteValidated: validateTheInput(event.siteVal, event.vehicleSite),
numberError: event.numberVal!.validationFailedMsg,
distanceError: event.distanceVal!.validationFailedMsg,
yearError: event.yearVal!.validationFailedMsg,
vinError: event.vinVal!.validationFailedMsg,
licensePlateError: event.licensePlateVal!.validationFailedMsg,
revsPerDistError: event.revsPerDistVal!.validationFailedMsg,
fuelTypeError: event.fuelTypeVal!.validationFailedMsg,
fuelCapacityError: event.fuelCapacityVal!.validationFailedMsg,
siteError: event.siteVal!.validationFailedMsg,
);
} else if (event is StoreDataEvent) {
toJson(StoreDataState(
vehicleNumber: event.vehicleNumber,
vehicleYear: event.vehicleYear,
vehicleRevsPerDist: event.vehicleRevsPerDist,
vehicleDistanceTraveled: event.vehicleDistanceTraveled,
vin: event.vin,
vehicleLicensePlate: event.vehicleLicensePlate,
fuelCapacity: event.fuelCapacity,
fuelType: event.fuelType,
vehicleSite: event.vehicleSite,
));
}
}
Map<String, dynamic>? toJson(ProgramDataTracSVTState state) {
if (state is StoreDataState) {
print("State was StoreDataState.");
print(state.toString());
return {
'vehicleNumber': state.vehicleNumber,
'vehicleDistanceTraveled': state.vehicleDistanceTraveled,
'vehicleLicensePlate': state.vehicleLicensePlate,
'vin': state.vin,
'rehicleRevsPerDist': state.vehicleRevsPerDist,
'vehicleSite': state.vehicleSite,
'vehicleYear': state.vehicleYear,
'fuelCapacity': state.fuelCapacity,
'fuelType': state.fuelType,
'distUnit': state.distUnits,
};
}
}
ProgramDataTracSVTState fromJson(Map<String, dynamic> json) {
print(json['vehicleDistance'] as String);
return ProgramDataTracSVTState(
vehicleNumber: (json['vehicleNumber'] as String?) == "" ||
json['vehicleNumber'] == null
? "HARDCODED"
: json['vehicleNumber'] as String,
vehicleDistanceTraveled:
(json['vehicleDistanceTraveled'] as String?) == "" ||
(json['vehicleDistanceTraveled'] as String?) == null
? "HARDCODED VALUE"
: json['vehicleDistanceTraveled'] as String,
vehicleLicensePlate: (json['vehicleLicensePlate'] as String?) == "" ||
(json['vehicleLicensePlate'] as String?) == null
? ""
: json['vehicleLicensePlate'] as String,
vin: (json['vin'] as String?) == "" || (json['vin'] as String?) == null
? "HARDCODED VALUE"
: json['vin'] as String,
vehicleRevsPerDist: (json['vehicleRevsPerDist'] as String?) == "" ||
(json['vehicleRevsPerDist'] as String?) == null
? "500"
: json['vehicleRevsPerDist'] as String,
vehicleSite: (json['vehicleSite'] as String?) == "" ||
(json['vehicleSite'] as String?) == null
? "Home Base Site"
: (json['vehicleSite'] as String),
vehicleYear: (json['vehicleYear'] as String?) == "" ||
(json['vehicleYear'] as String?) == null
? "2018"
: json['vehicleYear'] as String,
fuelCapacity: (json['fuelCapacity'] as String?) == "" ||
(json['fuelCapacity'] as String?) == null
? ""
: (json['fuelCapacity'] as String),
fuelType: (json['fuelType'] as String?) == "" ||
(json['fuelType'] as String?) == null
? "Diesel"
: (json['fuelType'] as String),
distUnits: (json['distUnit'] as String?) == "" ||
(json['distUnit'] as String?) == null
? "None"
: (json['distUnit'] as String),
);
}
我需要从存储中获取不止一个字符串的数据。我该怎么做?据我所知,我无法组合 JSON 字符串。
最佳答案
因此,如果我理解正确的话,除了能够以您之前所处的任何状态重新启动之外,您还需要一个完整的数据库。
我原来的回答是后者。查看您的代码后,我应该澄清您不需要手动调用 toJson
和 fromJson
方法。只要状态发生变化,它们就会自动调用,它会覆盖之前存储的任何内容,并且在应用重启时将恢复最新状态。
因此,您将在 fromJson
中返回一个 ProgramDataTracSVTState
。无论您在应用程序中的哪个位置显示 ClassABLoC
应用程序重新启动时的状态,都应该显示最后的事件状态,而无需设置专门的 classAInitialize
事件来手动调用 来自Json
。请参阅我的第一个示例以供引用。
为了为典型的本地数据库存储 VehicleModel
的多个实例(我假设您有一个模型类用于存储在 toJson
中的内容,如果不,你应该创建一个),我会创建单独的方法来调用 HydratedBloc.storage.write(key, value)
和 HydratedBloc.storage.read(key)
来访问您需要存储的任何车辆。
如果您的所有车辆编号都是唯一的,您可以将其用作访问主存储 map 中任何车辆的 key 。
这是一个带有基本 CarModel
的简化示例
class CarModel {
final int id;
final String car;
CarModel({required this.id, required this.car});
CarModel.fromJson(Map<String, dynamic> map)
: id = map['id'],
car = map['car'];
Map<String, dynamic> toJson() {
return {
'id': id,
'car': car,
};
}
}
事件
class PrintStorageData extends TestEvent {}
class StoreDataEvent extends TestEvent {
final CarModel car;
StoreDataEvent({required this.car});
}
状态
abstract class TestState extends Equatable {
final CarModel testModel;
const TestState(this.testModel);
@override
List<Object> get props => [testModel];
}
class State1 extends TestState {
State1({required CarModel model}) : super(model);
}
已更新 TestBloc
class TestBloc extends HydratedBloc<TestEvent, TestState> {
TestBloc()
: super(State1(model: CarModel(id: 0, car: 'no cars listed'))) {
on<PrintStorageData>(
(event, emit) => _printStorageData(),
);
on<StoreDataEvent>((event, emit) {
emit(State1(model: event.car)); // toJson gets called here and fromJson on app restart
_storeCarModel(event.car); // this is what stores in a database you can access later on
});
}
Future<void> _printStorageData() async {
final mapFromStorage =
await HydratedBloc.storage.read('vehicles') as Map? ?? {};
if (mapFromStorage.isNotEmpty) {
mapFromStorage.forEach((key, value) {
log('$key: $value');
});
} else {
log('No vehicles stored');
}
}
Future<void> _storeCarModel(CarModel model) async {
final id = model.id; // using id for storage key
final storageMap = await HydratedBloc.storage.read('vehicles') as Map? ??
{}; // initializing to empty map if nothing is stored
storageMap[id] = model.toJson();
await HydratedBloc.storage.write('vehicles', storageMap);
}
@override
TestState fromJson(Map<String, dynamic>? json) {
return State1(model: CarModel.fromJson(json!));
}
@override
Map<String, dynamic>? toJson(TestState state) {
return {'id': state.testModel.id, 'car': state.testModel.car};
}
}
要检查这一点,您可以在 UI 中添加几辆车并在重启后打印存储的列表
ElevatedButton(
onPressed: () => context.read<TestBloc>().add(PrintStorageData()),
child: Text('Read Data'),
),
ElevatedButton(
onPressed: () {
final car = CarModel(id: 1, car: 'tesla');
context.read<TestBloc>().add(StoreDataEvent(car: car));
},
child: Text('Store new car'),
),
由于您没有提供任何代码,因此这里是保存基本 String
的示例。
状态
abstract class TestState extends Equatable {
final String testString;
const TestState(this.testString);
@override
List<Object> get props => [testString];
}
class State1 extends TestState {
State1({required String testString}) : super(testString);
}
事件
abstract class TestEvent {}
class ChangeStringToSaved extends TestEvent {}
class UpdateSaved extends TestEvent {}
HydradedBloc
class TestBloc extends HydratedBloc<TestEvent, TestState> {
TestBloc() : super(State1(testString: 'not saved')) {
on<ChangeStringToSaved>(
(event, emit) => emit(State1(testString: 'this is saved')));
on<UpdateSaved>(
(event, emit) => emit(State1(testString: 'updated saved value')));
}
// Creating State1 from stored Map
@override
TestState? fromJson(Map<String, dynamic> json) {
return State1(testString: json['value'] as String);
}
// Saving to a basic Map
@override
Map<String, dynamic>? toJson(TestState state) {
return {'value': state.testString};
}
}
基本用户界面
class Home extends StatelessWidget {
@override
Widget build(BuildContext context) {
return Scaffold(
body: Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [
BlocConsumer<TestBloc, TestState>(
listener: (context, state) {},
builder: (context, state) {
return Text(state.testString);
}),
ElevatedButton(
onPressed: () =>
context.read<TestBloc>().add(ChangeStringToSaved()),
child: Text('Change to Saved'),
),
ElevatedButton(
onPressed: () => context.read<TestBloc>().add(UpdateSaved()),
child: Text('Update Saved'),
),
],
),
),
);
}
}
每当我更新 testString
时,它会在重启后自动保留。
因此,如果我理解正确的话,除了能够以您之前所处的任何状态重新启动之外,您还需要一个完整的数据库。
关于flutter - 使用 Hydrated Bloc 版本 5 及更高版本的持久状态,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/69619565/
我正在尝试从flutter应用程序构建apk,但出现此错误: Note: /mnt/Software/Linux/Flutter/flutter/.pub-cache/hosted/pub.dartl
我有一个名为 X 的较大应用程序,还有另一个名为 Y 的较小应用程序。他们现在彼此分开,并且工作正常。我想将应用程序 Y 集成到 X 中。我想将 Y 的代码放入 X 项目中,但它们应该有不同的 Mai
在android Studio中选择Create New Flutter Project,出现如下4个选项。 Flutter 应用程序 Flutter 插件 Flutter 包 flutter 模块
我看到我的 flutter 项目生成了一个文件 ios/Flutter/Flutter.podspec ,这个文件有什么用? 如果它与生成的 Flutter.framework 有关? 我应该将它包含
我尝试过的 在包含flutter SDK的位置添加/编辑.bash_profile.rtf文件。 导出PATH = / Users / temur / Documents / Projects / F
Flutter 日志会打印数千个详细/垃圾邮件日志。 我正在尝试调试一个复杂的应用程序,但是打印太冗长了,以至于我很难找到我自己打印的东西。 有没有办法禁用详细? 就像是: Logger.level.
在flutter 1.22更新之后,我在Lineargradient colors属性中遇到错误,这给我一个错误,即未定义名称colors参数。.在Android中更新flutter和flutter插
在下面的代码 widget.hintText 中给出错误,我试图将日期选择器作为单独的组件,并在从其他文件调用它时动态传递提示文本值。 import 'package:date_field/date_
在下面的代码 widget.hintText 中给出错误,我试图将日期选择器作为单独的组件,并在从其他文件调用它时动态传递提示文本值。 import 'package:date_field/date_
Flutter 1.0 发布后,我正在按照步骤搭建 Flutter 开发环境。 在步骤中(如所附屏幕截图所示),它说要更新 $PATH 两次,一次使用 flutter 工具的路径 export PAT
我有一个用 flutter 编写的移动应用程序,我想将其转换为 flutter_web 应用程序(集成 flutter_web 尚不可用)。我目前遇到包裹问题。 我已按照本网站中列出的说明进行操作 h
如何向我的 Flutter 路由添加自定义转换?这是我目前的路线结构。 class MyApp extends StatelessWidget { // This widget is the
我正在尝试通过 URL 在 webview 中显示网页。我试过 flutter_webview_plugin 插件,但是当我在浏览器上运行项目时它不起作用。 在 flutter web applica
我正在使用 animatedContainer 在按下按钮时显示 listView.builder()。这是一个要显示的简单子(monad)列表,但问题是我不知道 ListView 构建器的高度会传递
我目前正在我的应用程序中制作渐变背景动画......我正在使用 lottie 动画的帮助下这样做!我试图将它封装在一个容器中并成功地做到了。但是有一个问题,尽管我将高度更改为大于 2000 的东西,但
美好的一天! 我无法弄清楚如何使用 google 标签管理器设置 flutter。我找到了 this package包括标签管理器 api。但是我不知道如何正确配置它。 (在网络上我只需要复制粘贴一个
我的购物车模型如下 class Cart { String description; double unitCost; double amount; int quantity; S
在 Flutter 应用程序中,我想为在线托管的资源(图像、视频等)实现缓存。 我希望它能在原生平台 (Android/iOS)(例如使用文件系统)和网络(例如使用 IndexedDB)上运行。 Fl
我写了一个页面,在顶部一切都很好,应该是这样。在底部我有一个事件的历史。我的容器的宽度是自动确定的(取决于屏幕的宽度),而高度 - 不,在不同的设备上有不同的高度(底部的缩进是不同的)。是否可以自动确
我正在处理一个页面,其中有一些字段,例如 textfield 和 slider。在页面的末尾必须有一个用于进行下一步的按钮,该按钮被包裹在 Align 中以在页面之间具有固定位置。 另一方面,resi
我是一名优秀的程序员,十分优秀!