mirror of
https://github.com/inventree/inventree-app.git
synced 2025-04-28 05:26:47 +00:00
Fix for search screen
- Change input and controller - Add focus node - Add "searching" indicator
This commit is contained in:
parent
d7d8cefddd
commit
acf89426ce
@ -1,6 +1,11 @@
|
|||||||
## InvenTree App Release Notes
|
## InvenTree App Release Notes
|
||||||
---
|
---
|
||||||
|
|
||||||
|
### 0.7.1 - May 2022
|
||||||
|
---
|
||||||
|
|
||||||
|
- Fixes issue which prevented text input in search window
|
||||||
|
|
||||||
### 0.7.0 - May 2022
|
### 0.7.0 - May 2022
|
||||||
---
|
---
|
||||||
|
|
||||||
|
@ -797,6 +797,9 @@
|
|||||||
"description": "search"
|
"description": "search"
|
||||||
},
|
},
|
||||||
|
|
||||||
|
"searching": "Searching",
|
||||||
|
"@searching": {},
|
||||||
|
|
||||||
"searchLocation": "Search for location",
|
"searchLocation": "Search for location",
|
||||||
"@searchLocation": {},
|
"@searchLocation": {},
|
||||||
|
|
||||||
|
@ -36,6 +36,19 @@ class _SearchDisplayState extends RefreshableState<SearchWidget> {
|
|||||||
|
|
||||||
final bool hasAppBar;
|
final bool hasAppBar;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
|
||||||
|
_focusNode = FocusNode();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
_focusNode.dispose();
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String getAppBarTitle(BuildContext context) => L10().search;
|
String getAppBarTitle(BuildContext context) => L10().search;
|
||||||
|
|
||||||
@ -52,6 +65,17 @@ class _SearchDisplayState extends RefreshableState<SearchWidget> {
|
|||||||
|
|
||||||
Timer? debounceTimer;
|
Timer? debounceTimer;
|
||||||
|
|
||||||
|
bool isSearching() {
|
||||||
|
|
||||||
|
if (searchController.text.isEmpty) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return nSearchResults < 6;
|
||||||
|
}
|
||||||
|
|
||||||
|
int nSearchResults = 0;
|
||||||
|
|
||||||
int nPartResults = 0;
|
int nPartResults = 0;
|
||||||
|
|
||||||
int nCategoryResults = 0;
|
int nCategoryResults = 0;
|
||||||
@ -64,6 +88,8 @@ class _SearchDisplayState extends RefreshableState<SearchWidget> {
|
|||||||
|
|
||||||
int nPurchaseOrderResults = 0;
|
int nPurchaseOrderResults = 0;
|
||||||
|
|
||||||
|
late FocusNode _focusNode;
|
||||||
|
|
||||||
// Callback when the text is being edited
|
// Callback when the text is being edited
|
||||||
// Incorporates a debounce timer to restrict search frequency
|
// Incorporates a debounce timer to restrict search frequency
|
||||||
void onSearchTextChanged(String text, {bool immediate = false}) {
|
void onSearchTextChanged(String text, {bool immediate = false}) {
|
||||||
@ -84,6 +110,10 @@ class _SearchDisplayState extends RefreshableState<SearchWidget> {
|
|||||||
|
|
||||||
Future<void> search(String term) async {
|
Future<void> search(String term) async {
|
||||||
|
|
||||||
|
setState(() {
|
||||||
|
nSearchResults = 0;
|
||||||
|
});
|
||||||
|
|
||||||
if (term.isEmpty) {
|
if (term.isEmpty) {
|
||||||
setState(() {
|
setState(() {
|
||||||
// Do not search on an empty string
|
// Do not search on an empty string
|
||||||
@ -93,6 +123,8 @@ class _SearchDisplayState extends RefreshableState<SearchWidget> {
|
|||||||
nLocationResults = 0;
|
nLocationResults = 0;
|
||||||
nSupplierResults = 0;
|
nSupplierResults = 0;
|
||||||
nPurchaseOrderResults = 0;
|
nPurchaseOrderResults = 0;
|
||||||
|
|
||||||
|
nSearchResults = 0;
|
||||||
});
|
});
|
||||||
|
|
||||||
return;
|
return;
|
||||||
@ -104,6 +136,8 @@ class _SearchDisplayState extends RefreshableState<SearchWidget> {
|
|||||||
).then((int n) {
|
).then((int n) {
|
||||||
setState(() {
|
setState(() {
|
||||||
nPartResults = n;
|
nPartResults = n;
|
||||||
|
|
||||||
|
nSearchResults++;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -113,6 +147,8 @@ class _SearchDisplayState extends RefreshableState<SearchWidget> {
|
|||||||
).then((int n) {
|
).then((int n) {
|
||||||
setState(() {
|
setState(() {
|
||||||
nCategoryResults = n;
|
nCategoryResults = n;
|
||||||
|
|
||||||
|
nSearchResults++;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -122,6 +158,8 @@ class _SearchDisplayState extends RefreshableState<SearchWidget> {
|
|||||||
).then((int n) {
|
).then((int n) {
|
||||||
setState(() {
|
setState(() {
|
||||||
nStockResults = n;
|
nStockResults = n;
|
||||||
|
|
||||||
|
nSearchResults++;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -131,6 +169,8 @@ class _SearchDisplayState extends RefreshableState<SearchWidget> {
|
|||||||
).then((int n) {
|
).then((int n) {
|
||||||
setState(() {
|
setState(() {
|
||||||
nLocationResults = n;
|
nLocationResults = n;
|
||||||
|
|
||||||
|
nSearchResults++;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -143,6 +183,7 @@ class _SearchDisplayState extends RefreshableState<SearchWidget> {
|
|||||||
).then((int n) {
|
).then((int n) {
|
||||||
setState(() {
|
setState(() {
|
||||||
nSupplierResults = n;
|
nSupplierResults = n;
|
||||||
|
nSearchResults++;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -155,6 +196,7 @@ class _SearchDisplayState extends RefreshableState<SearchWidget> {
|
|||||||
).then((int n) {
|
).then((int n) {
|
||||||
setState(() {
|
setState(() {
|
||||||
nPurchaseOrderResults = n;
|
nPurchaseOrderResults = n;
|
||||||
|
nSearchResults++;
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -166,29 +208,31 @@ class _SearchDisplayState extends RefreshableState<SearchWidget> {
|
|||||||
|
|
||||||
// Search input
|
// Search input
|
||||||
tiles.add(
|
tiles.add(
|
||||||
InputDecorator(
|
TextFormField(
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
),
|
hintText: L10().queryEmpty,
|
||||||
child: ListTile(
|
prefixIcon: IconButton(
|
||||||
title: TextField(
|
icon: FaIcon(FontAwesomeIcons.search),
|
||||||
readOnly: false,
|
onPressed: null,
|
||||||
decoration: InputDecoration(
|
|
||||||
helperText: L10().queryEmpty,
|
|
||||||
),
|
|
||||||
controller: searchController,
|
|
||||||
onChanged: (String text) {
|
|
||||||
onSearchTextChanged(text);
|
|
||||||
},
|
|
||||||
),
|
),
|
||||||
trailing: IconButton(
|
suffixIcon: IconButton(
|
||||||
icon: FaIcon(FontAwesomeIcons.backspace, color: Colors.red),
|
icon: FaIcon(FontAwesomeIcons.backspace, color: Colors.red),
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
searchController.clear();
|
searchController.clear();
|
||||||
onSearchTextChanged("", immediate: true);
|
onSearchTextChanged("", immediate: true);
|
||||||
},
|
_focusNode.requestFocus();
|
||||||
|
}
|
||||||
),
|
),
|
||||||
)
|
),
|
||||||
)
|
readOnly: false,
|
||||||
|
autofocus: true,
|
||||||
|
autocorrect: false,
|
||||||
|
focusNode: _focusNode,
|
||||||
|
controller: searchController,
|
||||||
|
onChanged: (String text) {
|
||||||
|
onSearchTextChanged(text);
|
||||||
|
},
|
||||||
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
String query = searchController.text;
|
String query = searchController.text;
|
||||||
@ -335,7 +379,17 @@ class _SearchDisplayState extends RefreshableState<SearchWidget> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (results.isEmpty && searchController.text.isNotEmpty) {
|
if (isSearching()) {
|
||||||
|
tiles.add(
|
||||||
|
ListTile(
|
||||||
|
title: Text(L10().searching),
|
||||||
|
leading: FaIcon(FontAwesomeIcons.search),
|
||||||
|
trailing: CircularProgressIndicator(),
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isSearching() && results.isEmpty && searchController.text.isNotEmpty) {
|
||||||
tiles.add(
|
tiles.add(
|
||||||
ListTile(
|
ListTile(
|
||||||
title: Text(L10().queryNoResults),
|
title: Text(L10().queryNoResults),
|
||||||
|
Loading…
x
Reference in New Issue
Block a user