zero-trust-rps 0.0.5

Online Multiplayer Rock Paper Scissors
Documentation
use std::{future::Future, pin::pin, time::Duration};

use crossterm::event::{Event as CrosstermEvent, KeyEvent, MouseEvent};
use futures::{FutureExt as _, StreamExt as _};
use tokio::sync::mpsc;

use crate::common::client::{connection::read::SimplifiedServerMessage, state::ClientStateView};

/// Terminal events.
#[derive(Clone, Debug)]
pub enum Event {
    /// Terminal tick.
    Tick,
    /// Key press.
    Key(KeyEvent),
    /// Mouse click/scroll.
    Mouse(MouseEvent),
    /// Terminal resize.
    #[expect(unused)]
    Resize(u16, u16),
    /// Message from the server
    #[expect(clippy::enum_variant_names)]
    ServerEvent(ServerEvent),
    // New client state
    ClientStateUpdated(ClientStateView),
}

#[derive(Clone, Debug)]
pub enum ServerEvent {
    Other,
    Error(String),
}

impl From<SimplifiedServerMessage> for Event {
    fn from(value: SimplifiedServerMessage) -> Self {
        Event::ServerEvent(match value {
            SimplifiedServerMessage::NewRoomState(_) => ServerEvent::Other,
            SimplifiedServerMessage::Pong(_) => ServerEvent::Other,
            SimplifiedServerMessage::Error(error) => ServerEvent::Error(error),
        })
    }
}

impl From<ClientStateView> for Event {
    fn from(value: ClientStateView) -> Self {
        Event::ClientStateUpdated(value)
    }
}

/// Terminal event handler.
pub struct EventHandler {
    /// Event sender channel.
    sender: mpsc::UnboundedSender<Event>,
    /// Event receiver channel.
    receiver: mpsc::UnboundedReceiver<Event>,
    /// Event handler thread.
    #[expect(unused)]
    handler: tokio::task::JoinHandle<()>,
}

impl EventHandler {
    /// Constructs a new instance of [`EventHandler`].
    pub fn new(tick_rate: Duration) -> Self {
        let (sender, receiver) = mpsc::unbounded_channel();
        let _sender = sender.clone();
        let handler = tokio::task::spawn(async move {
            let mut reader = crossterm::event::EventStream::new();
            let mut tick = tokio::time::interval(tick_rate);

            loop {
                let mut crossterm_event = reader.next().fuse();
                let mut tick_delay = pin!(tick.tick().fuse());

                // _ = _sender.closed() => {
                //   break;
                // }
                futures::select! {
                  _ = tick_delay => {
                    _sender.send(Event::Tick).unwrap();
                  },
                  evt = crossterm_event => {
                        if let Some(Ok(evt)) = evt {
                            match evt {
                                CrosstermEvent::Key(key) => {
                                    if key.kind == crossterm::event::KeyEventKind::Press {
                                        _sender.send(Event::Key(key)).unwrap();
                                    }
                                },
                                CrosstermEvent::Mouse(mouse) => {
                                    _sender.send(Event::Mouse(mouse)).unwrap();
                                },
                                CrosstermEvent::Resize(x, y) => {
                                    _sender.send(Event::Resize(x, y)).unwrap();
                                },
                                CrosstermEvent::FocusLost => {
                                },
                                CrosstermEvent::FocusGained => {
                                },
                                CrosstermEvent::Paste(_) => {
                                },
                            }
                        }
                  }
                };
            }
        });
        Self {
            sender,
            receiver,
            handler,
        }
    }

    #[expect(clippy::needless_lifetimes)]
    pub fn next<'a>(&'a mut self) -> impl Future<Output = Option<Event>> + 'a {
        self.receiver.recv()
    }

    pub fn get_sender(&self) -> mpsc::UnboundedSender<Event> {
        self.sender.clone()
    }
}