use crate::common::{x11::Frame, TraitCapturer}; use crate::wayland::{capturable::*, *}; use std::{io, time::Duration}; pub struct Capturer(Display, Box, bool, Vec); fn map_err(err: E) -> io::Error { 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::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(false) .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() } }