import 'dart:async'; import 'dart:io'; import 'package:cloud_firestore/cloud_firestore.dart'; import 'package:fcs/data/services/services.dart'; import 'package:fcs/domain/constants.dart'; import 'package:fcs/domain/entities/package.dart'; import 'package:fcs/domain/entities/user.dart'; import 'package:fcs/domain/vo/delivery_address.dart'; import 'package:fcs/helpers/firebase_helper.dart'; import 'package:fcs/pages/main/model/base_model.dart'; import 'package:fcs/pagination/paginator_listener.dart'; import 'package:logging/logging.dart'; import 'package:path/path.dart' as Path; class PackageModel extends BaseModel { final log = Logger('PackageModel'); late PaginatorListener packages; late PaginatorListener customerPackages; late PaginatorListener activePackages; bool isLoading = false; int _menuSelectedIndex = 1; set menuSelectedIndex(int index) { _menuSelectedIndex = index; _loadPackages(_menuSelectedIndex == 2); _loadCustomerPackages(_menuSelectedIndex == 2); notifyListeners(); } int get menuSelectedIndex => _menuSelectedIndex; void privilegeChanged() { if (user != null) { _initData(); } } Future _initData() async { logout(); _menuSelectedIndex = 1; packages = PaginatorListener( (data, id) => Package.fromMap(data, id), onChange: () { notifyListeners(); }, rowPerLoad: 30, insertNewByListener: true); customerPackages = PaginatorListener( (data, id) => Package.fromMap(data, id), onChange: () { notifyListeners(); }, rowPerLoad: 30, insertNewByListener: true); activePackages = PaginatorListener( (data, id) => Package.fromMap(data, id), onChange: () { notifyListeners(); }, rowPerLoad: 30, insertNewByListener: true); _loadPackages(_menuSelectedIndex == 2); _loadCustomerPackages(_menuSelectedIndex == 2); _loadActivePackages(); } @override logout() async { if (customerPackages != null) customerPackages.close(); if (packages != null) packages.close(); if (activePackages != null) activePackages.close(); } Future _loadPackages(bool isDelivered) async { if (user == null) return; if (!((user!.hasPackages() || user!.hasReceiving() || user!.hasProcessing()))) return; String path = "/$packages_collection"; try { Query listenerQuery = FirebaseFirestore.instance .collection(path) .where("is_delivered", isEqualTo: isDelivered); Query pageQuery = FirebaseFirestore.instance .collection(path) .where("is_delivered", isEqualTo: isDelivered); pageQuery = pageQuery.orderBy("update_time", descending: true); packages.refresh(listeningQuery: listenerQuery, pageQuery: pageQuery); } catch (e) { log.warning("Error!! $e"); } } Future _loadCustomerPackages(bool isDelivered) async { if (user == null) return; String path = "/$packages_collection"; try { Query listenerQuery = FirebaseFirestore.instance .collection(path) .where("is_delivered", isEqualTo: isDelivered) .where("user_id", isEqualTo: user!.id); Query pageQuery = FirebaseFirestore.instance .collection(path) .where("is_delivered", isEqualTo: isDelivered) .where("user_id", isEqualTo: user!.id) .orderBy("update_time", descending: true); customerPackages.refresh( listeningQuery: listenerQuery, pageQuery: pageQuery); } catch (e) { log.warning("Error!! $e"); } } Future _loadActivePackages() async { if (user == null) return; if (!((user!.hasPackages() || user!.hasReceiving() || user!.hasProcessing()))) return; String path = "/$packages_collection"; try { Query listenerQuery = FirebaseFirestore.instance .collection(path) .where("is_delivered", isEqualTo: false); Query pageQuery = FirebaseFirestore.instance .collection(path) .where("is_delivered", isEqualTo: false); pageQuery = pageQuery.orderBy("update_time", descending: true); activePackages.refresh( listeningQuery: listenerQuery, pageQuery: pageQuery); } catch (e) { log.warning("Error!! $e"); } } Future getPackage(String id) async { if (user == null) return null; String path = "/$packages_collection"; try { DocumentSnapshot snap = await FirebaseFirestore.instance.collection("$path").doc(id).get(); if (snap.exists) { var package = Package.fromMap(snap.data as Map, snap.id); return package; } } catch (e) { log.warning("Error!! $e"); } return null; } Future getPackageByTrackingID(String trackingID) async { if (user == null) return null; String path = "/$packages_collection"; try { var snaps = await FirebaseFirestore.instance .collection("$path") .where("tracking_id", isEqualTo: trackingID) .where("is_deleted", isEqualTo: false) .get(const GetOptions(source: Source.server)); if (snaps.docs.length == 1) { var snap = snaps.docs[0]; var package = Package.fromMap(snap.data as Map, snap.id); return package; } } catch (e) { log.warning("Error!! $e"); } return null; } Future lookupPackage(String trackingID) async { if (user == null) return null; String path = "/$packages_collection"; try { var qsnap = await FirebaseFirestore.instance .collection("$path") .where("tracking_id", isEqualTo: trackingID) .where("has_user_id", isEqualTo: false) .where("is_deleted", isEqualTo: false) .get(const GetOptions(source: Source.server)); if (qsnap.docs.length > 0) { var snap = qsnap.docs[0]; if (snap.exists) { var package = Package.fromMap(snap.data as Map, snap.id); return package; } } qsnap = await FirebaseFirestore.instance .collection("$path") .where("tracking_id", isEqualTo: trackingID) .where("user_id", isEqualTo: user!.id) .where("is_deleted", isEqualTo: false) .get(const GetOptions(source: Source.server)); if (qsnap.docs.length > 0) { var snap = qsnap.docs[0]; if (snap.exists) { var package = Package.fromMap(snap.data as Map, snap.id); return package; } } } catch (e) { log.warning("Error!! $e"); } return null; } Future> getPackages(String userID, List status) async { List packages = []; try { var snaps = await FirebaseFirestore.instance .collection("/$packages_collection") .where("status", whereIn: status) .where("user_id", isEqualTo: userID) .where("is_deleted", isEqualTo: false) .get(const GetOptions(source: Source.server)); packages = snaps.docs.map((documentSnapshot) { var p = Package.fromMap( documentSnapshot.data as Map, documentSnapshot.id); return p; }).toList(); } catch (e) { log.warning("Error!! $e"); } return packages; } Future> searchUser(String term) { return Services.instance.userService.searchUser(term); } Future> searchPackage(String term) async { Future> packages = Services.instance.packageService.searchPackage(term); Future?> packagesFTS = Services.instance.packageService.ftsSearchPackage(term); // Package pkg = await getPackageByTrackingID(term); // if (pkg != null && !packages.contains(pkg)) { // packages.insert(0, pkg); // } List pkgs = await packages; List? ftsPkgs = await packagesFTS; pkgs.addAll(ftsPkgs!); final seen = Set(); return pkgs.where((e) => seen.add(e)).toList(); } Future createPackages(User user, List packages) { return Services.instance.packageService .createPackages(packages, user.fcsID!); } Future createReceiving( User user, Package package, List files) async { if (user != null) { package.fcsID = user.fcsID; } if (files != null) { if (files.length > uploadPhotoLimit) { throw Exception("Exceed number of file upload"); } package.photoUrls = package.photoUrls == null ? [] : package.photoUrls; String path = Path.join(pkg_files_path); List urls = await uploadFiles(path, files); urls.forEach((url) { package.photoUrls.add(url); }); } try { return Services.instance.packageService.createReceiving(package); } catch (e) { try { // delete uploaded photos if create fails deleteStorageFromUrls(package.photoUrls); } catch (e) {} throw e; } } Future updateReceiving(User user, Package package, List files, List deletedUrls) async { if (user != null) { package.fcsID = user.fcsID; } if (deletedUrls != null) { for (String url in deletedUrls) { package.photoUrls.remove(url); } } List uploadedURL = []; if (files != null) { var count = (package.photoUrls?.length ?? 0) + files.length - (deletedUrls?.length ?? 0); if (count > uploadPhotoLimit) throw Exception("Exceed number of file upload"); package.photoUrls = package.photoUrls == null ? [] : package.photoUrls; String path = Path.join(pkg_files_path); uploadedURL = await uploadFiles(path, files); uploadedURL.forEach((url) { package.photoUrls.add(url); }); } try { await Services.instance.packageService.updateReceiving(package); } catch (e) { // delete newly uploaded photos if fails try { deleteStorageFromUrls(uploadedURL); package.photoUrls.removeWhere((i) => uploadedURL.contains(i)); } catch (e) {} throw e; } return deleteStorageFromUrls(deletedUrls); } Future deleteReceiving(Package package) { return Services.instance.packageService.deleteReceiving(package); } Future updateProcessing( Package package, List files, List deletedUrls) async { if (deletedUrls != null) { for (String url in deletedUrls) { package.photoUrls.remove(url); } } List uploadedURL = []; if (files != null) { var count = (package.photoUrls?.length ?? 0) + files.length - (deletedUrls?.length ?? 0); if (count > uploadPhotoLimit) throw Exception("Exceed number of file upload"); package.photoUrls = package.photoUrls == null ? [] : package.photoUrls; String path = Path.join(pkg_files_path); uploadedURL = await uploadFiles(path, files); uploadedURL.forEach((url) { package.photoUrls.add(url); }); package.photoUrls.removeWhere((e) => deletedUrls.contains(e)); } try { await Services.instance.packageService.updateProcessing(package); } catch (e) { // delete newly uploaded photos if fails try { deleteStorageFromUrls(uploadedURL); package.photoUrls.removeWhere((i) => uploadedURL.contains(i)); } catch (e) {} throw e; } return deleteStorageFromUrls(deletedUrls); } Future deleteProcessing(Package package) { return Services.instance.packageService.deleteProcessing(package); } Future changeDeliveryAddress( Package package, DeliveryAddress deliveryAddress) { return Services.instance.packageService .changeDeliveryAddress(package.id!, deliveryAddress.id!); } Future packageReturn(Package package) { return Services.instance.packageService.packageReturn(package.id!); } }