summaryrefslogblamecommitdiffstats
path: root/examples/clock.rs
blob: b1ee8ab1a5c30217a869bb622ae6661e251d9a69 (plain) (tree)





































































































































































                                                                               
use iced::{
    Align, Application, Checkbox, Column, Command, Container, Element, Length,
    Settings, Subscription, Text,
};

pub fn main() {
    Clock::run(Settings::default())
}

#[derive(Debug)]
struct Clock {
    time: chrono::DateTime<chrono::Local>,
    enabled: bool,
}

#[derive(Debug, Clone)]
enum Message {
    Ticked(chrono::DateTime<chrono::Local>),
    Toggled(bool),
}

impl Application for Clock {
    type Message = Message;

    fn new() -> (Clock, Command<Message>) {
        (
            Clock {
                time: chrono::Local::now(),
                enabled: false,
            },
            Command::none(),
        )
    }

    fn title(&self) -> String {
        String::from("Clock - Iced")
    }

    fn update(&mut self, message: Message) -> Command<Message> {
        match message {
            Message::Ticked(time) => {
                self.time = time;
            }
            Message::Toggled(enabled) => {
                self.enabled = enabled;
            }
        };

        Command::none()
    }

    fn subscriptions(&self) -> Subscription<Message> {
        if self.enabled {
            time::every(std::time::Duration::from_millis(500), Message::Ticked)
        } else {
            Subscription::none()
        }
    }

    fn view(&mut self) -> Element<Message> {
        let clock = Text::new(format!("{}", self.time.format("%H:%M:%S")))
            .size(40)
            .width(Length::Shrink);

        let toggle = Checkbox::new(self.enabled, "Enabled", Message::Toggled)
            .width(Length::Shrink);

        let content = Column::new()
            .width(Length::Shrink)
            .align_items(Align::Center)
            .spacing(20)
            .push(clock)
            .push(toggle);

        Container::new(content)
            .width(Length::Fill)
            .height(Length::Fill)
            .center_x()
            .center_y()
            .into()
    }
}

mod time {
    use std::sync::{Arc, Mutex};

    pub fn every<Message>(
        duration: std::time::Duration,
        f: impl Fn(chrono::DateTime<chrono::Local>) -> Message
            + 'static
            + Send
            + Sync,
    ) -> iced::Subscription<Message>
    where
        Message: Send + 'static,
    {
        Tick {
            duration,
            message: Arc::new(f),
        }
        .into()
    }

    struct Tick<Message> {
        duration: std::time::Duration,
        message: Arc<
            dyn Fn(chrono::DateTime<chrono::Local>) -> Message + Send + Sync,
        >,
    }

    struct TickState {
        alive: Arc<Mutex<bool>>,
    }

    impl iced::subscription::Handle for TickState {
        fn cancel(&mut self) {
            match self.alive.lock() {
                Ok(mut guard) => *guard = false,
                _ => {}
            }
        }
    }

    impl<Message> iced::subscription::Definition for Tick<Message>
    where
        Message: 'static,
    {
        type Message = Message;

        fn id(&self) -> u64 {
            0
        }

        fn stream(
            &self,
        ) -> (
            futures::stream::BoxStream<'static, Message>,
            Box<dyn iced::subscription::Handle>,
        ) {
            use futures::StreamExt;

            let duration = self.duration.clone();
            let function = self.message.clone();
            let alive = Arc::new(Mutex::new(true));

            let state = TickState {
                alive: alive.clone(),
            };

            let stream = futures::stream::poll_fn(move |_| {
                std::thread::sleep(duration);

                if !*alive.lock().unwrap() {
                    return std::task::Poll::Ready(None);
                }

                let now = chrono::Local::now();

                std::task::Poll::Ready(Some(now))
            })
            .map(move |time| function(time));

            (stream.boxed(), Box::new(state))
        }
    }
}