add shipment in processing, update package, processing and receiving

This commit is contained in:
tzw
2025-03-06 17:59:15 +06:30
parent 9e6b288970
commit e75eacd1f9
20 changed files with 746 additions and 603 deletions

View File

@@ -231,6 +231,8 @@
"package.delivery.address":"Delivery address", "package.delivery.address":"Delivery address",
"package.popupmenu.active":"Active Packages", "package.popupmenu.active":"Active Packages",
"package.popupmenu.delivered":"Delivered Packages", "package.popupmenu.delivered":"Delivered Packages",
"package.shipment.title":"Shipment",
"package.processing.date":"Processing date",
"Package End ================================================================":"", "Package End ================================================================":"",
"Market Start ================================================================":"", "Market Start ================================================================":"",
@@ -665,10 +667,10 @@
"processing.delete.confirm":"Delete this package?", "processing.delete.confirm":"Delete this package?",
"processing.edit.sub_title":"Processing", "processing.edit.sub_title":"Processing",
"processing.edit.complete.btn":"Complete processing", "processing.edit.complete.btn":"Complete processing",
"processing.new":"New Processing", "processing.new":"New processing",
"processing.create":"New Processing", "processing.create":"New Processing",
"processing.update":"Update Processing", "processing.update":"Update Processing",
"processing.consignee.name":"Consignee", "processing.consignee.name":"Consignee name",
"processing.shipper.name":"Sender name", "processing.shipper.name":"Sender name",
"processing.package.select.btn":"Select", "processing.package.select.btn":"Select",
"processing.package.create":"New Package", "processing.package.create":"New Package",

View File

@@ -232,6 +232,8 @@
"package.delivery.address":"ပို့ဆောင်ရမည့်လိပ်စာ", "package.delivery.address":"ပို့ဆောင်ရမည့်လိပ်စာ",
"package.popupmenu.active":"လာမည့် အထုပ်များ", "package.popupmenu.active":"လာမည့် အထုပ်များ",
"package.popupmenu.delivered":"ပို့ပြီးသော အထုပ်များ", "package.popupmenu.delivered":"ပို့ပြီးသော အထုပ်များ",
"package.shipment.title":"တင်ပို့ခြင်း",
"package.processing.date":"လုပ်ဆောင်သည့်ရက်",
"Package End ================================================================":"", "Package End ================================================================":"",
"Market Start ================================================================":"", "Market Start ================================================================":"",
@@ -672,7 +674,7 @@
"processing.new":"လုပ်ဆောင်ခြင်း အသစ်", "processing.new":"လုပ်ဆောင်ခြင်း အသစ်",
"processing.create":"လုပ်ဆောင်ခြင်း အသစ်", "processing.create":"လုပ်ဆောင်ခြင်း အသစ်",
"processing.update":"လုပ်ဆောင်ခြင်း ပြင်ဆင်ခြင်း", "processing.update":"လုပ်ဆောင်ခြင်း ပြင်ဆင်ခြင်း",
"processing.consignee.name":"လက်ခံသူ", "processing.consignee.name":"လက်ခံသူ အမည်",
"processing.shipper.name":"တင်ပို့သူ အမည်", "processing.shipper.name":"တင်ပို့သူ အမည်",
"processing.package.select.btn":"ရွေးချယ်မည်", "processing.package.select.btn":"ရွေးချယ်မည်",
"processing.package.create":"အထုပ် အသစ်", "processing.package.create":"အထုပ် အသစ်",

View File

