summaryrefslogtreecommitdiffstats
path: root/examples/tour/src/main.rs
diff options
context:
space:
mode:
Diffstat (limited to 'examples/tour/src/main.rs')
-rw-r--r--examples/tour/src/main.rs534
1 files changed, 187 insertions, 347 deletions
diff --git a/examples/tour/src/main.rs b/examples/tour/src/main.rs
index d85f2916..75d2ce08 100644
--- a/examples/tour/src/main.rs
+++ b/examples/tour/src/main.rs
@@ -1,14 +1,11 @@
use iced::alignment;
-use iced::button;
-use iced::scrollable;
-use iced::slider;
-use iced::text_input;
use iced::theme;
-use iced::{
- Button, Checkbox, Color, Column, Container, ContentFit, Element, Image,
- Length, Radio, Row, Sandbox, Scrollable, Settings, Slider, Space, Text,
- TextInput, Toggler,
+use iced::widget::{
+ checkbox, column, container, horizontal_space, image, radio, row,
+ scrollable, slider, text, text_input, toggler, vertical_space,
};
+use iced::widget::{Button, Column, Container, Slider};
+use iced::{Color, Element, Length, Renderer, Sandbox, Settings};
pub fn main() -> iced::Result {
env_logger::init();
@@ -18,9 +15,6 @@ pub fn main() -> iced::Result {
pub struct Tour {
steps: Steps,
- scroll: scrollable::State,
- back_button: button::State,
- next_button: button::State,
debug: bool,
}
@@ -30,9 +24,6 @@ impl Sandbox for Tour {
fn new() -> Tour {
Tour {
steps: Steps::new(),
- scroll: scrollable::State::new(),
- back_button: button::State::new(),
- next_button: button::State::new(),
debug: false,
}
}
@@ -55,56 +46,41 @@ impl Sandbox for Tour {
}
}
- fn view(&mut self) -> Element<Message> {
- let Tour {
- steps,
- scroll,
- back_button,
- next_button,
- ..
- } = self;
+ fn view(&self) -> Element<Message> {
+ let Tour { steps, .. } = self;
- let mut controls = Row::new();
+ let mut controls = row![];
if steps.has_previous() {
controls = controls.push(
- button(back_button, "Back")
+ button("Back")
.on_press(Message::BackPressed)
.style(theme::Button::Secondary),
);
}
- controls = controls.push(Space::with_width(Length::Fill));
+ controls = controls.push(horizontal_space(Length::Fill));
if steps.can_continue() {
controls = controls.push(
- button(next_button, "Next")
+ button("Next")
.on_press(Message::NextPressed)
.style(theme::Button::Primary),
);
}
- let content: Element<_> = Column::new()
- .max_width(540)
- .spacing(20)
- .padding(20)
- .push(steps.view(self.debug).map(Message::StepMessage))
- .push(controls)
- .into();
-
- let content = if self.debug {
- content.explain(Color::BLACK)
- } else {
- content
- };
+ let content = column![
+ steps.view(self.debug).map(Message::StepMessage),
+ controls,
+ ]
+ .max_width(540)
+ .spacing(20)
+ .padding(20);
- let scrollable = Scrollable::new(scroll)
- .push(Container::new(content).width(Length::Fill).center_x());
+ let scrollable =
+ scrollable(container(content).width(Length::Fill).center_x());
- Container::new(scrollable)
- .height(Length::Fill)
- .center_y()
- .into()
+ container(scrollable).height(Length::Fill).center_y().into()
}
}
@@ -125,35 +101,24 @@ impl Steps {
Steps {
steps: vec![
Step::Welcome,
- Step::Slider {
- state: slider::State::new(),
- value: 50,
- },
+ Step::Slider { value: 50 },
Step::RowsAndColumns {
layout: Layout::Row,
- spacing_slider: slider::State::new(),
spacing: 20,
},
Step::Text {
- size_slider: slider::State::new(),
size: 30,
- color_sliders: [slider::State::new(); 3],
- color: Color::from_rgb(0.5, 0.5, 0.5),
+ color: Color::BLACK,
},
Step::Radio { selection: None },
Step::Toggler {
can_continue: false,
},
- Step::Image {
- height: 200,
- current_fit: ContentFit::Contain,
- slider: slider::State::new(),
- },
+ Step::Image { width: 300 },
Step::Scrollable,
Step::TextInput {
value: String::new(),
is_secure: false,
- state: text_input::State::new(),
},
Step::Debugger,
Step::End,
@@ -166,7 +131,7 @@ impl Steps {
self.steps[self.current].update(msg, debug);
}
- fn view(&mut self, debug: bool) -> Element<StepMessage> {
+ fn view(&self, debug: bool) -> Element<StepMessage> {
self.steps[self.current].view(debug)
}
@@ -198,38 +163,14 @@ impl Steps {
enum Step {
Welcome,
- Slider {
- state: slider::State,
- value: u8,
- },
- RowsAndColumns {
- layout: Layout,
- spacing_slider: slider::State,
- spacing: u16,
- },
- Text {
- size_slider: slider::State,
- size: u16,
- color_sliders: [slider::State; 3],
- color: Color,
- },
- Radio {
- selection: Option<Language>,
- },
- Toggler {
- can_continue: bool,
- },
- Image {
- height: u16,
- slider: slider::State,
- current_fit: ContentFit,
- },
+ Slider { value: u8 },
+ RowsAndColumns { layout: Layout, spacing: u16 },
+ Text { size: u16, color: Color },
+ Radio { selection: Option<Language> },
+ Toggler { can_continue: bool },
+ Image { width: u16 },
Scrollable,
- TextInput {
- value: String,
- is_secure: bool,
- state: text_input::State,
- },
+ TextInput { value: String, is_secure: bool },
Debugger,
End,
}
@@ -242,8 +183,7 @@ pub enum StepMessage {
TextSizeChanged(u16),
TextColorChanged(Color),
LanguageSelected(Language),
- ImageHeightChanged(u16),
- ImageFitSelected(ContentFit),
+ ImageWidthChanged(u16),
InputChanged(String),
ToggleSecureInput(bool),
DebugToggled(bool),
@@ -288,14 +228,9 @@ impl<'a> Step {
*spacing = new_spacing;
}
}
- StepMessage::ImageHeightChanged(new_height) => {
- if let Step::Image { height, .. } = self {
- *height = new_height;
- }
- }
- StepMessage::ImageFitSelected(fit) => {
- if let Step::Image { current_fit, .. } = self {
- *current_fit = fit;
+ StepMessage::ImageWidthChanged(new_width) => {
+ if let Step::Image { width, .. } = self {
+ *width = new_width;
}
}
StepMessage::InputChanged(new_value) => {
@@ -348,34 +283,21 @@ impl<'a> Step {
}
}
- fn view(&mut self, debug: bool) -> Element<StepMessage> {
+ fn view(&self, debug: bool) -> Element<StepMessage> {
match self {
Step::Welcome => Self::welcome(),
Step::Radio { selection } => Self::radio(*selection),
Step::Toggler { can_continue } => Self::toggler(*can_continue),
- Step::Slider { state, value } => Self::slider(state, *value),
- Step::Text {
- size_slider,
- size,
- color_sliders,
- color,
- } => Self::text(size_slider, *size, color_sliders, *color),
- Step::Image {
- height,
- slider,
- current_fit,
- } => Self::image(*height, slider, *current_fit),
- Step::RowsAndColumns {
- layout,
- spacing_slider,
- spacing,
- } => Self::rows_and_columns(*layout, spacing_slider, *spacing),
+ Step::Slider { value } => Self::slider(*value),
+ Step::Text { size, color } => Self::text(*size, *color),
+ Step::Image { width } => Self::image(*width),
+ Step::RowsAndColumns { layout, spacing } => {
+ Self::rows_and_columns(*layout, *spacing)
+ }
Step::Scrollable => Self::scrollable(),
- Step::TextInput {
- value,
- is_secure,
- state,
- } => Self::text_input(value, *is_secure, state),
+ Step::TextInput { value, is_secure } => {
+ Self::text_input(value, *is_secure)
+ }
Step::Debugger => Self::debugger(debug),
Step::End => Self::end(),
}
@@ -383,59 +305,51 @@ impl<'a> Step {
}
fn container(title: &str) -> Column<'a, StepMessage> {
- Column::new().spacing(20).push(Text::new(title).size(50))
+ column![text(title).size(50)].spacing(20)
}
fn welcome() -> Column<'a, StepMessage> {
Self::container("Welcome!")
- .push(Text::new(
+ .push(
"This is a simple tour meant to showcase a bunch of widgets \
that can be easily implemented on top of Iced.",
- ))
- .push(Text::new(
+ )
+ .push(
"Iced is a cross-platform GUI library for Rust focused on \
simplicity and type-safety. It is heavily inspired by Elm.",
- ))
- .push(Text::new(
+ )
+ .push(
"It was originally born as part of Coffee, an opinionated \
2D game engine for Rust.",
- ))
- .push(Text::new(
+ )
+ .push(
"On native platforms, Iced provides by default a renderer \
built on top of wgpu, a graphics library supporting Vulkan, \
Metal, DX11, and DX12.",
- ))
- .push(Text::new(
+ )
+ .push(
"Additionally, this tour can also run on WebAssembly thanks \
to dodrio, an experimental VDOM library for Rust.",
- ))
- .push(Text::new(
+ )
+ .push(
"You will need to interact with the UI in order to reach the \
end!",
- ))
+ )
}
- fn slider(
- state: &'a mut slider::State,
- value: u8,
- ) -> Column<'a, StepMessage> {
+ fn slider(value: u8) -> Column<'a, StepMessage> {
Self::container("Slider")
- .push(Text::new(
+ .push(
"A slider allows you to smoothly select a value from a range \
of values.",
- ))
- .push(Text::new(
+ )
+ .push(
"The following slider lets you choose an integer from \
0 to 100:",
- ))
- .push(Slider::new(
- state,
- 0..=100,
- value,
- StepMessage::SliderChanged,
- ))
+ )
+ .push(slider(0..=100, value, StepMessage::SliderChanged))
.push(
- Text::new(value.to_string())
+ text(value.to_string())
.width(Length::Fill)
.horizontal_alignment(alignment::Horizontal::Center),
)
@@ -443,257 +357,197 @@ impl<'a> Step {
fn rows_and_columns(
layout: Layout,
- spacing_slider: &'a mut slider::State,
spacing: u16,
) -> Column<'a, StepMessage> {
- let row_radio = Radio::new(
- Layout::Row,
- "Row",
- Some(layout),
- StepMessage::LayoutChanged,
- );
+ let row_radio =
+ radio("Row", Layout::Row, Some(layout), StepMessage::LayoutChanged);
- let column_radio = Radio::new(
- Layout::Column,
+ let column_radio = radio(
"Column",
+ Layout::Column,
Some(layout),
StepMessage::LayoutChanged,
);
let layout_section: Element<_> = match layout {
- Layout::Row => Row::new()
- .spacing(spacing)
- .push(row_radio)
- .push(column_radio)
- .into(),
- Layout::Column => Column::new()
- .spacing(spacing)
- .push(row_radio)
- .push(column_radio)
- .into(),
+ Layout::Row => {
+ row![row_radio, column_radio].spacing(spacing).into()
+ }
+ Layout::Column => {
+ column![row_radio, column_radio].spacing(spacing).into()
+ }
};
- let spacing_section = Column::new()
- .spacing(10)
- .push(Slider::new(
- spacing_slider,
- 0..=80,
- spacing,
- StepMessage::SpacingChanged,
- ))
- .push(
- Text::new(format!("{} px", spacing))
- .width(Length::Fill)
- .horizontal_alignment(alignment::Horizontal::Center),
- );
+ let spacing_section = column![
+ slider(0..=80, spacing, StepMessage::SpacingChanged),
+ text(format!("{} px", spacing))
+ .width(Length::Fill)
+ .horizontal_alignment(alignment::Horizontal::Center),
+ ]
+ .spacing(10);
Self::container("Rows and columns")
.spacing(spacing)
- .push(Text::new(
+ .push(
"Iced uses a layout model based on flexbox to position UI \
elements.",
- ))
- .push(Text::new(
+ )
+ .push(
"Rows and columns can be used to distribute content \
horizontally or vertically, respectively.",
- ))
+ )
.push(layout_section)
- .push(Text::new(
- "You can also easily change the spacing between elements:",
- ))
+ .push("You can also easily change the spacing between elements:")
.push(spacing_section)
}
- fn text(
- size_slider: &'a mut slider::State,
- size: u16,
- color_sliders: &'a mut [slider::State; 3],
- color: Color,
- ) -> Column<'a, StepMessage> {
- let size_section = Column::new()
- .padding(20)
- .spacing(20)
- .push(Text::new("You can change its size:"))
- .push(Text::new(format!("This text is {} pixels", size)).size(size))
- .push(Slider::new(
- size_slider,
- 10..=70,
- size,
- StepMessage::TextSizeChanged,
- ));
-
- let [red, green, blue] = color_sliders;
-
- let color_sliders = Row::new()
- .spacing(10)
- .push(color_slider(red, color.r, move |r| Color { r, ..color }))
- .push(color_slider(green, color.g, move |g| Color { g, ..color }))
- .push(color_slider(blue, color.b, move |b| Color { b, ..color }));
-
- let color_section = Column::new()
- .padding(20)
- .spacing(20)
- .push(Text::new("And its color:"))
- .push(Text::new(format!("{:?}", color)).style(color))
- .push(color_sliders);
+ fn text(size: u16, color: Color) -> Column<'a, StepMessage> {
+ let size_section = column![
+ "You can change its size:",
+ text(format!("This text is {} pixels", size)).size(size),
+ slider(10..=70, size, StepMessage::TextSizeChanged),
+ ]
+ .padding(20)
+ .spacing(20);
+
+ let color_sliders = row![
+ color_slider(color.r, move |r| Color { r, ..color }),
+ color_slider(color.g, move |g| Color { g, ..color }),
+ color_slider(color.b, move |b| Color { b, ..color }),
+ ]
+ .spacing(10);
+
+ let color_section = column![
+ "And its color:",
+ text(format!("{:?}", color)).style(color),
+ color_sliders,
+ ]
+ .padding(20)
+ .spacing(20);
Self::container("Text")
- .push(Text::new(
+ .push(
"Text is probably the most essential widget for your UI. \
It will try to adapt to the dimensions of its container.",
- ))
+ )
.push(size_section)
.push(color_section)
}
fn radio(selection: Option<Language>) -> Column<'a, StepMessage> {
- let question = Column::new()
- .padding(20)
- .spacing(10)
- .push(Text::new("Iced is written in...").size(24))
- .push(Language::all().iter().cloned().fold(
- Column::new().padding(10).spacing(20),
- |choices, language| {
- choices.push(Radio::new(
- language,
- language,
- selection,
- StepMessage::LanguageSelected,
- ))
- },
- ));
+ let question = column![
+ text("Iced is written in...").size(24),
+ column(
+ Language::all()
+ .iter()
+ .cloned()
+ .map(|language| {
+ radio(
+ language,
+ language,
+ selection,
+ StepMessage::LanguageSelected,
+ )
+ })
+ .map(Element::from)
+ .collect()
+ )
+ ]
+ .padding(20)
+ .spacing(10);
Self::container("Radio button")
- .push(Text::new(
+ .push(
"A radio button is normally used to represent a choice... \
Surprise test!",
- ))
+ )
.push(question)
- .push(Text::new(
+ .push(
"Iced works very well with iterators! The list above is \
basically created by folding a column over the different \
choices, creating a radio button for each one of them!",
- ))
+ )
}
fn toggler(can_continue: bool) -> Column<'a, StepMessage> {
Self::container("Toggler")
- .push(Text::new(
- "A toggler is mostly used to enable or disable something.",
- ))
+ .push("A toggler is mostly used to enable or disable something.")
.push(
- Container::new(Toggler::new(
+ Container::new(toggler(
+ "Toggle me to continue...".to_owned(),
can_continue,
- String::from("Toggle me to continue..."),
StepMessage::TogglerChanged,
))
.padding([0, 40]),
)
}
- fn image(
- height: u16,
- slider: &'a mut slider::State,
- current_fit: ContentFit,
- ) -> Column<'a, StepMessage> {
- const FIT_MODES: [(ContentFit, &str); 3] = [
- (ContentFit::Contain, "Contain"),
- (ContentFit::Cover, "Cover"),
- (ContentFit::Fill, "Fill"),
- ];
-
- let mode_selector = FIT_MODES.iter().fold(
- Column::new().padding(10).spacing(20),
- |choices, (mode, name)| {
- choices.push(Radio::new(
- *mode,
- *name,
- Some(current_fit),
- StepMessage::ImageFitSelected,
- ))
- },
- );
-
+ fn image(width: u16) -> Column<'a, StepMessage> {
Self::container("Image")
- .push(Text::new("Pictures of things in all shapes and sizes!"))
- .push(ferris(height, current_fit))
- .push(Slider::new(
- slider,
- 50..=500,
- height,
- StepMessage::ImageHeightChanged,
- ))
+ .push("An image that tries to keep its aspect ratio.")
+ .push(ferris(width))
+ .push(slider(100..=500, width, StepMessage::ImageWidthChanged))
.push(
- Text::new(format!("Height: {} px", height))
+ text(format!("Width: {} px", width))
.width(Length::Fill)
.horizontal_alignment(alignment::Horizontal::Center),
)
- .push(Text::new("Pick a content fit strategy:"))
- .push(mode_selector)
}
fn scrollable() -> Column<'a, StepMessage> {
Self::container("Scrollable")
- .push(Text::new(
+ .push(
"Iced supports scrollable content. Try it out! Find the \
button further below.",
- ))
+ )
.push(
- Text::new(
- "Tip: You can use the scrollbar to scroll down faster!",
- )
- .size(16),
+ text("Tip: You can use the scrollbar to scroll down faster!")
+ .size(16),
)
- .push(Column::new().height(Length::Units(4096)))
+ .push(vertical_space(Length::Units(4096)))
.push(
- Text::new("You are halfway there!")
+ text("You are halfway there!")
.width(Length::Fill)
.size(30)
.horizontal_alignment(alignment::Horizontal::Center),
)
- .push(Column::new().height(Length::Units(4096)))
- .push(ferris(200, ContentFit::Contain))
+ .push(vertical_space(Length::Units(4096)))
+ .push(ferris(300))
.push(
- Text::new("You made it!")
+ text("You made it!")
.width(Length::Fill)
.size(50)
.horizontal_alignment(alignment::Horizontal::Center),
)
}
- fn text_input(
- value: &str,
- is_secure: bool,
- state: &'a mut text_input::State,
- ) -> Column<'a, StepMessage> {
- let text_input = TextInput::new(
- state,
+ fn text_input(value: &str, is_secure: bool) -> Column<'a, StepMessage> {
+ let text_input = text_input(
"Type something to continue...",
value,
StepMessage::InputChanged,
)
.padding(10)
.size(30);
+
Self::container("Text input")
- .push(Text::new(
- "Use a text input to ask for different kinds of information.",
- ))
+ .push("Use a text input to ask for different kinds of information.")
.push(if is_secure {
text_input.password()
} else {
text_input
})
- .push(Checkbox::new(
- is_secure,
+ .push(checkbox(
"Enable password mode",
+ is_secure,
StepMessage::ToggleSecureInput,
))
- .push(Text::new(
+ .push(
"A text input produces a message every time it changes. It is \
very easy to keep track of its contents:",
- ))
+ )
.push(
- Text::new(if value.is_empty() {
+ text(if value.is_empty() {
"You have not typed anything yet..."
} else {
value
@@ -705,79 +559,65 @@ impl<'a> Step {
fn debugger(debug: bool) -> Column<'a, StepMessage> {
Self::container("Debugger")
- .push(Text::new(
+ .push(
"You can ask Iced to visually explain the layouting of the \
different elements comprising your UI!",
- ))
- .push(Text::new(
+ )
+ .push(
"Give it a shot! Check the following checkbox to be able to \
see element boundaries.",
- ))
+ )
.push(if cfg!(target_arch = "wasm32") {
Element::new(
- Text::new("Not available on web yet!")
+ text("Not available on web yet!")
.style(Color::from([0.7, 0.7, 0.7]))
.horizontal_alignment(alignment::Horizontal::Center),
)
} else {
- Element::new(Checkbox::new(
- debug,
- "Explain layout",
- StepMessage::DebugToggled,
- ))
+ checkbox("Explain layout", debug, StepMessage::DebugToggled)
+ .into()
})
- .push(Text::new("Feel free to go back and take a look."))
+ .push("Feel free to go back and take a look.")
}
fn end() -> Column<'a, StepMessage> {
Self::container("You reached the end!")
- .push(Text::new(
- "This tour will be updated as more features are added.",
- ))
- .push(Text::new("Make sure to keep an eye on it!"))
+ .push("This tour will be updated as more features are added.")
+ .push("Make sure to keep an eye on it!")
}
}
-fn ferris<'a>(
- height: u16,
- content_fit: ContentFit,
-) -> Container<'a, StepMessage> {
- Container::new(
+fn ferris<'a>(width: u16) -> Container<'a, StepMessage> {
+ container(
// This should go away once we unify resource loading on native
// platforms
if cfg!(target_arch = "wasm32") {
- Image::new("tour/images/ferris.png")
+ image("tour/images/ferris.png")
} else {
- Image::new(format!(
- "{}/images/ferris.png",
- env!("CARGO_MANIFEST_DIR"),
+ image(format!(
+ "{}/../../tour/images/ferris.png",
+ env!("CARGO_MANIFEST_DIR")
))
}
- .height(Length::Units(height))
- .content_fit(content_fit),
+ .width(Length::Units(width)),
)
.width(Length::Fill)
.center_x()
}
-fn button<'a, Message: Clone>(
- state: &'a mut button::State,
- label: &str,
-) -> Button<'a, Message> {
- Button::new(
- state,
- Text::new(label).horizontal_alignment(alignment::Horizontal::Center),
+fn button<'a, Message: Clone>(label: &str) -> Button<'a, Message> {
+ iced::widget::button(
+ text(label).horizontal_alignment(alignment::Horizontal::Center),
)
.padding(12)
.width(Length::Units(100))
}
-fn color_slider(
- state: &mut slider::State,
+fn color_slider<'a>(
component: f32,
- update: impl Fn(f32) -> Color + 'static,
-) -> Slider<f64, StepMessage, iced::Renderer> {
- Slider::new(state, 0.0..=1.0, f64::from(component), move |c| {
+ update: impl Fn(f32) -> Color + 'a,
+) -> Slider<'a, f64, StepMessage, Renderer> {
+ slider(0.0..=1.0, f64::from(component), move |c| {
StepMessage::TextColorChanged(update(c as f32))
})
.step(0.01)