Sometimes, we would like our widgets to handle their states by themselves. For example, a widget might change its states when receiving an Event.
To do so, we implement the on_event method of Widget.
fn on_event(
&mut self,
_state: &mut Tree,
event: Event,
_layout: Layout<'_>,
_cursor: mouse::Cursor,
_renderer: &Renderer,
_clipboard: &mut dyn Clipboard,
_shell: &mut Shell<'_, Message>,
_viewport: &Rectangle,
) -> event::Status {
match event {
Event::Keyboard(keyboard::Event::KeyPressed {
key: keyboard::Key::Named(Named::Space),
..
}) => {
self.highlight = !self.highlight;
event::Status::Captured
}
_ => event::Status::Ignored,
}
}
Our widget changes its highlight
field every time when the space bar is pressed.
If the Event passed to the on_event method is what the widget needs, we return Status::Captured. Otherwise, we return Status::Ignored to tell the system the event can be used by other widgets.
Since our widget maintains its own state, we do not need to pass the state from our app.
struct MyWidget {
highlight: bool,
}
impl MyWidget {
fn new() -> Self {
Self { highlight: false }
}
}
The full code is as follows:
use iced::{
advanced::{
graphics::core::{event, keyboard},
layout,
renderer::{self, Quad},
widget, Layout, Widget,
},
alignment,
keyboard::key::Named,
mouse,
widget::container,
Border, Color, Element, Event, Length, Rectangle, Shadow, Size, Theme,
};
fn main() -> iced::Result {
iced::application("updating widgets from events", MyApp::update, MyApp::view).run()
}
struct MyApp {
/* Since our widget maintains its own state, we do not need to pass the state from our app. */
// highlight: bool,
}
impl Default for MyApp {
fn default() -> Self {
MyApp::new()
}
}
#[derive(Debug, Clone)]
enum Message {
/*Since our widget maintains its own state, we do not need to pass the state from our app.*/
_Highlight(bool),
}
impl MyApp {
fn new() -> Self {
Self {}
}
fn update(&mut self, _message: Message) {}
fn view(&self) -> Element<Message> {
container(MyWidget::new())
.width(Length::Fill)
.height(Length::Fill)
.align_x(alignment::Horizontal::Center)
.align_y(alignment::Vertical::Center)
.into()
}
}
struct MyWidget {
highlight: bool,
}
impl MyWidget {
fn new() -> Self {
Self { highlight: false }
}
}
impl<Message, Renderer> Widget<Message, Theme, Renderer> for MyWidget
where
Renderer: iced::advanced::Renderer,
{
fn size(&self) -> Size<Length> {
Size {
width: Length::Shrink,
height: Length::Shrink,
}
}
fn layout(
&self,
_tree: &mut widget::Tree,
_renderer: &Renderer,
_limits: &layout::Limits,
) -> layout::Node {
layout::Node::new(Size::new(100., 100.))
}
fn draw(
&self,
_tree: &widget::Tree,
renderer: &mut Renderer,
_theme: &Theme,
_style: &renderer::Style,
layout: Layout<'_>,
_cursor: mouse::Cursor,
_viewport: &Rectangle,
) {
renderer.fill_quad(
Quad {
bounds: layout.bounds(),
border: Border {
color: Color::from_rgb(0.6, 0.8, 1.0),
width: 1.0,
radius: 10.0.into(),
},
shadow: Shadow::default(),
},
// Use the `highlight` variable to change the color of our widget in the draw method.
if self.highlight {
Color::from_rgb(0.6, 0.8, 1.0)
} else {
Color::from_rgb(0.0, 0.2, 0.4)
},
);
}
fn on_event(
&mut self,
_state: &mut widget::Tree,
event: iced::Event,
_layout: Layout<'_>,
_cursor: iced::advanced::mouse::Cursor,
_renderer: &Renderer,
_clipboard: &mut dyn iced::advanced::Clipboard,
_shell: &mut iced::advanced::Shell<'_, Message>,
_viewport: &Rectangle,
) -> iced::advanced::graphics::core::event::Status {
match event {
// When pressed "ENTER", trigger event::Status
Event::Keyboard(keyboard::Event::KeyPressed {
key: keyboard::Key::Named(Named::Space),
..
}) => {
self.highlight = !self.highlight;
event::Status::Captured
}
_ => event::Status::Ignored,
}
}
}
impl<'a, Message, Renderer> From<MyWidget> for Element<'a, Message, Theme, Renderer>
where
Renderer: iced::advanced::Renderer,
{
fn from(widget: MyWidget) -> Self {
Self::new(widget)
}
}
When the space bar is pressed, the widget color switches between light and dark.
➡️ Next: Producing Widget Messages
📘 Back: Table of contents