@@ -18,6 +18,7 @@ class FcsShipment {
String? reportName; String? reportName;
int packageCount; int packageCount;
int cartonCount; int cartonCount;
DateTime? processingDate;
FcsShipment( FcsShipment(
{this.id, {this.id,
@@ -36,7 +37,8 @@ class FcsShipment {
this.destinationPortName, this.destinationPortName,
this.reportName, this.reportName,
this.packageCount = 0, this.packageCount = 0,
this.cartonCount = 0}); this.cartonCount = 0,
this.processingDate});
factory FcsShipment.fromMap(Map<String, dynamic> map, String docID) { factory FcsShipment.fromMap(Map<String, dynamic> map, String docID) {
var cutoffDate = var cutoffDate =
@@ -44,6 +46,10 @@ class FcsShipment {
var arrivalDate = var arrivalDate =
map['eta_date'] == null ? null : (map['eta_date'] as Timestamp); map['eta_date'] == null ? null : (map['eta_date'] as Timestamp);
var processingDate = map['processing_date'] == null
? null
: (map['processing_date'] as Timestamp);
return FcsShipment( return FcsShipment(
id: docID, id: docID,
cutoffDate: cutoffDate?.toDate(), cutoffDate: cutoffDate?.toDate(),
@@ -59,7 +65,8 @@ class FcsShipment {
destinationPortId: map['destination_port_id'], destinationPortId: map['destination_port_id'],
destinationPortName: map['destination_port_name'], destinationPortName: map['destination_port_name'],
packageCount: map['package_count'] ?? 0, packageCount: map['package_count'] ?? 0,
cartonCount: map['carton_count'] ?? 0); cartonCount: map['carton_count'] ?? 0,
processingDate: processingDate?.toDate());
} }
Map<String, dynamic> toMap() { Map<String, dynamic> toMap() {

View File

@@ -7,21 +7,25 @@ import 'package:fcs/domain/vo/shipment_status.dart';
class Package { class Package {
String? id; String? id;
String? trackingID; String? trackingID;
String? userID;
String? fcsID;
String? userName;
String? phoneNumber;
DateTime? currentStatusDate; DateTime? currentStatusDate;
List<String> photoUrls; List<String> photoUrls;
List<ShipmentStatus> shipmentHistory; List<ShipmentStatus> shipmentHistory;
List<String> cartonIds; List<String> cartonIds;
String? desc; String? desc;
String? status; String? status;
String? shipmentId;
String? shipmentNumber; String? shipmentNumber;
String? userID;
String? fcsID;
String? userName;
String? phoneNumber;
String? senderFCSID; String? senderFCSID;
String? senderName; String? senderName;
String? senderPhoneNumber; String? senderPhoneNumber;
String? boxNumber; String? boxNumber;
String? cargoDesc; String? cargoDesc;
String? market; String? market;
@@ -50,6 +54,7 @@ class Package {
this.userName, this.userName,
this.fcsID, this.fcsID,
this.phoneNumber, this.phoneNumber,
this.shipmentId,
this.shipmentNumber, this.shipmentNumber,
this.senderFCSID, this.senderFCSID,
this.senderName, this.senderName,
@@ -77,7 +82,9 @@ class Package {
var currentStatusDate = var currentStatusDate =
map['status_date'] != null ? (map['status_date'] as Timestamp) : null; map['status_date'] != null ? (map['status_date'] as Timestamp) : null;
List<ShipmentStatus> shipmentStatus = List.from(map['all_status']) List<ShipmentStatus> shipmentStatus = map['all_status'] == null
? []
: List.from(map['all_status'])
.map((e) => ShipmentStatus.fromMap(Map<String, dynamic>.from(e))) .map((e) => ShipmentStatus.fromMap(Map<String, dynamic>.from(e)))
.toList(); .toList();
List<String> photoUrls = List<String> photoUrls =
@@ -96,14 +103,16 @@ class Package {
fcsID: map['fcs_id'], fcsID: map['fcs_id'],
trackingID: map['tracking_id'], trackingID: map['tracking_id'],
market: map['market'], market: map['market'],
userName: map['user_name'], userName: map['user_name'] ?? "",
phoneNumber: map['phone_number'], phoneNumber: map['phone_number'] ?? "",
remark: map['remark'], remark: map['remark'],
desc: map['desc'] ?? "", desc: map['desc'],
status: map['status'], status: map['status'],
senderFCSID: map['sender_fcs_id'], senderFCSID: map['sender_fcs_id'],
senderName: map['sender_name'], senderName: map['sender_name'] ?? "",
senderPhoneNumber: map['sender_phone_number'], senderPhoneNumber: map['sender_phone_number'] ?? "",
shipmentId: map['shipment_id'],
shipmentNumber: map['shipment_number'],
deliveryAddress: da, deliveryAddress: da,
currentStatusDate: currentStatusDate?.toDate().toLocal(), currentStatusDate: currentStatusDate?.toDate().toLocal(),
photoUrls: photoUrls, photoUrls: photoUrls,
@@ -136,19 +145,20 @@ class Package {
currentStatusDate: DateTime.parse(json['status_date'])); currentStatusDate: DateTime.parse(json['status_date']));
} }
bool isChangedForEdit(Package package) { bool isChangedForEditReceiving(Package package) {
return package.trackingID != this.trackingID || return package.trackingID != trackingID ||
package.remark != this.remark || package.remark != remark ||
package.fcsID != this.fcsID; package.fcsID != fcsID;
} }
bool isChangedForEditProcessing(Package package) { bool isChangedForEditProcessing(Package package) {
return package.fcsID != this.fcsID || return package.fcsID != fcsID ||
package.senderFCSID != this.senderFCSID || package.senderFCSID != senderFCSID ||
package.market != this.market || package.market != market ||
package.desc != this.desc || package.desc != desc ||
package.remark != this.remark || package.remark != remark ||
package.photoUrls != this.photoUrls; package.photoUrls != photoUrls ||
package.shipmentId != shipmentId;
} }
@override @override

View File

@@ -23,12 +23,12 @@ class DeliveryAddress {
factory DeliveryAddress.fromMap(Map<String, dynamic> map, String docID) { factory DeliveryAddress.fromMap(Map<String, dynamic> map, String docID) {
return DeliveryAddress( return DeliveryAddress(
id: docID, id: docID,
fullName: map['full_name'], fullName: map['full_name'] ?? "",
addressLine1: map['address_line1'], addressLine1: map['address_line1'] ?? "",
addressLine2: map['address_line2'], addressLine2: map['address_line2'] ?? "",
city: map['city'], city: map['city'] ?? "",
state: map['state'], state: map['state'] ?? "",
phoneNumber: map['phone_number'], phoneNumber: map['phone_number'] ?? "",
isDefault: map['is_default'] ?? false, isDefault: map['is_default'] ?? false,
); );
} }
@@ -51,11 +51,11 @@ class DeliveryAddress {
} }
bool isChangedForEdit(DeliveryAddress deliveryAddress) { bool isChangedForEdit(DeliveryAddress deliveryAddress) {
return deliveryAddress.fullName != this.fullName || return deliveryAddress.fullName != fullName ||
deliveryAddress.phoneNumber != this.phoneNumber || deliveryAddress.phoneNumber != phoneNumber ||
deliveryAddress.addressLine1 != this.addressLine1 || deliveryAddress.addressLine1 != addressLine1 ||
deliveryAddress.addressLine2 != this.addressLine2 || deliveryAddress.addressLine2 != addressLine2 ||
deliveryAddress.state != this.state || deliveryAddress.state != state ||
deliveryAddress.city != this.city; deliveryAddress.city != city;
} }
} }

View File

@@ -67,12 +67,16 @@ Future<dynamic> requestAPI(String path, method,
path, path,
data: payload, data: payload,
); );
if (response.statusCode == HttpStatus.ok) {
var data = Status.fromJson(response.data); var data = Status.fromJson(response.data);
if (data.status == 'Ok') { if (data.status == 'Ok') {
return response.data["data"]; return response.data["data"];
} else { } else {
throw Exception(data.message); throw Exception(data.message);
} }
}else{
throw Exception( "${response.statusCode} - ${response.statusMessage}");
}
} catch (e) { } catch (e) {
logger.warning("path:$path, api:$e"); logger.warning("path:$path, api:$e");
throw e; throw e;

View File

@@ -10,6 +10,7 @@ const labelColor = const Color(0xFF757575);
var dividerColor = Colors.grey.shade400; var dividerColor = Colors.grey.shade400;
const dangerColor = const Color(0xffff0606); const dangerColor = const Color(0xffff0606);
const hintTextColor = Color.fromARGB(255, 187, 187, 187); const hintTextColor = Color.fromARGB(255, 187, 187, 187);
const linkColor= Color(0xff0000EE);
const TextStyle labelStyle = const TextStyle labelStyle =
TextStyle(fontSize: 20, color: labelColor, fontWeight: FontWeight.w500); TextStyle(fontSize: 20, color: labelColor, fontWeight: FontWeight.w500);

View File

@@ -1,5 +1,4 @@
import 'package:fcs/helpers/theme.dart'; import 'package:fcs/helpers/theme.dart';
import 'package:fcs/pages/package/package_info.dart';
import 'package:fcs/pages/main/util.dart'; import 'package:fcs/pages/main/util.dart';
import 'package:fcs/pages/widgets/fcs_id_icon.dart'; import 'package:fcs/pages/widgets/fcs_id_icon.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
@@ -7,6 +6,7 @@ import 'package:intl/intl.dart';
DateFormat dayFormat = DateFormat("MMM dd yyyy"); DateFormat dayFormat = DateFormat("MMM dd yyyy");
DateFormat timeFormat = DateFormat("HH:mm"); DateFormat timeFormat = DateFormat("HH:mm");
final DateFormat dateFormat = DateFormat("d MMM yyyy");
typedef CallbackOnViewDetail(); typedef CallbackOnViewDetail();

View File

@@ -1,3 +1,6 @@
// ignore_for_file: deprecated_member_use
import 'package:fcs/domain/entities/user.dart';
import 'package:fcs/helpers/theme.dart'; import 'package:fcs/helpers/theme.dart';
import 'package:fcs/localization/app_translations.dart'; import 'package:fcs/localization/app_translations.dart';
import 'package:fcs/pages/main/model/language_model.dart'; import 'package:fcs/pages/main/model/language_model.dart';
@@ -16,11 +19,11 @@ Future showMsgDialog(BuildContext context, String title, String msg) {
context: context, context: context,
builder: (_) { builder: (_) {
return AlertDialog( return AlertDialog(
title: new Text(title), title: Text(title),
content: new Text(msg), content: Text(msg),
actions: <Widget>[ actions: <Widget>[
new TextButton( TextButton(
child: new Text( child: Text(
"Close", "Close",
style: TextStyle(color: primaryColor), style: TextStyle(color: primaryColor),
), ),
@@ -208,7 +211,7 @@ Widget getStatus(String status) {
call(BuildContext context, String phone) { call(BuildContext context, String phone) {
showConfirmDialog(context, "contact.phone.confim", () => launch("tel:$phone"), showConfirmDialog(context, "contact.phone.confim", () => launch("tel:$phone"),
translationVariables: ["$phone"]); translationVariables: [phone]);
} }
Widget nameWidget(String name) { Widget nameWidget(String name) {
@@ -297,7 +300,7 @@ Widget fcsDropDown(String label, IconData iconData,
child: Icon(iconData), child: Icon(iconData),
), ),
Expanded( Expanded(
child: Container( child: SizedBox(
height: 50.0, height: 50.0,
child: Row(children: <Widget>[ child: Row(children: <Widget>[
Expanded(child: _dropDown()), Expanded(child: _dropDown()),
@@ -393,7 +396,7 @@ bool hasUnicode(String text) {
final int maxBits = 128; final int maxBits = 128;
List<int> unicodeSymbols = List<int> unicodeSymbols =
text.codeUnits.where((ch) => ch > maxBits).toList(); text.codeUnits.where((ch) => ch > maxBits).toList();
return unicodeSymbols.length > 0; return unicodeSymbols.isNotEmpty;
} }
String removeTrailingZeros(double number) { String removeTrailingZeros(double number) {
@@ -416,3 +419,94 @@ String capitalizeFirstLetter(String text) {
if (text.isEmpty) return text; if (text.isEmpty) return text;
return text[0].toUpperCase() + text.substring(1); return text[0].toUpperCase() + text.substring(1);
} }
Widget userSearchBox(BuildContext context,
{required String lableKey,
IconData? icon,
User? user,
Function()? onSearch,
MainAxisAlignment rowMainAxisAlignment = MainAxisAlignment.start}) {
return Column(
children: [
Row(
mainAxisAlignment: rowMainAxisAlignment,
children: <Widget>[
Flexible(
child: Padding(
padding: const EdgeInsets.only(left: 2),
child:
LocalText(context, lableKey, color: labelColor, fontSize: 15),
)),
IconButton(
icon: Icon(Icons.search, color: Colors.black),
onPressed: onSearch),
],
),
Row(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Padding(
padding: const EdgeInsets.only(
right: 15,
),
child: user != null && user.fcsID != ""
? Icon(icon, color: primaryColor)
: const SizedBox(),
),
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(user?.name ?? ''),
Text(user?.fcsID ?? '',
style: TextStyle(fontSize: 13, color: labelColor)),
Text(user?.phoneNumber ?? '',
style: TextStyle(fontSize: 13, color: labelColor))
],
))
],
)
],
);
}
Widget userDisplayBox(BuildContext context,
{required String lableKey, IconData? icon, String? name, String? fcsID}) {
return fcsID != null && fcsID != ""
? Padding(
padding: const EdgeInsets.only(top: 8.0, bottom: 8),
child: Row(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Padding(
padding: const EdgeInsets.only(right: 15),
child: Icon(icon, color: primaryColor)),
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
LocalText(context, lableKey,
color: Colors.black54, fontSize: 15),
InkWell(
onTap: null,
child: Text(
name ?? '',
style: TextStyle(
shadows: [
Shadow(color: linkColor, offset: Offset(0, -2))
],
color: Colors.transparent,
decoration: TextDecoration.underline,
decorationColor: linkColor),
),
),
Text(fcsID,
style: TextStyle(fontSize: 13, color: labelColor))
],
),
),
],
),
)
: const SizedBox();
}

View File

@@ -163,34 +163,29 @@ class PackageModel extends BaseModel {
try { try {
var qsnap = await FirebaseFirestore.instance var qsnap = await FirebaseFirestore.instance
.collection("$path") .collection(path)
.where("tracking_id", isEqualTo: trackingID) .where("tracking_id", isEqualTo: trackingID)
.where("has_user_id", isEqualTo: false) .where("has_user_id", isEqualTo: false)
.where("is_deleted", isEqualTo: false) .where("is_deleted", isEqualTo: false)
.get(const GetOptions(source: Source.server)); .get(const GetOptions(source: Source.server));
if (qsnap.docs.length > 0) {
var snap = qsnap.docs[0]; if (qsnap.docs.isNotEmpty) {
if (snap.exists) { var snap = qsnap.docs.first;
var package = var package = Package.fromMap(snap.data(), snap.id);
Package.fromMap(snap.data as Map<String, dynamic>, snap.id);
return package; return package;
} }
}
qsnap = await FirebaseFirestore.instance qsnap = await FirebaseFirestore.instance
.collection("$path") .collection(path)
.where("tracking_id", isEqualTo: trackingID) .where("tracking_id", isEqualTo: trackingID)
.where("user_id", isEqualTo: user!.id) .where("user_id", isEqualTo: user!.id)
.where("is_deleted", isEqualTo: false) .where("is_deleted", isEqualTo: false)
.get(const GetOptions(source: Source.server)); .get(const GetOptions(source: Source.server));
if (qsnap.docs.length > 0) { if (qsnap.docs.isNotEmpty) {
var snap = qsnap.docs[0]; var snap = qsnap.docs.first;
if (snap.exists) { var package = Package.fromMap(snap.data(), snap.id);
var package =
Package.fromMap(snap.data as Map<String, dynamic>, snap.id);
return package; return package;
} }
}
} catch (e) { } catch (e) {
log.warning("Error!! $e"); log.warning("Error!! $e");
} }

View File

@@ -9,7 +9,6 @@ import 'package:fcs/pages/package/model/package_model.dart';
import 'package:fcs/pages/widgets/defalut_delivery_address.dart'; import 'package:fcs/pages/widgets/defalut_delivery_address.dart';
import 'package:fcs/pages/widgets/delivery_address_selection.dart'; import 'package:fcs/pages/widgets/delivery_address_selection.dart';
import 'package:fcs/pages/widgets/display_text.dart'; import 'package:fcs/pages/widgets/display_text.dart';
import 'package:fcs/pages/widgets/fcs_id_icon.dart';
import 'package:fcs/pages/widgets/local_button.dart'; import 'package:fcs/pages/widgets/local_button.dart';
import 'package:fcs/pages/widgets/local_text.dart'; import 'package:fcs/pages/widgets/local_text.dart';
import 'package:fcs/pages/widgets/multi_img_controller.dart'; import 'package:fcs/pages/widgets/multi_img_controller.dart';
@@ -22,15 +21,19 @@ 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';
final DateFormat dateFormat = DateFormat("d MMM yyyy"); import '../../domain/entities/fcs_shipment.dart';
import '../fcs_shipment/model/fcs_shipment_model.dart';
class PackageInfo extends StatefulWidget { class PackageInfo extends StatefulWidget {
final isCustomer; final bool isCustomer;
final isSearchResult; final bool isSearchResult;
final Package? package; final Package package;
PackageInfo( const PackageInfo(
{this.package, this.isSearchResult = false, this.isCustomer = false}); {super.key,
required this.package,
this.isSearchResult = false,
this.isCustomer = false});
@override @override
_PackageInfoState createState() => _PackageInfoState(); _PackageInfoState createState() => _PackageInfoState();
@@ -41,14 +44,16 @@ class _PackageInfoState extends State<PackageInfo> {
Package? _package = Package(); Package? _package = Package();
bool _isLoading = false; bool _isLoading = false;
MultiImgController multiImgController = MultiImgController(); MultiImgController multiImgController = MultiImgController();
FcsShipment? _shipment;
@override @override
void initState() { void initState() {
super.initState(); super.initState();
initPackage(widget.package!); _initPackage(widget.package);
_loadShipment();
} }
initPackage(Package? pkg) async { _initPackage(Package? pkg) async {
setState(() { setState(() {
_isLoading = true; _isLoading = true;
}); });
@@ -64,6 +69,17 @@ class _PackageInfoState extends State<PackageInfo> {
}); });
} }
_loadShipment() async {
if (widget.package.shipmentId == null) return;
var s = await context
.read<FcsShipmentModel>()
.getFcsShipment(widget.package.shipmentId!);
_shipment = s;
if (mounted) {
setState(() {});
}
}
@override @override
void dispose() { void dispose() {
super.dispose(); super.dispose();
@@ -83,41 +99,51 @@ class _PackageInfoState extends State<PackageInfo> {
labelTextKey: "package.tracking.id", labelTextKey: "package.tracking.id",
iconData: MaterialCommunityIcons.barcode_scan, iconData: MaterialCommunityIcons.barcode_scan,
); );
var fcsIDBox = DisplayText(
text: _package?.fcsID ?? "", final consigneeBox = userDisplayBox(context,
labelTextKey: "processing.fcs.id", lableKey: "box.consignee.title",
icon: FcsIDIcon(), icon: MaterialCommunityIcons.account_arrow_left,
); name: _package?.userName ?? "",
final customerNameBox = DisplayText( fcsID: _package?.fcsID ?? "");
text: _package?.userName ?? "",
labelTextKey: "package.create.name", final senderBox = userDisplayBox(context,
iconData: Icons.perm_identity, lableKey: "box.sender.title",
); icon: MaterialCommunityIcons.account_arrow_right,
name: _package?.senderName ?? "",
fcsID: _package?.senderFCSID ?? "");
final marketBox = DisplayText( final marketBox = DisplayText(
text: _package?.market ?? "-", text: _package?.market ?? "-",
labelTextKey: "package.create.market", labelTextKey: "package.create.market",
iconData: Icons.store, iconData: Icons.store,
); );
final descBox = DisplayText( final descBox = DisplayText(
text: _package?.desc ?? "-", text: _package?.desc ?? "-",
labelTextKey: "package.edit.desc", labelTextKey: "package.edit.desc",
iconData: MaterialCommunityIcons.message_text_outline, iconData: MaterialCommunityIcons.message_text_outline,
); );
final remarkBox = DisplayText( final remarkBox = DisplayText(
text: _package?.remark ?? "-", text: _package?.remark ?? "-",
labelTextKey: "package.edit.remark", labelTextKey: "package.edit.remark",
iconData: Entypo.new_message, iconData: Entypo.new_message,
); );
final img = MultiImageFile( final img = MultiImageFile(
enabled: false, enabled: false,
controller: multiImgController, controller: multiImgController,
title: "Receipt File", title: "Receipt File",
); );
final returnButton = LocalButton( final returnButton = Padding(
padding: const EdgeInsets.symmetric(horizontal: 10),
child: LocalButton(
textKey: "receiving.return.btn", textKey: "receiving.return.btn",
callBack: _return, callBack: _return,
),
); );
final deliveryAddressBox = DefaultDeliveryAddress( final deliveryAddressBox = DefaultDeliveryAddress(
deliveryAddress: _package!.deliveryAddress, deliveryAddress: _package!.deliveryAddress,
labelKey: "package.delivery.address", labelKey: "package.delivery.address",
@@ -136,13 +162,72 @@ class _PackageInfoState extends State<PackageInfo> {
: null, : null,
); );
final shipmentBox = Padding(
padding: const EdgeInsets.only(top: 15),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
LocalText(context, "package.shipment.title",
color: primaryColor, fontSize: 17, fontWeight: FontWeight.normal),
Row(
children: [
Flexible(
child: DisplayText(
text: _shipment?.shipmentNumber ?? '',
labelTextKey: "FCSshipment.number",
iconData: Ionicons.ios_airplane,
),
),
Flexible(
child: DisplayText(
text: _shipment != null
? _shipment!.processingDate != null
? dateFormatter.format(_shipment!.processingDate!)
: ""
: "",
labelTextKey: "package.processing.date",
iconData: Icons.date_range,
),
),
],
),
Row(
children: [
Flexible(
child: DisplayText(
text: _shipment != null
? _shipment!.cutoffDate != null
? dateFormatter.format(_shipment!.cutoffDate!)
: ""
: "",
labelTextKey: "FCSshipment.cutoff_date",
iconData: Icons.date_range,
),
),
Flexible(
child: DisplayText(
text: _shipment != null
? _shipment!.etaDate != null
? dateFormatter.format(_shipment!.etaDate!)
: ""
: "",
labelTextKey: "FCSshipment.ETA",
iconData: Icons.date_range,
),
),
],
),
],
),
);
return LocalProgress( return LocalProgress(
inAsyncCall: _isLoading, inAsyncCall: _isLoading,
child: Scaffold( child: Scaffold(
appBar: AppBar( appBar: AppBar(
centerTitle: true, centerTitle: true,
leading: new IconButton( leading: IconButton(
icon: new Icon(CupertinoIcons.back, color: primaryColor, size: 30), icon: Icon(CupertinoIcons.back, color: primaryColor, size: 30),
onPressed: () => Navigator.of(context).pop(), onPressed: () => Navigator.of(context).pop(),
), ),
shadowColor: Colors.transparent, shadowColor: Colors.transparent,
@@ -163,30 +248,34 @@ class _PackageInfoState extends State<PackageInfo> {
padding: const EdgeInsets.all(10.0), padding: const EdgeInsets.all(10.0),
child: ListView(children: <Widget>[ child: ListView(children: <Widget>[
trackingIdBox, trackingIdBox,
widget.isSearchResult ? Container() : fcsIDBox, Row(
widget.isSearchResult ? Container() : customerNameBox, children: [
widget.isSearchResult ? Container() : marketBox, Flexible(child: consigneeBox),
_package == null || _package!.photoUrls.length == 0 Flexible(child: senderBox)
? Container() ],
: img, ),
marketBox,
widget.isSearchResult ? Container() : descBox, widget.isSearchResult ? Container() : descBox,
remarkBox, remarkBox,
_package?.status == package_received_status &&
widget.isCustomer
? returnButton
: Container(),
widget.isSearchResult ? Container() : deliveryAddressBox, widget.isSearchResult ? Container() : deliveryAddressBox,
widget.isSearchResult _package == null || _package!.photoUrls.isEmpty
? Container() ? Container()
: Padding( : Padding(
padding: const EdgeInsets.only(top: 12), child: img),
shipmentBox,
Padding(
padding: const EdgeInsets.only(top: 15), padding: const EdgeInsets.only(top: 15),
child: StatusTree( child: StatusTree(
shipmentHistory: _package!.shipmentHistory, shipmentHistory: _package!.shipmentHistory,
currentStatus: _package!.status), currentStatus: _package!.status),
), ),
SizedBox( _package?.status == package_received_status &&
height: 20, widget.isCustomer
) ? Padding(
padding: EdgeInsets.only(top: 15),
child: returnButton)
: Container(),
SizedBox(height: 20)
]), ]),
)), )),
], ],

View File

@@ -33,10 +33,8 @@ class PackageListRow extends StatelessWidget {
Navigator.push( Navigator.push(
context, context,
CupertinoPageRoute( CupertinoPageRoute(
builder: (context) => PackageInfo( builder: (context) =>
package: package, PackageInfo(package: package, isCustomer: isCustomer)),
isCustomer: isCustomer,
)),
); );
}, },
child: Container( child: Container(

View File

@@ -1,3 +1,5 @@
// ignore_for_file: use_build_context_synchronously
import 'package:fcs/constants.dart'; import 'package:fcs/constants.dart';
import 'package:fcs/domain/entities/market.dart'; import 'package:fcs/domain/entities/market.dart';
import 'package:fcs/domain/entities/package.dart'; import 'package:fcs/domain/entities/package.dart';
@@ -20,19 +22,25 @@ import 'package:flutter/material.dart';
import 'package:flutter_vector_icons/flutter_vector_icons.dart'; import 'package:flutter_vector_icons/flutter_vector_icons.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
import '../../domain/entities/fcs_shipment.dart';
class PackageEditor extends StatefulWidget { class PackageEditor extends StatefulWidget {
final Package? package; final User consignee;
final User? consignee; final User sender;
final User? sender; final FcsShipment shipment;
PackageEditor({this.package, this.consignee, this.sender}); const PackageEditor(
{super.key,
required this.consignee,
required this.sender,
required this.shipment});
@override @override
_PackageEditorState createState() => _PackageEditorState(); _PackageEditorState createState() => _PackageEditorState();
} }
class _PackageEditorState extends State<PackageEditor> { class _PackageEditorState extends State<PackageEditor> {
TextEditingController _remarkCtl = new TextEditingController(); final _remarkCtl = TextEditingController();
TextEditingController _descCtl = new TextEditingController(); final _descCtl = TextEditingController();
bool _isLoading = false; bool _isLoading = false;
MultiImgController multiImgController = MultiImgController(); MultiImgController multiImgController = MultiImgController();
@@ -42,7 +50,6 @@ class _PackageEditorState extends State<PackageEditor> {
void initState() { void initState() {
super.initState(); super.initState();
_package = Package(); _package = Package();
_loadPackageData(widget.package?.id!);
} }
_loadPackageData(String? id) async { _loadPackageData(String? id) async {
@@ -75,7 +82,7 @@ class _PackageEditorState extends State<PackageEditor> {
children: <Widget>[ children: <Widget>[
Expanded( Expanded(
child: DisplayText( child: DisplayText(
text: _package!.trackingID != null ? _package!.trackingID : "", text: _package!.trackingID ?? "",
labelTextKey: "processing.tracking.id", labelTextKey: "processing.tracking.id",
iconData: MaterialCommunityIcons.barcode_scan, iconData: MaterialCommunityIcons.barcode_scan,
)), )),
@@ -114,8 +121,8 @@ class _PackageEditorState extends State<PackageEditor> {
child: Scaffold( child: Scaffold(
appBar: AppBar( appBar: AppBar(
centerTitle: true, centerTitle: true,
leading: new IconButton( leading: IconButton(
icon: new Icon(CupertinoIcons.back, color: primaryColor, size: 30), icon: Icon(CupertinoIcons.back, color: primaryColor, size: 30),
onPressed: () { onPressed: () {
if (isDataChanged()) { if (isDataChanged()) {
showConfirmDialog(context, "back.button_confirm", () { showConfirmDialog(context, "back.button_confirm", () {
@@ -128,20 +135,8 @@ class _PackageEditorState extends State<PackageEditor> {
), ),
shadowColor: Colors.transparent, shadowColor: Colors.transparent,
backgroundColor: Colors.white, backgroundColor: Colors.white,
title: widget.package == null title: LocalText(context, "processing.package.create",
? LocalText( fontSize: 20, color: primaryColor)),
context,
"processing.package.create",
fontSize: 20,
color: primaryColor,
)
: LocalText(
context,
"processing.package.update",
fontSize: 20,
color: primaryColor,
),
),
body: Padding( body: Padding(
padding: const EdgeInsets.all(8.0), padding: const EdgeInsets.all(8.0),
child: ListView( child: ListView(
@@ -256,9 +251,8 @@ class _PackageEditorState extends State<PackageEditor> {
_package!.desc = _descCtl.text; _package!.desc = _descCtl.text;
_package!.remark = _remarkCtl.text; _package!.remark = _remarkCtl.text;
_package!.photoFiles = multiImgController.getUpdatedFile; _package!.photoFiles = multiImgController.getUpdatedFile;
_package!.fcsID = widget.consignee?.fcsID; _package!.fcsID = widget.consignee.fcsID;
_package!.senderFCSID = _package!.senderFCSID = widget.sender.fcsID ?? "";
widget.sender?.fcsID != null ? widget.sender?.fcsID : "";
await packageModel.updateProcessing(_package!, await packageModel.updateProcessing(_package!,
multiImgController.getAddedFile, multiImgController.getDeletedUrl); multiImgController.getAddedFile, multiImgController.getDeletedUrl);

View File

@@ -9,7 +9,6 @@ import 'package:fcs/pages/package/tracking_id_page.dart';
import 'package:fcs/pages/main/util.dart'; import 'package:fcs/pages/main/util.dart';
import 'package:fcs/pages/user_search/user_search.dart'; import 'package:fcs/pages/user_search/user_search.dart';
import 'package:fcs/pages/widgets/display_text.dart'; import 'package:fcs/pages/widgets/display_text.dart';
import 'package:fcs/pages/widgets/fcs_id_icon.dart';
import 'package:fcs/pages/widgets/input_text.dart'; import 'package:fcs/pages/widgets/input_text.dart';
import 'package:fcs/pages/widgets/local_app_bar.dart'; import 'package:fcs/pages/widgets/local_app_bar.dart';
import 'package:fcs/pages/widgets/local_text.dart'; import 'package:fcs/pages/widgets/local_text.dart';
@@ -22,22 +21,32 @@ 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';
import '../../domain/entities/fcs_shipment.dart';
import '../fcs_shipment/model/fcs_shipment_model.dart';
import '../widgets/local_dropdown.dart';
class ProcessingEditEditor extends StatefulWidget { class ProcessingEditEditor extends StatefulWidget {
final Package? package; final Package package;
ProcessingEditEditor({this.package}); const ProcessingEditEditor({super.key, required this.package});
@override @override
_ProcessingEditEditorState createState() => _ProcessingEditEditorState(); _ProcessingEditEditorState createState() => _ProcessingEditEditorState();
} }
class _ProcessingEditEditorState extends State<ProcessingEditEditor> { class _ProcessingEditEditorState extends State<ProcessingEditEditor> {
TextEditingController _remarkCtl = new TextEditingController(); final DateFormat dateFormat = DateFormat("d MMM yyyy");
TextEditingController _descCtl = new TextEditingController();
TextEditingController _remarkCtl = TextEditingController();
TextEditingController _descCtl = TextEditingController();
Package? _package; Package? _package;
User? _consignee; User? _consignee;
User? _sender; User? _sender;
bool _isLoading = false; bool _isLoading = false;
List<FcsShipment> _shipments = [];
FcsShipment? _shipment;
MultiImgController multiImgController = MultiImgController();
@override @override
void initState() { void initState() {
@@ -56,57 +65,47 @@ class _ProcessingEditEditorState extends State<ProcessingEditEditor> {
fcsID: _package!.senderFCSID ?? "", fcsID: _package!.senderFCSID ?? "",
name: _package!.senderName ?? "", name: _package!.senderName ?? "",
phoneNumber: _package!.senderPhoneNumber ?? ""); phoneNumber: _package!.senderPhoneNumber ?? "");
_loadShipment();
} }
final DateFormat dateFormat = DateFormat("d MMM yyyy"); _loadShipment() async {
var fcsShipments =
await context.read<FcsShipmentModel>().getActiveFcsShipments();
_shipments = fcsShipments;
bool isNew = false; var s = FcsShipment(
MultiImgController multiImgController = MultiImgController(); id: widget.package.id, shipmentNumber: widget.package.shipmentNumber);
if (_shipments.contains(s)) {
_shipment = s;
} else {
_shipment = null;
}
if (mounted) {
setState(() {});
}
}
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
var fcsIDBox = Row(
children: <Widget>[
Expanded(
child: DisplayText(
text: _consignee!.fcsID,
labelTextKey: "processing.fcs.id",
icon: FcsIDIcon(),
)),
IconButton(
icon: Icon(Icons.search, color: primaryColor),
onPressed: () => searchUser(context, onUserSelect: (u) {
setState(() {
this._consignee = u;
});
})),
],
);
final namebox = DisplayText(
text: _consignee!.name,
labelTextKey: "processing.consignee.name",
iconData: Icons.person,
);
final phoneNumberBox = DisplayText(
text: _consignee!.phoneNumber,
labelTextKey: "processing.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,
); );
final completeProcessingBtn = fcsButton( final completeProcessingBtn = fcsButton(
context, context,
getLocalString(context, 'processing.edit.complete.btn'), getLocalString(context, 'processing.edit.complete.btn'),
callack: _completeProcessing, callack: _completeProcessing,
); );
final descBox = InputText( final descBox = InputText(
labelTextKey: 'processing.desc', labelTextKey: 'processing.desc',
iconData: MaterialCommunityIcons.message_text_outline, iconData: MaterialCommunityIcons.message_text_outline,
controller: _descCtl); controller: _descCtl);
final remarkBox = InputText( final remarkBox = InputText(
labelTextKey: 'processing.remark', labelTextKey: 'processing.remark',
iconData: Entypo.new_message, iconData: Entypo.new_message,
@@ -118,57 +117,40 @@ class _ProcessingEditEditorState extends State<ProcessingEditEditor> {
title: "Receipt File", title: "Receipt File",
); );
final consigneeBox = Container( final consigneeBox = userSearchBox(context,
child: Column( lableKey: 'box.consignee.title',
children: [ icon: MaterialCommunityIcons.account_arrow_left,
fcsIDBox, user: _consignee,
phoneNumberBox, onSearch: () => searchUser(context, onUserSelect: (u) {
namebox,
],
),
);
var shipperIDBox = Row(
children: <Widget>[
Expanded(
child: DisplayText(
text: _sender != null ? _sender!.fcsID : "",
labelTextKey: "processing.fcs.id",
icon: FcsIDIcon(),
)),
IconButton(
icon: Icon(Icons.search, color: primaryColor),
onPressed: () => searchUser(context, onUserSelect: (u) {
setState(() { setState(() {
this._sender = u; _consignee = u;
}); });
}, popPage: true)), }, popPage: true));
],
);
final shipperPhoneNumberBox = DisplayText( final senderBox = userSearchBox(context,
text: _sender != null ? _sender!.phoneNumber : "", lableKey: 'box.sender.title',
labelTextKey: "processing.phone", icon: MaterialCommunityIcons.account_arrow_right,
maxLines: 2, user: _sender,
iconData: Icons.phone, onSearch: () => searchUser(context, onUserSelect: (u) {
); setState(() {
_sender = u;
});
}, popPage: true));
final shipperNamebox = DisplayText( final fcsShipmentsBox = Container(
text: _sender != null ? _sender!.name : "", padding: EdgeInsets.symmetric(vertical: 15),
labelTextKey: "processing.shipper.name", child: LocalDropdown<FcsShipment>(
maxLines: 2, callback: (v) {
iconData: Icons.person, setState(() {
); _shipment = v;
});
final shipperBox = Container( },
child: Column( labelKey: "box.shipment",
children: [ iconData: Ionicons.ios_airplane,
shipperIDBox, display: (u) => u.shipmentNumber,
shipperPhoneNumberBox, selectedValue: _shipment,
shipperNamebox, values: _shipments,
], ));
),
);
return LocalProgress( return LocalProgress(
inAsyncCall: _isLoading, inAsyncCall: _isLoading,
@@ -189,24 +171,26 @@ class _ProcessingEditEditorState extends State<ProcessingEditEditor> {
}, },
), ),
body: Padding( body: Padding(
padding: const EdgeInsets.all(8.0), padding: const EdgeInsets.only(left: 12.0, right: 12, top: 10),
child: ListView( child: ListView(
children: [ children: [
trackingIdBox, trackingIdBox,
Row( Row(
children: [ children: [
Flexible(child: consigneeBox), Flexible(child: consigneeBox),
Flexible(child: shipperBox) Flexible(child: senderBox)
], ],
), ),
const SizedBox(height: 8),
fcsShipmentsBox,
marketDropdown(), marketDropdown(),
descBox, descBox,
remarkBox, remarkBox,
const SizedBox(height: 10),
img, img,
SizedBox(height: 20),
completeProcessingBtn, completeProcessingBtn,
SizedBox( SizedBox(height: 30)
height: 20,
)
], ],
), ),
), ),
@@ -319,23 +303,16 @@ class _ProcessingEditEditorState extends State<ProcessingEditEditor> {
} }
isDataChanged() { isDataChanged() {
if (isNew) { var package = Package(
return _consignee!.fcsID != "" ||
selectedMarket != null ||
_descCtl.text != "" ||
_remarkCtl.text != "" ||
multiImgController.getAddedFile.isNotEmpty;
} else {
var _package = Package(
fcsID: _consignee!.fcsID, fcsID: _consignee!.fcsID,
senderFCSID: _sender!.fcsID, senderFCSID: _sender!.fcsID,
market: selectedMarket, market: selectedMarket,
desc: _descCtl.text, desc: _descCtl.text,
remark: _remarkCtl.text, remark: _remarkCtl.text,
photoUrls: widget.package!.photoUrls); photoUrls: widget.package.photoUrls,
return widget.package!.isChangedForEditProcessing(_package) || shipmentId: _shipment?.id);
return widget.package.isChangedForEditProcessing(package) ||
multiImgController.getAddedFile.isNotEmpty || multiImgController.getAddedFile.isNotEmpty ||
multiImgController.getDeletedUrl.isNotEmpty; multiImgController.getDeletedUrl.isNotEmpty;
} }
}
} }

View File

@@ -1,3 +1,5 @@
// ignore_for_file: use_build_context_synchronously
import 'package:fcs/domain/entities/package.dart'; import 'package:fcs/domain/entities/package.dart';
import 'package:fcs/domain/entities/processing.dart'; import 'package:fcs/domain/entities/processing.dart';
import 'package:fcs/domain/entities/user.dart'; import 'package:fcs/domain/entities/user.dart';
@@ -6,7 +8,6 @@ import 'package:fcs/pages/main/util.dart';
import 'package:fcs/pages/package/package_info.dart'; import 'package:fcs/pages/package/package_info.dart';
import 'package:fcs/pages/user_search/user_search.dart'; import 'package:fcs/pages/user_search/user_search.dart';
import 'package:fcs/pages/widgets/display_text.dart'; import 'package:fcs/pages/widgets/display_text.dart';
import 'package:fcs/pages/widgets/fcs_id_icon.dart';
import 'package:fcs/pages/widgets/local_app_bar.dart'; import 'package:fcs/pages/widgets/local_app_bar.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';
@@ -14,12 +15,14 @@ import 'package:flutter/material.dart';
import 'package:flutter_vector_icons/flutter_vector_icons.dart'; import 'package:flutter_vector_icons/flutter_vector_icons.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
import '../../domain/entities/fcs_shipment.dart';
import '../fcs_shipment/model/fcs_shipment_model.dart';
import '../widgets/local_dropdown.dart';
import 'model/processing_model.dart'; import 'model/processing_model.dart';
import 'package_editor.dart'; import 'package_editor.dart';
class ProcesingEditor extends StatefulWidget { class ProcesingEditor extends StatefulWidget {
final Processing? processing; const ProcesingEditor({super.key});
const ProcesingEditor({this.processing});
@override @override
_ProcesingEditorState createState() => _ProcesingEditorState(); _ProcesingEditorState createState() => _ProcesingEditorState();
} }
@@ -27,117 +30,66 @@ class ProcesingEditor extends StatefulWidget {
class _ProcesingEditorState extends State<ProcesingEditor> { class _ProcesingEditorState extends State<ProcesingEditor> {
Processing processing = Processing(); Processing processing = Processing();
bool _isLoading = false; bool _isLoading = false;
late bool _isNew; User? _consignee;
User? consignee; User? _sender;
User? sender;
List<Package> packages = []; List<Package> packages = [];
List<FcsShipment> _shipments = [];
FcsShipment? _shipment;
@override @override
void initState() { void initState() {
_loadShipment();
super.initState(); super.initState();
_isNew = widget.processing == null; }
if (!_isNew) {
processing = widget.processing!; _loadShipment() async {
consignee = User( var fcsShipments =
fcsID: processing.userID, await context.read<FcsShipmentModel>().getActiveFcsShipments();
name: processing.userName, _shipments = fcsShipments;
phoneNumber: processing.userPhoneNumber);
sender = User( if (mounted) {
fcsID: processing.fcsID, setState(() {});
name: processing.shipperName,
phoneNumber: processing.shipperPhoneNumber);
packages = processing.packages;
} }
} }
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
var fcsIDBox = Row( final consigneeBox = userSearchBox(context,
children: <Widget>[ lableKey: 'box.consignee.title',
Expanded( icon: MaterialCommunityIcons.account_arrow_left,
child: DisplayText( user: _consignee,
text: consignee != null ? consignee!.fcsID : "", onSearch: () => searchUser(context, onUserSelect: (u) {
labelTextKey: "processing.fcs.id",
icon: FcsIDIcon(),
)),
IconButton(
icon: Icon(Icons.search, color: primaryColor),
onPressed: () => searchUser(context, onUserSelect: (u) {
setState(() { setState(() {
this.consignee = u; _consignee = u;
}); });
}, popPage: true)), }, popPage: true));
],
);
final phoneNumberBox = DisplayText( final senderBox = userSearchBox(context,
text: consignee != null ? consignee!.phoneNumber : "", lableKey: 'box.sender.title',
labelTextKey: "processing.phone", icon: MaterialCommunityIcons.account_arrow_right,
maxLines: 2, user: _sender,
iconData: Icons.phone, onSearch: () => searchUser(context, onUserSelect: (u) {
);
final namebox = DisplayText(
text: consignee != null ? consignee!.name : "",
labelTextKey: "processing.consignee.name",
maxLines: 2,
iconData: Icons.person,
);
final consigneeBox = Container(
child: Column(
children: [
fcsIDBox,
phoneNumberBox,
namebox,
],
),
);
var shipperIDBox = Row(
children: <Widget>[
Expanded(
child: DisplayText(
text: sender != null ? sender!.fcsID : "",
labelTextKey: "processing.fcs.id",
icon: FcsIDIcon(),
)),
IconButton(
icon: Icon(Icons.search, color: primaryColor),
onPressed: () => searchUser(context, onUserSelect: (u) {
setState(() { setState(() {
this.sender = u; _sender = u;
}); });
}, popPage: true)), }, popPage: true));
],
);
final shipperPhoneNumberBox = DisplayText( final fcsShipmentsBox = Container(
text: sender != null ? sender!.phoneNumber : "", padding: EdgeInsets.symmetric(vertical: 15),
labelTextKey: "processing.phone", child: LocalDropdown<FcsShipment>(
maxLines: 2, callback: (v) {
iconData: Icons.phone, setState(() {
); _shipment = v;
});
},
labelKey: "box.shipment",
iconData: Ionicons.ios_airplane,
display: (u) => u.shipmentNumber,
selectedValue: _shipment,
values: _shipments,
));
final shipperNamebox = DisplayText( final packageTitleBox = Row(
text: sender != null ? sender!.name : "",
labelTextKey: "processing.shipper.name",
maxLines: 2,
iconData: Icons.person,
);
final shipperBox = Container(
child: Column(
children: [
shipperIDBox,
shipperPhoneNumberBox,
shipperNamebox,
],
),
);
final packageTitleBox = Container(
child: Row(
mainAxisAlignment: MainAxisAlignment.end, mainAxisAlignment: MainAxisAlignment.end,
children: [ children: [
Text("Packages (${packages.length})"), Text("Packages (${packages.length})"),
@@ -147,25 +99,34 @@ class _ProcesingEditorState extends State<ProcesingEditor> {
color: primaryColor, color: primaryColor,
), ),
onPressed: () async { onPressed: () async {
if (this.consignee == null) { if (_consignee == null) {
showMsgDialog( showMsgDialog(context, "Error", "Please select consignee");
context, "Warning", "Please select 'Consignee'");
return; return;
} }
Package? _package = await Navigator.push<Package>(
if (_sender == null) {
showMsgDialog(context, "Error", "Please select sender");
return;
}
if (_shipment == null) {
showMsgDialog(context, "Error", "Please select shipment");
return;
}
Package? package = await Navigator.push<Package>(
context, context,
CupertinoPageRoute( CupertinoPageRoute(
builder: (context) => PackageEditor( builder: (context) => PackageEditor(
sender: this.sender, sender: _sender!,
consignee: this.consignee, consignee: _consignee!,
)), shipment: _shipment!)),
); );
_addPackage(_package); _addPackage(package);
// _savePackage(_package); // _savePackage(_package);
}), }),
], ],
),
); );
final createButton = fcsButton( final createButton = fcsButton(
@@ -178,7 +139,7 @@ class _ProcesingEditorState extends State<ProcesingEditor> {
inAsyncCall: _isLoading, inAsyncCall: _isLoading,
child: Scaffold( child: Scaffold(
appBar: LocalAppBar( appBar: LocalAppBar(
labelKey: _isNew ? "processing.create" : "processing.update", labelKey: "processing.create",
backgroundColor: Colors.white, backgroundColor: Colors.white,
labelColor: primaryColor, labelColor: primaryColor,
arrowColor: primaryColor, arrowColor: primaryColor,
@@ -193,10 +154,11 @@ class _ProcesingEditorState extends State<ProcesingEditor> {
Row( Row(
children: [ children: [
Flexible(child: consigneeBox), Flexible(child: consigneeBox),
Flexible(child: shipperBox) Flexible(child: senderBox)
], ],
), ),
Divider(), const SizedBox(height: 10),
fcsShipmentsBox,
packageTitleBox, packageTitleBox,
Column( Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
@@ -206,9 +168,7 @@ class _ProcesingEditorState extends State<ProcesingEditor> {
height: 20, height: 20,
), ),
createButton, createButton,
SizedBox( SizedBox(height: 10),
height: 10,
),
], ],
), ),
), ),
@@ -236,7 +196,7 @@ class _ProcesingEditorState extends State<ProcesingEditor> {
_addPackage(Package? package) { _addPackage(Package? package) {
if (package == null) return; if (package == null) return;
this.packages.add(package); packages.add(package);
setState(() {}); setState(() {});
} }
@@ -253,12 +213,7 @@ class _ProcesingEditorState extends State<ProcesingEditor> {
ProcessingModel processingModel = ProcessingModel processingModel =
Provider.of<ProcessingModel>(context, listen: false); Provider.of<ProcessingModel>(context, listen: false);
try { try {
if (_isNew) {
await processingModel.createProcessing(processing); await processingModel.createProcessing(processing);
} else {
processing.id = widget.processing!.id;
await processingModel.updateProcessing(processing);
}
Navigator.pop(context); Navigator.pop(context);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());

View File

@@ -1,10 +1,11 @@
import 'package:fcs/domain/entities/fcs_shipment.dart';
import 'package:fcs/domain/entities/package.dart'; import 'package:fcs/domain/entities/package.dart';
import 'package:fcs/helpers/theme.dart'; import 'package:fcs/helpers/theme.dart';
import 'package:fcs/pages/main/util.dart'; import 'package:fcs/pages/main/util.dart';
import 'package:fcs/pages/package/model/package_model.dart'; import 'package:fcs/pages/package/model/package_model.dart';
import 'package:fcs/pages/widgets/display_text.dart'; import 'package:fcs/pages/widgets/display_text.dart';
import 'package:fcs/pages/widgets/fcs_id_icon.dart';
import 'package:fcs/pages/widgets/local_app_bar.dart'; import 'package:fcs/pages/widgets/local_app_bar.dart';
import 'package:fcs/pages/widgets/local_text.dart';
import 'package:fcs/pages/widgets/multi_img_controller.dart'; import 'package:fcs/pages/widgets/multi_img_controller.dart';
import 'package:fcs/pages/widgets/multi_img_file.dart'; import 'package:fcs/pages/widgets/multi_img_file.dart';
import 'package:fcs/pages/widgets/progress.dart'; import 'package:fcs/pages/widgets/progress.dart';
@@ -15,13 +16,14 @@ 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';
import '../fcs_shipment/model/fcs_shipment_model.dart';
import 'processing_edit_editor.dart'; import 'processing_edit_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}); const ProcessingInfo({super.key, required this.package});
@override @override
_ProcessingInfoState createState() => _ProcessingInfoState(); _ProcessingInfoState createState() => _ProcessingInfoState();
@@ -32,21 +34,33 @@ class _ProcessingInfoState extends State<ProcessingInfo> {
Package? _package; Package? _package;
bool _isLoading = false; bool _isLoading = false;
MultiImgController multiImgController = MultiImgController(); MultiImgController multiImgController = MultiImgController();
FcsShipment? _shipment;
@override @override
void initState() { void initState() {
super.initState(); super.initState();
initPackage(widget.package!); _initPackage(widget.package);
_loadShipment();
} }
initPackage(Package? package) { _initPackage(Package package) {
if (package == null) return;
setState(() { setState(() {
_package = package; _package = package;
multiImgController.setImageUrls = package.photoUrls; multiImgController.setImageUrls = package.photoUrls;
}); });
} }
_loadShipment() async {
if (widget.package.shipmentId == null) return;
var s = await context
.read<FcsShipmentModel>()
.getFcsShipment(widget.package.shipmentId!);
_shipment = s;
if (mounted) {
setState(() {});
}
}
@override @override
void dispose() { void dispose() {
super.dispose(); super.dispose();
@@ -59,36 +73,7 @@ class _ProcessingInfoState extends State<ProcessingInfo> {
labelTextKey: "processing.tracking.id", labelTextKey: "processing.tracking.id",
iconData: MaterialCommunityIcons.barcode_scan, iconData: MaterialCommunityIcons.barcode_scan,
); );
var fcsIDBox = DisplayText(
text: _package != null ? _package!.fcsID : "",
labelTextKey: "processing.fcs.id",
icon: FcsIDIcon(),
);
final phoneNumberBox = DisplayText(
text: _package != null ? _package!.phoneNumber : "",
labelTextKey: "processing.phone",
iconData: Icons.phone,
);
final customerNameBox = DisplayText(
text: _package != null ? _package!.userName : "",
labelTextKey: "processing.consignee.name",
iconData: Icons.perm_identity,
);
var senderFcsIDBox = DisplayText(
text: _package != null ? _package!.senderFCSID : "",
labelTextKey: "processing.fcs.id",
icon: FcsIDIcon(),
);
final senderPhoneNumberBox = DisplayText(
text: _package != null ? _package!.senderPhoneNumber : "",
labelTextKey: "processing.phone",
iconData: Icons.phone,
);
final senderNameBox = DisplayText(
text: _package != null ? _package!.senderName : "",
labelTextKey: "processing.shipper.name",
iconData: Icons.perm_identity,
);
final marketBox = DisplayText( final marketBox = DisplayText(
text: _package != null ? _package!.market : "-", text: _package != null ? _package!.market : "-",
labelTextKey: "processing.market", labelTextKey: "processing.market",
@@ -110,24 +95,77 @@ class _ProcessingInfoState extends State<ProcessingInfo> {
title: "Receipt File", title: "Receipt File",
); );
final consigneeBox = Container( final consigneeBox = userDisplayBox(context,
lableKey: "box.consignee.title",
icon: MaterialCommunityIcons.account_arrow_left,
name: _package?.userName ?? "",
fcsID: _package?.fcsID ?? "");
final senderBox = userDisplayBox(context,
lableKey: "box.sender.title",
icon: MaterialCommunityIcons.account_arrow_right,
name: _package?.senderName ?? "",
fcsID: _package?.senderFCSID ?? "");
final shipmentBox = Padding(
padding: const EdgeInsets.only(top: 15),
child: Column( child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
fcsIDBox, LocalText(context, "package.shipment.title",
phoneNumberBox, color: primaryColor, fontSize: 17, fontWeight: FontWeight.normal),
customerNameBox, Row(
], children: [
), Flexible(
); child: DisplayText(
final shipperBox = Container( text: _shipment?.shipmentNumber ?? '',
child: Column( labelTextKey: "FCSshipment.number",
children: [ iconData: Ionicons.ios_airplane,
senderFcsIDBox, ),
senderPhoneNumberBox, ),
senderNameBox, Flexible(
child: DisplayText(
text: _shipment != null
? _shipment!.processingDate != null
? dateFormatter.format(_shipment!.processingDate!)
: ""
: "",
labelTextKey: "package.processing.date",
iconData: Icons.date_range,
),
),
],
),
Row(
children: [
Flexible(
child: DisplayText(
text: _shipment != null
? _shipment!.cutoffDate != null
? dateFormatter.format(_shipment!.cutoffDate!)
: ""
: "",
labelTextKey: "FCSshipment.cutoff_date",
iconData: Icons.date_range,
),
),
Flexible(
child: DisplayText(
text: _shipment != null
? _shipment!.etaDate != null
? dateFormatter.format(_shipment!.etaDate!)
: ""
: "",
labelTextKey: "FCSshipment.ETA",
iconData: Icons.date_range,
),
),
],
),
], ],
), ),
); );
return LocalProgress( return LocalProgress(
inAsyncCall: _isLoading, inAsyncCall: _isLoading,
child: Scaffold( child: Scaffold(
@@ -148,31 +186,32 @@ class _ProcessingInfoState extends State<ProcessingInfo> {
]), ]),
body: Card( body: Card(
elevation: 0, elevation: 0,
child: Column( child:
children: <Widget>[ ListView(padding: const EdgeInsets.all(10.0), children: <Widget>[
Expanded(
child: Padding(
padding: const EdgeInsets.all(10.0),
child: ListView(children: <Widget>[
trackingIdBox, trackingIdBox,
Row( Row(
children: [ children: [
Flexible(child: consigneeBox), Flexible(child: consigneeBox),
Flexible(child: shipperBox) Flexible(
child: _package?.senderFCSID != null &&
_package?.senderFCSID != ""
? senderBox
: const SizedBox())
], ],
), ),
marketBox, marketBox,
descBox, descBox,
remarkBox, remarkBox,
_package!.photoUrls.length == 0 ? Container() : img, _package!.photoUrls.isEmpty ? Container() : img,
StatusTree( shipmentBox,
Padding(
padding: const EdgeInsets.only(top: 15),
child: StatusTree(
shipmentHistory: _package!.shipmentHistory, shipmentHistory: _package!.shipmentHistory,
currentStatus: _package!.status ?? ""), currentStatus: _package!.status ?? ""),
),
SizedBox(height: 20) SizedBox(height: 20)
]), ]),
)),
],
),
), ),
), ),
); );
@@ -201,10 +240,11 @@ class _ProcessingInfoState extends State<ProcessingInfo> {
} }
_gotoEditor() async { _gotoEditor() async {
if (_package == null) return;
bool? deleted = await Navigator.push<bool>( bool? deleted = await Navigator.push<bool>(
context, context,
CupertinoPageRoute( CupertinoPageRoute(
builder: (context) => ProcessingEditEditor(package: _package))); builder: (context) => ProcessingEditEditor(package: _package!)));
if (deleted ?? false) { if (deleted ?? false) {
Navigator.pop(context); Navigator.pop(context);
} else { } else {
@@ -212,7 +252,7 @@ class _ProcessingInfoState extends State<ProcessingInfo> {
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!);
if (p == null) return; if (p == null) return;
initPackage(p); _initPackage(p);
} }
} }
} }

View File

@@ -76,12 +76,6 @@ class _ProfileState extends State<Profile> {
buildLanguage(languageModel); buildLanguage(languageModel);
var deliveryAddressModel = Provider.of<DeliveryAddressModel>(context); var deliveryAddressModel = Provider.of<DeliveryAddressModel>(context);
final namebox = DisplayText(
text: "${user.name ?? ''}" + " (${user.status ?? ''})",
labelTextKey: "profile.name",
iconData: Icons.person,
);
final currencyBox = Row( final currencyBox = Row(
children: <Widget>[ children: <Widget>[
Expanded( Expanded(
@@ -104,24 +98,6 @@ class _ProfileState extends State<Profile> {
iconData: MaterialCommunityIcons.account_remove, iconData: MaterialCommunityIcons.account_remove,
); );
final phonenumberBox = Row(
children: [
Expanded(
child: DisplayText(
text: user.phone,
labelTextKey: "profile.phone",
iconData: Icons.phone),
),
// IconButton(
// icon: Icon(Icons.edit, color: Colors.grey),
// onPressed: () {
// Navigator.of(context, rootNavigator: true).push(
// CupertinoPageRoute(
// builder: (context) => ChangePhoneNumber(user: user)));
// })
],
);
final fcsIDBox = Row( final fcsIDBox = Row(
children: [ children: [
Expanded( Expanded(
@@ -187,31 +163,48 @@ class _ProfileState extends State<Profile> {
}) })
], ],
); );
final titleBox = Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Row(
mainAxisAlignment: MainAxisAlignment.center,
mainAxisSize: MainAxisSize.min,
children: [
Text(
user.name ?? "",
style: TextStyle(fontSize: 18, color: Colors.black),
),
const SizedBox(width: 5),
InkResponse(
radius: 20,
onTap: _editName,
child: Icon(Icons.edit, color: Colors.grey, size: 23))
],
),
const SizedBox(height: 1),
Text(
user.phone,
style: TextStyle(fontSize: 15, color: labelColor),
),
],
);
return LocalProgress( return LocalProgress(
inAsyncCall: _isLoading, inAsyncCall: _isLoading,
child: Scaffold( child: Scaffold(
key: key, key: key,
appBar: LocalAppBar( appBar: LocalAppBar(
labelKey: "profile.title",
backgroundColor: Colors.white, backgroundColor: Colors.white,
labelColor: primaryColor, labelColor: primaryColor,
arrowColor: primaryColor), arrowColor: primaryColor,
titleWidget: titleBox
),
body: Padding( body: Padding(
padding: const EdgeInsets.all(8.0), padding: const EdgeInsets.all(8.0),
child: ListView( child: ListView(
children: <Widget>[ children: <Widget>[
Row( const SizedBox(height: 5),
children: <Widget>[
Expanded(child: namebox),
Padding(
padding: const EdgeInsets.only(right: 0),
child: IconButton(
icon: Icon(Icons.edit, color: Colors.grey),
onPressed: _editName),
)
],
),
phonenumberBox,
fcsIDBox, fcsIDBox,
usaShippingAddressBox, usaShippingAddressBox,
currencyBox, currencyBox,
@@ -327,13 +320,13 @@ class _ProfileState extends State<Profile> {
if (user == null || user.isCustomer()) return Container(); if (user == null || user.isCustomer()) return Container();
List<Privilege> privileges = []; List<Privilege> privileges = [];
user.privileges.forEach((e) { for (var e in user.privileges) {
Privilege? p = _privileges.firstWhereOrNull((p) => p.id == e); Privilege? p = _privileges.firstWhereOrNull((p) => p.id == e);
if (p != null) { if (p != null) {
privileges.add(p); privileges.add(p);
} }
}); }
return privileges.isEmpty return privileges.isEmpty
? const SizedBox() ? const SizedBox()
@@ -375,8 +368,7 @@ class _ProfileState extends State<Profile> {
style: TextStyle( style: TextStyle(
fontSize: 16.0, fontSize: 16.0,
fontStyle: FontStyle.normal, fontStyle: FontStyle.normal,
color: primaryColor, color: primaryColor)),
)),
Text( Text(
"${p.desc}", "${p.desc}",
style: TextStyle( style: TextStyle(
@@ -401,9 +393,7 @@ class _ProfileState extends State<Profile> {
_showToast(String title) { _showToast(String title) {
ScaffoldMessengerState? scaffold = key.currentState; ScaffoldMessengerState? scaffold = key.currentState;
if (scaffold == null) { scaffold ??= ScaffoldMessenger.of(context);
scaffold = ScaffoldMessenger.of(context);
}
scaffold.showSnackBar( scaffold.showSnackBar(
SnackBar( SnackBar(

View File

@@ -5,8 +5,6 @@ import 'package:fcs/pages/main/util.dart';
import 'package:fcs/pages/package/model/package_model.dart'; import 'package:fcs/pages/package/model/package_model.dart';
import 'package:fcs/pages/user_search/user_search.dart'; import 'package:fcs/pages/user_search/user_search.dart';
import 'package:fcs/pages/widgets/barcode_scanner.dart'; import 'package:fcs/pages/widgets/barcode_scanner.dart';
import 'package:fcs/pages/widgets/display_text.dart';
import 'package:fcs/pages/widgets/fcs_id_icon.dart';
import 'package:fcs/pages/widgets/input_text.dart'; import 'package:fcs/pages/widgets/input_text.dart';
import 'package:fcs/pages/widgets/local_app_bar.dart'; import 'package:fcs/pages/widgets/local_app_bar.dart';
import 'package:fcs/pages/widgets/multi_img_controller.dart'; import 'package:fcs/pages/widgets/multi_img_controller.dart';
@@ -16,11 +14,9 @@ import 'package:flutter/material.dart';
import 'package:flutter_vector_icons/flutter_vector_icons.dart'; import 'package:flutter_vector_icons/flutter_vector_icons.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
typedef void FindCallBack();
class ReceivingEditor extends StatefulWidget { class ReceivingEditor extends StatefulWidget {
final Package? package; final Package? package;
const ReceivingEditor({this.package}); const ReceivingEditor({super.key, this.package});
@override @override
_ReceivingEditorState createState() => _ReceivingEditorState(); _ReceivingEditorState createState() => _ReceivingEditorState();
} }
@@ -29,8 +25,8 @@ class _ReceivingEditorState extends State<ReceivingEditor> {
Package package = Package(); Package package = Package();
bool _isLoading = false; bool _isLoading = false;
late bool _isNew; late bool _isNew;
User? user; User? _consignee;
final _receiveFormKey=GlobalKey<FormState>(); final _receiveFormKey = GlobalKey<FormState>();
TextEditingController _trackingIDCtl = new TextEditingController(); TextEditingController _trackingIDCtl = new TextEditingController();
TextEditingController _remarkCtl = new TextEditingController(); TextEditingController _remarkCtl = new TextEditingController();
MultiImgController _multiImgController = MultiImgController(); MultiImgController _multiImgController = MultiImgController();
@@ -44,12 +40,12 @@ class _ReceivingEditorState extends State<ReceivingEditor> {
_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( _consignee = User(
fcsID: package.fcsID, fcsID: package.fcsID,
name: package.userName, name: package.userName,
phoneNumber: package.phoneNumber); phoneNumber: package.phoneNumber);
} else { } else {
package = new Package(); package = Package();
} }
_trackingIDCtl.addListener(() { _trackingIDCtl.addListener(() {
var text = _trackingIDCtl.text; var text = _trackingIDCtl.text;
@@ -66,24 +62,6 @@ class _ReceivingEditorState extends State<ReceivingEditor> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
var fcsIDBox = Row(
children: <Widget>[
Expanded(
child: DisplayText(
text: user != null ? user!.fcsID : "",
labelTextKey: "receiving.fcs.id",
icon: FcsIDIcon(),
)),
IconButton(
icon: Icon(Icons.search, color: primaryColor),
onPressed: () => searchUser(context, onUserSelect: (u) {
setState(() {
this.user = u;
});
}, popPage: true)),
],
);
final trackingIDBox = Container( final trackingIDBox = Container(
padding: EdgeInsets.only(left: 6), padding: EdgeInsets.only(left: 6),
child: Row( child: Row(
@@ -95,13 +73,12 @@ class _ReceivingEditorState extends State<ReceivingEditor> {
labelTextKey: "receiving.tracking.id", labelTextKey: "receiving.tracking.id",
controller: _trackingIDCtl, controller: _trackingIDCtl,
autovalidateMode: AutovalidateMode.onUserInteraction, autovalidateMode: AutovalidateMode.onUserInteraction,
validator: (value){ validator: (value) {
if(value==null || value.isEmpty){ if (value == null || value.isEmpty) {
return "invalid tracking ID"; return "invalid tracking ID";
} }
return null; return null;
}, },
)), )),
InkWell( InkWell(
onTap: _scan, onTap: _scan,
@@ -130,11 +107,6 @@ class _ReceivingEditorState extends State<ReceivingEditor> {
controller: _multiImgController, controller: _multiImgController,
title: "Receiving", title: "Receiving",
); );
final namebox = DisplayText(
text: user != null ? user!.name : "",
labelTextKey: "receiving.consignee.name",
iconData: Icons.person,
);
final createButton = fcsButton( final createButton = fcsButton(
context, context,
@@ -148,6 +120,17 @@ class _ReceivingEditorState extends State<ReceivingEditor> {
callack: _save, callack: _save,
); );
final consigneeBox = userSearchBox(context,
lableKey: 'box.consignee.title',
icon: MaterialCommunityIcons.account_arrow_left,
rowMainAxisAlignment: MainAxisAlignment.spaceBetween,
user: _consignee,
onSearch: () => searchUser(context, onUserSelect: (u) {
setState(() {
_consignee = u;
});
}, popPage: true));
return LocalProgress( return LocalProgress(
inAsyncCall: _isLoading, inAsyncCall: _isLoading,
child: Scaffold( child: Scaffold(
@@ -183,11 +166,8 @@ class _ReceivingEditorState extends State<ReceivingEditor> {
SizedBox( SizedBox(
height: 10, height: 10,
), ),
fcsIDBox, consigneeBox,
namebox, SizedBox(height: 20),
SizedBox(
height: 20,
),
_isNew ? createButton : updateButton, _isNew ? createButton : updateButton,
SizedBox( SizedBox(
height: 10, height: 10,
@@ -236,8 +216,8 @@ class _ReceivingEditorState extends State<ReceivingEditor> {
return; return;
} }
if (user == null) { if (_consignee == null) {
showMsgDialog(context, "Error", "Please select FCS ID"); showMsgDialog(context, "Error", "Please select consignee");
return; return;
} }
setState(() { setState(() {
@@ -248,12 +228,12 @@ class _ReceivingEditorState extends State<ReceivingEditor> {
try { try {
if (_isNew) { if (_isNew) {
await packageModel.createReceiving( await packageModel.createReceiving(
user!, _p, _multiImgController.getAddedFile); _consignee!, _p, _multiImgController.getAddedFile);
} else { } else {
_p.id = widget.package!.id; _p.id = widget.package!.id;
_p.photoUrls = package.photoUrls; _p.photoUrls = package.photoUrls;
await packageModel.updateReceiving( await packageModel.updateReceiving(
user!, _consignee!,
_p, _p,
_multiImgController.getAddedFile, _multiImgController.getAddedFile,
_multiImgController.getDeletedUrl); _multiImgController.getDeletedUrl);
@@ -272,15 +252,15 @@ class _ReceivingEditorState extends State<ReceivingEditor> {
if (_isNew) { if (_isNew) {
return _trackingIDCtl.text != "" || return _trackingIDCtl.text != "" ||
_remarkCtl.text != "" || _remarkCtl.text != "" ||
user != null || _consignee != null ||
_multiImgController.getAddedFile.isNotEmpty; _multiImgController.getAddedFile.isNotEmpty;
} else { } else {
var _package = Package( var package = Package(
trackingID: _trackingIDCtl.text, trackingID: _trackingIDCtl.text,
remark: _remarkCtl.text, remark: _remarkCtl.text,
fcsID: user!.fcsID, fcsID: _consignee!.fcsID,
photoUrls: widget.package!.photoUrls); photoUrls: widget.package!.photoUrls);
return widget.package!.isChangedForEdit(_package) || return widget.package!.isChangedForEditReceiving(package) ||
_multiImgController.getAddedFile.isNotEmpty || _multiImgController.getAddedFile.isNotEmpty ||
_multiImgController.getDeletedUrl.isNotEmpty; _multiImgController.getDeletedUrl.isNotEmpty;
} }

View File

@@ -4,7 +4,6 @@ import 'package:fcs/pages/main/model/main_model.dart';
import 'package:fcs/pages/main/util.dart'; import 'package:fcs/pages/main/util.dart';
import 'package:fcs/pages/package/model/package_model.dart'; import 'package:fcs/pages/package/model/package_model.dart';
import 'package:fcs/pages/widgets/display_text.dart'; import 'package:fcs/pages/widgets/display_text.dart';
import 'package:fcs/pages/widgets/fcs_id_icon.dart';
import 'package:fcs/pages/widgets/local_app_bar.dart'; import 'package:fcs/pages/widgets/local_app_bar.dart';
import 'package:fcs/pages/widgets/multi_img_controller.dart'; import 'package:fcs/pages/widgets/multi_img_controller.dart';
import 'package:fcs/pages/widgets/multi_img_file.dart'; import 'package:fcs/pages/widgets/multi_img_file.dart';
@@ -61,16 +60,13 @@ class _ReceivingInfoState extends State<ReceivingInfo> {
labelTextKey: "package.tracking.id", labelTextKey: "package.tracking.id",
iconData: MaterialCommunityIcons.barcode_scan, iconData: MaterialCommunityIcons.barcode_scan,
); );
var fcsIDBox = DisplayText(
text: _package!.fcsID, final consigneeBox = userDisplayBox(context,
labelTextKey: "processing.fcs.id", lableKey: "box.consignee.title",
icon: FcsIDIcon(), icon: MaterialCommunityIcons.account_arrow_left,
); name: _package?.userName ?? "",
final customerNameBox = DisplayText( fcsID: _package?.fcsID ?? "");
text: _package!.userName,
labelTextKey: "package.create.name",
iconData: Icons.perm_identity,
);
final remarkBox = DisplayText( final remarkBox = DisplayText(
text: _package!.remark ?? "-", text: _package!.remark ?? "-",
labelTextKey: "package.edit.remark", labelTextKey: "package.edit.remark",
@@ -112,16 +108,21 @@ class _ReceivingInfoState extends State<ReceivingInfo> {
padding: const EdgeInsets.all(10.0), padding: const EdgeInsets.all(10.0),
child: ListView(children: <Widget>[ child: ListView(children: <Widget>[
trackingIdBox, trackingIdBox,
fcsIDBox, consigneeBox,
customerNameBox,
remarkBox, remarkBox,
_package!.photoUrls.length == 0 ? Container() : img, _package!.photoUrls.isEmpty
StatusTree( ? Container()
: Padding(
padding: const EdgeInsets.only(top: 5),
child: img,
),
Padding(
padding: const EdgeInsets.only(top: 15),
child: StatusTree(
shipmentHistory: _package!.shipmentHistory, shipmentHistory: _package!.shipmentHistory,
currentStatus: _package!.status), currentStatus: _package!.status),
SizedBox( ),
height: 20, SizedBox(height: 20)
)
]), ]),
)), )),
], ],

View File

@@ -51,7 +51,7 @@ class StatusTree extends StatelessWidget {
children: <Widget>[ children: <Widget>[
Text( Text(
'Status', 'Status',
style: TextStyle(color: primaryColor, fontWeight: FontWeight.bold), style: TextStyle(color: primaryColor, fontSize: 17),
), ),
Container( Container(
child: Timeline.builder( child: Timeline.builder(
@@ -76,16 +76,20 @@ class StatusTree extends StatelessWidget {
Text(e.status, Text(e.status,
style: TextStyle( style: TextStyle(
color: e.done! ? primaryColor : Colors.grey, color: e.done! ? primaryColor : Colors.grey,
fontSize: 16, fontSize: 15,
fontWeight: FontWeight.bold)), fontWeight: FontWeight.bold)),
e.done! || isPacked e.done! || isPacked
? e.date != null ? e.date != null
? Text(dateFormatter.format(e.date!)) ? Text(
dateFormatter.format(e.date!),
style: TextStyle(fontSize: 13),
)
: const SizedBox() : const SizedBox()
: Container(), : Container(),
e.staffName == null e.staffName == null
? Container() ? Container()
: Text(e.staffName!) : Text(e.staffName!,
style: TextStyle(fontSize: 13))
], ],
), ),
), ),