From be6fad09935f7469f7e05e0a1328f961bdfe8b06 Mon Sep 17 00:00:00 2001 From: Oliver Date: Thu, 12 Dec 2024 10:52:45 +1100 Subject: [PATCH 01/21] Handle error on unexpected barcode response --- lib/barcode/barcode.dart | 15 ++++++++++----- 1 file changed, 10 insertions(+), 5 deletions(-) diff --git a/lib/barcode/barcode.dart b/lib/barcode/barcode.dart index dd2a6c8b..d4132bbf 100644 --- a/lib/barcode/barcode.dart +++ b/lib/barcode/barcode.dart @@ -4,6 +4,7 @@ import "package:flutter_speed_dial/flutter_speed_dial.dart"; import "package:flutter_tabler_icons/flutter_tabler_icons.dart"; import "package:inventree/helpers.dart"; import "package:inventree/inventree/sales_order.dart"; +import "package:inventree/inventree/sentry.dart"; import "package:inventree/preferences.dart"; import "package:inventree/widget/order/sales_order_detail.dart"; import "package:one_context/one_context.dart"; @@ -232,12 +233,16 @@ class BarcodeScanHandler extends BarcodeHandler { for (var key in validModels) { if (data.containsKey(key)) { - pk = (data[key]?["pk"] ?? -1) as int; + try { + pk = (data[key]?["pk"] ?? -1) as int; - // Break on the first valid match found - if (pk > 0) { - model = key; - break; + // Break on the first valid match found + if (pk > 0) { + model = key; + break; + } + } catch (error, stackTrace) { + sentryReportError("onBarcodeMatched", error, stackTrace); } } } From 47b74ef9a45afda4584ed8724ca1929d5bef3a07 Mon Sep 17 00:00:00 2001 From: Oliver Date: Thu, 12 Dec 2024 11:39:12 +1100 Subject: [PATCH 02/21] Add ManufacturerPart detail view --- lib/l10n/app_en.arb | 9 + .../company/manufacturer_part_detail.dart | 172 ++++++++++++++++++ 2 files changed, 181 insertions(+) create mode 100644 lib/widget/company/manufacturer_part_detail.dart diff --git a/lib/l10n/app_en.arb b/lib/l10n/app_en.arb index d6693bdc..99be420e 100644 --- a/lib/l10n/app_en.arb +++ b/lib/l10n/app_en.arb @@ -608,6 +608,9 @@ "itemDeleted": "Item has been removed", "@itemDeleted": {}, + "itemUpdated": "Item updated", + "@itemUpdated": {}, + "keywords": "Keywords", "@keywords": {}, @@ -684,6 +687,12 @@ "lost": "Lost", "@lost": {}, + "manufacturerPart": "Manufacturer Part", + "@manufacturerPart": {}, + + "manufacturerPartEdit": "Edit Manufacturer Part", + "@manufacturerPartEdit": {}, + "manufacturerPartNumber": "Manufacturer Part Number", "@manufacturerPartNumber": {}, diff --git a/lib/widget/company/manufacturer_part_detail.dart b/lib/widget/company/manufacturer_part_detail.dart new file mode 100644 index 00000000..7b62124c --- /dev/null +++ b/lib/widget/company/manufacturer_part_detail.dart @@ -0,0 +1,172 @@ + +import "package:flutter/material.dart"; +import "package:flutter_speed_dial/flutter_speed_dial.dart"; +import "package:flutter_tabler_icons/flutter_tabler_icons.dart"; + +import "package:inventree/l10.dart"; +import "package:inventree/api.dart"; +import "package:inventree/app_colors.dart"; + +import "package:inventree/inventree/company.dart"; +import "package:inventree/inventree/part.dart"; + +import "package:inventree/widget/refreshable_state.dart"; +import "package:inventree/widget/snacks.dart"; +import "package:inventree/widget/progress.dart"; + +import "package:inventree/widget/part/part_detail.dart"; +import "package:inventree/widget/company/company_detail.dart"; +import "package:url_launcher/url_launcher.dart"; + +/* + * Detail widget for viewing a single ManufacturerPart instance + */ +class ManufacturerPartDetailWidget extends StatefulWidget { + + const ManufacturerPartDetailWidget(this.manufacturerPart, {Key? key}) + : super(key: key); + + final InvenTreeManufacturerPart manufacturerPart; + + @override + _ManufacturerPartDisplayState createState() => _ManufacturerPartDisplayState(); +} + + +class _ManufacturerPartDisplayState extends RefreshableState { + + _ManufacturerPartDisplayState(); + + @override + String getAppBarTitle() => L10().manufacturerPart; + + @override + Future request(BuildContext context) async { + final bool result = widget.manufacturerPart.pk > 0 && + await widget.manufacturerPart.reload(); + + if (!result) { + Navigator.of(context).pop(); + } + } + + Future editManufacturerPart(BuildContext context) async { + widget.manufacturerPart.editForm( + context, + L10().manufacturerPartEdit, + onSuccess: (data) async { + refresh(context); + showSnackIcon(L10().itemUpdated, success: true); + } + ); + } + + @override + List barcodeButtons(BuildContext context) { + List actions = []; + + // TODO: Barcode actions? + + return actions; + } + + @override + List appBarActions(BuildContext context) { + List actions = []; + + if (widget.manufacturerPart.canEdit) { + actions.add( + IconButton( + icon: Icon(TablerIcons.edit), + tooltip: L10().edit, + onPressed: () { + editManufacturerPart(context); + } + ) + ); + } + + return actions; + } + + /* + * Build a set of tiles to display for this ManufacturerPart instance + */ + @override + List getTiles(BuildContext context) { + List tiles = []; + + if (loading) { + tiles.add(progressIndicator()); + return tiles; + } + + // Internal Part + tiles.add( + ListTile( + title: Text(L10().internalPart), + subtitle: Text(widget.manufacturerPart.partName), + leading: Icon(TablerIcons.box, color: COLOR_ACTION), + trailing: InvenTreeAPI().getThumbnail(widget.manufacturerPart.partImage), + onTap: () async { + showLoadingOverlay(); + final part = await InvenTreePart().get(widget.manufacturerPart.partId); + hideLoadingOverlay(); + + if (part is InvenTreePart) { + Navigator.push(context, MaterialPageRoute( + builder: (context) => PartDetailWidget(part))); + } + }, + ) + ); + + // Manufacturer details + tiles.add( + ListTile( + title: Text(L10().supplier), + subtitle: Text(widget.manufacturerPart.manufacturerName), + leading: Icon(TablerIcons.building_factory_2, color: COLOR_ACTION), + trailing: InvenTreeAPI().getThumbnail(widget.manufacturerPart.manufacturerImage), + onTap: () async { + showLoadingOverlay(); + var supplier = await InvenTreeCompany().get(widget.manufacturerPart.manufacturerId); + hideLoadingOverlay(); + + if (supplier is InvenTreeCompany) { + Navigator.push(context, MaterialPageRoute( + builder: (context) => CompanyDetailWidget(supplier) + )); + } + } + ) + ); + + // MPN (part number) + tiles.add( + ListTile( + title: Text(L10().manufacturerPartNumber), + subtitle: Text(widget.manufacturerPart.MPN), + leading: Icon(TablerIcons.hash), + ) + ); + + if (widget.manufacturerPart.link.isNotEmpty) { + tiles.add( + ListTile( + title: Text(widget.manufacturerPart.link), + leading: Icon(TablerIcons.link, color: COLOR_ACTION), + onTap: () async { + var uri = Uri.tryParse(widget.manufacturerPart.link); + if (uri != null && await canLaunchUrl(uri)) { + await launchUrl(uri); + } + }, + ) + ); + } + + return tiles; + } + +} From cbec71884e363b1efa78a8a7f19dfc5e10a6d99d Mon Sep 17 00:00:00 2001 From: Oliver Date: Thu, 12 Dec 2024 11:39:50 +1100 Subject: [PATCH 03/21] Support barcode scanning for manufacturer part --- lib/barcode/barcode.dart | 24 ++++++++++++++++---- lib/barcode/stock.dart | 4 ++-- lib/widget/company/supplier_part_detail.dart | 13 ++++++++++- 3 files changed, 34 insertions(+), 7 deletions(-) diff --git a/lib/barcode/barcode.dart b/lib/barcode/barcode.dart index d4132bbf..e0ce9a3c 100644 --- a/lib/barcode/barcode.dart +++ b/lib/barcode/barcode.dart @@ -6,6 +6,7 @@ import "package:inventree/helpers.dart"; import "package:inventree/inventree/sales_order.dart"; import "package:inventree/inventree/sentry.dart"; import "package:inventree/preferences.dart"; +import "package:inventree/widget/company/manufacturer_part_detail.dart"; import "package:inventree/widget/order/sales_order_detail.dart"; import "package:one_context/one_context.dart"; @@ -177,15 +178,27 @@ class BarcodeScanHandler extends BarcodeHandler { */ Future handleSupplierPart(int pk) async { - var supplierpart = await InvenTreeSupplierPart().get(pk); + var supplierPart = await InvenTreeSupplierPart().get(pk); - if (supplierpart is InvenTreeSupplierPart) { + if (supplierPart is InvenTreeSupplierPart) { OneContext().pop(); OneContext().push(MaterialPageRoute( - builder: (context) => SupplierPartDetailWidget(supplierpart))); + builder: (context) => SupplierPartDetailWidget(supplierPart))); } } + /* + * Response when a "ManufacturerPart" instance is scanned + */ + Future handleManufacturerPart(int pk) async { + var manufacturerPart = await InvenTreeManufacturerPart().get(pk); + + if (manufacturerPart is InvenTreeManufacturerPart) { + OneContext().pop(); + OneContext().push(MaterialPageRoute( + builder: (context) => ManufacturerPartDetailWidget(manufacturerPart))); + } + } /* * Response when a "PurchaseOrder" instance is scanned @@ -223,6 +236,7 @@ class BarcodeScanHandler extends BarcodeHandler { "stockitem", "stocklocation", "supplierpart", + "manufacturerpart", ]; @@ -271,7 +285,9 @@ class BarcodeScanHandler extends BarcodeHandler { case "salesorder": await handleSalesOrder(pk); return; - // TODO: Handle manufacturer part + case "manufacturerpart": + await handleManufacturerPart(pk); + return; default: // Fall through to failure state break; diff --git a/lib/barcode/stock.dart b/lib/barcode/stock.dart index d2309010..126d1d06 100644 --- a/lib/barcode/stock.dart +++ b/lib/barcode/stock.dart @@ -33,7 +33,7 @@ class BarcodeScanStockLocationHandler extends BarcodeHandler { // We expect that the barcode points to a 'stocklocation' if (data.containsKey("stocklocation")) { - int _loc = (data["stocklocation"]["pk"] ?? -1) as int; + int _loc = (data["stocklocation"]?["pk"] ?? -1) as int; // A valid stock location! if (_loc > 0) { @@ -83,7 +83,7 @@ class BarcodeScanStockItemHandler extends BarcodeHandler { Future onBarcodeMatched(Map data) async { // We expect that the barcode points to a 'stockitem' if (data.containsKey("stockitem")) { - int _item = (data["stockitem"]["pk"] ?? -1) as int; + int _item = (data["stockitem"]?["pk"] ?? -1) as int; // A valid stock location! if (_item > 0) { diff --git a/lib/widget/company/supplier_part_detail.dart b/lib/widget/company/supplier_part_detail.dart index c6c50468..1570c384 100644 --- a/lib/widget/company/supplier_part_detail.dart +++ b/lib/widget/company/supplier_part_detail.dart @@ -210,7 +210,18 @@ class _SupplierPartDisplayState extends RefreshableState ManufacturerPartDetailWidget(manufacturerPart) + )); + } + }, ) ); } From 4121e02242f9d04fd7c638fb5b6ef5b7e70e61a9 Mon Sep 17 00:00:00 2001 From: Oliver Date: Thu, 12 Dec 2024 11:40:02 +1100 Subject: [PATCH 04/21] Refactoring for null checks --- lib/inventree/company.dart | 39 ++++++++++++++++---- lib/inventree/stock.dart | 12 +++--- lib/widget/company/supplier_part_detail.dart | 12 +++--- 3 files changed, 45 insertions(+), 18 deletions(-) diff --git a/lib/inventree/company.dart b/lib/inventree/company.dart index b9bba772..18c980f7 100644 --- a/lib/inventree/company.dart +++ b/lib/inventree/company.dart @@ -171,7 +171,7 @@ class InvenTreeSupplierPart extends InvenTreeModel { String get MPN => getString("MPN", subKey: "manufacturer_part_detail"); - String get manufacturerImage => (jsondata["manufacturer_detail"]?["image"] ?? jsondata["manufacturer_detail"]["thumbnail"] ?? InvenTreeAPI.staticThumb) as String; + String get manufacturerImage => (jsondata["manufacturer_detail"]?["image"] ?? jsondata["manufacturer_detail"]?["thumbnail"] ?? InvenTreeAPI.staticThumb) as String; int get manufacturerPartId => getInt("manufacturer_part"); @@ -179,14 +179,14 @@ class InvenTreeSupplierPart extends InvenTreeModel { String get supplierName => getString("name", subKey: "supplier_detail"); - String get supplierImage => (jsondata["supplier_detail"]?["image"] ?? jsondata["supplier_detail"]["thumbnail"] ?? InvenTreeAPI.staticThumb) as String; + String get supplierImage => (jsondata["supplier_detail"]?["image"] ?? jsondata["supplier_detail"]?["thumbnail"] ?? InvenTreeAPI.staticThumb) as String; String get SKU => getString("SKU"); bool get active => getBool("active", backup: true); int get partId => getInt("part"); - + String get partImage => (jsondata["part_detail"]?["thumbnail"] ?? InvenTreeAPI.staticThumb) as String; String get partName => getString("name", subKey: "part_detail"); @@ -222,18 +222,43 @@ class InvenTreeManufacturerPart extends InvenTreeModel { String url = "company/part/manufacturer/"; @override - Map defaultListFilters() { + Map> formFields() { + Map> fields = { + "manufacturer": {}, + "MPN": {}, + }; + + return fields; + } + + @override + Map defaultFilters() { return { "manufacturer_detail": "true", + "part_detail": "true", }; } int get partId => getInt("part"); - + + String get partName => getString("name", subKey: "part_detail"); + + String get partDescription => getString("description", subKey: "part_detail"); + + String get partIPN => getString("IPN", subKey: "part_detail"); + + String get partImage => (jsondata["part_detail"]?["thumbnail"] ?? InvenTreeAPI.staticThumb) as String; + int get manufacturerId => getInt("manufacturer"); - + + String get manufacturerName => getString("name", subKey: "manufacturer_detail"); + + String get manufacturerDescription => getString("description", subKey: "manufacturer_detail"); + + String get manufacturerImage => (jsondata["manufacturer_detail"]?["image"] ?? jsondata["manufacturer_detail"]?["thumbnail"] ?? InvenTreeAPI.staticThumb) as String; + String get MPN => getString("MPN"); - + @override InvenTreeModel createFromJson(Map json) => InvenTreeManufacturerPart.fromJson(json); } diff --git a/lib/inventree/stock.dart b/lib/inventree/stock.dart index 7b03d18d..00e4cde9 100644 --- a/lib/inventree/stock.dart +++ b/lib/inventree/stock.dart @@ -98,7 +98,7 @@ class InvenTreeStockItemHistory extends InvenTreeModel { String get URL => "stock/track/"; @override - Map defaultListFilters() { + Map defaultFilters() { // By default, order by decreasing date return { @@ -206,7 +206,7 @@ class InvenTreeStockItem extends InvenTreeModel { if (isSerialized()) { // Prevent editing of 'quantity' field if the item is serialized - fields["quantity"]["hidden"] = true; + fields["quantity"]?["hidden"] = true; } // Old API does not support these fields @@ -395,7 +395,7 @@ class InvenTreeStockItem extends InvenTreeModel { // Use the detailed part information as priority if (jsondata.containsKey("part_detail")) { - nm = (jsondata["part_detail"]["full_name"] ?? "") as String; + nm = (jsondata["part_detail"]?["full_name"] ?? "") as String; } // Backup if first value fails @@ -411,7 +411,7 @@ class InvenTreeStockItem extends InvenTreeModel { // Use the detailed part description as priority if (jsondata.containsKey("part_detail")) { - desc = (jsondata["part_detail"]["description"] ?? "") as String; + desc = (jsondata["part_detail"]?["description"] ?? "") as String; } if (desc.isEmpty) { @@ -425,7 +425,7 @@ class InvenTreeStockItem extends InvenTreeModel { String img = ""; if (jsondata.containsKey("part_detail")) { - img = (jsondata["part_detail"]["thumbnail"] ?? "") as String; + img = (jsondata["part_detail"]?["thumbnail"] ?? "") as String; } if (img.isEmpty) { @@ -468,7 +468,7 @@ class InvenTreeStockItem extends InvenTreeModel { if (jsondata.containsKey("supplier_part_detail")) { thumb = (jsondata["supplier_part_detail"]?["supplier_detail"]?["image"] ?? "") as String; } else if (jsondata.containsKey("supplier_detail")) { - thumb = (jsondata["supplier_detail"]["image"] ?? "") as String; + thumb = (jsondata["supplier_detail"]?["image"] ?? "") as String; } return thumb; diff --git a/lib/widget/company/supplier_part_detail.dart b/lib/widget/company/supplier_part_detail.dart index 1570c384..c287f0a6 100644 --- a/lib/widget/company/supplier_part_detail.dart +++ b/lib/widget/company/supplier_part_detail.dart @@ -1,21 +1,23 @@ import "package:flutter/material.dart"; import "package:flutter_speed_dial/flutter_speed_dial.dart"; import "package:flutter_tabler_icons/flutter_tabler_icons.dart"; +import "package:url_launcher/url_launcher.dart"; import "package:inventree/api.dart"; import "package:inventree/app_colors.dart"; -import "package:inventree/barcode/barcode.dart"; import "package:inventree/l10.dart"; +import "package:inventree/barcode/barcode.dart"; + import "package:inventree/inventree/part.dart"; import "package:inventree/inventree/company.dart"; -import "package:inventree/widget/company/company_detail.dart"; -import "package:inventree/widget/part/part_detail.dart"; import "package:inventree/widget/progress.dart"; import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/snacks.dart"; -import "package:url_launcher/url_launcher.dart"; +import "package:inventree/widget/company/company_detail.dart"; +import "package:inventree/widget/company/manufacturer_part_detail.dart"; +import "package:inventree/widget/part/part_detail.dart"; /* @@ -180,7 +182,7 @@ class _SupplierPartDisplayState extends RefreshableState Date: Thu, 12 Dec 2024 11:52:28 +1100 Subject: [PATCH 05/21] Ignore selected errors in sentry --- lib/inventree/sentry.dart | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) diff --git a/lib/inventree/sentry.dart b/lib/inventree/sentry.dart index 34894ff4..a052ed57 100644 --- a/lib/inventree/sentry.dart +++ b/lib/inventree/sentry.dart @@ -158,6 +158,11 @@ Future sentryReportMessage(String message, {Map? context}) */ Future sentryReportError(String source, dynamic error, StackTrace? stackTrace, {Map context = const {}}) async { + if (sentryIgnoreError(error)) { + // No action on this error + return; + } + print("----- Sentry Intercepted error: $error -----"); print(stackTrace); @@ -228,3 +233,18 @@ Future sentryReportError(String source, dynamic error, StackTrace? stackTr print("Uploaded information to Sentry.io : ${response.toString()}"); }); } + + +/* + * Test if a certain error should be ignored by Sentry + */ +bool sentryIgnoreError(dynamic error) { + // Ignore 404 errors for media files + if (error is HttpException) { + if (error.uri.toString().contains("/media/") && error.message.contains("404")) { + return true; + } + } + + return false; +} \ No newline at end of file From 04cef7cfc449eadaca8c252eabe3d959f32b2de2 Mon Sep 17 00:00:00 2001 From: Oliver Date: Thu, 12 Dec 2024 11:52:46 +1100 Subject: [PATCH 06/21] Fix API implementation for ManufacturerPart --- lib/inventree/company.dart | 6 +++++- lib/widget/company/manufacturer_part_detail.dart | 11 +++++++++++ 2 files changed, 16 insertions(+), 1 deletion(-) diff --git a/lib/inventree/company.dart b/lib/inventree/company.dart index 18c980f7..eddbd5ae 100644 --- a/lib/inventree/company.dart +++ b/lib/inventree/company.dart @@ -219,13 +219,17 @@ class InvenTreeManufacturerPart extends InvenTreeModel { InvenTreeManufacturerPart.fromJson(Map json) : super.fromJson(json); @override - String url = "company/part/manufacturer/"; + String URL = "company/part/manufacturer/"; + + @override + List get rolesRequired => ["part"]; @override Map> formFields() { Map> fields = { "manufacturer": {}, "MPN": {}, + "link": {}, }; return fields; diff --git a/lib/widget/company/manufacturer_part_detail.dart b/lib/widget/company/manufacturer_part_detail.dart index 7b62124c..e9fec8fb 100644 --- a/lib/widget/company/manufacturer_part_detail.dart +++ b/lib/widget/company/manufacturer_part_detail.dart @@ -151,6 +151,17 @@ class _ManufacturerPartDisplayState extends RefreshableState Date: Thu, 12 Dec 2024 11:57:24 +1100 Subject: [PATCH 07/21] Update release notes --- assets/release_notes.md | 3 +++ lib/widget/company/manufacturer_part_detail.dart | 2 +- 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/assets/release_notes.md b/assets/release_notes.md index 1b226592..8b0b61d1 100644 --- a/assets/release_notes.md +++ b/assets/release_notes.md @@ -1,12 +1,15 @@ ### 0.17.1 - December 2024 --- +- Add support for ManufacturerPart model +- Support barcode scanning for ManufacturerPart - Fixes barcode scanning bug which prevents scanning of DataMatrix codes - Display "destination" information in PurchaseOrder detail view - Pre-fill "location" field when receiving items against PurchaseOrder - Fix display of part name in PurchaseOrderLineItem list - Adds "assigned to me" filter for Purchase Order list - Adds "assigned to me" filter for Sales Order list +- Updated translations ### 0.17.0 - December 2024 --- diff --git a/lib/widget/company/manufacturer_part_detail.dart b/lib/widget/company/manufacturer_part_detail.dart index e9fec8fb..046cc971 100644 --- a/lib/widget/company/manufacturer_part_detail.dart +++ b/lib/widget/company/manufacturer_part_detail.dart @@ -124,7 +124,7 @@ class _ManufacturerPartDisplayState extends RefreshableState Date: Thu, 12 Dec 2024 13:25:38 +1100 Subject: [PATCH 08/21] More error handling --- lib/barcode/handler.dart | 19 ++++++++++++++++--- 1 file changed, 16 insertions(+), 3 deletions(-) diff --git a/lib/barcode/handler.dart b/lib/barcode/handler.dart index 20dc9e92..2b9176d2 100644 --- a/lib/barcode/handler.dart +++ b/lib/barcode/handler.dart @@ -79,14 +79,27 @@ class BarcodeHandler { return; } - var response = await InvenTreeAPI().post( + APIResponse? response; + + try { + response = await InvenTreeAPI().post( url, body: { "barcode": barcode, ...extra_data, }, - expectedStatusCode: null, // Do not show an error on "unexpected code" - ); + expectedStatusCode: null, // Do not show an error on "unexpected code" + ); + } catch (error, stackTrace) { + sentryReportError("Barcode.processBarcode", error, stackTrace); + response = null; + } + + if (response == null) { + barcodeFailureTone(); + showSnackIcon(L10().barcodeError, success: false); + return; + } debug("Barcode scan response" + response.data.toString()); From f3dc87a1e5ac836a78035e87c7b95310f35baff4 Mon Sep 17 00:00:00 2001 From: Oliver Date: Thu, 12 Dec 2024 13:31:13 +1100 Subject: [PATCH 09/21] Decode quantity betterer --- lib/barcode/purchase_order.dart | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/lib/barcode/purchase_order.dart b/lib/barcode/purchase_order.dart index e8bba611..8a215d37 100644 --- a/lib/barcode/purchase_order.dart +++ b/lib/barcode/purchase_order.dart @@ -66,6 +66,8 @@ class POReceiveBarcodeHandler extends BarcodeHandler { showSnackIcon(L10().missingData, success: false); } + double quantity = double.tryParse((lineItemData["quantity"] ?? "0").toString()) ?? 0; + // Construct fields to receive Map fields = { "line_item": { @@ -77,7 +79,7 @@ class POReceiveBarcodeHandler extends BarcodeHandler { "quantity": { "parent": "items", "nested": true, - "value": lineItemData["quantity"] as double?, + "value": quantity, }, "status": { "parent": "items", From 3abb29e152ad6cfe16dbfc950d9d7fa3c4fdda94 Mon Sep 17 00:00:00 2001 From: Oliver Date: Thu, 12 Dec 2024 14:24:31 +1100 Subject: [PATCH 10/21] Refactoring --- lib/barcode/camera_controller.dart | 12 ++--- lib/barcode/handler.dart | 2 +- lib/barcode/purchase_order.dart | 81 ++++++++++++---------------- lib/inventree/purchase_order.dart | 81 ++++++++++++++++++++++++++++ lib/widget/order/po_line_detail.dart | 65 ++-------------------- 5 files changed, 125 insertions(+), 116 deletions(-) diff --git a/lib/barcode/camera_controller.dart b/lib/barcode/camera_controller.dart index 4c6edd7a..0ce08c7e 100644 --- a/lib/barcode/camera_controller.dart +++ b/lib/barcode/camera_controller.dart @@ -93,7 +93,7 @@ class _CameraBarcodeControllerState extends InvenTreeBarcodeControllerState { /* * Callback function when a barcode is scanned */ - void _onScanSuccess(Code? code) { + void _onScanSuccess(Code? code) async { if (scanning_paused) { return; @@ -122,18 +122,16 @@ class _CameraBarcodeControllerState extends InvenTreeBarcodeControllerState { if (mounted) { setState(() { scanned_code = barcode; - scanning_paused = barcode.isNotEmpty; }); } if (barcode.isNotEmpty) { - handleBarcodeData(barcode).then((_) { + pauseScan(); + + await handleBarcodeData(barcode).then((_) { if (!single_scanning && mounted) { - // Resume next scan - setState(() { - scanning_paused = false; - }); + resumeScan(); } }); } diff --git a/lib/barcode/handler.dart b/lib/barcode/handler.dart index 2b9176d2..48831f9a 100644 --- a/lib/barcode/handler.dart +++ b/lib/barcode/handler.dart @@ -107,7 +107,7 @@ class BarcodeHandler { // Handle strange response from the server if (!response.isValid() || !response.isMap()) { - onBarcodeUnknown({}); + await onBarcodeUnknown({}); showSnackIcon(L10().serverError, success: false); diff --git a/lib/barcode/purchase_order.dart b/lib/barcode/purchase_order.dart index 8a215d37..44a835b7 100644 --- a/lib/barcode/purchase_order.dart +++ b/lib/barcode/purchase_order.dart @@ -1,9 +1,6 @@ import "package:flutter/material.dart"; -import "package:flutter_tabler_icons/flutter_tabler_icons.dart"; import "package:one_context/one_context.dart"; - import "package:inventree/l10.dart"; -import "package:inventree/api_form.dart"; import "package:inventree/barcode/barcode.dart"; import "package:inventree/barcode/handler.dart"; @@ -66,51 +63,41 @@ class POReceiveBarcodeHandler extends BarcodeHandler { showSnackIcon(L10().missingData, success: false); } - double quantity = double.tryParse((lineItemData["quantity"] ?? "0").toString()) ?? 0; - - // Construct fields to receive - Map fields = { - "line_item": { - "parent": "items", - "nested": true, - "hidden": true, - "value": lineItemData["pk"] as int, - }, - "quantity": { - "parent": "items", - "nested": true, - "value": quantity, - }, - "status": { - "parent": "items", - "nested": true, - }, - "location": { - "value": lineItemData["location"] as int?, - }, - "barcode": { - "parent": "items", - "nested": true, - "hidden": true, - "type": "barcode", - "value": data["barcode_data"] as String, - } - }; + // At minimum, we need the line item ID value + final int? lineItemId = lineItemData["pk"] as int?; - final context = OneContext().context!; - final purchase_order_pk = lineItemData["purchase_order"]; - final receive_url = "${InvenTreePurchaseOrder().URL}${purchase_order_pk}/receive/"; - - launchApiForm( - context, - L10().receiveItem, - receive_url, - fields, - method: "POST", - icon: TablerIcons.transition_right, - onSuccess: (data) async { - showSnackIcon(L10().receivedItem, success: true); - } + if (lineItemId == null) { + barcodeFailureTone(); + return; + } + + InvenTreePOLineItem? lineItem = await InvenTreePOLineItem().get(lineItemId) as InvenTreePOLineItem?; + + if (lineItem == null) { + barcodeFailureTone(); + return; + } + + // Next, extract the "optional" fields + + // Extract information from the returned server response + double? quantity = double.tryParse((lineItemData["quantity"] ?? "0").toString()); + int? destination = lineItemData["location"] as int?; + String? barcode = data["barcode_data"] as String?; + + // Discard the barcode scanner at this stage + if (OneContext.hasContext) { + OneContext().pop(); + } + + await lineItem.receive( + OneContext().context!, + destination: destination, + quantity: quantity, + barcode: barcode, + onSuccess: () { + showSnackIcon(L10().receivedItem, success: true); + } ); } diff --git a/lib/inventree/purchase_order.dart b/lib/inventree/purchase_order.dart index 82cb0f34..69ed71ff 100644 --- a/lib/inventree/purchase_order.dart +++ b/lib/inventree/purchase_order.dart @@ -1,3 +1,5 @@ +import "package:flutter/cupertino.dart"; +import "package:flutter_tabler_icons/flutter_tabler_icons.dart"; import "package:inventree/api.dart"; import "package:inventree/helpers.dart"; import "package:inventree/inventree/company.dart"; @@ -5,6 +7,9 @@ import "package:inventree/inventree/model.dart"; import "package:inventree/inventree/orders.dart"; import "package:inventree/widget/progress.dart"; +import "package:inventree/api_form.dart"; +import "package:inventree/l10.dart"; + /* * Class representing an individual PurchaseOrder instance @@ -212,6 +217,16 @@ class InvenTreePOLineItem extends InvenTreeOrderLine { } } + InvenTreePurchaseOrder? get purchaseOrder { + dynamic detail = jsondata["order_detail"]; + + if (detail == null) { + return null; + } else { + return InvenTreePurchaseOrder.fromJson(detail as Map); + } + } + String get SKU => getString("SKU", subKey: "supplier_part_detail"); double get purchasePrice => getDouble("purchase_price"); @@ -223,6 +238,72 @@ class InvenTreePOLineItem extends InvenTreeOrderLine { Map get orderDetail => getMap("order_detail"); Map get destinationDetail => getMap("destination_detail"); + + // Receive this line item into stock + Future receive(BuildContext context, {int? destination, double? quantity, String? barcode, Function? onSuccess}) async { + // Infer the destination location from the line item if not provided + if (destinationId > 0) { + destination = destinationId; + } + + destination ??= (orderDetail["destination"]) as int?; + + quantity ??= outstanding; + + // Construct form fields + Map fields = { + "line_item": { + "parent": "items", + "nested": true, + "hidden": true, + "value": pk, + }, + "quantity": { + "parent": "items", + "nested": true, + "value": quantity, + }, + "location": {}, + "status": { + "parent": "items", + "nested": true, + }, + "batch_code": { + "parent": "items", + "nested": true, + }, + "barcode": { + "parent": "items", + "nested": true, + "type": "barcode", + "label": L10().barcodeAssign, + "value": barcode, + "required": false, + } + }; + + if (destination != null && destination > 0) { + fields["location"]?["value"] = destination; + } + + InvenTreePurchaseOrder? order = purchaseOrder; + + if (order != null) { + await launchApiForm( + context, + L10().receiveItem, + order.receive_url, + fields, + method: "POST", + icon: TablerIcons.transition_right, + onSuccess: (data) { + if (onSuccess != null) { + onSuccess(); + } + } + ); + } + } } /* diff --git a/lib/widget/order/po_line_detail.dart b/lib/widget/order/po_line_detail.dart index 06e0a735..7a0c9085 100644 --- a/lib/widget/order/po_line_detail.dart +++ b/lib/widget/order/po_line_detail.dart @@ -132,70 +132,13 @@ class _POLineDetailWidgetState extends RefreshableState { // Launch a form to 'receive' this line item Future receiveLineItem(BuildContext context) async { - - // Pre-fill the "destination" to receive into - int destination = widget.item.destinationId; - - if (destination < 0) { - destination = (widget.item.orderDetail["destination"] ?? -1) as int; - } - - // Construct fields to receive - Map fields = { - "line_item": { - "parent": "items", - "nested": true, - "hidden": true, - "value": widget.item.pk, - }, - "quantity": { - "parent": "items", - "nested": true, - "value": widget.item.outstanding, - }, - "status": { - "parent": "items", - "nested": true, - }, - "location": {}, - "batch_code": { - "parent": "items", - "nested": true, - }, - "barcode": { - "parent": "items", - "nested": true, - "type": "barcode", - "label": L10().barcodeAssign, - "required": false, - } - }; - - if (destination > 0) { - fields["location"]?["value"] = destination; - } - - showLoadingOverlay(); - var order = await InvenTreePurchaseOrder().get(widget.item.orderId); - hideLoadingOverlay(); - - if (order is InvenTreePurchaseOrder) { - launchApiForm( + widget.item.receive( context, - L10().receiveItem, - order.receive_url, - fields, - method: "POST", - icon: TablerIcons.transition_right, - onSuccess: (data) async { - showSnackIcon(L10().receivedItem, success: true); - refresh(context); + onSuccess: () => { + showSnackIcon(L10().receivedItem, success: true), + refresh(context) } ); - } else { - showSnackIcon(L10().error); - return; - } } @override From 2375739b58ddddd4191741e4d4a144cdc50f3d34 Mon Sep 17 00:00:00 2001 From: Oliver Date: Thu, 12 Dec 2024 14:36:42 +1100 Subject: [PATCH 11/21] Add option to confirm checkin details --- lib/barcode/purchase_order.dart | 6 +++++- lib/l10n/app_en.arb | 6 ++++++ lib/preferences.dart | 1 + lib/settings/purchase_order_settings.dart | 16 ++++++++++++++++ 4 files changed, 28 insertions(+), 1 deletion(-) diff --git a/lib/barcode/purchase_order.dart b/lib/barcode/purchase_order.dart index 44a835b7..7f908ea0 100644 --- a/lib/barcode/purchase_order.dart +++ b/lib/barcode/purchase_order.dart @@ -1,4 +1,5 @@ import "package:flutter/material.dart"; +import "package:inventree/preferences.dart"; import "package:one_context/one_context.dart"; import "package:inventree/l10.dart"; @@ -31,11 +32,14 @@ class POReceiveBarcodeHandler extends BarcodeHandler { @override Future processBarcode(String barcode, {String url = "barcode/po-receive/", - Map extra_data = const {}}) { + Map extra_data = const {}}) async { + + final bool confirm = await InvenTreeSettingsManager().getBool(INV_PO_CONFIRM_SCAN, true); final po_extra_data = { "purchase_order": purchaseOrder?.pk, "location": location?.pk, + "auto_allocate": !confirm, ...extra_data, }; diff --git a/lib/l10n/app_en.arb b/lib/l10n/app_en.arb index 99be420e..ef3b9d96 100644 --- a/lib/l10n/app_en.arb +++ b/lib/l10n/app_en.arb @@ -924,6 +924,12 @@ "projectCode": "Project Code", "@projectCode": {}, + "purchaseOrderConfirmScan": "Confirm Scan Data", + "@purchaseOrderConfirmScan": {}, + + "purchaseOrderConfirmScanDetail": "Confirm details when scanning in items", + "@purchaseOrderConfirmScanDetail": {}, + "purchaseOrderEnable": "Enable Purchase Orders", "@purchaseOrderEnable": {}, diff --git a/lib/preferences.dart b/lib/preferences.dart index 57004873..8507cd31 100644 --- a/lib/preferences.dart +++ b/lib/preferences.dart @@ -39,6 +39,7 @@ const String INV_STOCK_CONFIRM_SCAN = "stockConfirmScan"; // Purchase order settings const String INV_PO_ENABLE = "poEnable"; const String INV_PO_SHOW_CAMERA = "poShowCamera"; +const String INV_PO_CONFIRM_SCAN = "poConfirmScan"; // Sales order settings const String INV_SO_ENABLE = "soEnable"; diff --git a/lib/settings/purchase_order_settings.dart b/lib/settings/purchase_order_settings.dart index 5a3b430e..888b89b1 100644 --- a/lib/settings/purchase_order_settings.dart +++ b/lib/settings/purchase_order_settings.dart @@ -19,6 +19,7 @@ class _InvenTreePurchaseOrderSettingsState extends State loadSettings() async { poEnable = await InvenTreeSettingsManager().getBool(INV_PO_ENABLE, true); poShowCamera = await InvenTreeSettingsManager().getBool(INV_PO_SHOW_CAMERA, true); + poConfirmScan = await InvenTreeSettingsManager().getBool(INV_PO_CONFIRM_SCAN, true); if (mounted) { setState(() { @@ -75,6 +77,20 @@ class _InvenTreePurchaseOrderSettingsState extends State Date: Thu, 12 Dec 2024 15:57:44 +1100 Subject: [PATCH 12/21] Improve response handlign --- lib/barcode/purchase_order.dart | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/lib/barcode/purchase_order.dart b/lib/barcode/purchase_order.dart index 7f908ea0..77a352f9 100644 --- a/lib/barcode/purchase_order.dart +++ b/lib/barcode/purchase_order.dart @@ -48,11 +48,13 @@ class POReceiveBarcodeHandler extends BarcodeHandler { @override Future onBarcodeMatched(Map data) async { - if (!data.containsKey("lineitem")) { + + if (data.containsKey("lineitem") || data.containsKey("success")) { + barcodeSuccess(L10().receivedItem); + return; + } else { return onBarcodeUnknown(data); } - - barcodeSuccess(L10().receivedItem); } @override From 4f99f14bbbdfb4ce426edfa419f2047015c39878 Mon Sep 17 00:00:00 2001 From: Oliver Date: Thu, 12 Dec 2024 16:06:33 +1100 Subject: [PATCH 13/21] Cleanup --- lib/barcode/purchase_order.dart | 4 +++- lib/widget/order/po_line_detail.dart | 6 ++++-- 2 files changed, 7 insertions(+), 3 deletions(-) diff --git a/lib/barcode/purchase_order.dart b/lib/barcode/purchase_order.dart index 77a352f9..8b2f984b 100644 --- a/lib/barcode/purchase_order.dart +++ b/lib/barcode/purchase_order.dart @@ -21,10 +21,11 @@ import "package:inventree/widget/snacks.dart"; */ class POReceiveBarcodeHandler extends BarcodeHandler { - POReceiveBarcodeHandler({this.purchaseOrder, this.location}); + POReceiveBarcodeHandler({this.purchaseOrder, this.location, this.lineItem}); InvenTreePurchaseOrder? purchaseOrder; InvenTreeStockLocation? location; + InvenTreePOLineItem? lineItem; @override String getOverlayText(BuildContext context) => L10().barcodeReceivePart; @@ -39,6 +40,7 @@ class POReceiveBarcodeHandler extends BarcodeHandler { final po_extra_data = { "purchase_order": purchaseOrder?.pk, "location": location?.pk, + "line_item": lineItem?.pk, "auto_allocate": !confirm, ...extra_data, }; diff --git a/lib/widget/order/po_line_detail.dart b/lib/widget/order/po_line_detail.dart index 7a0c9085..eff805c6 100644 --- a/lib/widget/order/po_line_detail.dart +++ b/lib/widget/order/po_line_detail.dart @@ -2,12 +2,14 @@ import "package:flutter/material.dart"; import "package:flutter_speed_dial/flutter_speed_dial.dart"; import "package:flutter_tabler_icons/flutter_tabler_icons.dart"; -import "package:inventree/api_form.dart"; import "package:inventree/app_colors.dart"; import "package:inventree/helpers.dart"; import "package:inventree/inventree/model.dart"; import "package:inventree/l10.dart"; +import "package:inventree/barcode/barcode.dart"; +import "package:inventree/barcode/purchase_order.dart"; + import "package:inventree/inventree/company.dart"; import "package:inventree/inventree/part.dart"; import "package:inventree/inventree/purchase_order.dart"; @@ -140,7 +142,7 @@ class _POLineDetailWidgetState extends RefreshableState { } ); } - + @override List getTiles(BuildContext context) { List tiles = []; From de7dac251812efccc58edb0d86ab19963fdc372f Mon Sep 17 00:00:00 2001 From: Oliver Date: Thu, 12 Dec 2024 16:06:50 +1100 Subject: [PATCH 14/21] Remove unused imports --- lib/widget/order/po_line_detail.dart | 3 --- 1 file changed, 3 deletions(-) diff --git a/lib/widget/order/po_line_detail.dart b/lib/widget/order/po_line_detail.dart index eff805c6..6ae896e1 100644 --- a/lib/widget/order/po_line_detail.dart +++ b/lib/widget/order/po_line_detail.dart @@ -7,9 +7,6 @@ import "package:inventree/helpers.dart"; import "package:inventree/inventree/model.dart"; import "package:inventree/l10.dart"; -import "package:inventree/barcode/barcode.dart"; -import "package:inventree/barcode/purchase_order.dart"; - import "package:inventree/inventree/company.dart"; import "package:inventree/inventree/part.dart"; import "package:inventree/inventree/purchase_order.dart"; From be3d926c46b46df2ab3aa166d4b5aa81215f40b8 Mon Sep 17 00:00:00 2001 From: Oliver Walters Date: Thu, 12 Dec 2024 23:27:45 +1100 Subject: [PATCH 15/21] Fix async function --- lib/barcode/camera_controller.dart | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/lib/barcode/camera_controller.dart b/lib/barcode/camera_controller.dart index 0ce08c7e..ece027c2 100644 --- a/lib/barcode/camera_controller.dart +++ b/lib/barcode/camera_controller.dart @@ -93,7 +93,7 @@ class _CameraBarcodeControllerState extends InvenTreeBarcodeControllerState { /* * Callback function when a barcode is scanned */ - void _onScanSuccess(Code? code) async { + Future onScanSuccess(Code? code) async { if (scanning_paused) { return; @@ -184,7 +184,7 @@ class _CameraBarcodeControllerState extends InvenTreeBarcodeControllerState { Widget BarcodeReader(BuildContext context) { return ReaderWidget( - onScan: _onScanSuccess, + onScan: onScanSuccess, isMultiScan: false, tryHarder: true, tryInverted: true, From 05cf93644ee72226ec88b2ae0d42a52cf28e1625 Mon Sep 17 00:00:00 2001 From: Oliver Date: Fri, 13 Dec 2024 15:30:27 +1100 Subject: [PATCH 16/21] Fix for assigning custom barcode --- lib/barcode/barcode.dart | 21 ++++++--------------- 1 file changed, 6 insertions(+), 15 deletions(-) diff --git a/lib/barcode/barcode.dart b/lib/barcode/barcode.dart index e0ce9a3c..d46e6494 100644 --- a/lib/barcode/barcode.dart +++ b/lib/barcode/barcode.dart @@ -345,21 +345,6 @@ class UniqueBarcodeHandler extends BarcodeHandler { @override Future onBarcodeMatched(Map data) async { - - barcodeFailureTone(); - - // If the barcode is known, we can"t assign it to the stock item! - showSnackIcon( - L10().barcodeInUse, - icon: Icons.qr_code, - success: false - ); - } - - @override - Future onBarcodeUnknown(Map data) async { - // If the barcode is unknown, we *can* assign it to the stock item! - if (!data.containsKey("hash") && !data.containsKey("barcode_hash")) { showServerError( "barcode/", @@ -391,6 +376,12 @@ class UniqueBarcodeHandler extends BarcodeHandler { } } } + + @override + Future onBarcodeUnknown(Map data) async { + await onBarcodeMatched(data); + } + } From fc25c47a32f089f136df40d3fdb298a5551a02a0 Mon Sep 17 00:00:00 2001 From: Oliver Walters Date: Sat, 14 Dec 2024 11:39:34 +1100 Subject: [PATCH 17/21] Handle barcode scan result for company --- lib/barcode/barcode.dart | 20 +++++++++++++++++--- 1 file changed, 17 insertions(+), 3 deletions(-) diff --git a/lib/barcode/barcode.dart b/lib/barcode/barcode.dart index d46e6494..9bb5a32c 100644 --- a/lib/barcode/barcode.dart +++ b/lib/barcode/barcode.dart @@ -32,6 +32,7 @@ import "package:inventree/widget/order/purchase_order_detail.dart"; import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/snacks.dart"; import "package:inventree/widget/stock/stock_detail.dart"; +import "package:inventree/widget/company/company_detail.dart"; import "package:inventree/widget/company/supplier_part_detail.dart"; @@ -200,6 +201,16 @@ class BarcodeScanHandler extends BarcodeHandler { } } + Future handleCompany(int pk) async { + var company = await InvenTreeCompany().get(pk); + + if (company is InvenTreeCompany) { + OneContext().pop(); + OneContext().push(MaterialPageRoute( + builder: (context) => CompanyDetailWidget(company))); + } + } + /* * Response when a "PurchaseOrder" instance is scanned */ @@ -267,9 +278,6 @@ class BarcodeScanHandler extends BarcodeHandler { barcodeSuccessTone(); switch (model) { - case "part": - await handlePart(pk); - return; case "stockitem": await handleStockItem(pk); return; @@ -288,6 +296,12 @@ class BarcodeScanHandler extends BarcodeHandler { case "manufacturerpart": await handleManufacturerPart(pk); return; + case "part": + await handlePart(pk); + return; + case "company": + await handleCompany(pk); + return; default: // Fall through to failure state break; From 0c1a80f062aa8dc8f76091ef7339e136baeee57e Mon Sep 17 00:00:00 2001 From: Oliver Walters Date: Sat, 14 Dec 2024 11:42:34 +1100 Subject: [PATCH 18/21] Fix --- lib/barcode/barcode.dart | 1 + 1 file changed, 1 insertion(+) diff --git a/lib/barcode/barcode.dart b/lib/barcode/barcode.dart index 9bb5a32c..73e52f6b 100644 --- a/lib/barcode/barcode.dart +++ b/lib/barcode/barcode.dart @@ -244,6 +244,7 @@ class BarcodeScanHandler extends BarcodeHandler { // The following model types can be matched with barcodes List validModels = [ "part", + "company", "stockitem", "stocklocation", "supplierpart", From 3474d085f545e90e1e05d9151b4b5ed80806581f Mon Sep 17 00:00:00 2001 From: Oliver Walters Date: Sat, 14 Dec 2024 13:33:13 +1100 Subject: [PATCH 19/21] Adjust scan priority --- lib/barcode/barcode.dart | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/lib/barcode/barcode.dart b/lib/barcode/barcode.dart index 73e52f6b..f4f6396c 100644 --- a/lib/barcode/barcode.dart +++ b/lib/barcode/barcode.dart @@ -282,18 +282,18 @@ class BarcodeScanHandler extends BarcodeHandler { case "stockitem": await handleStockItem(pk); return; - case "stocklocation": - await handleStockLocation(pk); - return; - case "supplierpart": - await handleSupplierPart(pk); - return; case "purchaseorder": await handlePurchaseOrder(pk); return; case "salesorder": await handleSalesOrder(pk); return; + case "stocklocation": + await handleStockLocation(pk); + return; + case "supplierpart": + await handleSupplierPart(pk); + return; case "manufacturerpart": await handleManufacturerPart(pk); return; From d1a655f64309f2fb20a33c6cd1989e26552fdfdb Mon Sep 17 00:00:00 2001 From: Oliver Walters Date: Sat, 14 Dec 2024 14:56:41 +1100 Subject: [PATCH 20/21] Refactoring MODEL_TYPE - Use instead of duplicated const strings --- lib/api_form.dart | 31 +++++++++++++------------ lib/barcode/barcode.dart | 32 +++++++++++++------------- lib/inventree/company.dart | 9 ++++++++ lib/inventree/model.dart | 2 +- lib/inventree/part.dart | 8 ++++++- lib/inventree/project_code.dart | 3 +++ lib/inventree/purchase_order.dart | 2 +- lib/inventree/sales_order.dart | 5 +++- lib/inventree/stock.dart | 4 ++-- lib/widget/part/part_detail.dart | 2 +- lib/widget/stock/location_display.dart | 2 +- lib/widget/stock/stock_detail.dart | 2 +- 12 files changed, 62 insertions(+), 40 deletions(-) diff --git a/lib/api_form.dart b/lib/api_form.dart index f8e335d9..96af188b 100644 --- a/lib/api_form.dart +++ b/lib/api_form.dart @@ -562,11 +562,11 @@ class APIFormField { Map data = item as Map; switch (model) { - case "part": + case InvenTreePart.MODEL_TYPE: return InvenTreePart.fromJson(data).fullname; - case "partcategory": + case InvenTreePartCategory.MODEL_TYPE: return InvenTreePartCategory.fromJson(data).pathstring; - case "stocklocation": + case InvenTreeStockLocation.MODEL_TYPE: return InvenTreeStockLocation.fromJson(data).pathstring; default: return "itemAsString not implemented for '${model}'"; @@ -606,10 +606,12 @@ class APIFormField { Map _relatedFieldFilters() { switch (model) { - case "supplierpart": + case InvenTreeSupplierPart.MODEL_TYPE: return InvenTreeSupplierPart().defaultListFilters(); - case "stockitem": + case InvenTreeStockItem.MODEL_TYPE: return InvenTreeStockItem().defaultListFilters(); + default: + break; } return {}; @@ -643,7 +645,7 @@ class APIFormField { } switch (model) { - case "part": + case InvenTreePart.MODEL_TYPE: var part = InvenTreePart.fromJson(data); return ListTile( @@ -657,14 +659,14 @@ class APIFormField { ) : null, leading: extended ? InvenTreeAPI().getThumbnail(part.thumbnail) : null, ); - case "parttesttemplate": + case InvenTreePartTestTemplate.MODEL_TYPE: var template = InvenTreePartTestTemplate.fromJson(data); return ListTile( title: Text(template.testName), subtitle: Text(template.description), ); - case "supplierpart": + case InvenTreeSupplierPart.MODEL_TYPE: var part = InvenTreeSupplierPart.fromJson(data); return ListTile( @@ -673,7 +675,7 @@ class APIFormField { leading: extended ? InvenTreeAPI().getThumbnail(part.partImage) : null, trailing: extended && part.supplierImage.isNotEmpty ? InvenTreeAPI().getThumbnail(part.supplierImage) : null, ); - case "partcategory": + case InvenTreePartCategory.MODEL_TYPE: var cat = InvenTreePartCategory.fromJson(data); @@ -687,7 +689,7 @@ class APIFormField { style: TextStyle(fontWeight: selected ? FontWeight.bold : FontWeight.normal), ) : null, ); - case "stockitem": + case InvenTreeStockItem.MODEL_TYPE: var item = InvenTreeStockItem.fromJson(data); return ListTile( @@ -697,8 +699,7 @@ class APIFormField { leading: InvenTreeAPI().getThumbnail(item.partThumbnail), trailing: Text(item.quantityString()), ); - case "stocklocation": - + case InvenTreeStockLocation.MODEL_TYPE: var loc = InvenTreeStockLocation.fromJson(data); return ListTile( @@ -711,7 +712,7 @@ class APIFormField { style: TextStyle(fontWeight: selected ? FontWeight.bold : FontWeight.normal), ) : null, ); - case "salesordershipment": + case InvenTreeSalesOrderShipment.MODEL_TYPE: var shipment = InvenTreeSalesOrderShipment.fromJson(data); return ListTile( @@ -733,14 +734,14 @@ class APIFormField { title: Text(name), subtitle: Text(role), ); - case "company": + case InvenTreeCompany.MODEL_TYPE: var company = InvenTreeCompany.fromJson(data); return ListTile( title: Text(company.name), subtitle: extended ? Text(company.description) : null, leading: InvenTreeAPI().getThumbnail(company.thumbnail) ); - case "projectcode": + case InvenTreeProjectCode.MODEL_TYPE: var project_code = InvenTreeProjectCode.fromJson(data); return ListTile( title: Text(project_code.code), diff --git a/lib/barcode/barcode.dart b/lib/barcode/barcode.dart index f4f6396c..0de089e7 100644 --- a/lib/barcode/barcode.dart +++ b/lib/barcode/barcode.dart @@ -243,18 +243,18 @@ class BarcodeScanHandler extends BarcodeHandler { // The following model types can be matched with barcodes List validModels = [ - "part", - "company", - "stockitem", - "stocklocation", - "supplierpart", - "manufacturerpart", + InvenTreePart.MODEL_TYPE, + InvenTreeCompany.MODEL_TYPE, + InvenTreeStockItem.MODEL_TYPE, + InvenTreeStockLocation.MODEL_TYPE, + InvenTreeSupplierPart.MODEL_TYPE, + InvenTreeManufacturerPart.MODEL_TYPE, ]; if (InvenTreeAPI().supportsOrderBarcodes) { - validModels.add("purchaseorder"); - validModels.add("salesorder"); + validModels.add(InvenTreePurchaseOrder.MODEL_TYPE); + validModels.add(InvenTreeSalesOrder.MODEL_TYPE); } for (var key in validModels) { @@ -279,28 +279,28 @@ class BarcodeScanHandler extends BarcodeHandler { barcodeSuccessTone(); switch (model) { - case "stockitem": + case InvenTreeStockItem.MODEL_TYPE: await handleStockItem(pk); return; - case "purchaseorder": + case InvenTreePurchaseOrder.MODEL_TYPE: await handlePurchaseOrder(pk); return; - case "salesorder": + case InvenTreeSalesOrder.MODEL_TYPE: await handleSalesOrder(pk); return; - case "stocklocation": + case InvenTreeStockLocation.MODEL_TYPE: await handleStockLocation(pk); return; - case "supplierpart": + case InvenTreeSupplierPart.MODEL_TYPE: await handleSupplierPart(pk); return; - case "manufacturerpart": + case InvenTreeManufacturerPart.MODEL_TYPE: await handleManufacturerPart(pk); return; - case "part": + case InvenTreePart.MODEL_TYPE: await handlePart(pk); return; - case "company": + case InvenTreeCompany.MODEL_TYPE: await handleCompany(pk); return; default: diff --git a/lib/inventree/company.dart b/lib/inventree/company.dart index eddbd5ae..9be993e9 100644 --- a/lib/inventree/company.dart +++ b/lib/inventree/company.dart @@ -18,6 +18,9 @@ class InvenTreeCompany extends InvenTreeModel { @override String get URL => "company/"; + @override + static const String MODEL_TYPE = "company"; + @override List get rolesRequired => ["purchase_order", "sales_order", "return_order"]; @@ -128,6 +131,9 @@ class InvenTreeSupplierPart extends InvenTreeModel { @override String get URL => "company/part/"; + @override + static const String MODEL_TYPE = "supplierpart"; + @override List get rolesRequired => ["part", "purchase_order"]; @@ -221,6 +227,9 @@ class InvenTreeManufacturerPart extends InvenTreeModel { @override String URL = "company/part/manufacturer/"; + @override + static const String MODEL_TYPE = "manufacturerpart"; + @override List get rolesRequired => ["part"]; diff --git a/lib/inventree/model.dart b/lib/inventree/model.dart index 6d0e890a..42ced88e 100644 --- a/lib/inventree/model.dart +++ b/lib/inventree/model.dart @@ -66,7 +66,7 @@ class InvenTreeModel { String get WEB_URL => ""; // Return the "model type" of this model - String get MODEL_TYPE => ""; + static const String MODEL_TYPE = ""; // Helper function to set a value in the JSON data void setValue(String key, dynamic value) { diff --git a/lib/inventree/part.dart b/lib/inventree/part.dart index 103fa58c..375b2a24 100644 --- a/lib/inventree/part.dart +++ b/lib/inventree/part.dart @@ -23,6 +23,9 @@ class InvenTreePartCategory extends InvenTreeModel { @override String get URL => "part/category/"; + @override + static const String MODEL_TYPE = "partcategory"; + @override List get rolesRequired => ["part_category"]; @@ -79,6 +82,9 @@ class InvenTreePartTestTemplate extends InvenTreeModel { @override String get URL => "part/test-template/"; + @override + static const String MODEL_TYPE = "parttesttemplate"; + String get key => getString("key"); String get testName => getString("test_name"); @@ -193,7 +199,7 @@ class InvenTreePart extends InvenTreeModel { String get URL => "part/"; @override - String get MODEL_TYPE => "part"; + static const String MODEL_TYPE = "part"; @override List get rolesRequired => ["part"]; diff --git a/lib/inventree/project_code.dart b/lib/inventree/project_code.dart index 07a0c19d..fbd2ad17 100644 --- a/lib/inventree/project_code.dart +++ b/lib/inventree/project_code.dart @@ -16,6 +16,9 @@ class InvenTreeProjectCode extends InvenTreeModel { @override String get URL => "project-code/"; + @override + static const String MODEL_TYPE = "projectcode"; + @override Map> formFields() { return { diff --git a/lib/inventree/purchase_order.dart b/lib/inventree/purchase_order.dart index 69ed71ff..cfe808c5 100644 --- a/lib/inventree/purchase_order.dart +++ b/lib/inventree/purchase_order.dart @@ -27,7 +27,7 @@ class InvenTreePurchaseOrder extends InvenTreeOrder { String get URL => "order/po/"; @override - String get MODEL_TYPE => "purchaseorder"; + static const String MODEL_TYPE = "purchaseorder"; @override List get rolesRequired => ["purchase_order"]; diff --git a/lib/inventree/sales_order.dart b/lib/inventree/sales_order.dart index 9acc374b..b55db140 100644 --- a/lib/inventree/sales_order.dart +++ b/lib/inventree/sales_order.dart @@ -25,7 +25,7 @@ class InvenTreeSalesOrder extends InvenTreeOrder { String get URL => "order/so/"; @override - String get MODEL_TYPE => "salesorder"; + static const String MODEL_TYPE = "salesorder"; @override List get rolesRequired => ["sales_order"]; @@ -250,6 +250,9 @@ class InvenTreeSalesOrderShipment extends InvenTreeModel { @override String get URL => "/order/so/shipment/"; + @override + static const String MODEL_TYPE = "salesordershipment"; + @override Map> formFields() { Map> fields = { diff --git a/lib/inventree/stock.dart b/lib/inventree/stock.dart index 00e4cde9..be2bea3a 100644 --- a/lib/inventree/stock.dart +++ b/lib/inventree/stock.dart @@ -169,7 +169,7 @@ class InvenTreeStockItem extends InvenTreeModel { String get URL => "stock/"; @override - String get MODEL_TYPE => "stockitem"; + static const String MODEL_TYPE = "stockitem"; @override List get rolesRequired => ["stock"]; @@ -682,7 +682,7 @@ class InvenTreeStockLocation extends InvenTreeModel { String get URL => "stock/location/"; @override - String get MODEL_TYPE => "stocklocation"; + static const String MODEL_TYPE = "stocklocation"; @override List get rolesRequired => ["stock_location"]; diff --git a/lib/widget/part/part_detail.dart b/lib/widget/part/part_detail.dart index 0e2f2ca6..a11fe4c2 100644 --- a/lib/widget/part/part_detail.dart +++ b/lib/widget/part/part_detail.dart @@ -239,7 +239,7 @@ class _PartDisplayState extends RefreshableState { if (allowLabelPrinting) { - String model_type = api.supportsModernLabelPrinting ? InvenTreePart().MODEL_TYPE : "part"; + String model_type = api.supportsModernLabelPrinting ? InvenTreePart.MODEL_TYPE : "part"; String item_key = api.supportsModernLabelPrinting ? "items" : "part"; _labels = await getLabelTemplates( diff --git a/lib/widget/stock/location_display.dart b/lib/widget/stock/location_display.dart index 42e8907f..07a8464d 100644 --- a/lib/widget/stock/location_display.dart +++ b/lib/widget/stock/location_display.dart @@ -246,7 +246,7 @@ class _LocationDisplayState extends RefreshableState { if (widget.location != null) { - String model_type = api.supportsModernLabelPrinting ? InvenTreeStockLocation().MODEL_TYPE : "location"; + String model_type = api.supportsModernLabelPrinting ? InvenTreeStockLocation.MODEL_TYPE : "location"; String item_key = api.supportsModernLabelPrinting ? "items" : "location"; _labels = await getLabelTemplates( diff --git a/lib/widget/stock/stock_detail.dart b/lib/widget/stock/stock_detail.dart index 00817089..1b215988 100644 --- a/lib/widget/stock/stock_detail.dart +++ b/lib/widget/stock/stock_detail.dart @@ -298,7 +298,7 @@ class _StockItemDisplayState extends RefreshableState { // Request information on labels available for this stock item if (allowLabelPrinting) { - String model_type = api.supportsModernLabelPrinting ? InvenTreeStockItem().MODEL_TYPE : "stock"; + String model_type = api.supportsModernLabelPrinting ? InvenTreeStockItem.MODEL_TYPE : "stock"; String item_key = api.supportsModernLabelPrinting ? "items" : "item"; // Clear the existing labels list From f042e0668e0415390e47dcc8f8230db656309a09 Mon Sep 17 00:00:00 2001 From: Oliver Walters Date: Sat, 14 Dec 2024 15:02:45 +1100 Subject: [PATCH 21/21] @override fix --- lib/api_form.dart | 12 ++++++++++-- lib/inventree/company.dart | 3 --- lib/inventree/part.dart | 3 --- lib/inventree/project_code.dart | 1 - lib/inventree/purchase_order.dart | 1 - lib/inventree/sales_order.dart | 2 -- lib/inventree/stock.dart | 2 -- 7 files changed, 10 insertions(+), 14 deletions(-) diff --git a/lib/api_form.dart b/lib/api_form.dart index 96af188b..7b8cb426 100644 --- a/lib/api_form.dart +++ b/lib/api_form.dart @@ -11,15 +11,17 @@ import "package:inventree/api.dart"; import "package:inventree/app_colors.dart"; import "package:inventree/barcode/barcode.dart"; import "package:inventree/helpers.dart"; -import "package:inventree/inventree/sales_order.dart"; import "package:inventree/l10.dart"; import "package:inventree/inventree/company.dart"; import "package:inventree/inventree/part.dart"; import "package:inventree/inventree/project_code.dart"; -import "package:inventree/inventree/sentry.dart"; +import "package:inventree/inventree/purchase_order.dart"; +import "package:inventree/inventree/sales_order.dart"; import "package:inventree/inventree/stock.dart"; +import "package:inventree/inventree/sentry.dart"; + import "package:inventree/widget/dialogs.dart"; import "package:inventree/widget/fields.dart"; import "package:inventree/widget/progress.dart"; @@ -564,6 +566,12 @@ class APIFormField { switch (model) { case InvenTreePart.MODEL_TYPE: return InvenTreePart.fromJson(data).fullname; + case InvenTreeCompany.MODEL_TYPE: + return InvenTreeCompany.fromJson(data).name; + case InvenTreePurchaseOrder.MODEL_TYPE: + return InvenTreePurchaseOrder.fromJson(data).reference; + case InvenTreeSalesOrder.MODEL_TYPE: + return InvenTreeSalesOrder.fromJson(data).reference; case InvenTreePartCategory.MODEL_TYPE: return InvenTreePartCategory.fromJson(data).pathstring; case InvenTreeStockLocation.MODEL_TYPE: diff --git a/lib/inventree/company.dart b/lib/inventree/company.dart index 9be993e9..18e543e4 100644 --- a/lib/inventree/company.dart +++ b/lib/inventree/company.dart @@ -18,7 +18,6 @@ class InvenTreeCompany extends InvenTreeModel { @override String get URL => "company/"; - @override static const String MODEL_TYPE = "company"; @override @@ -131,7 +130,6 @@ class InvenTreeSupplierPart extends InvenTreeModel { @override String get URL => "company/part/"; - @override static const String MODEL_TYPE = "supplierpart"; @override @@ -227,7 +225,6 @@ class InvenTreeManufacturerPart extends InvenTreeModel { @override String URL = "company/part/manufacturer/"; - @override static const String MODEL_TYPE = "manufacturerpart"; @override diff --git a/lib/inventree/part.dart b/lib/inventree/part.dart index 375b2a24..c803c329 100644 --- a/lib/inventree/part.dart +++ b/lib/inventree/part.dart @@ -23,7 +23,6 @@ class InvenTreePartCategory extends InvenTreeModel { @override String get URL => "part/category/"; - @override static const String MODEL_TYPE = "partcategory"; @override @@ -82,7 +81,6 @@ class InvenTreePartTestTemplate extends InvenTreeModel { @override String get URL => "part/test-template/"; - @override static const String MODEL_TYPE = "parttesttemplate"; String get key => getString("key"); @@ -198,7 +196,6 @@ class InvenTreePart extends InvenTreeModel { @override String get URL => "part/"; - @override static const String MODEL_TYPE = "part"; @override diff --git a/lib/inventree/project_code.dart b/lib/inventree/project_code.dart index fbd2ad17..e5d45289 100644 --- a/lib/inventree/project_code.dart +++ b/lib/inventree/project_code.dart @@ -16,7 +16,6 @@ class InvenTreeProjectCode extends InvenTreeModel { @override String get URL => "project-code/"; - @override static const String MODEL_TYPE = "projectcode"; @override diff --git a/lib/inventree/purchase_order.dart b/lib/inventree/purchase_order.dart index cfe808c5..c258c8b6 100644 --- a/lib/inventree/purchase_order.dart +++ b/lib/inventree/purchase_order.dart @@ -26,7 +26,6 @@ class InvenTreePurchaseOrder extends InvenTreeOrder { @override String get URL => "order/po/"; - @override static const String MODEL_TYPE = "purchaseorder"; @override diff --git a/lib/inventree/sales_order.dart b/lib/inventree/sales_order.dart index b55db140..7372250b 100644 --- a/lib/inventree/sales_order.dart +++ b/lib/inventree/sales_order.dart @@ -24,7 +24,6 @@ class InvenTreeSalesOrder extends InvenTreeOrder { @override String get URL => "order/so/"; - @override static const String MODEL_TYPE = "salesorder"; @override @@ -250,7 +249,6 @@ class InvenTreeSalesOrderShipment extends InvenTreeModel { @override String get URL => "/order/so/shipment/"; - @override static const String MODEL_TYPE = "salesordershipment"; @override diff --git a/lib/inventree/stock.dart b/lib/inventree/stock.dart index be2bea3a..1ae5e7ff 100644 --- a/lib/inventree/stock.dart +++ b/lib/inventree/stock.dart @@ -168,7 +168,6 @@ class InvenTreeStockItem extends InvenTreeModel { @override String get URL => "stock/"; - @override static const String MODEL_TYPE = "stockitem"; @override @@ -681,7 +680,6 @@ class InvenTreeStockLocation extends InvenTreeModel { @override String get URL => "stock/location/"; - @override static const String MODEL_TYPE = "stocklocation"; @override