use std::os::unix::io::FromRawFd;
use std::time::Duration;
use serde::Deserialize;
use futures::StreamExt;
use tokio::fs::File;
use tokio::io::{AsyncBufReadExt, BufReader};
use crate::click::MouseButton;
use crate::BoxedStream;
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct I3BarEvent {
pub id: usize,
pub instance: Option<String>,
pub button: MouseButton,
}
fn unprocessed_events_stream(invert_scrolling: bool) -> BoxedStream<I3BarEvent> {
let stdin = unsafe { File::from_raw_fd(0) };
let lines = BufReader::new(stdin).lines();
futures::stream::unfold(lines, move |mut lines| async move {
loop {
let line = lines.next_line().await.ok().flatten()?;
let line = line.trim_start_matches(|c| c != '{');
let line = line.trim_end_matches(|c| c != '}');
if line.is_empty() {
continue;
}
#[derive(Deserialize)]
struct I3BarEventRaw {
instance: Option<String>,
button: MouseButton,
}
let event: I3BarEventRaw = match serde_json::from_str(line) {
Ok(event) => event,
Err(err) => {
eprintln!("Failed to deserialize click event.\nData: {line}\nError: {err}");
continue;
}
};
let (id, instance) = match event.instance {
Some(name) => {
let (id, instance) = name.split_once(':').unwrap();
let instance = if instance.is_empty() {
None
} else {
Some(instance.to_owned())
};
(id.parse().unwrap(), instance)
}
None => continue,
};
use MouseButton::*;
let button = match (event.button, invert_scrolling) {
(WheelUp, false) | (WheelDown, true) => WheelUp,
(WheelUp, true) | (WheelDown, false) => WheelDown,
(other, _) => other,
};
let event = I3BarEvent {
id,
instance,
button,
};
break Some((event, lines));
}
})
.boxed_local()
}
pub fn events_stream(
invert_scrolling: bool,
double_click_delay: Duration,
) -> BoxedStream<I3BarEvent> {
let events = unprocessed_events_stream(invert_scrolling);
futures::stream::unfold((events, None), move |(mut events, pending)| async move {
if let Some(pending) = pending {
return Some((pending, (events, None)));
}
let mut event = events.next().await?;
if event.button == MouseButton::Left && !double_click_delay.is_zero() {
if let Ok(new_event) = tokio::time::timeout(double_click_delay, events.next()).await {
let new_event = new_event?;
if event == new_event {
event.button = MouseButton::DoubleLeft;
} else {
return Some((event, (events, Some(new_event))));
}
}
}
Some((event, (events, None)))
})
.fuse()
.boxed_local()
}