update pickup

This commit is contained in:
tzw
2021-10-09 17:08:28 +06:30
parent 46da87dc0e
commit 50901992d7
21 changed files with 597 additions and 274 deletions

View File

@@ -126,6 +126,7 @@
"customer.invitation.request.confirm":"Accept customer", "customer.invitation.request.confirm":"Accept customer",
"customer.disable.btn":"Disable", "customer.disable.btn":"Disable",
"customer.enable.btn":"Enable", "customer.enable.btn":"Enable",
"customer.chat.btn":"Chat",
"Customer End ================================================================":"", "Customer End ================================================================":"",
"Invitation Start ================================================================":"", "Invitation Start ================================================================":"",
@@ -584,10 +585,15 @@
"pickup.to_time":"To Time", "pickup.to_time":"To Time",
"pickup.status":"Status", "pickup.status":"Status",
"pickup.desc":"Description", "pickup.desc":"Description",
"pickup.remark":"Remark", "pickup.complete.remark":"Complete Remark",
"pickup.staff.name":"Staff", "pickup.staff.name":"Staff",
"pickup.edit.complete.btn":"Complete pickup", "pickup.edit.complete.btn":"Complete pickup",
"pickup.continue.btn":"Continue to complete", "pickup.continue.btn":"Continue to complete",
"pickup.confirm.complete":"Complete confirm?", "pickup.confirm.complete":"Complete confirm?",
"pickup.zone":"Pickup Zone",
"pickup.customer":"Customer",
"pickup.customer.remark":"Customer Remark",
"pickup.reschedul.remark":"Reschedule Remark",
"pickup.delivery.address":"Delivery address",
"Pickup End ===================================================================":"" "Pickup End ===================================================================":""
} }

View File

@@ -126,6 +126,7 @@
"customer.invitation.request.confirm":"လက်ခံ လိုက်ပါ", "customer.invitation.request.confirm":"လက်ခံ လိုက်ပါ",
"customer.disable.btn":"ပိတ်ပါ", "customer.disable.btn":"ပိတ်ပါ",
"customer.enable.btn":"ဖွင့်ပါ", "customer.enable.btn":"ဖွင့်ပါ",
"customer.chat.btn":"Chat",
"Customer End ================================================================":"", "Customer End ================================================================":"",
"Invitation Start ================================================================":"", "Invitation Start ================================================================":"",
@@ -582,10 +583,15 @@
"pickup.to_time":"အချိန်(တွင်း)", "pickup.to_time":"အချိန်(တွင်း)",
"pickup.status":"အခြေအနေ", "pickup.status":"အခြေအနေ",
"pickup.desc":"ဖော်ပြချက်", "pickup.desc":"ဖော်ပြချက်",
"pickup.remark":"မှတ်ချက်", "pickup.complete.remark":"Complete Remark",
"pickup.staff.name":"ဝန်ထမ်း", "pickup.staff.name":"ဝန်ထမ်း",
"pickup.edit.complete.btn":"ပို့ဆောင်ခြင်း ပြီးဆုံးသည်", "pickup.edit.complete.btn":"ပို့ဆောင်ခြင်း ပြီးဆုံးသည်",
"pickup.continue.btn":"Continue to complete", "pickup.continue.btn":"Continue to complete",
"pickup.confirm.complete":"Complete confirm?", "pickup.confirm.complete":"Complete confirm?",
"pickup.zone":"Pickup Zone",
"pickup.customer":"ဝယ်သူ",
"pickup.customer.remark":"ဝယ်သူ မှတ်ချက်",
"pickup.reschedul.remark":"Reschedule Remark",
"pickup.delivery.address":"ပို့ဆောင်ရမည့်လိပ်စာ",
"Pickup End ===================================================================":"" "Pickup End ===================================================================":""
} }

View File

