2
0
mirror of https://github.com/inventree/inventree-app.git synced 2026-04-25 19:03:25 +00:00
Files
inventree-app/lib/widget/build/build_item_detail.dart
T
Ben Hagen ea132599d8 Build Order (#673)
* WIP

* Remove debug msg

* Add required roles

* More roles

* Fix refresh for BuildDetail widget

* Add attachments widget

* Translated text

* Further updates

* More translations

* Form field updates

* Cleanup

* Code formatting

* Fix duplicate import

* formatting

* Remove duplicate switch case

* Update to match modern app

* Improved required parts list

* Filtering for build outputs

* Display list of allocated stock items

* Display source and destination locations

* Fix typo

* Add build orders to drawer

* Fix hard-coded string

* Set default filter value

* Tweak build fields (remove "notes")

* Fixes

* Add "start_date" to build edit form

* Disable editing of build line

* Tweak build item / build detail views

* Remove unused func

* Remove unused import

---------

Co-authored-by: Asterix\Oliver <oliver@currawongeng.com>
Co-authored-by: Oliver Walters <oliver.henry.walters@gmail.com>
2026-04-21 15:45:37 +10:00

219 lines
5.7 KiB
Dart

import "package:flutter/material.dart";
import "package:flutter_speed_dial/flutter_speed_dial.dart";
import "package:flutter_tabler_icons/flutter_tabler_icons.dart";
import "package:inventree/api.dart";
import "package:inventree/app_colors.dart";
import "package:inventree/l10.dart";
import "package:inventree/inventree/build.dart";
import "package:inventree/inventree/stock.dart";
import "package:inventree/widget/progress.dart";
import "package:inventree/widget/refreshable_state.dart";
import "package:inventree/widget/snacks.dart";
import "package:inventree/widget/dialogs.dart";
/*
* Widget for displaying detail view of a single BuildItem (stock allocation)
*/
class BuildItemDetailWidget extends StatefulWidget {
const BuildItemDetailWidget(this.item, {Key? key}) : super(key: key);
final InvenTreeBuildItem item;
@override
_BuildItemDetailWidgetState createState() => _BuildItemDetailWidgetState();
}
/*
* State for the BuildItemDetailWidget
*/
class _BuildItemDetailWidgetState
extends RefreshableState<BuildItemDetailWidget> {
_BuildItemDetailWidgetState();
@override
String getAppBarTitle() => L10().allocatedItem;
@override
List<Widget> appBarActions(BuildContext context) {
List<Widget> actions = [];
actions.add(
IconButton(
icon: const Icon(TablerIcons.edit),
tooltip: L10().allocationEdit,
onPressed: () {
_editAllocation(context);
},
),
);
return actions;
}
@override
List<SpeedDialChild> actionButtons(BuildContext context) {
List<SpeedDialChild> buttons = [];
// Unallocate button
buttons.add(
SpeedDialChild(
child: const Icon(TablerIcons.minus, color: Colors.red),
label: L10().unallocate,
onTap: () async {
_unallocateStock(context);
},
),
);
return buttons;
}
@override
Future<void> request(BuildContext context) async {
await widget.item.reload();
}
// Edit this allocation
Future<void> _editAllocation(BuildContext context) async {
var fields = widget.item.formFields();
fields["stock_item"]?["hidden"] = true;
widget.item.editForm(
context,
L10().allocationEdit,
fields: fields,
onSuccess: (data) async {
refresh(context);
showSnackIcon(L10().itemUpdated, success: true);
},
);
}
// Deallocate this stock item
Future<void> _unallocateStock(BuildContext context) async {
confirmationDialog(
L10().unallocateStock,
L10().unallocateStockConfirm,
icon: TablerIcons.minus,
color: Colors.red,
acceptText: L10().unallocate,
onAccept: () async {
widget.item.delete().then((result) {
if (result) {
showSnackIcon(L10().stockItemUpdated, success: true);
Navigator.pop(context);
} else {
showSnackIcon(L10().error);
}
});
},
);
}
// Go to stock item detail page
Future<void> _viewStockItem(BuildContext context) async {
if (widget.item.stockItem != null) {
widget.item.stockItem!.goToDetailPage(context);
}
}
@override
List<Widget> getTiles(BuildContext context) {
List<Widget> tiles = [];
// Stock item information
if (widget.item.stockItem != null) {
tiles.add(
ListTile(
title: Text(L10().stockItem),
subtitle: Text(widget.item.partName),
leading: widget.item.partThumbnail.isNotEmpty
? SizedBox(
width: 32,
height: 32,
child: InvenTreeAPI().getThumbnail(widget.item.partThumbnail),
)
: Icon(TablerIcons.box, color: COLOR_ACTION),
trailing: Icon(TablerIcons.chevron_right, color: COLOR_ACTION),
onTap: () {
_viewStockItem(context);
},
),
);
// Location information
tiles.add(
ListTile(
title: Text(L10().stockLocation),
subtitle: Text(
widget.item.locationName.isNotEmpty
? widget.item.locationPath
: L10().locationNotSet,
),
leading: const Icon(TablerIcons.map_pin),
),
);
// Serial number if available
if (widget.item.serialNumber.isNotEmpty) {
tiles.add(
ListTile(
title: Text(L10().serialNumber),
subtitle: Text(widget.item.serialNumber),
leading: const Icon(TablerIcons.hash),
),
);
}
// Batch code if available
if (widget.item.batchCode.isNotEmpty) {
tiles.add(
ListTile(
title: Text(L10().batchCode),
subtitle: Text(widget.item.batchCode),
leading: const Icon(TablerIcons.barcode),
),
);
}
}
// Quantity allocated
tiles.add(
ListTile(
title: Text(L10().quantity),
subtitle: Text(widget.item.quantity.toString()),
leading: const Icon(TablerIcons.list),
),
);
// Install into (if specified)
if (widget.item.installIntoId > 0) {
tiles.add(
ListTile(
title: Text(L10().buildOutput),
subtitle: Text(L10().viewDetails),
leading: const Icon(TablerIcons.arrow_right),
trailing: const Icon(TablerIcons.chevron_right),
onTap: () async {
showLoadingOverlay();
var stockItem = await InvenTreeStockItem().get(
widget.item.installIntoId,
);
hideLoadingOverlay();
if (stockItem is InvenTreeStockItem) {
stockItem.goToDetailPage(context);
}
},
),
);
}
return tiles;
}
}