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:
@ -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),
|
||||
|
Reference in New Issue
Block a user