Merge remote-tracking branch 'ptd/master'

This commit is contained in:
tzw
2021-09-10 17:02:28 +06:30
69 changed files with 601 additions and 549 deletions

View File

@@ -22,6 +22,7 @@ import 'package:fcs/pages/widgets/local_title.dart';
import 'package:fcs/pages/widgets/progress.dart'; import 'package:fcs/pages/widgets/progress.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_vector_icons/flutter_vector_icons.dart';
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
@@ -59,9 +60,8 @@ class _DeliveryInfoState extends State<DeliveryInfo> {
@override @override
void initState() { void initState() {
super.initState(); super.initState();
if(widget.box != null) if (widget.box != null) _box = widget.box!;
_box = widget.box!; _selectedCartonType = _box.cartonType ?? '';
_selectedCartonType = _box.cartonType;
//for shipment weight //for shipment weight
volumetricRatio = Provider.of<ShipmentRateModel>(context, listen: false) volumetricRatio = Provider.of<ShipmentRateModel>(context, listen: false)
@@ -80,7 +80,7 @@ class _DeliveryInfoState extends State<DeliveryInfo> {
_heightController.text = _box.height.toString(); _heightController.text = _box.height.toString();
_lengthController.text = _box.length.toString(); _lengthController.text = _box.length.toString();
_cargoTypes = _box.cargoTypes; _cargoTypes = _box.cargoTypes;
_deliveryAddress = _box.deliveryAddress; _deliveryAddress = _box.deliveryAddress!;
isMixBox = _box.cartonType == carton_mix_box; isMixBox = _box.cartonType == carton_mix_box;
isFromShipments = _box.cartonType == carton_from_shipments; isFromShipments = _box.cartonType == carton_from_shipments;
isFromPackages = _box.cartonType == carton_from_packages; isFromPackages = _box.cartonType == carton_from_packages;
@@ -96,7 +96,7 @@ class _DeliveryInfoState extends State<DeliveryInfo> {
if (_box.cartonType == carton_from_packages && _box.userID == null) return; if (_box.cartonType == carton_from_packages && _box.userID == null) return;
PackageModel packageModel = PackageModel packageModel =
Provider.of<PackageModel>(context, listen: false); Provider.of<PackageModel>(context, listen: false);
List<Package> packages = await packageModel.getPackages(_box.userID, [ List<Package> packages = await packageModel.getPackages(_box.userID!, [
package_processed_status, package_processed_status,
package_packed_status, package_packed_status,
package_shipped_status, package_shipped_status,
@@ -135,8 +135,9 @@ class _DeliveryInfoState extends State<DeliveryInfo> {
final cartonTypeBox = LocalRadioButtons( final cartonTypeBox = LocalRadioButtons(
readOnly: true, readOnly: true,
values: cartonModel.cartonTypesInfo, values: cartonModel.cartonTypesInfo,
selectedValue: selectedValue: (_box.isShipmentCarton ?? false)
_box.isShipmentCarton ? carton_from_shipments : _box.cartonType); ? carton_from_shipments
: _box.cartonType);
final shipmentBox = DisplayText( final shipmentBox = DisplayText(
text: _box.fcsShipmentNumber, text: _box.fcsShipmentNumber,
labelTextKey: "box.fcs_shipment_num", labelTextKey: "box.fcs_shipment_num",
@@ -173,11 +174,11 @@ class _DeliveryInfoState extends State<DeliveryInfo> {
children: <Widget>[ children: <Widget>[
Expanded( Expanded(
child: new Text( child: new Text(
_selectedShipmentBox.shipmentNumber, _selectedShipmentBox.shipmentNumber ?? "",
style: textStyle, style: textStyle,
)), )),
new Text( new Text(
_selectedShipmentBox.desc, _selectedShipmentBox.desc ?? "",
style: textStyle, style: textStyle,
), ),
], ],
@@ -214,7 +215,7 @@ class _DeliveryInfoState extends State<DeliveryInfo> {
); );
final shipmentWeightBox = DisplayText( final shipmentWeightBox = DisplayText(
text: shipmentWeight.toStringAsFixed(0) : "", text: shipmentWeight.toStringAsFixed(0),
labelTextKey: "box.shipment_weight", labelTextKey: "box.shipment_weight",
iconData: MaterialCommunityIcons.weight, iconData: MaterialCommunityIcons.weight,
); );

View File

@@ -3,6 +3,7 @@ import 'package:fcs/helpers/theme.dart';
import 'package:fcs/pages/main/util.dart'; import 'package:fcs/pages/main/util.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_vector_icons/flutter_vector_icons.dart';
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
import 'delivery_info.dart'; import 'delivery_info.dart';

View File

@@ -14,18 +14,18 @@ class DeliveryModel extends BaseModel {
List<Carton> get cartons => List<Carton> get cartons =>
_selectedIndex == 1 ? _cartons : List<Carton>.from(_delivered.values); _selectedIndex == 1 ? _cartons : List<Carton>.from(_delivered.values);
Paginator _delivered; late Paginator _delivered;
int _selectedIndex = 1; int _selectedIndex = 1;
bool isLoading = false; bool isLoading = false;
List<Carton> _cartons = []; List<Carton> _cartons = [];
StreamSubscription<QuerySnapshot> listener; StreamSubscription<QuerySnapshot>? listener;
set selectedIndex(int index) { set selectedIndex(int index) {
_selectedIndex = index; _selectedIndex = index;
notifyListeners(); notifyListeners();
} }
get selectedIndex => _selectedIndex; int get selectedIndex => _selectedIndex;
initData() { initData() {
_selectedIndex = 1; _selectedIndex = 1;
@@ -37,12 +37,12 @@ class DeliveryModel extends BaseModel {
} }
Future<void> _loadCartons() async { Future<void> _loadCartons() async {
if (user == null || !user.hasDeliveries()) return; if (user == null || !user!.hasDeliveries()) return;
String path = "/$cartons_collection/"; String path = "/$cartons_collection/";
if (listener != null) listener.cancel(); if (listener != null) listener!.cancel();
_cartons = []; _cartons = [];
try { try {
listener = Firestore.instance listener = FirebaseFirestore.instance
.collection("$path") .collection("$path")
.where("status", isEqualTo: carton_shipped_status) .where("status", isEqualTo: carton_shipped_status)
.where("carton_type", whereIn: [ .where("carton_type", whereIn: [
@@ -55,9 +55,9 @@ class DeliveryModel extends BaseModel {
.snapshots() .snapshots()
.listen((QuerySnapshot snapshot) { .listen((QuerySnapshot snapshot) {
_cartons.clear(); _cartons.clear();
_cartons = snapshot.documents.map((documentSnapshot) { _cartons = snapshot.docs.map((documentSnapshot) {
var s = Carton.fromMap( var s = Carton.fromMap(
documentSnapshot.data, documentSnapshot.documentID); documentSnapshot.data as Map<String, dynamic>, documentSnapshot.id);
return s; return s;
}).toList(); }).toList();
notifyListeners(); notifyListeners();
@@ -68,9 +68,9 @@ class DeliveryModel extends BaseModel {
} }
Paginator _getDelivered() { Paginator _getDelivered() {
if (user == null || !user.hasDeliveries()) return null; if (user == null || !user!.hasDeliveries()) throw "No Privilege";
var pageQuery = Firestore.instance var pageQuery = FirebaseFirestore.instance
.collection("/$cartons_collection") .collection("/$cartons_collection")
.where("is_delivered", isEqualTo: true) .where("is_delivered", isEqualTo: true)
.where("status", whereIn: [carton_delivered_status]).where("is_deleted", .where("status", whereIn: [carton_delivered_status]).where("is_deleted",
@@ -105,7 +105,7 @@ class DeliveryModel extends BaseModel {
@override @override
logout() async { logout() async {
if (listener != null) await listener.cancel(); if (listener != null) await listener!.cancel();
if (_delivered != null) _delivered.close(); if (_delivered != null) _delivered.close();
_cartons = []; _cartons = [];
} }

View File

@@ -8,6 +8,7 @@ import 'package:fcs/pages/widgets/local_text.dart';
import 'package:fcs/pages/widgets/progress.dart'; import 'package:fcs/pages/widgets/progress.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_vector_icons/flutter_vector_icons.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
class DeliveryAddressEditor extends StatefulWidget { class DeliveryAddressEditor extends StatefulWidget {
@@ -38,12 +39,12 @@ class _DeliveryAddressEditorState extends State<DeliveryAddressEditor> {
if (widget.deliveryAddress != null) { if (widget.deliveryAddress != null) {
_isNew = false; _isNew = false;
_deliveryAddress = widget.deliveryAddress!; _deliveryAddress = widget.deliveryAddress!;
_nameController.text = _deliveryAddress.fullName; _nameController.text = _deliveryAddress.fullName ?? "";
_address1Controller.text = _deliveryAddress.addressLine1; _address1Controller.text = _deliveryAddress.addressLine1 ?? "";
_address2Controller.text = _deliveryAddress.addressLine2; _address2Controller.text = _deliveryAddress.addressLine2 ?? "";
_cityController.text = _deliveryAddress.city; _cityController.text = _deliveryAddress.city ?? "";
_stateController.text = _deliveryAddress.state; _stateController.text = _deliveryAddress.state ?? "";
_phoneController.text = _deliveryAddress.phoneNumber; _phoneController.text = _deliveryAddress.phoneNumber?? "";
} else { } else {
_cityController.text = "Yangon"; _cityController.text = "Yangon";
_stateController.text = "Yangon"; _stateController.text = "Yangon";

View File

@@ -3,6 +3,7 @@ import 'package:fcs/helpers/theme.dart';
import 'package:fcs/pages/widgets/local_text.dart'; import 'package:fcs/pages/widgets/local_text.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_vector_icons/flutter_vector_icons.dart';
typedef SelectionCallback(DeliveryAddress deliveryAddress); typedef SelectionCallback(DeliveryAddress deliveryAddress);

View File

@@ -1,4 +1,5 @@
import 'dart:async'; import 'dart:async';
import 'package:barcode_scan2/gen/protos/protos.pb.dart';
import 'package:cloud_firestore/cloud_firestore.dart'; import 'package:cloud_firestore/cloud_firestore.dart';
import 'package:fcs/data/services/services.dart'; import 'package:fcs/data/services/services.dart';
import 'package:fcs/domain/vo/delivery_address.dart'; import 'package:fcs/domain/vo/delivery_address.dart';
@@ -10,13 +11,13 @@ class DeliveryAddressModel extends BaseModel {
final log = Logger('FcsShipmentModel'); final log = Logger('FcsShipmentModel');
List<DeliveryAddress> deliveryAddresses = []; List<DeliveryAddress> deliveryAddresses = [];
StreamSubscription<QuerySnapshot> listener; StreamSubscription<QuerySnapshot>? listener;
DeliveryAddress get defalutAddress => DeliveryAddress get defalutAddress =>
deliveryAddresses.firstWhere((e) => e.isDefault, orElse: () => null); deliveryAddresses.firstWhere((e) => e.isDefault, orElse: () => DeliveryAddress());
DeliveryAddress getLocalDeliveryAddress(String id) => DeliveryAddress getLocalDeliveryAddress(String id) =>
deliveryAddresses.firstWhere((e) => e.id == id, orElse: () => null); deliveryAddresses.firstWhere((e) => e.id == id, orElse: () => DeliveryAddress());
@override @override
void privilegeChanged() { void privilegeChanged() {
@@ -26,26 +27,26 @@ class DeliveryAddressModel extends BaseModel {
@override @override
logout() async { logout() async {
if (listener != null) await listener.cancel(); if (listener != null) await listener!.cancel();
deliveryAddresses = []; deliveryAddresses = [];
} }
Future<void> _loadDeliveryAddresses() async { Future<void> _loadDeliveryAddresses() async {
if (user == null) return; if (user == null) return;
String path = "$delivery_address_collection/"; String path = "$delivery_address_collection/";
if (listener != null) listener.cancel(); if (listener != null) listener!.cancel();
deliveryAddresses = []; deliveryAddresses = [];
try { try {
listener = Firestore.instance listener = FirebaseFirestore.instance
.collection('users') .collection('users')
.document("${user.id}") .doc("${user!.id}")
.collection("$path") .collection("$path")
.snapshots() .snapshots()
.listen((QuerySnapshot snapshot) { .listen((QuerySnapshot snapshot) {
deliveryAddresses.clear(); deliveryAddresses.clear();
deliveryAddresses = snapshot.documents.map((documentSnapshot) { deliveryAddresses = snapshot.docs.map((documentSnapshot) {
var s = DeliveryAddress.fromMap( var s = DeliveryAddress.fromMap(
documentSnapshot.data, documentSnapshot.documentID); documentSnapshot.data as Map<String,dynamic>, documentSnapshot.id);
return s; return s;
}).toList(); }).toList();
notifyListeners(); notifyListeners();
@@ -56,9 +57,9 @@ class DeliveryAddressModel extends BaseModel {
} }
Future<DeliveryAddress> getDeliveryAddress(String id) async { Future<DeliveryAddress> getDeliveryAddress(String id) async {
String path = "/$user_collection/${user.id}/$delivery_address_collection"; String path = "/$user_collection/${user!.id}/$delivery_address_collection";
var snap = await Firestore.instance.collection(path).document(id).get(); var snap = await FirebaseFirestore.instance.collection(path).doc(id).get();
return DeliveryAddress.fromMap(snap.data, snap.documentID); return DeliveryAddress.fromMap(snap.data as Map<String,dynamic>, snap.id);
} }
void initUser(user) { void initUser(user) {
@@ -87,14 +88,14 @@ class DeliveryAddressModel extends BaseModel {
Future<List<DeliveryAddress>> getDeliveryAddresses(String userID) async { Future<List<DeliveryAddress>> getDeliveryAddresses(String userID) async {
String path = "$delivery_address_collection/"; String path = "$delivery_address_collection/";
var querySnap = await Firestore.instance var querySnap = await FirebaseFirestore.instance
.collection('users') .collection('users')
.document("$userID") .doc("$userID")
.collection("$path") .collection("$path")
.orderBy("full_name") .orderBy("full_name")
.getDocuments(); .get();
return querySnap.documents return querySnap.docs
.map((e) => DeliveryAddress.fromMap(e.data, e.documentID)) .map((e) => DeliveryAddress.fromMap(e.data as Map<String,dynamic>, e.id))
.toList(); .toList();
} }
} }

View File

@@ -9,6 +9,7 @@ import 'package:fcs/pages/widgets/input_text.dart';
import 'package:fcs/pages/widgets/progress.dart'; import 'package:fcs/pages/widgets/progress.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_vector_icons/flutter_vector_icons.dart';
import 'package:font_awesome_flutter/font_awesome_flutter.dart'; import 'package:font_awesome_flutter/font_awesome_flutter.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
@@ -36,11 +37,11 @@ class _DiscountEditorState extends State<DiscountEditor> {
super.initState(); super.initState();
if (widget.discount != null) { if (widget.discount != null) {
_discount = widget.discount!; _discount = widget.discount!;
_codeController.text = _discount.code; _codeController.text = _discount.code ?? "";
_amountController.text = _discount.amount.toStringAsFixed(2); _amountController.text = _discount.amount.toStringAsFixed(2);
_statusController.text = _discount.status; _statusController.text = _discount.status ?? '';
customerName = _discount.customerName; customerName = _discount.customerName ?? "";
customerId = _discount.customerId; customerId = _discount.customerId ?? "";
} else { } else {
_isNew = true; _isNew = true;
} }
@@ -76,8 +77,8 @@ class _DiscountEditorState extends State<DiscountEditor> {
icon: Icon(Icons.search, color: primaryColor), icon: Icon(Icons.search, color: primaryColor),
onPressed: () => searchUser(context, onUserSelect: (u) { onPressed: () => searchUser(context, onUserSelect: (u) {
setState(() { setState(() {
customerId = u.id; customerId = u.id ?? "";
customerName = u.name; customerName = u.name ?? "";
}); });
},popPage: true)), },popPage: true)),
], ],

View File

@@ -5,6 +5,7 @@ import 'package:fcs/pages/discount/discount_editor.dart';
import 'package:fcs/pages/main/util.dart'; import 'package:fcs/pages/main/util.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_vector_icons/flutter_vector_icons.dart';
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
typedef OnSelect(Discount discount); typedef OnSelect(Discount discount);

View File

@@ -30,9 +30,7 @@ class DiscountModel extends BaseModel {
initData() { initData() {
_selectedIndex = 1; _selectedIndex = 1;
_load(); _load();
if (_getUsed() != null) _used = _getUsed();
if (_used != null) _used.close();
if (_getUsed() != null) _used = _getUsed()!;
_used.load(); _used.load();
} }
@@ -44,14 +42,14 @@ class DiscountModel extends BaseModel {
_load() { _load() {
if (listener != null) listener!.cancel(); if (listener != null) listener!.cancel();
try { try {
listener = Firestore.instance listener = FirebaseFirestore.instance
.collection("/$discounts_collection") .collection("/$discounts_collection")
.orderBy("code", descending: false) .orderBy("code", descending: false)
.snapshots() .snapshots()
.listen((snaps) { .listen((snaps) {
_discounts.clear(); _discounts.clear();
snaps.documents.forEach((d) { snaps.docs.forEach((d) {
_discounts.add(Discount.fromMap(d.data, d.documentID)); _discounts.add(Discount.fromMap(d.data as Map<String,dynamic>, d.id));
}); });
notifyListeners(); notifyListeners();
}); });
@@ -60,10 +58,10 @@ class DiscountModel extends BaseModel {
} }
} }
Paginator? _getUsed() { Paginator _getUsed() {
if (user == null || !user.hasFcsShipments()) return null; if (user == null || !user!.hasFcsShipments()) throw "No Privilege";
var pageQuery = Firestore.instance var pageQuery = FirebaseFirestore.instance
.collection("/$discounts_collection") .collection("/$discounts_collection")
.where("status", isEqualTo: fcs_shipment_shipped_status) .where("status", isEqualTo: fcs_shipment_shipped_status)
.orderBy("code", descending: false); .orderBy("code", descending: false);
@@ -73,17 +71,17 @@ class DiscountModel extends BaseModel {
return paginator; return paginator;
} }
Future<List<Discount>?> getDiscount(String userID) async { Future<List<Discount?>?> getDiscount(String userID) async {
String path = "/$discounts_collection"; String path = "/$discounts_collection";
try { try {
var q = Firestore.instance var q = FirebaseFirestore.instance
.collection("$path") .collection("$path")
.where("customer_id", isEqualTo: userID) .where("customer_id", isEqualTo: userID)
.where("status", isEqualTo: "available"); .where("status", isEqualTo: "available");
var snaps = await q.getDocuments(source: Source.server); var snaps = await q.get(const GetOptions(source: Source.server));
List<Discount> discounts = snaps.documents.map((snap) { var discounts = snaps.docs.map((snap) {
if (snap.exists) { if (snap.exists) {
var s = Discount.fromMap(snap.data, snap.documentID); var s = Discount.fromMap(snap.data as Map<String,dynamic>, snap.id);
return s; return s;
} }
}).toList(); }).toList();
@@ -128,6 +126,6 @@ class DiscountModel extends BaseModel {
} }
Future<void> deleteDiscount(Discount discount) async { Future<void> deleteDiscount(Discount discount) async {
return Services.instance.commonService.deleteDiscount(discount.id); return Services.instance.commonService.deleteDiscount(discount.id!);
} }
} }

View File

@@ -28,7 +28,8 @@ class _FAQDetailPageState extends State<FAQDetailPage> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
faq = context.select((FAQModel m) => m.getFAQ(widget.faq.id)); if(widget.faq.id != null)
faq = context.select((FAQModel m) => m.getFAQ(widget.faq.id!));
if (faq == null) return Text("Deleted"); if (faq == null) return Text("Deleted");
bool isEditable = context.select((MainModel m) => m.faqEditable()); bool isEditable = context.select((MainModel m) => m.faqEditable());

View File

@@ -10,6 +10,7 @@ import 'package:fcs/pages/widgets/local_text.dart';
import 'package:fcs/pages/widgets/progress.dart'; import 'package:fcs/pages/widgets/progress.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_vector_icons/flutter_vector_icons.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
const info = "Select additional page"; const info = "Select additional page";
@@ -43,13 +44,13 @@ class _FAQEditorState extends State<FAQEditor> {
if (widget.faq != null) { if (widget.faq != null) {
_faq = widget.faq!; _faq = widget.faq!;
_sn.text = _faq.sn.toString(); _sn.text = _faq.sn.toString();
_engQ.text = _faq.questionEng; _engQ.text = _faq.questionEng ?? "";
_mmQ.text = _faq.questionMm; _mmQ.text = _faq.questionMm ?? '';
_engA.text = _faq.answerEng; _engA.text = _faq.answerEng ?? '';
_mmA.text = _faq.answerMm; _mmA.text = _faq.answerMm ?? '';
_pageLabelEng.text = _faq.pageLinkLabelEng; _pageLabelEng.text = _faq.pageLinkLabelEng ?? "";
_pageLabelMm.text = _faq.pageLinkLabelMm; _pageLabelMm.text = _faq.pageLinkLabelMm ?? "";
_pageLink = _faq.pageLink; _pageLink = _faq.pageLink ?? '';
} }
} }

View File

@@ -12,22 +12,22 @@ class FAQModel extends BaseModel {
List<FAQ> faqs = []; List<FAQ> faqs = [];
FAQ getFAQ(String id) { FAQ getFAQ(String id) {
return faqs.firstWhere((e) => e.id == id, orElse: () => null); return faqs.firstWhere((e) => e.id == id, orElse: () => FAQ());
} }
StreamSubscription<QuerySnapshot> listener; StreamSubscription<QuerySnapshot>? listener;
FAQModel() { FAQModel() {
if (listener != null) listener.cancel(); if (listener != null) listener!.cancel();
try { try {
listener = Firestore.instance listener = FirebaseFirestore.instance
.collection("/faqs") .collection("/faqs")
.orderBy("sn", descending: false) .orderBy("sn", descending: false)
.snapshots() .snapshots()
.listen((snaps) { .listen((snaps) {
faqs.clear(); faqs.clear();
snaps.documents.forEach((d) { snaps.docs.forEach((d) {
faqs.add(FAQ.fromMap(d.data, d.documentID)); faqs.add(FAQ.fromMap(d.data as Map<String,dynamic>, d.id));
}); });
notifyListeners(); notifyListeners();
}); });

View File

@@ -7,14 +7,14 @@ Widget itemTitle(BuildContext context, String textKey) {
return Padding( return Padding(
padding: const EdgeInsets.only(left: 18.0, top: 15, bottom: 0), padding: const EdgeInsets.only(left: 18.0, top: 15, bottom: 0),
child: Text( child: Text(
AppTranslations.of(context).text(textKey), AppTranslations.of(context)!.text(textKey),
style: TextStyle( style: TextStyle(
fontWeight: FontWeight.bold, fontSize: 18, color: Colors.black), fontWeight: FontWeight.bold, fontSize: 18, color: Colors.black),
), ),
); );
} }
Widget subItemTitle(BuildContext context, String textKey, {IconData iconData}) { Widget subItemTitle(BuildContext context, String textKey, {IconData? iconData}) {
return Padding( return Padding(
padding: const EdgeInsets.only(left: 0, top: 0, bottom: 0), padding: const EdgeInsets.only(left: 0, top: 0, bottom: 0),
child: Row( child: Row(
@@ -25,7 +25,7 @@ Widget subItemTitle(BuildContext context, String textKey, {IconData iconData}) {
), ),
SizedBox(width: 10), SizedBox(width: 10),
Text( Text(
AppTranslations.of(context).text(textKey), AppTranslations.of(context)!.text(textKey),
style: TextStyle( style: TextStyle(
fontWeight: FontWeight.w700, fontSize: 15, color: primaryColor), fontWeight: FontWeight.w700, fontSize: 15, color: primaryColor),
), ),
@@ -35,7 +35,7 @@ Widget subItemTitle(BuildContext context, String textKey, {IconData iconData}) {
} }
Widget contactItem(BuildContext context, String text, IconData iconData, Widget contactItem(BuildContext context, String text, IconData iconData,
{Function() onTap, String labelKey}) { {Function()? onTap, String? labelKey}) {
return Material( return Material(
child: Padding( child: Padding(
padding: const EdgeInsets.only(left: 18.0, bottom: 10, right: 18), padding: const EdgeInsets.only(left: 18.0, bottom: 10, right: 18),

View File

@@ -49,19 +49,22 @@ class _FcsShipmentEditorState extends State<FcsShipmentEditor> {
_isNew = widget.shipment == null; _isNew = widget.shipment == null;
if (widget.shipment != null) { if (widget.shipment != null) {
_shipment = widget.shipment!; _shipment = widget.shipment!;
_shipmentNumberController.text = _shipment.shipmentNumber; _shipmentNumberController.text = _shipment.shipmentNumber ?? "";
_cutoffDateController.text = dateFormatter.format(_shipment.cutoffDate); if(_shipment.cutoffDate != null)
_arrivalDateController.text = dateFormatter.format(_shipment.arrivalDate); _cutoffDateController.text = dateFormatter.format(_shipment.cutoffDate!);
if(_shipment.arrivalDate != null)
_arrivalDateController.text = dateFormatter.format(_shipment.arrivalDate!);
if(_shipment.departureDate != null)
_departureDateControler.text = _departureDateControler.text =
dateFormatter.format(_shipment.departureDate); dateFormatter.format(_shipment.departureDate!);
_statusController.text = _shipment.status; _statusController.text = _shipment.status ?? "";
_currentShipmentType = _shipment.shipType; _currentShipmentType = _shipment.shipType;
_consigneeController.text = _shipment.consignee; _consigneeController.text = _shipment.consignee ?? "";
_portController.text = _shipment.port; _portController.text = _shipment.port ?? "";
_destinationController.text = _shipment.destination; _destinationController.text = _shipment.destination ?? "";
} else { } else {
var mainModel = Provider.of<MainModel>(context, listen: false); var mainModel = Provider.of<MainModel>(context, listen: false);
_currentShipmentType = mainModel.setting.shipmentTypes[0]; _currentShipmentType = mainModel.setting!.shipmentTypes[0];
} }
} }
@@ -143,10 +146,10 @@ class _FcsShipmentEditorState extends State<FcsShipmentEditor> {
labelStyle: languageModel.isEng labelStyle: languageModel.isEng
? newLabelStyle(color: Colors.black54, fontSize: 20) ? newLabelStyle(color: Colors.black54, fontSize: 20)
: newLabelStyleMM(color: Colors.black54, fontSize: 20), : newLabelStyleMM(color: Colors.black54, fontSize: 20),
labelText: AppTranslations.of(context) labelText: AppTranslations.of(context)!
.text('FCSshipment.shipment_type'), .text('FCSshipment.shipment_type'),
icon: Icon(Ionicons.ios_airplane, color: primaryColor)), icon: Icon(Ionicons.ios_airplane, color: primaryColor)),
items: mainModel.setting.shipmentTypes items: mainModel.setting!.shipmentTypes
.map((e) => DropdownMenuItem(child: Text(e), value: e)) .map((e) => DropdownMenuItem(child: Text(e), value: e))
.toList(), .toList(),
onChanged: (String? selected) => { onChanged: (String? selected) => {
@@ -285,7 +288,7 @@ class _FcsShipmentEditorState extends State<FcsShipmentEditor> {
_consigneeController.text != "" || _consigneeController.text != "" ||
_portController.text != "" || _portController.text != "" ||
_destinationController.text != "" || _destinationController.text != "" ||
_currentShipmentType != mainModel.setting.shipmentTypes[0]; _currentShipmentType != mainModel.setting!.shipmentTypes[0];
} else { } else {
FcsShipment fcsShipment = _getPayload(); FcsShipment fcsShipment = _getPayload();
return widget.shipment!.isChangedForEdit(fcsShipment); return widget.shipment!.isChangedForEdit(fcsShipment);

View File

@@ -48,17 +48,20 @@ class _FcsShipmentInfoState extends State<FcsShipmentInfo> {
} }
_load() { _load() {
_shipmentNumberController.text = _fcsShipment!.shipmentNumber; _shipmentNumberController.text = _fcsShipment!.shipmentNumber ?? "";
_cutoffDateController.text = dateFormatter.format(_fcsShipment!.cutoffDate); if(_fcsShipment!.cutoffDate != null)
_cutoffDateController.text = dateFormatter.format(_fcsShipment!.cutoffDate!);
if(_fcsShipment!.arrivalDate != null)
_arrivalDateController.text = _arrivalDateController.text =
dateFormatter.format(_fcsShipment!.arrivalDate); dateFormatter.format(_fcsShipment!.arrivalDate!);
if(_fcsShipment!.departureDate != null)
_departureDateControler.text = _departureDateControler.text =
dateFormatter.format(_fcsShipment!.departureDate); dateFormatter.format(_fcsShipment!.departureDate!);
_shipmentTypeControler.text = _fcsShipment!.shipType; _shipmentTypeControler.text = _fcsShipment!.shipType ?? "";
_consigneeController.text = _fcsShipment!.consignee; _consigneeController.text = _fcsShipment!.consignee ?? "";
_portController.text = _fcsShipment!.port; _portController.text = _fcsShipment!.port ?? "";
_destinationController.text = _fcsShipment!.destination; _destinationController.text = _fcsShipment!.destination ?? "";
_statusController.text = _fcsShipment!.status; _statusController.text = _fcsShipment!.status ?? "";
} }
@override @override
@@ -182,6 +185,7 @@ class _FcsShipmentInfoState extends State<FcsShipmentInfo> {
} }
_edit() async { _edit() async {
var f;
bool? updated = await Navigator.push<bool>( bool? updated = await Navigator.push<bool>(
context, context,
CupertinoPageRoute( CupertinoPageRoute(
@@ -189,7 +193,8 @@ class _FcsShipmentInfoState extends State<FcsShipmentInfo> {
); );
if (updated ?? false) { if (updated ?? false) {
var shipmentModel = Provider.of<FcsShipmentModel>(context, listen: false); var shipmentModel = Provider.of<FcsShipmentModel>(context, listen: false);
var f = await shipmentModel.getFcsShipment(_fcsShipment!.id); if(_fcsShipment != null && _fcsShipment!.id != null )
f = await shipmentModel.getFcsShipment(_fcsShipment!.id!);
setState(() { setState(() {
_fcsShipment = f; _fcsShipment = f;
}); });

View File

@@ -43,9 +43,7 @@ class FcsShipmentListRow extends StatelessWidget {
Padding( Padding(
padding: const EdgeInsets.only(left: 8.0), padding: const EdgeInsets.only(left: 8.0),
child: new Text( child: new Text(
shipment!.shipmentNumber == null shipment?.shipmentNumber ?? '',
? ''
: shipment!.shipmentNumber,
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.black), fontSize: 15.0, color: Colors.black),
), ),
@@ -53,7 +51,7 @@ class FcsShipmentListRow extends StatelessWidget {
Padding( Padding(
padding: const EdgeInsets.only(left: 10.0, top: 10), padding: const EdgeInsets.only(left: 10.0, top: 10),
child: new Text( child: new Text(
dateFormatter.format(shipment!.cutoffDate), dateFormatter.format(shipment!.cutoffDate!),
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.grey), fontSize: 15.0, color: Colors.grey),
), ),
@@ -67,7 +65,7 @@ class FcsShipmentListRow extends StatelessWidget {
), ),
Padding( Padding(
padding: const EdgeInsets.all(0), padding: const EdgeInsets.all(0),
child: getStatus(shipment!.status), child: getStatus(shipment!.status ?? ''),
), ),
], ],
), ),

View File

@@ -11,13 +11,13 @@ import 'package:logging/logging.dart';
class FcsShipmentModel extends BaseModel { class FcsShipmentModel extends BaseModel {
final log = Logger('FcsShipmentModel'); final log = Logger('FcsShipmentModel');
StreamSubscription<QuerySnapshot> listener; StreamSubscription<QuerySnapshot>? listener;
List<FcsShipment> _fcsShipments = []; List<FcsShipment> _fcsShipments = [];
List<FcsShipment> get fcsShipments => _selectedIndex == 1 List<FcsShipment> get fcsShipments => _selectedIndex == 1
? _fcsShipments ? _fcsShipments
: List<FcsShipment>.from(_shipped.values); : List<FcsShipment>.from(_shipped.values);
Paginator _shipped; late Paginator _shipped;
bool isLoading = false; bool isLoading = false;
int _selectedIndex = 1; int _selectedIndex = 1;
set selectedIndex(int index) { set selectedIndex(int index) {
@@ -25,7 +25,7 @@ class FcsShipmentModel extends BaseModel {
notifyListeners(); notifyListeners();
} }
get selectedIndex => _selectedIndex; int get selectedIndex => _selectedIndex;
@override @override
void privilegeChanged() { void privilegeChanged() {
@@ -43,12 +43,12 @@ class FcsShipmentModel extends BaseModel {
} }
Future<void> _loadFcsShipments() async { Future<void> _loadFcsShipments() async {
if (user == null || !user.hasFcsShipments()) return; if (user == null || !user!.hasFcsShipments()) throw "No Privilege";
String path = "/$fcs_shipment_collection/"; String path = "/$fcs_shipment_collection/";
if (listener != null) listener.cancel(); if (listener != null) listener!.cancel();
_fcsShipments = []; _fcsShipments = [];
try { try {
listener = Firestore.instance listener = FirebaseFirestore.instance
.collection("$path") .collection("$path")
.where("status", isEqualTo: fcs_shipment_confirmed_status) .where("status", isEqualTo: fcs_shipment_confirmed_status)
.where("is_deleted", isEqualTo: false) .where("is_deleted", isEqualTo: false)
@@ -56,9 +56,9 @@ class FcsShipmentModel extends BaseModel {
.snapshots() .snapshots()
.listen((QuerySnapshot snapshot) { .listen((QuerySnapshot snapshot) {
_fcsShipments.clear(); _fcsShipments.clear();
_fcsShipments = snapshot.documents.map((documentSnapshot) { _fcsShipments = snapshot.docs.map((documentSnapshot) {
var s = FcsShipment.fromMap( var s = FcsShipment.fromMap(
documentSnapshot.data, documentSnapshot.documentID); documentSnapshot.data as Map<String,dynamic>, documentSnapshot.id);
return s; return s;
}).toList(); }).toList();
notifyListeners(); notifyListeners();
@@ -69,9 +69,9 @@ class FcsShipmentModel extends BaseModel {
} }
Paginator _getShipped() { Paginator _getShipped() {
if (user == null || !user.hasFcsShipments()) return null; if (user == null || !user!.hasFcsShipments()) throw "No Privilege";
var pageQuery = Firestore.instance var pageQuery = FirebaseFirestore.instance
.collection("/$fcs_shipment_collection") .collection("/$fcs_shipment_collection")
.where("status", isEqualTo: fcs_shipment_shipped_status) .where("status", isEqualTo: fcs_shipment_shipped_status)
.where("is_deleted", isEqualTo: false) .where("is_deleted", isEqualTo: false)
@@ -102,13 +102,13 @@ class FcsShipmentModel extends BaseModel {
Future<List<FcsShipment>> getActiveFcsShipments() async { Future<List<FcsShipment>> getActiveFcsShipments() async {
List<FcsShipment> fcsShipments = []; List<FcsShipment> fcsShipments = [];
try { try {
var snaps = await Firestore.instance var snaps = await FirebaseFirestore.instance
.collection("/$fcs_shipment_collection") .collection("/$fcs_shipment_collection")
.where("status", isEqualTo: fcs_shipment_confirmed_status) .where("status", isEqualTo: fcs_shipment_confirmed_status)
.getDocuments(source: Source.server); .get(const GetOptions(source: Source.server));
fcsShipments = snaps.documents.map((documentSnapshot) { fcsShipments = snaps.docs.map((documentSnapshot) {
var fcs = FcsShipment.fromMap( var fcs = FcsShipment.fromMap(
documentSnapshot.data, documentSnapshot.documentID); documentSnapshot.data as Map<String,dynamic>, documentSnapshot.id);
return fcs; return fcs;
}).toList(); }).toList();
} catch (e) { } catch (e) {
@@ -117,13 +117,13 @@ class FcsShipmentModel extends BaseModel {
return fcsShipments; return fcsShipments;
} }
Future<FcsShipment> getFcsShipment(String id) async { Future<FcsShipment?> getFcsShipment(String id) async {
try { try {
var snap = await Firestore.instance var snap = await FirebaseFirestore.instance
.collection("/$fcs_shipment_collection") .collection("/$fcs_shipment_collection")
.document(id) .doc(id)
.get(source: Source.server); .get(const GetOptions(source: Source.server));
var fcs = FcsShipment.fromMap(snap.data, snap.documentID); var fcs = FcsShipment.fromMap(snap.data as Map<String,dynamic>, snap.id);
return fcs; return fcs;
} catch (e) { } catch (e) {
@@ -135,13 +135,13 @@ class FcsShipmentModel extends BaseModel {
Future<List<FcsShipment>> getInvoiceFcsShipments() async { Future<List<FcsShipment>> getInvoiceFcsShipments() async {
List<FcsShipment> fcsShipments = []; List<FcsShipment> fcsShipments = [];
try { try {
var snaps = await Firestore.instance var snaps = await FirebaseFirestore.instance
.collection("/$fcs_shipment_collection") .collection("/$fcs_shipment_collection")
.where("pending_invoice_user_count", isGreaterThan: 0) .where("pending_invoice_user_count", isGreaterThan: 0)
.getDocuments(source: Source.server); .get(const GetOptions(source: Source.server));
fcsShipments = snaps.documents.map((documentSnapshot) { fcsShipments = snaps.docs.map((documentSnapshot) {
var fcs = FcsShipment.fromMap( var fcs = FcsShipment.fromMap(
documentSnapshot.data, documentSnapshot.documentID); documentSnapshot.data as Map<String,dynamic>, documentSnapshot.id);
return fcs; return fcs;
}).toList(); }).toList();
} catch (e) { } catch (e) {
@@ -156,7 +156,7 @@ class FcsShipmentModel extends BaseModel {
@override @override
logout() async { logout() async {
if (listener != null) await listener.cancel(); if (listener != null) await listener!.cancel();
if (_shipped != null) _shipped.close(); if (_shipped != null) _shipped.close();
_fcsShipments = []; _fcsShipments = [];
} }

View File

@@ -6,7 +6,7 @@ import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
class InvoiceDiscountList extends StatelessWidget { class InvoiceDiscountList extends StatelessWidget {
final List<Discount>? discounts; final List<Discount?>? discounts;
const InvoiceDiscountList({ const InvoiceDiscountList({
Key? key, Key? key,
@@ -68,14 +68,14 @@ class InvoiceDiscountList extends StatelessWidget {
onSelectChanged: (value) => Navigator.pop(context, c), onSelectChanged: (value) => Navigator.pop(context, c),
cells: [ cells: [
MyDataCell(new Text( MyDataCell(new Text(
c.code!, c?.code ?? '',
style: textStyle, style: textStyle,
)), )),
MyDataCell( MyDataCell(
Row( Row(
mainAxisAlignment: MainAxisAlignment.end, mainAxisAlignment: MainAxisAlignment.end,
children: [ children: [
Text(c.amount.toStringAsFixed(2), style: textStyle), Text(c?.amount.toStringAsFixed(2) ?? "", style: textStyle),
], ],
), ),
), ),

View File

@@ -113,7 +113,7 @@ class _InvoiceEditorState extends State<InvoiceEditor> {
}); });
} }
List<Discount> discounts = []; List<Discount?> discounts = [];
_loadDiscount() async { _loadDiscount() async {
DiscountModel discountModel = DiscountModel discountModel =
Provider.of<DiscountModel>(context, listen: false); Provider.of<DiscountModel>(context, listen: false);

View File

@@ -58,7 +58,9 @@ class InvoiceListRow extends StatelessWidget {
fontSize: 13.0, color: primaryColor), fontSize: 13.0, color: primaryColor),
), ),
new Text( new Text(
dateFormatter.format(invoice!.invoiceDate!), invoice!.invoiceDate != null
? dateFormatter.format(invoice!.invoiceDate!)
: '',
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.grey), fontSize: 15.0, color: Colors.grey),
) )

View File

@@ -86,7 +86,7 @@ class _InvoiceShipmentListRowState extends State<InvoiceShipmentListRow> {
), ),
Padding( Padding(
padding: const EdgeInsets.all(0), padding: const EdgeInsets.all(0),
child: getStatus(_fcsShipment.status!), child: getStatus(_fcsShipment.status??"")
), ),
], ],
), ),

View File

@@ -16,9 +16,9 @@ import 'package:path/path.dart' as Path;
class PackageModel extends BaseModel { class PackageModel extends BaseModel {
final log = Logger('PackageModel'); final log = Logger('PackageModel');
PaginatorListener? packages; late PaginatorListener<Package> packages;
PaginatorListener? customerPackages; late PaginatorListener<Package> customerPackages;
PaginatorListener? activePackages; late PaginatorListener<Package> activePackages;
bool isLoading = false; bool isLoading = false;
int _menuSelectedIndex = 1; int _menuSelectedIndex = 1;
@@ -63,15 +63,16 @@ class PackageModel extends BaseModel {
@override @override
logout() async { logout() async {
if (customerPackages != null) customerPackages!.close(); if (customerPackages != null) customerPackages.close();
if (packages != null) packages!.close(); if (packages != null) packages.close();
if (activePackages != null) activePackages!.close(); if (activePackages != null) activePackages.close();
} }
Future<void> _loadPackages(bool isDelivered) async { Future<void> _loadPackages(bool isDelivered) async {
if (user == null) return; if (user == null) return;
if (!((user!.hasPackages() || user!.hasReceiving() || user!.hasProcessing()))) if (!((user!.hasPackages() ||
return; user!.hasReceiving() ||
user!.hasProcessing()))) return;
String path = "/$packages_collection"; String path = "/$packages_collection";
try { try {
@@ -83,7 +84,7 @@ class PackageModel extends BaseModel {
.where("is_delivered", isEqualTo: isDelivered); .where("is_delivered", isEqualTo: isDelivered);
pageQuery = pageQuery.orderBy("update_time", descending: true); pageQuery = pageQuery.orderBy("update_time", descending: true);
packages!.refresh(listeningQuery: listenerQuery, pageQuery: pageQuery); packages.refresh(listeningQuery: listenerQuery, pageQuery: pageQuery);
} catch (e) { } catch (e) {
log.warning("Error!! $e"); log.warning("Error!! $e");
} }
@@ -104,7 +105,7 @@ class PackageModel extends BaseModel {
.where("user_id", isEqualTo: user!.id) .where("user_id", isEqualTo: user!.id)
.orderBy("update_time", descending: true); .orderBy("update_time", descending: true);
customerPackages!.refresh( customerPackages.refresh(
listeningQuery: listenerQuery, pageQuery: pageQuery); listeningQuery: listenerQuery, pageQuery: pageQuery);
} catch (e) { } catch (e) {
log.warning("Error!! $e"); log.warning("Error!! $e");
@@ -113,8 +114,9 @@ class PackageModel extends BaseModel {
Future<void> _loadActivePackages() async { Future<void> _loadActivePackages() async {
if (user == null) return; if (user == null) return;
if (!((user!.hasPackages() || user!.hasReceiving() || user!.hasProcessing()))) if (!((user!.hasPackages() ||
return; user!.hasReceiving() ||
user!.hasProcessing()))) return;
String path = "/$packages_collection"; String path = "/$packages_collection";
try { try {
@@ -126,7 +128,7 @@ class PackageModel extends BaseModel {
.where("is_delivered", isEqualTo: false); .where("is_delivered", isEqualTo: false);
pageQuery = pageQuery.orderBy("update_time", descending: true); pageQuery = pageQuery.orderBy("update_time", descending: true);
activePackages!.refresh( activePackages.refresh(
listeningQuery: listenerQuery, pageQuery: pageQuery); listeningQuery: listenerQuery, pageQuery: pageQuery);
} catch (e) { } catch (e) {
log.warning("Error!! $e"); log.warning("Error!! $e");
@@ -161,7 +163,8 @@ class PackageModel extends BaseModel {
.get(const GetOptions(source: Source.server)); .get(const GetOptions(source: Source.server));
if (snaps.docs.length == 1) { if (snaps.docs.length == 1) {
var snap = snaps.docs[0]; var snap = snaps.docs[0];
var package = Package.fromMap(snap.data as Map<String, dynamic>, snap.id); var package =
Package.fromMap(snap.data as Map<String, dynamic>, snap.id);
return package; return package;
} }
} catch (e) { } catch (e) {
@@ -184,7 +187,8 @@ class PackageModel extends BaseModel {
if (qsnap.docs.length > 0) { if (qsnap.docs.length > 0) {
var snap = qsnap.docs[0]; var snap = qsnap.docs[0];
if (snap.exists) { if (snap.exists) {
var package = Package.fromMap(snap.data as Map<String, dynamic>, snap.id); var package =
Package.fromMap(snap.data as Map<String, dynamic>, snap.id);
return package; return package;
} }
} }
@@ -198,7 +202,8 @@ class PackageModel extends BaseModel {
if (qsnap.docs.length > 0) { if (qsnap.docs.length > 0) {
var snap = qsnap.docs[0]; var snap = qsnap.docs[0];
if (snap.exists) { if (snap.exists) {
var package = Package.fromMap(snap.data as Map<String, dynamic>, snap.id); var package =
Package.fromMap(snap.data as Map<String, dynamic>, snap.id);
return package; return package;
} }
} }
@@ -218,8 +223,8 @@ class PackageModel extends BaseModel {
.where("is_deleted", isEqualTo: false) .where("is_deleted", isEqualTo: false)
.get(const GetOptions(source: Source.server)); .get(const GetOptions(source: Source.server));
packages = snaps.docs.map((documentSnapshot) { packages = snaps.docs.map((documentSnapshot) {
var p = var p = Package.fromMap(
Package.fromMap(documentSnapshot.data as Map<String, dynamic>, documentSnapshot.id); documentSnapshot.data as Map<String, dynamic>, documentSnapshot.id);
return p; return p;
}).toList(); }).toList();
} catch (e) { } catch (e) {

View File

@@ -38,8 +38,8 @@ class _PackageEditorPageState extends State<PackageEditorPage> {
super.initState(); super.initState();
_package = widget.package!; _package = widget.package!;
selectedMarket = _package!.market; selectedMarket = _package!.market;
_descCtl.text = _package!.desc; _descCtl.text = _package!.desc!;
_remarkCtl.text = _package!.remark; _remarkCtl.text = _package!.remark!;
multiImgController.setImageUrls = _package!.photoUrls; multiImgController.setImageUrls = _package!.photoUrls;
} }
@@ -157,7 +157,7 @@ class _PackageEditorPageState extends State<PackageEditorPage> {
String? selectedMarket; String? selectedMarket;
Widget marketDropdown() { Widget marketDropdown() {
List<Market> _markets = Provider.of<MarketModel>(context).markets; List<Market> _markets = Provider.of<MarketModel>(context).markets;
List<String> markets = _markets.map((e) => e.name).toList(); List<String?> markets = _markets.map((e) => e.name).toList();
markets.insert(0, MANAGE_MARKET); markets.insert(0, MANAGE_MARKET);
if (!markets.contains(selectedMarket)) { if (!markets.contains(selectedMarket)) {
markets.insert(0, selectedMarket!); markets.insert(0, selectedMarket!);
@@ -194,10 +194,10 @@ class _PackageEditorPageState extends State<PackageEditorPage> {
}); });
}, },
isExpanded: true, isExpanded: true,
items: markets.map<DropdownMenuItem<String>>((String value) { items: markets.map<DropdownMenuItem<String>>((String? value) {
return DropdownMenuItem<String>( return DropdownMenuItem<String>(
value: value, value: value,
child: Text(value, child: Text(value ?? "",
overflow: TextOverflow.ellipsis, overflow: TextOverflow.ellipsis,
style: TextStyle( style: TextStyle(
color: value == MANAGE_MARKET color: value == MANAGE_MARKET

View File

@@ -51,10 +51,10 @@ class _PackageInfoState extends State<PackageInfo> {
initPackage(Package pkg) async { initPackage(Package pkg) async {
PackageModel packageModel = PackageModel packageModel =
Provider.of<PackageModel>(context, listen: false); Provider.of<PackageModel>(context, listen: false);
Package package = await packageModel.getPackageByTrackingID(pkg.trackingID); Package? package = await packageModel.getPackageByTrackingID(pkg.trackingID!);
setState(() { setState(() {
_package = package; _package = package;
multiImgController.setImageUrls = package.photoUrls; multiImgController.setImageUrls = package!.photoUrls;
}); });
} }
@@ -65,7 +65,7 @@ class _PackageInfoState extends State<PackageInfo> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
String id = Provider.of<MainModel>(context).user.id; String? id = Provider.of<MainModel>(context).user!.id;
bool owner = _package?.userID == id; bool owner = _package?.userID == id;
bool canChangeDeliveryAddress = bool canChangeDeliveryAddress =
_package?.status == package_received_status || _package?.status == package_received_status ||
@@ -199,7 +199,7 @@ class _PackageInfoState extends State<PackageInfo> {
try { try {
await packageModel.changeDeliveryAddress(_package!, deliveryAddress); await packageModel.changeDeliveryAddress(_package!, deliveryAddress);
var da = var da =
await deliveryAddressModel.getDeliveryAddress(deliveryAddress.id); await deliveryAddressModel.getDeliveryAddress(deliveryAddress.id!);
setState(() { setState(() {
_package!.deliveryAddress = da; _package!.deliveryAddress = da;
}); });

View File

@@ -97,7 +97,7 @@ class _PackageListState extends State<PackageList> {
_searchCallback(Package package) async { _searchCallback(Package package) async {
var packageModel = Provider.of<PackageModel>(context, listen: false); var packageModel = Provider.of<PackageModel>(context, listen: false);
Package _package = await packageModel.getPackage(package.id); Package? _package = await packageModel.getPackage(package.id!);
if (_package == null) return; if (_package == null) return;
Navigator.push( Navigator.push(
context, context,

View File

@@ -60,7 +60,9 @@ class PackageListRow extends StatelessWidget {
Padding( Padding(
padding: const EdgeInsets.only(left: 8.0), padding: const EdgeInsets.only(left: 8.0),
child: new Text( child: new Text(
package!.id == null ? '' : package!.trackingID, package!.id == null
? ''
: package!.trackingID!,
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.black), fontSize: 15.0, color: Colors.black),
), ),
@@ -68,7 +70,9 @@ class PackageListRow extends StatelessWidget {
Padding( Padding(
padding: const EdgeInsets.only(left: 8.0), padding: const EdgeInsets.only(left: 8.0),
child: new Text( child: new Text(
package!.market == null ? '' : package!.market, package!.market == null
? ''
: package!.market!,
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.black), fontSize: 15.0, color: Colors.black),
), ),
@@ -84,12 +88,14 @@ class PackageListRow extends StatelessWidget {
children: <Widget>[ children: <Widget>[
Padding( Padding(
padding: const EdgeInsets.all(3.0), padding: const EdgeInsets.all(3.0),
child: getStatus(package!.status), child: getStatus(package!.status??""),
), ),
Padding( Padding(
padding: const EdgeInsets.all(0), padding: const EdgeInsets.all(0),
child: new Text( child: new Text(
dateFormat.format(package!.currentStatusDate), package!.currentStatusDate != null
? dateFormat.format(package!.currentStatusDate!)
: '',
style: style:
new TextStyle(fontSize: 15.0, color: Colors.grey), new TextStyle(fontSize: 15.0, color: Colors.grey),
), ),

View File

@@ -144,8 +144,8 @@ class _PackageNewState extends State<PackageNew> {
child: Column( child: Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
Text(packages[index].market), Text(packages[index].market!),
Text(packages[index].trackingID), Text(packages[index].trackingID!),
// DisplayText( // DisplayText(
// labelText: "Tracking ID", // labelText: "Tracking ID",
// text: packages[index].trackingID, // text: packages[index].trackingID,

View File

@@ -87,7 +87,7 @@ class _TrackingIDPageState extends State<TrackingIDPage> {
String? selectedMarket; String? selectedMarket;
Widget dropDown() { Widget dropDown() {
List<Market> _markets = Provider.of<MarketModel>(context).markets; List<Market> _markets = Provider.of<MarketModel>(context).markets;
List<String> markets = _markets.map((e) => e.name).toList(); List<String?> markets = _markets.map((e) => e.name).toList();
markets.insert(0, MANAGE_MARKET); markets.insert(0, MANAGE_MARKET);
markets.insert(0, SELECT_MARKET); markets.insert(0, SELECT_MARKET);
@@ -126,10 +126,10 @@ class _TrackingIDPageState extends State<TrackingIDPage> {
}); });
}, },
isExpanded: true, isExpanded: true,
items: markets.map<DropdownMenuItem<String>>((String value) { items: markets.map<DropdownMenuItem<String>>((String? value) {
return DropdownMenuItem<String>( return DropdownMenuItem<String>(
value: value, value: value,
child: Text(value, child: Text(value ?? "",
overflow: TextOverflow.ellipsis, overflow: TextOverflow.ellipsis,
style: TextStyle( style: TextStyle(
color: value == MANAGE_MARKET color: value == MANAGE_MARKET

View File

@@ -12,21 +12,22 @@ class PaymentMethodModel extends BaseModel {
List<PaymentMethod> paymentMethods = []; List<PaymentMethod> paymentMethods = [];
PaymentMethod getPaymentMethod(String id) { PaymentMethod getPaymentMethod(String id) {
return paymentMethods.firstWhere((e) => e.id == id, orElse: () => null); return paymentMethods.firstWhere((e) => e.id == id);
} }
StreamSubscription<QuerySnapshot> listener; StreamSubscription<QuerySnapshot>? listener;
PaymentMethodModel() { PaymentMethodModel() {
if (listener != null) listener.cancel(); if (listener != null) listener!.cancel();
try { try {
listener = Firestore.instance listener = FirebaseFirestore.instance
.collection("/payment_methods") .collection("/payment_methods")
.snapshots() .snapshots()
.listen((snaps) { .listen((snaps) {
paymentMethods.clear(); paymentMethods.clear();
snaps.documents.forEach((d) { snaps.docs.forEach((d) {
paymentMethods.add(PaymentMethod.fromMap(d.data, d.documentID)); paymentMethods
.add(PaymentMethod.fromMap(d.data as Map<String, dynamic>, d.id));
}); });
notifyListeners(); notifyListeners();
}); });

View File

@@ -38,12 +38,12 @@ class _PaymentMethodEditorState extends State<PaymentMethodEditor> {
if (widget.paymentMethod != null) { if (widget.paymentMethod != null) {
_paymentMethod = widget.paymentMethod; _paymentMethod = widget.paymentMethod;
_nameController.text = _paymentMethod!.name; _nameController.text = _paymentMethod!.name!;
_accountNameController.text = _paymentMethod!.accountName; _accountNameController.text = _paymentMethod!.accountName!;
_accountNumberController.text = _paymentMethod!.account; _accountNumberController.text = _paymentMethod!.account!;
_mailController.text = _paymentMethod!.email; _mailController.text = _paymentMethod!.email!;
_phoneController.text = _paymentMethod!.phone; _phoneController.text = _paymentMethod!.phone!;
_linkController.text = _paymentMethod!.link; _linkController.text = _paymentMethod!.link!;
} else { } else {
_paymentMethod = new PaymentMethod(); _paymentMethod = new PaymentMethod();
_nameController.text = ''; _nameController.text = '';
@@ -208,7 +208,7 @@ class _PaymentMethodEditorState extends State<PaymentMethodEditor> {
}); });
try { try {
await Provider.of<PaymentMethodModel>(context, listen: false) await Provider.of<PaymentMethodModel>(context, listen: false)
.deletePaymentMethod(_paymentMethod!.id); .deletePaymentMethod(_paymentMethod!.id!);
Navigator.pop(context); Navigator.pop(context);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());

View File

@@ -84,13 +84,13 @@ class _PaymentMethodPageState extends State<PaymentMethodPage> {
} }
_item(PaymentMethod method, bool isEditable) { _item(PaymentMethod method, bool isEditable) {
final accountName = _itemRow(method.accountName, "pm.account.name", final accountName = _itemRow(method.accountName!, "pm.account.name",
iconData: MaterialCommunityIcons.bank); iconData: MaterialCommunityIcons.bank);
final accountNumber = _itemRow(method.account, "pm.account.no", final accountNumber = _itemRow(method.account!, "pm.account.no",
iconData: MaterialCommunityIcons.checkbook); iconData: MaterialCommunityIcons.checkbook);
final phone = _itemRow(method.phone, "pm.phone", iconData: Icons.phone); final phone = _itemRow(method.phone!, "pm.phone", iconData: Icons.phone);
final email = _itemRow(method.email, "pm.email", iconData: Icons.mail); final email = _itemRow(method.email!, "pm.email", iconData: Icons.mail);
final link = _itemRow(method.link, "pm.link", iconData: Icons.link); final link = _itemRow(method.link!, "pm.link", iconData: Icons.link);
return InkWell( return InkWell(
onTap: isEditable onTap: isEditable
@@ -110,7 +110,7 @@ class _PaymentMethodPageState extends State<PaymentMethodPage> {
Padding( Padding(
padding: const EdgeInsets.all(8.0), padding: const EdgeInsets.all(8.0),
child: Text( child: Text(
method.name, method.name!,
style: TextStyle( style: TextStyle(
color: primaryColor, color: primaryColor,
fontWeight: FontWeight.bold, fontWeight: FontWeight.bold,

View File

@@ -11,7 +11,7 @@ class ProcessingModel extends BaseModel {
List<Processing> processings = []; List<Processing> processings = [];
final log = Logger('ProcessingModel'); final log = Logger('ProcessingModel');
StreamSubscription<QuerySnapshot> listener; StreamSubscription<QuerySnapshot>? listener;
void initUser(user) { void initUser(user) {
super.initUser(user); super.initUser(user);
@@ -19,12 +19,12 @@ class ProcessingModel extends BaseModel {
@override @override
logout() async { logout() async {
if (listener != null) await listener.cancel(); if (listener != null) await listener!.cancel();
processings = []; processings = [];
} }
Future<void> createProcessing(Processing processing) {} Future<void> createProcessing(Processing processing) async {}
Future<void> updateProcessing(Processing processing) {} Future<void> updateProcessing(Processing processing) async {}
} }

View File

@@ -44,14 +44,14 @@ class _PackageEditorState extends State<PackageEditor> {
void initState() { void initState() {
super.initState(); super.initState();
_package = Package(); _package = Package();
_loadPackageData(widget.package!.id); _loadPackageData(widget.package!.id!);
} }
_loadPackageData(String id) async { _loadPackageData(String id) async {
if (id != null) { if (id != null) {
PackageModel packageModel = PackageModel packageModel =
Provider.of<PackageModel>(context, listen: false); Provider.of<PackageModel>(context, listen: false);
Package package = await packageModel.getPackage(id); Package? package = await packageModel.getPackage(id);
if (package != null) { if (package != null) {
if (package.status != package_received_status) { if (package.status != package_received_status) {
showMsgDialog(context, "Error", showMsgDialog(context, "Error",
@@ -65,8 +65,8 @@ class _PackageEditorState extends State<PackageEditor> {
} }
setState(() { setState(() {
selectedMarket = _package!.market ?? ""; selectedMarket = _package!.market ?? "";
_descCtl.text = _package!.desc; _descCtl.text = _package!.desc!;
_remarkCtl.text = _package!.remark; _remarkCtl.text = _package!.remark!;
multiImgController.setImageUrls = _package!.photoUrls; multiImgController.setImageUrls = _package!.photoUrls;
}); });
} }
@@ -84,7 +84,7 @@ class _PackageEditorState extends State<PackageEditor> {
IconButton( IconButton(
icon: Icon(Icons.search, color: primaryColor), icon: Icon(Icons.search, color: primaryColor),
onPressed: () => searchPackage(context, callbackPackageSelect: (u) { onPressed: () => searchPackage(context, callbackPackageSelect: (u) {
_loadPackageData(u.id); _loadPackageData(u.id!);
Navigator.pop(context); Navigator.pop(context);
})), })),
], ],
@@ -177,7 +177,7 @@ class _PackageEditorState extends State<PackageEditor> {
String? selectedMarket; String? selectedMarket;
Widget marketDropdown() { Widget marketDropdown() {
List<Market> _markets = Provider.of<MarketModel>(context).markets; List<Market> _markets = Provider.of<MarketModel>(context).markets;
List<String> markets = _markets.map((e) => e.name).toList(); List<String?> markets = _markets.map((e) => e.name).toList();
markets.insert(0, MANAGE_MARKET); markets.insert(0, MANAGE_MARKET);
if (!markets.contains(selectedMarket)) { if (!markets.contains(selectedMarket)) {
markets.insert(0, selectedMarket!); markets.insert(0, selectedMarket!);
@@ -221,7 +221,7 @@ class _PackageEditorState extends State<PackageEditor> {
}); });
}, },
isExpanded: true, isExpanded: true,
items: markets.map<DropdownMenuItem<String>>((String value) { items: markets.map<DropdownMenuItem<String>>((String? value) {
return DropdownMenuItem<String>( return DropdownMenuItem<String>(
value: value, value: value,
child: Text(value ?? "", child: Text(value ?? "",

View File

@@ -42,8 +42,8 @@ class _ProcessingEditEditorState extends State<ProcessingEditEditor> {
super.initState(); super.initState();
_package = widget.package; _package = widget.package;
selectedMarket = _package!.market ?? ""; selectedMarket = _package!.market ?? "";
_descCtl.text = _package!.desc; _descCtl.text = _package!.desc!;
_remarkCtl.text = _package!.remark; _remarkCtl.text = _package!.remark!;
multiImgController.setImageUrls = _package!.photoUrls; multiImgController.setImageUrls = _package!.photoUrls;
_user = User( _user = User(
fcsID: _package!.fcsID ?? "", fcsID: _package!.fcsID ?? "",
@@ -162,7 +162,7 @@ class _ProcessingEditEditorState extends State<ProcessingEditEditor> {
String? selectedMarket; String? selectedMarket;
Widget marketDropdown() { Widget marketDropdown() {
List<Market> _markets = Provider.of<MarketModel>(context).markets; List<Market> _markets = Provider.of<MarketModel>(context).markets;
List<String> markets = _markets.map((e) => e.name).toList(); List<String?> markets = _markets.map((e) => e.name).toList();
markets.insert(0, MANAGE_MARKET); markets.insert(0, MANAGE_MARKET);
if (!markets.contains(selectedMarket)) { if (!markets.contains(selectedMarket)) {
markets.insert(0, selectedMarket!); markets.insert(0, selectedMarket!);
@@ -208,7 +208,7 @@ class _ProcessingEditEditorState extends State<ProcessingEditEditor> {
}); });
}, },
isExpanded: true, isExpanded: true,
items: markets.map<DropdownMenuItem<String>>((String value) { items: markets.map<DropdownMenuItem<String>>((String? value) {
return DropdownMenuItem<String>( return DropdownMenuItem<String>(
value: value, value: value,
child: Text(value ?? "", child: Text(value ?? "",

View File

@@ -175,7 +175,7 @@ class _ProcessingInfoState extends State<ProcessingInfo> {
_package!.photoUrls.length == 0 ? Container() : img, _package!.photoUrls.length == 0 ? Container() : img,
StatusTree( StatusTree(
shipmentHistory: _package!.shipmentHistory, shipmentHistory: _package!.shipmentHistory,
currentStatus: _package!.status), currentStatus: _package!.status??""),
SizedBox( SizedBox(
height: 20, height: 20,
) )
@@ -222,8 +222,8 @@ class _ProcessingInfoState extends State<ProcessingInfo> {
} else { } else {
PackageModel packageModel = PackageModel packageModel =
Provider.of<PackageModel>(context, listen: false); Provider.of<PackageModel>(context, listen: false);
Package p = await packageModel.getPackage(_package!.id); Package? p = await packageModel.getPackage(_package!.id!);
initPackage(p); initPackage(p!);
} }
} }
} }

View File

@@ -85,7 +85,7 @@ class _ProcessingListState extends State<ProcessingList> {
_searchCallback(Package package) async { _searchCallback(Package package) async {
var packageModel = Provider.of<PackageModel>(context, listen: false); var packageModel = Provider.of<PackageModel>(context, listen: false);
Package _package = await packageModel.getPackage(package.id); Package? _package = await packageModel.getPackage(package.id!);
if (_package == null) return; if (_package == null) return;
Navigator.push( Navigator.push(
context, context,

View File

@@ -57,7 +57,7 @@ class ProcessingListRow extends StatelessWidget {
Padding( Padding(
padding: const EdgeInsets.only(left: 8.0), padding: const EdgeInsets.only(left: 8.0),
child: new Text( child: new Text(
package!.id == null ? '' : package!.trackingID, package!.id == null ? '' : package!.trackingID!,
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.black), fontSize: 15.0, color: Colors.black),
), ),
@@ -65,7 +65,7 @@ class ProcessingListRow extends StatelessWidget {
Padding( Padding(
padding: const EdgeInsets.only(left: 8.0), padding: const EdgeInsets.only(left: 8.0),
child: new Text( child: new Text(
package!.market == null ? '' : package!.market, package!.market == null ? '' : package!.market!,
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.black), fontSize: 15.0, color: Colors.black),
), ),
@@ -81,12 +81,14 @@ class ProcessingListRow extends StatelessWidget {
children: <Widget>[ children: <Widget>[
Padding( Padding(
padding: const EdgeInsets.all(3.0), padding: const EdgeInsets.all(3.0),
child: getStatus(package!.status), child: getStatus(package!.status??""),
), ),
Padding( Padding(
padding: const EdgeInsets.all(0), padding: const EdgeInsets.all(0),
child: new Text( child: new Text(
dateFormat.format(package!.currentStatusDate), package!.currentStatusDate != null
? dateFormat.format(package!.currentStatusDate!)
: '',
style: new TextStyle(fontSize: 15.0, color: Colors.grey), style: new TextStyle(fontSize: 15.0, color: Colors.grey),
), ),
), ),

View File

@@ -18,12 +18,12 @@ import 'package:fcs/pages/widgets/multi_img_file.dart';
import 'package:fcs/pages/widgets/progress.dart'; import 'package:fcs/pages/widgets/progress.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_icons/flutter_icons.dart'; import 'package:flutter_icons_null_safety/flutter_icons_null_safety.dart';
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
class ProcessingEditor extends StatefulWidget { class ProcessingEditor extends StatefulWidget {
final Package package; final Package? package;
ProcessingEditor({this.package}); ProcessingEditor({this.package});
@override @override
@@ -34,22 +34,22 @@ class _ProcessingEditorState extends State<ProcessingEditor> {
TextEditingController _remarkCtl = new TextEditingController(); TextEditingController _remarkCtl = new TextEditingController();
TextEditingController _descCtl = new TextEditingController(); TextEditingController _descCtl = new TextEditingController();
Package _package; Package? _package;
User _user; User? _user;
bool _isLoading = false; bool _isLoading = false;
@override @override
void initState() { void initState() {
super.initState(); super.initState();
_package = widget.package; _package = widget.package;
selectedMarket = _package.market ?? ""; selectedMarket = _package!.market ?? "";
_descCtl.text = _package.desc; _descCtl.text = _package!.desc!;
_remarkCtl.text = _package.remark; _remarkCtl.text = _package!.remark!;
multiImgController.setImageUrls = _package.photoUrls; multiImgController.setImageUrls = _package!.photoUrls;
_user = User( _user = User(
fcsID: _package.fcsID ?? "", fcsID: _package!.fcsID ?? "",
name: _package.userName ?? "", name: _package!.userName ?? "",
phoneNumber: _package.phoneNumber ?? ""); phoneNumber: _package!.phoneNumber ?? "");
} }
final DateFormat dateFormat = DateFormat("d MMM yyyy"); final DateFormat dateFormat = DateFormat("d MMM yyyy");
@@ -63,7 +63,7 @@ class _ProcessingEditorState extends State<ProcessingEditor> {
children: <Widget>[ children: <Widget>[
Expanded( Expanded(
child: DisplayText( child: DisplayText(
text: _user.fcsID, text: _user!.fcsID??"",
labelTextKey: "processing.fcs.id", labelTextKey: "processing.fcs.id",
icon: FcsIDIcon(), icon: FcsIDIcon(),
)), )),
@@ -77,18 +77,18 @@ class _ProcessingEditorState extends State<ProcessingEditor> {
], ],
); );
final namebox = DisplayText( final namebox = DisplayText(
text: _user.name, text: _user!.name??"",
labelTextKey: "processing.name", labelTextKey: "processing.name",
iconData: Icons.person, iconData: Icons.person,
); );
final phoneNumberBox = DisplayText( final phoneNumberBox = DisplayText(
text: _user.phoneNumber, text: _user!.phoneNumber??"",
labelTextKey: "processing.phone", labelTextKey: "processing.phone",
iconData: Icons.phone, iconData: Icons.phone,
); );
final trackingIdBox = DisplayText( final trackingIdBox = DisplayText(
text: _package.trackingID, text: _package!.trackingID??"",
labelTextKey: "processing.tracking.id", labelTextKey: "processing.tracking.id",
iconData: MaterialCommunityIcons.barcode_scan, iconData: MaterialCommunityIcons.barcode_scan,
); );
@@ -152,13 +152,13 @@ class _ProcessingEditorState extends State<ProcessingEditor> {
); );
} }
String selectedMarket; String? selectedMarket;
Widget marketDropdown() { Widget marketDropdown() {
List<Market> _markets = Provider.of<MarketModel>(context).markets; List<Market> _markets = Provider.of<MarketModel>(context).markets;
List<String> markets = _markets.map((e) => e.name).toList(); List<String?> markets = _markets.map((e) => e.name).toList();
markets.insert(0, MANAGE_MARKET); markets.insert(0, MANAGE_MARKET);
if (!markets.contains(selectedMarket)) { if (!markets.contains(selectedMarket)) {
markets.insert(0, selectedMarket); markets.insert(0, selectedMarket!);
} }
return Padding( return Padding(
@@ -190,7 +190,7 @@ class _ProcessingEditorState extends State<ProcessingEditor> {
height: 1, height: 1,
color: Colors.grey, color: Colors.grey,
), ),
onChanged: (String newValue) { onChanged: (String? newValue) {
setState(() { setState(() {
if (newValue == MANAGE_MARKET) { if (newValue == MANAGE_MARKET) {
selectedMarket = null; selectedMarket = null;
@@ -201,7 +201,7 @@ class _ProcessingEditorState extends State<ProcessingEditor> {
}); });
}, },
isExpanded: true, isExpanded: true,
items: markets.map<DropdownMenuItem<String>>((String value) { items: markets.map<DropdownMenuItem<String>>((String? value) {
return DropdownMenuItem<String>( return DropdownMenuItem<String>(
value: value, value: value,
child: Text(value ?? "", child: Text(value ?? "",
@@ -229,7 +229,7 @@ class _ProcessingEditorState extends State<ProcessingEditor> {
} }
_completeProcessing() async { _completeProcessing() async {
if (_user.fcsID == null || _user.fcsID == "") { if (_user!.fcsID == null || _user!.fcsID == "") {
showMsgDialog(context, "Error", "Expected FCS-ID"); showMsgDialog(context, "Error", "Expected FCS-ID");
return; return;
} }
@@ -239,11 +239,11 @@ class _ProcessingEditorState extends State<ProcessingEditor> {
PackageModel packageModel = PackageModel packageModel =
Provider.of<PackageModel>(context, listen: false); Provider.of<PackageModel>(context, listen: false);
try { try {
_package.fcsID = _user.fcsID; _package!.fcsID = _user!.fcsID;
_package.desc = _descCtl.text; _package!.desc = _descCtl.text;
_package.remark = _remarkCtl.text; _package!.remark = _remarkCtl.text;
_package.market = selectedMarket; _package!.market = selectedMarket;
await packageModel.updateProcessing(_package, await packageModel.updateProcessing(_package!,
multiImgController.getAddedFile, multiImgController.getDeletedUrl); multiImgController.getAddedFile, multiImgController.getDeletedUrl);
Navigator.pop(context); Navigator.pop(context);
} catch (e) { } catch (e) {

View File

@@ -12,7 +12,8 @@ import 'package:fcs/pages/widgets/progress.dart';
import 'package:fcs/pages/widgets/status_tree.dart'; import 'package:fcs/pages/widgets/status_tree.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_icons/flutter_icons.dart'; import 'package:flutter_icons_null_safety/flutter_icons_null_safety.dart';
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
@@ -21,7 +22,7 @@ import 'processing_editor.dart';
final DateFormat dateFormat = DateFormat("d MMM yyyy"); final DateFormat dateFormat = DateFormat("d MMM yyyy");
class ProcessingInfo extends StatefulWidget { class ProcessingInfo extends StatefulWidget {
final Package package; final Package? package;
ProcessingInfo({this.package}); ProcessingInfo({this.package});
@override @override
@@ -30,14 +31,14 @@ class ProcessingInfo extends StatefulWidget {
class _ProcessingInfoState extends State<ProcessingInfo> { class _ProcessingInfoState extends State<ProcessingInfo> {
var dateFormatter = new DateFormat('dd MMM yyyy'); var dateFormatter = new DateFormat('dd MMM yyyy');
Package _package; Package? _package;
bool _isLoading = false; bool _isLoading = false;
MultiImgController multiImgController = MultiImgController(); MultiImgController multiImgController = MultiImgController();
@override @override
void initState() { void initState() {
super.initState(); super.initState();
initPackage(widget.package); initPackage(widget.package!);
} }
initPackage(Package package) { initPackage(Package package) {
@@ -55,37 +56,37 @@ class _ProcessingInfoState extends State<ProcessingInfo> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final trackingIdBox = DisplayText( final trackingIdBox = DisplayText(
text: _package.trackingID, text: _package!.trackingID ?? "",
labelTextKey: "processing.tracking.id", labelTextKey: "processing.tracking.id",
iconData: MaterialCommunityIcons.barcode_scan, iconData: MaterialCommunityIcons.barcode_scan,
); );
var fcsIDBox = DisplayText( var fcsIDBox = DisplayText(
text: _package.fcsID, text: _package!.fcsID??"",
labelTextKey: "processing.fcs.id", labelTextKey: "processing.fcs.id",
icon: FcsIDIcon(), icon: FcsIDIcon(),
); );
final phoneNumberBox = DisplayText( final phoneNumberBox = DisplayText(
text: _package.phoneNumber, text: _package!.phoneNumber??"",
labelTextKey: "processing.phone", labelTextKey: "processing.phone",
iconData: Icons.phone, iconData: Icons.phone,
); );
final customerNameBox = DisplayText( final customerNameBox = DisplayText(
text: _package.userName, text: _package!.userName??"",
labelTextKey: "processing.name", labelTextKey: "processing.name",
iconData: Icons.perm_identity, iconData: Icons.perm_identity,
); );
final marketBox = DisplayText( final marketBox = DisplayText(
text: _package.market ?? "-", text: _package!.market ?? "-",
labelTextKey: "processing.market", labelTextKey: "processing.market",
iconData: Icons.store, iconData: Icons.store,
); );
final descBox = DisplayText( final descBox = DisplayText(
text: _package.desc ?? "-", text: _package!.desc ?? "-",
labelTextKey: "processing.desc", labelTextKey: "processing.desc",
iconData: MaterialCommunityIcons.message_text_outline, iconData: MaterialCommunityIcons.message_text_outline,
); );
final remarkBox = DisplayText( final remarkBox = DisplayText(
text: _package.remark ?? "-", text: _package!.remark ?? "-",
labelTextKey: "processing.remark", labelTextKey: "processing.remark",
iconData: Entypo.new_message, iconData: Entypo.new_message,
); );
@@ -137,10 +138,10 @@ class _ProcessingInfoState extends State<ProcessingInfo> {
marketBox, marketBox,
descBox, descBox,
remarkBox, remarkBox,
_package.photoUrls.length == 0 ? Container() : img, _package!.photoUrls.length == 0 ? Container() : img,
StatusTree( StatusTree(
shipmentHistory: _package.shipmentHistory, shipmentHistory: _package!.shipmentHistory,
currentStatus: _package.status), currentStatus: _package!.status??""),
SizedBox( SizedBox(
height: 20, height: 20,
) )
@@ -164,7 +165,7 @@ class _ProcessingInfoState extends State<ProcessingInfo> {
PackageModel packageModel = PackageModel packageModel =
Provider.of<PackageModel>(context, listen: false); Provider.of<PackageModel>(context, listen: false);
try { try {
await packageModel.deleteProcessing(_package); await packageModel.deleteProcessing(_package!);
Navigator.pop<bool>(context, true); Navigator.pop<bool>(context, true);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());
@@ -176,19 +177,19 @@ class _ProcessingInfoState extends State<ProcessingInfo> {
} }
_gotoEditor() async { _gotoEditor() async {
bool deleted = await Navigator.push<bool>( bool? deleted = await Navigator.push<bool>(
context, context,
CupertinoPageRoute( CupertinoPageRoute(
builder: (context) => ProcessingEditor( builder: (context) => ProcessingEditor(
package: widget.package, package: widget.package!,
))); )));
if (deleted ?? false) { if (deleted ?? false) {
Navigator.pop(context); Navigator.pop(context);
} else { } else {
PackageModel packageModel = PackageModel packageModel =
Provider.of<PackageModel>(context, listen: false); Provider.of<PackageModel>(context, listen: false);
Package p = await packageModel.getPackage(_package.id); Package? p = await packageModel.getPackage(_package!.id!);
initPackage(p); initPackage(p!);
} }
} }
} }

View File

@@ -77,7 +77,7 @@ class _ProcessingListState extends State<ProcessingList> {
_searchCallback(Package package) async { _searchCallback(Package package) async {
var packageModel = Provider.of<PackageModel>(context, listen: false); var packageModel = Provider.of<PackageModel>(context, listen: false);
Package _package = await packageModel.getPackage(package.id); Package? _package = await packageModel.getPackage(package.id!);
if (_package == null) return; if (_package == null) return;
Navigator.push( Navigator.push(
context, context,

View File

@@ -3,7 +3,7 @@ import 'package:fcs/helpers/theme.dart';
import 'package:fcs/pages/main/util.dart'; import 'package:fcs/pages/main/util.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_icons/flutter_icons.dart'; import 'package:font_awesome_flutter/font_awesome_flutter.dart';
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
import 'processing_info.dart'; import 'processing_info.dart';
@@ -11,12 +11,12 @@ import 'processing_info.dart';
typedef CallbackPackageSelect(Package package); typedef CallbackPackageSelect(Package package);
class ProcessingListRow extends StatelessWidget { class ProcessingListRow extends StatelessWidget {
final Package package; final Package? package;
final CallbackPackageSelect callbackPackageSelect; final CallbackPackageSelect? callbackPackageSelect;
final double dotSize = 15.0; final double dotSize = 15.0;
final DateFormat dateFormat = new DateFormat("dd MMM yyyy"); final DateFormat dateFormat = new DateFormat("dd MMM yyyy");
ProcessingListRow({Key key, this.package, this.callbackPackageSelect}) ProcessingListRow({Key? key, this.package, this.callbackPackageSelect})
: super(key: key); : super(key: key);
@override @override
@@ -24,7 +24,7 @@ class ProcessingListRow extends StatelessWidget {
return InkWell( return InkWell(
onTap: () { onTap: () {
if (callbackPackageSelect != null) { if (callbackPackageSelect != null) {
callbackPackageSelect(package); callbackPackageSelect!(package!);
return; return;
} }
Navigator.push( Navigator.push(
@@ -45,7 +45,7 @@ class ProcessingListRow extends StatelessWidget {
Container( Container(
padding: EdgeInsets.only(left: 5, right: 10), padding: EdgeInsets.only(left: 5, right: 10),
child: Icon( child: Icon(
FontAwesome.dropbox, FontAwesomeIcons.dropbox,
color: primaryColor, color: primaryColor,
size: 30, size: 30,
), ),
@@ -57,7 +57,7 @@ class ProcessingListRow extends StatelessWidget {
Padding( Padding(
padding: const EdgeInsets.only(left: 8.0), padding: const EdgeInsets.only(left: 8.0),
child: new Text( child: new Text(
package.id == null ? '' : package.trackingID, package!.id == null ? '' : package!.trackingID!,
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.black), fontSize: 15.0, color: Colors.black),
), ),
@@ -65,7 +65,7 @@ class ProcessingListRow extends StatelessWidget {
Padding( Padding(
padding: const EdgeInsets.only(left: 8.0), padding: const EdgeInsets.only(left: 8.0),
child: new Text( child: new Text(
package.market == null ? '' : package.market, package!.market == null ? '' : package!.market!,
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.black), fontSize: 15.0, color: Colors.black),
), ),
@@ -81,12 +81,14 @@ class ProcessingListRow extends StatelessWidget {
children: <Widget>[ children: <Widget>[
Padding( Padding(
padding: const EdgeInsets.all(3.0), padding: const EdgeInsets.all(3.0),
child: getStatus(package.status), child: getStatus(package!.status ?? ""),
), ),
Padding( Padding(
padding: const EdgeInsets.all(0), padding: const EdgeInsets.all(0),
child: new Text( child: new Text(
dateFormat.format(package.currentStatusDate), package!.currentStatusDate != null
? dateFormat.format(package!.currentStatusDate!)
: '',
style: new TextStyle(fontSize: 15.0, color: Colors.grey), style: new TextStyle(fontSize: 15.0, color: Colors.grey),
), ),
), ),

View File

@@ -24,7 +24,7 @@ class _ProfileCurrencyEditState extends State<ProfileCurrencyEdit> {
void initState() { void initState() {
super.initState(); super.initState();
MainModel mainModel = Provider.of<MainModel>(context, listen: false); MainModel mainModel = Provider.of<MainModel>(context, listen: false);
if (mainModel.user.preferCurrency == "MMK") { if (mainModel.user!.preferCurrency == "MMK") {
_currency = Currency.MMK; _currency = Currency.MMK;
} else { } else {
_currency = Currency.USD; _currency = Currency.USD;
@@ -77,7 +77,7 @@ class _ProfileCurrencyEditState extends State<ProfileCurrencyEdit> {
onChanged: (Currency? value) { onChanged: (Currency? value) {
if(value != null) if(value != null)
setState(() { setState(() {
_currency = value!; _currency = value;
}); });
}, },
), ),
@@ -96,7 +96,7 @@ class _ProfileCurrencyEditState extends State<ProfileCurrencyEdit> {
onChanged: (Currency? value) { onChanged: (Currency? value) {
if(value != null) if(value != null)
setState(() { setState(() {
_currency = value!; _currency = value;
}); });
}, },
), ),

View File

@@ -24,7 +24,7 @@ class _ProfileEditState extends State<ProfileEdit> {
void initState() { void initState() {
super.initState(); super.initState();
MainModel mainModel = Provider.of<MainModel>(context, listen: false); MainModel mainModel = Provider.of<MainModel>(context, listen: false);
nameController.text = mainModel.user.name; nameController.text = mainModel.user!.name ?? "";
} }
@override @override

View File

@@ -18,6 +18,7 @@ import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter/services.dart'; import 'package:flutter/services.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
import 'package:flutter_vector_icons/flutter_vector_icons.dart';
import '../../helpers/theme.dart'; import '../../helpers/theme.dart';
@@ -31,7 +32,7 @@ class Profile extends StatefulWidget {
class _ProfileState extends State<Profile> { class _ProfileState extends State<Profile> {
GlobalKey key = GlobalKey(); GlobalKey key = GlobalKey();
bool _isLoading = false; bool _isLoading = false;
String? selectedLanguage ; String? selectedLanguage;
TextEditingController bizNameController = new TextEditingController(); TextEditingController bizNameController = new TextEditingController();
static final List<String> languagesList = Translation().supportedLanguages; static final List<String> languagesList = Translation().supportedLanguages;
@@ -67,18 +68,19 @@ class _ProfileState extends State<Profile> {
Provider.of<DeliveryAddressModel>(context); Provider.of<DeliveryAddressModel>(context);
final namebox = DisplayText( final namebox = DisplayText(
text: mainModel.user.name + " (${mainModel.user.status})", text: "${mainModel.user!.name ?? ''}" +
" (${mainModel.user!.status ?? ''})",
labelTextKey: "profile.name", labelTextKey: "profile.name",
iconData: Icons.person, iconData: Icons.person,
); );
final currencyBox = DisplayText( final currencyBox = DisplayText(
text: mainModel.user.preferCurrency, text: mainModel.user!.preferCurrency ?? "",
labelTextKey: "profile.currency", labelTextKey: "profile.currency",
iconData: FontAwesome5.money_bill_alt, iconData: FontAwesome5Regular.money_bill_alt,
); );
final phonenumberbox = DisplayText( final phonenumberbox = DisplayText(
text: mainModel.user.phone, text: mainModel.user!.phone ?? "",
labelTextKey: "profile.phone", labelTextKey: "profile.phone",
iconData: Icons.phone, iconData: Icons.phone,
); );
@@ -86,15 +88,15 @@ class _ProfileState extends State<Profile> {
children: [ children: [
Expanded( Expanded(
child: DisplayText( child: DisplayText(
text: mainModel.user.fcsID, text: mainModel.user!.fcsID ?? "",
labelTextKey: "customer.fcs.id", labelTextKey: "customer.fcs.id",
icon: FcsIDIcon(), icon: FcsIDIcon(),
), ),
), ),
IconButton( IconButton(
icon: Icon(Icons.content_copy, color: Colors.grey), icon: Icon(Icons.content_copy, color: Colors.grey),
onPressed: () => _copy( onPressed: () => _copy(getLocalString(context, "customer.fcs.id"),
getLocalString(context, "customer.fcs.id"), mainModel.user.fcsID), mainModel.user!.fcsID ?? ""),
) )
], ],
); );
@@ -103,7 +105,7 @@ class _ProfileState extends State<Profile> {
children: [ children: [
Expanded( Expanded(
child: DisplayText( child: DisplayText(
text: mainModel.setting.usaAddress, text: mainModel.setting!.usaAddress ?? "",
labelTextKey: "profile.usa.shipping.address", labelTextKey: "profile.usa.shipping.address",
iconData: Icons.location_on, iconData: Icons.location_on,
), ),
@@ -112,7 +114,7 @@ class _ProfileState extends State<Profile> {
icon: Icon(Icons.content_copy, color: Colors.grey), icon: Icon(Icons.content_copy, color: Colors.grey),
onPressed: () => _copy( onPressed: () => _copy(
getLocalString(context, "profile.usa.shipping.address"), getLocalString(context, "profile.usa.shipping.address"),
mainModel.setting.usaAddress), mainModel.setting!.usaAddress ?? ""),
) )
], ],
); );
@@ -195,14 +197,14 @@ class _ProfileState extends State<Profile> {
} }
Widget getPrivilegeBox(BuildContext context) { Widget getPrivilegeBox(BuildContext context) {
User user = Provider.of<MainModel>(context, listen: false).user; User? user = Provider.of<MainModel>(context, listen: false).user;
List<Privilege> _privileges = List<Privilege> _privileges =
Provider.of<StaffModel>(context, listen: false).privileges; Provider.of<StaffModel>(context, listen: false).privileges;
if (user == null || user.isCustomer()) return Container(); if (user == null || user.isCustomer()) return Container();
List<Privilege> privileges = []; List<Privilege> privileges = [];
user.privileges.forEach((e) { user.privileges.forEach((e) {
var p = _privileges.firstWhere((p) => p.id == e, orElse: () => null); var p = _privileges.firstWhere((p) => p.id == e);
if (p != null) { if (p != null) {
privileges.add(p); privileges.add(p);
} }
@@ -270,7 +272,7 @@ class _ProfileState extends State<Profile> {
} }
_showToast(String title) { _showToast(String title) {
final ScaffoldState scaffold = key.currentState; final ScaffoldState scaffold = key.currentState as ScaffoldState;
scaffold.showSnackBar( scaffold.showSnackBar(
SnackBar( SnackBar(
content: Text('copied "$title" data to clipboard'), content: Text('copied "$title" data to clipboard'),

View File

@@ -36,7 +36,7 @@ class _CustomEditorState extends State<CustomEditor> {
_productController.text = _custom.name??""; _productController.text = _custom.name??"";
_feeController.text = _custom.customDutyFee.toStringAsFixed(2); _feeController.text = _custom.customDutyFee.toStringAsFixed(2);
_shipmentRateController.text = _shipmentRateController.text =
_custom.rate == null ? "" : _custom.rate.toStringAsFixed(2); _custom.rate == null ? "" : _custom.rate?.toStringAsFixed(2) ?? '';
} else { } else {
_isNew = true; _isNew = true;
} }

View File

@@ -85,7 +85,8 @@ class _CustomListState extends State<CustomList> {
"Custom Fee \$ " + custom.customDutyFee.toStringAsFixed(2), "Custom Fee \$ " + custom.customDutyFee.toStringAsFixed(2),
custom.rate == null custom.rate == null
? "" ? ""
: "Shipment rate \$ " + custom.rate.toStringAsFixed(2)), : "Shipment rate \$ " +
custom.rate!.toStringAsFixed(2)),
), ),
); );
}), }),

View File

@@ -62,11 +62,11 @@ class _ShipmentRatesCalState extends State<ShipmentRatesCal> {
var amount = box.calAmount(rate); var amount = box.calAmount(rate);
var shipmentWeight = box.getShipmentWeight(rate.volumetricRatio); var shipmentWeight = box.getShipmentWeight(rate.volumetricRatio);
var effectiveWeight = var effectiveWeight =
_cargoType.weight > shipmentWeight ? _cargoType.weight : shipmentWeight; _cargoType.weight! > shipmentWeight ? _cargoType.weight : shipmentWeight;
setState(() { setState(() {
_deliveryFee = _deliveryFee =
effectiveWeight > rate.freeDeliveryWeight ? 0 : rate.deliveryFee; effectiveWeight! > rate.freeDeliveryWeight ? 0 : rate.deliveryFee;
_amount = amount == null ? 0 : amount + _deliveryFee; _amount = amount == null ? 0 : amount + _deliveryFee;
_shipmentWeight = shipmentWeight.toDouble(); _shipmentWeight = shipmentWeight.toDouble();
}); });

View File

@@ -42,8 +42,8 @@ class _ReceivingEditorState extends State<ReceivingEditor> {
_isNew = widget.package == null; _isNew = widget.package == null;
if (!_isNew) { if (!_isNew) {
package = widget.package!; package = widget.package!;
_trackingIDCtl.text = package.trackingID; _trackingIDCtl.text = package.trackingID!;
_remarkCtl.text = package.remark; _remarkCtl.text = package.remark!;
_multiImgController.setImageUrls = package.photoUrls; _multiImgController.setImageUrls = package.photoUrls;
user = User( user = User(
fcsID: package.fcsID, fcsID: package.fcsID,

View File

@@ -149,8 +149,8 @@ class _ReceivingInfoState extends State<ReceivingInfo> {
); );
PackageModel packageModel = PackageModel packageModel =
Provider.of<PackageModel>(context, listen: false); Provider.of<PackageModel>(context, listen: false);
var pkg = await packageModel.getPackage(widget.package!.id); var pkg = await packageModel.getPackage(widget.package!.id!);
initPackage(pkg); initPackage(pkg!);
} }
_delete() { _delete() {

View File

@@ -94,7 +94,7 @@ class _ReceivingListState extends State<ReceivingList> {
_searchCallback(Package package) async { _searchCallback(Package package) async {
var packageModel = Provider.of<PackageModel>(context, listen: false); var packageModel = Provider.of<PackageModel>(context, listen: false);
Package _package = await packageModel.getPackage(package.id); Package? _package = await packageModel.getPackage(package.id!);
if (_package == null) return; if (_package == null) return;
Navigator.push( Navigator.push(
context, context,

View File

@@ -55,7 +55,7 @@ class ReceivingListRow extends StatelessWidget {
Padding( Padding(
padding: const EdgeInsets.only(left: 8.0), padding: const EdgeInsets.only(left: 8.0),
child: new Text( child: new Text(
package!.id == null ? '' : package!.trackingID, package!.id == null ? '' : package!.trackingID!,
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.black), fontSize: 15.0, color: Colors.black),
), ),
@@ -63,7 +63,7 @@ class ReceivingListRow extends StatelessWidget {
Padding( Padding(
padding: const EdgeInsets.only(left: 8.0), padding: const EdgeInsets.only(left: 8.0),
child: new Text( child: new Text(
package!.market == null ? '' : package!.market, package!.market == null ? '' : package!.market!,
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.black), fontSize: 15.0, color: Colors.black),
), ),
@@ -79,12 +79,14 @@ class ReceivingListRow extends StatelessWidget {
children: <Widget>[ children: <Widget>[
Padding( Padding(
padding: const EdgeInsets.all(3.0), padding: const EdgeInsets.all(3.0),
child: getStatus(package!.status), child: getStatus(package!.status??""),
), ),
Padding( Padding(
padding: const EdgeInsets.all(0), padding: const EdgeInsets.all(0),
child: new Text( child: new Text(
dateFormat.format(package!.currentStatusDate), package!.currentStatusDate != null
? dateFormat.format(package!.currentStatusDate!)
: '',
style: new TextStyle(fontSize: 15.0, color: Colors.grey), style: new TextStyle(fontSize: 15.0, color: Colors.grey),
), ),
), ),

View File

@@ -5,7 +5,7 @@ import 'package:flutter/material.dart';
class BoxRow extends StatelessWidget { class BoxRow extends StatelessWidget {
final Carton box; final Carton box;
const BoxRow({Key key, this.box}) : super(key: key); const BoxRow({Key? key, required this.box}) : super(key: key);
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {

View File

@@ -11,7 +11,7 @@ import 'package:logging/logging.dart';
class ShipmentModel extends BaseModel { class ShipmentModel extends BaseModel {
final log = Logger('ShipmentModel'); final log = Logger('ShipmentModel');
StreamSubscription<QuerySnapshot> listener; StreamSubscription<QuerySnapshot>? listener;
List<Shipment> get shipments => _menuSelectedIndex == 1 List<Shipment> get shipments => _menuSelectedIndex == 1
? _shipments ? _shipments
@@ -19,7 +19,7 @@ class ShipmentModel extends BaseModel {
List<Shipment> _shipments = []; List<Shipment> _shipments = [];
Paginator _delivered; late Paginator _delivered;
bool isLoading = false; bool isLoading = false;
int _menuSelectedIndex = 1; int _menuSelectedIndex = 1;
@@ -28,7 +28,7 @@ class ShipmentModel extends BaseModel {
notifyListeners(); notifyListeners();
} }
get menuSelectedIndex => _menuSelectedIndex; int get menuSelectedIndex => _menuSelectedIndex;
initData(bool forCustomer, {bool myPickup = false}) { initData(bool forCustomer, {bool myPickup = false}) {
logout(); logout();
@@ -43,11 +43,11 @@ class ShipmentModel extends BaseModel {
@override @override
logout() async { logout() async {
if (_delivered != null) _delivered.close(); if (_delivered != null) _delivered.close();
if (listener != null) await listener.cancel(); if (listener != null) await listener!.cancel();
_shipments = []; _shipments = [];
} }
Future<void> loadMore({bool isCustomer}) async { Future<void> loadMore({bool? isCustomer}) async {
if (menuSelectedIndex == 1) if (menuSelectedIndex == 1)
return; // when delivered menu is not selected return return; // when delivered menu is not selected return
if (_delivered.ended) return; if (_delivered.ended) return;
@@ -59,7 +59,7 @@ class ShipmentModel extends BaseModel {
}); });
} }
Future<void> refresh({bool isCustomer}) async { Future<void> refresh({bool? isCustomer}) async {
if (menuSelectedIndex == 1) if (menuSelectedIndex == 1)
return; // when delivered menu is not selected return return; // when delivered menu is not selected return
await _delivered.refresh(onFinished: () { await _delivered.refresh(onFinished: () {
@@ -70,16 +70,16 @@ class ShipmentModel extends BaseModel {
Paginator _getDelivered(bool isCustomer) { Paginator _getDelivered(bool isCustomer) {
if (!isCustomer) { if (!isCustomer) {
if (user == null || if (user == null ||
!((user.hasPackages() || !((user!.hasPackages() ||
user.hasReceiving() || user!.hasReceiving() ||
user.hasProcessing()))) throw "No privilege"; user!.hasProcessing()))) throw "No privilege";
} }
var pageQuery = Firestore.instance var pageQuery = FirebaseFirestore.instance
.collection("/$shipments_collection") .collection("/$shipments_collection")
.where("is_delivered", isEqualTo: true) .where("is_delivered", isEqualTo: true)
.where("is_deleted", isEqualTo: false); .where("is_deleted", isEqualTo: false);
if (isCustomer) { if (isCustomer) {
pageQuery = pageQuery.where("user_id", isEqualTo: user.id); pageQuery = pageQuery.where("user_id", isEqualTo: user!.id);
} }
pageQuery = pageQuery.orderBy("status_date", descending: true); pageQuery = pageQuery.orderBy("status_date", descending: true);
var paginator = new Paginator(pageQuery, rowPerLoad: 20, toObj: (data, id) { var paginator = new Paginator(pageQuery, rowPerLoad: 20, toObj: (data, id) {
@@ -90,30 +90,31 @@ class ShipmentModel extends BaseModel {
Future<void> _loadShipments(bool forCustomer, bool myPickup) async { Future<void> _loadShipments(bool forCustomer, bool myPickup) async {
if (user == null) return; if (user == null) return;
if (!forCustomer && !user.hasShipment()) return; if (!forCustomer && !user!.hasShipment()) return;
if (listener != null) listener.cancel(); if (listener != null) listener!.cancel();
_shipments = []; _shipments = [];
try { try {
var q = Firestore.instance var q = FirebaseFirestore.instance
.collection("$shipments_collection") .collection("$shipments_collection")
.where("is_delivered", isEqualTo: false) .where("is_delivered", isEqualTo: false)
.where("is_canceled", isEqualTo: false) .where("is_canceled", isEqualTo: false)
.where("is_deleted", isEqualTo: false); .where("is_deleted", isEqualTo: false);
if (forCustomer) { if (forCustomer) {
q = q.where("user_id", isEqualTo: user.id); q = q.where("user_id", isEqualTo: user!.id);
} }
if (myPickup) { if (myPickup) {
q = q.where("pickup_user_id", isEqualTo: user.id); q = q.where("pickup_user_id", isEqualTo: user!.id);
} }
q = q.orderBy("created_at", descending: true); q = q.orderBy("created_at", descending: true);
listener = q.snapshots().listen((QuerySnapshot snapshot) { listener = q.snapshots().listen((QuerySnapshot snapshot) {
_shipments.clear(); _shipments.clear();
_shipments = snapshot.documents.map((documentSnapshot) { _shipments = snapshot.docs.map((documentSnapshot) {
var s = Shipment.fromMap( var s = Shipment.fromMap(
documentSnapshot.data, documentSnapshot.documentID); documentSnapshot.data as Map<String, dynamic>,
documentSnapshot.id);
return s; return s;
}).toList(); }).toList();
notifyListeners(); notifyListeners();
@@ -130,18 +131,17 @@ class ShipmentModel extends BaseModel {
shipment_courier_dropoff shipment_courier_dropoff
]; ];
Shipment getActiveShipment(String shipmentID) { Shipment? getActiveShipment(String shipmentID) {
return _shipments?.firstWhere((e) => e.id == shipmentID, return _shipments.firstWhere((e) => e.id == shipmentID);
orElse: () => null);
} }
Future<Shipment> getShipment(String shipmentID) async { Future<Shipment?> getShipment(String shipmentID) async {
String path = "/$shipments_collection"; String path = "/$shipments_collection";
try { try {
var ref = Firestore.instance.collection("$path").document(shipmentID); var ref = FirebaseFirestore.instance.collection("$path").doc(shipmentID);
var snap = await ref.get(source: Source.server); var snap = await ref.get(const GetOptions(source: Source.server));
if (snap.exists) { if (snap.exists) {
var s = Shipment.fromMap(snap.data, snap.documentID); var s = Shipment.fromMap(snap.data as Map<String, dynamic>, snap.id);
return s; return s;
} }
} catch (e) { } catch (e) {
@@ -150,20 +150,20 @@ class ShipmentModel extends BaseModel {
return null; return null;
} }
Future<List<Shipment>> getShipmentWithHandlingFee( Future<List<Shipment?>?> getShipmentWithHandlingFee(
String fcsShipmentID, String userID) async { String fcsShipmentID, String userID) async {
String path = "/$shipments_collection"; String path = "/$shipments_collection";
try { try {
var q = Firestore.instance var q = FirebaseFirestore.instance
.collection("$path") .collection("$path")
.where("user_id", isEqualTo: userID) .where("user_id", isEqualTo: userID)
.where("is_deleted", isEqualTo: false) .where("is_deleted", isEqualTo: false)
.where("handling_fee", isGreaterThan: 0) .where("handling_fee", isGreaterThan: 0)
.where("fcs_shipment_id", isEqualTo: fcsShipmentID); .where("fcs_shipment_id", isEqualTo: fcsShipmentID);
var snaps = await q.getDocuments(source: Source.server); var snaps = await q.get(const GetOptions(source: Source.server));
List<Shipment> shipments = snaps.documents.map((snap) { List<Shipment?> shipments = snaps.docs.map((snap) {
if (snap.exists) { if (snap.exists) {
var s = Shipment.fromMap(snap.data, snap.documentID); var s = Shipment.fromMap(snap.data as Map<String, dynamic>, snap.id);
return s; return s;
} }
}).toList(); }).toList();

View File

@@ -14,14 +14,14 @@ import 'package:fcs/pages/widgets/local_title.dart';
import 'package:fcs/pages/widgets/progress.dart'; import 'package:fcs/pages/widgets/progress.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_icons/flutter_icons.dart'; import 'package:flutter_icons_null_safety/flutter_icons_null_safety.dart';
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
import 'widgets.dart'; import 'widgets.dart';
class ShipmentAssign extends StatefulWidget { class ShipmentAssign extends StatefulWidget {
final Shipment shipment; final Shipment? shipment;
ShipmentAssign({this.shipment}); ShipmentAssign({this.shipment});
@override @override
@@ -38,13 +38,13 @@ class _ShipmentAssignState extends State<ShipmentAssign> {
TextEditingController _pickupDate = new TextEditingController(); TextEditingController _pickupDate = new TextEditingController();
TextEditingController _handlingFee = new TextEditingController(); TextEditingController _handlingFee = new TextEditingController();
Shipment _shipment; Shipment? _shipment;
bool _isLoading = false; bool _isLoading = false;
var now = new DateTime.now(); var now = new DateTime.now();
String _selectedShipmentType; String? _selectedShipmentType;
User _user; User? _user;
List<User> _users; List<User>? _users;
@override @override
void initState() { void initState() {
@@ -53,18 +53,19 @@ class _ShipmentAssignState extends State<ShipmentAssign> {
_shipment = widget.shipment; _shipment = widget.shipment;
_loadUsers(); _loadUsers();
_selectedShipmentType = _shipment.shipmentType; _selectedShipmentType = _shipment!.shipmentType;
_fromTimeEditingController.text = _shipment.pickupTimeStart; _fromTimeEditingController.text = _shipment!.pickupTimeStart!;
_toTimeEditingController.text = _shipment.pickupTimeEnd; _toTimeEditingController.text = _shipment!.pickupTimeEnd!;
_pickupDate.text = dateFormatter.format(_shipment.pickupDate ?? now); _pickupDate.text = dateFormatter.format(_shipment!.pickupDate! ?? now);
_handlingFee.text = _shipment.handlingFee?.toString() ?? "0"; _handlingFee.text = _shipment!.handlingFee != null
? _shipment!.handlingFee.toString()
: "0";
} }
_loadUsers() async { _loadUsers() async {
StaffModel staffModel = Provider.of<StaffModel>(context, listen: false); StaffModel staffModel = Provider.of<StaffModel>(context, listen: false);
var users = await staffModel.getPickupEmployees(); var users = await staffModel.getPickupEmployees();
var selectUser = users.firstWhere((e) => e.id == _shipment.pickupUserID, var selectUser = users.firstWhere((e) => e.id == _shipment!.pickupUserID);
orElse: () => null);
setState(() { setState(() {
_users = users; _users = users;
_user = selectUser; _user = selectUser;
@@ -74,7 +75,7 @@ class _ShipmentAssignState extends State<ShipmentAssign> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
ShipmentModel pickupModel = Provider.of<ShipmentModel>(context); ShipmentModel pickupModel = Provider.of<ShipmentModel>(context);
final shipmentNumberBox = getShipmentNumberStatus(context, _shipment); final shipmentNumberBox = getShipmentNumberStatus(context, _shipment!);
bool isLocalPickup = _selectedShipmentType == shipment_local_pickup; bool isLocalPickup = _selectedShipmentType == shipment_local_pickup;
bool isCourierPickup = _selectedShipmentType == shipment_courier_pickup; bool isCourierPickup = _selectedShipmentType == shipment_courier_pickup;
bool isLocalDropoff = _selectedShipmentType == shipment_local_dropoff; bool isLocalDropoff = _selectedShipmentType == shipment_local_dropoff;
@@ -146,15 +147,15 @@ class _ShipmentAssignState extends State<ShipmentAssign> {
} }
_save() async { _save() async {
_shipment.pickupUserID = this._user.id; _shipment!.pickupUserID = this._user!.id;
_shipment.handlingFee = double.tryParse(_handlingFee.text) ?? 0; _shipment!.handlingFee = double.tryParse(_handlingFee.text) ?? 0;
setState(() { setState(() {
_isLoading = true; _isLoading = true;
}); });
try { try {
ShipmentModel shipmentModel = ShipmentModel shipmentModel =
Provider.of<ShipmentModel>(context, listen: false); Provider.of<ShipmentModel>(context, listen: false);
await shipmentModel.assignShipment(_shipment); await shipmentModel.assignShipment(_shipment!);
Navigator.pop(context, true); Navigator.pop(context, true);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());

View File

@@ -17,11 +17,12 @@ import 'package:fcs/pages/widgets/my_data_table.dart';
import 'package:fcs/pages/widgets/progress.dart'; import 'package:fcs/pages/widgets/progress.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_icons/flutter_icons.dart'; import 'package:flutter_icons_null_safety/flutter_icons_null_safety.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
class ShipmentBoxEditor extends StatefulWidget { class ShipmentBoxEditor extends StatefulWidget {
final Carton box; final Carton? box;
ShipmentBoxEditor({this.box}); ShipmentBoxEditor({this.box});
@override @override
@@ -33,9 +34,9 @@ class _ShipmentBoxEditorState extends State<ShipmentBoxEditor> {
TextEditingController _widthCtl = new TextEditingController(); TextEditingController _widthCtl = new TextEditingController();
TextEditingController _heightCtl = new TextEditingController(); TextEditingController _heightCtl = new TextEditingController();
Carton _box; Carton? _box;
bool _isLoading = false; bool _isLoading = false;
bool _isNew; late bool _isNew;
double volumetricRatio = 0; double volumetricRatio = 0;
double shipmentWeight = 0; double shipmentWeight = 0;
@@ -50,16 +51,16 @@ class _ShipmentBoxEditorState extends State<ShipmentBoxEditor> {
if (widget.box != null) { if (widget.box != null) {
_box = widget.box; _box = widget.box;
_isNew = false; _isNew = false;
_lengthCtl.text = _box.length.toString(); _lengthCtl.text = _box!.length != null ? _box!.length.toString() : '';
_widthCtl.text = _box.width.toString(); _widthCtl.text = _box!.width != null ? _box!.width.toString() : '';
_heightCtl.text = _box.height.toString(); _heightCtl.text = _box!.height != null ? _box!.height.toString() : '';
} else { } else {
var shipmentModel = var shipmentModel =
Provider.of<DeliveryAddressModel>(context, listen: false); Provider.of<DeliveryAddressModel>(context, listen: false);
_isNew = true; _isNew = true;
_box = Carton(cargoTypes: []); _box = Carton(cargoTypes: []);
_box.deliveryAddress = shipmentModel.defalutAddress; _box!.deliveryAddress = shipmentModel.defalutAddress;
_lengthCtl.text = "12"; _lengthCtl.text = "12";
_widthCtl.text = "12"; _widthCtl.text = "12";
@@ -151,10 +152,11 @@ class _ShipmentBoxEditorState extends State<ShipmentBoxEditor> {
color: primaryColor, color: primaryColor,
), ),
onPressed: () async { onPressed: () async {
CargoType cargo = await Navigator.push<CargoType>( CargoType? cargo = await Navigator.push<CargoType>(
context, context,
CupertinoPageRoute( CupertinoPageRoute(
builder: (context) => CargoTypeEditor())); builder: (context) => CargoTypeEditor()));
if (cargo == null) return;
_addCargo(cargo); _addCargo(cargo);
}), }),
), ),
@@ -186,19 +188,19 @@ class _ShipmentBoxEditorState extends State<ShipmentBoxEditor> {
shipmentWeightBox, shipmentWeightBox,
LocalTitle(textKey: "shipment.box.delivery"), LocalTitle(textKey: "shipment.box.delivery"),
DefaultDeliveryAddress( DefaultDeliveryAddress(
deliveryAddress: _box.deliveryAddress, deliveryAddress: _box!.deliveryAddress,
labelKey: "shipment.box.delivery", labelKey: "shipment.box.delivery",
onTap: () async { onTap: () async {
DeliveryAddress d = await Navigator.push<DeliveryAddress>( DeliveryAddress? d = await Navigator.push<DeliveryAddress>(
context, context,
CupertinoPageRoute( CupertinoPageRoute(
builder: (context) => DeliveryAddressSelection( builder: (context) => DeliveryAddressSelection(
deliveryAddress: _box.deliveryAddress, deliveryAddress: _box!.deliveryAddress,
user: mainModel.user)), user: mainModel.user)),
); );
if (d == null) return; if (d == null) return;
setState(() { setState(() {
_box.deliveryAddress = d; _box!.deliveryAddress = d;
}); });
}), }),
createBtn createBtn
@@ -210,25 +212,26 @@ class _ShipmentBoxEditorState extends State<ShipmentBoxEditor> {
} }
List<MyDataRow> getCargoRows(BuildContext context) { List<MyDataRow> getCargoRows(BuildContext context) {
if (_box.cargoTypes == null) { if (_box!.cargoTypes == null) {
return []; return [];
} }
double total = 0; double total = 0;
var rows = _box.cargoTypes.map((c) { var rows = _box!.cargoTypes.map((c) {
total += c.weight; total += c.weight;
return MyDataRow( return MyDataRow(
onSelectChanged: (bool selected) async { onSelectChanged: (bool selected) async {
CargoType cargo = await Navigator.push<CargoType>( CargoType? cargo = await Navigator.push<CargoType>(
context, context,
CupertinoPageRoute( CupertinoPageRoute(
builder: (context) => CargoTypeEditor( builder: (context) => CargoTypeEditor(
cargo: c, cargo: c,
))); )));
if (cargo == null) return;
_addCargo(cargo); _addCargo(cargo);
}, },
cells: [ cells: [
MyDataCell(new Text( MyDataCell(new Text(
c.name == null ? "" : c.name, c.name == null ? "" : c.name!,
style: textStyle, style: textStyle,
)), )),
MyDataCell( MyDataCell(
@@ -281,14 +284,14 @@ class _ShipmentBoxEditorState extends State<ShipmentBoxEditor> {
_addCargo(CargoType cargo) { _addCargo(CargoType cargo) {
if (cargo == null) return; if (cargo == null) return;
setState(() { setState(() {
_box.cargoTypes.remove(cargo); _box!.cargoTypes.remove(cargo);
_box.cargoTypes.add(cargo); _box!.cargoTypes.add(cargo);
}); });
} }
_removeCargo(CargoType cargo) { _removeCargo(CargoType cargo) {
setState(() { setState(() {
_box.cargoTypes.remove(cargo); _box!.cargoTypes.remove(cargo);
}); });
} }
@@ -296,9 +299,9 @@ class _ShipmentBoxEditorState extends State<ShipmentBoxEditor> {
double l = double.parse(_lengthCtl.text, (s) => 0); double l = double.parse(_lengthCtl.text, (s) => 0);
double w = double.parse(_widthCtl.text, (s) => 0); double w = double.parse(_widthCtl.text, (s) => 0);
double h = double.parse(_heightCtl.text, (s) => 0); double h = double.parse(_heightCtl.text, (s) => 0);
_box.length = l; _box!.length = l;
_box.width = w; _box!.width = w;
_box.height = h; _box!.height = h;
Navigator.pop(context, _box); Navigator.pop(context, _box);
} }
} }

View File

@@ -8,14 +8,15 @@ import 'package:fcs/pages/widgets/local_text.dart';
import 'package:fcs/pages/widgets/progress.dart'; import 'package:fcs/pages/widgets/progress.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_icons/flutter_icons.dart'; import 'package:flutter_icons_null_safety/flutter_icons_null_safety.dart';
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
import 'widgets.dart'; import 'widgets.dart';
class ShipmentConfirm extends StatefulWidget { class ShipmentConfirm extends StatefulWidget {
final Shipment shipment; final Shipment? shipment;
ShipmentConfirm({this.shipment}); ShipmentConfirm({this.shipment});
@override @override
@@ -27,7 +28,7 @@ class _ShipmentConfirmState extends State<ShipmentConfirm> {
var timeFormatter = new DateFormat('jm'); var timeFormatter = new DateFormat('jm');
TextEditingController _handlingFee = new TextEditingController(); TextEditingController _handlingFee = new TextEditingController();
Shipment _shipment; Shipment? _shipment;
bool _isLoading = false; bool _isLoading = false;
var now = new DateTime.now(); var now = new DateTime.now();
@@ -36,17 +37,17 @@ class _ShipmentConfirmState extends State<ShipmentConfirm> {
super.initState(); super.initState();
_shipment = widget.shipment; _shipment = widget.shipment;
_handlingFee.text = _shipment.handlingFee?.toString() ?? "0"; _handlingFee.text = _shipment!.handlingFee?.toString() ?? "0";
} }
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final shipmentNumberBox = getShipmentNumberStatus(context, _shipment); final shipmentNumberBox = getShipmentNumberStatus(context, _shipment!);
final handlingFeeBox = InputText( final handlingFeeBox = InputText(
labelTextKey: "shipment.handling.fee", labelTextKey: "shipment.handling.fee",
controller: _handlingFee, controller: _handlingFee,
iconData: FontAwesome.truck, iconData: FontAwesomeIcons.truck,
); );
final confirmbtn = LocalButton( final confirmbtn = LocalButton(
textKey: "shipment.confirm.btn", textKey: "shipment.confirm.btn",
@@ -94,14 +95,14 @@ class _ShipmentConfirmState extends State<ShipmentConfirm> {
} }
_save() async { _save() async {
_shipment.handlingFee = double.tryParse(_handlingFee.text) ?? 0; _shipment!.handlingFee = double.tryParse(_handlingFee.text) ?? 0;
setState(() { setState(() {
_isLoading = true; _isLoading = true;
}); });
try { try {
ShipmentModel shipmentModel = ShipmentModel shipmentModel =
Provider.of<ShipmentModel>(context, listen: false); Provider.of<ShipmentModel>(context, listen: false);
await shipmentModel.confirmShipment(_shipment); await shipmentModel.confirmShipment(_shipment!);
Navigator.pop(context, true); Navigator.pop(context, true);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());

View File

@@ -29,7 +29,7 @@ import 'shipment_box_editor.dart';
import 'widgets.dart'; import 'widgets.dart';
class ShipmentEditor extends StatefulWidget { class ShipmentEditor extends StatefulWidget {
final Shipment shipment; final Shipment? shipment;
ShipmentEditor({this.shipment}); ShipmentEditor({this.shipment});
@override @override
@@ -45,12 +45,12 @@ class _ShipmentEditorState extends State<ShipmentEditor> {
TextEditingController _toTimeEditingController = new TextEditingController(); TextEditingController _toTimeEditingController = new TextEditingController();
TextEditingController _pickupDate = new TextEditingController(); TextEditingController _pickupDate = new TextEditingController();
Shipment _shipment; Shipment? _shipment;
bool _isLoading = false; bool _isLoading = false;
var now = new DateTime.now(); var now = new DateTime.now();
bool _isNew; late bool _isNew;
String _selectedShipmentType; String? _selectedShipmentType;
@override @override
void initState() { void initState() {
@@ -59,20 +59,21 @@ class _ShipmentEditorState extends State<ShipmentEditor> {
if (widget.shipment != null) { if (widget.shipment != null) {
_isNew = false; _isNew = false;
_shipment = widget.shipment; _shipment = widget.shipment;
_selectedShipmentType = _shipment.shipmentType; _selectedShipmentType = _shipment!.shipmentType;
_fromTimeEditingController.text = _shipment.pickupTimeStart; _fromTimeEditingController.text = _shipment!.pickupTimeStart ?? "";
_toTimeEditingController.text = _shipment.pickupTimeEnd; _toTimeEditingController.text = _shipment!.pickupTimeEnd ?? "";
_pickupDate.text = dateFormatter.format(_shipment.pickupDate ?? now); _pickupDate.text = dateFormatter.format(_shipment!.pickupDate ?? now);
} else { } else {
_isNew = true; _isNew = true;
_selectedShipmentType = shipment_local_pickup; _selectedShipmentType = shipment_local_pickup;
_pickupDate.text = dateFormatter.format(now); _pickupDate.text = dateFormatter.format(now);
_fromTimeEditingController.text = "${timeFormatter.format(now)}"; _fromTimeEditingController.text = "${timeFormatter.format(now)}";
_toTimeEditingController.text = "${timeFormatter.format(now)}"; _toTimeEditingController.text = "${timeFormatter.format(now)}";
_shipment = Shipment(boxes: []); // _shipment = Shipment(boxes: []);
var shipmentModel = var shipmentModel =
Provider.of<DeliveryAddressModel>(context, listen: false); Provider.of<DeliveryAddressModel>(context, listen: false);
_shipment.pickupAddress = shipmentModel.defalutAddress; _shipment!.pickupAddress = shipmentModel.defalutAddress;
_shipment!.boxes = [];
_pickupDate.text = dateFormatter.format(now); _pickupDate.text = dateFormatter.format(now);
} }
} }
@@ -86,7 +87,7 @@ class _ShipmentEditorState extends State<ShipmentEditor> {
Widget build(BuildContext context) { Widget build(BuildContext context) {
MainModel mainModel = Provider.of<MainModel>(context); MainModel mainModel = Provider.of<MainModel>(context);
ShipmentModel pickupModel = Provider.of<ShipmentModel>(context); ShipmentModel pickupModel = Provider.of<ShipmentModel>(context);
final shipmentNumberBox = getShipmentNumberStatus(context, _shipment); final shipmentNumberBox = getShipmentNumberStatus(context, _shipment!);
bool isLocalPickup = _selectedShipmentType == shipment_local_pickup; bool isLocalPickup = _selectedShipmentType == shipment_local_pickup;
bool isCourierPickup = _selectedShipmentType == shipment_courier_pickup; bool isCourierPickup = _selectedShipmentType == shipment_courier_pickup;
bool isLocalDropoff = _selectedShipmentType == shipment_local_dropoff; bool isLocalDropoff = _selectedShipmentType == shipment_local_dropoff;
@@ -121,20 +122,20 @@ class _ShipmentEditorState extends State<ShipmentEditor> {
backgroundColor: Colors.white, backgroundColor: Colors.white,
)); ));
final pickupAddressBox = DefaultDeliveryAddress( final pickupAddressBox = DefaultDeliveryAddress(
deliveryAddress: _shipment.pickupAddress, deliveryAddress: _shipment!.pickupAddress,
iconData: Icons.location_on, iconData: Icons.location_on,
labelKey: "shipment.location", labelKey: "shipment.location",
onTap: () async { onTap: () async {
DeliveryAddress address = await Navigator.push<DeliveryAddress>( DeliveryAddress? address = await Navigator.push<DeliveryAddress>(
context, context,
CupertinoPageRoute( CupertinoPageRoute(
builder: (context) => DeliveryAddressSelection( builder: (context) => DeliveryAddressSelection(
deliveryAddress: _shipment.pickupAddress, deliveryAddress: _shipment!.pickupAddress,
user: mainModel.user)), user: mainModel.user)),
); );
if (address == null) return; if (address == null) return;
setState(() { setState(() {
_shipment.pickupAddress = address; _shipment!.pickupAddress = address;
}); });
}, },
); );
@@ -142,7 +143,7 @@ class _ShipmentEditorState extends State<ShipmentEditor> {
FcsIDIcon(), FcsIDIcon(),
Expanded( Expanded(
child: DisplayText( child: DisplayText(
text: mainModel.setting.usaAddress, text: mainModel.setting!.usaAddress,
), ),
) )
]); ]);
@@ -194,7 +195,7 @@ class _ShipmentEditorState extends State<ShipmentEditor> {
LocalRadioButtons( LocalRadioButtons(
values: pickupModel.shipmentTypes, values: pickupModel.shipmentTypes,
selectedValue: _selectedShipmentType, selectedValue: _selectedShipmentType,
callback: (v) { callback: (String? v) {
setState(() { setState(() {
_selectedShipmentType = v; _selectedShipmentType = v;
}); });
@@ -239,7 +240,7 @@ class _ShipmentEditorState extends State<ShipmentEditor> {
), ),
), ),
Column( Column(
children: getBoxList(context, _shipment.boxes), children: getBoxList(context, _shipment!.boxes),
), ),
_isNew ? createBtn : updateBtn, _isNew ? createBtn : updateBtn,
], ],
@@ -276,7 +277,7 @@ class _ShipmentEditorState extends State<ShipmentEditor> {
_addBox(Carton box) { _addBox(Carton box) {
if (box == null) return; if (box == null) return;
box.cartonType = carton_from_shipments; box.cartonType = carton_from_shipments;
_shipment.boxes.add(box); _shipment!.boxes.add(box);
setState(() {}); setState(() {});
} }
@@ -287,15 +288,15 @@ class _ShipmentEditorState extends State<ShipmentEditor> {
_removeBox(Carton box) { _removeBox(Carton box) {
if (box == null) return; if (box == null) return;
_shipment.boxes.remove(box); _shipment!.boxes.remove(box);
setState(() {}); setState(() {});
} }
_save() async { _save() async {
_shipment.shipmentType = this._selectedShipmentType; _shipment!.shipmentType = this._selectedShipmentType;
_shipment.pickupDate = dateFormatter.parse(_pickupDate.text); _shipment!.pickupDate = dateFormatter.parse(_pickupDate.text);
_shipment.pickupTimeStart = _fromTimeEditingController.text; _shipment!.pickupTimeStart = _fromTimeEditingController.text;
_shipment.pickupTimeEnd = _toTimeEditingController.text; _shipment!.pickupTimeEnd = _toTimeEditingController.text;
setState(() { setState(() {
_isLoading = true; _isLoading = true;
}); });
@@ -303,9 +304,9 @@ class _ShipmentEditorState extends State<ShipmentEditor> {
ShipmentModel shipmentModel = ShipmentModel shipmentModel =
Provider.of<ShipmentModel>(context, listen: false); Provider.of<ShipmentModel>(context, listen: false);
if (_isNew) { if (_isNew) {
await shipmentModel.createShipment(_shipment); await shipmentModel.createShipment(_shipment!);
} else { } else {
await shipmentModel.updateShipment(_shipment); await shipmentModel.updateShipment(_shipment!);
} }
Navigator.pop(context, true); Navigator.pop(context, true);
} catch (e) { } catch (e) {
@@ -319,10 +320,10 @@ class _ShipmentEditorState extends State<ShipmentEditor> {
_openCourierWebsite() { _openCourierWebsite() {
MainModel mainModel = Provider.of<MainModel>(context, listen: false); MainModel mainModel = Provider.of<MainModel>(context, listen: false);
launch("${mainModel.setting.courierWebsite}"); launch("${mainModel.setting!.courierWebsite}");
} }
isDataChanged() { isDataChanged() {
return _shipment.boxes.isNotEmpty; return _shipment!.boxes.isNotEmpty;
} }
} }

View File

@@ -20,7 +20,7 @@ import 'package:fcs/pages/widgets/multi_img_controller.dart';
import 'package:fcs/pages/widgets/progress.dart'; import 'package:fcs/pages/widgets/progress.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_icons/flutter_icons.dart'; import 'package:flutter_icons_null_safety/flutter_icons_null_safety.dart';
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
import 'package:url_launcher/url_launcher.dart'; import 'package:url_launcher/url_launcher.dart';
@@ -32,8 +32,8 @@ import 'shipment_editor.dart';
import 'widgets.dart'; import 'widgets.dart';
class ShipmentInfo extends StatefulWidget { class ShipmentInfo extends StatefulWidget {
final bool isCustomer; final bool? isCustomer;
final Shipment shipment; final Shipment? shipment;
ShipmentInfo({this.shipment, this.isCustomer}); ShipmentInfo({this.shipment, this.isCustomer});
@override @override
@@ -62,39 +62,42 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
TextEditingController _pickupDate = new TextEditingController(); TextEditingController _pickupDate = new TextEditingController();
TextEditingController _handlingFeeController = new TextEditingController(); TextEditingController _handlingFeeController = new TextEditingController();
Shipment _shipment; Shipment? _shipment;
bool _isLoading = false; bool _isLoading = false;
bool _isCustomer; late bool _isCustomer;
var now = new DateTime.now(); var now = new DateTime.now();
@override @override
void initState() { void initState() {
super.initState(); super.initState();
_shipment = widget.shipment; _shipment = widget.shipment;
_isCustomer = widget.isCustomer; _isCustomer = widget.isCustomer!;
_loadCartons(_shipment.id); _loadCartons(_shipment!.id!);
_addressEditingController.text = _shipment.address; _addressEditingController.text = _shipment!.address ?? "";
_fromTimeEditingController.text = _shipment.pickupTimeStart; _fromTimeEditingController.text = _shipment!.pickupTimeStart ?? "";
_toTimeEditingController.text = _shipment.pickupTimeEnd; _toTimeEditingController.text = _shipment!.pickupTimeEnd ?? "";
_noOfPackageEditingController.text = _shipment.numberOfPackage.toString(); _noOfPackageEditingController.text = _shipment!.numberOfPackage != null
_weightEditingController.text = _shipment.weight.toString(); ? _shipment!.numberOfPackage.toString()
_pickupDate.text = dateFormatter.format(_shipment.pickupDate ?? now); : "";
_weightEditingController.text =
_shipment!.weight != null ? _shipment!.weight.toString() : "";
_pickupDate.text = dateFormatter.format(_shipment!.pickupDate ?? now);
} }
_loadCartons(String shipmentID) async { _loadCartons(String shipmentID) async {
CartonModel cartonModel = Provider.of<CartonModel>(context, listen: false); CartonModel cartonModel = Provider.of<CartonModel>(context, listen: false);
var cartons = await cartonModel.getCartons(shipmentID); var cartons = await cartonModel.getCartons(shipmentID);
setState(() { setState(() {
_shipment.boxes = cartons; _shipment!.boxes = cartons;
}); });
} }
_loadShipment(String id) async { _loadShipment(String id) async {
ShipmentModel shipmentModel = ShipmentModel shipmentModel =
Provider.of<ShipmentModel>(context, listen: false); Provider.of<ShipmentModel>(context, listen: false);
Shipment s = await shipmentModel.getShipment(id); Shipment? s = await shipmentModel.getShipment(id);
s.boxes = _shipment.boxes; s!.boxes = _shipment!.boxes;
setState(() { setState(() {
_shipment = s; _shipment = s;
}); });
@@ -109,31 +112,34 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
Widget build(BuildContext context) { Widget build(BuildContext context) {
MainModel mainModel = Provider.of<MainModel>(context); MainModel mainModel = Provider.of<MainModel>(context);
ShipmentModel pickupModel = Provider.of<ShipmentModel>(context); ShipmentModel pickupModel = Provider.of<ShipmentModel>(context);
bool isLocalPickup = _shipment.shipmentType == shipment_local_pickup; bool isLocalPickup = _shipment!.shipmentType == shipment_local_pickup;
bool isCourierPickup = _shipment.shipmentType == shipment_courier_pickup; bool isCourierPickup = _shipment!.shipmentType == shipment_courier_pickup;
bool isLocalDropoff = _shipment.shipmentType == shipment_local_dropoff; bool isLocalDropoff = _shipment!.shipmentType == shipment_local_dropoff;
bool isCourierDropoff = _shipment.shipmentType == shipment_courier_dropoff; bool isCourierDropoff = _shipment!.shipmentType == shipment_courier_dropoff;
bool isEditable = widget.isCustomer bool isEditable = widget.isCustomer!
? (_shipment.isPending || _shipment.isAssigned) ? (_shipment!.isPending || _shipment!.isAssigned)
: (_shipment.isPending || _shipment.isAssigned || _shipment.isPickuped); : (_shipment!.isPending ||
_shipment!.isAssigned ||
_shipment!.isPickuped);
bool canCancel = bool canCancel =
_shipment.isPending || _shipment.isConfirmed || _shipment.isAssigned; _shipment!.isPending || _shipment!.isConfirmed || _shipment!.isAssigned;
final popupMenu = LocalPopupMenuButton( final popupMenu = LocalPopupMenuButton(
popmenus: [ popmenus: [
LocalPopupMenu( LocalPopupMenu(
enabled: _shipment.isPending && isLocalPickup, enabled: _shipment!.isPending && isLocalPickup,
id: 1, id: 1,
textKey: "shipment.assign.for.pickup", textKey: "shipment.assign.for.pickup",
), ),
LocalPopupMenu( LocalPopupMenu(
enabled: enabled:
(_shipment.isPickuped && isLocalPickup) || _shipment.isReceived, (_shipment!.isPickuped && isLocalPickup) || _shipment!.isReceived,
id: 3, id: 3,
textKey: "shipment.pack.menuitem", textKey: "shipment.pack.menuitem",
), ),
LocalPopupMenu( LocalPopupMenu(
enabled: _shipment.isPending && (isCourierPickup || isCourierDropoff), enabled:
_shipment!.isPending && (isCourierPickup || isCourierDropoff),
id: 4, id: 4,
textKey: "shipment.confirm.menuitem", textKey: "shipment.confirm.menuitem",
), ),
@@ -149,7 +155,11 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
? _gotoAssign() ? _gotoAssign()
: p.id == 2 : p.id == 2
? _cancel() ? _cancel()
: p.id == 3 ? _gotoPack() : p.id == 4 ? _gotoConfirm() : null, : p.id == 3
? _gotoPack()
: p.id == 4
? _gotoConfirm()
: null,
); );
final popupMenuCustomer = LocalPopupMenuButton( final popupMenuCustomer = LocalPopupMenuButton(
popmenus: [ popmenus: [
@@ -163,29 +173,31 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
buttonIcon: Icons.more_vert, buttonIcon: Icons.more_vert,
popupMenuCallback: (p) => p.id == 2 ? _cancel() : null, popupMenuCallback: (p) => p.id == 2 ? _cancel() : null,
); );
final shipmentNumberBox = getShipmentNumberStatus(context, _shipment); final shipmentNumberBox = getShipmentNumberStatus(context, _shipment!);
final fromTimeBox = DisplayText( final fromTimeBox = DisplayText(
labelTextKey: 'shipment.from', labelTextKey: 'shipment.from',
iconData: Icons.timer, iconData: Icons.timer,
text: _shipment.pickupTimeStart, text: _shipment!.pickupTimeStart ?? "",
); );
final toTimeBox = DisplayText( final toTimeBox = DisplayText(
labelTextKey: 'shipment.to', labelTextKey: 'shipment.to',
iconData: Icons.timer_off, iconData: Icons.timer_off,
text: _shipment.pickupTimeEnd, text: _shipment!.pickupTimeEnd ?? "",
); );
final pickupDateBox = DisplayText( final pickupDateBox = DisplayText(
labelTextKey: "shipment.date", labelTextKey: "shipment.date",
iconData: Icons.date_range, iconData: Icons.date_range,
text: dateFormatter.format(_shipment.pickupDate), text: _shipment!.pickupDate != null
? dateFormatter.format(_shipment!.pickupDate!)
: '',
); );
var localDropoffAddressBox = Row(children: [ var localDropoffAddressBox = Row(children: [
FcsIDIcon(), FcsIDIcon(),
Expanded( Expanded(
child: DisplayText( child: DisplayText(
text: mainModel.setting.usaAddress, text: mainModel.setting!.usaAddress ?? "",
), ),
) )
]); ]);
@@ -201,17 +213,17 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
backgroundColor: Colors.white, backgroundColor: Colors.white,
)); ));
final pickupAddressBox = DefaultDeliveryAddress( final pickupAddressBox = DefaultDeliveryAddress(
deliveryAddress: _shipment.pickupAddress, deliveryAddress: _shipment!.pickupAddress,
iconData: Icons.location_on, iconData: Icons.location_on,
labelKey: "shipment.location", labelKey: "shipment.location",
); );
var usersBox = DisplayText( var usersBox = DisplayText(
labelTextKey: "shipment.staff", labelTextKey: "shipment.staff",
text: _shipment.pickupUserName, text: _shipment!.pickupUserName ?? "",
iconData: MaterialCommunityIcons.worker); iconData: MaterialCommunityIcons.worker);
var handlingFeeBox = DisplayText( var handlingFeeBox = DisplayText(
labelTextKey: "shipment.handling.fee", labelTextKey: "shipment.handling.fee",
text: (_shipment.handlingFee ?? 0).toString(), text: (_shipment!.handlingFee ?? 0).toString(),
iconData: FontAwesome.truck); iconData: FontAwesome.truck);
final assignCompleteBtn = LocalButton( final assignCompleteBtn = LocalButton(
@@ -232,7 +244,7 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
); );
final fcsShipmentNumberBox = DisplayText( final fcsShipmentNumberBox = DisplayText(
text: _shipment.fcsShipmentNumber, text: _shipment!.fcsShipmentNumber ?? "",
labelTextKey: "FCSshipment.number", labelTextKey: "FCSshipment.number",
iconData: Ionicons.ios_airplane, iconData: Ionicons.ios_airplane,
); );
@@ -267,7 +279,7 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
color: isEditable ? primaryColor : Colors.grey), color: isEditable ? primaryColor : Colors.grey),
onPressed: isEditable ? _edit : null, onPressed: isEditable ? _edit : null,
), ),
widget.isCustomer ? popupMenuCustomer : popupMenu, widget.isCustomer! ? popupMenuCustomer : popupMenu,
]), ]),
body: Padding( body: Padding(
padding: const EdgeInsets.all(10.0), padding: const EdgeInsets.all(10.0),
@@ -278,7 +290,7 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
LocalRadioButtons( LocalRadioButtons(
readOnly: true, readOnly: true,
values: pickupModel.shipmentTypes, values: pickupModel.shipmentTypes,
selectedValue: _shipment.shipmentType, selectedValue: _shipment!.shipmentType,
), ),
...(isLocalDropoff ...(isLocalDropoff
? [ ? [
@@ -305,13 +317,13 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
LocalTitle( LocalTitle(
textKey: "boxes.name", textKey: "boxes.name",
trailing: Text( trailing: Text(
"${_shipment.totalCount} Cartons - ${_shipment.totalWeight} lb"), "${_shipment!.totalCount} Cartons - ${_shipment!.totalWeight} lb"),
), ),
Column( Column(
children: getBoxList(context, _shipment.boxes), children: getBoxList(context, _shipment!.boxes),
), ),
!_isCustomer ? fcsShipmentNumberBox : Container(), !_isCustomer ? fcsShipmentNumberBox : Container(),
...(!_shipment.isPending ...(!_shipment!.isPending
? [ ? [
handlingFeeBox, handlingFeeBox,
] ]
@@ -324,22 +336,22 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
: []), : []),
...(isLocalPickup && !_isCustomer ...(isLocalPickup && !_isCustomer
? [ ? [
_shipment.isPending ? assignCompleteBtn : Container(), _shipment!.isPending ? assignCompleteBtn : Container(),
_shipment.isAssigned ? completePickupBtn : Container(), _shipment!.isAssigned ? completePickupBtn : Container(),
_shipment.isPickuped ? completePackBtn : Container(), _shipment!.isPickuped ? completePackBtn : Container(),
] ]
: []), : []),
...((isCourierPickup || isCourierDropoff) && !_isCustomer ...((isCourierPickup || isCourierDropoff) && !_isCustomer
? [ ? [
_shipment.isPending ? confirmCompleteBtn : Container(), _shipment!.isPending ? confirmCompleteBtn : Container(),
_shipment.isConfirmed ? completeReceiveBtn : Container(), _shipment!.isConfirmed ? completeReceiveBtn : Container(),
_shipment.isReceived ? completePackBtn : Container(), _shipment!.isReceived ? completePackBtn : Container(),
] ]
: []), : []),
...((isLocalDropoff) && !_isCustomer ...((isLocalDropoff) && !_isCustomer
? [ ? [
_shipment.isPending ? completeReceiveBtn : Container(), _shipment!.isPending ? completeReceiveBtn : Container(),
_shipment.isReceived ? completePackBtn : Container(), _shipment!.isReceived ? completePackBtn : Container(),
] ]
: []), : []),
], ],
@@ -361,7 +373,7 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
} }
_edit() async { _edit() async {
bool updated = await Navigator.push<bool>( bool? updated = await Navigator.push<bool>(
context, context,
CupertinoPageRoute( CupertinoPageRoute(
builder: (context) => ShipmentEditor( builder: (context) => ShipmentEditor(
@@ -369,8 +381,8 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
)), )),
); );
if (updated ?? false) { if (updated ?? false) {
await _loadShipment(_shipment.id); await _loadShipment(_shipment!.id!);
await _loadCartons(_shipment.id); await _loadCartons(_shipment!.id!);
} }
} }
@@ -387,7 +399,7 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
try { try {
ShipmentModel shipmentModel = ShipmentModel shipmentModel =
Provider.of<ShipmentModel>(context, listen: false); Provider.of<ShipmentModel>(context, listen: false);
await shipmentModel.cancelShipment(_shipment); await shipmentModel.cancelShipment(_shipment!);
Navigator.pop(context, true); Navigator.pop(context, true);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());
@@ -400,11 +412,11 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
_openCourierWebsite() { _openCourierWebsite() {
MainModel mainModel = Provider.of<MainModel>(context, listen: false); MainModel mainModel = Provider.of<MainModel>(context, listen: false);
launch("${mainModel.setting.courierWebsite}"); launch("${mainModel.setting!.courierWebsite}");
} }
_gotoAssign() async { _gotoAssign() async {
bool assigned = await Navigator.push<bool>( bool? assigned = await Navigator.push<bool>(
context, context,
CupertinoPageRoute( CupertinoPageRoute(
builder: (context) => ShipmentAssign( builder: (context) => ShipmentAssign(
@@ -412,7 +424,7 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
)), )),
); );
if (assigned ?? false) { if (assigned ?? false) {
await _loadShipment(_shipment.id); await _loadShipment(_shipment!.id!);
} }
} }
@@ -429,7 +441,7 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
try { try {
ShipmentModel shipmentModel = ShipmentModel shipmentModel =
Provider.of<ShipmentModel>(context, listen: false); Provider.of<ShipmentModel>(context, listen: false);
await shipmentModel.completeAssignShipment(_shipment); await shipmentModel.completeAssignShipment(_shipment!);
Navigator.pop(context, true); Navigator.pop(context, true);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());
@@ -453,7 +465,7 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
try { try {
ShipmentModel shipmentModel = ShipmentModel shipmentModel =
Provider.of<ShipmentModel>(context, listen: false); Provider.of<ShipmentModel>(context, listen: false);
await shipmentModel.completePickupShipment(_shipment); await shipmentModel.completePickupShipment(_shipment!);
Navigator.pop(context, true); Navigator.pop(context, true);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());
@@ -465,16 +477,16 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
} }
_gotoPack() async { _gotoPack() async {
bool assigned = await Navigator.push<bool>( bool? assigned = await Navigator.push<bool>(
context, context,
CupertinoPageRoute( CupertinoPageRoute(
builder: (context) => ShipmentPack( builder: (context) => ShipmentPack(
shipment: _shipment, shipment: _shipment!,
)), )),
); );
if (assigned ?? false) { if (assigned ?? false) {
await _loadShipment(_shipment.id); await _loadShipment(_shipment!.id!);
await _loadCartons(_shipment.id); await _loadCartons(_shipment!.id!);
} }
} }
@@ -491,7 +503,7 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
try { try {
ShipmentModel shipmentModel = ShipmentModel shipmentModel =
Provider.of<ShipmentModel>(context, listen: false); Provider.of<ShipmentModel>(context, listen: false);
await shipmentModel.completePackShipment(_shipment); await shipmentModel.completePackShipment(_shipment!);
Navigator.pop(context, true); Navigator.pop(context, true);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());
@@ -503,7 +515,7 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
} }
_gotoConfirm() async { _gotoConfirm() async {
bool assigned = await Navigator.push<bool>( bool? assigned = await Navigator.push<bool>(
context, context,
CupertinoPageRoute( CupertinoPageRoute(
builder: (context) => ShipmentConfirm( builder: (context) => ShipmentConfirm(
@@ -511,7 +523,7 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
)), )),
); );
if (assigned ?? false) { if (assigned ?? false) {
await _loadShipment(_shipment.id); await _loadShipment(_shipment!.id!);
} }
} }
@@ -528,7 +540,7 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
try { try {
ShipmentModel shipmentModel = ShipmentModel shipmentModel =
Provider.of<ShipmentModel>(context, listen: false); Provider.of<ShipmentModel>(context, listen: false);
await shipmentModel.completeConfirmShipment(_shipment); await shipmentModel.completeConfirmShipment(_shipment!);
Navigator.pop(context, true); Navigator.pop(context, true);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());
@@ -552,7 +564,7 @@ class _ShipmentInfoState extends State<ShipmentInfo> {
try { try {
ShipmentModel shipmentModel = ShipmentModel shipmentModel =
Provider.of<ShipmentModel>(context, listen: false); Provider.of<ShipmentModel>(context, listen: false);
await shipmentModel.completeReceiveShipment(_shipment); await shipmentModel.completeReceiveShipment(_shipment!);
Navigator.pop(context, true); Navigator.pop(context, true);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());

View File

@@ -17,7 +17,7 @@ import 'shipment_list_row.dart';
class ShipmentList extends StatefulWidget { class ShipmentList extends StatefulWidget {
final bool forCustomer; final bool forCustomer;
const ShipmentList({Key key, this.forCustomer = true}) : super(key: key); const ShipmentList({Key? key, this.forCustomer = true}) : super(key: key);
@override @override
_ShipmentListState createState() => _ShipmentListState(); _ShipmentListState createState() => _ShipmentListState();
} }

View File

@@ -2,15 +2,15 @@ import 'package:fcs/domain/entities/shipment.dart';
import 'package:fcs/helpers/theme.dart'; import 'package:fcs/helpers/theme.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_icons/flutter_icons.dart'; import 'package:flutter_icons_null_safety/flutter_icons_null_safety.dart';
import '../main/util.dart'; import '../main/util.dart';
import 'shipment_info.dart'; import 'shipment_info.dart';
class ShipmentListRow extends StatelessWidget { class ShipmentListRow extends StatelessWidget {
final Shipment shipment; final Shipment? shipment;
final bool isCustomer; final bool? isCustomer;
const ShipmentListRow({Key key, this.shipment, this.isCustomer}) const ShipmentListRow({Key? key, this.shipment, this.isCustomer})
: super(key: key); : super(key: key);
@override @override
@@ -45,7 +45,7 @@ class ShipmentListRow extends StatelessWidget {
Padding( Padding(
padding: const EdgeInsets.only(left: 8.0), padding: const EdgeInsets.only(left: 8.0),
child: new Text( child: new Text(
shipment.shipmentNumber ?? "", shipment!.shipmentNumber ?? "",
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.black), fontSize: 15.0, color: Colors.black),
), ),
@@ -69,7 +69,7 @@ class ShipmentListRow extends StatelessWidget {
children: <Widget>[ children: <Widget>[
Padding( Padding(
padding: const EdgeInsets.all(0), padding: const EdgeInsets.all(0),
child: getStatus(shipment.status ?? ""), child: getStatus(shipment!.status ?? ""),
), ),
// Padding( // Padding(
// padding: const EdgeInsets.only(left: 8.0, top: 5, bottom: 5), // padding: const EdgeInsets.only(left: 8.0, top: 5, bottom: 5),

View File

@@ -11,14 +11,14 @@ import 'package:fcs/pages/widgets/local_title.dart';
import 'package:fcs/pages/widgets/progress.dart'; import 'package:fcs/pages/widgets/progress.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_icons/flutter_icons.dart'; import 'package:flutter_icons_null_safety/flutter_icons_null_safety.dart';
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
import 'widgets.dart'; import 'widgets.dart';
class ShipmentPack extends StatefulWidget { class ShipmentPack extends StatefulWidget {
final Shipment shipment; final Shipment? shipment;
ShipmentPack({this.shipment}); ShipmentPack({this.shipment});
@override @override
@@ -29,12 +29,12 @@ class _ShipmentPackState extends State<ShipmentPack> {
var dateFormatter = new DateFormat('dd MMM yyyy'); var dateFormatter = new DateFormat('dd MMM yyyy');
var timeFormatter = new DateFormat('jm'); var timeFormatter = new DateFormat('jm');
Shipment _shipment; Shipment? _shipment;
bool _isLoading = false; bool _isLoading = false;
var now = new DateTime.now(); var now = new DateTime.now();
FcsShipment _fcsShipment; FcsShipment? _fcsShipment;
List<FcsShipment> _fcsShipments; List<FcsShipment>? _fcsShipments;
@override @override
void initState() { void initState() {
@@ -48,8 +48,8 @@ class _ShipmentPackState extends State<ShipmentPack> {
FcsShipmentModel fcsShipmentModel = FcsShipmentModel fcsShipmentModel =
Provider.of<FcsShipmentModel>(context, listen: false); Provider.of<FcsShipmentModel>(context, listen: false);
var fcsShipments = await fcsShipmentModel.getActiveFcsShipments(); var fcsShipments = await fcsShipmentModel.getActiveFcsShipments();
var fcsShipment = fcsShipments var fcsShipment =
.firstWhere((e) => e.id == _shipment.fcsShipmentID, orElse: () => null); fcsShipments.firstWhere((e) => e.id == _shipment?.fcsShipmentID);
setState(() { setState(() {
_fcsShipments = fcsShipments; _fcsShipments = fcsShipments;
_fcsShipment = fcsShipment; _fcsShipment = fcsShipment;
@@ -58,7 +58,7 @@ class _ShipmentPackState extends State<ShipmentPack> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final shipmentNumberBox = getShipmentNumberStatus(context, _shipment); final shipmentNumberBox = getShipmentNumberStatus(context, _shipment!);
var fcsShipmentsBox = LocalDropdown<FcsShipment>( var fcsShipmentsBox = LocalDropdown<FcsShipment>(
callback: (v) { callback: (v) {
@@ -120,14 +120,14 @@ class _ShipmentPackState extends State<ShipmentPack> {
} }
_save() async { _save() async {
_shipment.fcsShipmentID = this._fcsShipment.id; _shipment!.fcsShipmentID = this._fcsShipment!.id;
setState(() { setState(() {
_isLoading = true; _isLoading = true;
}); });
try { try {
ShipmentModel shipmentModel = ShipmentModel shipmentModel =
Provider.of<ShipmentModel>(context, listen: false); Provider.of<ShipmentModel>(context, listen: false);
await shipmentModel.packShipment(_shipment); await shipmentModel.packShipment(_shipment!);
Navigator.pop(context, true); Navigator.pop(context, true);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());

View File

@@ -11,8 +11,8 @@ import 'package:logging/logging.dart';
class StaffModel extends BaseModel { class StaffModel extends BaseModel {
final log = Logger('StaffModel'); final log = Logger('StaffModel');
StreamSubscription<QuerySnapshot> listener; StreamSubscription<QuerySnapshot>? listener;
StreamSubscription<QuerySnapshot> privilegeListener; StreamSubscription<QuerySnapshot>? privilegeListener;
List<User> employees = []; List<User> employees = [];
List<Privilege> privileges = []; List<Privilege> privileges = [];
@@ -26,28 +26,28 @@ class StaffModel extends BaseModel {
@override @override
logout() async { logout() async {
if (listener != null) listener.cancel(); if (listener != null) listener!.cancel();
if (privilegeListener != null) privilegeListener.cancel(); if (privilegeListener != null) privilegeListener!.cancel();
employees = []; employees = [];
privileges = []; privileges = [];
} }
Future<void> _loadEmployees() async { Future<void> _loadEmployees() async {
if (user == null || !user.hasStaffs()) return; if (user == null || !user!.hasStaffs()) throw "No Privilege";
try { try {
if (listener != null) listener.cancel(); if (listener != null) listener!.cancel();
listener = Firestore.instance listener = FirebaseFirestore.instance
.collection("/$user_collection") .collection("/$user_collection")
.where("is_employee", isEqualTo: true) .where("is_employee", isEqualTo: true)
.where("is_sys_admin", isEqualTo: false) .where("is_sys_admin", isEqualTo: false)
.snapshots() .snapshots()
.listen((QuerySnapshot snapshot) { .listen((QuerySnapshot snapshot) {
employees.clear(); employees.clear();
employees = snapshot.documents.map((documentSnapshot) { employees = snapshot.docs.map((documentSnapshot) {
var user = var user =
User.fromMap(documentSnapshot.data, documentSnapshot.documentID); User.fromMap(documentSnapshot.data as Map<String,dynamic>, documentSnapshot.id);
return user; return user;
}).toList(); }).toList();
notifyListeners(); notifyListeners();
@@ -59,14 +59,14 @@ class StaffModel extends BaseModel {
Future<void> _loadPrivileges() async { Future<void> _loadPrivileges() async {
try { try {
privilegeListener = Firestore.instance privilegeListener = FirebaseFirestore.instance
.collection("/$privilege_collection") .collection("/$privilege_collection")
.snapshots() .snapshots()
.listen((QuerySnapshot snapshot) { .listen((QuerySnapshot snapshot) {
privileges.clear(); privileges.clear();
privileges = snapshot.documents.map((documentSnapshot) { privileges = snapshot.docs.map((documentSnapshot) {
var privilege = Privilege.fromMap( var privilege = Privilege.fromMap(
documentSnapshot.data, documentSnapshot.documentID); documentSnapshot.data as Map<String,dynamic>, documentSnapshot.id);
return privilege; return privilege;
}).toList(); }).toList();
notifyListeners(); notifyListeners();
@@ -82,27 +82,27 @@ class StaffModel extends BaseModel {
token: await getToken()); token: await getToken());
} }
Future<User> findUser(String phoneNumber) { Future<User?> findUser(String phoneNumber) {
return Services.instance.userService.findUser(phoneNumber); return Services.instance.userService.findUser(phoneNumber);
} }
Future<List<User>> getPickupEmployees() async { Future<List<User>> getPickupEmployees() async {
if (user == null || !user.hasShipment()) return []; if (user == null || !user!.hasShipment()) return [];
return _getUsers(privilege_shipment); return _getUsers(privilege_shipment);
} }
Future<List<User>> _getUsers(String privilege) async { Future<List<User>> _getUsers(String privilege) async {
List<User> users = []; List<User> users = [];
try { try {
var snaps = await Firestore.instance var snaps = await FirebaseFirestore.instance
.collection("/$user_collection") .collection("/$user_collection")
.where("is_employee", isEqualTo: true) .where("is_employee", isEqualTo: true)
.where("is_sys_admin", isEqualTo: false) .where("is_sys_admin", isEqualTo: false)
.where("privileges", arrayContains: privilege) .where("privileges", arrayContains: privilege)
.getDocuments(source: Source.server); .get(const GetOptions(source: Source.server));
users = snaps.documents.map((documentSnapshot) { users = snaps.docs.map((documentSnapshot) {
var user = var user =
User.fromMap(documentSnapshot.data, documentSnapshot.documentID); User.fromMap(documentSnapshot.data as Map<String,dynamic>, documentSnapshot.id);
return user; return user;
}).toList(); }).toList();
} catch (e) { } catch (e) {

View File

@@ -15,7 +15,7 @@ import 'package:provider/provider.dart';
typedef void FindCallBack(); typedef void FindCallBack();
class StaffEditor extends StatefulWidget { class StaffEditor extends StatefulWidget {
final User staff; final User? staff;
const StaffEditor({this.staff}); const StaffEditor({this.staff});
@override @override
_StaffEditorState createState() => _StaffEditorState(); _StaffEditorState createState() => _StaffEditorState();
@@ -25,8 +25,8 @@ class _StaffEditorState extends State<StaffEditor> {
TextEditingController _phoneInput = new TextEditingController(); TextEditingController _phoneInput = new TextEditingController();
bool _isLoading = false; bool _isLoading = false;
User user; late User user;
User selectedUser; User? selectedUser;
List<Privilege> privileges = []; List<Privilege> privileges = [];
bool isNew = true; bool isNew = true;
@@ -38,8 +38,8 @@ class _StaffEditorState extends State<StaffEditor> {
user = User(); user = User();
if (!isNew) { if (!isNew) {
user = user =
User(name: widget.staff.name, phoneNumber: widget.staff.phoneNumber); User(name: widget.staff!.name, phoneNumber: widget.staff!.phoneNumber);
user.privileges = widget.staff.privileges; user.privileges = widget.staff!.privileges;
privileges.forEach((p) => user.privileges.contains(p.id) privileges.forEach((p) => user.privileges.contains(p.id)
? p.isChecked = true ? p.isChecked = true
: p.isChecked = false); : p.isChecked = false);
@@ -56,7 +56,7 @@ class _StaffEditorState extends State<StaffEditor> {
title: InkWell( title: InkWell(
onTap: () { onTap: () {
setState(() { setState(() {
p.isChecked = p.isChecked == null ? true : !p.isChecked; p.isChecked = p.isChecked == null ? true : !p.isChecked!;
}); });
}, },
child: new Row( child: new Row(
@@ -64,7 +64,8 @@ class _StaffEditorState extends State<StaffEditor> {
new Checkbox( new Checkbox(
value: p.isChecked == null ? false : p.isChecked, value: p.isChecked == null ? false : p.isChecked,
activeColor: primaryColor, activeColor: primaryColor,
onChanged: (bool value) { onChanged: (bool? value) {
if(value != null)
setState(() { setState(() {
p.isChecked = value; p.isChecked = value;
}); });
@@ -78,10 +79,10 @@ class _StaffEditorState extends State<StaffEditor> {
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: <Widget>[ children: <Widget>[
new Text( new Text(
p.name, p.name ??"",
style: TextStyle(fontSize: 15.0, color: primaryColor), style: TextStyle(fontSize: 15.0, color: primaryColor),
), ),
Text(p.desc, Text(p.desc ?? "",
style: TextStyle(fontSize: 13, color: Colors.grey[600])) style: TextStyle(fontSize: 13, color: Colors.grey[600]))
], ],
), ),
@@ -110,7 +111,7 @@ class _StaffEditorState extends State<StaffEditor> {
style: textStyle, style: textStyle,
decoration: new InputDecoration( decoration: new InputDecoration(
labelText: labelText:
AppTranslations.of(context).text('staff.phone.search'), AppTranslations.of(context)!.text('staff.phone.search'),
labelStyle: languageModel.isEng ? labelStyle : labelStyleMM, labelStyle: languageModel.isEng ? labelStyle : labelStyleMM,
// icon: Icon( // icon: Icon(
// Icons.search, // Icons.search,
@@ -150,7 +151,7 @@ class _StaffEditorState extends State<StaffEditor> {
? Container() ? Container()
: IconButton( : IconButton(
icon: Icon(Icons.open_in_new, color: primaryColor), icon: Icon(Icons.open_in_new, color: primaryColor),
onPressed: () => call(context, user.phoneNumber)), onPressed: () => call(context, user.phoneNumber!)),
], ],
); );
@@ -226,7 +227,7 @@ class _StaffEditorState extends State<StaffEditor> {
}); });
StaffModel staffModel = Provider.of<StaffModel>(context, listen: false); StaffModel staffModel = Provider.of<StaffModel>(context, listen: false);
try { try {
await staffModel.updatePrivileges(this.selectedUser.id, privilegesIDs()); await staffModel.updatePrivileges(this.selectedUser!.id!, privilegesIDs());
Navigator.pop(context); Navigator.pop(context);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());
@@ -238,7 +239,7 @@ class _StaffEditorState extends State<StaffEditor> {
} }
List<String> privilegesIDs() { List<String> privilegesIDs() {
return this.privileges.where((p) => p.isChecked).map((p) => p.id).toList(); return this.privileges.where((p) => p.isChecked!).map((p) => p.id).toList();
} }
_save() async { _save() async {
@@ -248,7 +249,7 @@ class _StaffEditorState extends State<StaffEditor> {
if (widget.staff == null) return; if (widget.staff == null) return;
StaffModel staffModel = Provider.of<StaffModel>(context, listen: false); StaffModel staffModel = Provider.of<StaffModel>(context, listen: false);
try { try {
await staffModel.updatePrivileges(widget.staff.id, privilegesIDs()); await staffModel.updatePrivileges(widget.staff!.id!, privilegesIDs());
Navigator.pop(context); Navigator.pop(context);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());
@@ -267,7 +268,7 @@ class _StaffEditorState extends State<StaffEditor> {
_isLoading = true; _isLoading = true;
}); });
try { try {
User _user = await staffModel.findUser(_phoneInput.text); User? _user = await staffModel.findUser(_phoneInput.text);
if (_user == null) { if (_user == null) {
showMsgDialog(context, "Error", _phoneInput.text + " not found!"); showMsgDialog(context, "Error", _phoneInput.text + " not found!");
return; return;

View File

@@ -5,7 +5,6 @@ import 'package:fcs/pages/widgets/local_text.dart';
import 'package:fcs/pages/widgets/progress.dart'; import 'package:fcs/pages/widgets/progress.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_icons/flutter_icons.dart';
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
@@ -84,7 +83,7 @@ class _StaffListState extends State<StaffList> {
padding: new EdgeInsets.symmetric( padding: new EdgeInsets.symmetric(
horizontal: 32.0 - dotSize / 2), horizontal: 32.0 - dotSize / 2),
child: Icon( child: Icon(
MaterialCommunityIcons.worker, Icons.person,
color: primaryColor, color: primaryColor,
size: 40, size: 40,
), ),
@@ -94,13 +93,13 @@ class _StaffListState extends State<StaffList> {
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: <Widget>[ children: <Widget>[
new Text( new Text(
user.name, user.name ?? "",
style: new TextStyle(fontSize: 15.0), style: new TextStyle(fontSize: 15.0),
), ),
Padding( Padding(
padding: const EdgeInsets.only(top: 8.0), padding: const EdgeInsets.only(top: 8.0),
child: new Text( child: new Text(
user.phoneNumber, user.phoneNumber ?? "",
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.grey), fontSize: 15.0, color: Colors.grey),
), ),

View File

@@ -17,7 +17,7 @@ Future<User?> searchUser(BuildContext context,
); );
class UserSearchDelegate extends SearchDelegate<User> { class UserSearchDelegate extends SearchDelegate<User> {
final OnUserSelect onUserSelect; final OnUserSelect? onUserSelect;
final bool popPage; final bool popPage;
UserSearchDelegate({required this.onUserSelect, required this.popPage}); UserSearchDelegate({required this.onUserSelect, required this.popPage});
@@ -119,7 +119,7 @@ class UserSearchDelegate extends SearchDelegate<User> {
_onUserRowSelect(BuildContext context, User user) { _onUserRowSelect(BuildContext context, User user) {
if (onUserSelect != null) { if (onUserSelect != null) {
onUserSelect(user); onUserSelect!(user);
} }
if (popPage) { if (popPage) {
Navigator.pop(context); Navigator.pop(context);

View File

@@ -78,7 +78,7 @@ class _LocalPopupMenuButtonState extends State<LocalPopupMenuButton> {
children: <Widget>[ children: <Widget>[
Icon( Icon(
widget.buttonIcon ?? Icons.filter_list, widget.buttonIcon ?? Icons.filter_list,
color: widget.buttonColor ?? primaryColor, color: widget.buttonColor,
), ),
hightlight hightlight
? Positioned( ? Positioned(
@@ -98,14 +98,12 @@ class _LocalPopupMenuButtonState extends State<LocalPopupMenuButton> {
)), )),
itemBuilder: (BuildContext context) { itemBuilder: (BuildContext context) {
return popmenus.map((LocalPopupMenu choice) { return popmenus.map((LocalPopupMenu choice) {
if (choice == null) return null;
return PopupMenuItem<LocalPopupMenu>( return PopupMenuItem<LocalPopupMenu>(
value: choice, value: choice,
child: Row( child: Row(
children: <Widget>[ children: <Widget>[
LocalText(context, choice.textKey, LocalText(context, choice.textKey ?? "",
color: color: choice.enabled ? primaryColor : Colors.grey),
choice?.enabled ?? true ? primaryColor : Colors.grey),
SizedBox( SizedBox(
width: 10, width: 10,
), ),
@@ -124,8 +122,8 @@ class _LocalPopupMenuButtonState extends State<LocalPopupMenuButton> {
bool _needHighlight() { bool _needHighlight() {
popmenus.forEach((e) { popmenus.forEach((e) {
if (e == null) return false; if (e == null) return;
if (e.selected && e.highlight) return true; if (e.selected && e.highlight) return;
}); });
return false; return false;
} }

View File

@@ -80,7 +80,7 @@ class MultiImgController {
List<File?> get getUpdatedFile { List<File?> get getUpdatedFile {
List<File?> _addfiles = getAddedFile; List<File?> _addfiles = getAddedFile;
this.imageFiles.addAll(_addfiles); this.imageFiles!.addAll(_addfiles);
return this.imageFiles; return this.imageFiles;
} }

View File

@@ -112,21 +112,21 @@ packages:
name: cloud_firestore name: cloud_firestore
url: "https://pub.dartlang.org" url: "https://pub.dartlang.org"
source: hosted source: hosted
version: "2.5.0" version: "2.5.2"
cloud_firestore_platform_interface: cloud_firestore_platform_interface:
dependency: transitive dependency: transitive
description: description:
name: cloud_firestore_platform_interface name: cloud_firestore_platform_interface
url: "https://pub.dartlang.org" url: "https://pub.dartlang.org"
source: hosted source: hosted
version: "5.4.0" version: "5.4.1"
cloud_firestore_web: cloud_firestore_web:
dependency: transitive dependency: transitive
description: description:
name: cloud_firestore_web name: cloud_firestore_web
url: "https://pub.dartlang.org" url: "https://pub.dartlang.org"
source: hosted source: hosted
version: "2.4.0" version: "2.4.2"
collection: collection:
dependency: transitive dependency: transitive
description: description:

View File

@@ -46,7 +46,6 @@ dependencies:
flutter_icons_null_safety: ^1.1.0 flutter_icons_null_safety: ^1.1.0
country_icons: ^2.0.2 country_icons: ^2.0.2
timeline_list: ^0.0.5 timeline_list: ^0.0.5
barcode_scan2: ^4.1.3
barcode_scan2: ^4.1.4 barcode_scan2: ^4.1.4
flutter_pdfview: ^1.2.1 flutter_pdfview: ^1.2.1
flutter_local_notifications: ^8.2.0 flutter_local_notifications: ^8.2.0