-
Notifications
You must be signed in to change notification settings - Fork 1.3k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add dropdown autocomplete on Market, Buy, Sell tabs; fix #2226
Basic autocomplete feature for all dropdowns on the major tabs: * Market / Offer Book * Market / Trades * Buy BTC * Sell BTC Known limitations: * Autocomplete still missing from Settings, Account, DAO tabs * Minor UX glitches remain despite lots of debugging and polishing Related issues: * fix #2226 * partially addressed #2712 * superseded #112
- Loading branch information
1 parent
3ccae69
commit 3f1b188
Showing
11 changed files
with
441 additions
and
225 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
194 changes: 194 additions & 0 deletions
194
desktop/src/main/java/bisq/desktop/components/AutocompleteComboBox.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,194 @@ | ||
/* | ||
* This file is part of Bisq. | ||
* | ||
* Bisq is free software: you can redistribute it and/or modify it | ||
* under the terms of the GNU Affero General Public License as published by | ||
* the Free Software Foundation, either version 3 of the License, or (at | ||
* your option) any later version. | ||
* | ||
* Bisq is distributed in the hope that it will be useful, but WITHOUT | ||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public | ||
* License for more details. | ||
* | ||
* You should have received a copy of the GNU Affero General Public License | ||
* along with Bisq. If not, see <http://www.gnu.org/licenses/>. | ||
*/ | ||
|
||
package bisq.desktop.components; | ||
|
||
import bisq.common.UserThread; | ||
|
||
import org.apache.commons.lang3.StringUtils; | ||
|
||
import com.jfoenix.controls.JFXComboBox; | ||
import com.jfoenix.skins.JFXComboBoxListViewSkin; | ||
|
||
import javafx.scene.input.KeyCode; | ||
import javafx.scene.input.KeyEvent; | ||
|
||
import javafx.event.Event; | ||
import javafx.event.EventHandler; | ||
|
||
import javafx.collections.FXCollections; | ||
import javafx.collections.ObservableList; | ||
|
||
import java.util.ArrayList; | ||
import java.util.List; | ||
|
||
/** | ||
* Implements searchable dropdown (an autocomplete like experience). | ||
* | ||
* Clients must use setAutocompleteItems() instead of setItems(). | ||
* | ||
* @param <T> type of the ComboBox item; in the simplest case this can be a String | ||
*/ | ||
public class AutocompleteComboBox<T> extends JFXComboBox<T> { | ||
private ArrayList<T> completeList; | ||
private ArrayList<T> matchingList; | ||
private JFXComboBoxListViewSkin comboBoxListViewSkin; | ||
|
||
public AutocompleteComboBox() { | ||
this(FXCollections.observableArrayList()); | ||
} | ||
|
||
private AutocompleteComboBox(ObservableList<T> items) { | ||
super(items); | ||
setEditable(true); | ||
clearOnFocus(); | ||
setEmptySkinToGetMoreControlOverListView(); | ||
fixSpaceKey(); | ||
setAutocompleteItems(items); | ||
reactToQueryChanges(); | ||
} | ||
|
||
/** | ||
* Set the complete list of ComboBox items. Use this instead of setItems(). | ||
*/ | ||
public void setAutocompleteItems(List<T> items) { | ||
completeList = new ArrayList<>(items); | ||
matchingList = new ArrayList<>(completeList); | ||
setValue(null); | ||
getSelectionModel().clearSelection(); | ||
setItems(FXCollections.observableList(matchingList)); | ||
getEditor().setText(""); | ||
} | ||
|
||
/** | ||
* Triggered when value change is *confirmed*. In practical terms | ||
* this is when user clicks item on the dropdown or hits [ENTER] | ||
* while typing in the text. | ||
* | ||
* This is in contrast to onAction event that is triggered | ||
* on every (unconfirmed) value change. The onAction is not really | ||
* suitable for the search enabled ComboBox. | ||
*/ | ||
public final void setOnChangeConfirmed(EventHandler<Event> eh) { | ||
setOnHidden(e -> { | ||
var inputText = getEditor().getText(); | ||
|
||
// Case 1: fire if input text selects (matches) an item | ||
var selectedItem = getSelectionModel().getSelectedItem(); | ||
var inputTextItem = getConverter().fromString(inputText); | ||
if (selectedItem != null && selectedItem.equals(inputTextItem)) { | ||
eh.handle(e); | ||
return; | ||
} | ||
|
||
// Case 2: fire if the text is empty to support special "show all" case | ||
if (inputText.isEmpty()) | ||
eh.handle(e); | ||
}); | ||
} | ||
|
||
// Clear selection and query when ComboBox gets new focus. This is usually what user | ||
// wants - to have a blank slate for a new search. The primary motivation though | ||
// was to work around UX glitches related to (starting) editing text when combobox | ||
// had specific item selected. | ||
private void clearOnFocus() { | ||
getEditor().focusedProperty().addListener((observableValue, hadFocus, hasFocus) -> { | ||
if (!hadFocus && hasFocus) { | ||
removeFilter(); | ||
forceRedraw(); | ||
} | ||
}); | ||
} | ||
|
||
// The ComboBox API does not provide enough control over the underlying | ||
// ListView that is used as a dropdown. The only way to get this control | ||
// is to set custom ListViewSkin. The default skin is null and so useless. | ||
private void setEmptySkinToGetMoreControlOverListView() { | ||
comboBoxListViewSkin = new JFXComboBoxListViewSkin<>(this); | ||
setSkin(comboBoxListViewSkin); | ||
} | ||
|
||
// By default pressing [SPACE] caused editor text to reset. The solution | ||
// is to suppress relevant event on the underlying ListViewSkin. | ||
private void fixSpaceKey() { | ||
comboBoxListViewSkin.getPopupContent().addEventFilter(KeyEvent.ANY, (KeyEvent event) -> { | ||
if (event.getCode() == KeyCode.SPACE) | ||
event.consume(); | ||
}); | ||
} | ||
|
||
private void filterBy(String query) { | ||
ArrayList<T> newMatchingList = new ArrayList<>(); | ||
for (T item : completeList) | ||
if (StringUtils.containsIgnoreCase(asString(item), query)) | ||
newMatchingList.add(item); | ||
matchingList = newMatchingList; | ||
setValue(null); | ||
getSelectionModel().clearSelection(); | ||
setItems(FXCollections.observableList(matchingList)); | ||
int pos = getEditor().getCaretPosition(); | ||
if (pos > query.length()) pos = query.length(); | ||
getEditor().setText(query); | ||
getEditor().positionCaret(pos); | ||
} | ||
|
||
private void reactToQueryChanges() { | ||
getEditor().addEventHandler(KeyEvent.KEY_RELEASED, (KeyEvent event) -> { | ||
UserThread.execute(() -> { | ||
String query = getEditor().getText(); | ||
var exactMatch = completeList.stream().anyMatch(item -> asString(item).equalsIgnoreCase(query)); | ||
if (!exactMatch) { | ||
if (query.isEmpty()) | ||
removeFilter(); | ||
else | ||
filterBy(query); | ||
forceRedraw(); | ||
} | ||
}); | ||
}); | ||
} | ||
|
||
private void removeFilter() { | ||
matchingList = new ArrayList<>(completeList); | ||
setValue(null); | ||
getSelectionModel().clearSelection(); | ||
setItems(FXCollections.observableList(matchingList)); | ||
getEditor().setText(""); | ||
} | ||
|
||
private void forceRedraw() { | ||
adjustVisibleRowCount(); | ||
if (matchingListSize() > 0) { | ||
comboBoxListViewSkin.getPopupContent().autosize(); | ||
show(); | ||
} else { | ||
hide(); | ||
} | ||
} | ||
|
||
private void adjustVisibleRowCount() { | ||
setVisibleRowCount(Math.min(10, matchingListSize())); | ||
} | ||
|
||
private String asString(T item) { | ||
return getConverter().toString(item); | ||
} | ||
|
||
private int matchingListSize() { | ||
return matchingList.size(); | ||
} | ||
} |
154 changes: 0 additions & 154 deletions
154
desktop/src/main/java/bisq/desktop/components/SearchComboBox.java
This file was deleted.
Oops, something went wrong.
Oops, something went wrong.