mirror of
https://github.com/xlmnxp/blue-recorder.git
synced 2025-04-03 08:14:55 +03:00
231 lines
7.4 KiB
Rust
231 lines
7.4 KiB
Rust
{6da6b2f8624c5f418d2db0fac0a00ae72a334913 true 7616 area_capture.rs 0xc001dea3f0}
extern crate regex;
|
|
|
|
#[cfg(any(target_os = "freebsd", target_os = "linux"))]
|
|
use anyhow::anyhow;
|
|
use anyhow::Result;
|
|
#[cfg(target_os = "windows")]
|
|
use display_info::DisplayInfo;
|
|
use glib::Continue;
|
|
use libadwaita::gtk::Label;
|
|
use libadwaita::Window;
|
|
use libadwaita::prelude::*;
|
|
#[cfg(any(target_os = "freebsd", target_os = "linux"))]
|
|
use regex::Regex;
|
|
use std::cell::RefCell;
|
|
#[cfg(any(target_os = "freebsd", target_os = "linux"))]
|
|
use std::process::Command;
|
|
use std::rc::Rc;
|
|
#[cfg(target_os = "windows")]
|
|
use x_win::get_active_window;
|
|
|
|
// This struct use "xwininfo" in linux & freebsd to get area x, y, width and height
|
|
#[derive(Debug, Copy, Clone)]
|
|
pub struct AreaCapture {
|
|
pub x: u16,
|
|
pub y: u16,
|
|
pub width: u16,
|
|
pub height: u16,
|
|
}
|
|
|
|
#[derive(Debug, Clone)]
|
|
pub struct Title {
|
|
pub title: String,
|
|
}
|
|
|
|
impl AreaCapture {
|
|
pub fn new() -> Result<AreaCapture> {
|
|
#[cfg(any(target_os = "freebsd", target_os = "linux"))]
|
|
let coordinate = xwininfo_to_coordinate(
|
|
String::from_utf8(Command::new("xwininfo").arg("-root").output()?.stdout)?
|
|
)?;
|
|
|
|
#[cfg(any(target_os = "freebsd", target_os = "linux"))]
|
|
let area_capture = AreaCapture {
|
|
x: coordinate.0,
|
|
y: coordinate.1,
|
|
width: coordinate.2,
|
|
height: coordinate.3,
|
|
};
|
|
|
|
#[cfg(target_os = "windows")]
|
|
let coordinate = DisplayInfo::all()?;
|
|
|
|
#[cfg(target_os = "windows")]
|
|
let area_capture = AreaCapture {
|
|
x: coordinate[0].x as u16,
|
|
y: coordinate[0].y as u16,
|
|
width: coordinate[0].width as u16,
|
|
height: coordinate[0].height as u16,
|
|
};
|
|
Ok(area_capture)
|
|
}
|
|
|
|
#[cfg(target_os = "windows")]
|
|
pub fn get_active_window(&mut self) -> Result<Self> {
|
|
let coordinate = get_active_window()?.position;
|
|
|
|
self.x = coordinate.x as u16;
|
|
self.y = coordinate.y as u16;
|
|
self.width = coordinate.width as u16;
|
|
self.height = coordinate.height as u16;
|
|
Ok(*self)
|
|
}
|
|
|
|
#[cfg(any(target_os = "freebsd", target_os = "linux"))]
|
|
pub fn get_area(&mut self) -> Result<Self> {
|
|
let coordinate = xwininfo_to_coordinate(
|
|
String::from_utf8(Command::new("xwininfo").output()?.stdout)?
|
|
)?;
|
|
self.x = coordinate.0;
|
|
self.y = coordinate.1;
|
|
self.width = coordinate.2;
|
|
self.height = coordinate.3;
|
|
Ok(*self)
|
|
}
|
|
|
|
#[cfg(any(target_os = "freebsd", target_os = "linux"))]
|
|
pub fn get_window_by_name(&mut self, name: &str) -> Result<Self> {
|
|
let coordinate = xwininfo_to_coordinate(
|
|
String::from_utf8(Command::new("xwininfo").arg("-name").arg(name).output()?.stdout)?,
|
|
)?;
|
|
self.x = coordinate.0;
|
|
self.y = coordinate.1;
|
|
self.width = coordinate.2;
|
|
self.height = coordinate.3;
|
|
Ok(*self)
|
|
}
|
|
|
|
pub fn reset(&mut self) -> Result<Self> {
|
|
#[cfg(target_os = "windows")]
|
|
{
|
|
let coordinate = DisplayInfo::all()?;
|
|
self.x = coordinate[0].x as u16;
|
|
self.y = coordinate[0].y as u16;
|
|
self.width = coordinate[0].width as u16;
|
|
self.height = coordinate[0].height as u16;
|
|
}
|
|
|
|
#[cfg(any(target_os = "freebsd", target_os = "linux"))]
|
|
{
|
|
let coordinate = xwininfo_to_coordinate(
|
|
String::from_utf8(Command::new("xwininfo").arg("-root").output()?.stdout)?
|
|
)?;
|
|
self.x = coordinate.0;
|
|
self.y = coordinate.1;
|
|
self.width = coordinate.2;
|
|
self.height = coordinate.3;
|
|
}
|
|
|
|
Ok(*self)
|
|
}
|
|
}
|
|
|
|
#[cfg(target_os = "windows")]
|
|
impl Title {
|
|
pub fn new() -> Result<Title> {
|
|
let title = Title {
|
|
title: String::new(),
|
|
};
|
|
Ok(title)
|
|
}
|
|
|
|
pub fn get_title(&mut self) -> Result<Self> {
|
|
self.title = get_active_window()?.title;
|
|
Ok(self.clone())
|
|
}
|
|
}
|
|
|
|
#[cfg(any(target_os = "freebsd", target_os = "linux"))]
|
|
fn xwininfo_to_coordinate(xwininfo_output: String) -> Result<(u16, u16, u16, u16)> {
|
|
let x: u16 = Regex::new(r"A.*X:\s+(\d+)\n")?
|
|
.captures(xwininfo_output.as_str())
|
|
.ok_or_else(|| anyhow!("Failed to capture string from xwininfo_output."))?
|
|
.get(1)
|
|
.ok_or_else(|| anyhow!("Failed to get x value from xwininfo_output."))?
|
|
.as_str()
|
|
.to_string()
|
|
.parse::<u16>()?;
|
|
let y: u16 = Regex::new(r"A.*Y:\s+(\d+)\n")?
|
|
.captures(xwininfo_output.as_str())
|
|
.ok_or_else(|| anyhow!("Failed to capture string from xwininfo_output."))?
|
|
.get(1)
|
|
.ok_or_else(|| anyhow!("Failed to get y value from xwininfo_output."))?
|
|
.as_str()
|
|
.to_string()
|
|
.parse::<u16>()?;
|
|
let width: u16 = Regex::new(r"Width:\s(\d+)\n")?
|
|
.captures(xwininfo_output.as_str())
|
|
.ok_or_else(|| anyhow!("Failed to capture string from xwininfo_output."))?
|
|
.get(1)
|
|
.ok_or_else(|| anyhow!("Failed to get width value from xwininfo_output."))?
|
|
.as_str()
|
|
.to_string()
|
|
.parse::<u16>()?;
|
|
let height: u16 = Regex::new(r"Height:\s(\d+)\n")?
|
|
.captures(xwininfo_output.as_str())
|
|
.ok_or_else(|| anyhow!("Failed to capture string from xwininfo_output."))?
|
|
.get(1)
|
|
.ok_or_else(|| anyhow!("Failed to get height value from xwininfo_output."))?
|
|
.as_str()
|
|
.to_string()
|
|
.parse::<u16>()?;
|
|
|
|
Ok((x, y, width, height))
|
|
}
|
|
|
|
// Display area chooser window size
|
|
pub fn show_size(area_chooser_window: Window, area_size_bottom_label: Label, area_size_top_label: Label) -> Result<()> {
|
|
// Create a shared state for the area size
|
|
let size_labels = Rc::new(RefCell::new((area_size_top_label, area_size_bottom_label)));
|
|
|
|
// Use a timeout to periodically check the window size
|
|
glib::timeout_add_local(1000, {
|
|
let area_chooser_window = area_chooser_window.clone();
|
|
let size_labels = size_labels.clone();
|
|
|
|
move || {
|
|
if !area_chooser_window.is_active() {
|
|
return Continue(false); // Stop the timeout
|
|
}
|
|
|
|
let mut area_capture = AreaCapture::new().unwrap();
|
|
#[cfg(any(target_os = "freebsd", target_os = "linux"))]
|
|
let size = area_capture.get_window_by_name(
|
|
area_chooser_window.title().unwrap().as_str()
|
|
).unwrap();
|
|
|
|
#[cfg(target_os = "windows")]
|
|
let size = area_capture.get_active_window().unwrap();
|
|
|
|
// Update the labels
|
|
let (top_label, bottom_label) = size_labels.borrow_mut().to_owned();
|
|
top_label.set_text(&format!("{}x{}", size.width, size.height));
|
|
bottom_label.set_text(&format!("{}x{}", size.width, size.height));
|
|
|
|
Continue(true) // Continue the timeout
|
|
}
|
|
});
|
|
|
|
Ok(())
|
|
}
|
|
|
|
// Returns `true` if the left mouse button is clicked, and `false` if the `Esc` key is pressed.
|
|
#[cfg(target_os = "windows")]
|
|
pub fn check_input() -> bool {
|
|
use winapi::um::winuser::{GetAsyncKeyState, VK_LBUTTON, VK_ESCAPE};
|
|
loop {
|
|
// Check if the left mouse button (VK_LBUTTON) is pressed
|
|
if unsafe { GetAsyncKeyState(VK_LBUTTON) } & 0x8000u16 as i16 != 0 {
|
|
break true;
|
|
}
|
|
|
|
// Check if the Esc key is pressed
|
|
if unsafe { GetAsyncKeyState(VK_ESCAPE) } & 0x8000u16 as i16 != 0 {
|
|
break false;
|
|
}
|
|
|
|
// Add a small delay to avoid busy-waiting
|
|
std::thread::sleep(std::time::Duration::from_millis(10));
|
|
}
|
|
}
|