2
0
mirror of https://github.com/inventree/inventree-app.git synced 2025-07-06 13:50:41 +00:00

Link icons (#677)

* Add LinkIcon component

* Visual UI updates

- Refactor some components
- Clearer text display
- Add obvious chevron icon when a "tile" will take the user somewhere else

* dart format

* Adjust release notes

* Add visual separator

* Cleanup unused imports
This commit is contained in:
Oliver
2025-07-04 21:16:04 +10:00
committed by GitHub
parent c30f1a19d1
commit 2adf8e3430
28 changed files with 261 additions and 195 deletions

View File

@ -1,3 +1,8 @@
### TBD - ???
---
- Improved UX across the entire app
### 0.19.1 - July 2025 ### 0.19.1 - July 2025
--- ---
- Fixes bug related to barcode scanning with certain devices - Fixes bug related to barcode scanning with certain devices

View File

@ -5,6 +5,7 @@ import "package:inventree/api.dart";
import "package:inventree/inventree/model.dart"; import "package:inventree/inventree/model.dart";
import "package:inventree/inventree/purchase_order.dart"; import "package:inventree/inventree/purchase_order.dart";
import "package:inventree/widget/company/company_detail.dart"; import "package:inventree/widget/company/company_detail.dart";
import "package:inventree/widget/company/supplier_part_detail.dart";
/* /*
* The InvenTreeCompany class represents the Company model in the InvenTree database. * The InvenTreeCompany class represents the Company model in the InvenTree database.
@ -152,6 +153,14 @@ class InvenTreeSupplierPart extends InvenTreeModel {
@override @override
List<String> get rolesRequired => ["part", "purchase_order"]; List<String> get rolesRequired => ["part", "purchase_order"];
@override
Future<Object?> goToDetailPage(BuildContext context) async {
return Navigator.push(
context,
MaterialPageRoute(builder: (context) => SupplierPartDetailWidget(this)),
);
}
@override @override
Map<String, Map<String, dynamic>> formFields() { Map<String, Map<String, dynamic>> formFields() {
Map<String, Map<String, dynamic>> fields = { Map<String, Map<String, dynamic>> fields = {

View File

@ -391,6 +391,9 @@
"editLineItem": "Edit Line Item", "editLineItem": "Edit Line Item",
"@editLineItem": {}, "@editLineItem": {},
"email": "Email",
"@email": {},
"enterPassword": "Enter password", "enterPassword": "Enter password",
"@enterPassword": {}, "@enterPassword": {},
@ -924,6 +927,9 @@
"permissionRequired": "Permission Required", "permissionRequired": "Permission Required",
"@permissionRequired": {}, "@permissionRequired": {},
"phone": "Phone",
"@phone": {},
"printLabel": "Print Label", "printLabel": "Print Label",
"@printLabel": {}, "@printLabel": {},

View File

@ -10,6 +10,7 @@ import "package:inventree/inventree/company.dart";
import "package:inventree/inventree/purchase_order.dart"; import "package:inventree/inventree/purchase_order.dart";
import "package:inventree/inventree/sales_order.dart"; import "package:inventree/inventree/sales_order.dart";
import "package:inventree/widget/attachment_widget.dart"; import "package:inventree/widget/attachment_widget.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/order/purchase_order_list.dart"; import "package:inventree/widget/order/purchase_order_list.dart";
import "package:inventree/widget/order/sales_order_list.dart"; import "package:inventree/widget/order/sales_order_list.dart";
import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/refreshable_state.dart";
@ -240,8 +241,10 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
if (widget.company.website.isNotEmpty) { if (widget.company.website.isNotEmpty) {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text("${widget.company.website}"), title: Text(L10().website),
subtitle: Text("${widget.company.website}"),
leading: Icon(TablerIcons.globe, color: COLOR_ACTION), leading: Icon(TablerIcons.globe, color: COLOR_ACTION),
trailing: LinkIcon(external: true),
onTap: () async { onTap: () async {
openLink(widget.company.website); openLink(widget.company.website);
}, },
@ -254,8 +257,10 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
if (widget.company.email.isNotEmpty) { if (widget.company.email.isNotEmpty) {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text("${widget.company.email}"), title: Text(L10().email),
subtitle: Text("${widget.company.email}"),
leading: Icon(TablerIcons.at, color: COLOR_ACTION), leading: Icon(TablerIcons.at, color: COLOR_ACTION),
trailing: LinkIcon(external: true),
onTap: () async { onTap: () async {
openLink("mailto:${widget.company.email}"); openLink("mailto:${widget.company.email}");
}, },
@ -268,8 +273,10 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
if (widget.company.phone.isNotEmpty) { if (widget.company.phone.isNotEmpty) {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text("${widget.company.phone}"), title: Text(L10().phone),
subtitle: Text("${widget.company.phone}"),
leading: Icon(TablerIcons.phone, color: COLOR_ACTION), leading: Icon(TablerIcons.phone, color: COLOR_ACTION),
trailing: LinkIcon(external: true),
onTap: () { onTap: () {
openLink("tel:${widget.company.phone}"); openLink("tel:${widget.company.phone}");
}, },
@ -283,8 +290,10 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
if (widget.company.link.isNotEmpty) { if (widget.company.link.isNotEmpty) {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text("${widget.company.link}"), title: Text(L10().link),
subtitle: Text("${widget.company.link}"),
leading: Icon(TablerIcons.link, color: COLOR_ACTION), leading: Icon(TablerIcons.link, color: COLOR_ACTION),
trailing: LinkIcon(external: true),
onTap: () { onTap: () {
widget.company.openLink(); widget.company.openLink();
}, },
@ -304,7 +313,7 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
ListTile( ListTile(
title: Text(L10().supplierParts), title: Text(L10().supplierParts),
leading: Icon(TablerIcons.building, color: COLOR_ACTION), leading: Icon(TablerIcons.building, color: COLOR_ACTION),
trailing: Text(supplierPartCount.toString()), trailing: LinkIcon(text: supplierPartCount.toString()),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -323,7 +332,7 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
ListTile( ListTile(
title: Text(L10().purchaseOrders), title: Text(L10().purchaseOrders),
leading: Icon(TablerIcons.shopping_cart, color: COLOR_ACTION), leading: Icon(TablerIcons.shopping_cart, color: COLOR_ACTION),
trailing: Text("${outstandingPurchaseOrders}"), trailing: LinkIcon(text: "${outstandingPurchaseOrders}"),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -343,7 +352,7 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
ListTile( ListTile(
title: Text(L10().suppliedParts), title: Text(L10().suppliedParts),
leading: Icon(TablerIcons.box), leading: Icon(TablerIcons.box),
trailing: Text("${company.partSuppliedCount}"), trailing: LargeText("${company.partSuppliedCount}"),
) )
); );
*/ */
@ -358,7 +367,7 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
ListTile( ListTile(
title: Text(L10().salesOrders), title: Text(L10().salesOrders),
leading: Icon(TablerIcons.truck, color: COLOR_ACTION), leading: Icon(TablerIcons.truck, color: COLOR_ACTION),
trailing: Text("${outstandingSalesOrders}"), trailing: LinkIcon(text: "${outstandingSalesOrders}"),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -377,6 +386,7 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().notes), title: Text(L10().notes),
subtitle: Text(widget.company.notes),
leading: Icon(TablerIcons.note), leading: Icon(TablerIcons.note),
onTap: null, onTap: null,
), ),
@ -387,7 +397,9 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
ListTile( ListTile(
title: Text(L10().attachments), title: Text(L10().attachments),
leading: Icon(TablerIcons.file, color: COLOR_ACTION), leading: Icon(TablerIcons.file, color: COLOR_ACTION),
trailing: attachmentCount > 0 ? Text(attachmentCount.toString()) : null, trailing: LinkIcon(
text: attachmentCount > 0 ? attachmentCount.toString() : null,
),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,

View File

@ -1,9 +1,9 @@
import "package:flutter/material.dart"; import "package:flutter/material.dart";
import "package:flutter_speed_dial/flutter_speed_dial.dart"; import "package:flutter_speed_dial/flutter_speed_dial.dart";
import "package:flutter_tabler_icons/flutter_tabler_icons.dart"; import "package:flutter_tabler_icons/flutter_tabler_icons.dart";
import "package:inventree/widget/link_icon.dart";
import "package:url_launcher/url_launcher.dart"; import "package:url_launcher/url_launcher.dart";
import "package:inventree/api.dart";
import "package:inventree/app_colors.dart"; import "package:inventree/app_colors.dart";
import "package:inventree/l10.dart"; import "package:inventree/l10.dart";
@ -117,7 +117,7 @@ class _SupplierPartDisplayState
title: Text(L10().internalPart), title: Text(L10().internalPart),
subtitle: Text(widget.supplierPart.partName), subtitle: Text(widget.supplierPart.partName),
leading: Icon(TablerIcons.box, color: COLOR_ACTION), leading: Icon(TablerIcons.box, color: COLOR_ACTION),
trailing: InvenTreeAPI().getThumbnail(widget.supplierPart.partImage), trailing: LinkIcon(),
onTap: () async { onTap: () async {
showLoadingOverlay(); showLoadingOverlay();
final part = await InvenTreePart().get(widget.supplierPart.partId); final part = await InvenTreePart().get(widget.supplierPart.partId);
@ -149,9 +149,7 @@ class _SupplierPartDisplayState
title: Text(L10().supplier), title: Text(L10().supplier),
subtitle: Text(widget.supplierPart.supplierName), subtitle: Text(widget.supplierPart.supplierName),
leading: Icon(TablerIcons.building, color: COLOR_ACTION), leading: Icon(TablerIcons.building, color: COLOR_ACTION),
trailing: InvenTreeAPI().getThumbnail( trailing: LinkIcon(),
widget.supplierPart.supplierImage,
),
onTap: () async { onTap: () async {
showLoadingOverlay(); showLoadingOverlay();
var supplier = await InvenTreeCompany().get( var supplier = await InvenTreeCompany().get(
@ -182,9 +180,7 @@ class _SupplierPartDisplayState
title: Text(L10().manufacturer), title: Text(L10().manufacturer),
subtitle: Text(widget.supplierPart.manufacturerName), subtitle: Text(widget.supplierPart.manufacturerName),
leading: Icon(TablerIcons.building_factory_2, color: COLOR_ACTION), leading: Icon(TablerIcons.building_factory_2, color: COLOR_ACTION),
trailing: InvenTreeAPI().getThumbnail( trailing: LinkIcon(),
widget.supplierPart.manufacturerImage,
),
onTap: () async { onTap: () async {
showLoadingOverlay(); showLoadingOverlay();
var supplier = await InvenTreeCompany().get( var supplier = await InvenTreeCompany().get(
@ -204,6 +200,7 @@ class _SupplierPartDisplayState
title: Text(L10().manufacturerPartNumber), title: Text(L10().manufacturerPartNumber),
subtitle: Text(widget.supplierPart.MPN), subtitle: Text(widget.supplierPart.MPN),
leading: Icon(TablerIcons.hash, color: COLOR_ACTION), leading: Icon(TablerIcons.hash, color: COLOR_ACTION),
trailing: LinkIcon(),
onTap: () async { onTap: () async {
showLoadingOverlay(); showLoadingOverlay();
var manufacturerPart = await InvenTreeManufacturerPart().get( var manufacturerPart = await InvenTreeManufacturerPart().get(
@ -236,7 +233,7 @@ class _SupplierPartDisplayState
: null, : null,
leading: Icon(TablerIcons.package), leading: Icon(TablerIcons.package),
trailing: widget.supplierPart.pack_quantity.isNotEmpty trailing: widget.supplierPart.pack_quantity.isNotEmpty
? Text(widget.supplierPart.pack_quantity) ? LargeText(widget.supplierPart.pack_quantity)
: null, : null,
), ),
); );
@ -245,8 +242,10 @@ class _SupplierPartDisplayState
if (widget.supplierPart.link.isNotEmpty) { if (widget.supplierPart.link.isNotEmpty) {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(widget.supplierPart.link), title: Text(L10().link),
subtitle: Text(widget.supplierPart.link),
leading: Icon(TablerIcons.link, color: COLOR_ACTION), leading: Icon(TablerIcons.link, color: COLOR_ACTION),
trailing: LinkIcon(external: true),
onTap: () async { onTap: () async {
var uri = Uri.tryParse(widget.supplierPart.link); var uri = Uri.tryParse(widget.supplierPart.link);
if (uri != null && await canLaunchUrl(uri)) { if (uri != null && await canLaunchUrl(uri)) {
@ -260,7 +259,8 @@ class _SupplierPartDisplayState
if (widget.supplierPart.note.isNotEmpty) { if (widget.supplierPart.note.isNotEmpty) {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(widget.supplierPart.note), title: Text(L10().notes),
subtitle: Text(widget.supplierPart.note),
leading: Icon(TablerIcons.pencil), leading: Icon(TablerIcons.pencil),
), ),
); );

View File

@ -8,7 +8,6 @@ import "package:inventree/inventree/model.dart";
import "package:inventree/widget/paginator.dart"; import "package:inventree/widget/paginator.dart";
import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/refreshable_state.dart";
import "package:inventree/widget/company/supplier_part_detail.dart";
/* /*
* Widget for displaying a list of Supplier Part instances * Widget for displaying a list of Supplier Part instances
@ -93,12 +92,7 @@ class _PaginatedSupplierPartListState
leading: InvenTreeAPI().getThumbnail(supplierPart.supplierImage), leading: InvenTreeAPI().getThumbnail(supplierPart.supplierImage),
trailing: InvenTreeAPI().getThumbnail(supplierPart.partImage), trailing: InvenTreeAPI().getThumbnail(supplierPart.partImage),
onTap: () { onTap: () {
Navigator.push( supplierPart.goToDetailPage(context);
context,
MaterialPageRoute(
builder: (context) => SupplierPartDetailWidget(supplierPart),
),
);
}, },
); );
} }

44
lib/widget/link_icon.dart Normal file
View File

@ -0,0 +1,44 @@
/*
* An icon component to indicate that pressing on an item will change the page.
*/
import "package:cached_network_image/cached_network_image.dart";
import "package:flutter/cupertino.dart";
import "package:flutter_tabler_icons/flutter_tabler_icons.dart";
import "package:inventree/app_colors.dart";
Widget LargeText(
String text, {
double size = 14.0,
bool bold = false,
Color? color,
}) {
// Return a large text widget with specified text
return Text(
text,
style: TextStyle(
fontSize: size,
fontWeight: bold ? FontWeight.bold : FontWeight.normal,
color: color,
),
);
}
Widget LinkIcon({
bool external = false,
String? text,
CachedNetworkImage? image,
}) {
// Return a row of items with an icon and text
return Row(
mainAxisSize: MainAxisSize.min,
children: [
if (text != null) ...[LargeText(text)],
if (image != null) ...[image],
Icon(
external ? TablerIcons.external_link : TablerIcons.chevron_right,
color: COLOR_ACTION,
),
],
);
}

View File

@ -3,6 +3,7 @@ import "package:flutter_tabler_icons/flutter_tabler_icons.dart";
import "package:inventree/helpers.dart"; import "package:inventree/helpers.dart";
import "package:inventree/l10.dart"; import "package:inventree/l10.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/refreshable_state.dart";
import "package:inventree/widget/snacks.dart"; import "package:inventree/widget/snacks.dart";
@ -70,36 +71,44 @@ class _ExtraLineDetailWidgetState
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().reference), title: Text(L10().reference),
trailing: Text(widget.item.reference), subtitle: Text(widget.item.reference),
leading: Icon(TablerIcons.hash),
), ),
); );
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().description), title: Text(L10().description),
trailing: Text(widget.item.description), subtitle: Text(widget.item.description),
leading: Icon(TablerIcons.info_circle),
), ),
); );
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().quantity), title: Text(L10().quantity),
trailing: Text(widget.item.quantity.toString()), trailing: LargeText(widget.item.quantity.toString()),
leading: Icon(TablerIcons.progress),
), ),
); );
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().unitPrice), title: Text(L10().unitPrice),
trailing: Text( trailing: LargeText(
renderCurrency(widget.item.price, widget.item.priceCurrency), renderCurrency(widget.item.price, widget.item.priceCurrency),
), ),
leading: Icon(TablerIcons.currency_dollar),
), ),
); );
if (widget.item.notes.isNotEmpty) { if (widget.item.notes.isNotEmpty) {
tiles.add( tiles.add(
ListTile(title: Text(L10().notes), subtitle: Text(widget.item.notes)), ListTile(
title: Text(L10().notes),
subtitle: Text(widget.item.notes),
leading: Icon(TablerIcons.note),
),
); );
} }