@@ -17,6 +17,7 @@ import 'package:fcs/pages/main/model/language_model.dart';
import 'package:fcs/pages/main/model/main_model.dart'; import 'package:fcs/pages/main/model/main_model.dart';
import 'package:fcs/pages/package/model/package_model.dart'; import 'package:fcs/pages/package/model/package_model.dart';
import 'package:fcs/pages/payment_methods/model/payment_method_model.dart'; import 'package:fcs/pages/payment_methods/model/payment_method_model.dart';
import 'package:fcs/pages/pickup/model/pickup_model.dart';
import 'package:fcs/pages/processing/model/processing_model.dart'; import 'package:fcs/pages/processing/model/processing_model.dart';
import 'package:fcs/pages/rates/model/shipment_rate_model.dart'; import 'package:fcs/pages/rates/model/shipment_rate_model.dart';
import 'package:fcs/pages/shipment/model/shipment_model.dart'; import 'package:fcs/pages/shipment/model/shipment_model.dart';
@@ -58,6 +59,7 @@ class _AppState extends State<App> {
final DeliveryModel deliveryModel = new DeliveryModel(); final DeliveryModel deliveryModel = new DeliveryModel();
final CartonSizeModel cartonSizeModel = new CartonSizeModel(); final CartonSizeModel cartonSizeModel = new CartonSizeModel();
final ProcessingModel processingModel = new ProcessingModel(); final ProcessingModel processingModel = new ProcessingModel();
final PickupModel pickupModel = new PickupModel();
late AppTranslationsDelegate _newLocaleDelegate; late AppTranslationsDelegate _newLocaleDelegate;
@@ -79,7 +81,8 @@ class _AppState extends State<App> {
..addModel(marketModel) ..addModel(marketModel)
..addModel(deliveryModel) ..addModel(deliveryModel)
..addModel(cartonSizeModel) ..addModel(cartonSizeModel)
..addModel(processingModel); ..addModel(processingModel)
..addModel(pickupModel);
_newLocaleDelegate = AppTranslationsDelegate( _newLocaleDelegate = AppTranslationsDelegate(
newLocale: Translation().supportedLocales().first); newLocale: Translation().supportedLocales().first);
@@ -127,6 +130,7 @@ class _AppState extends State<App> {
ChangeNotifierProvider.value(value: deliveryModel), ChangeNotifierProvider.value(value: deliveryModel),
ChangeNotifierProvider.value(value: cartonSizeModel), ChangeNotifierProvider.value(value: cartonSizeModel),
ChangeNotifierProvider.value(value: processingModel), ChangeNotifierProvider.value(value: processingModel),
ChangeNotifierProvider.value(value: pickupModel),
], ],
child: Consumer<LanguageModel>( child: Consumer<LanguageModel>(
builder: (context, value, child) { builder: (context, value, child) {

View File

@@ -1,13 +1,12 @@
import 'dart:convert'; import 'dart:convert';
import 'package:fcs/config.dart';
import 'package:fcs/domain/constants.dart';
import 'package:fcs/domain/entities/package.dart';
import 'package:fcs/domain/entities/pickup.dart'; import 'package:fcs/domain/entities/pickup.dart';
import 'package:fcs/helpers/api_helper.dart'; import 'package:fcs/helpers/api_helper.dart';
import 'package:fcs/helpers/firebase_helper.dart'; import 'package:fcs/helpers/firebase_helper.dart';
import 'package:logging/logging.dart'; import 'package:logging/logging.dart';
import '../../config.dart';
class PickupDataProvider { class PickupDataProvider {
final log = Logger('PickupDataProvider'); final log = Logger('PickupDataProvider');
static final PickupDataProvider instance = PickupDataProvider._(); static final PickupDataProvider instance = PickupDataProvider._();
@@ -17,4 +16,34 @@ class PickupDataProvider {
return await requestAPI("/pickups/complete", "PUT", return await requestAPI("/pickups/complete", "PUT",
payload: pickup.toMapForComplete(), token: await getToken()); payload: pickup.toMapForComplete(), token: await getToken());
} }
Future<List<Pickup>> searchPickup(String term) async {
if (term == null || term == '') return [];
List<Pickup> pickups = [];
try {
var data = {
"filters": [
{
"field": "pickup_number",
"compare": "like",
"value": "%" + term.toUpperCase() + "%"
}
]
};
var result = await requestAPI("/api/data/pickups", "POST",
token: await getToken(),
url: Config.instance.reportURL,
payload: jsonEncode(data));
if (result == null) return pickups;
result.forEach((d) {
var package = Pickup.fromJson(d);
pickups.add(package);
});
} catch (e) {
log.warning("Error >>>>${e.toString()}");
}
return pickups;
}
} }

View File

@@ -16,4 +16,9 @@ class PickupServiceImp implements PickupService {
Future<void> completePickup(Pickup pickup) { Future<void> completePickup(Pickup pickup) {
return pickupDataProvider.completePickup(pickup); return pickupDataProvider.completePickup(pickup);
} }
@override
Future<List<Pickup>> searchPickup(String term) {
return pickupDataProvider.searchPickup(term);
}
} }

View File

@@ -1,6 +1,6 @@
import 'package:fcs/domain/entities/package.dart';
import 'package:fcs/domain/entities/pickup.dart'; import 'package:fcs/domain/entities/pickup.dart';
abstract class PickupService { abstract class PickupService {
Future<void> completePickup(Pickup pickup); Future<void> completePickup(Pickup pickup);
Future<List<Pickup>> searchPickup(String term);
} }

View File

@@ -18,6 +18,8 @@ const discounts_by_weights_collection = "discounts_by_weight";
const shipments_collection = "shipments"; const shipments_collection = "shipments";
const cartons_collection = "cartons"; const cartons_collection = "cartons";
const discounts_collection = "discounts"; const discounts_collection = "discounts";
const pickup_collection = "pickups";
// docs // docs
const setting_doc_id = "setting"; const setting_doc_id = "setting";

View File

@@ -1,24 +1,28 @@
import 'package:cloud_firestore/cloud_firestore.dart'; import 'package:cloud_firestore/cloud_firestore.dart';
import 'package:fcs/domain/entities/package.dart'; import 'package:fcs/domain/entities/package.dart';
import 'package:fcs/domain/vo/delivery_address.dart';
class Pickup { class Pickup {
String? id; String? id;
DateTime? pickupDate; DateTime? pickupDate;
String? fromTime; String? pickupNumber;
String? toTime; DateTime? fromTime;
//for consignee DateTime? toTime;
String? userID; //for customer
String? userName; String? customerID;
String? userPhoneNumber; String? customerName;
String? customerRemark;
String? staffId; String? staffId;
String? staffName; String? staffName;
String? staffPhoneNumber; String? staffPhoneNumber;
//for shipper
String? fcsID; String? fcsID;
String? shipperName;
String? shipperPhoneNumber;
String? status; String? status;
String? rescheduleRemark;
String? zoneID;
String? zoneName;
DeliveryAddress? pickupAddress;
// for complete // for complete
String? completeRemark; String? completeRemark;
@@ -28,9 +32,8 @@ class Pickup {
Pickup( Pickup(
{this.id, {this.id,
this.userID, this.customerID,
this.userName, this.customerName,
this.userPhoneNumber,
this.staffId, this.staffId,
this.staffName, this.staffName,
this.staffPhoneNumber, this.staffPhoneNumber,
@@ -38,11 +41,16 @@ class Pickup {
this.fromTime, this.fromTime,
this.toTime, this.toTime,
this.fcsID, this.fcsID,
this.shipperName,
this.shipperPhoneNumber,
this.status, this.status,
this.packages = const [], this.packages = const [],
this.photoUrls = const []}); this.photoUrls = const [],
this.completeRemark,
this.customerRemark,
this.pickupNumber,
this.rescheduleRemark,
this.zoneID,
this.zoneName,
this.pickupAddress});
@override @override
bool operator ==(Object other) => other is Pickup && other.id == id; bool operator ==(Object other) => other is Pickup && other.id == id;
@@ -50,26 +58,63 @@ class Pickup {
@override @override
int get hashCode => id.hashCode; int get hashCode => id.hashCode;
bool isChangedForEdit(Pickup Pickup) { bool isChangedForEdit(Pickup pickup) {
return Pickup.userID != this.userID || return pickup.completeRemark != this.completeRemark;
Pickup.fcsID != this.fcsID ||
Pickup.packages != this.packages;
} }
factory Pickup.fromMap(Map<String, dynamic> map, String id) { factory Pickup.fromMap(Map<String, dynamic> map, String id) {
var _pickupDate = (map['pickup_date'] as Timestamp); var _pickupDate = (map['pickup_date'] as Timestamp);
var _fromTime = map['pickup_time_from'] == null
? null
: (map['pickup_time_from'] as Timestamp);
var _toTime = map['pickup_time_to'] == null
? null
: (map['pickup_time_to'] as Timestamp);
var da = map['pickup_address'];
var _da = da != null ? DeliveryAddress.fromMap(da, da["id"]) : null;
List<String> _photoUrls =
map['photo_urls'] == null ? [] : List.from(map['photo_urls']);
return Pickup( return Pickup(
id: id, id: id,
pickupDate: _pickupDate.toDate(), pickupDate: _pickupDate.toDate().toLocal(),
fromTime: map['from_time'], pickupNumber: map['pickup_number'],
toTime: map['to_time'], pickupAddress: _da,
staffId: map['staff_id'], fromTime: _fromTime?.toDate().toLocal() ?? null,
staffName: map['staff_name'], toTime: _toTime?.toDate().toLocal() ?? null,
staffPhoneNumber: map['staff_phone_number'], staffId: map['pickup_staff_id'],
userID: map['user_id'], staffName: map['pickup_staff_name'],
userName: map['user_name'], customerID: map['customer_id'],
userPhoneNumber: map['user_phone_number'], customerName: map['customer_name'],
status: map['status']); customerRemark: map['customer_remark'] ?? "",
completeRemark: map['complete_remark'] ?? "",
rescheduleRemark: map['reschedule_remark'],
zoneID: map['zone_id'],
zoneName: map['zone_name'],
status: map['status'],
photoUrls: _photoUrls);
}
factory Pickup.fromJson(Map<String, dynamic> json) {
return Pickup(
id: json['id'],
pickupDate: DateTime.parse(json['pickup_date']),
pickupNumber: json['pickup_number'],
fromTime: DateTime.parse(json['pickup_date']),
toTime: DateTime.parse(json['pickup_date']),
staffId: json['pickup_staff_id'],
staffName: json['pickup_staff_name'],
customerID: json['customer_id'],
customerName: json['customer_name'],
customerRemark: json['customer_remark'] ?? "",
completeRemark: json['complete_remark'] ?? "",
rescheduleRemark: json['reschedule_remark'],
zoneID: json['zone_id'],
zoneName: json['zone_name'],
status: json['status'],
);
} }
Map<String, dynamic> toMap() { Map<String, dynamic> toMap() {
@@ -81,9 +126,8 @@ class Pickup {
'staff_id': staffId, 'staff_id': staffId,
'staff_name': staffName, 'staff_name': staffName,
"staff_phone_number": staffPhoneNumber, "staff_phone_number": staffPhoneNumber,
'user_id': userID, 'user_id': customerID,
'user_name': userName, 'user_name': customerName,
'user_phone_number': userPhoneNumber,
'status': status, 'status': status,
}; };
} }
@@ -92,7 +136,7 @@ class Pickup {
return { return {
"id": id, "id": id,
"complete_remark": completeRemark, "complete_remark": completeRemark,
'photo_urls': photoUrls ?? [], 'photo_urls': photoUrls,
}; };
} }

View File

@@ -44,7 +44,7 @@ Future<String> uploadStorage(String path, File? file,
} }
Reference ref = FirebaseStorage.instance.ref().child('$path/$fileName'); Reference ref = FirebaseStorage.instance.ref().child('$path/$fileName');
UploadTask uploadTask = ref.putFile(file); UploadTask uploadTask = ref.putFile(file);
await uploadTask.resume(); await uploadTask;
String downloadUrl = await ref.getDownloadURL(); String downloadUrl = await ref.getDownloadURL();
return downloadUrl; return downloadUrl;
// StorageReference storageReference = // StorageReference storageReference =

View File

@@ -15,7 +15,7 @@ Future<void> main() async {
Config( Config(
flavor: Flavor.DEV, flavor: Flavor.DEV,
color: Colors.blue, color: Colors.blue,
apiURL: "https://asia-northeast1-fcs-dev1.cloudfunctions.net/API", apiURL: "https://asia-northeast1-fcs-dev1.cloudfunctions.net/API12",
reportURL: "http://petrok.mokkon.com:8091", reportURL: "http://petrok.mokkon.com:8091",
reportProjectID: "fcs-dev", reportProjectID: "fcs-dev",
bucketName: "gs://fcs-dev1-us", bucketName: "gs://fcs-dev1-us",

View File

@@ -1,6 +1,8 @@
import 'package:fcs/domain/constants.dart'; import 'package:fcs/domain/constants.dart';
import 'package:fcs/domain/entities/user.dart'; import 'package:fcs/domain/entities/user.dart';
import 'package:fcs/helpers/theme.dart'; import 'package:fcs/helpers/theme.dart';
import 'package:fcs/pages/chat/message_detail.dart';
import 'package:fcs/pages/chat/model/message_model.dart';
import 'package:fcs/pages/customer/model/customer_model.dart'; import 'package:fcs/pages/customer/model/customer_model.dart';
import 'package:fcs/pages/main/util.dart'; import 'package:fcs/pages/main/util.dart';
import 'package:fcs/pages/widgets/display_text.dart'; import 'package:fcs/pages/widgets/display_text.dart';
@@ -51,6 +53,14 @@ class _CustomerEditorState extends State<CustomerEditor> {
color: enabled ? primaryColor : Colors.grey, color: enabled ? primaryColor : Colors.grey,
callBack: () => _enable(!enabled), callBack: () => _enable(!enabled),
); );
final chatBox = LocalButton(
textKey: "customer.chat.btn",
iconData: Icons.chat,
color: primaryColor,
callBack: () => _gotoChat(),
);
return LocalProgress( return LocalProgress(
inAsyncCall: _isLoading, inAsyncCall: _isLoading,
child: SafeArea( child: SafeArea(
@@ -100,6 +110,7 @@ class _CustomerEditorState extends State<CustomerEditor> {
context, "customer.invitation.request.confirm"), context, "customer.invitation.request.confirm"),
callack: _add) callack: _add)
: Container(), : Container(),
chatBox,
widget.customer!.joined || widget.customer!.disabled widget.customer!.joined || widget.customer!.disabled
? enableBox ? enableBox
: Container() : Container()
@@ -149,4 +160,25 @@ class _CustomerEditorState extends State<CustomerEditor> {
}); });
} }
} }
_gotoChat() {
MessageModel messageModel =
Provider.of<MessageModel>(context, listen: false);
messageModel.initQuery(widget.customer!.id);
Navigator.of(context)
.push(CupertinoPageRoute(
builder: (context) => MessageDetail(
receiverID: widget.customer!.id,
receiverName: widget.customer!.name,
messageModel: messageModel,
)))
.then((value) {
if (widget.customer!.fcsUnseenCount > 0) {
messageModel.seenMessages(widget.customer!.id ?? "", false);
}
});
if (widget.customer!.fcsUnseenCount > 0) {
messageModel.seenMessages(widget.customer!.id ?? "", false);
}
}
} }

