mirror of
				https://github.com/inventree/inventree-app.git
				synced 2025-10-31 13:25:40 +00:00 
			
		
		
		
	Modern attachments (#505)
* Minimum API version is now 100 * Remove old API features - Anything below API v100 no longer supported * Reefactor attachment widget to support modern attachment API * Filter and display attachments correctly * Refactor
This commit is contained in:
		
							
								
								
									
										39
									
								
								lib/api.dart
									
									
									
									
									
								
							
							
						
						
									
										39
									
								
								lib/api.dart
									
									
									
									
									
								
							| @@ -187,7 +187,8 @@ class InvenTreeAPI { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Minimum required API version for server |   // Minimum required API version for server | ||||||
|   static const _minApiVersion = 20; |   // 2023-03-04 | ||||||
|  |   static const _minApiVersion = 100; | ||||||
|  |  | ||||||
|   bool _strictHttps = false; |   bool _strictHttps = false; | ||||||
|  |  | ||||||
| @@ -282,30 +283,6 @@ class InvenTreeAPI { | |||||||
|   String get serverVersion => (serverInfo["version"] ?? "") as String; |   String get serverVersion => (serverInfo["version"] ?? "") as String; | ||||||
|   int get apiVersion => (serverInfo["apiVersion"] ?? 1) as int; |   int get apiVersion => (serverInfo["apiVersion"] ?? 1) as int; | ||||||
|  |  | ||||||
|   // Plugins enabled at API v34 and above |  | ||||||
|   bool get pluginsEnabled => apiVersion >= 34 && (serverInfo["plugins_enabled"] ?? false) as bool; |  | ||||||
|  |  | ||||||
|   // API endpoint for receiving purchase order line items was introduced in v12 |  | ||||||
|   bool get supportsPoReceive => apiVersion >= 12; |  | ||||||
|  |  | ||||||
|   // Notification support requires API v25 or newer |  | ||||||
|   bool get supportsNotifications => isConnected() && apiVersion >= 25; |  | ||||||
|  |  | ||||||
|   // Return True if the API supports 'settings' (requires API v46) |  | ||||||
|   bool get supportsSettings => isConnected() && apiVersion >= 46; |  | ||||||
|  |  | ||||||
|   // Part parameter support requires API v56 or newer |  | ||||||
|   bool get supportsPartParameters => isConnected() && apiVersion >= 56; |  | ||||||
|  |  | ||||||
|   // Supports 'modern' barcode API (v80 or newer) |  | ||||||
|   bool get supportModernBarcodes => isConnected() && apiVersion >= 80; |  | ||||||
|  |  | ||||||
|   // Structural categories requires API v83 or newer |  | ||||||
|   bool get supportsStructuralCategories => isConnected() && apiVersion >= 83; |  | ||||||
|  |  | ||||||
|   // Company attachments require API v95 or newer |  | ||||||
|   bool get supportCompanyAttachments => isConnected() && apiVersion >= 95; |  | ||||||
|  |  | ||||||
|   // Consolidated search request API v102 or newer |   // Consolidated search request API v102 or newer | ||||||
|   bool get supportsConsolidatedSearch => isConnected() && apiVersion >= 102; |   bool get supportsConsolidatedSearch => isConnected() && apiVersion >= 102; | ||||||
|  |  | ||||||
| @@ -346,7 +323,11 @@ class InvenTreeAPI { | |||||||
|   bool get supportsCompanyActiveStatus => isConnected() && apiVersion >= 189; |   bool get supportsCompanyActiveStatus => isConnected() && apiVersion >= 189; | ||||||
|  |  | ||||||
|   // Does the server support the "modern" (consolidated) label printing API? |   // Does the server support the "modern" (consolidated) label printing API? | ||||||
|   bool get supportsModernLabelPrinting => isConnected() && apiVersion >= 198; |   bool get supportsModernLabelPrinting => isConnected() && apiVersion >= 201; | ||||||
|  |  | ||||||
|  |   // Does the server support the "modern" (consolidated) attachment API? | ||||||
|  |   // Ref: https://github.com/inventree/InvenTree/pull/7420 | ||||||
|  |   bool get supportsModernAttachments => isConnected() && apiVersion >= 207; | ||||||
|  |  | ||||||
|   // Cached list of plugins (refreshed when we connect to the server) |   // Cached list of plugins (refreshed when we connect to the server) | ||||||
|   List<InvenTreePlugin> _plugins = []; |   List<InvenTreePlugin> _plugins = []; | ||||||
| @@ -1517,7 +1498,6 @@ class InvenTreeAPI { | |||||||
|   Map<String, InvenTreeUserSetting> _userSettings = {}; |   Map<String, InvenTreeUserSetting> _userSettings = {}; | ||||||
|  |  | ||||||
|   Future<String> getGlobalSetting(String key) async { |   Future<String> getGlobalSetting(String key) async { | ||||||
|     if (!supportsSettings) return ""; |  | ||||||
|  |  | ||||||
|     InvenTreeGlobalSetting? setting = _globalSettings[key]; |     InvenTreeGlobalSetting? setting = _globalSettings[key]; | ||||||
|  |  | ||||||
| @@ -1543,7 +1523,6 @@ class InvenTreeAPI { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<String> getUserSetting(String key) async { |   Future<String> getUserSetting(String key) async { | ||||||
|     if (!supportsSettings) return ""; |  | ||||||
|  |  | ||||||
|     InvenTreeUserSetting? setting = _userSettings[key]; |     InvenTreeUserSetting? setting = _userSettings[key]; | ||||||
|  |  | ||||||
| @@ -1687,10 +1666,6 @@ class InvenTreeAPI { | |||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (!supportsNotifications) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     InvenTreeNotification().count(filters: {"read": "false"}).then((int n) { |     InvenTreeNotification().count(filters: {"read": "false"}).then((int n) { | ||||||
|       notification_counter = n; |       notification_counter = n; | ||||||
|     }); |     }); | ||||||
|   | |||||||
| @@ -341,12 +341,7 @@ class UniqueBarcodeHandler extends BarcodeHandler { | |||||||
|     } else { |     } else { | ||||||
|       String barcode; |       String barcode; | ||||||
|  |  | ||||||
|       if (InvenTreeAPI().supportModernBarcodes) { |  | ||||||
|       barcode = (data["barcode_data"] ?? "") as String; |       barcode = (data["barcode_data"] ?? "") as String; | ||||||
|       } else { |  | ||||||
|         // Legacy barcode API |  | ||||||
|         barcode = (data["hash"] ?? data["barcode_hash"] ?? "") as String; |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       if (barcode.isEmpty) { |       if (barcode.isEmpty) { | ||||||
|         barcodeFailureTone(); |         barcodeFailureTone(); | ||||||
|   | |||||||
| @@ -106,7 +106,10 @@ class InvenTreeCompanyAttachment extends InvenTreeAttachment { | |||||||
|   String get REFERENCE_FIELD => "company"; |   String get REFERENCE_FIELD => "company"; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String get URL => "company/attachment/"; |   String get REF_MODEL_TYPE => "company"; | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String get URL => InvenTreeAPI().supportsModernAttachments ? "attachment/" : "company/attachment/"; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   InvenTreeModel createFromJson(Map<String, dynamic> json) => InvenTreeCompanyAttachment.fromJson(json); |   InvenTreeModel createFromJson(Map<String, dynamic> json) => InvenTreeCompanyAttachment.fromJson(json); | ||||||
|   | |||||||
| @@ -938,9 +938,17 @@ class InvenTreeAttachment extends InvenTreeModel { | |||||||
|  |  | ||||||
|   InvenTreeAttachment.fromJson(Map<String, dynamic> json) : super.fromJson(json); |   InvenTreeAttachment.fromJson(Map<String, dynamic> json) : super.fromJson(json); | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String get URL => "attachment/"; | ||||||
|  |  | ||||||
|   // Override this reference field for any subclasses |   // Override this reference field for any subclasses | ||||||
|  |   // Note: This is used for the *legacy* attachment API | ||||||
|   String get REFERENCE_FIELD => ""; |   String get REFERENCE_FIELD => ""; | ||||||
|  |  | ||||||
|  |   // Override this reference field for any subclasses | ||||||
|  |   // Note: This is used for the *modern* attachment API | ||||||
|  |   String get REF_MODEL_TYPE => ""; | ||||||
|  |  | ||||||
|   String get attachment => getString("attachment"); |   String get attachment => getString("attachment"); | ||||||
|    |    | ||||||
|   // Return the filename of the attachment |   // Return the filename of the attachment | ||||||
| @@ -989,15 +997,39 @@ class InvenTreeAttachment extends InvenTreeModel { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   Future<bool> uploadAttachment(File attachment, int parentId, {String comment = "", Map<String, String> fields = const {}}) async { |   // Return a count of how many attachments exist against the specified model ID | ||||||
|  |   Future<int> countAttachments(int modelId) { | ||||||
|  |  | ||||||
|  |     Map<String, String> filters = {}; | ||||||
|  |  | ||||||
|  |     if (InvenTreeAPI().supportsModernAttachments) { | ||||||
|  |       filters["model_type"] = REF_MODEL_TYPE; | ||||||
|  |       filters["model_id"] = modelId.toString(); | ||||||
|  |     } else { | ||||||
|  |       filters[REFERENCE_FIELD] = modelId.toString(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return count(filters: filters); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   Future<bool> uploadAttachment(File attachment, String modelType, int modelId, {String comment = "", Map<String, String> fields = const {}}) async { | ||||||
|  |  | ||||||
|     // Ensure that the correct reference field is set |     // Ensure that the correct reference field is set | ||||||
|     Map<String, String> data = Map<String, String>.from(fields); |     Map<String, String> data = Map<String, String>.from(fields); | ||||||
|  |  | ||||||
|     data[REFERENCE_FIELD] = parentId.toString(); |     String url = URL; | ||||||
|  |  | ||||||
|  |     if (InvenTreeAPI().supportsModernAttachments) { | ||||||
|  |       // All attachments are stored in a consolidated table | ||||||
|  |       url = "attachment/"; | ||||||
|  |       data["model_id"] = modelId.toString(); | ||||||
|  |       data["model_type"] = modelType; | ||||||
|  |     } else { | ||||||
|  |       data[REFERENCE_FIELD] = modelId.toString(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     final APIResponse response = await InvenTreeAPI().uploadFile( |     final APIResponse response = await InvenTreeAPI().uploadFile( | ||||||
|         URL, |         url, | ||||||
|         attachment, |         attachment, | ||||||
|         method: "POST", |         method: "POST", | ||||||
|         name: "attachment", |         name: "attachment", | ||||||
|   | |||||||
| @@ -36,10 +36,6 @@ class InvenTreePartCategory extends InvenTreeModel { | |||||||
|       "structural": {}, |       "structural": {}, | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     if (!api.supportsStructuralCategories) { |  | ||||||
|       fields.remove("structural"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return fields; |     return fields; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -475,7 +471,10 @@ class InvenTreePartAttachment extends InvenTreeAttachment { | |||||||
|   String get REFERENCE_FIELD => "part"; |   String get REFERENCE_FIELD => "part"; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String get URL => "part/attachment/"; |   String get REF_MODEL_TYPE => "part"; | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String get URL => InvenTreeAPI().supportsModernAttachments ? "attachment/" : "part/attachment/"; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   InvenTreeModel createFromJson(Map<String, dynamic> json) => InvenTreePartAttachment.fromJson(json); |   InvenTreeModel createFromJson(Map<String, dynamic> json) => InvenTreePartAttachment.fromJson(json); | ||||||
|   | |||||||
| @@ -237,7 +237,10 @@ class InvenTreePurchaseOrderAttachment extends InvenTreeAttachment { | |||||||
|   String get REFERENCE_FIELD => "order"; |   String get REFERENCE_FIELD => "order"; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String get URL => "order/po/attachment/"; |   String get REF_MODEL_TYPE => "purchaseorder"; | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String get URL => InvenTreeAPI().supportsModernAttachments ? "attachment/" : "order/po/attachment/"; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   InvenTreeModel createFromJson(Map<String, dynamic> json) => InvenTreePurchaseOrderAttachment.fromJson(json); |   InvenTreeModel createFromJson(Map<String, dynamic> json) => InvenTreePurchaseOrderAttachment.fromJson(json); | ||||||
|   | |||||||
| @@ -280,6 +280,9 @@ class InvenTreeSalesOrderAttachment extends InvenTreeAttachment { | |||||||
|   String get REFERENCE_FIELD => "order"; |   String get REFERENCE_FIELD => "order"; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String get URL => "order/so/attachment/"; |   String get REF_MODEL_TYPE => "salesorder"; | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String get URL => InvenTreeAPI().supportsModernAttachments ? "attachment/" : "order/so/attachment/"; | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -635,11 +635,13 @@ class InvenTreeStockItemAttachment extends InvenTreeAttachment { | |||||||
|   String get REFERENCE_FIELD => "stock_item"; |   String get REFERENCE_FIELD => "stock_item"; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   String get URL => "stock/attachment/"; |   String get REF_MODEL_TYPE => "stockitem"; | ||||||
|  |  | ||||||
|  |   @override | ||||||
|  |   String get URL => InvenTreeAPI().supportsModernAttachments ? "attachment/" : "stock/attachment/"; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
|   InvenTreeModel createFromJson(Map<String, dynamic> json) => InvenTreeStockItemAttachment.fromJson(json); |   InvenTreeModel createFromJson(Map<String, dynamic> json) => InvenTreeStockItemAttachment.fromJson(json); | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -669,10 +671,6 @@ class InvenTreeStockLocation extends InvenTreeModel { | |||||||
|       "structural": {}, |       "structural": {}, | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     if (!api.supportsStructuralCategories) { |  | ||||||
|       fields.remove("structural"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return fields; |     return fields; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -121,7 +121,6 @@ class InvenTreeAboutWidget extends StatelessWidget { | |||||||
|       ); |       ); | ||||||
|  |  | ||||||
|       // Display extra tile if the server supports plugins |       // Display extra tile if the server supports plugins | ||||||
|       if (InvenTreeAPI().pluginsEnabled) { |  | ||||||
|       tiles.add( |       tiles.add( | ||||||
|         ListTile( |         ListTile( | ||||||
|           title: Text(L10().pluginSupport), |           title: Text(L10().pluginSupport), | ||||||
| @@ -129,7 +128,6 @@ class InvenTreeAboutWidget extends StatelessWidget { | |||||||
|           leading: FaIcon(FontAwesomeIcons.plug), |           leading: FaIcon(FontAwesomeIcons.plug), | ||||||
|         ) |         ) | ||||||
|       ); |       ); | ||||||
|       } |  | ||||||
|  |  | ||||||
|     } else { |     } else { | ||||||
|       tiles.add( |       tiles.add( | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ import "package:font_awesome_flutter/font_awesome_flutter.dart"; | |||||||
| import "package:one_context/one_context.dart"; | import "package:one_context/one_context.dart"; | ||||||
| import "package:url_launcher/url_launcher.dart"; | import "package:url_launcher/url_launcher.dart"; | ||||||
|  |  | ||||||
|  | import "package:inventree/api.dart"; | ||||||
| import "package:inventree/l10.dart"; | import "package:inventree/l10.dart"; | ||||||
| import "package:inventree/app_colors.dart"; | import "package:inventree/app_colors.dart"; | ||||||
|  |  | ||||||
| @@ -25,10 +26,10 @@ import "package:inventree/widget/refreshable_state.dart"; | |||||||
|  */ |  */ | ||||||
| class AttachmentWidget extends StatefulWidget { | class AttachmentWidget extends StatefulWidget { | ||||||
|  |  | ||||||
|   const AttachmentWidget(this.attachment, this.referenceId, this.hasUploadPermission) : super(); |   const AttachmentWidget(this.attachmentClass, this.modelId, this.hasUploadPermission) : super(); | ||||||
|  |  | ||||||
|   final InvenTreeAttachment attachment; |   final InvenTreeAttachment attachmentClass; | ||||||
|   final int referenceId; |   final int modelId; | ||||||
|   final bool hasUploadPermission; |   final bool hasUploadPermission; | ||||||
|  |  | ||||||
|   @override |   @override | ||||||
| @@ -74,7 +75,9 @@ class _AttachmentWidgetState extends RefreshableState<AttachmentWidget> { | |||||||
|     if (file == null) return; |     if (file == null) return; | ||||||
|  |  | ||||||
|     showLoadingOverlay(context); |     showLoadingOverlay(context); | ||||||
|     final bool result = await widget.attachment.uploadAttachment(file, widget.referenceId); |  | ||||||
|  |     final bool result = await widget.attachmentClass.uploadAttachment(file, widget.attachmentClass.MODEL_TYPE, widget.modelId); | ||||||
|  |  | ||||||
|     hideLoadingOverlay(); |     hideLoadingOverlay(); | ||||||
|  |  | ||||||
|     if (result) { |     if (result) { | ||||||
| @@ -131,14 +134,24 @@ class _AttachmentWidgetState extends RefreshableState<AttachmentWidget> { | |||||||
|   @override |   @override | ||||||
|   Future<void> request(BuildContext context) async { |   Future<void> request(BuildContext context) async { | ||||||
|  |  | ||||||
|     await widget.attachment.list( |     Map<String, String> filters = {}; | ||||||
|       filters: { |  | ||||||
|         widget.attachment.REFERENCE_FIELD: widget.referenceId.toString() |     if (InvenTreeAPI().supportsModernAttachments) { | ||||||
|  |       filters["model_type"] = widget.attachmentClass.MODEL_TYPE; | ||||||
|  |       filters["model_id"] = widget.modelId.toString(); | ||||||
|  |     } else { | ||||||
|  |       filters[widget.attachmentClass.REFERENCE_FIELD] = widget.modelId.toString(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     await widget.attachmentClass.list( | ||||||
|  |       filters: filters | ||||||
|     ).then((var results) { |     ).then((var results) { | ||||||
|       attachments.clear(); |       attachments.clear(); | ||||||
|  |  | ||||||
|  |       print("Found ${results.length} results:"); | ||||||
|  |  | ||||||
|       for (var result in results) { |       for (var result in results) { | ||||||
|  |         print(result.toString()); | ||||||
|         if (result is InvenTreeAttachment) { |         if (result is InvenTreeAttachment) { | ||||||
|           attachments.add(result); |           attachments.add(result); | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -188,12 +188,8 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> { | |||||||
|       } |       } | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     if (api.supportCompanyAttachments) { |     InvenTreeCompanyAttachment().countAttachments(widget.company.pk) | ||||||
|       InvenTreeCompanyAttachment().count( |     .then((value) { | ||||||
|         filters: { |  | ||||||
|           "company": widget.company.pk.toString() |  | ||||||
|         } |  | ||||||
|       ).then((value) { |  | ||||||
|       if (mounted) { |       if (mounted) { | ||||||
|         setState(() { |         setState(() { | ||||||
|           attachmentCount = value; |           attachmentCount = value; | ||||||
| @@ -201,7 +197,6 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> { | |||||||
|       } |       } | ||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
|   } |  | ||||||
|  |  | ||||||
|   Future <void> editCompany(BuildContext context) async { |   Future <void> editCompany(BuildContext context) async { | ||||||
|  |  | ||||||
| @@ -397,7 +392,7 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> { | |||||||
|       )); |       )); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (api.supportCompanyAttachments) { |  | ||||||
|     tiles.add(ListTile( |     tiles.add(ListTile( | ||||||
|       title: Text(L10().attachments), |       title: Text(L10().attachments), | ||||||
|       leading: FaIcon(FontAwesomeIcons.fileLines, color: COLOR_ACTION), |       leading: FaIcon(FontAwesomeIcons.fileLines, color: COLOR_ACTION), | ||||||
| @@ -415,7 +410,6 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> { | |||||||
|         ); |         ); | ||||||
|       } |       } | ||||||
|     )); |     )); | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return tiles; |     return tiles; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -174,7 +174,6 @@ class InvenTreeDrawer extends StatelessWidget { | |||||||
|       tiles.add(Divider()); |       tiles.add(Divider()); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     if (InvenTreeAPI().supportsNotifications) { |  | ||||||
|     int notification_count = InvenTreeAPI().notification_counter; |     int notification_count = InvenTreeAPI().notification_counter; | ||||||
|  |  | ||||||
|     tiles.add( |     tiles.add( | ||||||
| @@ -185,7 +184,6 @@ class InvenTreeDrawer extends StatelessWidget { | |||||||
|         onTap: _notifications, |         onTap: _notifications, | ||||||
|       ) |       ) | ||||||
|     ); |     ); | ||||||
|     } |  | ||||||
|  |  | ||||||
|     tiles.add( |     tiles.add( | ||||||
|       ListTile( |       ListTile( | ||||||
|   | |||||||
| @@ -228,9 +228,7 @@ class _PurchaseOrderDetailState extends RefreshableState<PurchaseOrderDetailWidg | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     InvenTreePurchaseOrderAttachment().count(filters: { |     InvenTreePurchaseOrderAttachment().countAttachments(widget.order.pk).then((int value) { | ||||||
|       "order": widget.order.pk.toString() |  | ||||||
|     }).then((int value) { |  | ||||||
|       if (mounted) { |       if (mounted) { | ||||||
|         setState(() { |         setState(() { | ||||||
|           attachmentCount = value; |           attachmentCount = value; | ||||||
|   | |||||||
| @@ -175,9 +175,7 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> { | |||||||
|  |  | ||||||
|     supportsProjectCodes = api.supportsProjectCodes && await api.getGlobalBooleanSetting("PROJECT_CODES_ENABLED"); |     supportsProjectCodes = api.supportsProjectCodes && await api.getGlobalBooleanSetting("PROJECT_CODES_ENABLED"); | ||||||
|  |  | ||||||
|     InvenTreeSalesOrderAttachment().count(filters: { |     InvenTreeSalesOrderAttachment().countAttachments(widget.order.pk).then((int value) { | ||||||
|       "order": widget.order.pk.toString() |  | ||||||
|     }).then((int value) { |  | ||||||
|       if (mounted) { |       if (mounted) { | ||||||
|         setState(() { |         setState(() { | ||||||
|           attachmentCount = value; |           attachmentCount = value; | ||||||
|   | |||||||
| @@ -90,7 +90,6 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> { | |||||||
|     List<SpeedDialChild> actions = []; |     List<SpeedDialChild> actions = []; | ||||||
|  |  | ||||||
|     if (InvenTreePart().canEdit) { |     if (InvenTreePart().canEdit) { | ||||||
|       if (api.supportModernBarcodes) { |  | ||||||
|       actions.add( |       actions.add( | ||||||
|           customBarcodeAction( |           customBarcodeAction( | ||||||
|               context, this, |               context, this, | ||||||
| @@ -99,7 +98,6 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> { | |||||||
|           ) |           ) | ||||||
|       ); |       ); | ||||||
|     } |     } | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return actions; |     return actions; | ||||||
|   } |   } | ||||||
| @@ -184,18 +182,10 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> { | |||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     // Request the number of parameters for this part |     // Request the number of parameters for this part | ||||||
|     if (api.supportsPartParameters) { |  | ||||||
|     showParameters = await InvenTreeSettingsManager().getValue(INV_PART_SHOW_PARAMETERS, true) as bool; |     showParameters = await InvenTreeSettingsManager().getValue(INV_PART_SHOW_PARAMETERS, true) as bool; | ||||||
|     } else { |  | ||||||
|       showParameters = false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // Request the number of attachments |     // Request the number of attachments | ||||||
|     InvenTreePartAttachment().count( |     InvenTreePartAttachment().countAttachments(part.pk).then((int value) { | ||||||
|       filters: { |  | ||||||
|         "part": part.pk.toString(), |  | ||||||
|       } |  | ||||||
|     ).then((int value) { |  | ||||||
|       if (mounted) { |       if (mounted) { | ||||||
|         setState(() { |         setState(() { | ||||||
|           attachmentCount = value; |           attachmentCount = value; | ||||||
|   | |||||||
| @@ -140,7 +140,6 @@ class _LocationDisplayState extends RefreshableState<LocationDisplayWidget> { | |||||||
|       } |       } | ||||||
|  |  | ||||||
|       // Assign or un-assign barcodes |       // Assign or un-assign barcodes | ||||||
|       if (api.supportModernBarcodes) { |  | ||||||
|       actions.add( |       actions.add( | ||||||
|           customBarcodeAction( |           customBarcodeAction( | ||||||
|               context, this, |               context, this, | ||||||
| @@ -149,7 +148,6 @@ class _LocationDisplayState extends RefreshableState<LocationDisplayWidget> { | |||||||
|           ) |           ) | ||||||
|       ); |       ); | ||||||
|     } |     } | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return actions; |     return actions; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -183,7 +183,6 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> { | |||||||
|           ) |           ) | ||||||
|       ); |       ); | ||||||
|  |  | ||||||
|       if (api.supportModernBarcodes) { |  | ||||||
|       actions.add( |       actions.add( | ||||||
|           customBarcodeAction( |           customBarcodeAction( | ||||||
|               context, this, |               context, this, | ||||||
| @@ -192,7 +191,6 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> { | |||||||
|           ) |           ) | ||||||
|       ); |       ); | ||||||
|     } |     } | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return actions; |     return actions; | ||||||
|   } |   } | ||||||
| @@ -246,12 +244,7 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Request the number of attachments |     // Request the number of attachments | ||||||
|     InvenTreeStockItemAttachment().count( |     InvenTreeStockItemAttachment().countAttachments(widget.item.pk).then((int value) { | ||||||
|       filters: { |  | ||||||
|         "stock_item": widget.item.pk.toString() |  | ||||||
|       } |  | ||||||
|     ).then((int value) { |  | ||||||
|  |  | ||||||
|       if (mounted) { |       if (mounted) { | ||||||
|         setState(() { |         setState(() { | ||||||
|           attachmentCount = value; |           attachmentCount = value; | ||||||
|   | |||||||
| @@ -110,10 +110,6 @@ void main() { | |||||||
|  |  | ||||||
|       // Check supported functions |       // Check supported functions | ||||||
|       assert(api.apiVersion >= 50); |       assert(api.apiVersion >= 50); | ||||||
|       assert(api.supportsSettings); |  | ||||||
|       assert(api.supportsNotifications); |  | ||||||
|       assert(api.supportsPoReceive); |  | ||||||
|  |  | ||||||
|       assert(api.serverInstance.isNotEmpty); |       assert(api.serverInstance.isNotEmpty); | ||||||
|       assert(api.serverVersion.isNotEmpty); |       assert(api.serverVersion.isNotEmpty); | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user