This commit is contained in:
TimonPost 2018-08-23 23:01:33 +02:00
commit c9c0a4efac
14 changed files with 259 additions and 81 deletions

View File

@ -15,7 +15,7 @@ readme = "README.md"
winapi = { version = "0.3.5", features = ["winbase","winuser","consoleapi","processenv","wincon", "handleapi","errhandlingapi"] } winapi = { version = "0.3.5", features = ["winbase","winuser","consoleapi","processenv","wincon", "handleapi","errhandlingapi"] }
[target.'cfg(unix)'.dependencies] [target.'cfg(unix)'.dependencies]
libc = "0.2.37" libc = "0.2.43"
termios = "0.3.0" termios = "0.3.0"
[lib] [lib]

View File

@ -19,5 +19,7 @@ use std::io::Write;
use std::{thread,time}; use std::{thread,time};
fn main() fn main()
{ {
input::keyboard::async_input::read_async_demo();
terminal::raw_mode::print_wait_screen_on_alternate_window();
thread::sleep(time::Duration::from_millis(2000));
} }

View File

@ -1,9 +1,9 @@
extern crate crossterm; extern crate crossterm;
use crossterm::{Screen, Crossterm}; use crossterm::{Screen, Crossterm, screen};
use crossterm::terminal::{terminal,Terminal, ClearType}; use crossterm::terminal::{terminal,Terminal, ClearType};
use crossterm::cursor::TerminalCursor; use crossterm::cursor::{TerminalCursor, cursor};
use crossterm::input::input;
use std::sync::{Arc,Mutex}; use std::sync::{Arc,Mutex};
use std::io::Read; use std::io::Read;
use std::{thread,time}; use std::{thread,time};
@ -13,35 +13,36 @@ fn main() {
let screen = Screen::new(true); let screen = Screen::new(true);
let crossterm = Crossterm::new(&screen); let crossterm = Crossterm::new(&screen);
let input = crossterm.input();
let terminal = crossterm.terminal();
let cursor = crossterm.cursor(); let cursor = crossterm.cursor();
let mut stdin = input.read_async().bytes();
cursor.hide(); cursor.hide();
let mut input_buf = Arc::new(Mutex::new(String::new())); let mut input_buf = Arc::new(Mutex::new(String::new()));
let threads = log(input_buf.clone(),&screen);
let mut count = 0; let mut count = 0;
let threads = log(input_buf.clone()); thread::spawn(move || {
let input = input(&screen);
let mut stdin = input.read_async().bytes();
loop loop
{ {
let a = stdin.next(); let a = stdin.next();
match a { match a {
Some(Ok(b'\n')) => Some(Ok(13)) =>
{ {
input_buf.lock().unwrap().clear(); input_buf.lock().unwrap().clear();
// need to start receiving again because if pressed enter then async reading will stop // need to start receiving again because if pressed enter then async reading will stop
stdin = input.read_async().bytes(); // stdin = input.read_async().bytes();
} }
Some(Ok(val)) => Some(Ok(val)) =>
{ {
input_buf.lock().unwrap().push(val as char); // println!("{:?}",a);
input_buf.lock().unwrap().push(a.unwrap().unwrap() as char);
} }
_ => {} _ => {}
} }
@ -49,6 +50,8 @@ fn main() {
thread::sleep(time::Duration::from_millis(100)); thread::sleep(time::Duration::from_millis(100));
count += 1; count += 1;
} }
}).join();
for thread in threads for thread in threads
{ {
@ -58,18 +61,20 @@ fn main() {
cursor.show(); cursor.show();
} }
fn log(input_buf: Arc<Mutex<String>>) -> Vec<thread::JoinHandle<()>> fn log(input_buf: Arc<Mutex<String>>, screen: &Screen) -> Vec<thread::JoinHandle<()>>
{ {
let mut threads = Vec::with_capacity(10); let mut threads = Vec::with_capacity(10);
let (_, term_height) = terminal(&Screen::default()).terminal_size();
for i in 0..10 let (_, term_height) = terminal(&screen).terminal_size();
for i in 0..1
{ {
let input_buffer = input_buf.clone(); let input_buffer = input_buf.clone();
let clone_stdout = screen.stdout.clone();
let crossterm = Crossterm::from(screen.stdout.clone());
let join = thread::spawn( move || { let join = thread::spawn( move || {
let crossterm = Crossterm::new(&Screen::new(true));
let cursor = crossterm.cursor(); let cursor = crossterm.cursor();
let terminal = crossterm.terminal(); let terminal = crossterm.terminal();

View File

@ -0,0 +1,98 @@
extern crate crossterm;
use crossterm::{Screen, Crossterm};
use crossterm::terminal::{terminal,Terminal, ClearType};
use crossterm::cursor::TerminalCursor;
use std::sync::{Arc,Mutex};
use std::io::Read;
use std::{thread,time};
fn main() {
use crossterm::color;
let screen = Screen::new(true);
let crossterm = Arc::new(Crossterm::new(&screen));
let cursor = crossterm.cursor();
cursor.hide();
let mut input_buf = Arc::new(Mutex::new(String::new()));
let mut count = 0;
let threads = log(input_buf.clone(),crossterm.clone());
let crossterm_clone = crossterm.clone();
thread::spawn(move || {
let input = crossterm_clone.input();
let mut stdin = input.read_async().bytes();
loop
{
let a = stdin.next();
match a {
Some(Ok(13)) =>
{
input_buf.lock().unwrap().clear();
// need to start receiving again because if pressed enter then async reading will stop
// stdin = input.read_async().bytes();
}
Some(Ok(val)) =>
{
// println!("{}",val);
input_buf.lock().unwrap().push(a.unwrap().unwrap() as u8 as char);
}
_ => {}
}
thread::sleep(time::Duration::from_millis(100));
count += 1;
}
}).join();
for thread in threads
{
thread.join();
}
cursor.show();
}
fn log(input_buf: Arc<Mutex<String>>, crossterm: Arc<Crossterm>) -> Vec<thread::JoinHandle<()>>
{
let mut threads = Vec::with_capacity(10);
let (_, term_height) = crossterm.terminal().terminal_size();
for i in 0..1
{
let input_buffer = input_buf.clone();
let crossterm_clone = crossterm.clone();
let join = thread::spawn( move || {
let cursor = crossterm_clone.cursor();
let terminal = crossterm_clone.terminal();
for j in 0..1000
{
swap_write(format!("Some output: {} from thread: {}", j, i).as_ref(), &input_buffer.lock().unwrap(), &terminal, &cursor, term_height);
thread::sleep(time::Duration::from_millis(300));
}
});
threads.push(join);
}
return threads;
}
pub fn swap_write(msg: &str, input_buf: &String, terminal: &Terminal, cursor: &TerminalCursor, term_height: u16) {
cursor.goto(0, term_height);
terminal.clear(ClearType::CurrentLine);
terminal.write(format!("{}\r\n", msg));
terminal.write(format!(">{}", input_buf));
}

View File

@ -78,7 +78,9 @@ fn main() {
map.spawn_food(&free_positions, &screen); map.spawn_food(&free_positions, &screen);
} }
} }
drop(screen);
} }
game_over_screen(); game_over_screen();
} }

View File

@ -42,6 +42,6 @@ pub fn print_wait_screen_on_alternate_window() {
{ {
print_wait_screen(&mut alternate.screen); print_wait_screen(&mut alternate.screen);
} }
drop(screen);
println!("Whe are back at the main screen"); println!("Whe are back at the main screen");
} }

View File

@ -2,9 +2,11 @@
use super::{ IStateCommand}; use super::{ IStateCommand};
use kernel::unix_kernel::terminal; use kernel::unix_kernel::terminal;
use termios::{tcsetattr, Termios, CREAD, ECHO, ICANON, TCSAFLUSH}; use termios::{tcsetattr, Termios, CREAD, ECHO, ICANON, TCSAFLUSH, BRKINT, ICRNL, INPCK, ISTRIP, IXON, OPOST, CS8, IEXTEN, ISIG,VTIME, VMIN};
use libc::STDIN_FILENO;
use std::sync::{Once, ONCE_INIT};
static TERMINAL_MODE: Once = ONCE_INIT;
const FD_STDIN: ::std::os::unix::io::RawFd = 1;
use std::io::{Error, ErrorKind, Result}; use std::io::{Error, ErrorKind, Result};
@ -17,16 +19,25 @@ impl NoncanonicalModeCommand {
NoncanonicalModeCommand {} NoncanonicalModeCommand {}
} }
} }
static mut ORIGINAL: Option<Termios> = None;
impl NoncanonicalModeCommand { impl NoncanonicalModeCommand {
pub fn enable(&mut self) -> Result<()> { pub fn enable(&mut self) -> Result<()> {
// Set noncanonical mode // Set noncanonical mode
if let Ok(orig) = Termios::from_fd(FD_STDIN) { if let Ok(orig) = Termios::from_fd(STDIN_FILENO) {
TERMINAL_MODE.call_once(|| {
unsafe { ORIGINAL = Some(orig.clone()); }
});
let mut noncan = orig.clone(); let mut noncan = orig.clone();
noncan.c_lflag &= !ICANON; noncan.c_iflag &= !(BRKINT | ICRNL | INPCK | ISTRIP | IXON);
noncan.c_lflag &= !ECHO; noncan.c_oflag &= !(OPOST);
noncan.c_lflag &= !CREAD; noncan.c_cflag |= (CS8);
tcsetattr(FD_STDIN, TCSAFLUSH, &noncan)?; noncan.c_lflag &= !(ECHO | ICANON | IEXTEN | ISIG);
noncan.c_cc[VMIN] = 0;
noncan.c_cc[VTIME] = 1;
tcsetattr(STDIN_FILENO, TCSAFLUSH, &noncan)?;
} else { } else {
return Err(Error::new( return Err(Error::new(
ErrorKind::Other, ErrorKind::Other,
@ -37,20 +48,36 @@ impl NoncanonicalModeCommand {
} }
pub fn disable(&self) -> Result<()> { pub fn disable(&self) -> Result<()> {
// Disable noncanonical mode
if let Ok(orig) = Termios::from_fd(FD_STDIN) {
let mut noncan = orig.clone();
noncan.c_lflag &= ICANON;
noncan.c_lflag &= ECHO;
noncan.c_lflag &= CREAD;
tcsetattr(FD_STDIN, TCSAFLUSH, &noncan)?; unsafe {
} else { if let Some(original) = ORIGINAL
return Err(Error::new( {
ErrorKind::Other, tcsetattr(STDIN_FILENO, TCSAFLUSH, &original)?;
"Could not set console mode when enabling raw mode",
));
} }
}
Ok(())
}
}
// This command is used for enabling and disabling raw mode for the terminal.
/// This command is used for enabling and disabling raw mode for the terminal.
pub struct RawModeCommand;
impl RawModeCommand {
pub fn new() -> Self {
return RawModeCommand {}
}
/// Enables raw mode.
pub fn enable(&mut self) -> Result<()> {
terminal::into_raw_mode();
Ok(())
}
/// Disables raw mode.
pub fn disable(&mut self) -> Result<()> {
terminal::disable_raw_mode();
Ok(()) Ok(())
} }
} }

View File

@ -123,3 +123,10 @@ impl<'crossterm> Crossterm {
style::ObjectStyle::new().apply_to(val) style::ObjectStyle::new().apply_to(val)
} }
} }
impl From<Arc<TerminalOutput>> for Crossterm
{
fn from(stdout: Arc<TerminalOutput>) -> Self {
Crossterm { stdout: stdout }
}
}

View File

@ -28,11 +28,12 @@ impl RawScreen {
pub fn into_raw_mode() -> io::Result<()> pub fn into_raw_mode() -> io::Result<()>
{ {
#[cfg(not(target_os = "windows"))] #[cfg(not(target_os = "windows"))]
let mut command = unix_command::NoncanonicalModeCommand::new(); let mut command = unix_command::RawModeCommand::new();
#[cfg(target_os = "windows")] #[cfg(target_os = "windows")]
let mut command = win_commands::RawModeCommand::new(); let mut command = win_commands::RawModeCommand::new();
command.enable()?; let result = command.enable();
Ok(()) Ok(())
} }
@ -40,9 +41,9 @@ impl RawScreen {
pub fn disable_raw_modes() -> io::Result<()> pub fn disable_raw_modes() -> io::Result<()>
{ {
#[cfg(not(target_os = "windows"))] #[cfg(not(target_os = "windows"))]
let command = unix_command::NoncanonicalModeCommand::new(); let mut command = unix_command::RawModeCommand::new();
#[cfg(target_os = "windows")] #[cfg(target_os = "windows")]
let command = win_commands::RawModeCommand::new(); let mut command = win_commands::RawModeCommand::new();
command.disable()?; command.disable()?;
return Ok(()) return Ok(())

View File

@ -57,7 +57,7 @@ impl Screen
if raw_mode if raw_mode
{ {
let screen = Screen { stdout: Arc::new(TerminalOutput::new(true)), buffer: Vec::new() }; let screen = Screen { stdout: Arc::new(TerminalOutput::new(true)), buffer: Vec::new() };
RawScreen::into_raw_mode(); RawScreen::into_raw_mode().unwrap();
return screen; return screen;
} }

View File

@ -107,35 +107,70 @@ pub fn make_raw(termios: &mut Termios) {
static mut ORIGINAL_TERMINAL_MODE: Option<Termios> = None; static mut ORIGINAL_TERMINAL_MODE: Option<Termios> = None;
/// Get the current terminal mode. pub fn into_raw_mode() -> io::Result<()>
pub fn get_terminal_mode() -> io::Result<Termios> { {
extern "C" { let tty_f;
pub fn tcgetattr(fd: c_int, termptr: *mut Termios) -> c_int;
let fd = unsafe {
if libc::isatty(libc::STDIN_FILENO) == 1 {
libc::STDIN_FILENO
} else {
tty_f = fs::File::open("/dev/tty")?;
tty_f.as_raw_fd()
} }
};
let mut termios = Termios::from_fd(fd)?;
let original = termios.clone();
unsafe { unsafe {
if let Some(original_mode) = ORIGINAL_TERMINAL_MODE if let None = ORIGINAL_TERMINAL_MODE
{ {
return Ok(original_mode.clone()) ORIGINAL_TERMINAL_MODE = Some(original.clone())
} }
else {
let mut termios = mem::zeroed();
is_true(tcgetattr(0, &mut termios))?;
ORIGINAL_TERMINAL_MODE = Some(termios.clone());
return Ok(termios)
} }
make_raw(&mut termios);
tcsetattr(fd, TCSADRAIN, &termios)?;
Ok(())
}
pub fn disable_raw_mode() -> io::Result<()>
{
let tty_f;
let fd = unsafe {
if libc::isatty(libc::STDIN_FILENO) == 1 {
libc::STDIN_FILENO
} else {
tty_f = fs::File::open("/dev/tty")?;
tty_f.as_raw_fd()
} }
};
if let Some(original) = unsafe { ORIGINAL_TERMINAL_MODE }
{
tcsetattr(fd, TCSADRAIN, &original)?;
}
Ok(())
} }
/// Get the TTY device. /// Get the TTY device.
/// ///
/// This allows for getting stdio representing _only_ the TTY, and not other streams. /// This allows for getting stdio representing _only_ the TTY, and not other streams.
pub fn get_tty() -> io::Result<fs::File> { pub fn get_tty() -> io::Result<fs::File> {
fs::OpenOptions::new() let mut tty_f: fs::File = unsafe { ::std::mem::zeroed() };
.read(true)
.write(true) let fd = unsafe {
.open("/dev/tty") if libc::isatty(libc::STDIN_FILENO) == 1 {
libc::STDIN_FILENO
} else {
tty_f = fs::File::open("/dev/tty")?;
tty_f.as_raw_fd()
}
};
return Ok(tty_f);
} }
pub fn read_char() -> io::Result<char> { pub fn read_char() -> io::Result<char> {

View File

@ -6,7 +6,6 @@ static mut HAS_BEEN_TRIED_TO_ENABLE: bool = false;
static mut IS_ANSI_ON_WINDOWS_ENABLED: Option<bool> = None; static mut IS_ANSI_ON_WINDOWS_ENABLED: Option<bool> = None;
static mut DOES_WINDOWS_SUPPORT_ANSI: Option<bool> = None; static mut DOES_WINDOWS_SUPPORT_ANSI: Option<bool> = None;
static ENABLE_ANSI: Once = ONCE_INIT; static ENABLE_ANSI: Once = ONCE_INIT;
use common::commands::win_commands::EnableAnsiCommand; use common::commands::win_commands::EnableAnsiCommand;
use common::commands::IEnableAnsiCommand; use common::commands::IEnableAnsiCommand;

View File

@ -5,6 +5,7 @@ use super::*;
/// This struct is an ansi escape code implementation for terminal related actions. /// This struct is an ansi escape code implementation for terminal related actions.
pub struct AnsiTerminal; pub struct AnsiTerminal;
use cursor::TerminalCursor;
impl AnsiTerminal { impl AnsiTerminal {
pub fn new() -> AnsiTerminal { pub fn new() -> AnsiTerminal {
@ -17,6 +18,7 @@ impl ITerminal for AnsiTerminal {
match clear_type { match clear_type {
ClearType::All => { ClearType::All => {
stdout.write_str(csi!("2J")); stdout.write_str(csi!("2J"));
TerminalCursor::new(stdout).goto(0,0);
} }
ClearType::FromCursorDown => { ClearType::FromCursorDown => {
stdout.write_str(csi!("J")); stdout.write_str(csi!("J"));