mirror of
https://github.com/inventree/inventree-app.git
synced 2025-04-27 21:16:48 +00:00
* Add new models for SalesOrder - Create generic Order and OrderLine models with common functionality * Refactor - Move some widgets around - Cleanup directory structure * Add link to home screen and nav drawer * Add SalesOrder list widget * Linting fixes * Fix string * Refactor PurchaseOrderDetailWidget * Tweaks to existing code * linting * Fixes for drawer widget * Add "detail" page for SalesOrder * Add more tiles to SalesOrder detail * Allow editing of salesorder * add list filters for sales orders * Display list of line items * Customer updates - Display customer icon on home screen - Fetch sales orders for customer detail page * Cleanup company detail view * Create new sales order from list * Stricter typing for formFields method * Create new PurchaseOrder and SalesOrder from company deatil * Status code updates - Add function for name comparison - Remove hard-coded values * Update view permission checks for home widget * Add ability to manually add SalesOrderLineItem * Add nice progress bar widgets * Display detail view for sales order line item * edit SalesOrderLineItem * Fix unused import * Hide "shipped items" tab - Will be added in a future update
233 lines
6.1 KiB
Dart
233 lines
6.1 KiB
Dart
import "package:flutter/material.dart";
|
|
import "package:flutter_speed_dial/flutter_speed_dial.dart";
|
|
import "package:font_awesome_flutter/font_awesome_flutter.dart";
|
|
|
|
import "package:inventree/api.dart";
|
|
import "package:inventree/app_colors.dart";
|
|
import "package:inventree/l10.dart";
|
|
|
|
import "package:inventree/inventree/part.dart";
|
|
import "package:inventree/inventree/stock.dart";
|
|
import "package:inventree/inventree/model.dart";
|
|
|
|
import "package:inventree/widget/progress.dart";
|
|
import "package:inventree/widget/refreshable_state.dart";
|
|
|
|
|
|
class StockItemTestResultsWidget extends StatefulWidget {
|
|
|
|
const StockItemTestResultsWidget(this.item, {Key? key}) : super(key: key);
|
|
|
|
final InvenTreeStockItem item;
|
|
|
|
@override
|
|
_StockItemTestResultDisplayState createState() => _StockItemTestResultDisplayState(item);
|
|
}
|
|
|
|
|
|
class _StockItemTestResultDisplayState extends RefreshableState<StockItemTestResultsWidget> {
|
|
|
|
_StockItemTestResultDisplayState(this.item);
|
|
|
|
@override
|
|
String getAppBarTitle() => L10().testResults;
|
|
|
|
@override
|
|
List<Widget> appBarActions(BuildContext context) => [];
|
|
|
|
@override
|
|
List<SpeedDialChild> actionButtons(BuildContext context) {
|
|
List<SpeedDialChild> actions = [];
|
|
|
|
if (InvenTreeStockItemTestResult().canCreate) {
|
|
actions.add(
|
|
SpeedDialChild(
|
|
child: FaIcon(FontAwesomeIcons.circlePlus),
|
|
label: L10().testResultAdd,
|
|
onTap: () {
|
|
addTestResult(context);
|
|
}
|
|
)
|
|
);
|
|
}
|
|
|
|
return actions;
|
|
}
|
|
|
|
@override
|
|
Future<void> request(BuildContext context) async {
|
|
await item.getTestTemplates();
|
|
await item.getTestResults();
|
|
}
|
|
|
|
final InvenTreeStockItem item;
|
|
|
|
Future <void> addTestResult(BuildContext context, {String name = "", bool nameIsEditable = true, bool result = false, String value = "", bool valueRequired = false, bool attachmentRequired = false}) async {
|
|
|
|
InvenTreeStockItemTestResult().createForm(
|
|
context,
|
|
L10().testResultAdd,
|
|
data: {
|
|
"stock_item": "${item.pk}",
|
|
"test": "${name}",
|
|
},
|
|
onSuccess: (data) {
|
|
refresh(context);
|
|
},
|
|
fileField: "attachment",
|
|
);
|
|
}
|
|
|
|
// Squish together templates and results
|
|
List<InvenTreeModel> getTestResults() {
|
|
var templates = item.testTemplates;
|
|
var results = item.testResults;
|
|
|
|
List<InvenTreeModel> outputs = [];
|
|
|
|
// Add each template to the list
|
|
for (var t in templates) {
|
|
outputs.add(t);
|
|
}
|
|
|
|
// Add each result (compare to existing items / templates
|
|
for (var result in results) {
|
|
bool match = false;
|
|
|
|
for (var ii = 0; ii < outputs.length; ii++) {
|
|
|
|
// Check against templates
|
|
if (outputs[ii] is InvenTreePartTestTemplate) {
|
|
var t = outputs[ii] as InvenTreePartTestTemplate;
|
|
|
|
if (result.key == t.key) {
|
|
t.results.add(result);
|
|
match = true;
|
|
break;
|
|
}
|
|
} else if (outputs[ii] is InvenTreeStockItemTestResult) {
|
|
var r = outputs[ii] as InvenTreeStockItemTestResult;
|
|
|
|
if (r.key == result.key) {
|
|
// Overwrite with a newer result
|
|
outputs[ii] = result;
|
|
match = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!match) {
|
|
outputs.add(result);
|
|
}
|
|
}
|
|
|
|
return outputs;
|
|
}
|
|
|
|
@override
|
|
List<Widget> getTiles(BuildContext context) {
|
|
List<Widget> tiles = [];
|
|
|
|
tiles.add(
|
|
Card(
|
|
child: ListTile(
|
|
title: Text(item.partName),
|
|
subtitle: Text(item.partDescription),
|
|
leading: InvenTreeAPI().getThumbnail(item.partImage),
|
|
)
|
|
)
|
|
);
|
|
|
|
tiles.add(
|
|
ListTile(
|
|
title: Text(L10().testResults,
|
|
style: TextStyle(fontWeight: FontWeight.bold)
|
|
)
|
|
)
|
|
);
|
|
|
|
if (loading) {
|
|
tiles.add(progressIndicator());
|
|
return tiles;
|
|
}
|
|
|
|
var results = getTestResults();
|
|
|
|
if (results.isEmpty) {
|
|
tiles.add(ListTile(
|
|
title: Text(L10().testResultNone),
|
|
subtitle: Text(L10().testResultNoneDetail),
|
|
));
|
|
|
|
return tiles;
|
|
}
|
|
|
|
for (var item in results) {
|
|
|
|
bool _hasResult = false;
|
|
bool _required = false;
|
|
String _test = "";
|
|
bool _result = false;
|
|
String _value = "";
|
|
String _notes = "";
|
|
|
|
FaIcon _icon = FaIcon(FontAwesomeIcons.circleQuestion, color: Colors.lightBlue);
|
|
bool _valueRequired = false;
|
|
bool _attachmentRequired = false;
|
|
|
|
if (item is InvenTreePartTestTemplate) {
|
|
_result = item.passFailStatus();
|
|
_test = item.testName;
|
|
_required = item.required;
|
|
_value = item.latestResult()?.value ?? L10().noResults;
|
|
_valueRequired = item.requiresValue;
|
|
_attachmentRequired = item.requiresAttachment;
|
|
_notes = item.latestResult()?.notes ?? item.description;
|
|
_hasResult = item.latestResult() != null;
|
|
} else if (item is InvenTreeStockItemTestResult) {
|
|
_result = item.result;
|
|
_test = item.testName;
|
|
_required = false;
|
|
_value = item.value;
|
|
_notes = item.notes;
|
|
_hasResult = true;
|
|
}
|
|
|
|
if (!_hasResult) {
|
|
_icon = FaIcon(FontAwesomeIcons.circleQuestion, color: Colors.blue);
|
|
} else if (_result == true) {
|
|
_icon = FaIcon(FontAwesomeIcons.circleCheck, color: COLOR_SUCCESS);
|
|
} else if (_result == false) {
|
|
_icon = FaIcon(FontAwesomeIcons.circleXmark, color: COLOR_DANGER);
|
|
}
|
|
|
|
tiles.add(ListTile(
|
|
title: Text(_test, style: TextStyle(
|
|
fontWeight: _required ? FontWeight.bold : FontWeight.normal,
|
|
fontStyle: _hasResult ? FontStyle.normal : FontStyle.italic
|
|
)),
|
|
subtitle: Text(_notes),
|
|
trailing: Text(_value),
|
|
leading: _icon,
|
|
onTap: () {
|
|
addTestResult(
|
|
context,
|
|
name: _test,
|
|
nameIsEditable: !_required,
|
|
valueRequired: _valueRequired,
|
|
attachmentRequired: _attachmentRequired
|
|
);
|
|
}
|
|
));
|
|
}
|
|
|
|
if (tiles.isEmpty) {
|
|
tiles.add(ListTile(
|
|
title: Text(L10().testResultNone),
|
|
));
|
|
}
|
|
|
|
return tiles;
|
|
}
|
|
} |