View File

@@ -271,9 +271,7 @@ class PackageModel extends BaseModel {
package.photoUrls = package.photoUrls == null ? [] : package.photoUrls; package.photoUrls = package.photoUrls == null ? [] : package.photoUrls;
String path = Path.join(pkg_files_path); String path = Path.join(pkg_files_path);
List<String> urls = await uploadFiles(path, files); List<String> urls = await uploadFiles(path, files);
urls.forEach((url) { package.photoUrls =urls;
package.photoUrls.add(url);
});
} }
try { try {

View File

@@ -1,13 +1,10 @@
import 'dart:async'; import 'dart:async';
import 'dart:io'; import 'dart:io';
import 'dart:math';
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/constants.dart'; import 'package:fcs/domain/constants.dart';
import 'package:fcs/domain/entities/package.dart';
import 'package:fcs/domain/entities/pickup.dart'; import 'package:fcs/domain/entities/pickup.dart';
import 'package:fcs/domain/entities/processing.dart';
import 'package:fcs/helpers/firebase_helper.dart'; import 'package:fcs/helpers/firebase_helper.dart';
import 'package:fcs/pages/main/model/base_model.dart'; import 'package:fcs/pages/main/model/base_model.dart';
import 'package:fcs/pagination/paginator_listener.dart'; import 'package:fcs/pagination/paginator_listener.dart';
@@ -17,22 +14,7 @@ import 'package:path/path.dart' as Path;
class PickupModel extends BaseModel { class PickupModel extends BaseModel {
final log = Logger('PickupModel'); final log = Logger('PickupModel');
StreamSubscription<QuerySnapshot>? listener; PaginatorListener<Pickup>? pickups;
PaginatorListener<Package>? pickups;
int _menuSelectedIndex = 1;
set menuSelectedIndex(int index) {
_menuSelectedIndex = index;
// _loadPackages(_menuSelectedIndex == 2);
// _loadCustomerPackages(_menuSelectedIndex == 2);
notifyListeners();
}
int get menuSelectedIndex => _menuSelectedIndex;
void initUser(user) { void initUser(user) {
super.initUser(user); super.initUser(user);
@@ -44,13 +26,34 @@ class PickupModel extends BaseModel {
} }
} }
Future<void> _initData() async {} Future<void> _initData() async {
logout();
pickups = PaginatorListener<Pickup>((data, id) => Pickup.fromMap(data, id),
onChange: () {
notifyListeners();
}, rowPerLoad: 30, insertNewByListener: true);
_loadPickups();
}
@override @override
logout() async { logout() async {
if (listener != null) await listener!.cancel();
if (pickups != null) pickups!.close(); if (pickups != null) pickups!.close();
// pickups = []; }
Future<void> _loadPickups() async {
if (user == null) return;
String path = "/$pickup_collection";
try {
Query listenerQuery = FirebaseFirestore.instance.collection(path);
Query pageQuery = FirebaseFirestore.instance
.collection(path)
.orderBy("update_time", descending: true);
pickups!.refresh(listeningQuery: listenerQuery, pageQuery: pageQuery);
} catch (e) {
log.warning("Error!! $e");
}
} }
Future<void> complete( Future<void> complete(
@@ -82,4 +85,27 @@ class PickupModel extends BaseModel {
} }
return deleteStorageFromUrls(deletedUrls); return deleteStorageFromUrls(deletedUrls);
} }
Future<Pickup?> getPickup(String id) async {
if (user == null) return null;
String path = "/$pickup_collection";
try {
DocumentSnapshot snap =
await FirebaseFirestore.instance.collection("$path").doc(id).get();
if (snap.exists) {
var package =
Pickup.fromMap(snap.data() as Map<String, dynamic>, snap.id);
return package;
}
} catch (e) {
log.warning("Error!! $e");
}
return null;
}
Future<List<Pickup>> searchPickup(String term) async {
Future<List<Pickup>> pickups =
Services.instance.pickupService.searchPickup(term);
return pickups;
}
} }

View File

@@ -1,15 +1,15 @@
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';
import 'package:fcs/domain/entities/pickup.dart';
import 'package:fcs/domain/entities/user.dart'; import 'package:fcs/domain/entities/user.dart';
import 'package:fcs/helpers/theme.dart'; import 'package:fcs/helpers/theme.dart';
import 'package:fcs/pages/market/market_editor.dart'; import 'package:fcs/pages/market/market_editor.dart';
import 'package:fcs/pages/market/model/market_model.dart'; import 'package:fcs/pages/market/model/market_model.dart';
import 'package:fcs/pages/package/model/package_model.dart';
import 'package:fcs/pages/package/tracking_id_page.dart'; 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_serach.dart'; import 'package:fcs/pages/pickup/model/pickup_model.dart';
import 'package:fcs/pages/widgets/defalut_delivery_address.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_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,49 +22,56 @@ import 'package:intl/intl.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
class PickupEditor extends StatefulWidget { class PickupEditor extends StatefulWidget {
final Package? package; final Pickup? pickup;
PickupEditor({this.package});
const PickupEditor({Key? key, this.pickup}) : super(key: key);
@override @override
_PickupEditorState createState() => _PickupEditorState(); _PickupEditorState createState() => _PickupEditorState();
} }
class _PickupEditorState extends State<PickupEditor> { class _PickupEditorState extends State<PickupEditor> {
TextEditingController _remarkCtl = new TextEditingController(); var dateFormatter = new DateFormat('dd MMM yyyy');
TextEditingController _descCtl = new TextEditingController(); var timeFormatter = new DateFormat('h:mm a');
Package? _package; TextEditingController _remarkCtl = new TextEditingController();
User? _user; MultiImgController multiImgController = MultiImgController();
Pickup? _pickup;
bool _isLoading = false; bool _isLoading = false;
@override @override
void initState() { void initState() {
super.initState(); super.initState();
_package = widget.package; _pickup = widget.pickup;
selectedMarket = _package!.market ?? ""; multiImgController.setImageUrls = _pickup!.photoUrls;
_descCtl.text = _package!.desc ?? "";
_remarkCtl.text = _package!.remark ?? "";
multiImgController.setImageUrls = _package!.photoUrls;
_user = User(
fcsID: _package!.fcsID ?? "",
name: _package!.userName ?? "",
phoneNumber: _package!.phoneNumber ?? "");
} }
final DateFormat dateFormat = DateFormat("d MMM yyyy");
bool isNew = false;
MultiImgController multiImgController = MultiImgController();
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final pickupNumberBox = DisplayText(
text: _pickup!.pickupNumber ?? "",
labelTextKey: "pickup.pickup_number",
iconData: SimpleLineIcons.direction,
);
final pickupDateBox = DisplayText(
text: _pickup!.pickupDate == null
? ""
: dateFormatter.format(_pickup!.pickupDate!),
labelTextKey: "pickup.date",
iconData: Icons.date_range,
);
var timeBox = Row( var timeBox = Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween, mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: <Widget>[ children: <Widget>[
Container( Container(
width: 150, width: 150,
child: DisplayText( child: DisplayText(
text: '9:00 AM', text: _pickup!.fromTime == null
? ""
: timeFormatter.format(_pickup!.fromTime!),
labelTextKey: "pickup.from_time", labelTextKey: "pickup.from_time",
iconData: MaterialCommunityIcons.clock_start, iconData: MaterialCommunityIcons.clock_start,
), ),
@@ -72,42 +79,46 @@ class _PickupEditorState extends State<PickupEditor> {
Container( Container(
width: 150, width: 150,
child: DisplayText( child: DisplayText(
text: '12:00 AM', text: _pickup!.toTime == null
? ""
: timeFormatter.format(_pickup!.toTime!),
labelTextKey: "pickup.to_time", labelTextKey: "pickup.to_time",
iconData: MaterialCommunityIcons.clock_end), iconData: MaterialCommunityIcons.clock_end,
),
) )
], ],
); );
final pickupDateBox = DisplayText( final customerBox = DisplayText(
text: "12-05-2021", text: _pickup!.customerName ?? "",
labelTextKey: "pickup.date", labelTextKey: "pickup.customer",
iconData: Icons.date_range, iconData: Icons.perm_identity,
); );
final pickupNumberBox = DisplayText( final staffNameBox = DisplayText(
text: "210502-ASDFRE", text: _pickup!.staffName ?? "",
labelTextKey: "pickup.pickup_number", labelTextKey: "pickup.staff.name",
iconData: SimpleLineIcons.direction, iconData: Icons.perm_identity,
); );
final deliveryAddressBox = DefaultDeliveryAddress(
deliveryAddress: _pickup!.pickupAddress,
labelKey: "pickup.delivery.address",
onTap: null);
final completeProcessingBtn = fcsButton( final completeProcessingBtn = fcsButton(
context, context,
getLocalString(context, 'pickup.edit.complete.btn'), getLocalString(context, 'pickup.edit.complete.btn'),
callack: _confirmComplete, callack: _confirmComplete,
); );
final staffNameBox = DisplayText( final completeRemarkBox = InputText(
text: _package != null ? _package!.userName : "", labelTextKey: 'pickup.complete.remark',
labelTextKey: "pickup.staff.name",
iconData: Icons.perm_identity,
);
final remarkBox = InputText(
labelTextKey: 'pickup.remark',
iconData: Entypo.new_message, iconData: Entypo.new_message,
controller: _remarkCtl); controller: _remarkCtl);
final statusBox = DisplayText( final statusBox = DisplayText(
text: _package != null ? _package!.status : "", text: _pickup != null ? _pickup!.status : "",
labelTextKey: "pickup.status", labelTextKey: "pickup.status",
iconData: Icons.av_timer, iconData: Icons.av_timer,
); );
@@ -117,6 +128,7 @@ class _PickupEditorState extends State<PickupEditor> {
controller: multiImgController, controller: multiImgController,
title: "Receipt File", title: "Receipt File",
); );
return LocalProgress( return LocalProgress(
inAsyncCall: _isLoading, inAsyncCall: _isLoading,
child: Scaffold( child: Scaffold(
@@ -150,11 +162,13 @@ class _PickupEditorState extends State<PickupEditor> {
pickupNumberBox, pickupNumberBox,
pickupDateBox, pickupDateBox,
timeBox, timeBox,
customerBox,
staffNameBox, staffNameBox,
statusBox, statusBox,
remarkBox, deliveryAddressBox,
completeRemarkBox,
img, img,
_package!.status == 'packed' _pickup!.status == "confirmed" || _pickup!.status == "rescheduled"
? completeProcessingBtn ? completeProcessingBtn
: Container(), : Container(),
SizedBox( SizedBox(
@@ -244,22 +258,15 @@ class _PickupEditorState extends State<PickupEditor> {
} }
_completePickup() async { _completePickup() async {
if (_user!.fcsID == null || _user!.fcsID == "") {
showMsgDialog(context, "Error", "Expected FCS-ID");
return;
}
setState(() { setState(() {
_isLoading = true; _isLoading = true;
}); });
PackageModel packageModel =
Provider.of<PackageModel>(context, listen: false);
try { try {
// _package!.fcsID = _user!.fcsID; _pickup?.completeRemark = _remarkCtl.text;
// _package!.desc = _descCtl.text; await context.read<PickupModel>().complete(_pickup!,
// _package!.remark = _remarkCtl.text; multiImgController.getAddedFile, multiImgController.getDeletedUrl);
// _package!.market = selectedMarket!;
// await packageModel.updateProcessing(_package!,
// multiImgController.getAddedFile, multiImgController.getDeletedUrl);
Navigator.pop(context); Navigator.pop(context);
} catch (e) { } catch (e) {
showMsgDialog(context, "Error", e.toString()); showMsgDialog(context, "Error", e.toString());
@@ -277,23 +284,10 @@ class _PickupEditorState extends State<PickupEditor> {
} }
isDataChanged() { isDataChanged() {
if (isNew) { var _pickup = Pickup(
return _user!.fcsID != "" || completeRemark: _remarkCtl.text, photoUrls: widget.pickup!.photoUrls);
selectedMarket != null || return widget.pickup!.isChangedForEdit(_pickup) ||
_descCtl.text != "" ||
_remarkCtl.text != "" ||
multiImgController.getAddedFile.isNotEmpty;
} else {
var _package = Package(
trackingID: widget.package!.trackingID,
fcsID: _user!.fcsID,
market: selectedMarket!,
desc: _descCtl.text,
remark: _remarkCtl.text,
photoUrls: widget.package!.photoUrls);
return widget.package!.isChangedForEditProcessing(_package) ||
multiImgController.getAddedFile.isNotEmpty || multiImgController.getAddedFile.isNotEmpty ||
multiImgController.getDeletedUrl.isNotEmpty; multiImgController.getDeletedUrl.isNotEmpty;
} }
} }
}

View File

@@ -1,15 +1,14 @@
import 'package:fcs/domain/entities/package.dart'; import 'package:fcs/domain/entities/pickup.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/pickup/model/pickup_model.dart';
import 'package:fcs/pages/pickup/pickup_editor.dart'; import 'package:fcs/pages/pickup/pickup_editor.dart';
import 'package:fcs/pages/widgets/defalut_delivery_address.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_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';
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';
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_vector_icons/flutter_vector_icons.dart'; import 'package:flutter_vector_icons/flutter_vector_icons.dart';
@@ -19,8 +18,8 @@ import 'package:provider/provider.dart';
final DateFormat dateFormat = DateFormat("d MMM yyyy"); final DateFormat dateFormat = DateFormat("d MMM yyyy");
class PickupInfo extends StatefulWidget { class PickupInfo extends StatefulWidget {
final Package? package; final Pickup pickup;
PickupInfo({this.package}); const PickupInfo({Key? key, required this.pickup}) : super(key: key);
@override @override
_PickupInfoState createState() => _PickupInfoState(); _PickupInfoState createState() => _PickupInfoState();
@@ -28,21 +27,22 @@ class PickupInfo extends StatefulWidget {
class _PickupInfoState extends State<PickupInfo> { class _PickupInfoState extends State<PickupInfo> {
var dateFormatter = new DateFormat('dd MMM yyyy'); var dateFormatter = new DateFormat('dd MMM yyyy');
Package? _package; var timeFormatter = new DateFormat('h:mm a');
Pickup? _pickup;
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.pickup);
} }
initPackage(Package? package) { initPackage(Pickup pickup) {
if (package == null) return;
setState(() { setState(() {
_package = package; _pickup = pickup;
multiImgController.setImageUrls = package.photoUrls; multiImgController.setImageUrls = pickup.photoUrls;
}); });
} }
@@ -54,25 +54,64 @@ class _PickupInfoState extends State<PickupInfo> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
final pickupNumberBox = DisplayText( final pickupNumberBox = DisplayText(
text: _package != null ? '210502-ASDFRE' : '210502-ASDFRE', text: _pickup!.pickupNumber ?? "",
labelTextKey: "pickup.pickup_number", labelTextKey: "pickup.pickup_number",
iconData: SimpleLineIcons.direction, iconData: SimpleLineIcons.direction,
); );
final pickupDateBox = DisplayText(
text: _pickup!.pickupDate == null
? ""
: dateFormatter.format(_pickup!.pickupDate!),
labelTextKey: "pickup.date",
iconData: Icons.date_range,
);
var timeBox = Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: <Widget>[
Container(
width: 150,
child: DisplayText(
text: _pickup!.fromTime == null
? ""
: timeFormatter.format(_pickup!.fromTime!),
labelTextKey: "pickup.from_time",
iconData: MaterialCommunityIcons.clock_start,
),
),
Container(
width: 150,
child: DisplayText(
text: _pickup!.toTime == null
? ""
: timeFormatter.format(_pickup!.toTime!),
labelTextKey: "pickup.to_time",
iconData: MaterialCommunityIcons.clock_end,
),
)
],
);
final customerBox = DisplayText(
text: _pickup!.customerName ?? "",
labelTextKey: "pickup.customer",
iconData: Icons.perm_identity,
);
final staffNameBox = DisplayText( final staffNameBox = DisplayText(
text: _package != null ? _package!.userName : "", text: _pickup!.staffName ?? "",
labelTextKey: "pickup.staff.name", labelTextKey: "pickup.staff.name",
iconData: Icons.perm_identity, iconData: Icons.perm_identity,
); );
final remarkBox = DisplayText( final deliveryAddressBox = DefaultDeliveryAddress(
text: _package != null ? _package!.remark : "-", deliveryAddress: _pickup!.pickupAddress,
labelTextKey: "pickup.remark", labelKey: "pickup.delivery.address",
iconData: Entypo.new_message, onTap: null);
);
final statusBox = DisplayText( final statusBox = DisplayText(
text: _package != null ? _package!.status : "", text: _pickup != null ? _pickup!.status : "",
labelTextKey: "pickup.status", labelTextKey: "pickup.status",
iconData: Icons.av_timer, iconData: Icons.av_timer,
); );
@@ -83,26 +122,22 @@ class _PickupInfoState extends State<PickupInfo> {
title: "Receipt File", title: "Receipt File",
); );
var timeBox = Row( final completeRemarkBox = DisplayText(
mainAxisAlignment: MainAxisAlignment.spaceBetween, text: _pickup!.completeRemark ?? "",
children: <Widget>[ labelTextKey: "pickup.complete.remark",
Container( iconData: Entypo.new_message,
width: 150, );
child: DisplayText(
text: '9:00 AM', final rescheduleRemarkBox = DisplayText(
labelTextKey: "pickup.from_time", text: _pickup!.rescheduleRemark ?? "",
iconData: MaterialCommunityIcons.clock_start, labelTextKey: "pickup.reschedul.remark",
), iconData: Entypo.new_message,
), );
Container(
width: 150, final customerRemarkBox = DisplayText(
child: DisplayText( text: _pickup!.customerRemark ?? "",
text: '12:00 AM', labelTextKey: "pickup.customer.remark",
labelTextKey: "pickup.to_time", iconData: Entypo.new_message,
iconData: MaterialCommunityIcons.clock_end,
),
)
],
); );
final continueBtn = fcsButton( final continueBtn = fcsButton(
@@ -111,12 +146,6 @@ class _PickupInfoState extends State<PickupInfo> {
callack: _gotoEditor, callack: _gotoEditor,
); );
final pickupDateBox = DisplayText(
text: "12-05-2021",
labelTextKey: "pickup.date",
iconData: Icons.date_range,
);
return LocalProgress( return LocalProgress(
inAsyncCall: _isLoading, inAsyncCall: _isLoading,
child: Scaffold( child: Scaffold(
@@ -145,11 +174,27 @@ class _PickupInfoState extends State<PickupInfo> {
pickupNumberBox, pickupNumberBox,
pickupDateBox, pickupDateBox,
timeBox, timeBox,
customerBox,
_pickup!.completeRemark == null ||
_pickup!.completeRemark == ""
? Container()
: customerRemarkBox,
staffNameBox, staffNameBox,
remarkBox,
statusBox, statusBox,
_package!.photoUrls.length == 0 ? Container() : img, _pickup!.rescheduleRemark == null ||
_package!.status == "packed" ? continueBtn : Container(), _pickup!.rescheduleRemark == ""
? Container()
: rescheduleRemarkBox,
_pickup!.completeRemark == null ||
_pickup!.completeRemark == ""
? Container()
: completeRemarkBox,
deliveryAddressBox,
_pickup!.photoUrls.length == 0 ? Container() : img,
_pickup!.status == "confirmed" ||
_pickup!.status == "rescheduled"
? continueBtn
: Container(),
SizedBox( SizedBox(
height: 20, height: 20,
) )
@@ -162,41 +207,16 @@ class _PickupInfoState extends State<PickupInfo> {
); );
} }
_delete() {
showConfirmDialog(context, "pickup.delete.confirm", _deletePackage);
}
_deletePackage() async {
setState(() {
_isLoading = true;
});
PackageModel packageModel =
Provider.of<PackageModel>(context, listen: false);
try {
await packageModel.deleteProcessing(_package!);
Navigator.pop<bool>(context, true);
} catch (e) {
showMsgDialog(context, "Error", e.toString());
} finally {
setState(() {
_isLoading = false;
});
}
}
_gotoEditor() async { _gotoEditor() async {
bool? deleted = await Navigator.push<bool>( bool? deleted = await Navigator.push<bool>(
context, context,
CupertinoPageRoute( CupertinoPageRoute(
builder: (context) => PickupEditor( builder: (context) => PickupEditor(pickup: widget.pickup)));
package: widget.package,
)));
if (deleted ?? false) { if (deleted ?? false) {
Navigator.pop(context); Navigator.pop(context);
} else { } else {
PackageModel packageModel = Pickup? p = await context.read<PickupModel>().getPickup(_pickup!.id!);
Provider.of<PackageModel>(context, listen: false); if (p == null) return;
Package? p = await packageModel.getPackage(_package!.id!);
initPackage(p); initPackage(p);
} }
} }

View File

@@ -1,10 +1,12 @@
import 'package:fcs/domain/entities/package.dart'; import 'package:fcs/domain/entities/package.dart';
import 'package:fcs/domain/entities/pickup.dart';
import 'package:fcs/helpers/theme.dart'; import 'package:fcs/helpers/theme.dart';
import 'package:fcs/pages/package/model/package_model.dart'; import 'package:fcs/pages/package/model/package_model.dart';
import 'package:fcs/pages/package_search/package_serach.dart'; import 'package:fcs/pages/package_search/package_serach.dart';
import 'package:fcs/pages/pickup/pickup_editor.dart'; import 'package:fcs/pages/pickup/pickup_editor.dart';
import 'package:fcs/pages/pickup/pickup_info.dart'; import 'package:fcs/pages/pickup/pickup_info.dart';
import 'package:fcs/pages/pickup/pickup_list_row.dart'; import 'package:fcs/pages/pickup/pickup_list_row.dart';
import 'package:fcs/pages/pickup_search/pickup_serach.dart';
import 'package:fcs/pages/widgets/local_text.dart'; import 'package:fcs/pages/widgets/local_text.dart';
import 'package:fcs/pages/widgets/progress.dart'; import 'package:fcs/pages/widgets/progress.dart';
import 'package:fcs/pagination/paginator_listview.dart'; import 'package:fcs/pagination/paginator_listview.dart';
@@ -12,6 +14,8 @@ import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:provider/provider.dart'; import 'package:provider/provider.dart';
import 'model/pickup_model.dart';
class PickupList extends StatefulWidget { class PickupList extends StatefulWidget {
@override @override
_PickupListState createState() => _PickupListState(); _PickupListState createState() => _PickupListState();
@@ -32,8 +36,8 @@ class _PickupListState extends State<PickupList> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
var packageModel = Provider.of<PackageModel>(context); var pickupModel = Provider.of<PickupModel>(context);
var packages = packageModel.activePackages; var pickups = pickupModel.pickups;
return LocalProgress( return LocalProgress(
inAsyncCall: _isLoading, inAsyncCall: _isLoading,
@@ -58,29 +62,25 @@ class _PickupListState extends State<PickupList> {
color: Colors.white, color: Colors.white,
), ),
iconSize: 30, iconSize: 30,
onPressed: () => searchPackage(context, onPressed: () => searchPickup(context,
callbackPackageSelect: _searchCallback), callbackPickupSelect: _searchCallback),
), ),
], ],
), ),
body: PaginatorListView<Package>( body: PaginatorListView<Pickup>(
paginatorListener: packages!, paginatorListener: pickups!,
rowBuilder: (p) => PickupListRow( rowBuilder: (p) => PickupListRow(key: ValueKey(p.id), pickup: p),
key: ValueKey(p.id),
package: p,
),
color: primaryColor, color: primaryColor,
)), )),
); );
} }
_searchCallback(Package package) async { _searchCallback(Pickup pickup) async {
var packageModel = Provider.of<PackageModel>(context, listen: false); Pickup? _pickup = await context.read<PickupModel>().getPickup(pickup.id!);
Package? _package = await packageModel.getPackage(package.id!); if (_pickup == null) return;
if (_package == null) return;
Navigator.push( Navigator.push(
context, context,
CupertinoPageRoute(builder: (context) => PickupInfo(package: _package)), CupertinoPageRoute(builder: (context) => PickupInfo(pickup: _pickup)),
); );
} }
} }

View File

@@ -1,4 +1,4 @@
import 'package:fcs/domain/entities/package.dart'; import 'package:fcs/domain/entities/pickup.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/pickup/pickup_info.dart'; import 'package:fcs/pages/pickup/pickup_info.dart';
@@ -7,29 +7,28 @@ 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:intl/intl.dart'; import 'package:intl/intl.dart';
typedef CallbackPackageSelect(Package package); typedef CallbackPickupSelect(Pickup pickup);
class PickupListRow extends StatelessWidget { class PickupListRow extends StatelessWidget {
final Package? package; final Pickup? pickup;
final CallbackPackageSelect? callbackPackageSelect; final CallbackPickupSelect? callbackPickupSelect;
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");
PickupListRow({Key? key, this.package, this.callbackPackageSelect}) PickupListRow({Key? key, this.pickup, this.callbackPickupSelect})
: super(key: key); : super(key: key);
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return InkWell( return InkWell(
onTap: () { onTap: () {
if (callbackPackageSelect != null) { if (callbackPickupSelect != null) {
callbackPackageSelect!(package!); callbackPickupSelect!(pickup!);
return; return;
} }
Navigator.push( Navigator.push(
context, context,
CupertinoPageRoute( CupertinoPageRoute(builder: (context) => PickupInfo(pickup: pickup!)),
builder: (context) => PickupInfo(package: package)),
); );
}, },
child: Container( child: Container(
@@ -56,19 +55,21 @@ class PickupListRow 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!, pickup?.pickupNumber ?? '',
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.black), fontSize: 15.0, color: Colors.black),
), ),
), ),
Padding( Padding(
padding: const EdgeInsets.only(left: 8.0), padding: const EdgeInsets.only(left: 10.0, top: 10),
child: new Text( child: new Text(
package!.market == null ? '' : package!.market!, pickup!.pickupDate != null
? dateFormat.format(pickup!.pickupDate!)
: "",
style: new TextStyle( style: new TextStyle(
fontSize: 15.0, color: Colors.black), fontSize: 15.0, color: Colors.grey),
),
), ),
)
], ],
), ),
), ),
@@ -78,19 +79,7 @@ class PickupListRow extends StatelessWidget {
), ),
Column( Column(
children: <Widget>[ children: <Widget>[
Padding( getStatus(pickup!.status ?? ""),
padding: const EdgeInsets.all(3.0),
child: getStatus(package!.status ?? ""),
),
Padding(
padding: const EdgeInsets.all(0),
child: new Text(
package!.currentStatusDate != null
? dateFormat.format(package!.currentStatusDate!)
: '',
style: new TextStyle(fontSize: 15.0, color: Colors.grey),
),
),
], ],
) )
], ],