View File

@ -5,6 +5,7 @@ import "package:flutter_tabler_icons/flutter_tabler_icons.dart";
import "package:inventree/l10.dart"; import "package:inventree/l10.dart";
import "package:inventree/inventree/model.dart"; import "package:inventree/inventree/model.dart";
import "package:inventree/inventree/purchase_order.dart"; import "package:inventree/inventree/purchase_order.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/paginator.dart"; import "package:inventree/widget/paginator.dart";
import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/refreshable_state.dart";
import "package:inventree/widget/snacks.dart"; import "package:inventree/widget/snacks.dart";
@ -110,7 +111,7 @@ class _PaginatedPOExtraLineListState
return ListTile( return ListTile(
title: Text(line.reference), title: Text(line.reference),
subtitle: Text(line.description), subtitle: Text(line.description),
trailing: Text(line.quantity.toString()), trailing: LargeText(line.quantity.toString(), size: 14),
onTap: () { onTap: () {
line.goToDetailPage(context).then((_) { line.goToDetailPage(context).then((_) {
refresh(); refresh();

View File

@ -11,11 +11,11 @@ import "package:inventree/inventree/company.dart";
import "package:inventree/inventree/part.dart"; import "package:inventree/inventree/part.dart";
import "package:inventree/inventree/purchase_order.dart"; import "package:inventree/inventree/purchase_order.dart";
import "package:inventree/inventree/stock.dart"; import "package:inventree/inventree/stock.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/progress.dart"; import "package:inventree/widget/progress.dart";
import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/refreshable_state.dart";
import "package:inventree/widget/snacks.dart"; import "package:inventree/widget/snacks.dart";
import "package:inventree/widget/company/supplier_part_detail.dart";
/* /*
* Widget for displaying detail view of a single PurchaseOrderLineItem * Widget for displaying detail view of a single PurchaseOrderLineItem
@ -145,7 +145,7 @@ class _POLineDetailWidgetState extends RefreshableState<POLineDetailWidget> {
title: Text(L10().internalPart), title: Text(L10().internalPart),
subtitle: Text(widget.item.partName), subtitle: Text(widget.item.partName),
leading: Icon(TablerIcons.box, color: COLOR_ACTION), leading: Icon(TablerIcons.box, color: COLOR_ACTION),
trailing: api.getThumbnail(widget.item.partImage), trailing: LinkIcon(image: api.getThumbnail(widget.item.partImage)),
onTap: () async { onTap: () async {
showLoadingOverlay(); showLoadingOverlay();
var part = await InvenTreePart().get(widget.item.partId); var part = await InvenTreePart().get(widget.item.partId);
@ -164,6 +164,7 @@ class _POLineDetailWidgetState extends RefreshableState<POLineDetailWidget> {
title: Text(L10().supplierPart), title: Text(L10().supplierPart),
subtitle: Text(widget.item.SKU), subtitle: Text(widget.item.SKU),
leading: Icon(TablerIcons.building, color: COLOR_ACTION), leading: Icon(TablerIcons.building, color: COLOR_ACTION),
trailing: LinkIcon(),
onTap: () async { onTap: () async {
showLoadingOverlay(); showLoadingOverlay();
var part = await InvenTreeSupplierPart().get( var part = await InvenTreeSupplierPart().get(
@ -172,12 +173,7 @@ class _POLineDetailWidgetState extends RefreshableState<POLineDetailWidget> {
hideLoadingOverlay(); hideLoadingOverlay();
if (part is InvenTreeSupplierPart) { if (part is InvenTreeSupplierPart) {
Navigator.push( part.goToDetailPage(context);
context,
MaterialPageRoute(
builder: (context) => SupplierPartDetailWidget(part),
),
);
} }
}, },
), ),
@ -200,11 +196,9 @@ class _POLineDetailWidgetState extends RefreshableState<POLineDetailWidget> {
ListTile( ListTile(
title: Text(L10().received), title: Text(L10().received),
subtitle: ProgressBar(widget.item.progressRatio), subtitle: ProgressBar(widget.item.progressRatio),
trailing: Text( trailing: LargeText(
widget.item.progressString, widget.item.progressString,
style: TextStyle( color: widget.item.isComplete ? COLOR_SUCCESS : COLOR_WARNING,
color: widget.item.isComplete ? COLOR_SUCCESS : COLOR_WARNING,
),
), ),
leading: Icon(TablerIcons.progress), leading: Icon(TablerIcons.progress),
), ),
@ -226,7 +220,7 @@ class _POLineDetailWidgetState extends RefreshableState<POLineDetailWidget> {
ListTile( ListTile(
title: Text(L10().unitPrice), title: Text(L10().unitPrice),
leading: Icon(TablerIcons.currency_dollar), leading: Icon(TablerIcons.currency_dollar),
trailing: Text( trailing: LargeText(
renderCurrency( renderCurrency(
widget.item.purchasePrice, widget.item.purchasePrice,
widget.item.purchasePriceCurrency, widget.item.purchasePriceCurrency,

View File

@ -7,6 +7,7 @@ import "package:inventree/l10.dart";
import "package:inventree/inventree/company.dart"; import "package:inventree/inventree/company.dart";
import "package:inventree/inventree/model.dart"; import "package:inventree/inventree/model.dart";
import "package:inventree/inventree/purchase_order.dart"; import "package:inventree/inventree/purchase_order.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/paginator.dart"; import "package:inventree/widget/paginator.dart";
import "package:inventree/widget/order/po_line_detail.dart"; import "package:inventree/widget/order/po_line_detail.dart";
@ -80,11 +81,9 @@ class _PaginatedPOLineListState
return ListTile( return ListTile(
title: Text(supplierPart.SKU), title: Text(supplierPart.SKU),
subtitle: Text(item.partName), subtitle: Text(item.partName),
trailing: Text( trailing: LargeText(
item.progressString, item.progressString,
style: TextStyle( color: item.isComplete ? COLOR_SUCCESS : COLOR_WARNING,
color: item.isComplete ? COLOR_SUCCESS : COLOR_WARNING,
),
), ),
leading: InvenTreeAPI().getThumbnail(supplierPart.partImage), leading: InvenTreeAPI().getThumbnail(supplierPart.partImage),
onTap: () async { onTap: () async {

View File

@ -14,8 +14,8 @@ import "package:inventree/inventree/stock.dart";
import "package:inventree/inventree/purchase_order.dart"; import "package:inventree/inventree/purchase_order.dart";
import "package:inventree/widget/dialogs.dart"; import "package:inventree/widget/dialogs.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/order/po_extra_line_list.dart"; 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/order/po_line_list.dart";
import "package:inventree/widget/attachment_widget.dart"; import "package:inventree/widget/attachment_widget.dart";
@ -347,11 +347,9 @@ class _PurchaseOrderDetailState
title: Text(widget.order.reference), title: Text(widget.order.reference),
subtitle: Text(widget.order.description), subtitle: Text(widget.order.description),
leading: supplier == null ? null : api.getThumbnail(supplier.thumbnail), leading: supplier == null ? null : api.getThumbnail(supplier.thumbnail),
trailing: Text( trailing: LargeText(
api.PurchaseOrderStatus.label(widget.order.status), api.PurchaseOrderStatus.label(widget.order.status),
style: TextStyle( color: api.PurchaseOrderStatus.color(widget.order.status),
color: api.PurchaseOrderStatus.color(widget.order.status),
),
), ),
), ),
); );
@ -382,6 +380,7 @@ class _PurchaseOrderDetailState
title: Text(L10().supplier), title: Text(L10().supplier),
subtitle: Text(supplier.name), subtitle: Text(supplier.name),
leading: Icon(TablerIcons.building, color: COLOR_ACTION), leading: Icon(TablerIcons.building, color: COLOR_ACTION),
trailing: LinkIcon(),
onTap: () { onTap: () {
supplier.goToDetailPage(context); supplier.goToDetailPage(context);
}, },
@ -406,14 +405,8 @@ class _PurchaseOrderDetailState
title: Text(L10().destination), title: Text(L10().destination),
subtitle: Text(destination!.name), subtitle: Text(destination!.name),
leading: Icon(TablerIcons.map_pin, color: COLOR_ACTION), leading: Icon(TablerIcons.map_pin, color: COLOR_ACTION),
onTap: () => { trailing: LinkIcon(),
Navigator.push( onTap: () => {destination!.goToDetailPage(context)},
context,
MaterialPageRoute(
builder: (context) => LocationDisplayWidget(destination),
),
),
},
), ),
); );
} }
@ -430,9 +423,9 @@ class _PurchaseOrderDetailState
maximum: widget.order.lineItemCount.toDouble(), maximum: widget.order.lineItemCount.toDouble(),
), ),
leading: Icon(TablerIcons.clipboard_check), leading: Icon(TablerIcons.clipboard_check),
trailing: Text( trailing: LargeText(
"${completedLines} / ${widget.order.lineItemCount}", "${completedLines} / ${widget.order.lineItemCount}",
style: TextStyle(color: lineColor), color: lineColor,
), ),
), ),
); );
@ -442,7 +435,7 @@ class _PurchaseOrderDetailState
ListTile( ListTile(
title: Text(L10().extraLineItems), title: Text(L10().extraLineItems),
leading: Icon(TablerIcons.clipboard_list, color: COLOR_ACTION), leading: Icon(TablerIcons.clipboard_list, color: COLOR_ACTION),
trailing: Text(extraLineCount.toString()), trailing: LinkIcon(text: extraLineCount.toString()),
onTap: () => { onTap: () => {
Navigator.push( Navigator.push(
context, context,
@ -461,7 +454,7 @@ class _PurchaseOrderDetailState
ListTile( ListTile(
title: Text(L10().totalPrice), title: Text(L10().totalPrice),
leading: Icon(TablerIcons.currency_dollar), leading: Icon(TablerIcons.currency_dollar),
trailing: Text( trailing: LargeText(
renderCurrency( renderCurrency(
widget.order.totalPrice, widget.order.totalPrice,
widget.order.totalPriceCurrency, widget.order.totalPriceCurrency,
@ -474,7 +467,7 @@ class _PurchaseOrderDetailState
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().issueDate), title: Text(L10().issueDate),
trailing: Text(widget.order.issueDate), trailing: LargeText(widget.order.issueDate),
leading: Icon(TablerIcons.calendar), leading: Icon(TablerIcons.calendar),
), ),
); );
@ -484,7 +477,7 @@ class _PurchaseOrderDetailState
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().startDate), title: Text(L10().startDate),
trailing: Text(widget.order.startDate), trailing: LargeText(widget.order.startDate),
leading: Icon(TablerIcons.calendar), leading: Icon(TablerIcons.calendar),
), ),
); );
@ -494,7 +487,7 @@ class _PurchaseOrderDetailState
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().targetDate), title: Text(L10().targetDate),
trailing: Text(widget.order.targetDate), trailing: LargeText(widget.order.targetDate),
leading: Icon(TablerIcons.calendar), leading: Icon(TablerIcons.calendar),
), ),
); );
@ -504,7 +497,7 @@ class _PurchaseOrderDetailState
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().completionDate), title: Text(L10().completionDate),
trailing: Text(widget.order.completionDate), trailing: LargeText(widget.order.completionDate),
leading: Icon(TablerIcons.calendar), leading: Icon(TablerIcons.calendar),
), ),
); );
@ -521,7 +514,7 @@ class _PurchaseOrderDetailState
? TablerIcons.users ? TablerIcons.users
: TablerIcons.user, : TablerIcons.user,
), ),
trailing: Text(widget.order.responsibleName), trailing: LargeText(widget.order.responsibleName),
), ),
); );
} }
@ -531,6 +524,7 @@ class _PurchaseOrderDetailState
ListTile( ListTile(
title: Text(L10().notes), title: Text(L10().notes),
leading: Icon(TablerIcons.note, color: COLOR_ACTION), leading: Icon(TablerIcons.note, color: COLOR_ACTION),
trailing: LinkIcon(),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -545,7 +539,9 @@ class _PurchaseOrderDetailState
ListTile( ListTile(
title: Text(L10().attachments), title: Text(L10().attachments),
leading: Icon(TablerIcons.file, color: COLOR_ACTION), leading: Icon(TablerIcons.file, color: COLOR_ACTION),
trailing: attachmentCount > 0 ? Text(attachmentCount.toString()) : null, trailing: LinkIcon(
text: attachmentCount > 0 ? attachmentCount.toString() : null,
),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,

View File

@ -4,6 +4,7 @@ import "package:flutter_tabler_icons/flutter_tabler_icons.dart";
import "package:inventree/inventree/company.dart"; import "package:inventree/inventree/company.dart";
import "package:inventree/inventree/model.dart"; import "package:inventree/inventree/model.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/paginator.dart"; import "package:inventree/widget/paginator.dart";
import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/refreshable_state.dart";
import "package:inventree/l10.dart"; import "package:inventree/l10.dart";
@ -173,11 +174,9 @@ class _PaginatedPurchaseOrderListState
leading: supplier == null leading: supplier == null
? null ? null
: InvenTreeAPI().getThumbnail(supplier.thumbnail), : InvenTreeAPI().getThumbnail(supplier.thumbnail),
trailing: Text( trailing: LargeText(
InvenTreeAPI().PurchaseOrderStatus.label(order.status), InvenTreeAPI().PurchaseOrderStatus.label(order.status),
style: TextStyle( color: InvenTreeAPI().PurchaseOrderStatus.color(order.status),
color: InvenTreeAPI().PurchaseOrderStatus.color(order.status),
),
), ),
onTap: () async { onTap: () async {
order.goToDetailPage(context); order.goToDetailPage(context);

View File

@ -6,6 +6,7 @@ import "package:inventree/barcode/sales_order.dart";
import "package:inventree/inventree/company.dart"; import "package:inventree/inventree/company.dart";
import "package:inventree/inventree/sales_order.dart"; import "package:inventree/inventree/sales_order.dart";
import "package:inventree/preferences.dart"; import "package:inventree/preferences.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/order/so_extra_line_list.dart"; import "package:inventree/widget/order/so_extra_line_list.dart";
import "package:inventree/widget/order/so_line_list.dart"; import "package:inventree/widget/order/so_line_list.dart";
import "package:inventree/widget/order/so_shipment_list.dart"; import "package:inventree/widget/order/so_shipment_list.dart";
@ -323,11 +324,9 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
title: Text(widget.order.reference), title: Text(widget.order.reference),
subtitle: Text(widget.order.description), subtitle: Text(widget.order.description),
leading: customer == null ? null : api.getThumbnail(customer.thumbnail), leading: customer == null ? null : api.getThumbnail(customer.thumbnail),
trailing: Text( trailing: LargeText(
api.SalesOrderStatus.label(widget.order.status), api.SalesOrderStatus.label(widget.order.status),
style: TextStyle( color: api.SalesOrderStatus.color(widget.order.status),
color: api.SalesOrderStatus.color(widget.order.status),
),
), ),
), ),
); );
@ -356,6 +355,7 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
title: Text(L10().customer), title: Text(L10().customer),
subtitle: Text(customer.name), subtitle: Text(customer.name),
leading: Icon(TablerIcons.user, color: COLOR_ACTION), leading: Icon(TablerIcons.user, color: COLOR_ACTION),
trailing: LinkIcon(),
onTap: () { onTap: () {
customer.goToDetailPage(context); customer.goToDetailPage(context);
}, },
@ -367,7 +367,7 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().customerReference), title: Text(L10().customerReference),
trailing: Text(widget.order.customerReference), trailing: LargeText(widget.order.customerReference),
leading: Icon(TablerIcons.hash), leading: Icon(TablerIcons.hash),
), ),
); );
@ -375,6 +375,24 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
Color lineColor = widget.order.complete ? COLOR_SUCCESS : COLOR_WARNING; Color lineColor = widget.order.complete ? COLOR_SUCCESS : COLOR_WARNING;
// Shipment progress
if (widget.order.shipmentCount > 0) {
tiles.add(
ListTile(
title: Text(L10().shipments),
subtitle: ProgressBar(
widget.order.completedShipmentCount.toDouble(),
maximum: widget.order.shipmentCount.toDouble(),
),
leading: Icon(TablerIcons.truck_delivery),
trailing: LargeText(
"${widget.order.completedShipmentCount} / ${widget.order.shipmentCount}",
color: lineColor,
),
),
);
}
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().lineItems), title: Text(L10().lineItems),
@ -383,9 +401,9 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
maximum: widget.order.lineItemCount.toDouble(), maximum: widget.order.lineItemCount.toDouble(),
), ),
leading: Icon(TablerIcons.clipboard_check), leading: Icon(TablerIcons.clipboard_check),
trailing: Text( trailing: LargeText(
"${widget.order.completedLineItemCount} / ${widget.order.lineItemCount}", "${widget.order.completedLineItemCount} / ${widget.order.lineItemCount}",
style: TextStyle(color: lineColor), color: lineColor,
), ),
), ),
); );
@ -395,7 +413,7 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
ListTile( ListTile(
title: Text(L10().extraLineItems), title: Text(L10().extraLineItems),
leading: Icon(TablerIcons.clipboard_list, color: COLOR_ACTION), leading: Icon(TablerIcons.clipboard_list, color: COLOR_ACTION),
trailing: Text(extraLineCount.toString()), trailing: LinkIcon(text: extraLineCount.toString()),
onTap: () => { onTap: () => {
Navigator.push( Navigator.push(
context, context,
@ -410,31 +428,13 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
), ),
); );
// Shipment progress
if (widget.order.shipmentCount > 0) {
tiles.add(
ListTile(
title: Text(L10().shipments),
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),
),
),
);
}
// TODO: total price // TODO: total price
if (widget.order.startDate.isNotEmpty) { if (widget.order.startDate.isNotEmpty) {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().startDate), title: Text(L10().startDate),
trailing: Text(widget.order.startDate), trailing: LargeText(widget.order.startDate),
leading: Icon(TablerIcons.calendar), leading: Icon(TablerIcons.calendar),
), ),
); );
@ -444,7 +444,7 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().targetDate), title: Text(L10().targetDate),
trailing: Text(widget.order.targetDate), trailing: LargeText(widget.order.targetDate),
leading: Icon(TablerIcons.calendar), leading: Icon(TablerIcons.calendar),
), ),
); );
@ -454,7 +454,7 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().completionDate), title: Text(L10().completionDate),
trailing: Text(widget.order.shipmentDate), trailing: LargeText(widget.order.shipmentDate),
leading: Icon(TablerIcons.calendar), leading: Icon(TablerIcons.calendar),
), ),
); );
@ -471,7 +471,7 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
? TablerIcons.users ? TablerIcons.users
: TablerIcons.user, : TablerIcons.user,
), ),
trailing: Text(widget.order.responsibleName), trailing: LargeText(widget.order.responsibleName),
), ),
); );
} }
@ -481,6 +481,7 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
ListTile( ListTile(
title: Text(L10().notes), title: Text(L10().notes),
leading: Icon(TablerIcons.note, color: COLOR_ACTION), leading: Icon(TablerIcons.note, color: COLOR_ACTION),
trailing: LinkIcon(),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -495,7 +496,9 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
ListTile( ListTile(
title: Text(L10().attachments), title: Text(L10().attachments),
leading: Icon(TablerIcons.file, color: COLOR_ACTION), leading: Icon(TablerIcons.file, color: COLOR_ACTION),
trailing: attachmentCount > 0 ? Text(attachmentCount.toString()) : null, trailing: LinkIcon(
text: attachmentCount > 0 ? attachmentCount.toString() : null,
),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -519,8 +522,8 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
List<Widget> getTabIcons(BuildContext context) { List<Widget> getTabIcons(BuildContext context) {
return [ return [
Tab(text: L10().details), Tab(text: L10().details),
Tab(text: L10().lineItems),
Tab(text: L10().shipments), Tab(text: L10().shipments),
Tab(text: L10().lineItems),
]; ];
} }
@ -528,8 +531,8 @@ class _SalesOrderDetailState extends RefreshableState<SalesOrderDetailWidget> {
List<Widget> getTabs(BuildContext context) { List<Widget> getTabs(BuildContext context) {
return [ return [
ListView(children: orderTiles(context)), ListView(children: orderTiles(context)),
PaginatedSOLineList({"order": widget.order.pk.toString()}),
PaginatedSOShipmentList({"order": widget.order.pk.toString()}), PaginatedSOShipmentList({"order": widget.order.pk.toString()}),
PaginatedSOLineList({"order": widget.order.pk.toString()}),
]; ];
} }
} }

View File

@ -2,6 +2,7 @@ import "package:flutter/material.dart";
import "package:flutter_speed_dial/flutter_speed_dial.dart"; import "package:flutter_speed_dial/flutter_speed_dial.dart";
import "package:flutter_tabler_icons/flutter_tabler_icons.dart"; import "package:flutter_tabler_icons/flutter_tabler_icons.dart";
import "package:inventree/inventree/sales_order.dart"; import "package:inventree/inventree/sales_order.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/paginator.dart"; import "package:inventree/widget/paginator.dart";
import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/refreshable_state.dart";
@ -155,11 +156,9 @@ class _PaginatedSalesOrderListState
leading: customer == null leading: customer == null
? null ? null
: InvenTreeAPI().getThumbnail(customer.thumbnail), : InvenTreeAPI().getThumbnail(customer.thumbnail),
trailing: Text( trailing: LargeText(
InvenTreeAPI().SalesOrderStatus.label(order.status), InvenTreeAPI().SalesOrderStatus.label(order.status),
style: TextStyle( color: InvenTreeAPI().SalesOrderStatus.color(order.status),
color: InvenTreeAPI().SalesOrderStatus.color(order.status),
),
), ),
onTap: () async { onTap: () async {
order.goToDetailPage(context); order.goToDetailPage(context);

View File

@ -6,6 +6,7 @@ import "package:inventree/l10.dart";
import "package:inventree/inventree/model.dart"; import "package:inventree/inventree/model.dart";
import "package:inventree/inventree/sales_order.dart"; import "package:inventree/inventree/sales_order.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/paginator.dart"; import "package:inventree/widget/paginator.dart";
import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/refreshable_state.dart";
@ -112,7 +113,7 @@ class _PaginatedSOExtraLineListState
return ListTile( return ListTile(
title: Text(line.reference), title: Text(line.reference),
subtitle: Text(line.description), subtitle: Text(line.description),
trailing: Text(line.quantity.toString()), trailing: LargeText(line.quantity.toString()),
onTap: () { onTap: () {
line.goToDetailPage(context).then((_) { line.goToDetailPage(context).then((_) {
refresh(); refresh();

View File

@ -10,6 +10,7 @@ import "package:inventree/barcode/sales_order.dart";
import "package:inventree/inventree/part.dart"; import "package:inventree/inventree/part.dart";
import "package:inventree/inventree/sales_order.dart"; import "package:inventree/inventree/sales_order.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/refreshable_state.dart";
import "package:inventree/widget/progress.dart"; import "package:inventree/widget/progress.dart";
@ -172,7 +173,7 @@ class _SOLineDetailWidgetState extends RefreshableState<SoLineDetailWidget> {
title: Text(L10().part), title: Text(L10().part),
subtitle: Text(widget.item.partName), subtitle: Text(widget.item.partName),
leading: Icon(TablerIcons.box, color: COLOR_ACTION), leading: Icon(TablerIcons.box, color: COLOR_ACTION),
trailing: api.getThumbnail(widget.item.partImage), trailing: LinkIcon(image: api.getThumbnail(widget.item.partImage)),
onTap: () async { onTap: () async {
showLoadingOverlay(); showLoadingOverlay();
var part = await InvenTreePart().get(widget.item.partId); var part = await InvenTreePart().get(widget.item.partId);
@ -190,7 +191,7 @@ class _SOLineDetailWidgetState extends RefreshableState<SoLineDetailWidget> {
ListTile( ListTile(
title: Text(L10().availableStock), title: Text(L10().availableStock),
leading: Icon(TablerIcons.packages), leading: Icon(TablerIcons.packages),
trailing: Text(simpleNumberString(widget.item.availableStock)), trailing: LargeText(simpleNumberString(widget.item.availableStock)),
), ),
); );
@ -200,11 +201,9 @@ class _SOLineDetailWidgetState extends RefreshableState<SoLineDetailWidget> {
leading: Icon(TablerIcons.clipboard_check), leading: Icon(TablerIcons.clipboard_check),
title: Text(L10().allocated), title: Text(L10().allocated),
subtitle: ProgressBar(widget.item.allocatedRatio), subtitle: ProgressBar(widget.item.allocatedRatio),
trailing: Text( trailing: LargeText(
widget.item.allocatedString, widget.item.allocatedString,
style: TextStyle( color: widget.item.isAllocated ? COLOR_SUCCESS : COLOR_WARNING,
color: widget.item.isAllocated ? COLOR_SUCCESS : COLOR_WARNING,
),
), ),
), ),
); );
@ -214,11 +213,9 @@ class _SOLineDetailWidgetState extends RefreshableState<SoLineDetailWidget> {
ListTile( ListTile(
title: Text(L10().shipped), title: Text(L10().shipped),
subtitle: ProgressBar(widget.item.progressRatio), subtitle: ProgressBar(widget.item.progressRatio),
trailing: Text( trailing: LargeText(
widget.item.progressString, 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),
), ),

View File

@ -1,5 +1,6 @@
import "package:flutter/material.dart"; import "package:flutter/material.dart";
import "package:inventree/l10.dart"; import "package:inventree/l10.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/order/so_line_detail.dart"; import "package:inventree/widget/order/so_line_detail.dart";
import "package:inventree/widget/paginator.dart"; import "package:inventree/widget/paginator.dart";
import "package:inventree/inventree/model.dart"; import "package:inventree/inventree/model.dart";
@ -67,11 +68,9 @@ class _PaginatedSOLineListState
title: Text(part.name), title: Text(part.name),
subtitle: Text(part.description), subtitle: Text(part.description),
leading: InvenTreeAPI().getThumbnail(part.thumbnail), leading: InvenTreeAPI().getThumbnail(part.thumbnail),
trailing: Text( trailing: LargeText(
item.progressString, item.progressString,
style: TextStyle( color: item.isComplete ? COLOR_SUCCESS : COLOR_WARNING,
color: item.isComplete ? COLOR_SUCCESS : COLOR_WARNING,
),
), ),
onTap: () async { onTap: () async {
showLoadingOverlay(); showLoadingOverlay();

View File

@ -2,6 +2,7 @@ import "package:flutter/material.dart";
import "package:flutter_tabler_icons/flutter_tabler_icons.dart"; import "package:flutter_tabler_icons/flutter_tabler_icons.dart";
import "package:inventree/app_colors.dart"; import "package:inventree/app_colors.dart";
import "package:inventree/inventree/sales_order.dart"; import "package:inventree/inventree/sales_order.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/paginator.dart"; import "package:inventree/widget/paginator.dart";
import "package:inventree/inventree/model.dart"; import "package:inventree/inventree/model.dart";
@ -56,7 +57,9 @@ class _PaginatedSOShipmentListState
leading: shipment.shipped leading: shipment.shipped
? Icon(TablerIcons.calendar_check, color: COLOR_SUCCESS) ? Icon(TablerIcons.calendar_check, color: COLOR_SUCCESS)
: Icon(TablerIcons.calendar_cancel, color: COLOR_WARNING), : Icon(TablerIcons.calendar_cancel, color: COLOR_WARNING),
trailing: shipment.shipped ? Text(shipment.shipment_date ?? "") : null, trailing: shipment.shipped
? LargeText(shipment.shipment_date ?? "")
: null,
); );
} }
} }

View File

@ -6,6 +6,7 @@ import "package:inventree/app_colors.dart";
import "package:inventree/l10.dart"; import "package:inventree/l10.dart";
import "package:inventree/inventree/part.dart"; import "package:inventree/inventree/part.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/part/category_list.dart"; import "package:inventree/widget/part/category_list.dart";
import "package:inventree/widget/part/part_list.dart"; import "package:inventree/widget/part/part_list.dart";
@ -144,6 +145,7 @@ class _CategoryDisplayState extends RefreshableState<CategoryDisplayWidget> {
title: Text(L10().parentCategory), title: Text(L10().parentCategory),
subtitle: Text("${widget.category?.parentPathString}"), subtitle: Text("${widget.category?.parentPathString}"),
leading: Icon(TablerIcons.arrow_move_up, color: COLOR_ACTION), leading: Icon(TablerIcons.arrow_move_up, color: COLOR_ACTION),
trailing: LinkIcon(),
onTap: () async { onTap: () async {
int parentId = widget.category?.parentId ?? -1; int parentId = widget.category?.parentId ?? -1;

View File

@ -2,6 +2,7 @@ import "package:flutter/material.dart";
import "package:inventree/inventree/model.dart"; import "package:inventree/inventree/model.dart";
import "package:inventree/inventree/part.dart"; import "package:inventree/inventree/part.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/paginator.dart"; import "package:inventree/widget/paginator.dart";
import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/refreshable_state.dart";
@ -96,7 +97,7 @@ class _PaginatedPartCategoryListState
return ListTile( return ListTile(
title: Text(category.name), title: Text(category.name),
subtitle: Text(category.pathstring), subtitle: Text(category.pathstring),
trailing: Text("${category.partcount}"), trailing: LargeText("${category.partcount}", size: 14),
leading: category.customIcon == null ? null : Icon(category.customIcon), leading: category.customIcon == null ? null : Icon(category.customIcon),
onTap: () { onTap: () {
category.goToDetailPage(context); category.goToDetailPage(context);

View File

@ -14,6 +14,7 @@ import "package:inventree/labels.dart";
import "package:inventree/preferences.dart"; import "package:inventree/preferences.dart";
import "package:inventree/widget/attachment_widget.dart"; import "package:inventree/widget/attachment_widget.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/part/bom_list.dart"; import "package:inventree/widget/part/bom_list.dart";
import "package:inventree/widget/part/part_list.dart"; import "package:inventree/widget/part/part_list.dart";
import "package:inventree/widget/notes_widget.dart"; import "package:inventree/widget/notes_widget.dart";
@ -334,6 +335,7 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
title: Text(L10().templatePart), title: Text(L10().templatePart),
subtitle: Text(parentPart!.fullname), subtitle: Text(parentPart!.fullname),
leading: api.getImage(parentPart!.thumbnail, width: 32, height: 32), leading: api.getImage(parentPart!.thumbnail, width: 32, height: 32),
trailing: LinkIcon(),
onTap: () { onTap: () {
parentPart?.goToDetailPage(context); parentPart?.goToDetailPage(context);
}, },
@ -348,6 +350,7 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
title: Text(L10().partCategory), title: Text(L10().partCategory),
subtitle: Text("${part.categoryName}"), subtitle: Text("${part.categoryName}"),
leading: Icon(TablerIcons.sitemap, color: COLOR_ACTION), leading: Icon(TablerIcons.sitemap, color: COLOR_ACTION),
trailing: LinkIcon(),
onTap: () async { onTap: () async {
if (part.categoryId > 0) { if (part.categoryId > 0) {
showLoadingOverlay(); showLoadingOverlay();
@ -385,7 +388,7 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
ListTile( ListTile(
title: Text(L10().variants), title: Text(L10().variants),
leading: Icon(TablerIcons.versions, color: COLOR_ACTION), leading: Icon(TablerIcons.versions, color: COLOR_ACTION),
trailing: Text(variantCount.toString()), trailing: LinkIcon(text: variantCount.toString()),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -405,10 +408,7 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
title: Text(L10().availableStock), title: Text(L10().availableStock),
subtitle: Text(L10().stockDetails), subtitle: Text(L10().stockDetails),
leading: Icon(TablerIcons.packages), leading: Icon(TablerIcons.packages),
trailing: Text( trailing: LargeText(part.stockString()),
part.stockString(),
style: TextStyle(fontWeight: FontWeight.bold),
),
), ),
); );
@ -422,11 +422,11 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().partPricing), title: Text(L10().partPricing),
leading: Icon(TablerIcons.currency_dollar, color: COLOR_ACTION), subtitle: Text(
trailing: Text(
pricing.isNotEmpty ? pricing : L10().noPricingAvailable, pricing.isNotEmpty ? pricing : L10().noPricingAvailable,
style: TextStyle(fontWeight: FontWeight.bold),
), ),
leading: Icon(TablerIcons.currency_dollar, color: COLOR_ACTION),
trailing: LinkIcon(),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -448,7 +448,7 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
title: Text(L10().onOrder), title: Text(L10().onOrder),
subtitle: Text(L10().onOrderDetails), subtitle: Text(L10().onOrderDetails),
leading: Icon(TablerIcons.shopping_cart), leading: Icon(TablerIcons.shopping_cart),
trailing: Text("${part.onOrderString}"), trailing: LargeText("${part.onOrderString}"),
onTap: () { onTap: () {
// TODO - Order views // TODO - Order views
}, },
@ -463,7 +463,7 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
ListTile( ListTile(
title: Text(L10().billOfMaterials), title: Text(L10().billOfMaterials),
leading: Icon(TablerIcons.list_tree, color: COLOR_ACTION), leading: Icon(TablerIcons.list_tree, color: COLOR_ACTION),
trailing: Text(bomCount.toString()), trailing: LinkIcon(text: bomCount.toString()),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -482,9 +482,9 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
ListTile( ListTile(
title: Text(L10().building), title: Text(L10().building),
leading: Icon(TablerIcons.tools), leading: Icon(TablerIcons.tools),
trailing: Text("${simpleNumberString(part.building)}"), trailing: LargeText("${simpleNumberString(part.building)}"),
onTap: () { onTap: () {
// TODO // TODO: List of active build orders?
}, },
), ),
); );
@ -498,7 +498,7 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
title: Text(L10().usedIn), title: Text(L10().usedIn),
subtitle: Text(L10().usedInDetails), subtitle: Text(L10().usedInDetails),
leading: Icon(TablerIcons.stack_2, color: COLOR_ACTION), leading: Icon(TablerIcons.stack_2, color: COLOR_ACTION),
trailing: Text(usedInCount.toString()), trailing: LinkIcon(text: usedInCount.toString()),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -543,9 +543,9 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
title: Text(L10().usedIn), title: Text(L10().usedIn),
subtitle: Text(L10().usedInDetails), subtitle: Text(L10().usedInDetails),
leading: Icon(TablerIcons.sitemap), leading: Icon(TablerIcons.sitemap),
trailing: Text("${part.usedInCount}"), trailing: LargeText("${part.usedInCount}"),
onTap: () { onTap: () {
// TODO // TODO: Show assemblies which use this part
}, },
), ),
); );
@ -557,7 +557,7 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
ListTile( ListTile(
title: Text(L10().suppliers), title: Text(L10().suppliers),
leading: Icon(TablerIcons.building_factory, color: COLOR_ACTION), leading: Icon(TablerIcons.building_factory, color: COLOR_ACTION),
trailing: Text("${part.supplierCount}"), trailing: LinkIcon(text: "${part.supplierCount}"),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -577,7 +577,7 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
ListTile( ListTile(
title: Text(L10().notes), title: Text(L10().notes),
leading: Icon(TablerIcons.note, color: COLOR_ACTION), leading: Icon(TablerIcons.note, color: COLOR_ACTION),
trailing: Text(""), trailing: LinkIcon(),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -591,7 +591,9 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
ListTile( ListTile(
title: Text(L10().attachments), title: Text(L10().attachments),
leading: Icon(TablerIcons.file, color: COLOR_ACTION), leading: Icon(TablerIcons.file, color: COLOR_ACTION),
trailing: attachmentCount > 0 ? Text(attachmentCount.toString()) : null, trailing: LinkIcon(
text: attachmentCount > 0 ? attachmentCount.toString() : null,
),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,

View File

@ -5,6 +5,7 @@ import "package:inventree/l10.dart";
import "package:inventree/inventree/model.dart"; import "package:inventree/inventree/model.dart";
import "package:inventree/inventree/part.dart"; import "package:inventree/inventree/part.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/paginator.dart"; import "package:inventree/widget/paginator.dart";
import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/refreshable_state.dart";
@ -121,10 +122,7 @@ class _PaginatedPartListState extends PaginatedSearchState<PaginatedPartList> {
return ListTile( return ListTile(
title: Text(part.fullname), title: Text(part.fullname),
subtitle: Text(part.description), subtitle: Text(part.description),
trailing: Text( trailing: LargeText(part.stockString()),
part.stockString(),
style: TextStyle(fontSize: 16, fontWeight: FontWeight.bold),
),
leading: InvenTreeAPI().getThumbnail(part.thumbnail), leading: InvenTreeAPI().getThumbnail(part.thumbnail),
onTap: () { onTap: () {
part.goToDetailPage(context); part.goToDetailPage(context);

View File

@ -42,7 +42,7 @@ mixin BaseWidgetProperties {
// Override getTiles to replace the internal context // Override getTiles to replace the internal context
return ListView( return ListView(
physics: AlwaysScrollableScrollPhysics(), physics: AlwaysScrollableScrollPhysics(),
children: getTiles(context), children: [Divider(), ...getTiles(context)],
); );
} }

View File

@ -10,6 +10,7 @@ import "package:inventree/l10.dart";
import "package:inventree/inventree/stock.dart"; import "package:inventree/inventree/stock.dart";
import "package:inventree/preferences.dart"; import "package:inventree/preferences.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/stock/location_list.dart"; import "package:inventree/widget/stock/location_list.dart";
import "package:inventree/widget/progress.dart"; import "package:inventree/widget/progress.dart";
@ -341,6 +342,7 @@ class _LocationDisplayState extends RefreshableState<LocationDisplayWidget> {
title: Text(L10().parentLocation), title: Text(L10().parentLocation),
subtitle: Text("${location!.parentPathString}"), subtitle: Text("${location!.parentPathString}"),
leading: Icon(TablerIcons.arrow_move_up, color: COLOR_ACTION), leading: Icon(TablerIcons.arrow_move_up, color: COLOR_ACTION),
trailing: LinkIcon(),
onTap: () async { onTap: () async {
int parentId = location?.parentId ?? -1; int parentId = location?.parentId ?? -1;

View File

@ -2,6 +2,7 @@ import "package:flutter/material.dart";
import "package:inventree/inventree/model.dart"; import "package:inventree/inventree/model.dart";
import "package:inventree/inventree/stock.dart"; import "package:inventree/inventree/stock.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/paginator.dart"; import "package:inventree/widget/paginator.dart";
import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/refreshable_state.dart";
@ -86,7 +87,7 @@ class _PaginatedStockLocationListState
return ListTile( return ListTile(
title: Text(location.name), title: Text(location.name),
subtitle: Text(location.pathstring), subtitle: Text(location.pathstring),
trailing: Text("${location.itemcount}"), trailing: LargeText("${location.itemcount}", size: 14),
leading: location.customIcon == null ? null : Icon(location.customIcon), leading: location.customIcon == null ? null : Icon(location.customIcon),
onTap: () { onTap: () {
location.goToDetailPage(context); location.goToDetailPage(context);

View File

@ -18,9 +18,9 @@ import "package:inventree/inventree/company.dart";
import "package:inventree/inventree/stock.dart"; import "package:inventree/inventree/stock.dart";
import "package:inventree/inventree/part.dart"; import "package:inventree/inventree/part.dart";
import "package:inventree/widget/company/supplier_part_detail.dart";
import "package:inventree/widget/dialogs.dart"; import "package:inventree/widget/dialogs.dart";
import "package:inventree/widget/attachment_widget.dart"; import "package:inventree/widget/attachment_widget.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/progress.dart"; import "package:inventree/widget/progress.dart";
import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/refreshable_state.dart";
import "package:inventree/widget/snacks.dart"; import "package:inventree/widget/snacks.dart";
@ -490,14 +490,11 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
Widget? trailing; Widget? trailing;
if (!widget.item.isInStock) { if (!widget.item.isInStock) {
trailing = Text(L10().unavailable, style: TextStyle(color: COLOR_DANGER)); trailing = LargeText(L10().unavailable, color: COLOR_DANGER);
} else if (!widget.item.isSerialized()) { } else if (!widget.item.isSerialized()) {
trailing = Text( trailing = LargeText(
widget.item.quantityString(), widget.item.quantityString(),
style: TextStyle( color: api.StockStatus.color(widget.item.status),
fontSize: 20,
color: api.StockStatus.color(widget.item.status),
),
); );
} }
@ -518,7 +515,7 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
} }
} }
}, },
//trailing: Text(item.serialOrQuantityDisplay()), //trailing: LargeText(item.serialOrQuantityDisplay()),
), ),
); );
} }
@ -546,6 +543,7 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
title: Text(L10().stockLocation), title: Text(L10().stockLocation),
subtitle: Text("${widget.item.locationPathString}"), subtitle: Text("${widget.item.locationPathString}"),
leading: Icon(TablerIcons.location, color: COLOR_ACTION), leading: Icon(TablerIcons.location, color: COLOR_ACTION),
trailing: LinkIcon(),
onTap: () async { onTap: () async {
if (widget.item.locationId > 0) { if (widget.item.locationId > 0) {
showLoadingOverlay(); showLoadingOverlay();
@ -587,7 +585,7 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
? Text(L10().quantityAvailable) ? Text(L10().quantityAvailable)
: Text(L10().quantity), : Text(L10().quantity),
leading: Icon(TablerIcons.packages), leading: Icon(TablerIcons.packages),
trailing: Text("${widget.item.quantityString()}"), trailing: LargeText("${widget.item.quantityString()}"),
), ),
); );
} }
@ -613,9 +611,9 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
ListTile( ListTile(
title: Text(L10().status), title: Text(L10().status),
leading: Icon(TablerIcons.help_circle), leading: Icon(TablerIcons.help_circle),
trailing: Text( trailing: LargeText(
api.StockStatus.label(widget.item.status), api.StockStatus.label(widget.item.status),
style: TextStyle(color: api.StockStatus.color(widget.item.status)), color: api.StockStatus.color(widget.item.status),
), ),
), ),
); );
@ -627,10 +625,7 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
title: Text(L10().supplierPart), title: Text(L10().supplierPart),
subtitle: Text(widget.item.supplierSKU), subtitle: Text(widget.item.supplierSKU),
leading: Icon(TablerIcons.building, color: COLOR_ACTION), leading: Icon(TablerIcons.building, color: COLOR_ACTION),
trailing: InvenTreeAPI().getThumbnail( trailing: LinkIcon(),
widget.item.supplierImage,
hideIfNull: true,
),
onTap: () async { onTap: () async {
showLoadingOverlay(); showLoadingOverlay();
var sp = await InvenTreeSupplierPart().get( var sp = await InvenTreeSupplierPart().get(
@ -638,12 +633,7 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
); );
hideLoadingOverlay(); hideLoadingOverlay();
if (sp is InvenTreeSupplierPart) { if (sp is InvenTreeSupplierPart) {
Navigator.push( sp.goToDetailPage(context);
context,
MaterialPageRoute(
builder: (context) => SupplierPartDetailWidget(sp),
),
);
} }
}, },
), ),
@ -667,9 +657,11 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().salesOrder), title: Text(L10().salesOrder),
subtitle: Text(salesOrder?.description ?? ""), subtitle: Text(
salesOrder?.reference ?? salesOrder?.description ?? "",
),
leading: Icon(TablerIcons.truck_delivery, color: COLOR_ACTION), leading: Icon(TablerIcons.truck_delivery, color: COLOR_ACTION),
trailing: Text(salesOrder?.reference ?? ""), trailing: LinkIcon(),
onTap: () { onTap: () {
salesOrder?.goToDetailPage(context); salesOrder?.goToDetailPage(context);
}, },
@ -681,9 +673,9 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().customer), title: Text(L10().customer),
subtitle: Text(customer?.description ?? ""), subtitle: Text("${customer?.name} - ${customer?.description}"),
leading: Icon(TablerIcons.building_store, color: COLOR_ACTION), leading: Icon(TablerIcons.building_store, color: COLOR_ACTION),
trailing: Text(customer?.name ?? ""), trailing: LinkIcon(),
onTap: () { onTap: () {
customer?.goToDetailPage(context); customer?.goToDetailPage(context);
}, },
@ -741,7 +733,7 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
tiles.add( tiles.add(
ListTile( ListTile(
title: Text(L10().lastUpdated), title: Text(L10().lastUpdated),
subtitle: Text(widget.item.updatedDateString), trailing: LargeText(widget.item.updatedDateString),
leading: Icon(TablerIcons.calendar), leading: Icon(TablerIcons.calendar),
), ),
); );
@ -775,7 +767,7 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
ListTile( ListTile(
title: Text(L10().testResults), title: Text(L10().testResults),
leading: Icon(TablerIcons.list_check, color: COLOR_ACTION), leading: Icon(TablerIcons.list_check, color: COLOR_ACTION),
trailing: Text("${widget.item.testResultCount}"), trailing: LinkIcon(text: "${widget.item.testResultCount}"),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -795,7 +787,7 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
ListTile( ListTile(
title: Text(L10().purchasePrice), title: Text(L10().purchasePrice),
leading: Icon(TablerIcons.currency_dollar), leading: Icon(TablerIcons.currency_dollar),
trailing: Text( trailing: LargeText(
renderCurrency( renderCurrency(
widget.item.purchasePrice, widget.item.purchasePrice,
widget.item.purchasePriceCurrency, widget.item.purchasePriceCurrency,
@ -812,7 +804,7 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
ListTile( ListTile(
title: Text(L10().history), title: Text(L10().history),
leading: Icon(TablerIcons.history, color: COLOR_ACTION), leading: Icon(TablerIcons.history, color: COLOR_ACTION),
trailing: Text("${widget.item.trackingItemCount}"), trailing: LinkIcon(text: "${widget.item.trackingItemCount}"),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -832,6 +824,7 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
ListTile( ListTile(
title: Text(L10().notes), title: Text(L10().notes),
leading: Icon(TablerIcons.note, color: COLOR_ACTION), leading: Icon(TablerIcons.note, color: COLOR_ACTION),
trailing: LinkIcon(),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,
@ -845,7 +838,9 @@ class _StockItemDisplayState extends RefreshableState<StockDetailWidget> {
ListTile( ListTile(
title: Text(L10().attachments), title: Text(L10().attachments),
leading: Icon(TablerIcons.file, color: COLOR_ACTION), leading: Icon(TablerIcons.file, color: COLOR_ACTION),
trailing: attachmentCount > 0 ? Text(attachmentCount.toString()) : null, trailing: LinkIcon(
text: attachmentCount > 0 ? attachmentCount.toString() : null,
),
onTap: () { onTap: () {
Navigator.push( Navigator.push(
context, context,

View File

@ -2,6 +2,7 @@ import "package:flutter/material.dart";
import "package:inventree/inventree/model.dart"; import "package:inventree/inventree/model.dart";
import "package:inventree/inventree/stock.dart"; import "package:inventree/inventree/stock.dart";
import "package:inventree/widget/link_icon.dart";
import "package:inventree/widget/paginator.dart"; import "package:inventree/widget/paginator.dart";
import "package:inventree/widget/refreshable_state.dart"; import "package:inventree/widget/refreshable_state.dart";
import "package:inventree/l10.dart"; import "package:inventree/l10.dart";
@ -130,16 +131,10 @@ class _PaginatedStockItemListState
title: Text("${item.partName}"), title: Text("${item.partName}"),
subtitle: Text(item.locationPathString), subtitle: Text(item.locationPathString),
leading: InvenTreeAPI().getThumbnail(item.partThumbnail), leading: InvenTreeAPI().getThumbnail(item.partThumbnail),
trailing: SizedBox( trailing: LargeText(
width: 48, item.displayQuantity,
child: Text( size: 14,
"${item.displayQuantity}", color: InvenTreeAPI().StockStatus.color(item.status),
style: TextStyle(
fontWeight: FontWeight.bold,
fontSize: 14,
color: InvenTreeAPI().StockStatus.color(item.status),
),
),
), ),
onTap: () { onTap: () {
item.goToDetailPage(context); item.goToDetailPage(context);