In addition to the build-in widgets, we can also design our own custom widgets.
To do so, we need to enable the advanced feature.
The dependencies of the Cargo.toml
file should look like this:
[dependencies]
iced = { version = "0.12.1", features = ["advanced"] }
Then, we need a struct that implement Widget trait.
struct MyWidget;
impl<Message, Renderer> Widget<Message, Theme, Renderer> for MyWidget
where
Renderer: iced::advanced::Renderer,
{
fn size(&self) -> Size<Length> {
// ...
}
fn layout(
&self,
_tree: &mut Tree,
_renderer: &Renderer,
_limits: &layout::Limits,
) -> layout::Node {
// ...
}
fn draw(
&self,
_state: &Tree,
_renderer: &mut Renderer,
_theme: &Theme,
_style: &renderer::Style,
_layout: Layout<'_>,
_cursor: mouse::Cursor,
_viewport: &Rectangle,
) {
// ...
}
}
We define the size of MyWidget
by the methods: size and layout.
Currently, we set the width and height to Length::Shrink, to tell the layout system that we use the least space for this widget.
fn size(&self) -> Size<Length> {
Size {
width: Length::Shrink,
height: Length::Shrink,
}
}
Then, we tell the layout system the precise size we are going to use for the widget.
In this example, our widget is of size (100, 100)
.
fn layout(&self, _renderer: &Renderer, _limits: &layout::Limits) -> layout::Node {
layout::Node::new([100, 100].into())
}
Usually, the layout method would consider the Limits parameter, which is the constraints from the layout system. But now, we ignore it for simplicity.
Next, we draw our widget in the draw method. We use the given Renderer to do so. One may refer to the given Theme and Style for the colors of the widget.
fn draw(
&self,
_state: &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(),
},
Color::from_rgb(0.0, 0.2, 0.4),
);
}
The given Layout parameter would be calculated automatically by the layout system according to the size and layout methods we defined before.
For convenience, we can implement From<MyWidget>
for Element.
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)
}
}
Finally, the widget can be added to our app by the following code.
fn view(&self) -> iced::Element<Self::Message> {
container(MyWidget)
.width(Length::Fill)
.height(Length::Fill)
.center_x()
.center_y()
.into()
}
Note that it is not necessary to put MyWidget
in a Container.
We can add the widget directly into our app.
fn view(&self) -> iced::Element<Self::Message> {
MyWidget.into()
}
The full code is as follows:
use iced::{
advanced::{
layout, mouse,
renderer::{self, Quad},
widget::Tree,
Layout, Widget,
},
widget::container,
Border, Color, Element, Length, Rectangle, Sandbox, Settings, Shadow, Size, Theme,
};
fn main() -> iced::Result {
MyApp::run(Settings::default())
}
struct MyApp;
impl Sandbox for MyApp {
type Message = ();
fn new() -> Self {
Self
}
fn title(&self) -> String {
String::from("My App")
}
fn update(&mut self, _message: Self::Message) {}
fn view(&self) -> iced::Element<Self::Message> {
container(MyWidget)
.width(Length::Fill)
.height(Length::Fill)
.center_x()
.center_y()
.into()
}
}
struct MyWidget;
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 Tree,
_renderer: &Renderer,
_limits: &layout::Limits,
) -> layout::Node {
layout::Node::new([100, 100].into())
}
fn draw(
&self,
_state: &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(),
},
Color::from_rgb(0.0, 0.2, 0.4),
);
}
}
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)
}
}
➡️ Next: Updating Widgets From Outside
📘 Back: Table of contents