View File

@@ -0,0 +1,162 @@
import 'package:fcs/domain/entities/pickup.dart';
import 'package:fcs/helpers/theme.dart';
import 'package:fcs/pages/pickup/model/pickup_model.dart';
import 'package:fcs/pages/pickup/pickup_list_row.dart';
import 'package:fcs/pages/widgets/barcode_scanner.dart';
import 'package:flutter/material.dart';
import 'package:flutter_vector_icons/flutter_vector_icons.dart';
import 'package:provider/provider.dart';
Future<Pickup?> searchPickup(BuildContext context,
{CallbackPickupSelect? callbackPickupSelect}) async =>
await showSearch<Pickup>(
context: context,
delegate:
PackageSearchDelegate(callbackPickupSelect: callbackPickupSelect),
);
class PackageSearchDelegate extends SearchDelegate<Pickup> {
final CallbackPickupSelect? callbackPickupSelect;
PackageSearchDelegate({this.callbackPickupSelect});
@override
String get searchFieldLabel => 'Search by Pickup Number';
@override
ThemeData appBarTheme(BuildContext context) {
final ThemeData theme = Theme.of(context);
return theme.copyWith(
appBarTheme: AppBarTheme(color: primaryColor),
inputDecorationTheme: InputDecorationTheme(
border: InputBorder.none,
hintStyle: TextStyle(
color: theme.primaryTextTheme.caption?.color, fontSize: 14)),
textTheme: TextTheme(
headline1: TextStyle(
color: theme.primaryTextTheme.headline1?.color,
fontSize: 16,
backgroundColor: primaryColor)),
primaryColor: primaryColor,
);
}
@override
List<Widget> buildActions(BuildContext context) {
return [
IconButton(
icon: Icon(MaterialCommunityIcons.barcode_scan,
size: 30, color: Colors.white),
onPressed: () => _scan(context),
),
IconButton(
icon: Icon(Icons.clear),
onPressed: () => query = '',
),
];
}
@override
Widget buildLeading(BuildContext context) {
return IconButton(
icon: Icon(Icons.arrow_back),
onPressed: () => close(context, new Pickup()),
);
}
@override
Widget buildResults(BuildContext context) {
final pickupModel = Provider.of<PickupModel>(context);
return FutureBuilder(
future: pickupModel.searchPickup(query),
builder: (context, AsyncSnapshot<List<Pickup>> snapshot) {
if (snapshot.hasData) {
if (snapshot.data!.length == 0) {
return Container(
child: Center(
child: Text(
"No result found",
textAlign: TextAlign.center,
),
),
);
}
return Container(
padding: EdgeInsets.only(top: 15),
child: ListView(
children: snapshot.data!.map((e) {
return PickupListRow(
pickup: e,
callbackPickupSelect: callbackPickupSelect,
);
}).toList(),
),
);
} else if (snapshot.hasError) {
return Container(
child: Center(
child: Text(
'${snapshot.error}',
textAlign: TextAlign.center,
),
),
);
} else {
return Container(
child: Center(
child: CircularProgressIndicator(
valueColor:
new AlwaysStoppedAnimation<Color>(primaryColor)),
),
);
}
});
}
@override
Widget buildSuggestions(BuildContext context) {
return Container(
child: Center(
child: Opacity(
opacity: 0.2,
child: Icon(SimpleLineIcons.direction, size: 200, color: primaryColor)),
),
);
}
_scan(BuildContext context) async {
// PermissionStatus permission =
// await PermissionHandler().checkPermissionStatus(PermissionGroup.camera);
// if (permission != PermissionStatus.granted) {
// Map<PermissionGroup, PermissionStatus> permissions =
// await PermissionHandler()
// .requestPermissions([PermissionGroup.camera]);
// if (permissions[PermissionGroup.camera] != PermissionStatus.granted) {
// showMsgDialog(context, "Error", "Camera permission is not granted");
// return null;
// }
// }
try {
// PickedFile pickedFile =
// await ImagePicker().getImage(source: ImageSource.camera);
// FirebaseVisionImage visionImage =
// FirebaseVisionImage.fromFile(File(pickedFile.path));
// final BarcodeDetector barcodeDetector =
// FirebaseVision.instance.barcodeDetector();
// final List<Barcode> barcodes =
// await barcodeDetector.detectInImage(visionImage);
// Barcode bc = barcodes.firstWhere((element) => true);
// String barcode;
// if (bc != null) barcode = bc.rawValue;
String? barcode = await scanBarcode();
if (barcode != null) {
query = barcode;
showResults(context);
}
} catch (e) {
print('error: $e');
}
}
}

