import 'dart:async'; import 'dart:io'; import 'package:cloud_firestore/cloud_firestore.dart'; import 'package:fcs/data/services/services.dart'; import 'package:fcs/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; import '../../../helpers/shared_pref.dart'; class PackageModel extends BaseModel { final log = Logger('PackageModel'); PaginatorListener? packages; PaginatorListener? customerPackages; PaginatorListener? activePackages; bool isLoading = false; int selectedIndexForCustomer = 1; int selectedIndex = 0; initDataForCustomer(int index) { selectedIndexForCustomer = index; _loadPaginationCustomerPackages(selectedIndexForCustomer == 2); } initData(int index) { selectedIndex = index; _loadPaginationPackages(selectedIndex); } @override void privilegeChanged() { if (user != null) { _loadPaginationActivePackages(); } } @override logout() async { if (customerPackages != null) customerPackages!.close(); if (packages != null) packages!.close(); if (activePackages != null) activePackages!.close(); } onChangedForCustomer(int index) { selectedIndexForCustomer = index; _loadPaginationCustomerPackages(selectedIndexForCustomer == 2); notifyListeners(); } _loadPaginationPackages(int index) { if (user == null) return; if (!((user!.hasPackages() || user!.hasReceiving() || user!.hasProcessing()))) { return; } String path = "/$packages_collection"; Query col = FirebaseFirestore.instance.collection(path); Query pageQuery = FirebaseFirestore.instance.collection(path); // received status if (index == 1) { col = col.where("status", isEqualTo: package_received_status); pageQuery = pageQuery.where("status", isEqualTo: package_received_status); } // processed status if (index == 2) { col = col.where("status", isEqualTo: package_processed_status); pageQuery = pageQuery.where("status", isEqualTo: package_processed_status); } // packed status if (index == 3) { col = col.where("status", isEqualTo: package_packed_status); pageQuery = pageQuery.where("status", isEqualTo: package_packed_status); } // shipped status if (index == 4) { col = col.where("status", isEqualTo: package_shipped_status); pageQuery = pageQuery.where("status", isEqualTo: package_shipped_status); } // delivered status if (index == 5) { col = col.where("status", isEqualTo: package_delivered_status); pageQuery = pageQuery.where("status", isEqualTo: package_delivered_status); } pageQuery = pageQuery.orderBy("update_time", descending: true); packages?.close(); packages = PaginatorListener( col, pageQuery, (data, id) => Package.fromMap(data, id), rowPerLoad: 30); } onChanged(int index) { selectedIndex = index; _loadPaginationPackages(index); notifyListeners(); } _loadPaginationCustomerPackages(bool isDelivered) { if (user == null) return; String path = "/$packages_collection"; Query col = 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?.close(); customerPackages = PaginatorListener( col, pageQuery, (data, id) => Package.fromMap(data, id), rowPerLoad: 30); } _loadPaginationActivePackages() { if (user == null) return; if (!((user!.hasPackages() || user!.hasReceiving() || user!.hasProcessing()))) { return; } String path = "/$packages_collection"; Query col = FirebaseFirestore.instance .collection(path) .where("is_delivered", isEqualTo: false); Query pageQuery = FirebaseFirestore.instance .collection(path) .where("is_delivered", isEqualTo: false) .orderBy("update_time", descending: true); activePackages?.close(); activePackages = PaginatorListener( col, pageQuery, (data, id) => Package.fromMap(data, id), rowPerLoad: 30); } 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(), 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.isNotEmpty) { var snap = qsnap.docs.first; var package = Package.fromMap(snap.data(), 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.isNotEmpty) { var snap = qsnap.docs.first; var package = Package.fromMap(snap.data(), 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(), documentSnapshot.id); return p; }).toList(); } catch (e) { log.warning("Error!! $e"); } return packages; } Future> searchUser(String term) async { if (term != '') { await SharedPref.saveRecentSearch('account_search', term); } return Services.instance.userService.searchUser(term); } Future> searchPackage(String term) async { if (term != '') { await SharedPref.saveRecentSearch('package_search', term); } 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 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; String path = Path.join(pkg_files_path); List urls = await uploadFiles(path, files); package.photoUrls = urls; } 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.isNotEmpty) { for (String? url in deletedUrls) { package.photoUrls.remove(url); } } List uploadedURL = []; if (files.isNotEmpty) { var count = (package.photoUrls.length) + files.length - (deletedUrls.length); if (count > uploadPhotoLimit) throw Exception("Exceed number of file upload"); package.photoUrls = 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.isNotEmpty) { for (String? url in deletedUrls) { package.photoUrls.remove(url); } } List uploadedURL = []; if (files.isNotEmpty) { var count = (package.photoUrls.length) + files.length - (deletedUrls.length); if (count > uploadPhotoLimit) { throw Exception("Exceed number of file upload"); } package.photoUrls = package.photoUrls; String path = Path.join(pkg_files_path); uploadedURL = await uploadFiles(path, files); for (var url in uploadedURL) { 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!); } Future> getPackagesByIds(List packageIds) async { List packages = []; try { for (var e in packageIds) { Package? p = await getPackage(e); if (p != null) { packages.add(p); } } } catch (e) { log.warning("Error!! $e"); } return packages; } }