mirror of
https://github.com/inventree/inventree-app.git
synced 2025-06-16 20:25:26 +00:00
Format code
This commit is contained in:
@ -8,7 +8,6 @@ import "package:inventree/widget/snacks.dart";
|
||||
|
||||
import "package:inventree/inventree/orders.dart";
|
||||
|
||||
|
||||
class ExtraLineDetailWidget extends StatefulWidget {
|
||||
const ExtraLineDetailWidget(this.item, {Key? key}) : super(key: key);
|
||||
|
||||
@ -18,8 +17,8 @@ class ExtraLineDetailWidget extends StatefulWidget {
|
||||
_ExtraLineDetailWidgetState createState() => _ExtraLineDetailWidgetState();
|
||||
}
|
||||
|
||||
class _ExtraLineDetailWidgetState extends RefreshableState<ExtraLineDetailWidget> {
|
||||
|
||||
class _ExtraLineDetailWidgetState
|
||||
extends RefreshableState<ExtraLineDetailWidget> {
|
||||
_ExtraLineDetailWidgetState();
|
||||
|
||||
@override
|
||||
@ -30,14 +29,11 @@ class _ExtraLineDetailWidgetState extends RefreshableState<ExtraLineDetailWidget
|
||||
List<Widget> actions = [];
|
||||
|
||||
if (widget.item.canEdit) {
|
||||
actions.add(
|
||||
IconButton(
|
||||
actions.add(IconButton(
|
||||
icon: Icon(TablerIcons.edit),
|
||||
onPressed: () {
|
||||
_editLineItem(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
}));
|
||||
}
|
||||
|
||||
return actions;
|
||||
@ -53,60 +49,44 @@ class _ExtraLineDetailWidgetState extends RefreshableState<ExtraLineDetailWidget
|
||||
Future<void> _editLineItem(BuildContext context) async {
|
||||
var fields = widget.item.formFields();
|
||||
|
||||
widget.item.editForm(
|
||||
context,
|
||||
L10().editLineItem,
|
||||
fields: fields,
|
||||
widget.item.editForm(context, L10().editLineItem, fields: fields,
|
||||
onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().lineItemUpdated, success: true);
|
||||
}
|
||||
);
|
||||
refresh(context);
|
||||
showSnackIcon(L10().lineItemUpdated, success: true);
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
List<Widget> getTiles(BuildContext context) {
|
||||
List<Widget> tiles = [];
|
||||
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().reference),
|
||||
trailing: Text(widget.item.reference),
|
||||
)
|
||||
);
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().reference),
|
||||
trailing: Text(widget.item.reference),
|
||||
));
|
||||
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().description),
|
||||
trailing: Text(widget.item.description),
|
||||
)
|
||||
);
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().description),
|
||||
trailing: Text(widget.item.description),
|
||||
));
|
||||
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().quantity),
|
||||
trailing: Text(widget.item.quantity.toString()),
|
||||
)
|
||||
);
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().quantity),
|
||||
trailing: Text(widget.item.quantity.toString()),
|
||||
));
|
||||
|
||||
tiles.add(
|
||||
ListTile(
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().unitPrice),
|
||||
trailing: Text(
|
||||
renderCurrency(widget.item.price, widget.item.priceCurrency)
|
||||
)
|
||||
)
|
||||
);
|
||||
renderCurrency(widget.item.price, widget.item.priceCurrency))));
|
||||
|
||||
if (widget.item.notes.isNotEmpty) {
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().notes),
|
||||
subtitle: Text(widget.item.notes),
|
||||
)
|
||||
);
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().notes),
|
||||
subtitle: Text(widget.item.notes),
|
||||
));
|
||||
}
|
||||
|
||||
return tiles;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -9,41 +9,36 @@ import "package:inventree/widget/paginator.dart";
|
||||
import "package:inventree/widget/refreshable_state.dart";
|
||||
import "package:inventree/widget/snacks.dart";
|
||||
|
||||
|
||||
class POExtraLineListWidget extends StatefulWidget {
|
||||
|
||||
const POExtraLineListWidget(this.order, {this.filters = const {}, Key? key}) : super(key: key);
|
||||
const POExtraLineListWidget(this.order, {this.filters = const {}, Key? key})
|
||||
: super(key: key);
|
||||
|
||||
final InvenTreePurchaseOrder order;
|
||||
|
||||
final Map<String, String> filters;
|
||||
|
||||
@override
|
||||
_PurchaseOrderExtraLineListWidgetState createState() => _PurchaseOrderExtraLineListWidgetState();
|
||||
_PurchaseOrderExtraLineListWidgetState createState() =>
|
||||
_PurchaseOrderExtraLineListWidgetState();
|
||||
}
|
||||
|
||||
class _PurchaseOrderExtraLineListWidgetState extends RefreshableState<POExtraLineListWidget> {
|
||||
|
||||
class _PurchaseOrderExtraLineListWidgetState
|
||||
extends RefreshableState<POExtraLineListWidget> {
|
||||
_PurchaseOrderExtraLineListWidgetState();
|
||||
|
||||
@override
|
||||
String getAppBarTitle() => L10().extraLineItems;
|
||||
|
||||
Future<void> _addLineItem(BuildContext context) async {
|
||||
|
||||
var fields = InvenTreePOExtraLineItem().formFields();
|
||||
|
||||
fields["order"]?["value"] = widget.order.pk;
|
||||
|
||||
InvenTreePOExtraLineItem().createForm(
|
||||
context,
|
||||
L10().lineItemAdd,
|
||||
fields: fields,
|
||||
onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().lineItemUpdated, success: true);
|
||||
}
|
||||
);
|
||||
InvenTreePOExtraLineItem().createForm(context, L10().lineItemAdd,
|
||||
fields: fields, onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().lineItemUpdated, success: true);
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
@ -51,15 +46,12 @@ class _PurchaseOrderExtraLineListWidgetState extends RefreshableState<POExtraLin
|
||||
List<SpeedDialChild> actions = [];
|
||||
|
||||
if (widget.order.canEdit) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.circle_plus, color: Colors.green),
|
||||
label: L10().lineItemAdd,
|
||||
onTap: () {
|
||||
_addLineItem(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
}));
|
||||
}
|
||||
|
||||
return actions;
|
||||
@ -71,35 +63,35 @@ class _PurchaseOrderExtraLineListWidgetState extends RefreshableState<POExtraLin
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
class PaginatedPOExtraLineList extends PaginatedSearchWidget {
|
||||
|
||||
const PaginatedPOExtraLineList(Map<String, String> filters) : super(filters: filters);
|
||||
const PaginatedPOExtraLineList(Map<String, String> filters)
|
||||
: super(filters: filters);
|
||||
|
||||
@override
|
||||
String get searchTitle => L10().extraLineItems;
|
||||
|
||||
@override
|
||||
_PaginatedPOExtraLineListState createState() => _PaginatedPOExtraLineListState();
|
||||
|
||||
_PaginatedPOExtraLineListState createState() =>
|
||||
_PaginatedPOExtraLineListState();
|
||||
}
|
||||
|
||||
class _PaginatedPOExtraLineListState extends PaginatedSearchState<PaginatedPOExtraLineList> {
|
||||
|
||||
class _PaginatedPOExtraLineListState
|
||||
extends PaginatedSearchState<PaginatedPOExtraLineList> {
|
||||
_PaginatedPOExtraLineListState() : super();
|
||||
|
||||
@override
|
||||
String get prefix => "po_extra_line_";
|
||||
|
||||
@override
|
||||
Future<InvenTreePageResponse?> requestPage(int limit, int offset, Map<String, String> params) async {
|
||||
final page = await InvenTreePOExtraLineItem().listPaginated(limit, offset, filters: params);
|
||||
Future<InvenTreePageResponse?> requestPage(
|
||||
int limit, int offset, Map<String, String> params) async {
|
||||
final page = await InvenTreePOExtraLineItem()
|
||||
.listPaginated(limit, offset, filters: params);
|
||||
return page;
|
||||
}
|
||||
|
||||
@override
|
||||
Widget buildItem(BuildContext context, InvenTreeModel model) {
|
||||
|
||||
InvenTreePOExtraLineItem line = model as InvenTreePOExtraLineItem;
|
||||
|
||||
return ListTile(
|
||||
@ -113,4 +105,4 @@ class _PaginatedPOExtraLineListState extends PaginatedSearchState<PaginatedPOExt
|
||||
},
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -21,22 +21,18 @@ import "package:inventree/widget/company/supplier_part_detail.dart";
|
||||
* Widget for displaying detail view of a single PurchaseOrderLineItem
|
||||
*/
|
||||
class POLineDetailWidget extends StatefulWidget {
|
||||
|
||||
const POLineDetailWidget(this.item, {Key? key}) : super(key: key);
|
||||
|
||||
final InvenTreePOLineItem item;
|
||||
|
||||
@override
|
||||
_POLineDetailWidgetState createState() => _POLineDetailWidgetState();
|
||||
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
* State for the POLineDetailWidget
|
||||
*/
|
||||
class _POLineDetailWidgetState extends RefreshableState<POLineDetailWidget> {
|
||||
|
||||
_POLineDetailWidgetState();
|
||||
|
||||
InvenTreeStockLocation? destination;
|
||||
@ -49,14 +45,12 @@ class _POLineDetailWidgetState extends RefreshableState<POLineDetailWidget> {
|
||||
List<Widget> actions = [];
|
||||
|
||||
if (widget.item.canEdit) {
|
||||
actions.add(
|
||||
IconButton(
|
||||
icon: Icon(TablerIcons.edit),
|
||||
onPressed: () {
|
||||
_editLineItem(context);
|
||||
},
|
||||
)
|
||||
);
|
||||
actions.add(IconButton(
|
||||
icon: Icon(TablerIcons.edit),
|
||||
onPressed: () {
|
||||
_editLineItem(context);
|
||||
},
|
||||
));
|
||||
}
|
||||
|
||||
return actions;
|
||||
@ -69,15 +63,12 @@ class _POLineDetailWidgetState extends RefreshableState<POLineDetailWidget> {
|
||||
if (widget.item.canCreate) {
|
||||
// Receive items
|
||||
if (!widget.item.isComplete) {
|
||||
buttons.add(
|
||||
SpeedDialChild(
|
||||
buttons.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.transition_right, color: Colors.blue),
|
||||
label: L10().receiveItem,
|
||||
onTap: () async {
|
||||
receiveLineItem(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
}));
|
||||
}
|
||||
}
|
||||
|
||||
@ -89,7 +80,9 @@ class _POLineDetailWidgetState extends RefreshableState<POLineDetailWidget> {
|
||||
await widget.item.reload();
|
||||
|
||||
if (widget.item.destinationId > 0) {
|
||||
InvenTreeStockLocation().get(widget.item.destinationId).then((InvenTreeModel? loc) {
|
||||
InvenTreeStockLocation()
|
||||
.get(widget.item.destinationId)
|
||||
.then((InvenTreeModel? loc) {
|
||||
if (mounted) {
|
||||
if (loc != null && loc is InvenTreeStockLocation) {
|
||||
setState(() {
|
||||
@ -109,75 +102,68 @@ class _POLineDetailWidgetState extends RefreshableState<POLineDetailWidget> {
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// Callback to edit this line item
|
||||
Future<void> _editLineItem(BuildContext context) async {
|
||||
var fields = widget.item.formFields();
|
||||
|
||||
widget.item.editForm(
|
||||
context,
|
||||
L10().editLineItem,
|
||||
fields: fields,
|
||||
onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().lineItemUpdated, success: true);
|
||||
}
|
||||
);
|
||||
widget.item.editForm(context, L10().editLineItem, fields: fields,
|
||||
onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().lineItemUpdated, success: true);
|
||||
});
|
||||
}
|
||||
|
||||
// Launch a form to 'receive' this line item
|
||||
// Launch a form to 'receive' this line item
|
||||
Future<void> receiveLineItem(BuildContext context) async {
|
||||
widget.item.receive(
|
||||
context,
|
||||
onSuccess: () => {
|
||||
showSnackIcon(L10().receivedItem, success: true),
|
||||
refresh(context)
|
||||
}
|
||||
);
|
||||
widget.item.receive(context,
|
||||
onSuccess: () => {
|
||||
showSnackIcon(L10().receivedItem, success: true),
|
||||
refresh(context)
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
@override
|
||||
List<Widget> getTiles(BuildContext context) {
|
||||
List<Widget> tiles = [];
|
||||
|
||||
// Reference to the part
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().internalPart),
|
||||
subtitle: Text(widget.item.partName),
|
||||
leading: Icon(TablerIcons.box, color: COLOR_ACTION),
|
||||
trailing: api.getThumbnail(widget.item.partImage),
|
||||
onTap: () async {
|
||||
showLoadingOverlay();
|
||||
var part = await InvenTreePart().get(widget.item.partId);
|
||||
hideLoadingOverlay();
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().internalPart),
|
||||
subtitle: Text(widget.item.partName),
|
||||
leading: Icon(TablerIcons.box, color: COLOR_ACTION),
|
||||
trailing: api.getThumbnail(widget.item.partImage),
|
||||
onTap: () async {
|
||||
showLoadingOverlay();
|
||||
var part = await InvenTreePart().get(widget.item.partId);
|
||||
hideLoadingOverlay();
|
||||
|
||||
if (part is InvenTreePart) {
|
||||
part.goToDetailPage(context);
|
||||
}
|
||||
},
|
||||
)
|
||||
);
|
||||
if (part is InvenTreePart) {
|
||||
part.goToDetailPage(context);
|
||||
}
|
||||
},
|
||||
));
|
||||
|
||||
// Reference to the supplier part
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().supplierPart),
|
||||
subtitle: Text(widget.item.SKU),
|
||||
leading: Icon(TablerIcons.building, color: COLOR_ACTION),
|
||||
onTap: () async {
|
||||
showLoadingOverlay();
|
||||
var part = await InvenTreeSupplierPart().get(widget.item.supplierPartId);
|
||||
hideLoadingOverlay();
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().supplierPart),
|
||||
subtitle: Text(widget.item.SKU),
|
||||
leading: Icon(TablerIcons.building, color: COLOR_ACTION),
|
||||
onTap: () async {
|
||||
showLoadingOverlay();
|
||||
var part =
|
||||
await InvenTreeSupplierPart().get(widget.item.supplierPartId);
|
||||
hideLoadingOverlay();
|
||||
|
||||
if (part is InvenTreeSupplierPart) {
|
||||
Navigator.push(context, MaterialPageRoute(builder: (context) => SupplierPartDetailWidget(part)));
|
||||
}
|
||||
},
|
||||
)
|
||||
);
|
||||
if (part is InvenTreeSupplierPart) {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => SupplierPartDetailWidget(part)));
|
||||
}
|
||||
},
|
||||
));
|
||||
|
||||
// Destination
|
||||
if (destination != null) {
|
||||
@ -185,75 +171,57 @@ class _POLineDetailWidgetState extends RefreshableState<POLineDetailWidget> {
|
||||
title: Text(L10().destination),
|
||||
subtitle: Text(destination!.name),
|
||||
leading: Icon(TablerIcons.map_pin, color: COLOR_ACTION),
|
||||
onTap: () => {
|
||||
destination!.goToDetailPage(context)
|
||||
}
|
||||
));
|
||||
onTap: () => {destination!.goToDetailPage(context)}));
|
||||
}
|
||||
|
||||
// Received quantity
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().received),
|
||||
subtitle: ProgressBar(widget.item.progressRatio),
|
||||
trailing: Text(
|
||||
widget.item.progressString,
|
||||
style: TextStyle(
|
||||
color: widget.item.isComplete ? COLOR_SUCCESS: COLOR_WARNING
|
||||
)
|
||||
),
|
||||
leading: Icon(TablerIcons.progress),
|
||||
)
|
||||
);
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().received),
|
||||
subtitle: ProgressBar(widget.item.progressRatio),
|
||||
trailing: Text(widget.item.progressString,
|
||||
style: TextStyle(
|
||||
color: widget.item.isComplete ? COLOR_SUCCESS : COLOR_WARNING)),
|
||||
leading: Icon(TablerIcons.progress),
|
||||
));
|
||||
|
||||
// Reference
|
||||
if (widget.item.reference.isNotEmpty) {
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().reference),
|
||||
subtitle: Text(widget.item.reference),
|
||||
leading: Icon(TablerIcons.hash),
|
||||
)
|
||||
);
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().reference),
|
||||
subtitle: Text(widget.item.reference),
|
||||
leading: Icon(TablerIcons.hash),
|
||||
));
|
||||
}
|
||||
|
||||
// Pricing information
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().unitPrice),
|
||||
leading: Icon(TablerIcons.currency_dollar),
|
||||
trailing: Text(
|
||||
renderCurrency(widget.item.purchasePrice, widget.item.purchasePriceCurrency)
|
||||
),
|
||||
)
|
||||
);
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().unitPrice),
|
||||
leading: Icon(TablerIcons.currency_dollar),
|
||||
trailing: Text(renderCurrency(
|
||||
widget.item.purchasePrice, widget.item.purchasePriceCurrency)),
|
||||
));
|
||||
|
||||
// Note
|
||||
if (widget.item.notes.isNotEmpty) {
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().notes),
|
||||
subtitle: Text(widget.item.notes),
|
||||
leading: Icon(TablerIcons.note),
|
||||
)
|
||||
);
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().notes),
|
||||
subtitle: Text(widget.item.notes),
|
||||
leading: Icon(TablerIcons.note),
|
||||
));
|
||||
}
|
||||
|
||||
// External link
|
||||
if (widget.item.link.isNotEmpty) {
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().link),
|
||||
subtitle: Text(widget.item.link),
|
||||
leading: Icon(TablerIcons.link, color: COLOR_ACTION),
|
||||
onTap: () async {
|
||||
await openLink(widget.item.link);
|
||||
},
|
||||
)
|
||||
);
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().link),
|
||||
subtitle: Text(widget.item.link),
|
||||
leading: Icon(TablerIcons.link, color: COLOR_ACTION),
|
||||
onTap: () async {
|
||||
await openLink(widget.item.link);
|
||||
},
|
||||
));
|
||||
}
|
||||
|
||||
return tiles;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
@ -16,22 +16,21 @@ import "package:inventree/widget/progress.dart";
|
||||
* Paginated widget class for displaying a list of purchase order line items
|
||||
*/
|
||||
class PaginatedPOLineList extends PaginatedSearchWidget {
|
||||
|
||||
const PaginatedPOLineList(Map<String, String> filters) : super(filters: filters);
|
||||
const PaginatedPOLineList(Map<String, String> filters)
|
||||
: super(filters: filters);
|
||||
|
||||
@override
|
||||
String get searchTitle => L10().lineItems;
|
||||
|
||||
@override
|
||||
_PaginatedPOLineListState createState() => _PaginatedPOLineListState();
|
||||
|
||||
}
|
||||
|
||||
/*
|
||||
* State class for PaginatedPOLineList
|
||||
*/
|
||||
class _PaginatedPOLineListState extends PaginatedSearchState<PaginatedPOLineList> {
|
||||
|
||||
class _PaginatedPOLineListState
|
||||
extends PaginatedSearchState<PaginatedPOLineList> {
|
||||
_PaginatedPOLineListState() : super();
|
||||
|
||||
@override
|
||||
@ -39,29 +38,30 @@ class _PaginatedPOLineListState extends PaginatedSearchState<PaginatedPOLineList
|
||||
|
||||
@override
|
||||
Map<String, String> get orderingOptions => {
|
||||
"part": L10().part,
|
||||
"SKU": L10().sku,
|
||||
"quantity": L10().quantity,
|
||||
};
|
||||
"part": L10().part,
|
||||
"SKU": L10().sku,
|
||||
"quantity": L10().quantity,
|
||||
};
|
||||
|
||||
@override
|
||||
Map<String, Map<String, dynamic>> get filterOptions => {
|
||||
"pending": {
|
||||
"label": L10().outstanding,
|
||||
"help_text": L10().outstandingOrderDetail,
|
||||
"tristate": true,
|
||||
},
|
||||
"received": {
|
||||
"label": L10().received,
|
||||
"help_text": L10().receivedFilterDetail,
|
||||
"tristate": true,
|
||||
}
|
||||
};
|
||||
"pending": {
|
||||
"label": L10().outstanding,
|
||||
"help_text": L10().outstandingOrderDetail,
|
||||
"tristate": true,
|
||||
},
|
||||
"received": {
|
||||
"label": L10().received,
|
||||
"help_text": L10().receivedFilterDetail,
|
||||
"tristate": true,
|
||||
}
|
||||
};
|
||||
|
||||
@override
|
||||
Future<InvenTreePageResponse?> requestPage(int limit, int offset, Map<String, String> params) async {
|
||||
|
||||
final page = await InvenTreePOLineItem().listPaginated(limit, offset, filters: params);
|
||||
Future<InvenTreePageResponse?> requestPage(
|
||||
int limit, int offset, Map<String, String> params) async {
|
||||
final page = await InvenTreePOLineItem()
|
||||
.listPaginated(limit, offset, filters: params);
|
||||
return page;
|
||||
}
|
||||
|
||||
@ -71,24 +71,29 @@ class _PaginatedPOLineListState extends PaginatedSearchState<PaginatedPOLineList
|
||||
InvenTreeSupplierPart? supplierPart = item.supplierPart;
|
||||
|
||||
if (supplierPart != null) {
|
||||
|
||||
return ListTile(
|
||||
title: Text(supplierPart.SKU),
|
||||
subtitle: Text(item.partName),
|
||||
trailing: Text(item.progressString, style: TextStyle(color: item.isComplete ? COLOR_SUCCESS : COLOR_WARNING)),
|
||||
trailing: Text(item.progressString,
|
||||
style: TextStyle(
|
||||
color: item.isComplete ? COLOR_SUCCESS : COLOR_WARNING)),
|
||||
leading: InvenTreeAPI().getThumbnail(supplierPart.partImage),
|
||||
onTap: () async {
|
||||
showLoadingOverlay();
|
||||
await item.reload();
|
||||
hideLoadingOverlay();
|
||||
Navigator.push(context, MaterialPageRoute(builder: (context) => POLineDetailWidget(item)));
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => POLineDetailWidget(item)));
|
||||
},
|
||||
);
|
||||
} else {
|
||||
// Return an error tile
|
||||
return ListTile(
|
||||
title: Text(L10().error),
|
||||
subtitle: Text("supplier part not defined", style: TextStyle(color: COLOR_DANGER)),
|
||||
subtitle: Text("supplier part not defined",
|
||||
style: TextStyle(color: COLOR_DANGER)),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
@ -18,7 +18,6 @@ import "package:inventree/widget/order/po_extra_line_list.dart";
|
||||
import "package:inventree/widget/stock/location_display.dart";
|
||||
import "package:inventree/widget/order/po_line_list.dart";
|
||||
|
||||
|
||||
import "package:inventree/widget/attachment_widget.dart";
|
||||
import "package:inventree/widget/notes_widget.dart";
|
||||
import "package:inventree/widget/progress.dart";
|
||||
@ -27,13 +26,11 @@ import "package:inventree/widget/snacks.dart";
|
||||
import "package:inventree/widget/stock/stock_list.dart";
|
||||
import "package:inventree/preferences.dart";
|
||||
|
||||
|
||||
/*
|
||||
* Widget for viewing a single PurchaseOrder instance
|
||||
*/
|
||||
class PurchaseOrderDetailWidget extends StatefulWidget {
|
||||
|
||||
const PurchaseOrderDetailWidget(this.order, {Key? key}): super(key: key);
|
||||
const PurchaseOrderDetailWidget(this.order, {Key? key}) : super(key: key);
|
||||
|
||||
final InvenTreePurchaseOrder order;
|
||||
|
||||
@ -41,11 +38,10 @@ class PurchaseOrderDetailWidget extends StatefulWidget {
|
||||
_PurchaseOrderDetailState createState() => _PurchaseOrderDetailState();
|
||||
}
|
||||
|
||||
|
||||
class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidget> {
|
||||
|
||||
class _PurchaseOrderDetailState
|
||||
extends RefreshableState<PurchaseOrderDetailWidget> {
|
||||
_PurchaseOrderDetailState();
|
||||
|
||||
|
||||
List<InvenTreePOLineItem> lines = [];
|
||||
int extraLineCount = 0;
|
||||
|
||||
@ -73,15 +69,12 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
List<Widget> actions = [];
|
||||
|
||||
if (widget.order.canEdit) {
|
||||
actions.add(
|
||||
IconButton(
|
||||
actions.add(IconButton(
|
||||
icon: Icon(TablerIcons.edit),
|
||||
tooltip: L10().purchaseOrderEdit,
|
||||
onPressed: () {
|
||||
editOrder(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
}));
|
||||
}
|
||||
|
||||
return actions;
|
||||
@ -92,51 +85,38 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
List<SpeedDialChild> actions = [];
|
||||
|
||||
if (showCameraShortcut && widget.order.canEdit) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
child: Icon(TablerIcons.camera, color: Colors.blue),
|
||||
label: L10().takePicture,
|
||||
onTap: () async {
|
||||
_uploadImage(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.camera, color: Colors.blue),
|
||||
label: L10().takePicture,
|
||||
onTap: () async {
|
||||
_uploadImage(context);
|
||||
}));
|
||||
}
|
||||
|
||||
if (widget.order.canCreate) {
|
||||
if (widget.order.isPending) {
|
||||
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.circle_plus, color: Colors.green),
|
||||
label: L10().lineItemAdd,
|
||||
onTap: () async {
|
||||
_addLineItem(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
}));
|
||||
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.send, color: Colors.blue),
|
||||
label: L10().issueOrder,
|
||||
onTap: () async {
|
||||
_issueOrder(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
}));
|
||||
}
|
||||
|
||||
if (widget.order.isOpen) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.circle_x, color: Colors.red),
|
||||
label: L10().cancelOrder,
|
||||
onTap: () async {
|
||||
_cancelOrder(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
}));
|
||||
}
|
||||
}
|
||||
|
||||
@ -145,67 +125,53 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
|
||||
/// Add a new line item to this order
|
||||
Future<void> _addLineItem(BuildContext context) async {
|
||||
|
||||
var fields = InvenTreePOLineItem().formFields();
|
||||
|
||||
// Update part field definition
|
||||
fields["part"]?["hidden"] = false;
|
||||
fields["part"]?["filters"] = {
|
||||
"supplier": widget.order.supplierId
|
||||
};
|
||||
fields["part"]?["filters"] = {"supplier": widget.order.supplierId};
|
||||
|
||||
fields["order"]?["value"] = widget.order.pk;
|
||||
|
||||
InvenTreePOLineItem().createForm(
|
||||
context,
|
||||
L10().lineItemAdd,
|
||||
fields: fields,
|
||||
onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().lineItemUpdated, success: true);
|
||||
}
|
||||
);
|
||||
InvenTreePOLineItem().createForm(context, L10().lineItemAdd, fields: fields,
|
||||
onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().lineItemUpdated, success: true);
|
||||
});
|
||||
}
|
||||
|
||||
/// Upload an image against the current PurchaseOrder
|
||||
Future<void> _uploadImage(BuildContext context) async {
|
||||
|
||||
InvenTreePurchaseOrderAttachment().uploadImage(
|
||||
widget.order.pk,
|
||||
prefix: widget.order.reference,
|
||||
).then((result) => refresh(context));
|
||||
InvenTreePurchaseOrderAttachment()
|
||||
.uploadImage(
|
||||
widget.order.pk,
|
||||
prefix: widget.order.reference,
|
||||
)
|
||||
.then((result) => refresh(context));
|
||||
}
|
||||
|
||||
/// Issue this order
|
||||
Future<void> _issueOrder(BuildContext context) async {
|
||||
|
||||
confirmationDialog(
|
||||
L10().issueOrder, "",
|
||||
icon: TablerIcons.send,
|
||||
color: Colors.blue,
|
||||
acceptText: L10().issue,
|
||||
onAccept: () async {
|
||||
widget.order.issueOrder().then((dynamic) {
|
||||
refresh(context);
|
||||
});
|
||||
}
|
||||
);
|
||||
confirmationDialog(L10().issueOrder, "",
|
||||
icon: TablerIcons.send,
|
||||
color: Colors.blue,
|
||||
acceptText: L10().issue, onAccept: () async {
|
||||
widget.order.issueOrder().then((dynamic) {
|
||||
refresh(context);
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
/// Cancel this order
|
||||
Future<void> _cancelOrder(BuildContext context) async {
|
||||
|
||||
confirmationDialog(
|
||||
L10().cancelOrder, "",
|
||||
icon: TablerIcons.circle_x,
|
||||
color: Colors.red,
|
||||
acceptText: L10().cancel,
|
||||
onAccept: () async {
|
||||
widget.order.cancelOrder().then((dynamic) {
|
||||
refresh(context);
|
||||
});
|
||||
}
|
||||
);
|
||||
confirmationDialog(L10().cancelOrder, "",
|
||||
icon: TablerIcons.circle_x,
|
||||
color: Colors.red,
|
||||
acceptText: L10().cancel, onAccept: () async {
|
||||
widget.order.cancelOrder().then((dynamic) {
|
||||
refresh(context);
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
@ -213,25 +179,22 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
List<SpeedDialChild> actions = [];
|
||||
|
||||
if (api.supportsBarcodePOReceiveEndpoint && widget.order.isPlaced) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
child: Icon(Icons.barcode_reader),
|
||||
label: L10().scanReceivedParts,
|
||||
onTap:() async {
|
||||
scanBarcode(
|
||||
context,
|
||||
handler: POReceiveBarcodeHandler(purchaseOrder: widget.order),
|
||||
).then((value) {
|
||||
refresh(context);
|
||||
});
|
||||
},
|
||||
)
|
||||
);
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(Icons.barcode_reader),
|
||||
label: L10().scanReceivedParts,
|
||||
onTap: () async {
|
||||
scanBarcode(
|
||||
context,
|
||||
handler: POReceiveBarcodeHandler(purchaseOrder: widget.order),
|
||||
).then((value) {
|
||||
refresh(context);
|
||||
});
|
||||
},
|
||||
));
|
||||
}
|
||||
|
||||
if (widget.order.isPending && api.supportsBarcodePOAddLineEndpoint) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.circle_plus, color: COLOR_SUCCESS),
|
||||
label: L10().lineItemAdd,
|
||||
onTap: () async {
|
||||
@ -239,15 +202,12 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
context,
|
||||
handler: POAllocateBarcodeHandler(purchaseOrder: widget.order),
|
||||
);
|
||||
}
|
||||
)
|
||||
);
|
||||
}));
|
||||
}
|
||||
|
||||
return actions;
|
||||
}
|
||||
|
||||
|
||||
@override
|
||||
Future<void> request(BuildContext context) async {
|
||||
await widget.order.reload();
|
||||
@ -256,8 +216,11 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
|
||||
lines = await widget.order.getLineItems();
|
||||
|
||||
showCameraShortcut = await InvenTreeSettingsManager().getBool(INV_PO_SHOW_CAMERA, true);
|
||||
supportProjectCodes = api.supportsProjectCodes && await api.getGlobalBooleanSetting("PROJECT_CODES_ENABLED", backup: true);
|
||||
showCameraShortcut =
|
||||
await InvenTreeSettingsManager().getBool(INV_PO_SHOW_CAMERA, true);
|
||||
supportProjectCodes = api.supportsProjectCodes &&
|
||||
await api.getGlobalBooleanSetting("PROJECT_CODES_ENABLED",
|
||||
backup: true);
|
||||
|
||||
completedLines = 0;
|
||||
|
||||
@ -267,7 +230,9 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
}
|
||||
}
|
||||
|
||||
InvenTreePurchaseOrderAttachment().countAttachments(widget.order.pk).then((int value) {
|
||||
InvenTreePurchaseOrderAttachment()
|
||||
.countAttachments(widget.order.pk)
|
||||
.then((int value) {
|
||||
if (mounted) {
|
||||
setState(() {
|
||||
attachmentCount = value;
|
||||
@ -275,8 +240,11 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
}
|
||||
});
|
||||
|
||||
if (api.supportsPurchaseOrderDestination && widget.order.destinationId > 0) {
|
||||
InvenTreeStockLocation().get(widget.order.destinationId).then((InvenTreeModel? loc) {
|
||||
if (api.supportsPurchaseOrderDestination &&
|
||||
widget.order.destinationId > 0) {
|
||||
InvenTreeStockLocation()
|
||||
.get(widget.order.destinationId)
|
||||
.then((InvenTreeModel? loc) {
|
||||
if (mounted) {
|
||||
if (loc != null && loc is InvenTreeStockLocation) {
|
||||
setState(() {
|
||||
@ -298,7 +266,8 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
}
|
||||
|
||||
// Count number of "extra line items" against this order
|
||||
InvenTreePOExtraLineItem().count(filters: {"order": widget.order.pk.toString() }).then((int value) {
|
||||
InvenTreePOExtraLineItem().count(
|
||||
filters: {"order": widget.order.pk.toString()}).then((int value) {
|
||||
if (mounted) {
|
||||
setState(() {
|
||||
extraLineCount = value;
|
||||
@ -308,7 +277,7 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
}
|
||||
|
||||
// Edit the currently displayed PurchaseOrder
|
||||
Future <void> editOrder(BuildContext context) async {
|
||||
Future<void> editOrder(BuildContext context) async {
|
||||
var fields = widget.order.formFields();
|
||||
|
||||
// Cannot edit supplier field from here
|
||||
@ -324,39 +293,30 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
fields.remove("project_code");
|
||||
}
|
||||
|
||||
widget.order.editForm(
|
||||
context,
|
||||
L10().purchaseOrderEdit,
|
||||
fields: fields,
|
||||
onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().purchaseOrderUpdated, success: true);
|
||||
}
|
||||
);
|
||||
widget.order.editForm(context, L10().purchaseOrderEdit, fields: fields,
|
||||
onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().purchaseOrderUpdated, success: true);
|
||||
});
|
||||
}
|
||||
|
||||
Widget headerTile(BuildContext context) {
|
||||
|
||||
InvenTreeCompany? supplier = widget.order.supplier;
|
||||
|
||||
return Card(
|
||||
child: ListTile(
|
||||
title: Text(widget.order.reference),
|
||||
subtitle: Text(widget.order.description),
|
||||
leading: supplier == null ? null : api.getThumbnail(supplier.thumbnail),
|
||||
trailing: Text(
|
||||
api.PurchaseOrderStatus.label(widget.order.status),
|
||||
style: TextStyle(
|
||||
color: api.PurchaseOrderStatus.color(widget.order.status)
|
||||
),
|
||||
)
|
||||
)
|
||||
);
|
||||
|
||||
title: Text(widget.order.reference),
|
||||
subtitle: Text(widget.order.description),
|
||||
leading:
|
||||
supplier == null ? null : api.getThumbnail(supplier.thumbnail),
|
||||
trailing: Text(
|
||||
api.PurchaseOrderStatus.label(widget.order.status),
|
||||
style: TextStyle(
|
||||
color: api.PurchaseOrderStatus.color(widget.order.status)),
|
||||
)));
|
||||
}
|
||||
|
||||
List<Widget> orderTiles(BuildContext context) {
|
||||
|
||||
List<Widget> tiles = [];
|
||||
|
||||
InvenTreeCompany? supplier = widget.order.supplier;
|
||||
@ -366,7 +326,8 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
if (supportProjectCodes && widget.order.hasProjectCode) {
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().projectCode),
|
||||
subtitle: Text("${widget.order.projectCode} - ${widget.order.projectCodeDescription}"),
|
||||
subtitle: Text(
|
||||
"${widget.order.projectCode} - ${widget.order.projectCodeDescription}"),
|
||||
leading: Icon(TablerIcons.list),
|
||||
));
|
||||
}
|
||||
@ -393,21 +354,21 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
// Order destination
|
||||
if (destination != null) {
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().destination),
|
||||
subtitle: Text(destination!.name),
|
||||
leading: Icon(TablerIcons.map_pin, color: COLOR_ACTION),
|
||||
onTap: () => {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => LocationDisplayWidget(destination)
|
||||
)
|
||||
)
|
||||
}
|
||||
));
|
||||
title: Text(L10().destination),
|
||||
subtitle: Text(destination!.name),
|
||||
leading: Icon(TablerIcons.map_pin, color: COLOR_ACTION),
|
||||
onTap: () => {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) =>
|
||||
LocationDisplayWidget(destination)))
|
||||
}));
|
||||
}
|
||||
|
||||
Color lineColor = completedLines < widget.order.lineItemCount ? COLOR_WARNING : COLOR_SUCCESS;
|
||||
Color lineColor = completedLines < widget.order.lineItemCount
|
||||
? COLOR_WARNING
|
||||
: COLOR_SUCCESS;
|
||||
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().lineItems),
|
||||
@ -416,7 +377,8 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
maximum: widget.order.lineItemCount.toDouble(),
|
||||
),
|
||||
leading: Icon(TablerIcons.clipboard_check),
|
||||
trailing: Text("${completedLines} / ${widget.order.lineItemCount}", style: TextStyle(color: lineColor)),
|
||||
trailing: Text("${completedLines} / ${widget.order.lineItemCount}",
|
||||
style: TextStyle(color: lineColor)),
|
||||
));
|
||||
|
||||
// Extra line items
|
||||
@ -426,20 +388,18 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
trailing: Text(extraLineCount.toString()),
|
||||
onTap: () => {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => POExtraLineListWidget(widget.order, filters: {"order": widget.order.pk.toString()})
|
||||
)
|
||||
)
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => POExtraLineListWidget(widget.order,
|
||||
filters: {"order": widget.order.pk.toString()})))
|
||||
},
|
||||
));
|
||||
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().totalPrice),
|
||||
leading: Icon(TablerIcons.currency_dollar),
|
||||
trailing: Text(
|
||||
renderCurrency(widget.order.totalPrice, widget.order.totalPriceCurrency)
|
||||
),
|
||||
trailing: Text(renderCurrency(
|
||||
widget.order.totalPrice, widget.order.totalPriceCurrency)),
|
||||
));
|
||||
|
||||
if (widget.order.issueDate.isNotEmpty) {
|
||||
@ -475,54 +435,44 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
}
|
||||
|
||||
// Responsible "owner"
|
||||
if (widget.order.responsibleName.isNotEmpty && widget.order.responsibleLabel.isNotEmpty) {
|
||||
if (widget.order.responsibleName.isNotEmpty &&
|
||||
widget.order.responsibleLabel.isNotEmpty) {
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().responsible),
|
||||
leading: Icon(widget.order.responsibleLabel == "group" ? TablerIcons.users : TablerIcons.user),
|
||||
trailing: Text(widget.order.responsibleName)
|
||||
));
|
||||
title: Text(L10().responsible),
|
||||
leading: Icon(widget.order.responsibleLabel == "group"
|
||||
? TablerIcons.users
|
||||
: TablerIcons.user),
|
||||
trailing: Text(widget.order.responsibleName)));
|
||||
}
|
||||
|
||||
// Notes tile
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().notes),
|
||||
leading: Icon(TablerIcons.note, color: COLOR_ACTION),
|
||||
onTap: () {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => NotesWidget(widget.order)
|
||||
)
|
||||
);
|
||||
},
|
||||
)
|
||||
);
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().notes),
|
||||
leading: Icon(TablerIcons.note, color: COLOR_ACTION),
|
||||
onTap: () {
|
||||
Navigator.push(context,
|
||||
MaterialPageRoute(builder: (context) => NotesWidget(widget.order)));
|
||||
},
|
||||
));
|
||||
|
||||
// Attachments
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().attachments),
|
||||
leading: Icon(TablerIcons.file, color: COLOR_ACTION),
|
||||
trailing: attachmentCount > 0 ? Text(attachmentCount.toString()) : null,
|
||||
onTap: () {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().attachments),
|
||||
leading: Icon(TablerIcons.file, color: COLOR_ACTION),
|
||||
trailing: attachmentCount > 0 ? Text(attachmentCount.toString()) : null,
|
||||
onTap: () {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => AttachmentWidget(
|
||||
InvenTreePurchaseOrderAttachment(),
|
||||
widget.order.pk,
|
||||
widget.order.reference,
|
||||
widget.order.canEdit
|
||||
)
|
||||
)
|
||||
);
|
||||
},
|
||||
)
|
||||
);
|
||||
widget.order.canEdit)));
|
||||
},
|
||||
));
|
||||
|
||||
return tiles;
|
||||
|
||||
}
|
||||
|
||||
@override
|
||||
@ -533,7 +483,7 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg
|
||||
Tab(text: L10().received)
|
||||
];
|
||||
}
|
||||
|
||||
|
||||
@override
|
||||
List<Widget> getTabs(BuildContext context) {
|
||||
return [
|
||||
|
@ -16,18 +16,18 @@ import "package:inventree/inventree/purchase_order.dart";
|
||||
* Widget class for displaying a list of Purchase Orders
|
||||
*/
|
||||
class PurchaseOrderListWidget extends StatefulWidget {
|
||||
|
||||
const PurchaseOrderListWidget({this.filters = const {}, Key? key}) : super(key: key);
|
||||
const PurchaseOrderListWidget({this.filters = const {}, Key? key})
|
||||
: super(key: key);
|
||||
|
||||
final Map<String, String> filters;
|
||||
|
||||
@override
|
||||
_PurchaseOrderListWidgetState createState() => _PurchaseOrderListWidgetState();
|
||||
_PurchaseOrderListWidgetState createState() =>
|
||||
_PurchaseOrderListWidgetState();
|
||||
}
|
||||
|
||||
|
||||
class _PurchaseOrderListWidgetState extends RefreshableState<PurchaseOrderListWidget> {
|
||||
|
||||
class _PurchaseOrderListWidgetState
|
||||
extends RefreshableState<PurchaseOrderListWidget> {
|
||||
_PurchaseOrderListWidgetState();
|
||||
|
||||
@override
|
||||
@ -38,15 +38,12 @@ class _PurchaseOrderListWidgetState extends RefreshableState<PurchaseOrderListWi
|
||||
List<SpeedDialChild> actions = [];
|
||||
|
||||
if (InvenTreePurchaseOrder().canCreate) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.circle_plus),
|
||||
label: L10().purchaseOrderCreate,
|
||||
onTap: () {
|
||||
_createPurchaseOrder(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
}));
|
||||
}
|
||||
|
||||
return actions;
|
||||
@ -59,19 +56,15 @@ class _PurchaseOrderListWidgetState extends RefreshableState<PurchaseOrderListWi
|
||||
// Cannot set contact until company is locked in
|
||||
fields.remove("contact");
|
||||
|
||||
InvenTreePurchaseOrder().createForm(
|
||||
context,
|
||||
L10().purchaseOrderCreate,
|
||||
fields: fields,
|
||||
onSuccess: (result) async {
|
||||
Map<String, dynamic> data = result as Map<String, dynamic>;
|
||||
InvenTreePurchaseOrder().createForm(context, L10().purchaseOrderCreate,
|
||||
fields: fields, onSuccess: (result) async {
|
||||
Map<String, dynamic> data = result as Map<String, dynamic>;
|
||||
|
||||
if (data.containsKey("pk")) {
|
||||
var order = InvenTreePurchaseOrder.fromJson(data);
|
||||
order.goToDetailPage(context);
|
||||
}
|
||||
if (data.containsKey("pk")) {
|
||||
var order = InvenTreePurchaseOrder.fromJson(data);
|
||||
order.goToDetailPage(context);
|
||||
}
|
||||
);
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
@ -79,18 +72,16 @@ class _PurchaseOrderListWidgetState extends RefreshableState<PurchaseOrderListWi
|
||||
List<SpeedDialChild> actions = [];
|
||||
|
||||
if (api.supportsBarcodePOReceiveEndpoint) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
child: Icon(Icons.barcode_reader),
|
||||
label: L10().scanReceivedParts,
|
||||
onTap:() async {
|
||||
scanBarcode(
|
||||
context,
|
||||
handler: POReceiveBarcodeHandler(),
|
||||
);
|
||||
},
|
||||
)
|
||||
);
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(Icons.barcode_reader),
|
||||
label: L10().scanReceivedParts,
|
||||
onTap: () async {
|
||||
scanBarcode(
|
||||
context,
|
||||
handler: POReceiveBarcodeHandler(),
|
||||
);
|
||||
},
|
||||
));
|
||||
}
|
||||
|
||||
return actions;
|
||||
@ -102,22 +93,20 @@ class _PurchaseOrderListWidgetState extends RefreshableState<PurchaseOrderListWi
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
class PaginatedPurchaseOrderList extends PaginatedSearchWidget {
|
||||
|
||||
const PaginatedPurchaseOrderList(Map<String, String> filters) : super(filters: filters);
|
||||
const PaginatedPurchaseOrderList(Map<String, String> filters)
|
||||
: super(filters: filters);
|
||||
|
||||
@override
|
||||
String get searchTitle => L10().purchaseOrders;
|
||||
|
||||
@override
|
||||
_PaginatedPurchaseOrderListState createState() => _PaginatedPurchaseOrderListState();
|
||||
|
||||
_PaginatedPurchaseOrderListState createState() =>
|
||||
_PaginatedPurchaseOrderListState();
|
||||
}
|
||||
|
||||
|
||||
class _PaginatedPurchaseOrderListState extends PaginatedSearchState<PaginatedPurchaseOrderList> {
|
||||
|
||||
class _PaginatedPurchaseOrderListState
|
||||
extends PaginatedSearchState<PaginatedPurchaseOrderList> {
|
||||
_PaginatedPurchaseOrderListState() : super();
|
||||
|
||||
@override
|
||||
@ -125,51 +114,53 @@ class _PaginatedPurchaseOrderListState extends PaginatedSearchState<PaginatedPur
|
||||
|
||||
@override
|
||||
Map<String, String> get orderingOptions => {
|
||||
"reference": L10().reference,
|
||||
"supplier__name": L10().supplier,
|
||||
"status": L10().status,
|
||||
"target_date": L10().targetDate,
|
||||
};
|
||||
"reference": L10().reference,
|
||||
"supplier__name": L10().supplier,
|
||||
"status": L10().status,
|
||||
"target_date": L10().targetDate,
|
||||
};
|
||||
|
||||
@override
|
||||
Map<String, Map<String, dynamic>> get filterOptions => {
|
||||
"outstanding": {
|
||||
"label": L10().outstanding,
|
||||
"help_text": L10().outstandingOrderDetail,
|
||||
"tristate": true,
|
||||
},
|
||||
"overdue": {
|
||||
"label": L10().overdue,
|
||||
"help_text": L10().overdueDetail,
|
||||
"tristate": true,
|
||||
},
|
||||
"assigned_to_me": {
|
||||
"label": L10().assignedToMe,
|
||||
"help_text": L10().assignedToMeDetail,
|
||||
"tristate": true,
|
||||
}
|
||||
};
|
||||
"outstanding": {
|
||||
"label": L10().outstanding,
|
||||
"help_text": L10().outstandingOrderDetail,
|
||||
"tristate": true,
|
||||
},
|
||||
"overdue": {
|
||||
"label": L10().overdue,
|
||||
"help_text": L10().overdueDetail,
|
||||
"tristate": true,
|
||||
},
|
||||
"assigned_to_me": {
|
||||
"label": L10().assignedToMe,
|
||||
"help_text": L10().assignedToMeDetail,
|
||||
"tristate": true,
|
||||
}
|
||||
};
|
||||
|
||||
@override
|
||||
Future<InvenTreePageResponse?> requestPage(int limit, int offset, Map<String, String> params) async {
|
||||
|
||||
Future<InvenTreePageResponse?> requestPage(
|
||||
int limit, int offset, Map<String, String> params) async {
|
||||
await InvenTreeAPI().PurchaseOrderStatus.load();
|
||||
final page = await InvenTreePurchaseOrder().listPaginated(limit, offset, filters: params);
|
||||
final page = await InvenTreePurchaseOrder()
|
||||
.listPaginated(limit, offset, filters: params);
|
||||
|
||||
return page;
|
||||
}
|
||||
|
||||
@override
|
||||
Widget buildItem(BuildContext context, InvenTreeModel model) {
|
||||
|
||||
InvenTreePurchaseOrder order = model as InvenTreePurchaseOrder;
|
||||
|
||||
InvenTreeCompany? supplier = order.supplier;
|
||||
|
||||
|
||||
return ListTile(
|
||||
title: Text(order.reference),
|
||||
subtitle: Text(order.description),
|
||||
leading: supplier == null ? null : InvenTreeAPI().getThumbnail(supplier.thumbnail),
|
||||
leading: supplier == null
|
||||
? null
|
||||
: InvenTreeAPI().getThumbnail(supplier.thumbnail),
|
||||
trailing: Text(
|
||||
InvenTreeAPI().PurchaseOrderStatus.label(order.status),
|
||||
style: TextStyle(
|
||||
@ -181,4 +172,4 @@ class _PaginatedPurchaseOrderListState extends PaginatedSearchState<PaginatedPur
|
||||
},
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1,4 +1,3 @@
|
||||
|
||||
import "package:flutter/material.dart";
|
||||
import "package:flutter_speed_dial/flutter_speed_dial.dart";
|
||||
import "package:flutter_tabler_icons/flutter_tabler_icons.dart";
|
||||
@ -25,7 +24,6 @@ import "package:inventree/widget/progress.dart";
|
||||
* Widget for viewing a single SalesOrder instance
|
||||
*/
|
||||
class SalesOrderDetailWidget extends StatefulWidget {
|
||||
|
||||
const SalesOrderDetailWidget(this.order, {Key? key}) : super(key: key);
|
||||
|
||||
final InvenTreeSalesOrder order;
|
||||
@ -34,9 +32,7 @@ class SalesOrderDetailWidget extends StatefulWidget {
|
||||
_SalesOrderDetailState createState() => _SalesOrderDetailState();
|
||||
}
|
||||
|
||||
|
||||
class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
|
||||
_SalesOrderDetailState();
|
||||
|
||||
List<InvenTreeSOLineItem> lines = [];
|
||||
@ -62,14 +58,12 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
List<Widget> actions = [];
|
||||
|
||||
if (widget.order.canEdit) {
|
||||
actions.add(
|
||||
IconButton(
|
||||
icon: Icon(TablerIcons.edit),
|
||||
onPressed: () {
|
||||
editOrder(context);
|
||||
},
|
||||
)
|
||||
);
|
||||
actions.add(IconButton(
|
||||
icon: Icon(TablerIcons.edit),
|
||||
onPressed: () {
|
||||
editOrder(context);
|
||||
},
|
||||
));
|
||||
}
|
||||
|
||||
return actions;
|
||||
@ -77,21 +71,15 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
|
||||
// Add a new shipment against this sales order
|
||||
Future<void> _addShipment(BuildContext context) async {
|
||||
|
||||
var fields = InvenTreeSalesOrderShipment().formFields();
|
||||
|
||||
fields["order"]?["value"] = widget.order.pk;
|
||||
fields["order"]?["hidden"] = true;
|
||||
|
||||
InvenTreeSalesOrderShipment().createForm(
|
||||
context,
|
||||
L10().shipmentAdd,
|
||||
fields: fields,
|
||||
onSuccess: (result) async {
|
||||
refresh(context);
|
||||
}
|
||||
);
|
||||
|
||||
InvenTreeSalesOrderShipment().createForm(context, L10().shipmentAdd,
|
||||
fields: fields, onSuccess: (result) async {
|
||||
refresh(context);
|
||||
});
|
||||
}
|
||||
|
||||
// Add a new line item to this sales order
|
||||
@ -101,54 +89,44 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
fields["order"]?["value"] = widget.order.pk;
|
||||
fields["order"]?["hidden"] = true;
|
||||
|
||||
InvenTreeSOLineItem().createForm(
|
||||
context,
|
||||
L10().lineItemAdd,
|
||||
fields: fields,
|
||||
InvenTreeSOLineItem().createForm(context, L10().lineItemAdd, fields: fields,
|
||||
onSuccess: (result) async {
|
||||
refresh(context);
|
||||
}
|
||||
);
|
||||
refresh(context);
|
||||
});
|
||||
}
|
||||
|
||||
/// Upload an image for this order
|
||||
Future<void> _uploadImage(BuildContext context) async {
|
||||
InvenTreeSalesOrderAttachment().uploadImage(
|
||||
widget.order.pk,
|
||||
prefix: widget.order.reference,
|
||||
).then((result) => refresh(context));
|
||||
InvenTreeSalesOrderAttachment()
|
||||
.uploadImage(
|
||||
widget.order.pk,
|
||||
prefix: widget.order.reference,
|
||||
)
|
||||
.then((result) => refresh(context));
|
||||
}
|
||||
|
||||
/// Issue this order
|
||||
Future<void> _issueOrder(BuildContext context) async {
|
||||
|
||||
confirmationDialog(
|
||||
L10().issueOrder, "",
|
||||
confirmationDialog(L10().issueOrder, "",
|
||||
icon: TablerIcons.send,
|
||||
color: Colors.blue,
|
||||
acceptText: L10().issue,
|
||||
onAccept: () async {
|
||||
widget.order.issueOrder().then((dynamic) {
|
||||
refresh(context);
|
||||
});
|
||||
}
|
||||
);
|
||||
acceptText: L10().issue, onAccept: () async {
|
||||
widget.order.issueOrder().then((dynamic) {
|
||||
refresh(context);
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
/// Cancel this order
|
||||
Future<void> _cancelOrder(BuildContext context) async {
|
||||
|
||||
confirmationDialog(
|
||||
L10().cancelOrder, "",
|
||||
confirmationDialog(L10().cancelOrder, "",
|
||||
icon: TablerIcons.circle_x,
|
||||
color: Colors.red,
|
||||
acceptText: L10().cancel,
|
||||
onAccept: () async {
|
||||
await widget.order.cancelOrder().then((dynamic) {
|
||||
refresh(context);
|
||||
});
|
||||
}
|
||||
);
|
||||
acceptText: L10().cancel, onAccept: () async {
|
||||
await widget.order.cancelOrder().then((dynamic) {
|
||||
refresh(context);
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
@ -156,62 +134,48 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
List<SpeedDialChild> actions = [];
|
||||
|
||||
if (showCameraShortcut && widget.order.canEdit) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
child: Icon(TablerIcons.camera, color: Colors.blue),
|
||||
label: L10().takePicture,
|
||||
onTap: () async {
|
||||
_uploadImage(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.camera, color: Colors.blue),
|
||||
label: L10().takePicture,
|
||||
onTap: () async {
|
||||
_uploadImage(context);
|
||||
}));
|
||||
}
|
||||
|
||||
if (widget.order.isPending) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
child: Icon(TablerIcons.send, color: Colors.blue),
|
||||
label: L10().issueOrder,
|
||||
onTap: () async {
|
||||
_issueOrder(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.send, color: Colors.blue),
|
||||
label: L10().issueOrder,
|
||||
onTap: () async {
|
||||
_issueOrder(context);
|
||||
}));
|
||||
}
|
||||
|
||||
if (widget.order.isOpen) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
child: Icon(TablerIcons.circle_x, color: Colors.red),
|
||||
label: L10().cancelOrder,
|
||||
onTap: () async {
|
||||
_cancelOrder(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.circle_x, color: Colors.red),
|
||||
label: L10().cancelOrder,
|
||||
onTap: () async {
|
||||
_cancelOrder(context);
|
||||
}));
|
||||
}
|
||||
|
||||
// Add line item
|
||||
if ((widget.order.isPending || widget.order.isInProgress) && InvenTreeSOLineItem().canCreate) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
if ((widget.order.isPending || widget.order.isInProgress) &&
|
||||
InvenTreeSOLineItem().canCreate) {
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.circle_plus, color: Colors.green),
|
||||
label: L10().lineItemAdd,
|
||||
onTap: () async {
|
||||
_addLineItem(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
}));
|
||||
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.circle_plus, color: Colors.green),
|
||||
label: L10().shipmentAdd,
|
||||
onTap: () async {
|
||||
_addShipment(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
}));
|
||||
}
|
||||
|
||||
return actions;
|
||||
@ -221,9 +185,9 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
List<SpeedDialChild> barcodeButtons(BuildContext context) {
|
||||
List<SpeedDialChild> actions = [];
|
||||
|
||||
if ((widget.order.isInProgress || widget.order.isPending) && InvenTreeSOLineItem().canCreate) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
if ((widget.order.isInProgress || widget.order.isPending) &&
|
||||
InvenTreeSOLineItem().canCreate) {
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(Icons.barcode_reader),
|
||||
label: L10().lineItemAdd,
|
||||
onTap: () async {
|
||||
@ -231,25 +195,18 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
context,
|
||||
handler: SOAddItemBarcodeHandler(salesOrder: widget.order),
|
||||
);
|
||||
}
|
||||
)
|
||||
);
|
||||
}));
|
||||
|
||||
if (api.supportsBarcodeSOAllocateEndpoint) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.transition_right),
|
||||
label: L10().allocateStock,
|
||||
onTap: () async {
|
||||
scanBarcode(
|
||||
context,
|
||||
handler: SOAllocateStockHandler(
|
||||
salesOrder: widget.order,
|
||||
)
|
||||
);
|
||||
}
|
||||
)
|
||||
);
|
||||
scanBarcode(context,
|
||||
handler: SOAllocateStockHandler(
|
||||
salesOrder: widget.order,
|
||||
));
|
||||
}));
|
||||
}
|
||||
}
|
||||
|
||||
@ -261,10 +218,15 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
await widget.order.reload();
|
||||
await api.SalesOrderStatus.load();
|
||||
|
||||
supportsProjectCodes = api.supportsProjectCodes && await api.getGlobalBooleanSetting("PROJECT_CODES_ENABLED", backup: true);
|
||||
showCameraShortcut = await InvenTreeSettingsManager().getBool(INV_SO_SHOW_CAMERA, true);
|
||||
supportsProjectCodes = api.supportsProjectCodes &&
|
||||
await api.getGlobalBooleanSetting("PROJECT_CODES_ENABLED",
|
||||
backup: true);
|
||||
showCameraShortcut =
|
||||
await InvenTreeSettingsManager().getBool(INV_SO_SHOW_CAMERA, true);
|
||||
|
||||
InvenTreeSalesOrderAttachment().countAttachments(widget.order.pk).then((int value) {
|
||||
InvenTreeSalesOrderAttachment()
|
||||
.countAttachments(widget.order.pk)
|
||||
.then((int value) {
|
||||
if (mounted) {
|
||||
setState(() {
|
||||
attachmentCount = value;
|
||||
@ -273,7 +235,8 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
});
|
||||
|
||||
// Count number of "extra line items" against this order
|
||||
InvenTreeSOExtraLineItem().count(filters: {"order": widget.order.pk.toString() }).then((int value) {
|
||||
InvenTreeSOExtraLineItem().count(
|
||||
filters: {"order": widget.order.pk.toString()}).then((int value) {
|
||||
if (mounted) {
|
||||
setState(() {
|
||||
extraLineCount = value;
|
||||
@ -298,15 +261,11 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
fields.remove("project_code");
|
||||
}
|
||||
|
||||
widget.order.editForm(
|
||||
context,
|
||||
L10().salesOrderEdit,
|
||||
fields: fields,
|
||||
onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().salesOrderUpdated, success: true);
|
||||
}
|
||||
);
|
||||
widget.order.editForm(context, L10().salesOrderEdit, fields: fields,
|
||||
onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().salesOrderUpdated, success: true);
|
||||
});
|
||||
}
|
||||
|
||||
// Construct header tile
|
||||
@ -314,45 +273,40 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
InvenTreeCompany? customer = widget.order.customer;
|
||||
|
||||
return Card(
|
||||
child: ListTile(
|
||||
title: Text(widget.order.reference),
|
||||
subtitle: Text(widget.order.description),
|
||||
leading: customer == null ? null : api.getThumbnail(customer.thumbnail),
|
||||
trailing: Text(
|
||||
api.SalesOrderStatus.label(widget.order.status),
|
||||
style: TextStyle(
|
||||
color: api.SalesOrderStatus.color(widget.order.status)
|
||||
),
|
||||
),
|
||||
)
|
||||
);
|
||||
child: ListTile(
|
||||
title: Text(widget.order.reference),
|
||||
subtitle: Text(widget.order.description),
|
||||
leading: customer == null ? null : api.getThumbnail(customer.thumbnail),
|
||||
trailing: Text(
|
||||
api.SalesOrderStatus.label(widget.order.status),
|
||||
style:
|
||||
TextStyle(color: api.SalesOrderStatus.color(widget.order.status)),
|
||||
),
|
||||
));
|
||||
}
|
||||
|
||||
List<Widget> orderTiles(BuildContext context) {
|
||||
|
||||
List<Widget> tiles = [
|
||||
headerTile(context)
|
||||
];
|
||||
List<Widget> tiles = [headerTile(context)];
|
||||
|
||||
InvenTreeCompany? customer = widget.order.customer;
|
||||
|
||||
if (supportsProjectCodes && widget.order.hasProjectCode) {
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().projectCode),
|
||||
subtitle: Text("${widget.order.projectCode} - ${widget.order.projectCodeDescription}"),
|
||||
subtitle: Text(
|
||||
"${widget.order.projectCode} - ${widget.order.projectCodeDescription}"),
|
||||
leading: Icon(TablerIcons.list),
|
||||
));
|
||||
}
|
||||
|
||||
if (customer != null) {
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().customer),
|
||||
subtitle: Text(customer.name),
|
||||
leading: Icon(TablerIcons.user, color: COLOR_ACTION),
|
||||
onTap: () {
|
||||
customer.goToDetailPage(context);
|
||||
}
|
||||
));
|
||||
title: Text(L10().customer),
|
||||
subtitle: Text(customer.name),
|
||||
leading: Icon(TablerIcons.user, color: COLOR_ACTION),
|
||||
onTap: () {
|
||||
customer.goToDetailPage(context);
|
||||
}));
|
||||
}
|
||||
|
||||
if (widget.order.customerReference.isNotEmpty) {
|
||||
@ -367,12 +321,12 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().lineItems),
|
||||
subtitle: ProgressBar(
|
||||
widget.order.completedLineItemCount.toDouble(),
|
||||
maximum: widget.order.lineItemCount.toDouble()
|
||||
),
|
||||
subtitle: ProgressBar(widget.order.completedLineItemCount.toDouble(),
|
||||
maximum: widget.order.lineItemCount.toDouble()),
|
||||
leading: Icon(TablerIcons.clipboard_check),
|
||||
trailing: Text("${widget.order.completedLineItemCount} / ${widget.order.lineItemCount}", style: TextStyle(color: lineColor)),
|
||||
trailing: Text(
|
||||
"${widget.order.completedLineItemCount} / ${widget.order.lineItemCount}",
|
||||
style: TextStyle(color: lineColor)),
|
||||
));
|
||||
|
||||
// Extra line items
|
||||
@ -384,9 +338,8 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => SOExtraLineListWidget(widget.order, filters: {"order": widget.order.pk.toString()})
|
||||
)
|
||||
)
|
||||
builder: (context) => SOExtraLineListWidget(widget.order,
|
||||
filters: {"order": widget.order.pk.toString()})))
|
||||
},
|
||||
));
|
||||
|
||||
@ -394,12 +347,12 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
if (widget.order.shipmentCount > 0) {
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().shipments),
|
||||
subtitle: ProgressBar(
|
||||
widget.order.completedShipmentCount.toDouble(),
|
||||
maximum: widget.order.shipmentCount.toDouble()
|
||||
),
|
||||
subtitle: ProgressBar(widget.order.completedShipmentCount.toDouble(),
|
||||
maximum: widget.order.shipmentCount.toDouble()),
|
||||
leading: Icon(TablerIcons.truck_delivery),
|
||||
trailing: Text("${widget.order.completedShipmentCount} / ${widget.order.shipmentCount}", style: TextStyle(color: lineColor)),
|
||||
trailing: Text(
|
||||
"${widget.order.completedShipmentCount} / ${widget.order.shipmentCount}",
|
||||
style: TextStyle(color: lineColor)),
|
||||
));
|
||||
}
|
||||
|
||||
@ -430,51 +383,42 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
}
|
||||
|
||||
// Responsible "owner"
|
||||
if (widget.order.responsibleName.isNotEmpty && widget.order.responsibleLabel.isNotEmpty) {
|
||||
if (widget.order.responsibleName.isNotEmpty &&
|
||||
widget.order.responsibleLabel.isNotEmpty) {
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().responsible),
|
||||
leading: Icon(widget.order.responsibleLabel == "group" ? TablerIcons.users : TablerIcons.user),
|
||||
trailing: Text(widget.order.responsibleName)
|
||||
));
|
||||
leading: Icon(widget.order.responsibleLabel == "group"
|
||||
? TablerIcons.users
|
||||
: TablerIcons.user),
|
||||
trailing: Text(widget.order.responsibleName)));
|
||||
}
|
||||
|
||||
// Notes tile
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().notes),
|
||||
leading: Icon(TablerIcons.note, color: COLOR_ACTION),
|
||||
onTap: () {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => NotesWidget(widget.order)
|
||||
)
|
||||
);
|
||||
},
|
||||
)
|
||||
);
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().notes),
|
||||
leading: Icon(TablerIcons.note, color: COLOR_ACTION),
|
||||
onTap: () {
|
||||
Navigator.push(context,
|
||||
MaterialPageRoute(builder: (context) => NotesWidget(widget.order)));
|
||||
},
|
||||
));
|
||||
|
||||
// Attachments
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().attachments),
|
||||
leading: Icon(TablerIcons.file, color: COLOR_ACTION),
|
||||
trailing: attachmentCount > 0 ? Text(attachmentCount.toString()) : null,
|
||||
onTap: () {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => AttachmentWidget(
|
||||
InvenTreeSalesOrderAttachment(),
|
||||
widget.order.pk,
|
||||
widget.order.reference,
|
||||
widget.order.canEdit
|
||||
)
|
||||
)
|
||||
);
|
||||
},
|
||||
)
|
||||
);
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().attachments),
|
||||
leading: Icon(TablerIcons.file, color: COLOR_ACTION),
|
||||
trailing: attachmentCount > 0 ? Text(attachmentCount.toString()) : null,
|
||||
onTap: () {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => AttachmentWidget(
|
||||
InvenTreeSalesOrderAttachment(),
|
||||
widget.order.pk,
|
||||
widget.order.reference,
|
||||
widget.order.canEdit)));
|
||||
},
|
||||
));
|
||||
|
||||
return tiles;
|
||||
}
|
||||
@ -496,5 +440,4 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
|
||||
PaginatedSOShipmentList({"order": widget.order.pk.toString()}),
|
||||
];
|
||||
}
|
||||
|
||||
}
|
||||
|
@ -1,4 +1,3 @@
|
||||
|
||||
import "package:flutter/material.dart";
|
||||
import "package:flutter_speed_dial/flutter_speed_dial.dart";
|
||||
import "package:flutter_tabler_icons/flutter_tabler_icons.dart";
|
||||
@ -12,20 +11,18 @@ import "package:inventree/api.dart";
|
||||
import "package:inventree/inventree/company.dart";
|
||||
import "package:inventree/inventree/model.dart";
|
||||
|
||||
|
||||
class SalesOrderListWidget extends StatefulWidget {
|
||||
|
||||
const SalesOrderListWidget({this.filters = const {}, Key? key}) : super(key: key);
|
||||
const SalesOrderListWidget({this.filters = const {}, Key? key})
|
||||
: super(key: key);
|
||||
|
||||
final Map<String, String> filters;
|
||||
|
||||
@override
|
||||
_SalesOrderListWidgetState createState() => _SalesOrderListWidgetState();
|
||||
|
||||
}
|
||||
|
||||
class _SalesOrderListWidgetState extends RefreshableState<SalesOrderListWidget> {
|
||||
|
||||
class _SalesOrderListWidgetState
|
||||
extends RefreshableState<SalesOrderListWidget> {
|
||||
_SalesOrderListWidgetState();
|
||||
|
||||
@override
|
||||
@ -36,15 +33,12 @@ class _SalesOrderListWidgetState extends RefreshableState<SalesOrderListWidget>
|
||||
List<SpeedDialChild> actions = [];
|
||||
|
||||
if (InvenTreeSalesOrder().canCreate) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
child: Icon(TablerIcons.circle_plus),
|
||||
label: L10().salesOrderCreate,
|
||||
onTap: () {
|
||||
_createSalesOrder(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.circle_plus),
|
||||
label: L10().salesOrderCreate,
|
||||
onTap: () {
|
||||
_createSalesOrder(context);
|
||||
}));
|
||||
}
|
||||
|
||||
return actions;
|
||||
@ -57,19 +51,15 @@ class _SalesOrderListWidgetState extends RefreshableState<SalesOrderListWidget>
|
||||
// Cannot set contact until company is locked in
|
||||
fields.remove("contact");
|
||||
|
||||
InvenTreeSalesOrder().createForm(
|
||||
context,
|
||||
L10().salesOrderCreate,
|
||||
fields: fields,
|
||||
onSuccess: (result) async {
|
||||
Map<String, dynamic> data = result as Map<String, dynamic>;
|
||||
InvenTreeSalesOrder().createForm(context, L10().salesOrderCreate,
|
||||
fields: fields, onSuccess: (result) async {
|
||||
Map<String, dynamic> data = result as Map<String, dynamic>;
|
||||
|
||||
if (data.containsKey("pk")) {
|
||||
var order = InvenTreeSalesOrder.fromJson(data);
|
||||
order.goToDetailPage(context);
|
||||
}
|
||||
}
|
||||
);
|
||||
if (data.containsKey("pk")) {
|
||||
var order = InvenTreeSalesOrder.fromJson(data);
|
||||
order.goToDetailPage(context);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
@ -82,25 +72,22 @@ class _SalesOrderListWidgetState extends RefreshableState<SalesOrderListWidget>
|
||||
Widget getBody(BuildContext context) {
|
||||
return PaginatedSalesOrderList(widget.filters);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
|
||||
class PaginatedSalesOrderList extends PaginatedSearchWidget {
|
||||
|
||||
const PaginatedSalesOrderList(Map<String, String> filters) : super(filters: filters);
|
||||
const PaginatedSalesOrderList(Map<String, String> filters)
|
||||
: super(filters: filters);
|
||||
|
||||
@override
|
||||
String get searchTitle => L10().salesOrders;
|
||||
|
||||
@override
|
||||
_PaginatedSalesOrderListState createState() => _PaginatedSalesOrderListState();
|
||||
|
||||
_PaginatedSalesOrderListState createState() =>
|
||||
_PaginatedSalesOrderListState();
|
||||
}
|
||||
|
||||
|
||||
class _PaginatedSalesOrderListState extends PaginatedSearchState<PaginatedSalesOrderList> {
|
||||
|
||||
class _PaginatedSalesOrderListState
|
||||
extends PaginatedSearchState<PaginatedSalesOrderList> {
|
||||
_PaginatedSalesOrderListState() : super();
|
||||
|
||||
@override
|
||||
@ -108,62 +95,59 @@ class _PaginatedSalesOrderListState extends PaginatedSearchState<PaginatedSalesO
|
||||
|
||||
@override
|
||||
Map<String, String> get orderingOptions => {
|
||||
"reference": L10().reference,
|
||||
"status": L10().status,
|
||||
"target_date": L10().targetDate,
|
||||
"customer__name": L10().customer,
|
||||
};
|
||||
"reference": L10().reference,
|
||||
"status": L10().status,
|
||||
"target_date": L10().targetDate,
|
||||
"customer__name": L10().customer,
|
||||
};
|
||||
|
||||
@override
|
||||
Map<String, Map<String, dynamic>> get filterOptions => {
|
||||
"outstanding": {
|
||||
"label": L10().outstanding,
|
||||
"help_text": L10().outstandingOrderDetail,
|
||||
"tristate": true,
|
||||
},
|
||||
"overdue": {
|
||||
"label": L10().overdue,
|
||||
"help_text": L10().overdueDetail,
|
||||
"tristate": true,
|
||||
},
|
||||
"assigned_to_me": {
|
||||
"label": L10().assignedToMe,
|
||||
"help_text": L10().assignedToMeDetail,
|
||||
"tristate": true,
|
||||
}
|
||||
};
|
||||
"outstanding": {
|
||||
"label": L10().outstanding,
|
||||
"help_text": L10().outstandingOrderDetail,
|
||||
"tristate": true,
|
||||
},
|
||||
"overdue": {
|
||||
"label": L10().overdue,
|
||||
"help_text": L10().overdueDetail,
|
||||
"tristate": true,
|
||||
},
|
||||
"assigned_to_me": {
|
||||
"label": L10().assignedToMe,
|
||||
"help_text": L10().assignedToMeDetail,
|
||||
"tristate": true,
|
||||
}
|
||||
};
|
||||
|
||||
@override
|
||||
Future<InvenTreePageResponse?> requestPage(int limit, int offset, Map<String, String> params) async {
|
||||
|
||||
Future<InvenTreePageResponse?> requestPage(
|
||||
int limit, int offset, Map<String, String> params) async {
|
||||
await InvenTreeAPI().SalesOrderStatus.load();
|
||||
final page = await InvenTreeSalesOrder().listPaginated(limit, offset, filters: params);
|
||||
final page = await InvenTreeSalesOrder()
|
||||
.listPaginated(limit, offset, filters: params);
|
||||
|
||||
return page;
|
||||
}
|
||||
|
||||
@override
|
||||
Widget buildItem(BuildContext context, InvenTreeModel model) {
|
||||
|
||||
InvenTreeSalesOrder order = model as InvenTreeSalesOrder;
|
||||
|
||||
InvenTreeCompany? customer = order.customer;
|
||||
|
||||
return ListTile(
|
||||
title: Text(order.reference),
|
||||
subtitle: Text(order.description),
|
||||
leading: customer == null ? null : InvenTreeAPI().getThumbnail(customer.thumbnail),
|
||||
trailing: Text(
|
||||
InvenTreeAPI().SalesOrderStatus.label(order.status),
|
||||
style: TextStyle(
|
||||
color: InvenTreeAPI().SalesOrderStatus.color(order.status),
|
||||
)
|
||||
),
|
||||
onTap: () async {
|
||||
order.goToDetailPage(context);
|
||||
}
|
||||
);
|
||||
|
||||
title: Text(order.reference),
|
||||
subtitle: Text(order.description),
|
||||
leading: customer == null
|
||||
? null
|
||||
: InvenTreeAPI().getThumbnail(customer.thumbnail),
|
||||
trailing: Text(InvenTreeAPI().SalesOrderStatus.label(order.status),
|
||||
style: TextStyle(
|
||||
color: InvenTreeAPI().SalesOrderStatus.color(order.status),
|
||||
)),
|
||||
onTap: () async {
|
||||
order.goToDetailPage(context);
|
||||
});
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
@ -11,41 +11,36 @@ import "package:inventree/widget/paginator.dart";
|
||||
import "package:inventree/widget/refreshable_state.dart";
|
||||
import "package:inventree/widget/snacks.dart";
|
||||
|
||||
|
||||
class SOExtraLineListWidget extends StatefulWidget {
|
||||
|
||||
const SOExtraLineListWidget(this.order, {this.filters = const {}, Key? key}) : super(key: key);
|
||||
const SOExtraLineListWidget(this.order, {this.filters = const {}, Key? key})
|
||||
: super(key: key);
|
||||
|
||||
final InvenTreeSalesOrder order;
|
||||
|
||||
final Map<String, String> filters;
|
||||
|
||||
@override
|
||||
_SalesOrderExtraLineListWidgetState createState() => _SalesOrderExtraLineListWidgetState();
|
||||
_SalesOrderExtraLineListWidgetState createState() =>
|
||||
_SalesOrderExtraLineListWidgetState();
|
||||
}
|
||||
|
||||
class _SalesOrderExtraLineListWidgetState extends RefreshableState<SOExtraLineListWidget> {
|
||||
|
||||
class _SalesOrderExtraLineListWidgetState
|
||||
extends RefreshableState<SOExtraLineListWidget> {
|
||||
_SalesOrderExtraLineListWidgetState();
|
||||
|
||||
@override
|
||||
String getAppBarTitle() => L10().extraLineItems;
|
||||
|
||||
Future<void> _addLineItem(BuildContext context) async {
|
||||
|
||||
var fields = InvenTreeSOExtraLineItem().formFields();
|
||||
|
||||
fields["order"]?["value"] = widget.order.pk;
|
||||
|
||||
InvenTreeSOExtraLineItem().createForm(
|
||||
context,
|
||||
L10().lineItemAdd,
|
||||
fields: fields,
|
||||
onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().lineItemUpdated, success: true);
|
||||
}
|
||||
);
|
||||
InvenTreeSOExtraLineItem().createForm(context, L10().lineItemAdd,
|
||||
fields: fields, onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().lineItemUpdated, success: true);
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
@ -53,15 +48,12 @@ class _SalesOrderExtraLineListWidgetState extends RefreshableState<SOExtraLineLi
|
||||
List<SpeedDialChild> actions = [];
|
||||
|
||||
if (widget.order.canEdit) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
child: Icon(TablerIcons.circle_plus, color: Colors.green),
|
||||
label: L10().lineItemAdd,
|
||||
onTap: () {
|
||||
_addLineItem(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.circle_plus, color: Colors.green),
|
||||
label: L10().lineItemAdd,
|
||||
onTap: () {
|
||||
_addLineItem(context);
|
||||
}));
|
||||
}
|
||||
|
||||
return actions;
|
||||
@ -73,35 +65,35 @@ class _SalesOrderExtraLineListWidgetState extends RefreshableState<SOExtraLineLi
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
class PaginatedSOExtraLineList extends PaginatedSearchWidget {
|
||||
|
||||
const PaginatedSOExtraLineList(Map<String, String> filters) : super(filters: filters);
|
||||
const PaginatedSOExtraLineList(Map<String, String> filters)
|
||||
: super(filters: filters);
|
||||
|
||||
@override
|
||||
String get searchTitle => L10().extraLineItems;
|
||||
|
||||
@override
|
||||
_PaginatedSOExtraLineListState createState() => _PaginatedSOExtraLineListState();
|
||||
|
||||
_PaginatedSOExtraLineListState createState() =>
|
||||
_PaginatedSOExtraLineListState();
|
||||
}
|
||||
|
||||
class _PaginatedSOExtraLineListState extends PaginatedSearchState<PaginatedSOExtraLineList> {
|
||||
|
||||
class _PaginatedSOExtraLineListState
|
||||
extends PaginatedSearchState<PaginatedSOExtraLineList> {
|
||||
_PaginatedSOExtraLineListState() : super();
|
||||
|
||||
@override
|
||||
String get prefix => "so_extra_line_";
|
||||
|
||||
@override
|
||||
Future<InvenTreePageResponse?> requestPage(int limit, int offset, Map<String, String> params) async {
|
||||
final page = await InvenTreeSOExtraLineItem().listPaginated(limit, offset, filters: params);
|
||||
Future<InvenTreePageResponse?> requestPage(
|
||||
int limit, int offset, Map<String, String> params) async {
|
||||
final page = await InvenTreeSOExtraLineItem()
|
||||
.listPaginated(limit, offset, filters: params);
|
||||
return page;
|
||||
}
|
||||
|
||||
@override
|
||||
Widget buildItem(BuildContext context, InvenTreeModel model) {
|
||||
|
||||
InvenTreeSOExtraLineItem line = model as InvenTreeSOExtraLineItem;
|
||||
|
||||
return ListTile(
|
||||
@ -115,4 +107,4 @@ class _PaginatedSOExtraLineListState extends PaginatedSearchState<PaginatedSOExt
|
||||
},
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -1,5 +1,3 @@
|
||||
|
||||
|
||||
/*
|
||||
* Widget for displaying detail view of a single SalesOrderLineItem
|
||||
*/
|
||||
@ -22,21 +20,16 @@ import "package:inventree/l10.dart";
|
||||
import "package:inventree/helpers.dart";
|
||||
import "package:inventree/api_form.dart";
|
||||
|
||||
|
||||
class SoLineDetailWidget extends StatefulWidget {
|
||||
|
||||
const SoLineDetailWidget(this.item, {Key? key}) : super(key: key);
|
||||
|
||||
final InvenTreeSOLineItem item;
|
||||
|
||||
@override
|
||||
_SOLineDetailWidgetState createState() => _SOLineDetailWidgetState();
|
||||
|
||||
}
|
||||
|
||||
|
||||
class _SOLineDetailWidgetState extends RefreshableState<SoLineDetailWidget> {
|
||||
|
||||
_SOLineDetailWidgetState();
|
||||
|
||||
InvenTreeSalesOrder? order;
|
||||
@ -62,7 +55,6 @@ class _SOLineDetailWidgetState extends RefreshableState<SoLineDetailWidget> {
|
||||
}
|
||||
|
||||
Future<void> _allocateStock(BuildContext context) async {
|
||||
|
||||
if (order == null) {
|
||||
return;
|
||||
}
|
||||
@ -76,22 +68,13 @@ class _SOLineDetailWidgetState extends RefreshableState<SoLineDetailWidget> {
|
||||
"part": widget.item.partId.toString()
|
||||
};
|
||||
fields["quantity"]?["value"] = widget.item.unallocatedQuantity.toString();
|
||||
fields["shipment"]?["filters"] = {
|
||||
"order": order!.pk.toString()
|
||||
};
|
||||
|
||||
launchApiForm(
|
||||
context,
|
||||
L10().allocateStock,
|
||||
order!.allocate_url,
|
||||
fields,
|
||||
method: "POST",
|
||||
icon: TablerIcons.transition_right,
|
||||
onSuccess: (data) async {
|
||||
refresh(context);
|
||||
}
|
||||
);
|
||||
fields["shipment"]?["filters"] = {"order": order!.pk.toString()};
|
||||
|
||||
launchApiForm(context, L10().allocateStock, order!.allocate_url, fields,
|
||||
method: "POST",
|
||||
icon: TablerIcons.transition_right, onSuccess: (data) async {
|
||||
refresh(context);
|
||||
});
|
||||
}
|
||||
|
||||
Future<void> _editLineItem(BuildContext context) async {
|
||||
@ -102,32 +85,24 @@ class _SOLineDetailWidgetState extends RefreshableState<SoLineDetailWidget> {
|
||||
fields["part"]?["hidden"] = true;
|
||||
}
|
||||
|
||||
widget.item.editForm(
|
||||
context,
|
||||
L10().editLineItem,
|
||||
fields: fields,
|
||||
onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().lineItemUpdated, success: true);
|
||||
}
|
||||
);
|
||||
widget.item.editForm(context, L10().editLineItem, fields: fields,
|
||||
onSuccess: (data) async {
|
||||
refresh(context);
|
||||
showSnackIcon(L10().lineItemUpdated, success: true);
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
List<SpeedDialChild> actionButtons(BuildContext context) {
|
||||
|
||||
List<SpeedDialChild> buttons = [];
|
||||
|
||||
if (order != null && order!.isOpen) {
|
||||
buttons.add(
|
||||
SpeedDialChild(
|
||||
buttons.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.transition_right, color: Colors.blue),
|
||||
label: L10().allocateStock,
|
||||
onTap: () async {
|
||||
_allocateStock(context);
|
||||
}
|
||||
)
|
||||
);
|
||||
}));
|
||||
}
|
||||
|
||||
return buttons;
|
||||
@ -138,23 +113,15 @@ class _SOLineDetailWidgetState extends RefreshableState<SoLineDetailWidget> {
|
||||
List<SpeedDialChild> actions = [];
|
||||
|
||||
if (order != null && order!.isOpen && InvenTreeSOLineItem().canCreate) {
|
||||
|
||||
if (api.supportsBarcodeSOAllocateEndpoint) {
|
||||
actions.add(
|
||||
SpeedDialChild(
|
||||
child: Icon(TablerIcons.transition_right),
|
||||
label: L10().allocateStock,
|
||||
onTap: () async {
|
||||
scanBarcode(
|
||||
context,
|
||||
actions.add(SpeedDialChild(
|
||||
child: Icon(TablerIcons.transition_right),
|
||||
label: L10().allocateStock,
|
||||
onTap: () async {
|
||||
scanBarcode(context,
|
||||
handler: SOAllocateStockHandler(
|
||||
salesOrder: order,
|
||||
lineItem: widget.item
|
||||
)
|
||||
);
|
||||
}
|
||||
)
|
||||
);
|
||||
salesOrder: order, lineItem: widget.item));
|
||||
}));
|
||||
}
|
||||
}
|
||||
|
||||
@ -179,8 +146,7 @@ class _SOLineDetailWidgetState extends RefreshableState<SoLineDetailWidget> {
|
||||
List<Widget> tiles = [];
|
||||
|
||||
// Reference to the part
|
||||
tiles.add(
|
||||
ListTile(
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().part),
|
||||
subtitle: Text(widget.item.partName),
|
||||
leading: Icon(TablerIcons.box, color: COLOR_ACTION),
|
||||
@ -193,85 +159,64 @@ class _SOLineDetailWidgetState extends RefreshableState<SoLineDetailWidget> {
|
||||
if (part is InvenTreePart) {
|
||||
part.goToDetailPage(context);
|
||||
}
|
||||
}
|
||||
)
|
||||
);
|
||||
}));
|
||||
|
||||
// Available quantity
|
||||
tiles.add(
|
||||
ListTile(
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().availableStock),
|
||||
leading: Icon(TablerIcons.packages),
|
||||
trailing: Text(simpleNumberString(widget.item.availableStock))
|
||||
)
|
||||
);
|
||||
trailing: Text(simpleNumberString(widget.item.availableStock))));
|
||||
|
||||
// Allocated quantity
|
||||
tiles.add(
|
||||
ListTile(
|
||||
tiles.add(ListTile(
|
||||
leading: Icon(TablerIcons.clipboard_check),
|
||||
title: Text(L10().allocated),
|
||||
subtitle: ProgressBar(widget.item.allocatedRatio),
|
||||
trailing: Text(
|
||||
widget.item.allocatedString,
|
||||
style: TextStyle(
|
||||
color: widget.item.isAllocated ? COLOR_SUCCESS : COLOR_WARNING
|
||||
)
|
||||
)
|
||||
)
|
||||
);
|
||||
trailing: Text(widget.item.allocatedString,
|
||||
style: TextStyle(
|
||||
color:
|
||||
widget.item.isAllocated ? COLOR_SUCCESS : COLOR_WARNING))));
|
||||
|
||||
// Shipped quantity
|
||||
tiles.add(
|
||||
ListTile(
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().shipped),
|
||||
subtitle: ProgressBar(widget.item.progressRatio),
|
||||
trailing: Text(
|
||||
widget.item.progressString,
|
||||
style: TextStyle(
|
||||
color: widget.item.isComplete ? COLOR_SUCCESS : COLOR_WARNING
|
||||
),
|
||||
color: widget.item.isComplete ? COLOR_SUCCESS : COLOR_WARNING),
|
||||
),
|
||||
leading: Icon(TablerIcons.truck)
|
||||
)
|
||||
);
|
||||
leading: Icon(TablerIcons.truck)));
|
||||
|
||||
// Reference
|
||||
if (widget.item.reference.isNotEmpty) {
|
||||
tiles.add(
|
||||
ListTile(
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().reference),
|
||||
subtitle: Text(widget.item.reference),
|
||||
leading: Icon(TablerIcons.hash)
|
||||
)
|
||||
);
|
||||
leading: Icon(TablerIcons.hash)));
|
||||
}
|
||||
|
||||
// Note
|
||||
if (widget.item.notes.isNotEmpty) {
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().notes),
|
||||
subtitle: Text(widget.item.notes),
|
||||
leading: Icon(TablerIcons.note),
|
||||
)
|
||||
);
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().notes),
|
||||
subtitle: Text(widget.item.notes),
|
||||
leading: Icon(TablerIcons.note),
|
||||
));
|
||||
}
|
||||
|
||||
// External link
|
||||
if (widget.item.link.isNotEmpty) {
|
||||
tiles.add(
|
||||
ListTile(
|
||||
title: Text(L10().link),
|
||||
subtitle: Text(widget.item.link),
|
||||
leading: Icon(TablerIcons.link, color: COLOR_ACTION),
|
||||
onTap: () async {
|
||||
await openLink(widget.item.link);
|
||||
},
|
||||
)
|
||||
);
|
||||
tiles.add(ListTile(
|
||||
title: Text(L10().link),
|
||||
subtitle: Text(widget.item.link),
|
||||
leading: Icon(TablerIcons.link, color: COLOR_ACTION),
|
||||
onTap: () async {
|
||||
await openLink(widget.item.link);
|
||||
},
|
||||
));
|
||||
}
|
||||
|
||||
return tiles;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -9,28 +9,26 @@ import "package:inventree/api.dart";
|
||||
import "package:inventree/app_colors.dart";
|
||||
import "package:inventree/widget/progress.dart";
|
||||
|
||||
|
||||
/*
|
||||
* Paginated widget class for displaying a list of sales order line items
|
||||
*/
|
||||
|
||||
class PaginatedSOLineList extends PaginatedSearchWidget {
|
||||
const PaginatedSOLineList(Map<String, String> filters) : super(filters: filters);
|
||||
const PaginatedSOLineList(Map<String, String> filters)
|
||||
: super(filters: filters);
|
||||
|
||||
@override
|
||||
String get searchTitle => L10().lineItems;
|
||||
|
||||
@override
|
||||
_PaginatedSOLineListState createState() => _PaginatedSOLineListState();
|
||||
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
* State class for PaginatedSOLineList
|
||||
*/
|
||||
class _PaginatedSOLineListState extends PaginatedSearchState<PaginatedSOLineList> {
|
||||
|
||||
class _PaginatedSOLineListState
|
||||
extends PaginatedSearchState<PaginatedSOLineList> {
|
||||
_PaginatedSOLineListState() : super();
|
||||
|
||||
@override
|
||||
@ -38,18 +36,18 @@ class _PaginatedSOLineListState extends PaginatedSearchState<PaginatedSOLineList
|
||||
|
||||
@override
|
||||
Map<String, String> get orderingOptions => {
|
||||
"part": L10().part,
|
||||
"quantity": L10().quantity,
|
||||
};
|
||||
"part": L10().part,
|
||||
"quantity": L10().quantity,
|
||||
};
|
||||
|
||||
@override
|
||||
Map<String, Map<String, dynamic>> get filterOptions => {
|
||||
|
||||
};
|
||||
Map<String, Map<String, dynamic>> get filterOptions => {};
|
||||
|
||||
@override
|
||||
Future<InvenTreePageResponse?> requestPage(int limit, int offset, Map<String, String> params) async {
|
||||
final page = await InvenTreeSOLineItem().listPaginated(limit, offset, filters: params);
|
||||
Future<InvenTreePageResponse?> requestPage(
|
||||
int limit, int offset, Map<String, String> params) async {
|
||||
final page = await InvenTreeSOLineItem()
|
||||
.listPaginated(limit, offset, filters: params);
|
||||
return page;
|
||||
}
|
||||
|
||||
@ -60,27 +58,27 @@ class _PaginatedSOLineListState extends PaginatedSearchState<PaginatedSOLineList
|
||||
|
||||
if (part != null) {
|
||||
return ListTile(
|
||||
title: Text(part.name),
|
||||
subtitle: Text(part.description),
|
||||
leading: InvenTreeAPI().getThumbnail(part.thumbnail),
|
||||
trailing: Text(item.progressString, style: TextStyle(color: item.isComplete ? COLOR_SUCCESS : COLOR_WARNING)),
|
||||
onTap: () async {
|
||||
showLoadingOverlay();
|
||||
await item.reload();
|
||||
hideLoadingOverlay();
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => SoLineDetailWidget(item))
|
||||
);
|
||||
}
|
||||
);
|
||||
title: Text(part.name),
|
||||
subtitle: Text(part.description),
|
||||
leading: InvenTreeAPI().getThumbnail(part.thumbnail),
|
||||
trailing: Text(item.progressString,
|
||||
style: TextStyle(
|
||||
color: item.isComplete ? COLOR_SUCCESS : COLOR_WARNING)),
|
||||
onTap: () async {
|
||||
showLoadingOverlay();
|
||||
await item.reload();
|
||||
hideLoadingOverlay();
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(
|
||||
builder: (context) => SoLineDetailWidget(item)));
|
||||
});
|
||||
} else {
|
||||
return ListTile(
|
||||
title: Text(L10().error),
|
||||
subtitle: Text("Missing part detail", style: TextStyle(color: COLOR_DANGER)),
|
||||
subtitle:
|
||||
Text("Missing part detail", style: TextStyle(color: COLOR_DANGER)),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
@ -1,4 +1,3 @@
|
||||
|
||||
import "package:flutter/material.dart";
|
||||
import "package:flutter_tabler_icons/flutter_tabler_icons.dart";
|
||||
import "package:inventree/app_colors.dart";
|
||||
@ -9,19 +8,19 @@ import "package:inventree/inventree/model.dart";
|
||||
import "package:inventree/l10.dart";
|
||||
|
||||
class PaginatedSOShipmentList extends PaginatedSearchWidget {
|
||||
|
||||
const PaginatedSOShipmentList(Map<String, String> filters) : super(filters: filters);
|
||||
const PaginatedSOShipmentList(Map<String, String> filters)
|
||||
: super(filters: filters);
|
||||
|
||||
@override
|
||||
String get searchTitle => L10().shipments;
|
||||
|
||||
@override
|
||||
_PaginatedSOShipmentListState createState() => _PaginatedSOShipmentListState();
|
||||
_PaginatedSOShipmentListState createState() =>
|
||||
_PaginatedSOShipmentListState();
|
||||
}
|
||||
|
||||
|
||||
class _PaginatedSOShipmentListState extends PaginatedSearchState<PaginatedSOShipmentList> {
|
||||
|
||||
class _PaginatedSOShipmentListState
|
||||
extends PaginatedSearchState<PaginatedSOShipmentList> {
|
||||
_PaginatedSOShipmentListState() : super();
|
||||
|
||||
@override
|
||||
@ -34,22 +33,23 @@ class _PaginatedSOShipmentListState extends PaginatedSearchState<PaginatedSOShip
|
||||
Map<String, Map<String, dynamic>> get filterOptions => {};
|
||||
|
||||
@override
|
||||
Future<InvenTreePageResponse?> requestPage(int limit, int offset, Map<String, String> params) async {
|
||||
final page = await InvenTreeSalesOrderShipment().listPaginated(limit, offset, filters: params);
|
||||
Future<InvenTreePageResponse?> requestPage(
|
||||
int limit, int offset, Map<String, String> params) async {
|
||||
final page = await InvenTreeSalesOrderShipment()
|
||||
.listPaginated(limit, offset, filters: params);
|
||||
return page;
|
||||
}
|
||||
|
||||
@override
|
||||
Widget buildItem(BuildContext context, InvenTreeModel model) {
|
||||
|
||||
InvenTreeSalesOrderShipment shipment = model as InvenTreeSalesOrderShipment;
|
||||
|
||||
return ListTile(
|
||||
title: Text(shipment.reference),
|
||||
subtitle: Text(shipment.tracking_number),
|
||||
leading: shipment.shipped ? Icon(TablerIcons.calendar_check, color: COLOR_SUCCESS) : Icon(TablerIcons.calendar_cancel, color: COLOR_WARNING),
|
||||
trailing: shipment.shipped ? Text(shipment.shipment_date ?? "") : null
|
||||
);
|
||||
|
||||
title: Text(shipment.reference),
|
||||
subtitle: Text(shipment.tracking_number),
|
||||
leading: shipment.shipped
|
||||
? Icon(TablerIcons.calendar_check, color: COLOR_SUCCESS)
|
||||
: Icon(TablerIcons.calendar_cancel, color: COLOR_WARNING),
|
||||
trailing: shipment.shipped ? Text(shipment.shipment_date ?? "") : null);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
Reference in New Issue
Block a user