use crate::common::{x11::Frame, TraitCapturer}; use crate::wayland::{capturable::*, *}; use std::{io, sync::RwLock, time::Duration}; pub struct Capturer(Display, Box, bool, Vec); pub static mut IS_CURSOR_EMBEDDED: bool = true; lazy_static::lazy_static! { static ref MAP_ERR: RwLock io::Error>> = Default::default(); } pub fn detect_cursor_embedded() { if unsafe { IS_CURSOR_EMBEDDED } { use crate::common::wayland::pipewire::get_available_cursor_modes; match get_available_cursor_modes() { Ok(modes) => unsafe { IS_CURSOR_EMBEDDED = (modes & 0x02) > 0; }, Err(..) => unsafe { IS_CURSOR_EMBEDDED = false; }, } } } pub fn set_map_err(f: fn(err: String) -> io::Error) { *MAP_ERR.write().unwrap() = Some(f); } fn map_err(err: E) -> io::Error { if let Some(f) = *MAP_ERR.read().unwrap() { f(err.to_string()) } else { io::Error::new(io::ErrorKind::Other, err.to_string()) } } impl Capturer { pub fn new(display: Display, yuv: bool) -> io::Result { let r = display.0.recorder(false).map_err(map_err)?; Ok(Capturer(display, r, yuv, Default::default())) } pub fn width(&self) -> usize { self.0.width() } pub fn height(&self) -> usize { self.0.height() } } impl TraitCapturer for Capturer { fn set_use_yuv(&mut self, use_yuv: bool) { self.2 = use_yuv; } fn frame<'a>(&'a mut self, timeout: Duration) -> io::Result> { match self.1.capture(timeout.as_millis() as _).map_err(map_err)? { PixelProvider::BGR0(w, h, x) => Ok(Frame(if self.2 { crate::common::bgra_to_i420(w as _, h as _, &x, &mut self.3); &self.3[..] } else { x })), PixelProvider::RGB0(w, h, x) => Ok(Frame(if self.2 { crate::common::rgba_to_i420(w as _, h as _, &x, &mut self.3); &self.3[..] } else { x })), PixelProvider::NONE => Err(std::io::ErrorKind::WouldBlock.into()), _ => Err(map_err("Invalid data")), } } } pub struct Display(pipewire::PipeWireCapturable); impl Display { pub fn primary() -> io::Result { let mut all = Display::all()?; if all.is_empty() { return Err(io::ErrorKind::NotFound.into()); } Ok(all.remove(0)) } pub fn all() -> io::Result> { Ok(pipewire::get_capturables(unsafe { IS_CURSOR_EMBEDDED }) .map_err(map_err)? .drain(..) .map(|x| Display(x)) .collect()) } pub fn width(&self) -> usize { self.0.size.0 } pub fn height(&self) -> usize { self.0.size.1 } pub fn origin(&self) -> (i32, i32) { self.0.position } pub fn is_online(&self) -> bool { true } pub fn is_primary(&self) -> bool { false } pub fn name(&self) -> String { "".to_owned() } }