use std::{path::Path, process};
use avio::{
BitrateMode, CRF_MAX, HardwareEncoder, Preset, VideoCodec, VideoCodecEncodeExt, VideoDecoder,
VideoEncoder,
};
fn main() {
let mut args = std::env::args().skip(1);
let mut input = None::<String>;
let mut output = None::<String>;
let mut preset_str = "medium".to_string();
let mut crf: u32 = 23;
let mut vbr: Option<u64> = None;
while let Some(flag) = args.next() {
match flag.as_str() {
"--input" | "-i" => input = Some(args.next().unwrap_or_default()),
"--output" | "-o" => output = Some(args.next().unwrap_or_default()),
"--preset" => preset_str = args.next().unwrap_or_else(|| "medium".to_string()),
"--crf" => {
let v = args.next().unwrap_or_default();
crf = v.parse().unwrap_or(23);
}
"--vbr" => {
let v = args.next().unwrap_or_default();
vbr = v.parse().ok();
}
other => {
eprintln!("Unknown flag: {other}");
process::exit(1);
}
}
}
let input = input.unwrap_or_else(|| {
eprintln!(
"Usage: encode_video_direct --input <file> --output <file> \
[--preset ultrafast|faster|fast|medium|slow|slower|veryslow] \
[--crf N] [--vbr N]"
);
process::exit(1);
});
let output = output.unwrap_or_else(|| {
eprintln!("--output is required");
process::exit(1);
});
let preset = match preset_str.to_lowercase().as_str() {
"ultrafast" => Preset::Ultrafast,
"faster" => Preset::Faster,
"fast" => Preset::Fast,
"medium" => Preset::Medium,
"slow" => Preset::Slow,
"slower" => Preset::Slower,
"veryslow" => Preset::Veryslow,
other => {
eprintln!(
"Unknown preset '{other}' \
(try ultrafast, faster, fast, medium, slow, slower, veryslow)"
);
process::exit(1);
}
};
let crf = crf.min(CRF_MAX);
let bmode = match vbr {
Some(target) => BitrateMode::Vbr {
target,
max: target * 2,
},
None => BitrateMode::Crf(crf),
};
let probe = match VideoDecoder::open(&input).build() {
Ok(d) => d,
Err(e) => {
eprintln!("Error opening input: {e}");
process::exit(1);
}
};
let width = probe.width();
let height = probe.height();
let fps = probe.frame_rate();
let in_codec = probe.stream_info().codec_name().to_string();
drop(probe);
let out_codec = VideoCodec::H264;
let in_name = Path::new(&input)
.file_name()
.and_then(|n| n.to_str())
.unwrap_or(&input);
let out_name = Path::new(&output)
.file_name()
.and_then(|n| n.to_str())
.unwrap_or(&output);
println!("Input: {in_name} {width}×{height} {fps:.2} fps codec={in_codec}");
println!(
"Codec: {} lgpl_compatible={}",
out_codec.default_extension(),
out_codec.is_lgpl_compatible()
);
let hw_list: Vec<String> = HardwareEncoder::available()
.iter()
.map(|hw| format!("{hw:?}"))
.collect();
println!("HW encoders available: {}", hw_list.join(", "));
let quality_str = match &bmode {
BitrateMode::Cbr(bps) => format!("cbr={bps}"),
BitrateMode::Crf(q) => format!("crf={q}"),
BitrateMode::Vbr { target, max } => format!("vbr target={target} max={max}"),
};
println!("Output: {out_name} {width}×{height} preset={preset_str} {quality_str}");
println!();
let mut encoder = match VideoEncoder::create(&output)
.video(width, height, fps)
.video_codec(out_codec)
.preset(preset)
.bitrate_mode(bmode)
.build()
{
Ok(e) => e,
Err(e) => {
eprintln!("Error building encoder: {e}");
process::exit(1);
}
};
println!(
"Encoder opened: actual_codec={} hardware={}",
encoder.actual_video_codec(),
encoder.is_hardware_encoding()
);
println!("Encoding...");
let mut decoder = match VideoDecoder::open(&input).build() {
Ok(d) => d,
Err(e) => {
eprintln!("Error opening decoder: {e}");
process::exit(1);
}
};
let mut frames: u64 = 0;
loop {
let frame = match decoder.decode_one() {
Ok(Some(f)) => f,
Ok(None) => break,
Err(e) => {
eprintln!("Decode error: {e}");
process::exit(1);
}
};
if let Err(e) = encoder.push_video(&frame) {
eprintln!("Encode error: {e}");
process::exit(1);
}
frames += 1;
}
if let Err(e) = encoder.finish() {
eprintln!("Error finalising output: {e}");
process::exit(1);
}
let size_str = match std::fs::metadata(&output) {
Ok(m) => {
#[allow(clippy::cast_precision_loss)]
let kb = m.len() as f64 / 1024.0;
if kb < 1024.0 {
format!("{kb:.0} KB")
} else {
format!("{:.1} MB", kb / 1024.0)
}
}
Err(_) => "(unknown size)".to_string(),
};
println!("Done. {out_name} {size_str} {frames} frames");
}