extern crate libc;
extern crate atty;
use self::libc::{STDOUT_FILENO, TIOCGWINSZ, c_ushort};
use self::libc::ioctl;
use self::super::Size;
#[repr(C)]
#[derive(Debug)]
pub struct UnixSize {
pub rows: c_ushort,
pub cols: c_ushort,
x: c_ushort,
y: c_ushort,
}
pub fn get() -> Option<Size> {
if atty::isnt(atty::Stream::Stdout) {
return None;
}
let us = UnixSize {
rows: 0,
cols: 0,
x: 0,
y: 0,
};
let r = unsafe { ioctl(STDOUT_FILENO, TIOCGWINSZ.into(), &us) };
if r == 0 {
Some(Size {
rows: us.rows,
cols: us.cols,
})
} else {
None
}
}
#[cfg(test)]
mod tests {
use super::get;
use super::super::Size;
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()
.unwrap()
}
#[test]
fn test_shell() {
let output = stty_size();
assert!(output.status.success());
let stdout = String::from_utf8(output.stdout).unwrap();
let mut data = stdout.split_whitespace();
let rs = data.next().unwrap().parse::<u16>().unwrap();
let cs = data.next().unwrap().parse::<u16>().unwrap();
if let Some(Size { rows, cols }) = get() {
assert_eq!(rows, rs);
assert_eq!(cols, cs);
}
}
}