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