From e0bdb203f2596a65b6ce8c6fb939f82a6dc24048 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?H=C3=A9ctor=20Ram=C3=B3n=20Jim=C3=A9nez?= Date: Sun, 17 Nov 2019 07:02:38 +0100 Subject: [PATCH 1/6] Implement future-based `Command` in `iced_core` --- core/Cargo.toml | 7 +++++++ core/src/command.rs | 49 +++++++++++++++++++++++++++++++++++++++++++++ core/src/lib.rs | 4 ++++ native/Cargo.toml | 2 +- native/src/lib.rs | 2 +- 5 files changed, 62 insertions(+), 2 deletions(-) create mode 100644 core/src/command.rs diff --git a/core/Cargo.toml b/core/Cargo.toml index a244bcba4c..f24923456d 100644 --- a/core/Cargo.toml +++ b/core/Cargo.toml @@ -6,3 +6,10 @@ edition = "2018" description = "The essential concepts of Iced" license = "MIT" repository = "https://github.com/hecrj/iced" + +[features] +# Exposes a future-based `Command` type +command = ["futures"] + +[dependencies] +futures = { version = "0.3", optional = true } diff --git a/core/src/command.rs b/core/src/command.rs new file mode 100644 index 0000000000..ae901bd833 --- /dev/null +++ b/core/src/command.rs @@ -0,0 +1,49 @@ +use futures::future::{BoxFuture, Future, FutureExt}; + +pub struct Command { + futures: Vec>, +} + +impl Command { + pub fn none() -> Self { + Self { + futures: Vec::new(), + } + } + + pub fn attempt( + future: impl Future + 'static + Send, + f: impl Fn(T) -> A + 'static + Send, + ) -> Command { + Command { + futures: vec![future.map(f).boxed()], + } + } + + pub fn batch(commands: impl Iterator>) -> Self { + Self { + futures: commands.flat_map(|command| command.futures).collect(), + } + } + + pub fn futures(self) -> Vec> { + self.futures + } +} + +impl From for Command +where + A: Future + 'static + Send, +{ + fn from(future: A) -> Self { + Self { + futures: vec![future.boxed()], + } + } +} + +impl std::fmt::Debug for Command { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + f.debug_struct("Command").finish() + } +} diff --git a/core/src/lib.rs b/core/src/lib.rs index b61f2eae00..3816f8a2e8 100644 --- a/core/src/lib.rs +++ b/core/src/lib.rs @@ -3,6 +3,8 @@ pub mod widget; mod align; mod background; mod color; +#[cfg(feature = "command")] +mod command; mod font; mod length; mod point; @@ -12,6 +14,8 @@ mod vector; pub use align::Align; pub use background::Background; pub use color::Color; +#[cfg(feature = "command")] +pub use command::Command; pub use font::Font; pub use length::Length; pub use point::Point; diff --git a/native/Cargo.toml b/native/Cargo.toml index 38db1610ca..5dc3ae1afa 100644 --- a/native/Cargo.toml +++ b/native/Cargo.toml @@ -8,6 +8,6 @@ license = "MIT" repository = "https://github.com/hecrj/iced" [dependencies] -iced_core = { version = "0.1.0-alpha", path = "../core" } +iced_core = { version = "0.1.0-alpha", path = "../core", features = ["command"] } twox-hash = "1.5" raw-window-handle = "0.3" diff --git a/native/src/lib.rs b/native/src/lib.rs index 7e55064c51..bd03ddcd49 100644 --- a/native/src/lib.rs +++ b/native/src/lib.rs @@ -216,7 +216,7 @@ mod size; mod user_interface; pub use iced_core::{ - Align, Background, Color, Font, Length, Point, Rectangle, Vector, + Align, Background, Color, Command, Font, Length, Point, Rectangle, Vector, }; pub use element::Element; From e640b875900a3833fd38efa195e99b40ec3f6820 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?H=C3=A9ctor=20Ram=C3=B3n=20Jim=C3=A9nez?= Date: Sun, 17 Nov 2019 07:03:22 +0100 Subject: [PATCH 2/6] Derive `Clone` for `text_input::State` --- core/src/widget/text_input.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/src/widget/text_input.rs b/core/src/widget/text_input.rs index 450a7caef0..16c67954a1 100644 --- a/core/src/widget/text_input.rs +++ b/core/src/widget/text_input.rs @@ -80,7 +80,7 @@ where } } -#[derive(Debug, Default)] +#[derive(Debug, Default, Clone)] pub struct State { pub is_focused: bool, cursor_position: usize, From 02c20e6202f1c8c28753f3233cc635790707937a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?H=C3=A9ctor=20Ram=C3=B3n=20Jim=C3=A9nez?= Date: Sun, 17 Nov 2019 07:09:46 +0100 Subject: [PATCH 3/6] Support async actions in `iced_winit` --- examples/scroll.rs | 14 ++++++--- examples/todos.rs | 14 ++++++--- examples/tour.rs | 36 ++++++++++++---------- src/lib.rs | 22 +++++++++----- src/winit.rs | 4 +-- winit/Cargo.toml | 1 + winit/src/application.rs | 66 +++++++++++++++++++++++++++++++--------- 7 files changed, 110 insertions(+), 47 deletions(-) diff --git a/examples/scroll.rs b/examples/scroll.rs index 5070187991..61ad2a5339 100644 --- a/examples/scroll.rs +++ b/examples/scroll.rs @@ -1,12 +1,12 @@ use iced::{ - button, scrollable, Align, Application, Button, Container, Element, Image, - Length, Scrollable, Text, + button, scrollable, Align, Application, Button, Command, Container, + Element, Image, Length, Scrollable, Text, }; pub fn main() { env_logger::init(); - Example::default().run() + Example::run() } #[derive(Default)] @@ -25,16 +25,22 @@ pub enum Message { impl Application for Example { type Message = Message; + fn new() -> (Example, Command) { + (Example::default(), Command::none()) + } + fn title(&self) -> String { String::from("Scroll - Iced") } - fn update(&mut self, message: Message) { + fn update(&mut self, message: Message) -> Command { match message { Message::AddItem => { self.item_count += 1; } } + + Command::none() } fn view(&mut self) -> Element { diff --git a/examples/todos.rs b/examples/todos.rs index f921a666a6..d97a9e0830 100644 --- a/examples/todos.rs +++ b/examples/todos.rs @@ -1,11 +1,11 @@ use iced::{ button, scrollable, text::HorizontalAlignment, text_input, Align, - Application, Background, Button, Checkbox, Color, Column, Container, - Element, Font, Length, Row, Scrollable, Text, TextInput, + Application, Background, Button, Checkbox, Color, Column, Command, + Container, Element, Font, Length, Row, Scrollable, Text, TextInput, }; pub fn main() { - Todos::default().run() + Todos::run() } #[derive(Debug, Default)] @@ -29,11 +29,15 @@ pub enum Message { impl Application for Todos { type Message = Message; + fn new() -> (Todos, Command) { + (Todos::default(), Command::none()) + } + fn title(&self) -> String { String::from("Todos - Iced") } - fn update(&mut self, message: Message) { + fn update(&mut self, message: Message) -> Command { match message { Message::InputChanged(value) => { self.input_value = value; @@ -58,6 +62,8 @@ impl Application for Todos { } dbg!(self); + + Command::none() } fn view(&mut self) -> Element { diff --git a/examples/tour.rs b/examples/tour.rs index 34ad0a343a..6d7a080f8d 100644 --- a/examples/tour.rs +++ b/examples/tour.rs @@ -1,13 +1,14 @@ use iced::{ button, scrollable, slider, text::HorizontalAlignment, text_input, - Application, Background, Button, Checkbox, Color, Column, Container, - Element, Image, Length, Radio, Row, Scrollable, Slider, Text, TextInput, + Application, Background, Button, Checkbox, Color, Column, Command, + Container, Element, Image, Length, Radio, Row, Scrollable, Slider, Text, + TextInput, }; pub fn main() { env_logger::init(); - Tour::new().run() + Tour::run() } pub struct Tour { @@ -18,26 +19,27 @@ pub struct Tour { debug: bool, } -impl Tour { - pub fn new() -> Tour { - Tour { - steps: Steps::new(), - scroll: scrollable::State::new(), - back_button: button::State::new(), - next_button: button::State::new(), - debug: true, - } - } -} - impl Application for Tour { type Message = Message; + fn new() -> (Tour, Command) { + ( + Tour { + steps: Steps::new(), + scroll: scrollable::State::new(), + back_button: button::State::new(), + next_button: button::State::new(), + debug: true, + }, + Command::none(), + ) + } + fn title(&self) -> String { format!("{} - Iced", self.steps.title()) } - fn update(&mut self, event: Message) { + fn update(&mut self, event: Message) -> Command { match event { Message::BackPressed => { self.steps.go_back(); @@ -49,6 +51,8 @@ impl Application for Tour { self.steps.update(step_msg, &mut self.debug); } } + + Command::none() } fn view(&mut self) -> Element { diff --git a/src/lib.rs b/src/lib.rs index 52b0ff8c02..945af42106 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -4,21 +4,23 @@ mod platform; pub use platform::*; -pub trait Application { - type Message: std::fmt::Debug; +pub trait Application: Sized { + type Message: std::fmt::Debug + Send; + + fn new() -> (Self, Command); fn title(&self) -> String; - fn update(&mut self, message: Self::Message); + fn update(&mut self, message: Self::Message) -> Command; fn view(&mut self) -> Element; - fn run(self) + fn run() where Self: 'static + Sized, { #[cfg(not(target_arch = "wasm32"))] - iced_winit::Application::run(Instance(self)); + as iced_winit::Application>::run(); #[cfg(target_arch = "wasm32")] iced_web::Application::run(Instance(self)); @@ -35,12 +37,18 @@ where type Renderer = Renderer; type Message = A::Message; + fn new() -> (Self, Command) { + let (app, command) = A::new(); + + (Instance(app), command) + } + fn title(&self) -> String { self.0.title() } - fn update(&mut self, message: Self::Message) { - self.0.update(message); + fn update(&mut self, message: Self::Message) -> Command { + self.0.update(message) } fn view(&mut self) -> Element { diff --git a/src/winit.rs b/src/winit.rs index d35a339f8a..c869a269ad 100644 --- a/src/winit.rs +++ b/src/winit.rs @@ -2,8 +2,8 @@ pub use iced_wgpu::{Primitive, Renderer}; pub use iced_winit::{ button, scrollable, slider, text, text_input, winit, Align, Background, - Checkbox, Color, Font, Image, Length, Radio, Scrollable, Slider, Text, - TextInput, + Checkbox, Color, Command, Font, Image, Length, Radio, Scrollable, Slider, + Text, TextInput, }; pub type Element<'a, Message> = iced_winit::Element<'a, Message, Renderer>; diff --git a/winit/Cargo.toml b/winit/Cargo.toml index 2831ba2f01..2a33255deb 100644 --- a/winit/Cargo.toml +++ b/winit/Cargo.toml @@ -13,4 +13,5 @@ debug = [] [dependencies] iced_native = { version = "0.1.0-alpha", path = "../native" } winit = { version = "0.20.0-alpha3", git = "https://github.com/rust-windowing/winit", rev = "709808eb4e69044705fcb214bcc30556db761405"} +futures = { version = "0.3", features = ["thread-pool"] } log = "0.4" diff --git a/winit/src/application.rs b/winit/src/application.rs index 331bafa0cf..bf41d0c8c6 100644 --- a/winit/src/application.rs +++ b/winit/src/application.rs @@ -2,24 +2,26 @@ use crate::{ conversion, input::{keyboard, mouse}, renderer::{Target, Windowed}, - Cache, Container, Debug, Element, Event, Length, MouseCursor, + Cache, Command, Container, Debug, Element, Event, Length, MouseCursor, UserInterface, }; -pub trait Application { +pub trait Application: Sized { type Renderer: Windowed; - type Message: std::fmt::Debug; + type Message: std::fmt::Debug + Send; + + fn new() -> (Self, Command); fn title(&self) -> String; - fn update(&mut self, message: Self::Message); + fn update(&mut self, message: Self::Message) -> Command; fn view(&mut self) -> Element; - fn run(mut self) + fn run() where - Self: 'static + Sized, + Self: 'static, { use winit::{ event::{self, WindowEvent}, @@ -28,10 +30,18 @@ pub trait Application { }; let mut debug = Debug::new(); - let mut title = self.title(); debug.startup_started(); - let event_loop = EventLoop::new(); + let event_loop = EventLoop::with_user_event(); + let proxy = event_loop.create_proxy(); + let mut thread_pool = + futures::executor::ThreadPool::new().expect("Create thread pool"); + let mut external_messages = Vec::new(); + + let (mut application, init_command) = Self::new(); + spawn(init_command, &mut thread_pool, &proxy); + + let mut title = application.title(); // TODO: Ask for window settings and configure this properly let window = WindowBuilder::new() @@ -59,7 +69,7 @@ pub trait Application { debug.layout_started(); let user_interface = UserInterface::build( - document(&mut self, size, &mut debug), + document(&mut application, size, &mut debug), Cache::default(), &mut renderer, ); @@ -85,15 +95,16 @@ pub trait Application { // handled. debug.layout_started(); let mut user_interface = UserInterface::build( - document(&mut self, size, &mut debug), + document(&mut application, size, &mut debug), cache.take().unwrap(), &mut renderer, ); debug.layout_finished(); debug.event_processing_started(); - let messages = + let mut messages = user_interface.update(&renderer, events.drain(..)); + messages.extend(external_messages.drain(..)); debug.event_processing_finished(); if messages.is_empty() { @@ -113,12 +124,14 @@ pub trait Application { debug.log_message(&message); debug.update_started(); - self.update(message); + let command = application.update(message); + + spawn(command, &mut thread_pool, &proxy); debug.update_finished(); } // Update window title - let new_title = self.title(); + let new_title = application.title(); if title != new_title { window.set_title(&new_title); @@ -128,7 +141,7 @@ pub trait Application { debug.layout_started(); let user_interface = UserInterface::build( - document(&mut self, size, &mut debug), + document(&mut application, size, &mut debug), temp_cache, &mut renderer, ); @@ -143,6 +156,9 @@ pub trait Application { window.request_redraw(); } + event::Event::UserEvent(message) => { + external_messages.push(message); + } event::Event::RedrawRequested(_) => { debug.render_started(); @@ -288,3 +304,25 @@ where .height(Length::Units(size.height.round() as u16)) .into() } + +fn spawn( + command: Command, + thread_pool: &mut futures::executor::ThreadPool, + proxy: &winit::event_loop::EventLoopProxy, +) { + use futures::FutureExt; + + let futures = command.futures(); + + for future in futures { + let proxy = proxy.clone(); + + let future = future.map(move |message| { + proxy + .send_event(message) + .expect("Send command result to event loop"); + }); + + thread_pool.spawn_ok(future); + } +} From b2392d28bd7bf968b43cbdb6ce973160a275a681 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?H=C3=A9ctor=20Ram=C3=B3n=20Jim=C3=A9nez?= Date: Sun, 17 Nov 2019 07:38:01 +0100 Subject: [PATCH 4/6] Save `todos` async in a JSON file and load on boot --- Cargo.toml | 3 + examples/todos.rs | 340 ++++++++++++++++++++++++++++++++++------------ 2 files changed, 257 insertions(+), 86 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index 8f9769b26f..8766d8143e 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -36,6 +36,9 @@ iced_web = { version = "0.1.0-alpha", path = "web" } [dev-dependencies] env_logger = "0.7" +serde = { version = "1.0", features = ["derive"] } +serde_json = "1.0" +directories = "2.0" [target.'cfg(target_arch = "wasm32")'.dev-dependencies] wasm-bindgen = "0.2.51" diff --git a/examples/todos.rs b/examples/todos.rs index d97a9e0830..2cdf1ea1a6 100644 --- a/examples/todos.rs +++ b/examples/todos.rs @@ -3,23 +3,34 @@ use iced::{ Application, Background, Button, Checkbox, Color, Column, Command, Container, Element, Font, Length, Row, Scrollable, Text, TextInput, }; +use serde::{Deserialize, Serialize}; pub fn main() { Todos::run() } +#[derive(Debug)] +enum Todos { + Loading, + Loaded(State), +} + #[derive(Debug, Default)] -struct Todos { +struct State { scroll: scrollable::State, input: text_input::State, input_value: String, filter: Filter, tasks: Vec, controls: Controls, + dirty: bool, + saving: bool, } #[derive(Debug, Clone)] -pub enum Message { +enum Message { + Loaded(Result), + Saved(Result<(), SaveError>), InputChanged(String), CreateTask, FilterChanged(Filter), @@ -30,113 +41,175 @@ impl Application for Todos { type Message = Message; fn new() -> (Todos, Command) { - (Todos::default(), Command::none()) + (Todos::Loading, Command::attempt(load(), Message::Loaded)) } fn title(&self) -> String { - String::from("Todos - Iced") + let dirty = match self { + Todos::Loading => false, + Todos::Loaded(state) => state.dirty, + }; + + format!("Todos{} - Iced", if dirty { "*" } else { "" }) } fn update(&mut self, message: Message) -> Command { - match message { - Message::InputChanged(value) => { - self.input_value = value; - } - Message::CreateTask => { - if !self.input_value.is_empty() { - self.tasks.push(Task::new(self.input_value.clone())); - self.input_value.clear(); + match self { + Todos::Loading => { + match message { + Message::Loaded(Ok(state)) => { + *self = Todos::Loaded(State { + input_value: state.input_value, + filter: state.filter, + tasks: state.tasks, + ..State::default() + }); + } + Message::Loaded(Err(_)) => { + *self = Todos::Loaded(State::default()); + } + _ => {} } + + Command::none() } - Message::FilterChanged(filter) => { - self.filter = filter; - } - Message::TaskMessage(i, TaskMessage::Delete) => { - self.tasks.remove(i); - } - Message::TaskMessage(i, task_message) => { - if let Some(task) = self.tasks.get_mut(i) { - task.update(task_message); + Todos::Loaded(state) => { + let mut saved = false; + + match message { + Message::InputChanged(value) => { + state.input_value = value; + } + Message::CreateTask => { + if !state.input_value.is_empty() { + state + .tasks + .push(Task::new(state.input_value.clone())); + state.input_value.clear(); + } + } + Message::FilterChanged(filter) => { + state.filter = filter; + } + Message::TaskMessage(i, TaskMessage::Delete) => { + state.tasks.remove(i); + } + Message::TaskMessage(i, task_message) => { + if let Some(task) = state.tasks.get_mut(i) { + task.update(task_message); + } + } + Message::Saved(_) => { + state.saving = false; + saved = true; + } + _ => {} } - } - } - dbg!(self); + if !saved { + state.dirty = true; + } - Command::none() + if state.dirty && !state.saving { + state.dirty = false; + state.saving = true; + + Command::attempt( + save(SavedState { + input_value: state.input_value.clone(), + filter: state.filter, + tasks: state.tasks.clone(), + }), + Message::Saved, + ) + } else { + Command::none() + } + } + } } fn view(&mut self) -> Element { - let Todos { - scroll, - input, - input_value, - filter, - tasks, - controls, - } = self; - - let title = Text::new("todos") - .size(100) - .color([0.5, 0.5, 0.5]) - .horizontal_alignment(HorizontalAlignment::Center); - - let input = TextInput::new( - input, - "What needs to be done?", - input_value, - Message::InputChanged, - ) - .padding(15) - .size(30) - .on_submit(Message::CreateTask); - - let controls = controls.view(&tasks, *filter); - let filtered_tasks = tasks.iter().filter(|task| filter.matches(task)); - - let tasks: Element<_> = - if filtered_tasks.count() > 0 { - tasks - .iter_mut() - .enumerate() - .filter(|(_, task)| filter.matches(task)) - .fold(Column::new().spacing(20), |column, (i, task)| { - column.push(task.view().map(move |message| { - Message::TaskMessage(i, message) - })) + match self { + Todos::Loading => loading_message(), + Todos::Loaded(State { + scroll, + input, + input_value, + filter, + tasks, + controls, + .. + }) => { + let title = Text::new("todos") + .size(100) + .color([0.5, 0.5, 0.5]) + .horizontal_alignment(HorizontalAlignment::Center); + + let input = TextInput::new( + input, + "What needs to be done?", + input_value, + Message::InputChanged, + ) + .padding(15) + .size(30) + .on_submit(Message::CreateTask); + + let controls = controls.view(&tasks, *filter); + let filtered_tasks = + tasks.iter().filter(|task| filter.matches(task)); + + let tasks: Element<_> = if filtered_tasks.count() > 0 { + tasks + .iter_mut() + .enumerate() + .filter(|(_, task)| filter.matches(task)) + .fold(Column::new().spacing(20), |column, (i, task)| { + column.push(task.view().map(move |message| { + Message::TaskMessage(i, message) + })) + }) + .into() + } else { + empty_message(match filter { + Filter::All => "You have not created a task yet...", + Filter::Active => "All your tasks are done! :D", + Filter::Completed => { + "You have not completed a task yet..." + } }) - .into() - } else { - empty_message(match filter { - Filter::All => "You have not created a task yet...", - Filter::Active => "All your tasks are done! :D", - Filter::Completed => "You have not completed a task yet...", - }) - }; + }; - let content = Column::new() - .max_width(800) - .spacing(20) - .push(title) - .push(input) - .push(controls) - .push(tasks); - - Scrollable::new(scroll) - .padding(40) - .push(Container::new(content).width(Length::Fill).center_x()) - .into() + let content = Column::new() + .max_width(800) + .spacing(20) + .push(title) + .push(input) + .push(controls) + .push(tasks); + + Scrollable::new(scroll) + .padding(40) + .push( + Container::new(content).width(Length::Fill).center_x(), + ) + .into() + } + } } } -#[derive(Debug)] +#[derive(Debug, Clone, Serialize, Deserialize)] struct Task { description: String, completed: bool, + + #[serde(skip)] state: TaskState, } -#[derive(Debug)] +#[derive(Debug, Clone)] pub enum TaskState { Idle { edit_button: button::State, @@ -147,6 +220,14 @@ pub enum TaskState { }, } +impl Default for TaskState { + fn default() -> Self { + TaskState::Idle { + edit_button: button::State::new(), + } + } +} + #[derive(Debug, Clone)] pub enum TaskMessage { Completed(bool), @@ -255,7 +336,7 @@ impl Task { } } -#[derive(Debug, Default)] +#[derive(Debug, Default, Clone)] pub struct Controls { all_button: button::State, active_button: button::State, @@ -324,7 +405,7 @@ impl Controls { } } -#[derive(Debug, Clone, Copy, PartialEq, Eq)] +#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize)] pub enum Filter { All, Active, @@ -347,6 +428,18 @@ impl Filter { } } +fn loading_message() -> Element<'static, Message> { + Container::new( + Text::new("Loading...") + .horizontal_alignment(HorizontalAlignment::Center) + .size(50), + ) + .width(Length::Fill) + .height(Length::Fill) + .center_y() + .into() +} + fn empty_message(message: &str) -> Element<'static, Message> { Container::new( Text::new(message) @@ -381,3 +474,78 @@ fn edit_icon() -> Text { fn delete_icon() -> Text { icon('\u{F1F8}') } + +// Persistence +#[derive(Debug, Clone, Serialize, Deserialize)] +struct SavedState { + input_value: String, + filter: Filter, + tasks: Vec, +} + +fn save_path() -> std::path::PathBuf { + let mut path = if let Some(project_dirs) = + directories::ProjectDirs::from("rs", "Iced", "Todos") + { + project_dirs.data_dir().into() + } else { + std::env::current_dir() + .expect("The current directory is not accessible") + }; + + path.push("todos.json"); + + path +} + +#[derive(Debug, Clone)] +enum LoadError { + FileError, + FormatError, +} + +async fn load() -> Result { + use std::io::Read; + + let mut contents = String::new(); + + let mut file = + std::fs::File::open(save_path()).map_err(|_| LoadError::FileError)?; + + file.read_to_string(&mut contents) + .map_err(|_| LoadError::FileError)?; + + serde_json::from_str(&contents).map_err(|_| LoadError::FormatError) +} + +#[derive(Debug, Clone)] +enum SaveError { + DirectoryError, + FileError, + WriteError, + FormatError, +} + +async fn save(state: SavedState) -> Result<(), SaveError> { + use std::io::Write; + + let json = serde_json::to_string_pretty(&state) + .map_err(|_| SaveError::FormatError)?; + + let save_path = save_path(); + let save_dir = save_path.parent().ok_or(SaveError::DirectoryError)?; + + std::fs::create_dir_all(save_dir).map_err(|_| SaveError::DirectoryError)?; + + let mut file = + std::fs::File::create(save_path).map_err(|_| SaveError::FileError)?; + + file.write_all(json.as_bytes()) + .map_err(|_| SaveError::WriteError)?; + + // This is a simple way to save at most once every couple seconds + // We will be able to get rid of it once we implement event subscriptions + std::thread::sleep(std::time::Duration::from_secs(2)); + + Ok(()) +} From a803ab240b9b5d1a073454f814229bad9d8bdafc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?H=C3=A9ctor=20Ram=C3=B3n=20Jim=C3=A9nez?= Date: Mon, 18 Nov 2019 00:10:40 +0100 Subject: [PATCH 5/6] Rename `Command::attempt` to `Command::perform` --- core/src/command.rs | 2 +- examples/todos.rs | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/core/src/command.rs b/core/src/command.rs index ae901bd833..e1c865ddb4 100644 --- a/core/src/command.rs +++ b/core/src/command.rs @@ -11,7 +11,7 @@ impl Command { } } - pub fn attempt( + pub fn perform( future: impl Future + 'static + Send, f: impl Fn(T) -> A + 'static + Send, ) -> Command { diff --git a/examples/todos.rs b/examples/todos.rs index 2cdf1ea1a6..05142fb2e3 100644 --- a/examples/todos.rs +++ b/examples/todos.rs @@ -41,7 +41,7 @@ impl Application for Todos { type Message = Message; fn new() -> (Todos, Command) { - (Todos::Loading, Command::attempt(load(), Message::Loaded)) + (Todos::Loading, Command::perform(load(), Message::Loaded)) } fn title(&self) -> String { @@ -114,7 +114,7 @@ impl Application for Todos { state.dirty = false; state.saving = true; - Command::attempt( + Command::perform( save(SavedState { input_value: state.input_value.clone(), filter: state.filter, From 63dbf078fefea444073813e834c2d35fa25eb3a7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?H=C3=A9ctor=20Ram=C3=B3n=20Jim=C3=A9nez?= Date: Mon, 18 Nov 2019 00:13:18 +0100 Subject: [PATCH 6/6] Improve persistence in `todos` --- examples/todos.rs | 100 ++++++++++++++++++++++++---------------------- 1 file changed, 53 insertions(+), 47 deletions(-) diff --git a/examples/todos.rs b/examples/todos.rs index 05142fb2e3..a1dfc5aa5f 100644 --- a/examples/todos.rs +++ b/examples/todos.rs @@ -41,7 +41,10 @@ impl Application for Todos { type Message = Message; fn new() -> (Todos, Command) { - (Todos::Loading, Command::perform(load(), Message::Loaded)) + ( + Todos::Loading, + Command::perform(SavedState::load(), Message::Loaded), + ) } fn title(&self) -> String { @@ -115,11 +118,12 @@ impl Application for Todos { state.saving = true; Command::perform( - save(SavedState { + SavedState { input_value: state.input_value.clone(), filter: state.filter, tasks: state.tasks.clone(), - }), + } + .save(), Message::Saved, ) } else { @@ -483,41 +487,12 @@ struct SavedState { tasks: Vec, } -fn save_path() -> std::path::PathBuf { - let mut path = if let Some(project_dirs) = - directories::ProjectDirs::from("rs", "Iced", "Todos") - { - project_dirs.data_dir().into() - } else { - std::env::current_dir() - .expect("The current directory is not accessible") - }; - - path.push("todos.json"); - - path -} - #[derive(Debug, Clone)] enum LoadError { FileError, FormatError, } -async fn load() -> Result { - use std::io::Read; - - let mut contents = String::new(); - - let mut file = - std::fs::File::open(save_path()).map_err(|_| LoadError::FileError)?; - - file.read_to_string(&mut contents) - .map_err(|_| LoadError::FileError)?; - - serde_json::from_str(&contents).map_err(|_| LoadError::FormatError) -} - #[derive(Debug, Clone)] enum SaveError { DirectoryError, @@ -526,26 +501,57 @@ enum SaveError { FormatError, } -async fn save(state: SavedState) -> Result<(), SaveError> { - use std::io::Write; +impl SavedState { + fn path() -> std::path::PathBuf { + let mut path = if let Some(project_dirs) = + directories::ProjectDirs::from("rs", "Iced", "Todos") + { + project_dirs.data_dir().into() + } else { + std::env::current_dir() + .expect("The current directory is not accessible") + }; - let json = serde_json::to_string_pretty(&state) - .map_err(|_| SaveError::FormatError)?; + path.push("todos.json"); - let save_path = save_path(); - let save_dir = save_path.parent().ok_or(SaveError::DirectoryError)?; + path + } + + async fn load() -> Result { + use std::io::Read; + + let mut contents = String::new(); - std::fs::create_dir_all(save_dir).map_err(|_| SaveError::DirectoryError)?; + let mut file = std::fs::File::open(Self::path()) + .map_err(|_| LoadError::FileError)?; + + file.read_to_string(&mut contents) + .map_err(|_| LoadError::FileError)?; + + serde_json::from_str(&contents).map_err(|_| LoadError::FormatError) + } - let mut file = - std::fs::File::create(save_path).map_err(|_| SaveError::FileError)?; + async fn save(self) -> Result<(), SaveError> { + use std::io::Write; - file.write_all(json.as_bytes()) - .map_err(|_| SaveError::WriteError)?; + let json = serde_json::to_string_pretty(&self) + .map_err(|_| SaveError::FormatError)?; - // This is a simple way to save at most once every couple seconds - // We will be able to get rid of it once we implement event subscriptions - std::thread::sleep(std::time::Duration::from_secs(2)); + let path = Self::path(); + let dir = path.parent().ok_or(SaveError::DirectoryError)?; - Ok(()) + std::fs::create_dir_all(dir).map_err(|_| SaveError::DirectoryError)?; + + let mut file = + std::fs::File::create(path).map_err(|_| SaveError::FileError)?; + + file.write_all(json.as_bytes()) + .map_err(|_| SaveError::WriteError)?; + + // This is a simple way to save at most once every couple seconds + // We will be able to get rid of it once we implement event subscriptions + std::thread::sleep(std::time::Duration::from_secs(2)); + + Ok(()) + } }