1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
use crate::{
Btn, Button, Choice, Color, Dropdown, EventCtx, GeomBatch, GfxCtx, JustDraw, MultiKey, Outcome,
ScreenDims, ScreenPt, Widget, WidgetImpl, WidgetOutput,
};
use geom::Polygon;
pub struct PersistentSplit<T: Clone + PartialEq> {
current_value: T,
btn: Button,
spacer: JustDraw,
dropdown: Dropdown<T>,
}
impl<T: 'static + PartialEq + Clone + std::fmt::Debug> PersistentSplit<T> {
pub fn new<MK: Into<Option<MultiKey>>>(
ctx: &EventCtx,
label: &str,
default_value: T,
hotkey: MK,
choices: Vec<Choice<T>>,
) -> Widget {
let dropdown = Dropdown::new(ctx, "change", default_value, choices, true);
let btn = Btn::plaintext(dropdown.current_value_label())
.build(ctx, label, hotkey)
.take_btn();
Widget::new(Box::new(PersistentSplit {
current_value: dropdown.current_value(),
spacer: JustDraw::wrap(
ctx,
GeomBatch::from(vec![(
Color::WHITE.alpha(0.5),
Polygon::rectangle(3.0, btn.get_dims().height),
)]),
)
.take_just_draw(),
btn,
dropdown,
}))
.named(label)
}
}
impl<T: 'static + PartialEq + Clone> PersistentSplit<T> {
pub fn current_value(&self) -> T {
self.current_value.clone()
}
}
impl<T: 'static + Clone + PartialEq> WidgetImpl for PersistentSplit<T> {
fn get_dims(&self) -> ScreenDims {
let dims1 = self.btn.get_dims();
let dims2 = self.spacer.get_dims();
let dims3 = self.dropdown.get_dims();
ScreenDims::new(
dims1.width + dims2.width + dims3.width,
dims1.height.max(dims2.height).max(dims3.height),
)
}
fn set_pos(&mut self, top_left: ScreenPt) {
self.btn.set_pos(top_left);
self.spacer
.set_pos(ScreenPt::new(top_left.x + self.btn.dims.width, top_left.y));
self.dropdown.set_pos(ScreenPt::new(
top_left.x + self.btn.dims.width + self.spacer.get_dims().width,
top_left.y,
));
}
fn event(&mut self, ctx: &mut EventCtx, output: &mut WidgetOutput) {
self.btn.event(ctx, output);
if let Outcome::Clicked(_) = output.outcome {
return;
}
self.dropdown.event(ctx, &mut WidgetOutput::new());
let new_value = self.dropdown.current_value();
if new_value != self.current_value {
self.current_value = new_value;
let hotkey = self.btn.hotkey.take();
let label = self.btn.action.clone();
self.btn = Btn::plaintext(self.dropdown.current_value_label())
.build(ctx, label, hotkey)
.take_btn();
output.redo_layout = true;
}
}
fn draw(&self, g: &mut GfxCtx) {
self.btn.draw(g);
self.spacer.draw(g);
self.dropdown.draw(g);
}
}