#![allow(clippy::useless_conversion)]
use libc::{STDERR_FILENO, STDIN_FILENO, STDOUT_FILENO, TIOCGWINSZ, ioctl, winsize};
use std::{fs::File, mem::zeroed, os::fd::AsRawFd};
unsafe fn get_dimensions_any() -> winsize {
let mut window: winsize = unsafe { zeroed() };
let mut result = unsafe { ioctl(STDOUT_FILENO, TIOCGWINSZ.into(), &mut window) };
if result == -1 {
window = unsafe { zeroed() };
result = unsafe { ioctl(STDIN_FILENO, TIOCGWINSZ.into(), &mut window) };
if result == -1 {
window = unsafe { zeroed() };
result = unsafe { ioctl(STDERR_FILENO, TIOCGWINSZ.into(), &mut window) };
if result == -1 {
window = unsafe { zeroed() };
let Ok(tty) = File::options().read(true).open("/dev/tty") else {
return unsafe { zeroed() };
};
result = unsafe { ioctl(tty.as_raw_fd(), TIOCGWINSZ.into(), &mut window) };
if result == -1 {
return unsafe { zeroed() };
}
}
}
}
window
}
unsafe fn get_dimensions_out() -> winsize {
let mut window: winsize = unsafe { zeroed() };
let result = unsafe { ioctl(STDOUT_FILENO, TIOCGWINSZ.into(), &mut window) };
if result != -1 {
return window;
}
unsafe { zeroed() }
}
unsafe fn get_dimensions_in() -> winsize {
let mut window: winsize = unsafe { zeroed() };
let result = unsafe { ioctl(STDIN_FILENO, TIOCGWINSZ.into(), &mut window) };
if result != -1 {
return window;
}
unsafe { zeroed() }
}
unsafe fn get_dimensions_err() -> winsize {
let mut window: winsize = unsafe { zeroed() };
let result = unsafe { ioctl(STDERR_FILENO, TIOCGWINSZ.into(), &mut window) };
if result != -1 {
return window;
}
unsafe { zeroed() }
}
pub fn dimensions() -> Option<(usize, usize)> {
let w = unsafe { get_dimensions_any() };
if w.ws_col == 0 || w.ws_row == 0 {
None
} else {
Some((w.ws_col as usize, w.ws_row as usize))
}
}
pub fn dimensions_stdout() -> Option<(usize, usize)> {
let w = unsafe { get_dimensions_out() };
if w.ws_col == 0 || w.ws_row == 0 {
None
} else {
Some((w.ws_col as usize, w.ws_row as usize))
}
}
pub fn dimensions_stdin() -> Option<(usize, usize)> {
let w = unsafe { get_dimensions_in() };
if w.ws_col == 0 || w.ws_row == 0 {
None
} else {
Some((w.ws_col as usize, w.ws_row as usize))
}
}
pub fn dimensions_stderr() -> Option<(usize, usize)> {
let w = unsafe { get_dimensions_err() };
if w.ws_col == 0 || w.ws_row == 0 {
None
} else {
Some((w.ws_col as usize, w.ws_row as usize))
}
}
#[cfg(test)]
mod tests {
use super::dimensions;
use std::process::{Command, Output, Stdio};
#[cfg(target_os = "macos")]
fn stty_size() -> Output {
Command::new("stty")
.arg("-f")
.arg("/dev/stderr")
.arg("size")
.stderr(Stdio::inherit())
.output()
.unwrap()
}
#[cfg(not(target_os = "macos"))]
fn stty_size() -> Output {
Command::new("stty")
.arg("-F")
.arg("/dev/stderr")
.arg("size")
.stderr(Stdio::inherit())
.output()
.expect("failed to run `stty_size()`")
}
#[test]
fn test_shell() {
let output = stty_size();
let stdout = String::from_utf8(output.stdout).expect("failed to turn into String");
let mut data = stdout.split_whitespace();
let rs = data
.next()
.unwrap_or("0")
.parse::<usize>()
.expect("failed to parse rows");
let cs = data
.next()
.unwrap_or("0")
.parse::<usize>()
.expect("failed to parse cols");
println!("stdout: {}", stdout);
println!("rows: {}\ncols: {}", rs, cs);
if let Some((w, h)) = dimensions() {
assert_eq!(rs, h);
assert_eq!(cs, w);
}
}
}