1
0
mirror of https://github.com/xlmnxp/blue-recorder.git synced 2025-04-03 16:24:54 +03:00
blue-recorder/src/ffmpeg_interface.rs
2022-10-18 15:03:17 +03:00
Ask

348 lines
12 KiB
Rust

{3202e21ff72d82be0c256924959c33fc97b7643a true 12004 ffmpeg_interface.rs 0xc0030ba4d0}

extern crate subprocess;
use chrono::prelude::*;
use gettextrs::gettext;
use gio::File;
use gtk::prelude::*;
use gtk::{Button, CheckButton, ComboBoxText, Entry, ProgressBar, SpinButton, Window};
use gtk::{ButtonsType, DialogFlags, MessageDialog, MessageType, ResponseType};
use std::path::PathBuf;
use std::process::Command;
use std::sync::mpsc::Sender;
use std::thread::sleep;
use std::time::Duration;
use subprocess::Exec;
#[derive(Clone)]
pub struct ProgressWidget {
pub progress_dialog: MessageDialog,
pub progressbar: ProgressBar,
pub progress_button: Button,
}
impl ProgressWidget {
pub fn new(
progress_dialog: MessageDialog,
progressbar: ProgressBar,
progress_button: Button,
) -> ProgressWidget {
ProgressWidget {
progress_dialog,
progressbar,
progress_button,
}
}
pub fn set_progress(&self, title: String, value: i32, max: i32) {
let progress_precentage: f64 = value as f64 / max as f64;
self.progressbar.set_text(Some(&title));
self.progressbar.set_fraction(progress_precentage);
}
pub fn show(&self) {
self.progressbar.set_fraction(0.0);
self.progress_dialog.show();
}
pub fn hide(&self) {
self.progress_button.set_sensitive(true);
self.progress_dialog.hide();
}
}
#[derive(Clone)]
pub struct Ffmpeg {
pub filename: (File, Entry, ComboBoxText),
pub record_video: CheckButton,
pub record_audio: CheckButton,
pub audio_id: ComboBoxText,
pub record_mouse: CheckButton,
pub follow_mouse: CheckButton,
pub record_frames: SpinButton,
pub record_delay: SpinButton,
pub command: Entry,
pub video_process_id: Option<u32>,
pub audio_process_id: Option<u32>,
pub saved_filename: Option<String>,
pub unbound: Option<Sender<bool>>,
pub progress_widget: ProgressWidget,
pub window: Window,
pub overwrite: CheckButton,
}
impl Ffmpeg {
pub fn start_record(
&mut self,
x: u16,
y: u16,
width: u16,
height: u16,
) -> (Option<u32>, Option<u32>) {
self.saved_filename = Some(
self.filename
.0
.path()
.unwrap()
.join(PathBuf::from(format!(
"{}.{}",
if self.filename.1.text().to_string().trim().eq("") {
Utc::now().to_string().replace(" UTC", "").replace(' ', "-")
} else {
self.filename.1.text().to_string().trim().to_string()
},
self.filename.2.active_id().unwrap()
)))
.as_path()
.display()
.to_string(),
);
let is_file_already_exists =
std::path::Path::new(&self.saved_filename.clone().unwrap()).exists();
if !self.overwrite.is_active() && is_file_already_exists {
let message_dialog = MessageDialog::new(
Some(&self.window),
DialogFlags::empty(),
MessageType::Warning,
ButtonsType::Ok,
&gettext("File already exist."),
);
message_dialog.show();
message_dialog.connect_response(
glib::clone!(@strong message_dialog => move |_, response| {
if response == ResponseType::Ok {
message_dialog.hide();
}
message_dialog.hide();
}),
);
return (None, None);
}
if self.record_audio.is_active() {
let mut ffmpeg_command = Command::new("ffmpeg");
ffmpeg_command.arg("-f");
ffmpeg_command.arg("pulse");
ffmpeg_command.arg("-i");
ffmpeg_command.arg(&self.audio_id.active_id().unwrap());
ffmpeg_command.arg("-f");
ffmpeg_command.arg("ogg");
ffmpeg_command.arg(format!(
"{}.temp.audio",
self.saved_filename.as_ref().unwrap()
));
ffmpeg_command.arg("-y");
self.audio_process_id = Some(ffmpeg_command.spawn().unwrap().id());
}
if self.record_video.is_active() {
let mut ffmpeg_command: Command = Command::new("ffmpeg");
// record video with specified width and hight
ffmpeg_command.arg("-video_size");
ffmpeg_command.arg(format!("{}x{}", width, height));
ffmpeg_command.arg("-framerate");
ffmpeg_command.arg(format!("{}", self.record_frames.value()));
ffmpeg_command.arg("-f");
ffmpeg_command.arg("x11grab");
ffmpeg_command.arg("-i");
ffmpeg_command.arg(format!(
"{}+{},{}",
std::env::var("DISPLAY")
.unwrap_or_else(|_| ":0".to_string())
.as_str(),
x,
y
));
// if show mouse switch is enabled, draw the mouse to video
ffmpeg_command.arg("-draw_mouse");
if self.record_mouse.is_active() {
ffmpeg_command.arg("1");
} else {
ffmpeg_command.arg("0");
}
// if follow mouse switch is enabled, follow the mouse
if self.follow_mouse.is_active() {
ffmpeg_command.arg("-follow_mouse");
ffmpeg_command.arg("centered");
}
ffmpeg_command.arg("-crf");
ffmpeg_command.arg("1");
ffmpeg_command.arg(self.saved_filename.as_ref().unwrap());
ffmpeg_command.arg("-y");
// sleep for delay
sleep(Duration::from_secs(self.record_delay.value() as u64));
// start recording and return the process id
self.video_process_id = Some(ffmpeg_command.spawn().unwrap().id());
return (self.video_process_id, self.audio_process_id);
}
(None, None)
}
pub fn stop_record(&self) {
self.progress_widget.show();
// kill the process to stop recording
self.progress_widget.set_progress("".to_string(), 1, 6);
if self.video_process_id.is_some() {
self.progress_widget
.set_progress("Stop Recording Video".to_string(), 1, 6);
Command::new("kill")
.arg(format!("{}", self.video_process_id.unwrap()))
.output()
.unwrap();
}
self.progress_widget.set_progress("".to_string(), 2, 6);
if self.audio_process_id.is_some() {
self.progress_widget
.set_progress("Stop Recording Audio".to_string(), 2, 6);
Command::new("kill")
.arg(format!("{}", self.audio_process_id.unwrap()))
.output()
.unwrap();
}
let is_video_record = std::path::Path::new(
format!(
"{}{}",
self.saved_filename.as_ref().unwrap_or(&String::from("")),
{ "" }
)
.as_str(),
)
.exists();
let is_audio_record = std::path::Path::new(
format!(
"{}.temp.audio",
self.saved_filename.as_ref().unwrap_or(&String::from(""))
)
.as_str(),
)
.exists();
if is_video_record {
let mut move_command = Command::new("mv");
move_command.arg(format!("{}{}", self.saved_filename.as_ref().unwrap(), {
""
}));
move_command.arg(format!(
"{}{}",
self.saved_filename.as_ref().unwrap_or(&String::new()),
if is_audio_record {
format!(
".temp.without.audio.{}",
self.filename.2.active_id().unwrap()
)
} else {
"".to_string()
}
));
move_command.output().unwrap();
self.progress_widget.set_progress("".to_string(), 4, 6);
// if audio record, then merge video with audio
if is_audio_record {
self.progress_widget
.set_progress("Save Audio Recording".to_string(), 4, 6);
let mut ffmpeg_audio_merge_command = Command::new("ffmpeg");
ffmpeg_audio_merge_command.arg("-i");
ffmpeg_audio_merge_command.arg(format!(
"{}.temp.without.audio.{}",
self.saved_filename.as_ref().unwrap(),
self.filename.2.active_id().unwrap()
));
ffmpeg_audio_merge_command.arg("-i");
ffmpeg_audio_merge_command.arg(format!(
"{}.temp.audio",
self.saved_filename.as_ref().unwrap()
));
ffmpeg_audio_merge_command.arg("-c:v");
ffmpeg_audio_merge_command.arg("copy");
ffmpeg_audio_merge_command.arg("-c:a");
ffmpeg_audio_merge_command.arg("aac");
ffmpeg_audio_merge_command.arg(self.saved_filename.as_ref().unwrap());
ffmpeg_audio_merge_command.arg("-y");
sleep(Duration::from_secs(1));
ffmpeg_audio_merge_command.output().unwrap();
std::fs::remove_file(format!(
"{}.temp.audio",
self.saved_filename.as_ref().unwrap()
))
.unwrap();
std::fs::remove_file(format!(
"{}.temp.without.audio.{}",
self.saved_filename.as_ref().unwrap(),
self.filename.2.active_id().unwrap()
))
.unwrap();
}
}
// if only audio is recording then convert it to chosen format
else if is_audio_record {
self.progress_widget
.set_progress("Convert Audio to choosen format".to_string(), 4, 6);
sleep(Duration::from_secs(1));
Command::new("ffmpeg")
.arg("-f")
.arg("ogg")
.arg("-i")
.arg(format!(
"{}.temp.audio",
self.saved_filename.as_ref().unwrap()
))
.arg(self.saved_filename.as_ref().unwrap())
.output()
.unwrap();
std::fs::remove_file(format!(
"{}.temp.audio",
self.saved_filename.as_ref().unwrap()
))
.unwrap();
}
self.progress_widget.set_progress("".to_string(), 5, 6);
// execute command after finish recording
if self.command.text().trim() != "" {
self.progress_widget.set_progress(
"execute custom command after finish".to_string(),
5,
6,
);
Exec::shell(self.command.text().trim()).popen().unwrap();
}
self.progress_widget
.set_progress("Finish".to_string(), 6, 6);
self.progress_widget.hide();
}
pub fn play_record(self) {
if self.saved_filename.is_some() {
if is_snap() {
// open the video using snapctrl for snap package
Command::new("snapctl")
.arg("user-open")
.arg(self.saved_filename.unwrap())
.spawn()
.unwrap();
} else {
Command::new("xdg-open")
.arg(self.saved_filename.unwrap())
.spawn()
.unwrap();
}
}
}
}
fn is_snap() -> bool {
!std::env::var("SNAP").unwrap_or_default().is_empty()
}