View File

@@ -73,7 +73,9 @@ class _CargoEditorState extends State<CargoEditor> {
backgroundColor: primaryColor, backgroundColor: primaryColor,
title: Text(AppTranslations.of(context)!.text("cargo.form.title")), title: Text(AppTranslations.of(context)!.text("cargo.form.title")),
actions: [ actions: [
IconButton( _isNew
? Container()
: IconButton(
icon: Icon(Icons.delete), icon: Icon(Icons.delete),
onPressed: _delete, onPressed: _delete,
) )

View File

@@ -85,7 +85,9 @@ class _CustomEditorState extends State<CustomEditor> {
title: title:
Text(AppTranslations.of(context)!.text("rate.custom.form.title")), Text(AppTranslations.of(context)!.text("rate.custom.form.title")),
actions: [ actions: [
IconButton( _isNew
? Container()
: IconButton(
icon: Icon(Icons.delete), icon: Icon(Icons.delete),
onPressed: _delete, onPressed: _delete,
) )

View File

@@ -75,7 +75,9 @@ class _DiscountByWeightEditorState extends State<DiscountByWeightEditor> {
backgroundColor: primaryColor, backgroundColor: primaryColor,
title: Text(AppTranslations.of(context)!.text("discount.new")), title: Text(AppTranslations.of(context)!.text("discount.new")),
actions: [ actions: [
IconButton( _isNew
? Container()
: IconButton(
icon: Icon(Icons.delete), icon: Icon(Icons.delete),
onPressed: _delete, onPressed: _delete,
) )