use egui::{
text::{CCursor, CCursorRange},
Id, PopupCloseBehavior, Response, TextEdit, Ui, Widget, WidgetText,
};
use std::hash::Hash;
pub struct DropDownBox<
'a,
F: FnMut(&mut Ui, &str) -> Response,
V: AsRef<str>,
I: Iterator<Item = V>,
> {
buf: &'a mut String,
popup_id: Id,
display: F,
it: I,
hint_text: WidgetText,
filter_by_input: bool,
select_on_focus: bool,
desired_width: Option<f32>,
}
impl<'a, F: FnMut(&mut Ui, &str) -> Response, V: AsRef<str>, I: Iterator<Item = V>>
DropDownBox<'a, F, V, I>
{
pub fn from_iter(
it: impl IntoIterator<IntoIter = I>,
id_source: impl Hash,
buf: &'a mut String,
display: F,
) -> Self {
Self {
popup_id: Id::new(id_source),
it: it.into_iter(),
display,
buf,
hint_text: WidgetText::default(),
filter_by_input: true,
select_on_focus: false,
desired_width: None,
}
}
pub fn hint_text(mut self, hint_text: impl Into<WidgetText>) -> Self {
self.hint_text = hint_text.into();
self
}
pub fn filter_by_input(mut self, filter_by_input: bool) -> Self {
self.filter_by_input = filter_by_input;
self
}
pub fn select_on_focus(mut self, select_on_focus: bool) -> Self {
self.select_on_focus = select_on_focus;
self
}
pub fn desired_width(mut self, desired_width: f32) -> Self {
self.desired_width = desired_width.into();
self
}
}
impl<'a, F: FnMut(&mut Ui, &str) -> Response, V: AsRef<str>, I: Iterator<Item = V>> Widget
for DropDownBox<'a, F, V, I>
{
fn ui(self, ui: &mut Ui) -> Response {
let Self {
popup_id,
buf,
it,
mut display,
hint_text,
filter_by_input,
select_on_focus,
desired_width,
} = self;
let mut edit = TextEdit::singleline(buf).hint_text(hint_text);
if let Some(dw) = desired_width {
edit = edit.desired_width(dw);
}
let mut edit_output = edit.show(ui);
let mut r = edit_output.response;
if r.gained_focus() {
if select_on_focus {
edit_output
.state
.cursor
.set_char_range(Some(CCursorRange::two(
CCursor::new(0),
CCursor::new(buf.len()),
)));
edit_output.state.store(ui.ctx(), r.id);
}
ui.memory_mut(|m| m.open_popup(popup_id));
}
let mut changed = false;
egui::popup_below_widget(ui, popup_id, &r, PopupCloseBehavior::CloseOnClick, |ui| {
egui::ScrollArea::vertical().show(ui, |ui| {
for var in it {
let text = var.as_ref();
if filter_by_input
&& !buf.is_empty()
&& !text.to_lowercase().contains(&buf.to_lowercase())
{
continue;
}
if display(ui, text).clicked() {
*buf = text.to_owned();
changed = true;
ui.memory_mut(|m| m.close_popup());
}
}
});
});
if changed {
r.mark_changed();
}
r
}
}