2
0
mirror of https://github.com/inventree/inventree-app.git synced 2025-04-30 22:46:49 +00:00

Refactoring of paginated views

This commit is contained in:
Oliver Walters 2021-03-03 23:58:59 +11:00
parent 989e0e81b3
commit c1312e4e5d
4 changed files with 200 additions and 90 deletions

View File

@ -13,6 +13,7 @@ import 'package:InvenTree/widget/snacks.dart';
import 'package:InvenTree/widget/part_detail.dart'; import 'package:InvenTree/widget/part_detail.dart';
import 'package:InvenTree/widget/drawer.dart'; import 'package:InvenTree/widget/drawer.dart';
import 'package:InvenTree/widget/refreshable_state.dart'; import 'package:InvenTree/widget/refreshable_state.dart';
import 'package:InvenTree/widget/paginator.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/foundation.dart'; import 'package:flutter/foundation.dart';
@ -155,7 +156,7 @@ class _CategoryDisplayState extends RefreshableState<CategoryDisplayWidget> {
}); });
} }
Widget getCategoryDescriptionCard() { Widget getCategoryDescriptionCard({bool extra = true}) {
if (category == null) { if (category == null) {
return Card( return Card(
child: ListTile( child: ListTile(
@ -163,16 +164,18 @@ class _CategoryDisplayState extends RefreshableState<CategoryDisplayWidget> {
) )
); );
} else { } else {
return Card(
child: Column( List<Widget> children = [
children: <Widget>[ ListTile(
ListTile( title: Text("${category.name}",
title: Text("${category.name}", style: TextStyle(fontWeight: FontWeight.bold)
style: TextStyle(fontWeight: FontWeight.bold) ),
), subtitle: Text("${category.description}"),
subtitle: Text("${category.description}"), ),
), ];
Divider(),
if (extra) {
children.add(
ListTile( ListTile(
title: Text(I18N.of(context).parentCategory), title: Text(I18N.of(context).parentCategory),
subtitle: Text("${category.parentpathstring}"), subtitle: Text("${category.parentpathstring}"),
@ -190,7 +193,12 @@ class _CategoryDisplayState extends RefreshableState<CategoryDisplayWidget> {
} }
}, },
) )
] );
}
return Card(
child: Column(
children: children
), ),
); );
} }
@ -250,7 +258,7 @@ class _CategoryDisplayState extends RefreshableState<CategoryDisplayWidget> {
List<Widget> actionTiles() { List<Widget> actionTiles() {
List<Widget> tiles = [ List<Widget> tiles = [
getCategoryDescriptionCard(), getCategoryDescriptionCard(extra: false),
ListTile( ListTile(
title: Text(I18N.of(context).actions, title: Text(I18N.of(context).actions,
style: TextStyle(fontWeight: FontWeight.bold) style: TextStyle(fontWeight: FontWeight.bold)
@ -263,6 +271,8 @@ class _CategoryDisplayState extends RefreshableState<CategoryDisplayWidget> {
return tiles; return tiles;
} }
int partCount = 0;
@override @override
Widget getBody(BuildContext context) { Widget getBody(BuildContext context) {
@ -272,7 +282,33 @@ class _CategoryDisplayState extends RefreshableState<CategoryDisplayWidget> {
children: detailTiles() children: detailTiles()
); );
case 1: case 1:
return PaginatedPartList({"category": "${category?.pk ?? null}"}); return Column(
mainAxisAlignment: MainAxisAlignment.start,
children: [
getCategoryDescriptionCard(extra: false),
ListTile(
title: Text(
I18N.of(context).parts,
style: TextStyle(fontWeight: FontWeight.bold),
),
trailing: Text(
"${partCount}",
style: TextStyle(fontWeight: FontWeight.bold),
),
),
Divider(height: 3),
Expanded(
child: PaginatedPartList(
{"category": "${category?.pk ?? null}"},
onTotalChanged: (int total) {
setState(() {
partCount = total;
});
},
)
)
],
);
case 2: case 2:
return ListView( return ListView(
children: actionTiles() children: actionTiles()
@ -335,10 +371,12 @@ class PaginatedPartList extends StatefulWidget {
final Map<String, String> filters; final Map<String, String> filters;
PaginatedPartList(this.filters); Function onTotalChanged;
PaginatedPartList(this.filters, {this.onTotalChanged});
@override @override
_PaginatedPartListState createState() => _PaginatedPartListState(filters); _PaginatedPartListState createState() => _PaginatedPartListState(filters, onTotalChanged);
} }
@ -348,9 +386,11 @@ class _PaginatedPartListState extends State<PaginatedPartList> {
String _searchTerm; String _searchTerm;
Function onTotalChanged;
final Map<String, String> filters; final Map<String, String> filters;
_PaginatedPartListState(this.filters); _PaginatedPartListState(this.filters, this.onTotalChanged);
final PagingController<int, InvenTreePart> _pagingController = PagingController(firstPageKey: 0); final PagingController<int, InvenTreePart> _pagingController = PagingController(firstPageKey: 0);
@ -397,6 +437,10 @@ class _PaginatedPartListState extends State<PaginatedPartList> {
_pagingController.appendPage(parts, nextPageKey); _pagingController.appendPage(parts, nextPageKey);
} }
if (onTotalChanged != null) {
onTotalChanged(page.count);
}
} catch (error) { } catch (error) {
print("Error! - ${error.toString()}"); print("Error! - ${error.toString()}");
_pagingController.error = error; _pagingController.error = error;
@ -437,34 +481,25 @@ class _PaginatedPartListState extends State<PaginatedPartList> {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return CustomScrollView( return CustomScrollView(
shrinkWrap: true,
physics: ClampingScrollPhysics(),
scrollDirection: Axis.vertical,
slivers: <Widget>[ slivers: <Widget>[
// TODO: Introduce searching within the list // TODO: Introduce searching within the list
/* //PaginatedSearch(callback: updateSearchTerm),
SliverToBoxAdapter(child: TextField(
onChanged: updateSearchTerm,
)
),
*/
PagedSliverList.separated( PagedSliverList.separated(
pagingController: _pagingController, pagingController: _pagingController,
builderDelegate: PagedChildBuilderDelegate<InvenTreePart>( builderDelegate: PagedChildBuilderDelegate<InvenTreePart>(
itemBuilder: (context, item, index) { itemBuilder: (context, item, index) {
return _buildPart(context, item); return _buildPart(context, item);
} },
noItemsFoundIndicatorBuilder: (context) {
return NoResultsWidget("No parts found");
}
), ),
separatorBuilder: (context, index) => const Divider(height: 1), separatorBuilder: (context, index) => const Divider(height: 1),
), ),
] ]
); );
return PagedListView<int, InvenTreePart>.separated(
pagingController: _pagingController,
builderDelegate: PagedChildBuilderDelegate<InvenTreePart>(
itemBuilder: (context, item, index) {
return _buildPart(context, item);
}
),
separatorBuilder: (context, index) => const Divider(height: 1),
);
} }
} }

View File

@ -10,6 +10,7 @@ import 'package:InvenTree/widget/dialogs.dart';
import 'package:InvenTree/widget/search.dart'; import 'package:InvenTree/widget/search.dart';
import 'package:InvenTree/widget/snacks.dart'; import 'package:InvenTree/widget/snacks.dart';
import 'package:InvenTree/widget/stock_detail.dart'; import 'package:InvenTree/widget/stock_detail.dart';
import 'package:InvenTree/widget/paginator.dart';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
@ -131,8 +132,6 @@ class _LocationDisplayState extends RefreshableState<LocationDisplayWidget> {
} }
} }
List<InvenTreeStockItem> _items = List<InvenTreeStockItem>();
@override @override
Future<void> onBuild(BuildContext context) async { Future<void> onBuild(BuildContext context) async {
refresh(); refresh();
@ -162,7 +161,7 @@ class _LocationDisplayState extends RefreshableState<LocationDisplayWidget> {
setState(() {}); setState(() {});
} }
Widget locationDescriptionCard() { Widget locationDescriptionCard({bool includeActions = true}) {
if (location == null) { if (location == null) {
return Card( return Card(
child: ListTile( child: ListTile(
@ -170,13 +169,16 @@ class _LocationDisplayState extends RefreshableState<LocationDisplayWidget> {
) )
); );
} else { } else {
return Card(
child: Column( List<Widget> children = [
children: <Widget> [ ListTile(
ListTile( title: Text("${location.name}"),
title: Text("${location.name}"), subtitle: Text("${location.description}"),
subtitle: Text("${location.description}"), ),
), ];
if (includeActions) {
children.add(
ListTile( ListTile(
title: Text("Parent Category"), title: Text("Parent Category"),
subtitle: Text("${location.parentpathstring}"), subtitle: Text("${location.parentpathstring}"),
@ -193,7 +195,12 @@ class _LocationDisplayState extends RefreshableState<LocationDisplayWidget> {
} }
}, },
) )
] );
}
return Card(
child: Column(
children: children,
) )
); );
} }
@ -222,6 +229,8 @@ class _LocationDisplayState extends RefreshableState<LocationDisplayWidget> {
); );
} }
int stockItemCount = 0;
Widget getSelectedWidget(int index) { Widget getSelectedWidget(int index) {
switch (index) { switch (index) {
case 0: case 0:
@ -229,7 +238,30 @@ class _LocationDisplayState extends RefreshableState<LocationDisplayWidget> {
children: detailTiles(), children: detailTiles(),
); );
case 1: case 1:
return PaginatedStockList({"location": "${location?.pk ?? -1}"}); return Column(
mainAxisAlignment: MainAxisAlignment.start,
children: [
locationDescriptionCard(includeActions: false),
ListTile(
title: Text(
I18N.of(context).stockItems,
style: TextStyle(fontWeight: FontWeight.bold),
),
trailing: Text("${stockItemCount}")
),
Divider(height: 3),
Expanded(
child: PaginatedStockList(
{"location": "${location?.pk ?? -1}"},
onTotalChanged: (int total) {
setState(() {
stockItemCount = total;
});
}
),
)
]
);
case 2: case 2:
return ListView( return ListView(
children: ListTile.divideTiles( children: ListTile.divideTiles(
@ -275,40 +307,10 @@ List<Widget> detailTiles() {
} }
List<Widget> stockTiles() {
List<Widget> tiles = [
locationDescriptionCard(),
ListTile(
title: Text(
I18N.of(context).stockItems,
style: TextStyle(fontWeight: FontWeight.bold)
),
trailing: _items.length > 0 ? Text("${_items.length}") : null,
)
];
/*
if (loading) {
tiles.add(progressIndicator());
} else if (_items.length > 0) {
tiles.add(StockList(_items));
} else {
tiles.add(ListTile(
title: Text("No Stock Items"),
subtitle: Text("No stock items available in this location")
));
}
*/
return tiles;
}
List<Widget> actionTiles() { List<Widget> actionTiles() {
List<Widget> tiles = []; List<Widget> tiles = [];
tiles.add(locationDescriptionCard()); tiles.add(locationDescriptionCard(includeActions: false));
// Stock adjustment actions // Stock adjustment actions
if (InvenTreeAPI().checkPermission('stock', 'change')) { if (InvenTreeAPI().checkPermission('stock', 'change')) {
@ -403,10 +405,12 @@ class PaginatedStockList extends StatefulWidget {
final Map<String, String> filters; final Map<String, String> filters;
PaginatedStockList(this.filters); Function onTotalChanged;
PaginatedStockList(this.filters, {this.onTotalChanged});
@override @override
_PaginatedStockListState createState() => _PaginatedStockListState(filters); _PaginatedStockListState createState() => _PaginatedStockListState(filters, onTotalChanged);
} }
@ -418,7 +422,9 @@ class _PaginatedStockListState extends State<PaginatedStockList> {
final Map<String, String> filters; final Map<String, String> filters;
_PaginatedStockListState(this.filters); Function onTotalChanged;
_PaginatedStockListState(this.filters, this.onTotalChanged);
final PagingController<int, InvenTreeStockItem> _pagingController = PagingController(firstPageKey: 0); final PagingController<int, InvenTreeStockItem> _pagingController = PagingController(firstPageKey: 0);
@ -464,6 +470,11 @@ class _PaginatedStockListState extends State<PaginatedStockList> {
final int nextPageKey = pageKey + page.length; final int nextPageKey = pageKey + page.length;
_pagingController.appendPage(items, nextPageKey); _pagingController.appendPage(items, nextPageKey);
} }
if (onTotalChanged != null) {
onTotalChanged(page.count);
}
} catch (error) { } catch (error) {
_pagingController.error = error; _pagingController.error = error;
} }
@ -498,6 +509,9 @@ class _PaginatedStockListState extends State<PaginatedStockList> {
@override @override
Widget build (BuildContext context) { Widget build (BuildContext context) {
return CustomScrollView( return CustomScrollView(
shrinkWrap: true,
physics: ClampingScrollPhysics(),
scrollDirection: Axis.vertical,
slivers: <Widget>[ slivers: <Widget>[
// TODO - Search input // TODO - Search input
PagedSliverList.separated( PagedSliverList.separated(
@ -505,6 +519,9 @@ class _PaginatedStockListState extends State<PaginatedStockList> {
builderDelegate: PagedChildBuilderDelegate<InvenTreeStockItem>( builderDelegate: PagedChildBuilderDelegate<InvenTreeStockItem>(
itemBuilder: (context, item, index) { itemBuilder: (context, item, index) {
return _buildItem(context, item); return _buildItem(context, item);
},
noItemsFoundIndicatorBuilder: (context) {
return NoResultsWidget("No stock items found");
} }
), ),
separatorBuilder: (context, item) => const Divider(height: 1), separatorBuilder: (context, item) => const Divider(height: 1),

44
lib/widget/paginator.dart Normal file
View File

@ -0,0 +1,44 @@
// Pagination related widgets
import 'package:flutter/material.dart';
import 'package:font_awesome_flutter/font_awesome_flutter.dart';
import 'package:flutter_gen/gen_l10n/app_localizations.dart';
class PaginatedSearch extends StatelessWidget {
Function callback;
PaginatedSearch({this.callback});
@override
Widget build(BuildContext context) {
return SliverToBoxAdapter(
child: TextField(
onChanged: callback,
decoration: InputDecoration(
hintText: "Search",
),
)
);
}
}
class NoResultsWidget extends StatelessWidget {
final String description;
NoResultsWidget(this.description);
@override
Widget build(BuildContext context) {
return ListTile(
title: Text(I18N.of(context).noResults),
subtitle: Text(description),
leading: FaIcon(FontAwesomeIcons.exclamationCircle),
);
}
}

View File

@ -427,6 +427,7 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
return tiles; return tiles;
} }
int stockItemCount = 0;
Widget getSelectedWidget(int index) { Widget getSelectedWidget(int index) {
switch (index) { switch (index) {
@ -440,17 +441,30 @@ class _PartDisplayState extends RefreshableState<PartDetailWidget> {
), ),
); );
case 1: case 1:
return PaginatedStockList({"part": "${part.pk}"}); return Column(
/* mainAxisAlignment: MainAxisAlignment.start,
return Center( children: [
child: ListView( headerTile(),
children: ListTile.divideTiles( ListTile(
context: context, title: Text(
tiles: stockTiles() I18N.of(context).stockItems,
).toList() style: TextStyle(fontWeight: FontWeight.bold),
) ),
trailing: Text("${stockItemCount}")
),
Divider(height: 3),
Expanded(
child: PaginatedStockList(
{"part": "${part.pk}"},
onTotalChanged: (int total) {
setState(() {
stockItemCount = total;
});
},
)
)
],
); );
*/
case 2: case 2:
return Center( return Center(
child: ListView( child: ListView(