mirror of
https://github.com/inventree/inventree-app.git
synced 2025-04-28 13:36:50 +00:00
Major overhaul of "paginated list" widget class
- Simplify implementation - Create mixin class for code reuse - Allow custom app-bar - Allow custom ordering / sorting options - Improve code commenting / readability
This commit is contained in:
parent
979f950129
commit
7301243ed6
@ -17,48 +17,6 @@ import "package:inventree/widget/refreshable_state.dart";
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Widget for displaying a list of BomItems for the specified 'parent' Part instance
|
|
||||||
*/
|
|
||||||
class BomList extends StatefulWidget {
|
|
||||||
|
|
||||||
const BomList(this.parent);
|
|
||||||
|
|
||||||
final InvenTreePart parent;
|
|
||||||
|
|
||||||
@override
|
|
||||||
_BomListState createState() => _BomListState(parent);
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class _BomListState extends PaginatedState<BomList> {
|
|
||||||
|
|
||||||
_BomListState(this.parent);
|
|
||||||
|
|
||||||
final InvenTreePart parent;
|
|
||||||
|
|
||||||
@override
|
|
||||||
String getAppBarTitle(BuildContext context) => L10().billOfMaterials;
|
|
||||||
|
|
||||||
@override
|
|
||||||
String get prefix => "bom_";
|
|
||||||
|
|
||||||
@override
|
|
||||||
Map<String, String> get orderingOptions => {
|
|
||||||
"quantity": L10().quantity,
|
|
||||||
"sub_part": L10().part,
|
|
||||||
};
|
|
||||||
|
|
||||||
@override
|
|
||||||
Widget getBody(BuildContext context) {
|
|
||||||
return PaginatedBomList({
|
|
||||||
"part": parent.pk.toString(),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Create a paginated widget displaying a list of BomItem objects
|
* Create a paginated widget displaying a list of BomItem objects
|
||||||
*/
|
*/
|
||||||
@ -78,13 +36,23 @@ class PaginatedBomList extends StatefulWidget {
|
|||||||
|
|
||||||
class _PaginatedBomListState extends PaginatedSearchState<PaginatedBomList> {
|
class _PaginatedBomListState extends PaginatedSearchState<PaginatedBomList> {
|
||||||
|
|
||||||
_PaginatedBomListState(Map<String, String> filters, this.onTotalChanged) : super(filters);
|
_PaginatedBomListState(Map<String, String> filters, this.onTotalChanged) : super(filters, fullscreen: true);
|
||||||
|
|
||||||
Function(int)? onTotalChanged;
|
Function(int)? onTotalChanged;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get prefix => "bom_";
|
String get prefix => "bom_";
|
||||||
|
|
||||||
|
@override
|
||||||
|
Map<String, String> get orderingOptions => {
|
||||||
|
"quantity": L10().quantity,
|
||||||
|
"sub_part": L10().part,
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@override
|
||||||
|
String getAppBarTitle(BuildContext context) => L10().billOfMaterials;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Future<InvenTreePageResponse?> requestPage(int limit, int offset, Map<String, String> params) async {
|
Future<InvenTreePageResponse?> requestPage(int limit, int offset, Map<String, String> params) async {
|
||||||
|
|
||||||
|
@ -24,7 +24,7 @@ class _NotificationState extends RefreshableState<NotificationWidget> {
|
|||||||
List<InvenTreeNotification> notifications = [];
|
List<InvenTreeNotification> notifications = [];
|
||||||
|
|
||||||
@override
|
@override
|
||||||
AppBar? buildAppBar(BuildContext context) {
|
AppBar? buildAppBar(BuildContext context, GlobalKey<ScaffoldState> key) {
|
||||||
// No app bar for the notification widget
|
// No app bar for the notification widget
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
@ -14,32 +14,32 @@ import "package:inventree/widget/refreshable_state.dart";
|
|||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Generic widget class for displaying a "paginated list".
|
* Generic stateful widget for displaying paginated data retrieved via the API
|
||||||
* Provides some basic functionality for adjusting ordering and filtering options
|
*
|
||||||
|
* - Can be displayed as "full screen" (with app-bar and drawer)
|
||||||
|
* - Can be displayed as a standalone widget
|
||||||
*/
|
*/
|
||||||
abstract class PaginatedState<T extends StatefulWidget> extends RefreshableState<T> {
|
class PaginatedSearchState<T extends StatefulWidget> extends State<T> with BaseWidgetProperties {
|
||||||
|
|
||||||
|
PaginatedSearchState(this.filters, {this.fullscreen = true});
|
||||||
|
|
||||||
|
final _key = GlobalKey<ScaffoldState>();
|
||||||
|
|
||||||
|
final Map<String, String> filters;
|
||||||
|
|
||||||
|
static const _pageSize = 25;
|
||||||
|
|
||||||
|
// Determine if this widget is shown "fullscreen" (i.e. with appbar)
|
||||||
|
final bool fullscreen;
|
||||||
|
|
||||||
// Prefix for storing and loading pagination options
|
// Prefix for storing and loading pagination options
|
||||||
|
// Override in implementing class
|
||||||
String get prefix => "prefix_";
|
String get prefix => "prefix_";
|
||||||
|
|
||||||
// Ordering options for this paginated state (override in implementing class)
|
// Return a map of sorting options available for this list
|
||||||
|
// Should be overridden by an implementing subclass
|
||||||
Map<String, String> get orderingOptions => {};
|
Map<String, String> get orderingOptions => {};
|
||||||
|
|
||||||
@override
|
|
||||||
List<Widget> getAppBarActions(BuildContext context) {
|
|
||||||
List<Widget> actions = [];
|
|
||||||
|
|
||||||
// If ordering options have been provided
|
|
||||||
if (orderingOptions.isNotEmpty) {
|
|
||||||
actions.add(IconButton(
|
|
||||||
icon: FaIcon(FontAwesomeIcons.sort),
|
|
||||||
onPressed: () => _updateFilters(context),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
return actions;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Return the selected ordering "field" for this list widget
|
// Return the selected ordering "field" for this list widget
|
||||||
Future<String> orderingField() async {
|
Future<String> orderingField() async {
|
||||||
dynamic field = await InvenTreeSettingsManager().getValue("${prefix}ordering_field", null);
|
dynamic field = await InvenTreeSettingsManager().getValue("${prefix}ordering_field", null);
|
||||||
@ -62,9 +62,21 @@ abstract class PaginatedState<T extends StatefulWidget> extends RefreshableState
|
|||||||
return order == "+" ? "+" : "-";
|
return order == "+" ? "+" : "-";
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update the (configurable) filters for this paginated list
|
// Return string for determining 'ordering' of paginated list
|
||||||
Future<void> _updateFilters(BuildContext context) async {
|
Future<String> get orderingString async {
|
||||||
|
dynamic field = await orderingField();
|
||||||
|
dynamic order = await orderingOrder();
|
||||||
|
|
||||||
|
// Return an empty string if no field is provided
|
||||||
|
if (field.toString().isEmpty) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
return "${order}${field}";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update the (configurable) filters for this paginated list
|
||||||
|
Future<void> _saveOrderingOptions(BuildContext context) async {
|
||||||
// Retrieve stored setting
|
// Retrieve stored setting
|
||||||
dynamic _field = await orderingField();
|
dynamic _field = await orderingField();
|
||||||
dynamic _order = await orderingOrder();
|
dynamic _order = await orderingOrder();
|
||||||
@ -96,12 +108,12 @@ abstract class PaginatedState<T extends StatefulWidget> extends RefreshableState
|
|||||||
"value": _order,
|
"value": _order,
|
||||||
"choices": [
|
"choices": [
|
||||||
{
|
{
|
||||||
"value": "+",
|
"value": "+",
|
||||||
"display_name": "Ascending",
|
"display_name": "Ascending",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"value": "-",
|
"value": "-",
|
||||||
"display_name": "Descending",
|
"display_name": "Descending",
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
@ -123,27 +135,12 @@ abstract class PaginatedState<T extends StatefulWidget> extends RefreshableState
|
|||||||
await InvenTreeSettingsManager().setValue("${prefix}ordering_field", f);
|
await InvenTreeSettingsManager().setValue("${prefix}ordering_field", f);
|
||||||
await InvenTreeSettingsManager().setValue("${prefix}ordering_order", o);
|
await InvenTreeSettingsManager().setValue("${prefix}ordering_order", o);
|
||||||
|
|
||||||
// Refresh the widget
|
// Refresh data from the server
|
||||||
setState(() {});
|
_pagingController.refresh();
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class PaginatedSearchState<T extends StatefulWidget> extends State<T> {
|
|
||||||
|
|
||||||
PaginatedSearchState(this.filters);
|
|
||||||
|
|
||||||
final Map<String, String> filters;
|
|
||||||
|
|
||||||
static const _pageSize = 25;
|
|
||||||
|
|
||||||
// Prefix for storing and loading pagination options
|
|
||||||
// Override in implementing class
|
|
||||||
String get prefix => "prefix_";
|
|
||||||
|
|
||||||
// Search query term
|
// Search query term
|
||||||
String searchTerm = "";
|
String searchTerm = "";
|
||||||
|
|
||||||
@ -176,19 +173,12 @@ class PaginatedSearchState<T extends StatefulWidget> extends State<T> {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<String> get ordering async {
|
|
||||||
dynamic field = await InvenTreeSettingsManager().getValue("${prefix}ordering_field", "");
|
|
||||||
dynamic order = await InvenTreeSettingsManager().getValue("${prefix}ordering_order", "+");
|
|
||||||
|
|
||||||
return "${order}${field}";
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<void> _fetchPage(int pageKey) async {
|
Future<void> _fetchPage(int pageKey) async {
|
||||||
try {
|
try {
|
||||||
Map<String, String> params = filters;
|
Map<String, String> params = filters;
|
||||||
|
|
||||||
params["search"] = "${searchTerm}";
|
params["search"] = "${searchTerm}";
|
||||||
params["ordering"] = await ordering;
|
params["ordering"] = await orderingString;
|
||||||
|
|
||||||
final page = await requestPage(
|
final page = await requestPage(
|
||||||
_pageSize,
|
_pageSize,
|
||||||
@ -234,6 +224,8 @@ class PaginatedSearchState<T extends StatefulWidget> extends State<T> {
|
|||||||
_pagingController.refresh();
|
_pagingController.refresh();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Function to construct a single paginated item
|
||||||
|
// Must be overridden in an implementing subclass
|
||||||
Widget buildItem(BuildContext context, InvenTreeModel item) {
|
Widget buildItem(BuildContext context, InvenTreeModel item) {
|
||||||
|
|
||||||
// This method must be overridden by the child class
|
// This method must be overridden by the child class
|
||||||
@ -243,10 +235,31 @@ class PaginatedSearchState<T extends StatefulWidget> extends State<T> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Return a string which is displayed when there are no results
|
||||||
|
// Can be overridden by an implementing subclass
|
||||||
String get noResultsText => L10().noResults;
|
String get noResultsText => L10().noResults;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build (BuildContext context) {
|
Widget build (BuildContext context) {
|
||||||
|
|
||||||
|
if (fullscreen) {
|
||||||
|
return Scaffold(
|
||||||
|
key: _key,
|
||||||
|
appBar: buildAppBar(context, _key),
|
||||||
|
drawer: getDrawer(context),
|
||||||
|
body: Builder(
|
||||||
|
builder: (BuildContext ctx) {
|
||||||
|
return getBody(ctx);
|
||||||
|
}
|
||||||
|
)
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return getBody(context);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget getBody(BuildContext context) {
|
||||||
return Column(
|
return Column(
|
||||||
mainAxisAlignment: MainAxisAlignment.start,
|
mainAxisAlignment: MainAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
@ -277,6 +290,21 @@ class PaginatedSearchState<T extends StatefulWidget> extends State<T> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
List<Widget> getAppBarActions(BuildContext context) {
|
||||||
|
List<Widget> actions = [];
|
||||||
|
|
||||||
|
// If ordering options have been provided
|
||||||
|
if (orderingOptions.isNotEmpty) {
|
||||||
|
actions.add(IconButton(
|
||||||
|
icon: FaIcon(FontAwesomeIcons.sort),
|
||||||
|
onPressed: () => _saveOrderingOptions(context),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
return actions;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -354,7 +354,9 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
|
|||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
MaterialPageRoute(
|
MaterialPageRoute(
|
||||||
builder: (context) => BomList(part)
|
builder: (context) => PaginatedBomList({
|
||||||
|
"part": part.pk.toString(),
|
||||||
|
})
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -3,7 +3,53 @@ import "package:inventree/widget/drawer.dart";
|
|||||||
import "package:flutter/material.dart";
|
import "package:flutter/material.dart";
|
||||||
|
|
||||||
|
|
||||||
abstract class RefreshableState<T extends StatefulWidget> extends State<T> {
|
/*
|
||||||
|
* Simple mixin class which defines simple methods for defining widget properties
|
||||||
|
*/
|
||||||
|
mixin BaseWidgetProperties {
|
||||||
|
|
||||||
|
// Return a list of appBar actions (default = None)
|
||||||
|
List<Widget> getAppBarActions(BuildContext context) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return a title for the appBar
|
||||||
|
String getAppBarTitle(BuildContext context) { return "--- app bar ---"; }
|
||||||
|
|
||||||
|
// Function to construct a drawer (override if needed)
|
||||||
|
Widget getDrawer(BuildContext context) {
|
||||||
|
return InvenTreeDrawer(context);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Function to construct a body (MUST BE PROVIDED)
|
||||||
|
Widget getBody(BuildContext context) {
|
||||||
|
|
||||||
|
// Default return is an empty ListView
|
||||||
|
return ListView();
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget? getBottomNavBar(BuildContext context) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
AppBar? buildAppBar(BuildContext context, GlobalKey<ScaffoldState> key) {
|
||||||
|
return AppBar(
|
||||||
|
title: Text(getAppBarTitle(context)),
|
||||||
|
actions: getAppBarActions(context),
|
||||||
|
leading: backButton(context, key),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Abstract base class which provides generic "refresh" functionality.
|
||||||
|
*
|
||||||
|
* - Drag down and release to 'refresh' the widget
|
||||||
|
* - Define some method which runs to 'refresh' the widget state
|
||||||
|
*/
|
||||||
|
abstract class RefreshableState<T extends StatefulWidget> extends State<T> with BaseWidgetProperties {
|
||||||
|
|
||||||
final refreshableKey = GlobalKey<ScaffoldState>();
|
final refreshableKey = GlobalKey<ScaffoldState>();
|
||||||
|
|
||||||
@ -25,12 +71,6 @@ abstract class RefreshableState<T extends StatefulWidget> extends State<T> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
List<Widget> getAppBarActions(BuildContext context) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
String getAppBarTitle(BuildContext context) { return "App Bar Title"; }
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
@ -60,34 +100,6 @@ abstract class RefreshableState<T extends StatefulWidget> extends State<T> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Function to construct a drawer (override if needed)
|
|
||||||
Widget getDrawer(BuildContext context) {
|
|
||||||
return InvenTreeDrawer(context);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Function to construct a body (MUST BE PROVIDED)
|
|
||||||
Widget getBody(BuildContext context) {
|
|
||||||
|
|
||||||
// Default return is an empty ListView
|
|
||||||
return ListView();
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget? getBottomNavBar(BuildContext context) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget? getFab(BuildContext context) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
AppBar? buildAppBar(BuildContext context) {
|
|
||||||
return AppBar(
|
|
||||||
title: Text(getAppBarTitle(context)),
|
|
||||||
actions: getAppBarActions(context),
|
|
||||||
leading: backButton(context, refreshableKey),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
|
|
||||||
@ -96,9 +108,8 @@ abstract class RefreshableState<T extends StatefulWidget> extends State<T> {
|
|||||||
|
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
key: refreshableKey,
|
key: refreshableKey,
|
||||||
appBar: buildAppBar(context),
|
appBar: buildAppBar(context, refreshableKey),
|
||||||
drawer: getDrawer(context),
|
drawer: getDrawer(context),
|
||||||
floatingActionButton: getFab(context),
|
|
||||||
body: Builder(
|
body: Builder(
|
||||||
builder: (BuildContext context) {
|
builder: (BuildContext context) {
|
||||||
return RefreshIndicator(
|
return RefreshIndicator(
|
||||||
|
@ -53,9 +53,9 @@ class _SearchDisplayState extends RefreshableState<SearchWidget> {
|
|||||||
String getAppBarTitle(BuildContext context) => L10().search;
|
String getAppBarTitle(BuildContext context) => L10().search;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
AppBar? buildAppBar(BuildContext context) {
|
AppBar? buildAppBar(BuildContext context, GlobalKey<ScaffoldState> key) {
|
||||||
if (hasAppBar) {
|
if (hasAppBar) {
|
||||||
return super.buildAppBar(context);
|
return super.buildAppBar(context, key);
|
||||||
} else {
|
} else {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user