This is the 24th day of my participation in the August More Text Challenge

preface

In the last post, we completed the basic functionality of the shopping list, but there were several problems:

  • You can add duplicate shopping items to the CheckBox. This will cause two duplicate items to be operated at the same time.
  • There’s no offline storage. If it’s a real shopping list, the data is lost when you quit the App, and the App doesn’t work at all.
  • Unable to delete shopping item.

In this article, we’ll address these issues.

Repeat the process of adding shopping items

When repeating the addition, we can handle this problem well by adding 1 to the original shopping item and displaying the number of shopping items in the list. The processing of repeated addition is relatively simple. One is to add a count attribute to the ShoppingItem; the other is that when we respond to AddItemAction in the Reducer, when there are repeated items, we can add 1 to the number of the item.

Here we take out two general methods addItemActionHandler and toggleItemStateActionHandler, so that other place also can call.

List<ShoppingItem> addItemActionHandler(
    List<ShoppingItem> oldItems, ShoppingItem newItem) {
  List<ShoppingItem> newItems = [];

  if (oldItems.length > 0) {
    bool duplicated = false;
    newItems = oldItems.map((item) {
      if (item == newItem) {
        duplicated = true;
        return ShoppingItem(
            name: item.name, selected: item.selected, count: item.count + 1);
      }
      return item;
    }).toList();
    if (!duplicated) {
      newItems.add(newItem);
    }
  } else {
    newItems.add(newItem);
  }

  return newItems;
}

List<ShoppingItem> toggleItemStateActionHandler(
    List<ShoppingItem> oldItems, ShoppingItem newItem) {
  List<ShoppingItem> newItems = oldItems.map((item) {
    if (item == newItem)
      returnShoppingItem( name: item.name, selected: ! item.selected, count: item.count);return item;
  }).toList();

  return newItems;
}
Copy the code

Offline storage

Offline storage We use the shared_preferences plugin to store offline shopping lists. This plugin is introduced when we write a persistent CookieManager. Shared_preferences can only store bool, int, double, String, and List

.

class ShoppingItem {
  final String name;
  final bool selected;
  final int count;

  ShoppingItem({required this.name, this.selected = false.this.count = 1});

  bool operator= = (Object? other) {
    if (other == null| |! (otheris ShoppingItem)) return false;
    return other.name == this.name;
  }

  @override
  get hashCode => name.hashCode;

  Map<String.String> toJson() {
    return {
      'name': name,
      'selected': selected.toString(),
      'count': count.toString(),
    };
  }

  factory ShoppingItem.fromJson(Map<String.dynamic> json) {
    return ShoppingItem(
      name: json['name']! , selected: json['selected'] = ='true',
      count: int.parse(json['count']!) ,); }}Copy the code

Because offline storage is an asynchronous operation, middleware is required to complete the asynchronous storage operation. When a shopping item is added or the status of the shopping item is changed, the latest list is stored offline.

/ / middleware
const SHOPPLINT_LIST_KEY = 'shoppingList';
void shoppingListMiddleware(
    Store<ShoppingListState> store, dynamic action, NextDispatcher next) async {
	/ /...
  if (action is AddItemAction || action is ToggleItemStateAction) {
    List<Map<String.String>> listToSave =
            _prepareForSave(store.state.shoppingItems, action);
    SharedPreferences.getInstance().then(
        (prefs) => prefs.setString(SHOPPLINT_LIST_KEY, json.encode(listToSave)));
  }
  / /...
  next(action);
}

// Get the List that needs to be stored for different actions
List<Map<String.String>> _prepareForSave(
    List<ShoppingItem> oldItems, dynamic action) {
  List<ShoppingItem> newItems = [];
  if (action is AddItemAction) {
    newItems = addItemActionHandler(oldItems, action.item);
  }
  if (action is ToggleItemStateAction) {
    newItems = toggleItemStateActionHandler(oldItems, action.item);
  }

  return newItems.map((item) => item.toJson()).toList();
}
Copy the code

Recover listings from offline data

With offline storage done, the next problem is how to recover the manifest from offline data. This recovery is done when the App starts. That is, after startup, the shopping list needs to be read from offline storage to populate the state. Again, we need two actions here:

  • ReadOfflineAction: Reads data from the offline cache. Offline reading is an asynchronous operation and therefore needs to be done in the middleware. Post-completion schedulingReadOfflineSuccessAction.
  • ReadOfflineSuccessAction: The data is successfully read and carries offline data to update status data.

With these two operations, the middleware code becomes:

void shoppingListMiddleware(
    Store<ShoppingListState> store, dynamic action, NextDispatcher next) async {
  if (action is ReadOfflineAction) {
    SharedPreferences.getInstance().then((prefs) {
      dynamic offlineList = prefs.get(SHOPPLINT_LIST_KEY'shoppingList');
      if(offlineList ! =null && offlineList is String) { store.dispatch( ReadOfflineSuccessAction(offlineList: json.decode(offlineList))); }}); }else if (action is AddItemAction || action is ToggleItemStateAction) {
    List<Map<String.String>> listToSave =
        _prepareForSave(store.state.shoppingItems, action);
    SharedPreferences.getInstance().then(
        (prefs) => prefs.setString(SHOPPLINT_LIST_KEY, json.encode(listToSave)));
  } else {
    // ReadOfflineSuccessAction: No operation
  }

  next(action);
}
Copy the code

Here is a pit I stepped in when debugging, when the middleware code was written:

if (action is ReadOfflineAction) {
} else {
  // Store data offline
}
Copy the code

Results every other startup, the data is lost, scratching your head! Then played a breakpoint in offline storage that code, only to find that because ReadOfflineSuccessAction jump into this, when the store. State. ShoppingItems because haven’t updated to, is an empty array, save the empty array 😅 directly. How do I schedule ReadOfflineAction when I start the App? This is where StoreBuilder comes in handy. StoreBuilder provides callback Settings for state lifecycle functions. You can use StoreBuilder to build lower-level state-dependent components and then specify the corresponding lifecycle callback methods:

const StoreBuilder({
  Key? key,
  required this.builder,
  this.onInit,
  this.onDispose,
  this.rebuildOnChange = true.this.onWillChange,
  this.onDidChange,
  this.onInitialBuild,
}) : super(key: key);
Copy the code

In this case, we specify onInit to initialize the callback method and schedule the ReadOfflineAction in onInit to read the offline data once we start.

home: StoreBuilder<ShoppingListState>(
  onInit: (store) => store.dispatch(ReadOfflineAction()),
  builder: (context, store) => ShoppingListHome(),
),
Copy the code

That’s it! Done!

Running effect

Now you don’t have to worry about losing your shopping list! Source code has been uploaded to: Redux state management source code.

conclusion

This article introduces the use of StoreBuilder to introduce state life cycle hook functions and read offline data during initialization. Redux’s middleware is then used to store the data and load the offline data, thus completing a shopping list that supports offline storage. There’s also the problem that there’s no way to reduce or delete items. It’s not scientific. It’s not your girlfriend’s shopping cart. In the next post, we’ll take a general shopping quantity increment/subtraction component.


This is a column about the entry and practice of Flutter. The corresponding source code is here: Entry and Practice of Flutter.

👍🏻 : feel the harvest please point a praise to encourage!

🌟 : Collect articles, easy to look back!

💬 : Comment exchange, mutual progress!