mirror of
https://github.com/openfoodfacts/smooth-app.git
synced 2025-08-06 18:25:11 +08:00

Impacted files: * `app_en.arb`: added 1 label for "scan history" * `continuous_scan_model.dart`: added the product to "scan history" * `product_list.dart`: added "scan history"; refactored `ProductListType` * `product_list_page.dart`: added "scan history" * `product_query_page_helper.dart`: added "scan history" * `user_preferences_account.dart`: added an access to the "scan history" page
298 lines
8.4 KiB
Dart
298 lines
8.4 KiB
Dart
import 'dart:async';
|
|
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/widgets.dart';
|
|
import 'package:openfoodfacts/openfoodfacts.dart';
|
|
import 'package:smooth_app/data_models/fetched_product.dart';
|
|
import 'package:smooth_app/data_models/product_list.dart';
|
|
import 'package:smooth_app/database/dao_product.dart';
|
|
import 'package:smooth_app/database/dao_product_list.dart';
|
|
import 'package:smooth_app/database/local_database.dart';
|
|
import 'package:smooth_app/generic_lib/duration_constants.dart';
|
|
import 'package:smooth_app/helpers/analytics_helper.dart';
|
|
import 'package:smooth_app/query/barcode_product_query.dart';
|
|
import 'package:smooth_app/services/smooth_services.dart';
|
|
|
|
enum ScannedProductState {
|
|
FOUND,
|
|
NOT_FOUND,
|
|
LOADING,
|
|
THANKS,
|
|
CACHED,
|
|
ERROR_INTERNET,
|
|
ERROR_INVALID_CODE,
|
|
}
|
|
|
|
class ContinuousScanModel with ChangeNotifier {
|
|
ContinuousScanModel();
|
|
|
|
final Map<String, ScannedProductState> _states =
|
|
<String, ScannedProductState>{};
|
|
final List<String> _barcodes = <String>[];
|
|
final ProductList _productList = ProductList.scanSession();
|
|
final ProductList _scanHistory = ProductList.scanHistory();
|
|
final ProductList _history = ProductList.history();
|
|
|
|
String? _latestScannedBarcode;
|
|
String? _latestFoundBarcode;
|
|
String? _latestConsultedBarcode;
|
|
late DaoProduct _daoProduct;
|
|
late DaoProductList _daoProductList;
|
|
|
|
ProductList get productList => _productList;
|
|
|
|
List<String> getBarcodes() => _barcodes;
|
|
|
|
String? get latestConsultedBarcode => _latestConsultedBarcode;
|
|
|
|
set lastConsultedBarcode(String? barcode) {
|
|
_latestConsultedBarcode = barcode;
|
|
if (barcode != null) {
|
|
notifyListeners();
|
|
}
|
|
}
|
|
|
|
Future<ContinuousScanModel?> load(final LocalDatabase localDatabase) async {
|
|
try {
|
|
_daoProduct = DaoProduct(localDatabase);
|
|
_daoProductList = DaoProductList(localDatabase);
|
|
if (!await _refresh()) {
|
|
return null;
|
|
}
|
|
return this;
|
|
} catch (e) {
|
|
Logs.e('Load database error', ex: e);
|
|
}
|
|
return null;
|
|
}
|
|
|
|
Future<bool> _refresh() async {
|
|
try {
|
|
_latestScannedBarcode = null;
|
|
_latestFoundBarcode = null;
|
|
_barcodes.clear();
|
|
_states.clear();
|
|
_latestScannedBarcode = null;
|
|
await refreshProductList();
|
|
for (final String barcode in _productList.barcodes) {
|
|
_barcodes.add(barcode);
|
|
_states[barcode] = ScannedProductState.CACHED;
|
|
_latestScannedBarcode = barcode;
|
|
}
|
|
return true;
|
|
} catch (e) {
|
|
Logs.e('Refresh database error', ex: e);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
Future<void> refreshProductList() async => _daoProductList.get(_productList);
|
|
|
|
void _setBarcodeState(
|
|
final String barcode,
|
|
final ScannedProductState state,
|
|
) {
|
|
_states[barcode] = state;
|
|
notifyListeners();
|
|
}
|
|
|
|
ScannedProductState? getBarcodeState(final String barcode) =>
|
|
_states[barcode];
|
|
|
|
/// Adds a barcode
|
|
/// Will return [true] if this barcode is successfully added
|
|
Future<bool> onScan(String? code) async {
|
|
if (code == null) {
|
|
return false;
|
|
}
|
|
|
|
code = _fixBarcodeIfNecessary(code);
|
|
|
|
if (_latestScannedBarcode == code || _barcodes.contains(code)) {
|
|
lastConsultedBarcode = code;
|
|
return false;
|
|
}
|
|
AnalyticsHelper.trackEvent(
|
|
AnalyticsEvent.scanAction,
|
|
barcode: code,
|
|
);
|
|
|
|
_latestScannedBarcode = code;
|
|
return _addBarcode(code);
|
|
}
|
|
|
|
Future<bool> onCreateProduct(String? barcode) async {
|
|
if (barcode == null) {
|
|
return false;
|
|
}
|
|
return _addBarcode(barcode);
|
|
}
|
|
|
|
Future<void> retryBarcodeFetch(String barcode) async {
|
|
_setBarcodeState(barcode, ScannedProductState.LOADING);
|
|
await _updateBarcode(barcode);
|
|
}
|
|
|
|
Future<bool> _addBarcode(final String barcode) async {
|
|
final ScannedProductState? state = getBarcodeState(barcode);
|
|
if (state == null || state == ScannedProductState.NOT_FOUND) {
|
|
if (!_barcodes.contains(barcode)) {
|
|
_barcodes.add(barcode);
|
|
}
|
|
_setBarcodeState(barcode, ScannedProductState.LOADING);
|
|
_cacheOrLoadBarcode(barcode);
|
|
lastConsultedBarcode = barcode;
|
|
return true;
|
|
}
|
|
if (state == ScannedProductState.FOUND ||
|
|
state == ScannedProductState.CACHED) {
|
|
_barcodes.remove(barcode);
|
|
_barcodes.add(barcode);
|
|
_addProduct(barcode, state);
|
|
if (state == ScannedProductState.CACHED) {
|
|
_updateBarcode(barcode);
|
|
}
|
|
lastConsultedBarcode = barcode;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
Future<void> _cacheOrLoadBarcode(final String barcode) async {
|
|
final bool cached = await _cachedBarcode(barcode);
|
|
if (!cached) {
|
|
_loadBarcode(barcode);
|
|
}
|
|
}
|
|
|
|
Future<bool> _cachedBarcode(final String barcode) async {
|
|
final Product? product = await _daoProduct.get(barcode);
|
|
if (product != null) {
|
|
try {
|
|
// We try to load the fresh copy of product from the server
|
|
final FetchedProduct fetchedProduct =
|
|
await _queryBarcode(barcode).timeout(SnackBarDuration.long);
|
|
if (fetchedProduct.product != null) {
|
|
_addProduct(barcode, ScannedProductState.CACHED);
|
|
return true;
|
|
}
|
|
} on TimeoutException {
|
|
// We tried to load the product from the server,
|
|
// but it was taking more than 5 seconds.
|
|
// So we'll just show the already cached product.
|
|
_addProduct(barcode, ScannedProductState.CACHED);
|
|
return true;
|
|
}
|
|
_addProduct(barcode, ScannedProductState.CACHED);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
Future<FetchedProduct> _queryBarcode(
|
|
final String barcode,
|
|
) async =>
|
|
BarcodeProductQuery(
|
|
barcode: barcode,
|
|
daoProduct: _daoProduct,
|
|
isScanned: true,
|
|
).getFetchedProduct();
|
|
|
|
Future<void> _loadBarcode(
|
|
final String barcode,
|
|
) async {
|
|
final FetchedProduct fetchedProduct = await _queryBarcode(barcode);
|
|
switch (fetchedProduct.status) {
|
|
case FetchedProductStatus.ok:
|
|
_addProduct(barcode, ScannedProductState.FOUND);
|
|
return;
|
|
case FetchedProductStatus.internetNotFound:
|
|
_setBarcodeState(barcode, ScannedProductState.NOT_FOUND);
|
|
return;
|
|
case FetchedProductStatus.internetError:
|
|
_setBarcodeState(barcode, ScannedProductState.ERROR_INTERNET);
|
|
return;
|
|
case FetchedProductStatus.codeInvalid:
|
|
_setBarcodeState(barcode, ScannedProductState.ERROR_INVALID_CODE);
|
|
return;
|
|
case FetchedProductStatus.userCancelled:
|
|
// we do nothing
|
|
return;
|
|
}
|
|
}
|
|
|
|
Future<void> _updateBarcode(
|
|
final String barcode,
|
|
) async {
|
|
final FetchedProduct fetchedProduct = await _queryBarcode(barcode);
|
|
switch (fetchedProduct.status) {
|
|
case FetchedProductStatus.ok:
|
|
_addProduct(barcode, ScannedProductState.FOUND);
|
|
return;
|
|
case FetchedProductStatus.internetNotFound:
|
|
_setBarcodeState(barcode, ScannedProductState.NOT_FOUND);
|
|
return;
|
|
case FetchedProductStatus.internetError:
|
|
_setBarcodeState(barcode, ScannedProductState.ERROR_INTERNET);
|
|
return;
|
|
case FetchedProductStatus.codeInvalid:
|
|
_setBarcodeState(barcode, ScannedProductState.ERROR_INVALID_CODE);
|
|
return;
|
|
case FetchedProductStatus.userCancelled:
|
|
// we do nothing
|
|
return;
|
|
}
|
|
}
|
|
|
|
Future<void> _addProduct(
|
|
final String barcode,
|
|
final ScannedProductState state,
|
|
) async {
|
|
if (_latestFoundBarcode != barcode) {
|
|
_latestFoundBarcode = barcode;
|
|
await _daoProductList.push(productList, _latestFoundBarcode!);
|
|
await _daoProductList.push(_scanHistory, _latestFoundBarcode!);
|
|
await _daoProductList.push(_history, _latestFoundBarcode!);
|
|
_daoProductList.localDatabase.notifyListeners();
|
|
}
|
|
_setBarcodeState(barcode, state);
|
|
}
|
|
|
|
Future<void> clearScanSession() async {
|
|
await _daoProductList.clear(productList);
|
|
await refresh();
|
|
}
|
|
|
|
Future<void> removeBarcode(
|
|
final String barcode,
|
|
) async {
|
|
await _daoProductList.set(
|
|
productList,
|
|
barcode,
|
|
false,
|
|
);
|
|
_barcodes.remove(barcode);
|
|
|
|
if (barcode == _latestScannedBarcode) {
|
|
_latestScannedBarcode = null;
|
|
}
|
|
|
|
notifyListeners();
|
|
}
|
|
|
|
Future<void> refresh() async {
|
|
await _refresh();
|
|
notifyListeners();
|
|
}
|
|
|
|
/// Sometimes the scanner may fail, this is a simple fix for now
|
|
/// But could be improved in the future
|
|
String _fixBarcodeIfNecessary(String code) {
|
|
if (code.length == 12) {
|
|
return '0$code';
|
|
} else {
|
|
return code;
|
|
}
|
|
}
|
|
}
|