mirror of
https://github.com/xlmnxp/blue-recorder.git
synced 2025-04-05 01:04:54 +03:00
split audio from video recording in Xorg then merge it with video
This commit is contained in:
parent
4e70f62b6b
commit
e033addfaf
@ -1,7 +1,10 @@
|
|||||||
extern crate subprocess;
|
extern crate subprocess;
|
||||||
use chrono::prelude::*;
|
use chrono::prelude::*;
|
||||||
use gtk::prelude::*;
|
use gtk::prelude::*;
|
||||||
use gtk::{CheckButton, ComboBoxText, Entry, FileChooser, ProgressBar, SpinButton, Window, WindowType, WindowPosition};
|
use gtk::{
|
||||||
|
CheckButton, ComboBoxText, Entry, FileChooser, ProgressBar, SpinButton, Window, WindowPosition,
|
||||||
|
WindowType,
|
||||||
|
};
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
use std::process::Command;
|
use std::process::Command;
|
||||||
@ -96,15 +99,22 @@ pub struct Ffmpeg {
|
|||||||
pub record_frames: SpinButton,
|
pub record_frames: SpinButton,
|
||||||
pub record_delay: SpinButton,
|
pub record_delay: SpinButton,
|
||||||
pub command: Entry,
|
pub command: Entry,
|
||||||
pub process_id: Option<u32>,
|
pub video_process_id: Option<u32>,
|
||||||
|
pub audio_process_id: Option<u32>,
|
||||||
pub saved_filename: Option<String>,
|
pub saved_filename: Option<String>,
|
||||||
pub unbound: Option<Sender<bool>>,
|
pub unbound: Option<Sender<bool>>,
|
||||||
pub progress_widget: ProgressWidget,
|
pub progress_widget: ProgressWidget,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Ffmpeg {
|
impl Ffmpeg {
|
||||||
pub fn start_record(&mut self, x: u16, y: u16, width: u16, height: u16) -> u32 {
|
pub fn start_record(
|
||||||
if self.process_id.is_some() {
|
&mut self,
|
||||||
|
x: u16,
|
||||||
|
y: u16,
|
||||||
|
width: u16,
|
||||||
|
height: u16,
|
||||||
|
) -> (Option<u32>, Option<u32>) {
|
||||||
|
if self.video_process_id.is_some() || self.audio_process_id.is_some() {
|
||||||
self.stop_record();
|
self.stop_record();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -127,6 +137,22 @@ impl Ffmpeg {
|
|||||||
.to_string(),
|
.to_string(),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
if self.record_audio.get_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.get_active_id().unwrap().to_string());
|
||||||
|
ffmpeg_command.arg("-f");
|
||||||
|
ffmpeg_command.arg("ogg");
|
||||||
|
ffmpeg_command.arg(format!(
|
||||||
|
"{}.temp.audio",
|
||||||
|
self.saved_filename.as_ref().unwrap().to_string()
|
||||||
|
));
|
||||||
|
ffmpeg_command.arg("-y");
|
||||||
|
self.audio_process_id = Some(ffmpeg_command.spawn().unwrap().id());
|
||||||
|
}
|
||||||
|
|
||||||
if is_wayland() {
|
if is_wayland() {
|
||||||
if self.record_video.get_active() {
|
if self.record_video.get_active() {
|
||||||
if self.unbound.is_some() {
|
if self.unbound.is_some() {
|
||||||
@ -145,118 +171,106 @@ impl Ffmpeg {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if self.record_audio.get_active() {
|
return (None, self.audio_process_id);
|
||||||
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.get_active_id().unwrap().to_string());
|
|
||||||
ffmpeg_command.arg("-f");
|
|
||||||
ffmpeg_command.arg("ogg");
|
|
||||||
ffmpeg_command.arg(format!(
|
|
||||||
"{}.temp.audio",
|
|
||||||
self.saved_filename.as_ref().unwrap().to_string()
|
|
||||||
));
|
|
||||||
ffmpeg_command.arg("-y");
|
|
||||||
self.process_id = Some(ffmpeg_command.spawn().unwrap().id());
|
|
||||||
}
|
|
||||||
|
|
||||||
return self.process_id.unwrap();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut ffmpeg_command: Command = Command::new("ffmpeg");
|
|
||||||
|
|
||||||
// if recorder video switch is enabled, record video with specified width and hight
|
|
||||||
if self.record_video.get_active() {
|
if self.record_video.get_active() {
|
||||||
|
let mut ffmpeg_command: Command = Command::new("ffmpeg");
|
||||||
|
|
||||||
|
// record video with specified width and hight
|
||||||
ffmpeg_command.arg("-video_size");
|
ffmpeg_command.arg("-video_size");
|
||||||
ffmpeg_command.arg(format!("{}x{}", width, height));
|
ffmpeg_command.arg(format!("{}x{}", width, height));
|
||||||
}
|
// if show mouse switch is enabled, draw the mouse to video
|
||||||
|
ffmpeg_command.arg("-draw_mouse");
|
||||||
// if show mouse switch is enabled, draw the mouse to video
|
if self.record_mouse.get_active() {
|
||||||
ffmpeg_command.arg("-draw_mouse");
|
ffmpeg_command.arg("1");
|
||||||
if self.record_mouse.get_active() {
|
} else {
|
||||||
ffmpeg_command.arg("1");
|
ffmpeg_command.arg("0");
|
||||||
} else {
|
}
|
||||||
ffmpeg_command.arg("0");
|
// if follow mouse switch is enabled, follow the mouse
|
||||||
}
|
if self.follow_mouse.get_active() {
|
||||||
|
ffmpeg_command.arg("-follow_mouse");
|
||||||
// if follow mouse switch is enabled, follow the mouse
|
ffmpeg_command.arg("centered");
|
||||||
if self.follow_mouse.get_active() {
|
}
|
||||||
ffmpeg_command.arg("-follow_mouse");
|
ffmpeg_command.arg("-framerate");
|
||||||
ffmpeg_command.arg("centered");
|
ffmpeg_command.arg(format!("{}", self.record_frames.get_value()));
|
||||||
}
|
|
||||||
|
|
||||||
ffmpeg_command.arg("-framerate");
|
|
||||||
ffmpeg_command.arg(format!("{}", self.record_frames.get_value()));
|
|
||||||
ffmpeg_command.arg("-f");
|
|
||||||
ffmpeg_command.arg("x11grab");
|
|
||||||
ffmpeg_command.arg("-i");
|
|
||||||
ffmpeg_command.arg(format!(
|
|
||||||
"{}+{},{}",
|
|
||||||
std::env::var("DISPLAY")
|
|
||||||
.unwrap_or(":1".to_string())
|
|
||||||
.as_str(),
|
|
||||||
x,
|
|
||||||
y
|
|
||||||
));
|
|
||||||
|
|
||||||
// if follow audio switch is enabled, record the audio
|
|
||||||
if self.record_audio.get_active() {
|
|
||||||
ffmpeg_command.arg("-f");
|
ffmpeg_command.arg("-f");
|
||||||
ffmpeg_command.arg("pulse");
|
ffmpeg_command.arg("x11grab");
|
||||||
ffmpeg_command.arg("-i");
|
ffmpeg_command.arg("-i");
|
||||||
ffmpeg_command.arg(self.audio_id.get_active_id().unwrap().to_string());
|
ffmpeg_command.arg(format!(
|
||||||
ffmpeg_command.arg("-strict");
|
"{}+{},{}",
|
||||||
ffmpeg_command.arg("-2");
|
std::env::var("DISPLAY")
|
||||||
|
.unwrap_or(":1".to_string())
|
||||||
|
.as_str(),
|
||||||
|
x,
|
||||||
|
y
|
||||||
|
));
|
||||||
|
ffmpeg_command.arg("-q");
|
||||||
|
ffmpeg_command.arg("1");
|
||||||
|
ffmpeg_command.arg(self.saved_filename.as_ref().unwrap().to_string());
|
||||||
|
ffmpeg_command.arg("-y");
|
||||||
|
// sleep for delay
|
||||||
|
sleep(Duration::from_secs(self.record_delay.get_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);
|
||||||
}
|
}
|
||||||
|
|
||||||
ffmpeg_command.arg("-q");
|
(None, None)
|
||||||
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.get_value() as u64));
|
|
||||||
|
|
||||||
// start recording and return the process id
|
|
||||||
self.process_id = Some(ffmpeg_command.spawn().unwrap().id());
|
|
||||||
self.process_id.unwrap()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn stop_record(&self) {
|
pub fn stop_record(&self) {
|
||||||
&self.progress_widget.show();
|
&self.progress_widget.show();
|
||||||
// kill the process to stop recording
|
// kill the process to stop recording
|
||||||
if self.process_id.is_some() {
|
if self.video_process_id.is_some() {
|
||||||
&self
|
&self
|
||||||
.progress_widget
|
.progress_widget
|
||||||
.set_progress("Stop Recording".to_string(), 1, 5);
|
.set_progress("Stop Recording Video".to_string(), 1, 6);
|
||||||
Command::new("kill")
|
Command::new("kill")
|
||||||
.arg(format!("{}", self.process_id.unwrap()))
|
.arg(format!("{}", self.video_process_id.unwrap()))
|
||||||
.output()
|
.output()
|
||||||
.unwrap();
|
.unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
if is_wayland() {
|
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("")),
|
||||||
|
if is_wayland() { ".temp" } else { "" }
|
||||||
|
)
|
||||||
|
.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 && is_wayland() {
|
||||||
// create new dbus session
|
// create new dbus session
|
||||||
let connection = zbus::Connection::new_session().unwrap();
|
let connection = zbus::Connection::new_session().unwrap();
|
||||||
// bind the connection to gnome screencast proxy
|
// bind the connection to gnome screencast proxy
|
||||||
let gnome_screencast_proxy = GnomeScreencastProxy::new(&connection).unwrap();
|
let gnome_screencast_proxy = GnomeScreencastProxy::new(&connection).unwrap();
|
||||||
gnome_screencast_proxy.stop_screencast().unwrap();
|
gnome_screencast_proxy.stop_screencast().unwrap();
|
||||||
let is_audio_record = std::path::Path::new(
|
|
||||||
format!("{}.temp.audio", self.saved_filename.as_ref().unwrap_or(&String::from(""))).as_str()
|
|
||||||
)
|
|
||||||
.exists();
|
|
||||||
let is_video_record = std::path::Path::new(
|
|
||||||
format!("{}.temp", self.saved_filename.as_ref().unwrap_or(&String::from(""))).as_str()
|
|
||||||
)
|
|
||||||
.exists();
|
|
||||||
|
|
||||||
if self.unbound.is_some() {
|
if self.unbound.is_some() {
|
||||||
&self.progress_widget.set_progress(
|
&self.progress_widget.set_progress(
|
||||||
"Stop Wayland Video Recording".to_string(),
|
"Stop Wayland Video Recording".to_string(),
|
||||||
2,
|
3,
|
||||||
5,
|
6,
|
||||||
);
|
);
|
||||||
self.unbound
|
self.unbound
|
||||||
.as_ref()
|
.as_ref()
|
||||||
@ -288,86 +302,106 @@ impl Ffmpeg {
|
|||||||
ffmpeg_convert_command.output().unwrap();
|
ffmpeg_convert_command.output().unwrap();
|
||||||
std::fs::remove_file(format!("{}.temp", self.saved_filename.as_ref().unwrap()))
|
std::fs::remove_file(format!("{}.temp", self.saved_filename.as_ref().unwrap()))
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
if is_audio_record {
|
|
||||||
&self.progress_widget.set_progress(
|
|
||||||
"Stop Wayland Audio Recording".to_string(),
|
|
||||||
3,
|
|
||||||
5,
|
|
||||||
);
|
|
||||||
|
|
||||||
// merge audio with video
|
|
||||||
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.get_active_id().unwrap().to_string()
|
|
||||||
));
|
|
||||||
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");
|
|
||||||
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.get_active_id().unwrap().to_string()
|
|
||||||
))
|
|
||||||
.unwrap();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else if is_audio_record {
|
}
|
||||||
&self.progress_widget.set_progress(
|
}
|
||||||
"Convert Audio to choosen format".to_string(),
|
if is_video_record {
|
||||||
3,
|
let mut move_command = Command::new("mv");
|
||||||
5,
|
move_command.arg(format!(
|
||||||
);
|
"{}{}",
|
||||||
println!("convert audio");
|
self.saved_filename.as_ref().unwrap(),
|
||||||
Command::new("ffmpeg")
|
if is_wayland() { ".temp" } else { "" }
|
||||||
.arg("-f")
|
));
|
||||||
.arg("ogg")
|
move_command.arg(format!(
|
||||||
.arg("-i")
|
"{}{}",
|
||||||
.arg(format!(
|
self.saved_filename.as_ref().unwrap_or(&String::new()),
|
||||||
"{}.temp.audio",
|
if is_audio_record {
|
||||||
self.saved_filename.as_ref().unwrap()
|
format!(
|
||||||
))
|
".temp.without.audio.{}",
|
||||||
.arg(format!("{}", self.saved_filename.as_ref().unwrap()))
|
self.filename.2.get_active_id().unwrap().to_string()
|
||||||
.output()
|
)
|
||||||
.unwrap();
|
} else {
|
||||||
|
"".to_string()
|
||||||
|
}
|
||||||
|
));
|
||||||
|
move_command.output().unwrap();
|
||||||
|
|
||||||
|
// 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.get_active_id().unwrap().to_string()
|
||||||
|
));
|
||||||
|
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!(
|
std::fs::remove_file(format!(
|
||||||
"{}.temp.audio",
|
"{}.temp.audio",
|
||||||
self.saved_filename.as_ref().unwrap()
|
self.saved_filename.as_ref().unwrap()
|
||||||
))
|
))
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
std::fs::remove_file(format!(
|
||||||
|
"{}.temp.without.audio.{}",
|
||||||
|
self.saved_filename.as_ref().unwrap(),
|
||||||
|
self.filename.2.get_active_id().unwrap().to_string()
|
||||||
|
))
|
||||||
|
.unwrap();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// if only audio is recording then convert it to chosen fromat
|
||||||
|
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(format!("{}", self.saved_filename.as_ref().unwrap()))
|
||||||
|
.output()
|
||||||
|
.unwrap();
|
||||||
|
std::fs::remove_file(format!(
|
||||||
|
"{}.temp.audio",
|
||||||
|
self.saved_filename.as_ref().unwrap()
|
||||||
|
))
|
||||||
|
.unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
// execute command after finish recording
|
// execute command after finish recording
|
||||||
if !(self.command.get_text().trim() == "") {
|
if !(self.command.get_text().trim() == "") {
|
||||||
&self.progress_widget.set_progress(
|
&self.progress_widget.set_progress(
|
||||||
"execute custom command after finish".to_string(),
|
"execute custom command after finish".to_string(),
|
||||||
4,
|
|
||||||
5,
|
5,
|
||||||
|
6,
|
||||||
);
|
);
|
||||||
Exec::shell(self.command.get_text().trim()).popen().unwrap();
|
Exec::shell(self.command.get_text().trim()).popen().unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
&self
|
&self
|
||||||
.progress_widget
|
.progress_widget
|
||||||
.set_progress("Finish".to_string(), 5, 5);
|
.set_progress("Finish".to_string(), 6, 6);
|
||||||
&self.progress_widget.hide();
|
&self.progress_widget.hide();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -290,7 +290,8 @@ fn main() {
|
|||||||
record_frames: frames_spin,
|
record_frames: frames_spin,
|
||||||
record_delay: delay_spin,
|
record_delay: delay_spin,
|
||||||
command: command_entry,
|
command: command_entry,
|
||||||
process_id: None,
|
video_process_id: None,
|
||||||
|
audio_process_id: None,
|
||||||
saved_filename: None,
|
saved_filename: None,
|
||||||
unbound: None,
|
unbound: None,
|
||||||
progress_widget: ProgressWidget::new(&main_window),
|
progress_widget: ProgressWidget::new(&main_window),
|
||||||
|
Loading…
Reference in New Issue
Block a user