|
use pyo3::prelude::*; |
|
use std::process::Command; |
|
use std::path::Path; |
|
|
|
#[pyfunction] |
|
fn render_video( |
|
id: usize, |
|
image_path: String, |
|
audio_path: String, |
|
duration: f64, |
|
words: Vec<(String, (u32, u32, u32, u32))>, |
|
output_dir: String, |
|
) -> PyResult<String> { |
|
|
|
if !Path::new(&image_path).exists() { |
|
return Err(pyo3::exceptions::PyFileNotFoundError::new_err(format!( |
|
"Image not found: {}", |
|
image_path |
|
))); |
|
} |
|
if !Path::new(&audio_path).exists() { |
|
return Err(pyo3::exceptions::PyFileNotFoundError::new_err(format!( |
|
"Audio not found: {}", |
|
audio_path |
|
))); |
|
} |
|
|
|
let output_path = format!("{}/clip{}.mp4", output_dir, id); |
|
let duration_str = duration.to_string(); |
|
let n_words = words.len() as f64; |
|
let highlight_duration = duration / n_words; |
|
|
|
|
|
let mut filters = vec![]; |
|
for (i, (_, (x, y, w, h))) in words.iter().enumerate() { |
|
let start = i as f64 * highlight_duration; |
|
let end = start + highlight_duration; |
|
filters.push(format!( |
|
"drawbox=x={x}:y={y}:w={w}:h={h}:color=yellow@0.5:t=fill:enable='between(t,{start:.2},{end:.2})'" |
|
)); |
|
} |
|
|
|
let filter_chain = filters.join(","); |
|
|
|
|
|
let status = Command::new("ffmpeg") |
|
.args(&[ |
|
"-y", |
|
"-loop", "1", |
|
"-i", &image_path, |
|
"-i", &audio_path, |
|
"-vf", &filter_chain, |
|
"-c:v", "libx264", |
|
"-preset", "fast", |
|
"-crf", "18", |
|
"-pix_fmt", "yuv420p", |
|
"-movflags", "+faststart", |
|
"-c:a", "aac", |
|
"-b:a", "192k", |
|
"-r", "60", |
|
"-fps_mode", "vfr", |
|
"-t", &duration_str, |
|
&output_path, |
|
]) |
|
.status(); |
|
|
|
match status { |
|
Ok(exit_status) if exit_status.success() => Ok(output_path), |
|
Ok(_) => Err(pyo3::exceptions::PyRuntimeError::new_err( |
|
"FFmpeg command failed. Check filter syntax.", |
|
)), |
|
Err(e) => Err(pyo3::exceptions::PyRuntimeError::new_err(format!( |
|
"Failed to execute FFmpeg: {}", |
|
e |
|
))), |
|
} |
|
} |
|
|
|
#[pymodule] |
|
fn rust_highlight(_py: Python, m: &PyModule) -> PyResult<()> { |
|
m.add_function(wrap_pyfunction!(render_video, m)?)?; |
|
Ok(()) |
|
} |