forked from amethyst/amethyst
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathselection.rs
373 lines (329 loc) · 14.9 KB
/
selection.rs
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
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
use std::marker::PhantomData;
use derivative::Derivative;
use derive_new::new;
use serde::{Deserialize, Serialize};
use winit::{ElementState, Event, KeyboardInput, VirtualKeyCode, WindowEvent};
use amethyst_core::{
ecs::{
Component, DenseVecStorage, Entities, FlaggedStorage, Join, Read, ReadStorage, ReaderId,
System, SystemData, World, Write, WriteStorage,
},
shrev::EventChannel,
SystemDesc,
};
use amethyst_input::{BindingTypes, InputHandler};
use crate::{CachedSelectionOrder, UiEvent, UiEventType};
// TODO: If none selected and there is a Selectable in the World, select the lower ordered one automatically?
/// Component indicating that a Ui entity is selectable.
/// Generic Type:
/// - G: Selection Group. Used to determine which entities can be selected together at the same time.
#[derive(Debug, Serialize, Deserialize, new)]
pub struct Selectable<G> {
/// The order in which entities are selected when pressing the `Tab` key or the "go to next" input action.
pub order: u32,
#[new(default)]
/// A multi selection group. When multiple entities are in the same selection group, they can be selected at
/// the same time by holding shift or control and clicking them.
/// You can also select the first element, then hold shift and press the keyboard arrow keys.
// TODO: Holding shift + arrow keys to select more.
// TODO: Pressing the arrow keys could optionally be binded to change the selected ui element.
pub multi_select_group: Option<G>,
#[new(default)]
/// Indicates if you can select multiple entities at once without having to press the shift or control key.
pub auto_multi_select: bool,
/// Indicates if this consumes the inputs. If enabled, all inputs (except Tab) will be ignored when the component is focused.
/// For example, the arrow keys will not change the selected ui element.
/// Example usage: Ui Editable Text.
#[new(default)]
pub consumes_inputs: bool,
}
impl<G: Send + Sync + 'static> Component for Selectable<G> {
type Storage = FlaggedStorage<Self, DenseVecStorage<Self>>;
}
/// Component indicating that a Ui entity is currently selected.
#[derive(Debug, Serialize, Deserialize)]
pub struct Selected;
impl Component for Selected {
type Storage = DenseVecStorage<Self>;
}
/// Builds a `SelectionKeyboardSystem`.
#[derive(Derivative, Debug)]
#[derivative(Default(bound = ""))]
pub struct SelectionKeyboardSystemDesc<G> {
marker: PhantomData<G>,
}
impl<'a, 'b, G> SystemDesc<'a, 'b, SelectionKeyboardSystem<G>> for SelectionKeyboardSystemDesc<G>
where
G: Send + Sync + 'static + PartialEq,
{
fn build(self, world: &mut World) -> SelectionKeyboardSystem<G> {
<SelectionKeyboardSystem<G> as System<'_>>::SystemData::setup(world);
let window_reader_id = world.fetch_mut::<EventChannel<Event>>().register_reader();
SelectionKeyboardSystem::new(window_reader_id)
}
}
/// System managing the selection of entities.
/// Reacts to `UiEvent`.
/// Reacts to Tab and Shift+Tab.
#[derive(Debug)]
pub struct SelectionKeyboardSystem<G> {
window_reader_id: ReaderId<Event>,
phantom: PhantomData<G>,
}
impl<G> SelectionKeyboardSystem<G>
where
G: Send + Sync + 'static + PartialEq,
{
/// Creates a new `SelectionKeyboardSystem`.
pub fn new(window_reader_id: ReaderId<Event>) -> Self {
Self {
window_reader_id,
phantom: PhantomData,
}
}
}
impl<'a, G> System<'a> for SelectionKeyboardSystem<G>
where
G: Send + Sync + 'static + PartialEq,
{
type SystemData = (
Read<'a, EventChannel<Event>>,
Read<'a, CachedSelectionOrder>,
WriteStorage<'a, Selected>,
Write<'a, EventChannel<UiEvent>>,
Entities<'a>,
);
fn run(
&mut self,
(window_events, cached, mut selecteds, mut ui_events, entities): Self::SystemData,
) {
/*
Algorithm in use:
Add clicked elements + shift + ctrl status.
If tab or shift-tab
remove clicked buf
add replace: select higher or lower id closes to previous highest old id
if clicked buf isn't empty
if check currently highest selected multiselect group
// if shift && ctrl -> shift only
if shift
add multiple
else if ctrl || auto_multi_select
add single
else
add replace
else
add replace
*/
// Checks if tab was pressed.
// TODO: Controller support/Use InputEvent in addition to keys.
for event in window_events.read(&mut self.window_reader_id) {
if let Event::WindowEvent {
event:
WindowEvent::KeyboardInput {
input:
KeyboardInput {
state: ElementState::Pressed,
virtual_keycode: Some(VirtualKeyCode::Tab),
modifiers,
..
},
..
},
..
} = *event
{
// Get index of highest selected ui element
let highest = cached.highest_order_selected_index(&selecteds);
if let Some(highest) = highest {
// If Some, an element was currently selected. We move the cursor to the next or previous element depending if Shift was pressed.
// Select Replace
for (entity, _) in (&*entities, &selecteds).join() {
ui_events.single_write(UiEvent::new(UiEventType::Blur, entity));
}
selecteds.clear();
let target = if !modifiers.shift {
// Up
if highest > 0 {
cached.cache.get(highest - 1).unwrap_or_else(|| cached.cache.last()
.expect("unreachable: A highest ui element was selected, but none exist in the cache."))
} else {
cached.cache.last()
.expect("unreachable: A highest ui element was selected, but none exist in the cache.")
}
} else {
// Down
cached.cache.get(highest + 1).unwrap_or_else(|| cached.cache.first()
.expect("unreachable: A highest ui element was selected, but none exist in the cache."))
};
selecteds
.insert(target.1, Selected)
.expect("unreachable: We are inserting");
ui_events.single_write(UiEvent::new(UiEventType::Focus, target.1));
} else if let Some(lowest) = cached.cache.first() {
// If None, nothing was selected. Try to take lowest if it exists.
selecteds
.insert(lowest.1, Selected)
.expect("unreachable: We are inserting");
ui_events.single_write(UiEvent::new(UiEventType::Focus, lowest.1));
}
}
}
}
}
/// Builds a `SelectionMouseSystem`.
#[derive(Derivative, Debug)]
#[derivative(Default(bound = ""))]
pub struct SelectionMouseSystemDesc<G, T>
where
G: Send + Sync + 'static + PartialEq,
T: BindingTypes,
{
marker: PhantomData<(G, T)>,
}
impl<'a, 'b, G, T> SystemDesc<'a, 'b, SelectionMouseSystem<G, T>> for SelectionMouseSystemDesc<G, T>
where
G: Send + Sync + 'static + PartialEq,
T: BindingTypes,
{
fn build(self, world: &mut World) -> SelectionMouseSystem<G, T> {
<SelectionMouseSystem<G, T> as System<'_>>::SystemData::setup(world);
let ui_reader_id = world.fetch_mut::<EventChannel<UiEvent>>().register_reader();
SelectionMouseSystem::new(ui_reader_id)
}
}
/// System handling the clicks on ui entities and selecting them, if applicable.
#[derive(Debug)]
pub struct SelectionMouseSystem<G, T>
where
T: BindingTypes,
{
ui_reader_id: ReaderId<UiEvent>,
phantom: PhantomData<(G, T)>,
}
impl<G, T: BindingTypes> SelectionMouseSystem<G, T>
where
G: Send + Sync + 'static + PartialEq,
{
/// Creates a new `SelectionMouseSystem`.
pub fn new(ui_reader_id: ReaderId<UiEvent>) -> Self {
Self {
ui_reader_id,
phantom: PhantomData,
}
}
}
impl<'a, G, T: BindingTypes> System<'a> for SelectionMouseSystem<G, T>
where
G: Send + Sync + 'static + PartialEq,
{
type SystemData = (
Write<'a, EventChannel<UiEvent>>,
Read<'a, CachedSelectionOrder>,
WriteStorage<'a, Selected>,
ReadStorage<'a, Selectable<G>>,
Read<'a, InputHandler<T>>,
Entities<'a>,
);
fn run(
&mut self,
(mut ui_events, cached, mut selecteds, selectables, input_handler, entities): Self::SystemData,
) {
let shift = input_handler.key_is_down(VirtualKeyCode::LShift)
|| input_handler.key_is_down(VirtualKeyCode::RShift);
let ctrl = input_handler.key_is_down(VirtualKeyCode::LControl)
|| input_handler.key_is_down(VirtualKeyCode::RControl);
let mut emitted: Vec<UiEvent> = Vec::new();
// Add clicked elements to clicked buffer
for ev in ui_events.read(&mut self.ui_reader_id) {
if let UiEventType::ClickStart = ev.event_type {
// Ignore events from elements removed between the event emission and now.
if selectables.get(ev.target).is_some() {
let clicked = ev.target;
// Inside of the loop because its possible that the user clicks two times in a frame while pressing shift.
let highest = cached.highest_order_selected_index(&selecteds);
if let Some(highest) = highest {
let (highest_is_select, auto_multi_select) = {
let highest_multi_select_group = &selectables
.get(
cached
.cache
.get(highest)
.expect(
"unreachable: we just got those values from the cache.",
)
.1,
)
.expect("unreachable: we just got those values from the cache.")
.multi_select_group;
let (target_multi_select_group, auto_multi_select) = {
let target_selectable = selectables.get(clicked).expect("unreachable: Because when filling the buffer we checked that the component still exist on the entity.");
(
&target_selectable.multi_select_group,
target_selectable.auto_multi_select,
)
};
(
highest_multi_select_group == target_multi_select_group,
auto_multi_select,
)
};
if highest_is_select {
if shift {
// Add from latest selected to target for all that have same multi_select_group
let cached_index_clicked = cached.index_of(clicked)
.expect("unreachable: Entity has to be in the cache, otherwise it wouldn't have been added.");
// When multi-selecting, you remove everything that was previously selected, and then add everything in the range.
for (entity, _) in (&*entities, &selecteds).join() {
emitted.push(UiEvent::new(UiEventType::Blur, entity));
}
selecteds.clear();
let min = cached_index_clicked.min(highest);
let max = cached_index_clicked.max(highest);
for i in min..=max {
let target_entity = cached.cache.get(i).expect(
"unreachable: Range has to be inside of the cache range.",
);
selecteds
.insert(target_entity.1, Selected)
.expect("unreachable: We are inserting");
emitted.push(UiEvent::new(UiEventType::Focus, target_entity.1));
}
} else if ctrl || auto_multi_select {
// Select adding single element
selecteds
.insert(clicked, Selected)
.expect("unreachable: We are inserting");
emitted.push(UiEvent::new(UiEventType::Focus, clicked));
} else {
// Select replace, because we don't want to be adding elements.
selecteds.clear();
selecteds
.insert(clicked, Selected)
.expect("unreachable: We are inserting");
emitted.push(UiEvent::new(UiEventType::Focus, clicked));
}
} else {
for (entity, _) in (&*entities, &selecteds).join() {
emitted.push(UiEvent::new(UiEventType::Blur, entity));
}
// Different multi select group than the latest one selected. Execute Select replace
selecteds.clear();
selecteds
.insert(clicked, Selected)
.expect("unreachable: We are inserting");
emitted.push(UiEvent::new(UiEventType::Focus, clicked));
}
} else {
// Nothing was previously selected, let's just select single.
selecteds
.insert(clicked, Selected)
.expect("unreachable: We are inserting");
emitted.push(UiEvent::new(UiEventType::Focus, clicked));
}
}
}
}
ui_events.iter_write(emitted.into_iter());
}
}