From 2916817fd597aa310e427118d421d182747fca98 Mon Sep 17 00:00:00 2001 From: polyesterswing Date: Tue, 16 Jul 2024 16:55:32 +0530 Subject: [PATCH 1/4] feat: flush stdout buffer before next frame --- src/main.rs | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/main.rs b/src/main.rs index b6dd071..e771858 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,6 +1,7 @@ use opencv::core::VecN; use opencv::prelude::*; use opencv::{core, highgui, imgproc, videoio, Result}; +use std::io::{self, Write}; fn map_range(from_range: (i32, i32), to_range: (i32, i32), s: i32) -> i32 { to_range.0 + (s - from_range.0) * (to_range.1 - to_range.0) / (from_range.1 - from_range.0) @@ -76,6 +77,9 @@ fn main() -> Result<()> { "{}", find_colors(&smaller, &gray, ascii_table, ascii_table_len)? ); + + io::stdout().flush().unwrap(); + } if highgui::wait_key(10)? > 0 { From 5eb52c7497be5c23c478f5c269cabf9509ac3436 Mon Sep 17 00:00:00 2001 From: polyesterswing Date: Tue, 16 Jul 2024 16:56:38 +0530 Subject: [PATCH 2/4] feat: remove debug code --- src/main.rs | 14 +------------- 1 file changed, 1 insertion(+), 13 deletions(-) diff --git a/src/main.rs b/src/main.rs index e771858..7279f51 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,6 +1,6 @@ use opencv::core::VecN; use opencv::prelude::*; -use opencv::{core, highgui, imgproc, videoio, Result}; +use opencv::{core, imgproc, videoio, Result}; use std::io::{self, Write}; fn map_range(from_range: (i32, i32), to_range: (i32, i32), s: i32) -> i32 { @@ -41,9 +41,6 @@ fn main() -> Result<()> { let term_size = termion::terminal_size().unwrap(); - let window = "Video"; - highgui::named_window(window, 1)?; - let mut cam = videoio::VideoCapture::from_file("baby-shark.webm", videoio::CAP_ANY)?; if !cam.is_opened()? { @@ -65,14 +62,9 @@ fn main() -> Result<()> { imgproc::INTER_AREA, )?; - //let _ = find_colors(&smaller)?; - //println!("{}", find_colors(&smaller)?); - let mut gray = Mat::default(); imgproc::cvt_color_def(&smaller, &mut gray, imgproc::COLOR_BGR2GRAY)?; - highgui::imshow(window, &smaller)?; - println!( "{}", find_colors(&smaller, &gray, ascii_table, ascii_table_len)? @@ -82,9 +74,5 @@ fn main() -> Result<()> { } - if highgui::wait_key(10)? > 0 { - break; - } } - Ok(()) } From 7a78e1fb628356320c915bb51a6f22efcc49c154 Mon Sep 17 00:00:00 2001 From: polyesterswing Date: Fri, 19 Jul 2024 22:16:00 +0530 Subject: [PATCH 3/4] feat: add threads --- src/main.rs | 39 +++++++++++++++++++++++++-------------- 1 file changed, 25 insertions(+), 14 deletions(-) diff --git a/src/main.rs b/src/main.rs index 7279f51..101db91 100644 --- a/src/main.rs +++ b/src/main.rs @@ -2,13 +2,15 @@ use opencv::core::VecN; use opencv::prelude::*; use opencv::{core, imgproc, videoio, Result}; use std::io::{self, Write}; +use std::sync::mpsc; +use std::{thread, time}; fn map_range(from_range: (i32, i32), to_range: (i32, i32), s: i32) -> i32 { to_range.0 + (s - from_range.0) * (to_range.1 - to_range.0) / (from_range.1 - from_range.0) } fn find_colors(frame: &Mat, gray: &Mat, table: &str, table_len: usize) -> Result { - let mut out_colors = String::new(); + let mut out_colors = String::with_capacity((frame.rows() * frame.cols()) as usize * 4); let rows = frame.rows(); let cols = frame.cols(); @@ -28,7 +30,7 @@ fn find_colors(frame: &Mat, gray: &Mat, table: &str, table_len: usize) -> Result )) } - out_colors.push_str("\n"); + out_colors.push_str("\r\n"); } Ok(out_colors) @@ -47,11 +49,17 @@ fn main() -> Result<()> { panic!("Unable to open video file"); } - loop { + let fps = cam.get(videoio::CAP_PROP_FPS)?; + // time b/w each frame + let frame_delay = time::Duration::from_millis((1000.0 / fps) as u64); + + let (tx, rx) = mpsc::channel(); + + thread::spawn(move || loop { let mut frame = Mat::default(); - cam.read(&mut frame)?; + cam.read(&mut frame).unwrap(); - if frame.size()?.width > 0 { + if frame.size().unwrap().width > 0 { let mut smaller = Mat::default(); imgproc::resize( &frame, @@ -60,19 +68,22 @@ fn main() -> Result<()> { 0.0, 0.0, imgproc::INTER_AREA, - )?; + ) + .unwrap(); let mut gray = Mat::default(); - imgproc::cvt_color_def(&smaller, &mut gray, imgproc::COLOR_BGR2GRAY)?; - - println!( - "{}", - find_colors(&smaller, &gray, ascii_table, ascii_table_len)? - ); - - io::stdout().flush().unwrap(); + imgproc::cvt_color_def(&smaller, &mut gray, imgproc::COLOR_BGR2GRAY).unwrap(); + tx.send(find_colors(&smaller, &gray, ascii_table, ascii_table_len).unwrap()) + .unwrap(); } + }); + for received in rx { + print!("{}", received); + thread::sleep(frame_delay); + print!("{}", termion::clear::All); } + + Ok(()) } From 1362d5ac6e6102f77c12219d765ff5ed6c088e87 Mon Sep 17 00:00:00 2001 From: polyesterswing Date: Fri, 19 Jul 2024 22:48:55 +0530 Subject: [PATCH 4/4] feat: move to crossterm and fix flickering --- Cargo.lock | 277 ++++++++++++++++++++++++++++++++++++++++++---------- Cargo.toml | 2 +- src/main.rs | 12 ++- 3 files changed, 236 insertions(+), 55 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 1126f24..17b853a 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -15,8 +15,8 @@ dependencies = [ name = "ascii-video-player" version = "0.1.0" dependencies = [ + "crossterm", "opencv", - "termion", ] [[package]] @@ -25,12 +25,6 @@ version = "1.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0c4b4d0bd25bd0b74681c0ad21497610ce1b7c91b1022cd21c80c6fbdd9476b0" -[[package]] -name = "bitflags" -version = "1.3.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" - [[package]] name = "bitflags" version = "2.6.0" @@ -48,6 +42,12 @@ dependencies = [ "once_cell", ] +[[package]] +name = "cfg-if" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" + [[package]] name = "clang" version = "2.0.0" @@ -68,6 +68,31 @@ dependencies = [ "libc", ] +[[package]] +name = "crossterm" +version = "0.27.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f476fe445d41c9e991fd07515a6f463074b782242ccf4a5b7b1d1012e70824df" +dependencies = [ + "bitflags", + "crossterm_winapi", + "libc", + "mio", + "parking_lot", + "signal-hook", + "signal-hook-mio", + "winapi", +] + +[[package]] +name = "crossterm_winapi" +version = "0.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "acdd7c62a3665c7f6830a51635d9ac9b23ed385797f70a83bb8bafe9c572ab2b" +dependencies = [ + "winapi", +] + [[package]] name = "dunce" version = "1.0.4" @@ -96,22 +121,39 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "97b3888a4aecf77e811145cadf6eef5901f4782c53886191b2f693f24761847c" [[package]] -name = "libredox" -version = "0.0.2" +name = "lock_api" +version = "0.4.12" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3af92c55d7d839293953fcd0fda5ecfe93297cfde6ffbdec13b41d99c0ba6607" +checksum = "07af8b9cdd281b7915f413fa73f29ebd5d55d0d3f0155584dade1ff18cea1b17" dependencies = [ - "bitflags 2.6.0", - "libc", - "redox_syscall", + "autocfg", + "scopeguard", ] +[[package]] +name = "log" +version = "0.4.22" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a7a70ba024b9dc04c27ea2f0c0548feb474ec5c54bba33a7f72f873a39d07b24" + [[package]] name = "memchr" version = "2.7.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "78ca9ab1a0babb1e7d5695e3530886289c18cf2f87ec19a575a0abdce112e3a3" +[[package]] +name = "mio" +version = "0.8.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a4a650543ca06a924e8b371db273b2756685faae30f8487da1b56505a8f78b0c" +dependencies = [ + "libc", + "log", + "wasi", + "windows-sys", +] + [[package]] name = "num-traits" version = "0.2.19" @@ -121,12 +163,6 @@ dependencies = [ "autocfg", ] -[[package]] -name = "numtoa" -version = "0.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b8f8bdf33df195859076e54ab11ee78a1b208382d3a26ec40d142ffc1ecc49ef" - [[package]] name = "once_cell" version = "1.19.0" @@ -167,6 +203,29 @@ dependencies = [ "regex", ] +[[package]] +name = "parking_lot" +version = "0.12.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f1bf18183cf54e8d6059647fc3063646a1801cf30896933ec2311622cc4b9a27" +dependencies = [ + "lock_api", + "parking_lot_core", +] + +[[package]] +name = "parking_lot_core" +version = "0.9.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e401f977ab385c9e4e3ab30627d6f26d00e2c73eef317493c4ec6d468726cf8" +dependencies = [ + "cfg-if", + "libc", + "redox_syscall", + "smallvec", + "windows-targets 0.52.5", +] + [[package]] name = "percent-encoding" version = "2.3.1" @@ -199,19 +258,13 @@ dependencies = [ [[package]] name = "redox_syscall" -version = "0.4.1" +version = "0.5.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4722d768eff46b75989dd134e5c353f0d6296e5aaa3132e776cbdb56be7731aa" +checksum = "2a908a6e00f1fdd0dfd9c0eb08ce85126f6d8bbda50017e74bc4a4b7d4a926a4" dependencies = [ - "bitflags 1.3.2", + "bitflags", ] -[[package]] -name = "redox_termios" -version = "0.1.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "20145670ba436b55d91fc92d25e71160fbfbdd57831631c8d7d36377a476f1cb" - [[package]] name = "regex" version = "1.10.5" @@ -241,6 +294,12 @@ version = "0.8.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7a66a03ae7c801facd77a29370b4faec201768915ac14a721ba36f20bc9c209b" +[[package]] +name = "scopeguard" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49" + [[package]] name = "semver" version = "1.0.23" @@ -254,26 +313,50 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64" [[package]] -name = "syn" -version = "2.0.68" +name = "signal-hook" +version = "0.3.17" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "901fa70d88b9d6c98022e23b4136f9f3e54e4662c3bc1bd1d84a42a9a0f0c1e9" +checksum = "8621587d4798caf8eb44879d42e56b9a93ea5dcd315a6487c357130095b62801" dependencies = [ - "proc-macro2", - "quote", - "unicode-ident", + "libc", + "signal-hook-registry", +] + +[[package]] +name = "signal-hook-mio" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "29ad2e15f37ec9a6cc544097b78a1ec90001e9f71b81338ca39f430adaca99af" +dependencies = [ + "libc", + "mio", + "signal-hook", ] [[package]] -name = "termion" -version = "4.0.2" +name = "signal-hook-registry" +version = "1.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1ccce68e518d1173e80876edd54760b60b792750d0cab6444a79101c6ea03848" +checksum = "a9e9e0b4211b72e7b8b6e85c807d36c212bdb33ea8587f7569562a84df5465b1" dependencies = [ "libc", - "libredox", - "numtoa", - "redox_termios", +] + +[[package]] +name = "smallvec" +version = "1.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67" + +[[package]] +name = "syn" +version = "2.0.68" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "901fa70d88b9d6c98022e23b4136f9f3e54e4662c3bc1bd1d84a42a9a0f0c1e9" +dependencies = [ + "proc-macro2", + "quote", + "unicode-ident", ] [[package]] @@ -288,6 +371,34 @@ version = "0.2.15" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "accd4ea62f7bb7a82fe23066fb0957d48ef677f6eeb8215f372f52e48bb32426" +[[package]] +name = "wasi" +version = "0.11.0+wasi-snapshot-preview1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" + +[[package]] +name = "winapi" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419" +dependencies = [ + "winapi-i686-pc-windows-gnu", + "winapi-x86_64-pc-windows-gnu", +] + +[[package]] +name = "winapi-i686-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" + +[[package]] +name = "winapi-x86_64-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" + [[package]] name = "windows" version = "0.56.0" @@ -295,7 +406,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1de69df01bdf1ead2f4ac895dc77c9351aefff65b2f3db429a343f9cbf05e132" dependencies = [ "windows-core", - "windows-targets", + "windows-targets 0.52.5", ] [[package]] @@ -307,7 +418,7 @@ dependencies = [ "windows-implement", "windows-interface", "windows-result", - "windows-targets", + "windows-targets 0.52.5", ] [[package]] @@ -338,7 +449,31 @@ version = "0.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "5e383302e8ec8515204254685643de10811af0ed97ea37210dc26fb0032647f8" dependencies = [ - "windows-targets", + "windows-targets 0.52.5", +] + +[[package]] +name = "windows-sys" +version = "0.48.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "677d2418bec65e3338edb076e806bc1ec15693c5d0104683f2efe857f61056a9" +dependencies = [ + "windows-targets 0.48.5", +] + +[[package]] +name = "windows-targets" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9a2fa6e2155d7247be68c096456083145c183cbbbc2764150dda45a87197940c" +dependencies = [ + "windows_aarch64_gnullvm 0.48.5", + "windows_aarch64_msvc 0.48.5", + "windows_i686_gnu 0.48.5", + "windows_i686_msvc 0.48.5", + "windows_x86_64_gnu 0.48.5", + "windows_x86_64_gnullvm 0.48.5", + "windows_x86_64_msvc 0.48.5", ] [[package]] @@ -347,28 +482,46 @@ version = "0.52.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6f0713a46559409d202e70e28227288446bf7841d3211583a4b53e3f6d96e7eb" dependencies = [ - "windows_aarch64_gnullvm", - "windows_aarch64_msvc", - "windows_i686_gnu", + "windows_aarch64_gnullvm 0.52.5", + "windows_aarch64_msvc 0.52.5", + "windows_i686_gnu 0.52.5", "windows_i686_gnullvm", - "windows_i686_msvc", - "windows_x86_64_gnu", - "windows_x86_64_gnullvm", - "windows_x86_64_msvc", + "windows_i686_msvc 0.52.5", + "windows_x86_64_gnu 0.52.5", + "windows_x86_64_gnullvm 0.52.5", + "windows_x86_64_msvc 0.52.5", ] +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2b38e32f0abccf9987a4e3079dfb67dcd799fb61361e53e2882c3cbaf0d905d8" + [[package]] name = "windows_aarch64_gnullvm" version = "0.52.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7088eed71e8b8dda258ecc8bac5fb1153c5cffaf2578fc8ff5d61e23578d3263" +[[package]] +name = "windows_aarch64_msvc" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc35310971f3b2dbbf3f0690a219f40e2d9afcf64f9ab7cc1be722937c26b4bc" + [[package]] name = "windows_aarch64_msvc" version = "0.52.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9985fd1504e250c615ca5f281c3f7a6da76213ebd5ccc9561496568a2752afb6" +[[package]] +name = "windows_i686_gnu" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a75915e7def60c94dcef72200b9a8e58e5091744960da64ec734a6c6e9b3743e" + [[package]] name = "windows_i686_gnu" version = "0.52.5" @@ -381,24 +534,48 @@ version = "0.52.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "87f4261229030a858f36b459e748ae97545d6f1ec60e5e0d6a3d32e0dc232ee9" +[[package]] +name = "windows_i686_msvc" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f55c233f70c4b27f66c523580f78f1004e8b5a8b659e05a4eb49d4166cca406" + [[package]] name = "windows_i686_msvc" version = "0.52.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "db3c2bf3d13d5b658be73463284eaf12830ac9a26a90c717b7f771dfe97487bf" +[[package]] +name = "windows_x86_64_gnu" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "53d40abd2583d23e4718fddf1ebec84dbff8381c07cae67ff7768bbf19c6718e" + [[package]] name = "windows_x86_64_gnu" version = "0.52.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "4e4246f76bdeff09eb48875a0fd3e2af6aada79d409d33011886d3e1581517d9" +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b7b52767868a23d5bab768e390dc5f5c55825b6d30b86c844ff2dc7414044cc" + [[package]] name = "windows_x86_64_gnullvm" version = "0.52.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "852298e482cd67c356ddd9570386e2862b5673c85bd5f88df9ab6802b334c596" +[[package]] +name = "windows_x86_64_msvc" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed94fce61571a4006852b7389a063ab983c02eb1bb37b47f8272ce92d06d9538" + [[package]] name = "windows_x86_64_msvc" version = "0.52.5" diff --git a/Cargo.toml b/Cargo.toml index 1e23cd1..01080d6 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -4,5 +4,5 @@ version = "0.1.0" edition = "2021" [dependencies] +crossterm = "0.27.0" opencv = "0.92.0" -termion = "4.0.2" diff --git a/src/main.rs b/src/main.rs index 101db91..ac60fea 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,10 +1,12 @@ use opencv::core::VecN; use opencv::prelude::*; use opencv::{core, imgproc, videoio, Result}; -use std::io::{self, Write}; use std::sync::mpsc; use std::{thread, time}; +use crossterm::{execute, cursor::{MoveTo, Hide}, terminal::{Clear, ClearType}}; +use std::io::{stdout, Write}; + fn map_range(from_range: (i32, i32), to_range: (i32, i32), s: i32) -> i32 { to_range.0 + (s - from_range.0) * (to_range.1 - to_range.0) / (from_range.1 - from_range.0) } @@ -41,7 +43,7 @@ fn main() -> Result<()> { " .'`^\",:;Il!i><~+_-?][}{1)(|\\/tfjrxnuvczXYUJCLQ0OZmwqpdbkhao*#MW&8%B@$"; let ascii_table_len = ascii_table.len(); - let term_size = termion::terminal_size().unwrap(); + let term_size = crossterm::terminal::size().unwrap(); let mut cam = videoio::VideoCapture::from_file("baby-shark.webm", videoio::CAP_ANY)?; @@ -54,6 +56,7 @@ fn main() -> Result<()> { let frame_delay = time::Duration::from_millis((1000.0 / fps) as u64); let (tx, rx) = mpsc::channel(); + execute!(stdout(), Hide).unwrap(); thread::spawn(move || loop { let mut frame = Mat::default(); @@ -64,7 +67,7 @@ fn main() -> Result<()> { imgproc::resize( &frame, &mut smaller, - core::Size::new(term_size.0.into(), term_size.1.into()), + core::Size::new(term_size.0.into(), (term_size.1 - 2).into()), 0.0, 0.0, imgproc::INTER_AREA, @@ -80,9 +83,10 @@ fn main() -> Result<()> { }); for received in rx { + execute!(stdout(), MoveTo(0, 0)).unwrap(); + execute!(stdout(), Clear(ClearType::Purge)).unwrap(); print!("{}", received); thread::sleep(frame_delay); - print!("{}", termion::clear::All); } Ok(())