2
0
mirror of https://github.com/inventree/inventree-app.git synced 2025-06-17 04:35:26 +00:00

Supplier part support (#253)

* Bump version and release noes

* Add barebone list and detail widgets for the SupplierPart model

* Launch into SupplierPartList from CompanyDetail

* Update StockDetail widget

* Fixes for SupplierPart model

* Add images to supplier part list

* Add search functionality to SupplierPart list

* Added details to SupplierPartDetail widget

* Link through to supplier company

* Add some more details

* Adds ability to edit SupplierPart information

* Navigate to supplier part list from part detail page

* Display supplier part information on stock item detail page

* Add barcode scan response for SupplierPart

* Refactor barcode scanning code

* Navigate to supplier part detail from stock item page

* Support custom barcode for SupplierPart via app

* Cleanup comment

* linting

* Fix override

* Enable display of supplier list on home screen

* Code cleanup

* Update release noets
This commit is contained in:
Oliver
2023-02-04 09:05:36 +11:00
committed by GitHub
parent ce37d0e757
commit 15bf109296
22 changed files with 798 additions and 295 deletions

View File

@ -1,16 +1,22 @@
import "package:inventree/api.dart";
import "package:inventree/app_colors.dart";
import "package:inventree/inventree/company.dart";
import "package:inventree/inventree/purchase_order.dart";
import "package:inventree/widget/purchase_order_list.dart";
import "package:inventree/widget/refreshable_state.dart";
import "package:flutter/material.dart";
import "package:font_awesome_flutter/font_awesome_flutter.dart";
import "package:inventree/l10.dart";
import "package:inventree/api.dart";
import "package:inventree/app_colors.dart";
import "package:inventree/inventree/company.dart";
import "package:inventree/inventree/purchase_order.dart";
import "package:inventree/widget/purchase_order_list.dart";
import "package:inventree/widget/refreshable_state.dart";
import "package:inventree/widget/snacks.dart";
import "package:inventree/widget/supplier_part_list.dart";
/*
* Widget for displaying detail view of a single Company instance
*/
class CompanyDetailWidget extends StatefulWidget {
const CompanyDetailWidget(this.company, {Key? key}) : super(key: key);
@ -18,18 +24,18 @@ class CompanyDetailWidget extends StatefulWidget {
final InvenTreeCompany company;
@override
_CompanyDetailState createState() => _CompanyDetailState(company);
_CompanyDetailState createState() => _CompanyDetailState();
}
class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
_CompanyDetailState(this.company);
final InvenTreeCompany company;
_CompanyDetailState();
List<InvenTreePurchaseOrder> outstandingOrders = [];
int supplierPartCount = 0;
@override
String getAppBarTitle(BuildContext context) => L10().company;
@ -43,7 +49,7 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
IconButton(
icon: FaIcon(FontAwesomeIcons.globe),
onPressed: () async {
company.goToInvenTreePage();
widget.company.goToInvenTreePage();
},
)
);
@ -64,16 +70,35 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
@override
Future<void> request(BuildContext context) async {
await company.reload();
final bool result = await widget.company.reload();
if (company.isSupplier) {
outstandingOrders = await company.getPurchaseOrders(outstanding: true);
if (!result || widget.company.pk <= 0) {
// Company could not be loaded for some reason
Navigator.of(context).pop();
return;
}
if (widget.company.isSupplier) {
outstandingOrders = await widget.company.getPurchaseOrders(outstanding: true);
}
InvenTreeSupplierPart().count(
filters: {
"supplier": widget.company.pk.toString()
}
).then((value) {
if (mounted) {
setState(() {
supplierPartCount = value;
});
}
});
}
Future <void> editCompany(BuildContext context) async {
company.editForm(
widget.company.editForm(
context,
L10().companyEdit,
onSuccess: (data) async {
@ -83,6 +108,9 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
);
}
/*
* Construct a list of tiles to display for this Company instance
*/
List<Widget> _companyTiles() {
List<Widget> tiles = [];
@ -91,15 +119,15 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
tiles.add(Card(
child: ListTile(
title: Text("${company.name}"),
subtitle: Text("${company.description}"),
leading: InvenTreeAPI().getImage(company.image, width: 40, height: 40),
title: Text("${widget.company.name}"),
subtitle: Text("${widget.company.description}"),
leading: InvenTreeAPI().getImage(widget.company.image, width: 40, height: 40),
),
));
if (company.website.isNotEmpty) {
if (widget.company.website.isNotEmpty) {
tiles.add(ListTile(
title: Text("${company.website}"),
title: Text("${widget.company.website}"),
leading: FaIcon(FontAwesomeIcons.globe),
onTap: () {
// TODO - Open website
@ -109,9 +137,9 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
sep = true;
}
if (company.email.isNotEmpty) {
if (widget.company.email.isNotEmpty) {
tiles.add(ListTile(
title: Text("${company.email}"),
title: Text("${widget.company.email}"),
leading: FaIcon(FontAwesomeIcons.at),
onTap: () {
// TODO - Open email
@ -121,9 +149,9 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
sep = true;
}
if (company.phone.isNotEmpty) {
if (widget.company.phone.isNotEmpty) {
tiles.add(ListTile(
title: Text("${company.phone}"),
title: Text("${widget.company.phone}"),
leading: FaIcon(FontAwesomeIcons.phone),
onTap: () {
// TODO - Call phone number
@ -134,12 +162,12 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
}
// External link
if (company.link.isNotEmpty) {
if (widget.company.link.isNotEmpty) {
tiles.add(ListTile(
title: Text("${company.link}"),
title: Text("${widget.company.link}"),
leading: FaIcon(FontAwesomeIcons.link, color: COLOR_CLICK),
onTap: () {
company.openLink();
widget.company.openLink();
},
));
@ -150,11 +178,27 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
tiles.add(Divider());
}
if (company.isSupplier) {
// TODO - Add list of supplier parts
// TODO - Add list of purchase orders
if (widget.company.isSupplier) {
tiles.add(Divider());
if (supplierPartCount > 0) {
tiles.add(
ListTile(
title: Text(L10().supplierParts),
leading: FaIcon(FontAwesomeIcons.building, color: COLOR_CLICK),
trailing: Text(supplierPartCount.toString()),
onTap: () {
Navigator.push(
context,
MaterialPageRoute(
builder: (context) => SupplierPartList({
"supplier": widget.company.pk.toString()
})
)
);
}
)
);
}
tiles.add(
ListTile(
@ -167,7 +211,7 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
MaterialPageRoute(
builder: (context) => PurchaseOrderListWidget(
filters: {
"supplier": "${company.pk}"
"supplier": "${widget.company.pk}"
}
)
)
@ -188,18 +232,18 @@ class _CompanyDetailState extends RefreshableState<CompanyDetailWidget> {
*/
}
if (company.isManufacturer) {
if (widget.company.isManufacturer) {
// TODO - Add list of manufacturer parts
}
if (company.isCustomer) {
if (widget.company.isCustomer) {
// TODO - Add list of sales orders
tiles.add(Divider());
}
if (company.notes.isNotEmpty) {
if (widget.company.notes.isNotEmpty) {
tiles.add(ListTile(
title: Text(L10().notes),
leading: FaIcon(FontAwesomeIcons.stickyNote),