1use crate::core::alignment;
3use crate::core::border::{self, Border};
4use crate::core::layout::{self, Layout};
5use crate::core::mouse;
6use crate::core::overlay;
7use crate::core::renderer;
8use crate::core::text::{self, Text};
9use crate::core::touch;
10use crate::core::widget::tree::{self, Tree};
11use crate::core::window;
12use crate::core::{
13 Background, Clipboard, Color, Event, Length, Padding, Pixels, Point,
14 Rectangle, Shadow, Size, Theme, Vector,
15};
16use crate::core::{Element, Shell, Widget};
17use crate::scrollable::{self, Scrollable};
18
19pub struct Menu<
21 'a,
22 'b,
23 T,
24 Message,
25 Theme = crate::Theme,
26 Renderer = crate::Renderer,
27> where
28 Theme: Catalog,
29 Renderer: text::Renderer,
30 'b: 'a,
31{
32 state: &'a mut State,
33 options: &'a [T],
34 hovered_option: &'a mut Option<usize>,
35 on_selected: Box<dyn FnMut(T) -> Message + 'a>,
36 on_option_hovered: Option<&'a dyn Fn(T) -> Message>,
37 width: f32,
38 padding: Padding,
39 text_size: Option<Pixels>,
40 text_line_height: text::LineHeight,
41 text_shaping: text::Shaping,
42 font: Option<Renderer::Font>,
43 class: &'a <Theme as Catalog>::Class<'b>,
44}
45
46impl<'a, 'b, T, Message, Theme, Renderer>
47 Menu<'a, 'b, T, Message, Theme, Renderer>
48where
49 T: ToString + Clone,
50 Message: 'a,
51 Theme: Catalog + 'a,
52 Renderer: text::Renderer + 'a,
53 'b: 'a,
54{
55 pub fn new(
58 state: &'a mut State,
59 options: &'a [T],
60 hovered_option: &'a mut Option<usize>,
61 on_selected: impl FnMut(T) -> Message + 'a,
62 on_option_hovered: Option<&'a dyn Fn(T) -> Message>,
63 class: &'a <Theme as Catalog>::Class<'b>,
64 ) -> Self {
65 Menu {
66 state,
67 options,
68 hovered_option,
69 on_selected: Box::new(on_selected),
70 on_option_hovered,
71 width: 0.0,
72 padding: Padding::ZERO,
73 text_size: None,
74 text_line_height: text::LineHeight::default(),
75 text_shaping: text::Shaping::default(),
76 font: None,
77 class,
78 }
79 }
80
81 pub fn width(mut self, width: f32) -> Self {
83 self.width = width;
84 self
85 }
86
87 pub fn padding<P: Into<Padding>>(mut self, padding: P) -> Self {
89 self.padding = padding.into();
90 self
91 }
92
93 pub fn text_size(mut self, text_size: impl Into<Pixels>) -> Self {
95 self.text_size = Some(text_size.into());
96 self
97 }
98
99 pub fn text_line_height(
101 mut self,
102 line_height: impl Into<text::LineHeight>,
103 ) -> Self {
104 self.text_line_height = line_height.into();
105 self
106 }
107
108 pub fn text_shaping(mut self, shaping: text::Shaping) -> Self {
110 self.text_shaping = shaping;
111 self
112 }
113
114 pub fn font(mut self, font: impl Into<Renderer::Font>) -> Self {
116 self.font = Some(font.into());
117 self
118 }
119
120 pub fn overlay(
127 self,
128 position: Point,
129 viewport: Rectangle,
130 target_height: f32,
131 menu_height: Length,
132 ) -> overlay::Element<'a, Message, Theme, Renderer> {
133 overlay::Element::new(Box::new(Overlay::new(
134 position,
135 viewport,
136 self,
137 target_height,
138 menu_height,
139 )))
140 }
141}
142
143#[derive(Debug)]
145pub struct State {
146 tree: Tree,
147}
148
149impl State {
150 pub fn new() -> Self {
152 Self {
153 tree: Tree::empty(),
154 }
155 }
156}
157
158impl Default for State {
159 fn default() -> Self {
160 Self::new()
161 }
162}
163
164struct Overlay<'a, 'b, Message, Theme, Renderer>
165where
166 Theme: Catalog,
167 Renderer: text::Renderer,
168{
169 position: Point,
170 viewport: Rectangle,
171 tree: &'a mut Tree,
172 list: Scrollable<'a, Message, Theme, Renderer>,
173 width: f32,
174 target_height: f32,
175 class: &'a <Theme as Catalog>::Class<'b>,
176}
177
178impl<'a, 'b, Message, Theme, Renderer> Overlay<'a, 'b, Message, Theme, Renderer>
179where
180 Message: 'a,
181 Theme: Catalog + scrollable::Catalog + 'a,
182 Renderer: text::Renderer + 'a,
183 'b: 'a,
184{
185 pub fn new<T>(
186 position: Point,
187 viewport: Rectangle,
188 menu: Menu<'a, 'b, T, Message, Theme, Renderer>,
189 target_height: f32,
190 menu_height: Length,
191 ) -> Self
192 where
193 T: Clone + ToString,
194 {
195 let Menu {
196 state,
197 options,
198 hovered_option,
199 on_selected,
200 on_option_hovered,
201 width,
202 padding,
203 font,
204 text_size,
205 text_line_height,
206 text_shaping,
207 class,
208 } = menu;
209
210 let list = Scrollable::new(List {
211 options,
212 hovered_option,
213 on_selected,
214 on_option_hovered,
215 font,
216 text_size,
217 text_line_height,
218 text_shaping,
219 padding,
220 class,
221 })
222 .height(menu_height);
223
224 state.tree.diff(&list as &dyn Widget<_, _, _>);
225
226 Self {
227 position,
228 viewport,
229 tree: &mut state.tree,
230 list,
231 width,
232 target_height,
233 class,
234 }
235 }
236}
237
238impl<Message, Theme, Renderer> crate::core::Overlay<Message, Theme, Renderer>
239 for Overlay<'_, '_, Message, Theme, Renderer>
240where
241 Theme: Catalog,
242 Renderer: text::Renderer,
243{
244 fn layout(&mut self, renderer: &Renderer, bounds: Size) -> layout::Node {
245 let space_below =
246 bounds.height - (self.position.y + self.target_height);
247 let space_above = self.position.y;
248
249 let limits = layout::Limits::new(
250 Size::ZERO,
251 Size::new(
252 bounds.width - self.position.x,
253 if space_below > space_above {
254 space_below
255 } else {
256 space_above
257 },
258 ),
259 )
260 .width(self.width);
261
262 let node = self.list.layout(self.tree, renderer, &limits);
263 let size = node.size();
264
265 node.move_to(if space_below > space_above {
266 self.position + Vector::new(0.0, self.target_height)
267 } else {
268 self.position - Vector::new(0.0, size.height)
269 })
270 }
271
272 fn update(
273 &mut self,
274 event: &Event,
275 layout: Layout<'_>,
276 cursor: mouse::Cursor,
277 renderer: &Renderer,
278 clipboard: &mut dyn Clipboard,
279 shell: &mut Shell<'_, Message>,
280 ) {
281 let bounds = layout.bounds();
282
283 self.list.update(
284 self.tree, event, layout, cursor, renderer, clipboard, shell,
285 &bounds,
286 );
287 }
288
289 fn mouse_interaction(
290 &self,
291 layout: Layout<'_>,
292 cursor: mouse::Cursor,
293 renderer: &Renderer,
294 ) -> mouse::Interaction {
295 self.list.mouse_interaction(
296 self.tree,
297 layout,
298 cursor,
299 &self.viewport,
300 renderer,
301 )
302 }
303
304 fn draw(
305 &self,
306 renderer: &mut Renderer,
307 theme: &Theme,
308 defaults: &renderer::Style,
309 layout: Layout<'_>,
310 cursor: mouse::Cursor,
311 ) {
312 let bounds = layout.bounds();
313
314 let style = Catalog::style(theme, self.class);
315
316 renderer.fill_quad(
317 renderer::Quad {
318 bounds,
319 border: style.border,
320 shadow: style.shadow,
321 ..renderer::Quad::default()
322 },
323 style.background,
324 );
325
326 self.list.draw(
327 self.tree, renderer, theme, defaults, layout, cursor, &bounds,
328 );
329 }
330}
331
332struct List<'a, 'b, T, Message, Theme, Renderer>
333where
334 Theme: Catalog,
335 Renderer: text::Renderer,
336{
337 options: &'a [T],
338 hovered_option: &'a mut Option<usize>,
339 on_selected: Box<dyn FnMut(T) -> Message + 'a>,
340 on_option_hovered: Option<&'a dyn Fn(T) -> Message>,
341 padding: Padding,
342 text_size: Option<Pixels>,
343 text_line_height: text::LineHeight,
344 text_shaping: text::Shaping,
345 font: Option<Renderer::Font>,
346 class: &'a <Theme as Catalog>::Class<'b>,
347}
348
349struct ListState {
350 is_hovered: Option<bool>,
351}
352
353impl<T, Message, Theme, Renderer> Widget<Message, Theme, Renderer>
354 for List<'_, '_, T, Message, Theme, Renderer>
355where
356 T: Clone + ToString,
357 Theme: Catalog,
358 Renderer: text::Renderer,
359{
360 fn tag(&self) -> tree::Tag {
361 tree::Tag::of::<Option<bool>>()
362 }
363
364 fn state(&self) -> tree::State {
365 tree::State::new(ListState { is_hovered: None })
366 }
367
368 fn size(&self) -> Size<Length> {
369 Size {
370 width: Length::Fill,
371 height: Length::Shrink,
372 }
373 }
374
375 fn layout(
376 &mut self,
377 _tree: &mut Tree,
378 renderer: &Renderer,
379 limits: &layout::Limits,
380 ) -> layout::Node {
381 use std::f32;
382
383 let text_size =
384 self.text_size.unwrap_or_else(|| renderer.default_size());
385
386 let text_line_height = self.text_line_height.to_absolute(text_size);
387
388 let size = {
389 let intrinsic = Size::new(
390 0.0,
391 (f32::from(text_line_height) + self.padding.y())
392 * self.options.len() as f32,
393 );
394
395 limits.resolve(Length::Fill, Length::Shrink, intrinsic)
396 };
397
398 layout::Node::new(size)
399 }
400
401 fn update(
402 &mut self,
403 tree: &mut Tree,
404 event: &Event,
405 layout: Layout<'_>,
406 cursor: mouse::Cursor,
407 renderer: &Renderer,
408 _clipboard: &mut dyn Clipboard,
409 shell: &mut Shell<'_, Message>,
410 _viewport: &Rectangle,
411 ) {
412 match event {
413 Event::Mouse(mouse::Event::ButtonPressed(mouse::Button::Left)) => {
414 if cursor.is_over(layout.bounds())
415 && let Some(index) = *self.hovered_option
416 && let Some(option) = self.options.get(index)
417 {
418 shell.publish((self.on_selected)(option.clone()));
419 shell.capture_event();
420 }
421 }
422 Event::Mouse(mouse::Event::CursorMoved { .. }) => {
423 if let Some(cursor_position) =
424 cursor.position_in(layout.bounds())
425 {
426 let text_size = self
427 .text_size
428 .unwrap_or_else(|| renderer.default_size());
429
430 let option_height =
431 f32::from(self.text_line_height.to_absolute(text_size))
432 + self.padding.y();
433
434 let new_hovered_option =
435 (cursor_position.y / option_height) as usize;
436
437 if *self.hovered_option != Some(new_hovered_option)
438 && let Some(option) =
439 self.options.get(new_hovered_option)
440 {
441 if let Some(on_option_hovered) = self.on_option_hovered
442 {
443 shell.publish(on_option_hovered(option.clone()));
444 }
445
446 shell.request_redraw();
447 }
448
449 *self.hovered_option = Some(new_hovered_option);
450 }
451 }
452 Event::Touch(touch::Event::FingerPressed { .. }) => {
453 if let Some(cursor_position) =
454 cursor.position_in(layout.bounds())
455 {
456 let text_size = self
457 .text_size
458 .unwrap_or_else(|| renderer.default_size());
459
460 let option_height =
461 f32::from(self.text_line_height.to_absolute(text_size))
462 + self.padding.y();
463
464 *self.hovered_option =
465 Some((cursor_position.y / option_height) as usize);
466
467 if let Some(index) = *self.hovered_option
468 && let Some(option) = self.options.get(index)
469 {
470 shell.publish((self.on_selected)(option.clone()));
471 shell.capture_event();
472 }
473 }
474 }
475 _ => {}
476 }
477
478 let state = tree.state.downcast_mut::<ListState>();
479
480 if let Event::Window(window::Event::RedrawRequested(_now)) = event {
481 state.is_hovered = Some(cursor.is_over(layout.bounds()));
482 } else if state.is_hovered.is_some_and(|is_hovered| {
483 is_hovered != cursor.is_over(layout.bounds())
484 }) {
485 shell.request_redraw();
486 }
487 }
488
489 fn mouse_interaction(
490 &self,
491 _tree: &Tree,
492 layout: Layout<'_>,
493 cursor: mouse::Cursor,
494 _viewport: &Rectangle,
495 _renderer: &Renderer,
496 ) -> mouse::Interaction {
497 let is_mouse_over = cursor.is_over(layout.bounds());
498
499 if is_mouse_over {
500 mouse::Interaction::Pointer
501 } else {
502 mouse::Interaction::default()
503 }
504 }
505
506 fn draw(
507 &self,
508 _tree: &Tree,
509 renderer: &mut Renderer,
510 theme: &Theme,
511 _style: &renderer::Style,
512 layout: Layout<'_>,
513 _cursor: mouse::Cursor,
514 viewport: &Rectangle,
515 ) {
516 let style = Catalog::style(theme, self.class);
517 let bounds = layout.bounds();
518
519 let text_size =
520 self.text_size.unwrap_or_else(|| renderer.default_size());
521 let option_height =
522 f32::from(self.text_line_height.to_absolute(text_size))
523 + self.padding.y();
524
525 let offset = viewport.y - bounds.y;
526 let start = (offset / option_height) as usize;
527 let end = ((offset + viewport.height) / option_height).ceil() as usize;
528
529 let visible_options = &self.options[start..end.min(self.options.len())];
530
531 for (i, option) in visible_options.iter().enumerate() {
532 let i = start + i;
533 let is_selected = *self.hovered_option == Some(i);
534
535 let bounds = Rectangle {
536 x: bounds.x,
537 y: bounds.y + (option_height * i as f32),
538 width: bounds.width,
539 height: option_height,
540 };
541
542 if is_selected {
543 renderer.fill_quad(
544 renderer::Quad {
545 bounds: Rectangle {
546 x: bounds.x + style.border.width,
547 width: bounds.width - style.border.width * 2.0,
548 ..bounds
549 },
550 border: border::rounded(style.border.radius),
551 ..renderer::Quad::default()
552 },
553 style.selected_background,
554 );
555 }
556
557 renderer.fill_text(
558 Text {
559 content: option.to_string(),
560 bounds: Size::new(f32::INFINITY, bounds.height),
561 size: text_size,
562 line_height: self.text_line_height,
563 font: self.font.unwrap_or_else(|| renderer.default_font()),
564 align_x: text::Alignment::Default,
565 align_y: alignment::Vertical::Center,
566 shaping: self.text_shaping,
567 wrapping: text::Wrapping::default(),
568 },
569 Point::new(bounds.x + self.padding.left, bounds.center_y()),
570 if is_selected {
571 style.selected_text_color
572 } else {
573 style.text_color
574 },
575 *viewport,
576 );
577 }
578 }
579}
580
581impl<'a, 'b, T, Message, Theme, Renderer>
582 From<List<'a, 'b, T, Message, Theme, Renderer>>
583 for Element<'a, Message, Theme, Renderer>
584where
585 T: ToString + Clone,
586 Message: 'a,
587 Theme: 'a + Catalog,
588 Renderer: 'a + text::Renderer,
589 'b: 'a,
590{
591 fn from(list: List<'a, 'b, T, Message, Theme, Renderer>) -> Self {
592 Element::new(list)
593 }
594}
595
596#[derive(Debug, Clone, Copy, PartialEq)]
598pub struct Style {
599 pub background: Background,
601 pub border: Border,
603 pub text_color: Color,
605 pub selected_text_color: Color,
607 pub selected_background: Background,
609 pub shadow: Shadow,
611}
612
613pub trait Catalog: scrollable::Catalog {
615 type Class<'a>;
617
618 fn default<'a>() -> <Self as Catalog>::Class<'a>;
620
621 fn default_scrollable<'a>() -> <Self as scrollable::Catalog>::Class<'a> {
623 <Self as scrollable::Catalog>::default()
624 }
625
626 fn style(&self, class: &<Self as Catalog>::Class<'_>) -> Style;
628}
629
630pub type StyleFn<'a, Theme> = Box<dyn Fn(&Theme) -> Style + 'a>;
632
633impl Catalog for Theme {
634 type Class<'a> = StyleFn<'a, Self>;
635
636 fn default<'a>() -> StyleFn<'a, Self> {
637 Box::new(default)
638 }
639
640 fn style(&self, class: &StyleFn<'_, Self>) -> Style {
641 class(self)
642 }
643}
644
645pub fn default(theme: &Theme) -> Style {
647 let palette = theme.extended_palette();
648
649 Style {
650 background: palette.background.weak.color.into(),
651 border: Border {
652 width: 1.0,
653 radius: 0.0.into(),
654 color: palette.background.strong.color,
655 },
656 text_color: palette.background.weak.text,
657 selected_text_color: palette.primary.strong.text,
658 selected_background: palette.primary.strong.color.into(),
659 shadow: Shadow::default(),
660 }
661}