Initial commit
This commit is contained in:
commit
4800758796
1
.gitignore
vendored
Normal file
1
.gitignore
vendored
Normal file
|
@ -0,0 +1 @@
|
||||||
|
/target
|
5
Cargo.lock
generated
Normal file
5
Cargo.lock
generated
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
# This file is automatically @generated by Cargo.
|
||||||
|
# It is not intended for manual editing.
|
||||||
|
[[package]]
|
||||||
|
name = "stream-screen"
|
||||||
|
version = "0.1.0"
|
9
Cargo.toml
Normal file
9
Cargo.toml
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
[package]
|
||||||
|
name = "stream-screen"
|
||||||
|
version = "0.1.0"
|
||||||
|
authors = ["Benjamin Bädorf <hello@benjaminbaedorf.eu>"]
|
||||||
|
edition = "2018"
|
||||||
|
|
||||||
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
|
||||||
|
[dependencies]
|
177
src/main.rs
Normal file
177
src/main.rs
Normal file
|
@ -0,0 +1,177 @@
|
||||||
|
use std::env;
|
||||||
|
use std::io::{BufRead, BufReader, Error, ErrorKind, Write};
|
||||||
|
use std::process::{Child, Command, Stdio};
|
||||||
|
|
||||||
|
struct Config {
|
||||||
|
current_screen: String,
|
||||||
|
output: String,
|
||||||
|
screen_blacklist: Vec<String>,
|
||||||
|
workspace_blacklist: Vec<String>,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn help() {
|
||||||
|
println!("Wrapper around wf-recorder and ffmpeg that automatically switches the screen being recorded based on current window focus");
|
||||||
|
println!("");
|
||||||
|
println!("Usage: stream-screen [options]");
|
||||||
|
println!("");
|
||||||
|
println!("Options:");
|
||||||
|
println!(" --not-ws <ws-num> Do not show this workspace. Can be used multiple times. Example: 3");
|
||||||
|
println!(" --not-screen <screen> Do not show this screen. Can be used multiple times. Example: HDMI-A-1");
|
||||||
|
println!(" -o|--output <output> Output to this device. Defaults to /dev/video0");
|
||||||
|
std::process::exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn record_screen(config: &mut Config, valid_screens: &Vec<String>) -> Result<Child, Error> {
|
||||||
|
if valid_screens.len() == 0 {
|
||||||
|
let cmd = Command::new("ffmpeg")
|
||||||
|
.args(&[
|
||||||
|
"--i",
|
||||||
|
"color=size=1920x1080:rate=25:color=black",
|
||||||
|
"--f=lavfi",
|
||||||
|
"-vcodec",
|
||||||
|
"rawvideo",
|
||||||
|
"-pix_fmt",
|
||||||
|
"yuv420p",
|
||||||
|
"-f",
|
||||||
|
"v4l2",
|
||||||
|
config.output.as_str(),
|
||||||
|
])
|
||||||
|
.spawn()?;
|
||||||
|
|
||||||
|
config.current_screen = "".to_string();
|
||||||
|
|
||||||
|
return Ok(cmd);
|
||||||
|
} else {
|
||||||
|
let index = get_screen_index(&valid_screens[0]);
|
||||||
|
let output_str = format!("--file={}", config.output.as_str());
|
||||||
|
let mut cmd = Command::new("wf-recorder")
|
||||||
|
.args(&[
|
||||||
|
"--muxer=v4l2",
|
||||||
|
"--codec=rawvideo",
|
||||||
|
"--pixel-format=yuv420p",
|
||||||
|
output_str.as_str(),
|
||||||
|
])
|
||||||
|
.stdin(Stdio::piped())
|
||||||
|
.spawn()?;
|
||||||
|
|
||||||
|
cmd.stdin
|
||||||
|
.as_mut()
|
||||||
|
.ok_or(Error::new(ErrorKind::Other, "Recorder process failed"))?
|
||||||
|
.write_all(index.as_bytes())?;
|
||||||
|
|
||||||
|
config.current_screen = valid_screens[0].as_str().to_string();
|
||||||
|
|
||||||
|
return Ok(cmd);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_valid_screens_for_recording(config: &Config) -> Vec<String> {
|
||||||
|
let mut command =
|
||||||
|
"swaymsg -t get_workspaces | jq -r 'sort_by(.focused != true) | map(select(.visible"
|
||||||
|
.to_string();
|
||||||
|
for screen in &config.screen_blacklist {
|
||||||
|
command.push_str(" and .output != \"");
|
||||||
|
command.push_str(screen.as_str());
|
||||||
|
command.push_str("\"");
|
||||||
|
}
|
||||||
|
for workspace in &config.workspace_blacklist {
|
||||||
|
command.push_str(" and .num != ");
|
||||||
|
command.push_str(workspace.as_str());
|
||||||
|
}
|
||||||
|
command.push_str(")) | map(.output) | .[]'");
|
||||||
|
let output = Command::new("bash")
|
||||||
|
.args(&["-c", command.as_str()])
|
||||||
|
.output()
|
||||||
|
.expect("Couldn't get current focus");
|
||||||
|
|
||||||
|
let stdout = String::from_utf8(output.stdout).expect("Invalid UTF-8 from get_workspaces");
|
||||||
|
let valid_screens = stdout.split('\n').map(|s| s.to_string()).collect();
|
||||||
|
for screen in &valid_screens {
|
||||||
|
println!("{}", screen);
|
||||||
|
}
|
||||||
|
return valid_screens;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_screen_index(screen: &String) -> String {
|
||||||
|
let mut command = "swaymsg -t get_outputs | jq -r 'map(.name==\"".to_string();
|
||||||
|
command.push_str(screen.as_str());
|
||||||
|
command.push_str("\") | index(true) + 1'");
|
||||||
|
let output = Command::new("bash")
|
||||||
|
.args(&["-c", command.as_str()])
|
||||||
|
.output()
|
||||||
|
.expect("Couldn't get screen index");
|
||||||
|
|
||||||
|
let stdout = String::from_utf8(output.stdout).expect("Invalid UTF-8 from get_workspaces");
|
||||||
|
return stdout;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() -> Result<(), Box<dyn std::error::Error>> {
|
||||||
|
let mut config = Config {
|
||||||
|
current_screen: "".to_string(),
|
||||||
|
output: "/dev/video0".to_string(),
|
||||||
|
screen_blacklist: Vec::new(),
|
||||||
|
workspace_blacklist: Vec::new(),
|
||||||
|
};
|
||||||
|
let args: Vec<String> = env::args().collect();
|
||||||
|
println!("{:?}", args);
|
||||||
|
|
||||||
|
let mut i = 1;
|
||||||
|
loop {
|
||||||
|
if i >= args.len() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
let arg = &args[i];
|
||||||
|
if arg == "--not-ws" {
|
||||||
|
i += 1;
|
||||||
|
config.workspace_blacklist.push(args[i].clone());
|
||||||
|
} else if arg == "--not-screen" {
|
||||||
|
i += 1;
|
||||||
|
config.screen_blacklist.push(args[i].clone());
|
||||||
|
} else if arg == "-o" || arg == "--output" {
|
||||||
|
i += 1;
|
||||||
|
config.output = args[i].clone();
|
||||||
|
} else if arg == "-h" || arg == "--help" {
|
||||||
|
help();
|
||||||
|
} else {
|
||||||
|
println!("Unknown option: {}", arg);
|
||||||
|
help();
|
||||||
|
}
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
let valid_screens = get_valid_screens_for_recording(&config);
|
||||||
|
let mut recorder = record_screen(&mut config, &valid_screens)?;
|
||||||
|
|
||||||
|
let stdout = match Command::new("bash")
|
||||||
|
.args(&["-c", "swaymsg -t subscribe -m \"['window']\""])
|
||||||
|
.stdout(Stdio::piped())
|
||||||
|
.spawn()?
|
||||||
|
.stdout
|
||||||
|
{
|
||||||
|
Some(stdout) => stdout,
|
||||||
|
None => panic!("Could not open stdout"),
|
||||||
|
};
|
||||||
|
|
||||||
|
let reader = BufReader::new(stdout);
|
||||||
|
reader.lines().filter_map(|line| line.ok()).for_each(|_| {
|
||||||
|
println!("Switched focus");
|
||||||
|
let valid_screens = get_valid_screens_for_recording(&config);
|
||||||
|
if valid_screens.len() > 0 && valid_screens[0] == config.current_screen {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
match recorder.kill() {
|
||||||
|
Ok(_) => {}
|
||||||
|
Err(err) => panic!("{:?}", err),
|
||||||
|
};
|
||||||
|
|
||||||
|
recorder = match record_screen(&mut config, &valid_screens) {
|
||||||
|
Ok(recorder) => recorder,
|
||||||
|
Err(err) => panic!("{:?}", err),
|
||||||
|
};
|
||||||
|
println!("Recording {}", config.current_screen);
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
Loading…
Reference in a new issue