Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Initial gtk4 migration #39

Merged
merged 50 commits into from
Nov 22, 2023
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
50 commits
Select commit Hold shift + click to select a range
6f48df3
Initial gtk4 migration
ochibani Sep 24, 2022
3c0ea6e
fix window size by changing main.ui
ochibani Sep 28, 2022
78d3fb6
display home icons in folder chooser
ochibani Sep 29, 2022
84cc6b3
area chooser window redesigned
ochibani Sep 30, 2022
8092950
spell check
ochibani Oct 1, 2022
835e466
remove wayland from ffmpeg-interface
ochibani Oct 14, 2022
3428d6d
Remove buggy video formats
ochibani Oct 14, 2022
56f5598
fix is_file_already_exists & play button
ochibani Oct 16, 2022
d9e7458
use system theme colors
xlmnxp Oct 18, 2022
a6ca6d3
look like fixed the video and audio merge
xlmnxp Oct 18, 2022
828074a
fix Progress dialog
xlmnxp Oct 18, 2022
336e8e4
hide window selection in wayland
ochibani Oct 18, 2022
71edab6
folder chooser parent
ochibani Oct 19, 2022
cec2d85
folder chooser hide
ochibani Oct 19, 2022
d1c4706
fix FileChooserButton
ochibani Oct 20, 2022
257c5b4
remove progress button
ochibani Oct 20, 2022
ad01c74
remove audio only record code
ochibani Oct 24, 2022
a3dbd29
add delay timer
ochibani Nov 1, 2022
f7c72ff
add recording timer
ochibani Nov 2, 2022
cbcb44d
add hours to timer
ochibani Nov 2, 2022
3e37cc9
minor change
ochibani Nov 4, 2022
258cda8
Edit indicator.rs
ochibani Nov 26, 2022
78ec81d
Working X11 version
ochibani Nov 26, 2022
fb32ebe
use 60fps as default
xlmnxp Feb 22, 2023
86303da
Merge branch 'blue-recorder-gtk4' of https://github.com/ochibani/blue…
xlmnxp Feb 22, 2023
c5189de
bug fixes, force select audio or video to record
xlmnxp Feb 22, 2023
a8f025e
use of Childs instead of kill command (Work cross multiple operating …
xlmnxp Feb 22, 2023
d82d71b
Simplify the implementation of Ffmpeg
xlmnxp Feb 23, 2023
b5b10cf
fix video and audio merges and fix audio recording issue
xlmnxp Feb 28, 2023
2ed389d
include tr.po with migration to GTK4
xlmnxp Mar 1, 2023
af982be
auto detect dark mode and change icons
xlmnxp Mar 1, 2023
3695530
improve style of select buttons
xlmnxp Oct 14, 2023
f857ba5
remove overwrite switch and fix message dialog
xlmnxp Oct 14, 2023
329c4a7
add wayland recorder
xlmnxp Oct 14, 2023
29b0a61
little improvements
xlmnxp Oct 15, 2023
aa9213a
little improvements
xlmnxp Oct 15, 2023
5d8f19f
update ui
ochibani Oct 16, 2023
7688528
update ui
ochibani Oct 17, 2023
8b7343c
add auto hide
ochibani Oct 19, 2023
b959262
add wayland record and mouse toggle
xlmnxp Oct 19, 2023
0e95762
refactor of code
xlmnxp Oct 21, 2023
5f847a0
remove progress dialog
xlmnxp Oct 21, 2023
5f079b8
fix borrow_mut
xlmnxp Oct 21, 2023
75df858
enable recording of window or screen
xlmnxp Oct 21, 2023
6c049e4
remove temporary video file when recording video only
xlmnxp Oct 21, 2023
510a1e5
enable recording of window or screen
xlmnxp Oct 21, 2023
5af98f9
remember last decision to record window or screen
xlmnxp Oct 22, 2023
4e4756e
update ui
ochibani Oct 23, 2023
591a671
remove appid from gtk part
xlmnxp Oct 27, 2023
d10fdf7
upgrade snap packaging
xlmnxp Oct 28, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
add wayland record and mouse toggle
  • Loading branch information
xlmnxp committed Oct 19, 2023
commit b959262fbad0530962749ab753e12b38c6d16b68
2 changes: 1 addition & 1 deletion Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -22,4 +22,4 @@ subprocess = "0.2.6"
dark-light = "1.0.0"
async-std = {version = "1.12.0", features = ["attributes"]}
gstreamer = "0.20.5"
zbus = "3.12.0"
zbus = "3.12.0"
183 changes: 128 additions & 55 deletions src/ffmpeg_interface.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,6 @@
extern crate subprocess;
use crate::utils::{is_snap, is_wayland};
use crate::wayland_record::{CursorModeTypes, RecordTypes, WaylandRecorder};
use chrono::prelude::*;
use gettextrs::gettext;
use gtk::{prelude::*, ResponseType};
Expand All @@ -12,7 +14,6 @@ use std::sync::mpsc::Sender;
use std::thread::sleep;
use std::time::Duration;
use subprocess::Exec;
use crate::wayland_record::WaylandRecorder;

#[derive(Clone)]
pub struct ProgressWidget {
Expand Down Expand Up @@ -61,7 +62,8 @@ pub struct Ffmpeg {
pub unbound: Option<Sender<bool>>,
pub progress_widget: ProgressWidget,
pub window: Window,
pub record_wayland: WaylandRecorder
pub record_wayland: WaylandRecorder,
pub main_context: gtk::glib::MainContext,
}

impl Ffmpeg {
Expand Down Expand Up @@ -99,31 +101,15 @@ impl Ffmpeg {
&gettext("File already exist. Do you want to overwrite it?"),
);

let answer = glib::MainContext::default().block_on(message_dialog.run_future());
let answer = self.main_context.block_on(message_dialog.run_future());
message_dialog.close();

if answer != ResponseType::Yes {
return 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 = Some(Rc::new(RefCell::new(ffmpeg_command.spawn().unwrap())));
}

if self.record_video.is_active() {
if self.record_video.is_active() && !is_wayland() {
let mut ffmpeg_command: Command = Command::new("ffmpeg");

// record video with specified width and hight
Expand Down Expand Up @@ -183,19 +169,57 @@ impl Ffmpeg {

// start recording and return the process id
self.video_process = Some(Rc::new(RefCell::new(ffmpeg_command.spawn().unwrap())));
} else if self.record_video.is_active() && is_wayland() {
sleep(Duration::from_secs(self.record_delay.value() as u64));
if !self
.main_context
.block_on(self.record_wayland.start(
format!(
"{}.temp.without.audio.webm",
self.saved_filename.as_ref().unwrap()
),
RecordTypes::Monitor,
{
if self.record_mouse.is_active() {
CursorModeTypes::Show
} else {
CursorModeTypes::Hidden
}
},
))
{
println!("failed to start recording");
return 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 = Some(Rc::new(RefCell::new(ffmpeg_command.spawn().unwrap())));
}

Some(())
}

pub fn stop_record(&self) {
pub fn stop_record(&mut self) {
self.progress_widget.show();
// kill the process to stop recording
self.progress_widget.set_progress("".to_string(), 1, 6);
self.progress_widget.set_progress("".to_string(), 1, 7);

// kill the process to stop recording
if self.video_process.is_some() {
self.progress_widget
.set_progress("Stop Recording Video".to_string(), 1, 6);
.set_progress("Stop Recording Video".to_string(), 1, 7);

Command::new("kill")
.arg(format!(
Expand All @@ -213,13 +237,16 @@ impl Ffmpeg {
.unwrap();

println!("video killed");
} else if is_wayland() {
self.main_context
.block_on(self.record_wayland.stop());
}

self.progress_widget.set_progress("".to_string(), 2, 6);
self.progress_widget.set_progress("".to_string(), 2, 7);

if self.audio_process.is_some() {
self.progress_widget
.set_progress("Stop Recording Audio".to_string(), 2, 6);
.set_progress("Stop Recording Audio".to_string(), 2, 7);

Command::new("kill")
.arg(format!(
Expand All @@ -238,35 +265,83 @@ impl Ffmpeg {
println!("audio killed");
}

let video_filename = format!(
"{}.temp.without.audio.{}",
self.saved_filename.as_ref().unwrap(),
self.filename.2.active_id().unwrap()
);
let video_filename = {
if is_wayland() {
format!(
"{}.temp.without.audio.webm",
self.saved_filename.as_ref().unwrap()
)
} else {
format!(
"{}.temp.without.audio.{}",
self.saved_filename.as_ref().unwrap(),
self.filename.2.active_id().unwrap()
)
}
};

let audio_filename = format!("{}.temp.audio", self.saved_filename.as_ref().unwrap());

let is_video_record = std::path::Path::new(video_filename.as_str()).exists();
let is_video_record = {
if is_wayland() {
std::path::Path::new(&format!(
"{}.temp.without.audio.webm",
self.saved_filename.as_ref().unwrap()
))
.exists()
} else {
std::path::Path::new(video_filename.as_str()).exists()
}
};
let is_audio_record = std::path::Path::new(audio_filename.as_str()).exists();

if is_video_record {
let mut move_command = Command::new("mv");
move_command.args([
self.saved_filename.as_ref().unwrap().as_str(),
if is_audio_record {
video_filename.as_str()
} else {
self.saved_filename.as_ref().unwrap()
},
]);
move_command.output().unwrap();
if !is_wayland() {
let mut move_command = Command::new("mv");
move_command.args([
self.saved_filename.as_ref().unwrap().as_str(),
if is_audio_record {
video_filename.as_str()
} else {
self.saved_filename.as_ref().unwrap()
},
]);
move_command.output().unwrap();
} else {
println!("convert webm to specified format");

// convert webm to specified format
self.progress_widget.set_progress(
"Convert screen-cast to specified format".to_string(),
4,
7,
);

Command::new("ffmpeg")
.args([
"-i",
format!(
"{}.temp.without.audio.webm",
self.saved_filename.as_ref().unwrap()
)
.as_str(),
"-crf",
"23", // default quality
"-c:a",
self.filename.2.active_id().unwrap().as_str(),
self.saved_filename.as_ref().unwrap(),
"-y",
])
.output()
.unwrap();
}

self.progress_widget.set_progress("".to_string(), 4, 6);
self.progress_widget.set_progress("".to_string(), 5, 7);

// if audio record, then merge video with audio
if is_audio_record {
self.progress_widget
.set_progress("Save Audio Recording".to_string(), 4, 6);
.set_progress("Save Audio Recording".to_string(), 5, 7);

Command::new("ffmpeg")
.args([
Expand All @@ -276,15 +351,17 @@ impl Ffmpeg {
"ogg",
"-i",
audio_filename.as_str(),
"-c:v",
"copy",
"-crf",
"23", // default quality
"-c:a",
"aac",
self.saved_filename.as_ref().unwrap(),
"-y",
])
.output()
.expect("failed to merge video with audio");
.spawn()
.expect("failed to merge video with audio")
.wait()
.unwrap();

std::fs::remove_file(video_filename).unwrap();
std::fs::remove_file(audio_filename).unwrap();
Expand All @@ -293,7 +370,7 @@ impl Ffmpeg {
// 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);
.set_progress("Convert Audio to choosen format".to_string(), 5, 7);

Command::new("ffmpeg")
.args([
Expand All @@ -309,7 +386,7 @@ impl Ffmpeg {
std::fs::remove_file(audio_filename).unwrap();
}

self.progress_widget.set_progress("".to_string(), 5, 6);
self.progress_widget.set_progress("".to_string(), 6, 7);

// execute command after finish recording
if self.command.text().trim() != "" {
Expand All @@ -322,7 +399,7 @@ impl Ffmpeg {
}

self.progress_widget
.set_progress("Finish".to_string(), 6, 6);
.set_progress("Finish".to_string(), 7, 7);
self.progress_widget.hide();
}

Expand All @@ -344,7 +421,3 @@ impl Ffmpeg {
}
}
}

fn is_snap() -> bool {
!std::env::var("SNAP").unwrap_or_default().is_empty()
}
5 changes: 4 additions & 1 deletion src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -448,6 +448,8 @@ pub fn build_ui(application: &Application) {

let _delay_spin = delay_spin.clone();

let main_context = glib::MainContext::default();
let wayland_record = main_context.block_on(WaylandRecorder::new());
// Init record struct
let ffmpeg_record_interface: Rc<RefCell<Ffmpeg>> = Rc::new(RefCell::new(Ffmpeg {
filename: (
Expand All @@ -469,7 +471,8 @@ pub fn build_ui(application: &Application) {
progress_widget: ProgressWidget::new(progress_dialog, progress_bar),
window: main_window.clone(),
record_delay: delay_spin,
record_wayland: glib::MainContext::default().block_on(WaylandRecorder::new())
record_wayland: wayland_record,
main_context: main_context,
}));

// Record Button
Expand Down
4 changes: 4 additions & 0 deletions src/utils.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,4 +2,8 @@ pub fn is_wayland() -> bool {
std::env::var("XDG_SESSION_TYPE")
.unwrap_or_default()
.eq_ignore_ascii_case("wayland")
}

pub fn is_snap() -> bool {
!std::env::var("SNAP").unwrap_or_default().is_empty()
}
Loading