mirror of
https://github.com/inventree/inventree-app.git
synced 2025-04-28 05:26:47 +00:00
300 lines
7.5 KiB
Dart
300 lines
7.5 KiB
Dart
import 'package:InvenTree/inventree/part.dart';
|
|
import 'package:InvenTree/inventree/stock.dart';
|
|
import 'package:InvenTree/inventree/model.dart';
|
|
import 'package:InvenTree/api.dart';
|
|
import 'package:InvenTree/widget/dialogs.dart';
|
|
import 'package:InvenTree/widget/fields.dart';
|
|
import 'package:InvenTree/widget/progress.dart';
|
|
import 'package:InvenTree/widget/snacks.dart';
|
|
|
|
import 'package:InvenTree/l10.dart';
|
|
|
|
import 'dart:io';
|
|
|
|
import 'package:flutter/cupertino.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:InvenTree/widget/refreshable_state.dart';
|
|
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
|
|
|
|
|
|
class StockItemTestResultsWidget extends StatefulWidget {
|
|
|
|
StockItemTestResultsWidget(this.item, {Key key}) : super(key: key);
|
|
|
|
final InvenTreeStockItem item;
|
|
|
|
@override
|
|
_StockItemTestResultDisplayState createState() => _StockItemTestResultDisplayState(item);
|
|
}
|
|
|
|
|
|
class _StockItemTestResultDisplayState extends RefreshableState<StockItemTestResultsWidget> {
|
|
|
|
final _addResultKey = GlobalKey<FormState>();
|
|
|
|
@override
|
|
String getAppBarTitle(BuildContext context) => L10().testResults;
|
|
|
|
@override
|
|
Future<void> request(BuildContext context) async {
|
|
await item.getTestTemplates(context);
|
|
await item.getTestResults(context);
|
|
}
|
|
|
|
final InvenTreeStockItem item;
|
|
|
|
_StockItemTestResultDisplayState(this.item);
|
|
|
|
void uploadTestResult(String name, bool result, String value, String notes, File attachment) async {
|
|
|
|
final success = await item.uploadTestResult(
|
|
context, name, result,
|
|
value: value,
|
|
notes: notes,
|
|
attachment: attachment
|
|
);
|
|
|
|
showSnackIcon(
|
|
success ? "Test result uploaded" : "Could not upload test result",
|
|
success: success
|
|
);
|
|
|
|
refresh();
|
|
}
|
|
|
|
void addTestResult({String name = '', bool nameIsEditable = true, bool result = false, String value = '', bool valueRequired = false, bool attachmentRequired = false}) async {
|
|
|
|
String _name;
|
|
bool _result;
|
|
String _value;
|
|
String _notes;
|
|
File _attachment;
|
|
|
|
showFormDialog("Add Test Data",
|
|
key: _addResultKey,
|
|
callback: () {
|
|
uploadTestResult(_name, _result, _value, _notes, _attachment);
|
|
},
|
|
fields: <Widget>[
|
|
StringField(
|
|
label: L10().testName,
|
|
initial: name,
|
|
isEnabled: nameIsEditable,
|
|
onSaved: (value) => _name = value,
|
|
),
|
|
CheckBoxField(
|
|
label: L10().result,
|
|
hint: L10().testPassedOrFailed,
|
|
initial: true,
|
|
onSaved: (value) => _result = value,
|
|
),
|
|
StringField(
|
|
label: L10().value,
|
|
initial: value,
|
|
allowEmpty: true,
|
|
onSaved: (value) => _value = value,
|
|
validator: (String value) {
|
|
if (valueRequired && (value == null || value.isEmpty)) {
|
|
return L10().valueRequired;
|
|
}
|
|
return null;
|
|
},
|
|
),
|
|
ImagePickerField(
|
|
context,
|
|
label: L10().attachImage,
|
|
required: attachmentRequired,
|
|
onSaved: (attachment) => _attachment = attachment,
|
|
),
|
|
StringField(
|
|
allowEmpty: true,
|
|
label: L10().notes,
|
|
onSaved: (value) => _notes = value,
|
|
),
|
|
]
|
|
);
|
|
}
|
|
|
|
// 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;
|
|
}
|
|
|
|
List<Widget> resultsList() {
|
|
List<Widget> tiles = [];
|
|
|
|
tiles.add(
|
|
Card(
|
|
child: ListTile(
|
|
title: Text(item.partName),
|
|
subtitle: Text(item.partDescription),
|
|
leading: InvenTreeAPI().getImage(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.length == 0) {
|
|
tiles.add(ListTile(
|
|
title: Text(L10().testResultNone),
|
|
subtitle: Text(L10().testResultNoneDetail),
|
|
));
|
|
|
|
return tiles;
|
|
}
|
|
|
|
for (var item in results) {
|
|
|
|
bool _required = false;
|
|
String _test;
|
|
bool _result = null;
|
|
String _value;
|
|
String _notes;
|
|
FaIcon _icon = FaIcon(FontAwesomeIcons.questionCircle, color: Color.fromRGBO(0, 0, 250, 1));
|
|
bool _valueRequired = false;
|
|
bool _attachmentRequired = false;
|
|
|
|
if (item is InvenTreePartTestTemplate) {
|
|
_result = item.passFailStatus();
|
|
_test = item.testName;
|
|
_required = item.required;
|
|
_value = item.latestResult()?.value ?? '';
|
|
_notes = item.latestResult()?.notes ?? '';
|
|
_valueRequired = item.requiresValue;
|
|
_attachmentRequired = item.requiresAttachment;
|
|
} else if (item is InvenTreeStockItemTestResult) {
|
|
_result = item.result;
|
|
_test = item.testName;
|
|
_required = false;
|
|
_value = item.value;
|
|
_notes = item.notes;
|
|
}
|
|
|
|
if (_result == true) {
|
|
_icon = FaIcon(FontAwesomeIcons.checkCircle,
|
|
color: Color.fromRGBO(0, 250, 0, 0.8)
|
|
);
|
|
} else if (_result == false) {
|
|
_icon = FaIcon(FontAwesomeIcons.timesCircle,
|
|
color: Color.fromRGBO(250, 0, 0, 0.8)
|
|
);
|
|
}
|
|
|
|
tiles.add(ListTile(
|
|
title: Text(_test, style: TextStyle(fontWeight: _required ? FontWeight.bold : FontWeight.normal)),
|
|
subtitle: Text(_value),
|
|
trailing: _icon,
|
|
onLongPress: () {
|
|
addTestResult(
|
|
name: _test,
|
|
nameIsEditable: !_required,
|
|
valueRequired: _valueRequired,
|
|
attachmentRequired: _attachmentRequired
|
|
);
|
|
}
|
|
));
|
|
}
|
|
|
|
if (tiles.isEmpty) {
|
|
tiles.add(ListTile(
|
|
title: Text(L10().testResultNone),
|
|
));
|
|
}
|
|
|
|
return tiles;
|
|
}
|
|
|
|
@override
|
|
Widget getBody(BuildContext context) {
|
|
|
|
return ListView(
|
|
children: ListTile.divideTiles(
|
|
context: context,
|
|
tiles: resultsList()
|
|
).toList()
|
|
);
|
|
}
|
|
|
|
/*
|
|
List<SpeedDialChild> actionButtons() {
|
|
|
|
var buttons = List<SpeedDialChild>();
|
|
|
|
buttons.add(SpeedDialChild(
|
|
child: Icon(FontAwesomeIcons.plusCircle),
|
|
label: L10().testResultAdd,
|
|
onTap: () {
|
|
addTestResult();
|
|
},
|
|
));
|
|
|
|
return buttons;
|
|
}
|
|
*/
|
|
|
|
@override
|
|
Widget getFab(BuildContext context) {
|
|
return FloatingActionButton(
|
|
child: Icon(FontAwesomeIcons.plus),
|
|
onPressed: () {
|
|
addTestResult();
|
|
},
|
|
);
|
|
}
|
|
} |