blue-recorder/core/src/utils.rs
2025-01-23 04:43:48 +02:00
Ask

212 lines
5.9 KiB
Rust

{52ad353ec4450b51264498b1fabc973bcf075581 true 6087 utils.rs 0xc001e19c00}

use anyhow::Result;
use std::process::Command;
// Select recording mode
#[derive(Clone, Copy)]
pub enum RecordMode {
Area,
Screen,
Window,
}
// Get audio output source
#[cfg(feature = "gtk")]
pub fn audio_output_source() -> Result<String> {
// Get the default sink
let default_sink_output = Command::new("pactl")
.arg("get-default-sink")
.output()?;
let default_sink = String::from_utf8_lossy(&default_sink_output.stdout)
.trim()
.to_string();
// List sinks and filter for the monitor of the default sink
let sinks_output = Command::new("pactl")
.arg("list")
.arg("sinks")
.output()?;
let sinks = String::from_utf8_lossy(&sinks_output.stdout);
let monitor_line = sinks
.lines()
.find(|line| line.contains(&format!("{}.monitor", default_sink)))
.unwrap_or("");
// Extract the part after the colon
let output_source = monitor_line.split(':')
.nth(1)
.unwrap_or("")
.trim().to_string();
Ok(output_source)
}
#[cfg(feature = "gtk")]
// Disable GtkWidget
pub fn disable_input_widgets(input_widgets: Vec<adw::gtk::Widget>) {
use adw::gtk::prelude::WidgetExt;
for widget in input_widgets {
widget.set_sensitive(false);
}
}
#[cfg(feature = "gtk")]
// Enable GtkWidget
pub fn enable_input_widgets(input_widgets: Vec<adw::gtk::Widget>) {
use adw::gtk::prelude::WidgetExt;
for widget in input_widgets {
widget.set_sensitive(true);
}
}
#[cfg(feature = "gtk")]
// Execute command after finish recording
pub fn exec(command: &str) -> Result<()> {
if !command.trim().is_empty() {
subprocess::Exec::shell(command.trim()).popen()?;
}
Ok(())
}
// Check if tmp input video file exist
pub fn is_input_audio_record(audio_filename: &str) -> bool {
std::path::Path::new(audio_filename).exists()
}
// Check if tmp output video file exist
pub fn is_output_audio_record(audio_filename: &str) -> bool {
std::path::Path::new(audio_filename).exists()
}
#[cfg(feature = "gtk")]
// Overwrite file if exists or not
pub fn is_overwrite(msg_bundle: &str, filename: &str, window: adw::Window) -> bool {
let is_file_already_exists = std::path::Path::new(filename).exists();
if is_file_already_exists {
let message_dialog = adw::gtk::MessageDialog::new(
Some(&window),
adw::gtk::DialogFlags::all(),
adw::gtk::MessageType::Warning,
adw::gtk::ButtonsType::YesNo,
msg_bundle,
);
let main_context = glib::MainContext::default();
use adw::prelude::*;
let answer = main_context.block_on(message_dialog.run_future());
message_dialog.close();
if answer != adw::gtk::ResponseType::Yes {
return false;
} else {
return true;
}
} else {
return true;
}
}
// Detect if snap package is used
pub fn is_snap() -> bool {
!std::env::var("SNAP").unwrap_or_default().is_empty()
}
// Validate audio/video file integrity
pub fn is_valid(filename: &str) -> Result<bool> {
let validate = Command::new("ffmpeg")
.args(["-v", "error",
"-i", filename,
"-c", "copy",
"-f", "null", "-"
]).output()?;
if validate.status.success() {
Ok(true)
} else {
Ok(false)
}
}
// Check if tmp video file exist
pub fn is_video_record(video_filename: &str) -> bool {
std::path::Path::new(video_filename).exists()
}
// Detect wayland session
pub fn is_wayland() -> bool {
std::env::var("XDG_SESSION_TYPE")
.unwrap_or_default()
.eq_ignore_ascii_case("wayland")
}
#[cfg(feature = "gtk")]
// Play recorded file
pub fn play_record(file_name: &str) -> Result<()> {
if is_snap() {
// open the video using snapctrl for snap package
Command::new("snapctl").arg("user-open")
.arg(file_name)
.spawn()?;
} else {
open::that(file_name)?;
}
Ok(())
}
#[cfg(feature = "gtk")]
// Get audio input source list
pub fn sources_descriptions_list() -> Result<Vec<String>> {
let sources_descriptions: Vec<String> = {
let list_sources_child = std::process::Command::new("pactl")
.args(["list", "sources"])
.stdout(std::process::Stdio::piped())
.spawn();
let sources_descriptions = String::from_utf8(if let Ok(..) = list_sources_child {
std::process::Command::new("grep")
.args(["-e", "device.description"])
.stdin(list_sources_child?.stdout.take()
.ok_or_else(|| anyhow::anyhow!("Failed to get audio input source descriptions."))?)
.output()?
.stdout
} else {
Vec::new()
})?;
sources_descriptions
.split('\n')
.map(|s| {
s.trim()
.replace("device.description = ", "")
.replace('\"', "")
})
.filter(|s| !s.is_empty())
.collect()
};
Ok(sources_descriptions)
}
#[cfg(feature = "gtk")]
// Validate video file
pub fn validate_video_file(filename: String) -> Result<()> {
let start_time = std::time::Instant::now();
let duration = std::time::Duration::from_secs(300);
let main_loop = glib::MainLoop::new(None, false);
let filename_clone = filename.clone();
let _main_loop = main_loop.clone();
async_std::task::spawn_blocking(move || {
while std::time::Instant::now().duration_since(start_time) < duration {
if is_valid(&filename_clone).unwrap_or(false) {
_main_loop.quit();
return Ok(());
}
// Sleep a bit to avoid busy waiting
std::thread::sleep(std::time::Duration::from_millis(100));
}
Err(anyhow::Error::msg("Unable to validate tmp video file."))
});
main_loop.run();
Ok(())
}