Created terminal type and changed alternate and raw screen
This commit is contained in:
parent
d38f406dc7
commit
403d0668a7
@ -12,93 +12,22 @@
|
||||
//! - Run program with: `cargo run`
|
||||
extern crate crossterm;
|
||||
|
||||
use crossterm::Context;
|
||||
use crossterm::Terminal;
|
||||
|
||||
// mod terminal;
|
||||
// mod color;
|
||||
// mod cursor;
|
||||
// mod crossterm_type;
|
||||
mod input;
|
||||
// mod input;
|
||||
|
||||
use input::keyboard::{async_input, input as stdin};
|
||||
|
||||
use crossterm::raw::IntoRawMode;
|
||||
//use input::keyboard::{async_input, input as stdin};
|
||||
|
||||
use std::{thread, time};
|
||||
|
||||
fn main() {
|
||||
// let context = Context::new();
|
||||
crossterm();
|
||||
{
|
||||
// let screen = ::crossterm::screen::AlternateScreen::from(context.clone());
|
||||
// screen.into_raw_mode(context.clone());
|
||||
|
||||
// async_input::async_reading_on_alternate_screen();
|
||||
// async_input::test();
|
||||
// stdin::t();
|
||||
// stdin::read_line();
|
||||
// stdin::read_char();
|
||||
// stdin::read_char();
|
||||
}
|
||||
}
|
||||
|
||||
use crossterm::raw::RawTerminal;
|
||||
use crossterm::Crossterm;
|
||||
|
||||
|
||||
use crossterm::cursor::cursor::TerminalCursor;
|
||||
use crossterm::terminal::terminal::Terminal;
|
||||
use crossterm::terminal::ClearType;
|
||||
use std::io::Read;
|
||||
|
||||
pub fn crossterm() {
|
||||
let crossterm = Crossterm::new();
|
||||
let mut term = crossterm.terminal();
|
||||
let mut cursor = crossterm.cursor();
|
||||
let input = crossterm.input();
|
||||
|
||||
// clear screen
|
||||
term.clear(ClearType::All);
|
||||
|
||||
let mut raw_screen = RawTerminal::new(&crossterm.context());
|
||||
raw_screen.enable();
|
||||
|
||||
let mut stdin = input.read_until_async().bytes();
|
||||
|
||||
let mut buf = String::new();
|
||||
|
||||
let (term_x, term_y) = term.terminal_size();
|
||||
let mut command_bar_y = term_y;
|
||||
let (curs_x, curs_y) = cursor.pos();
|
||||
|
||||
let mut counter: u16 = 0 + curs_y;
|
||||
loop {
|
||||
cursor.goto(0, counter);
|
||||
let (curs_x, curs_y) = cursor.pos();
|
||||
term.write(format!("cursor pos {} term pos: {} command pos: {}", curs_y, term_y, command_bar_y));
|
||||
cursor.goto(0, counter + 1);
|
||||
|
||||
if (curs_y >= term_y - 1 )
|
||||
{
|
||||
cursor.goto(0, counter + 1);
|
||||
term.clear(ClearType::CurrentLine);
|
||||
cursor.goto(0, counter + 2);
|
||||
term.write(format!("> {}", buf));
|
||||
term.scroll_up(1);
|
||||
}
|
||||
|
||||
while let Some(b) = stdin.next() {
|
||||
if let Ok(b) = b {
|
||||
if b == 3 {
|
||||
term.exit();
|
||||
} else if b == 13 {
|
||||
buf.clear();
|
||||
} else {
|
||||
buf.push(b as char);
|
||||
}
|
||||
}
|
||||
}
|
||||
counter += 1;
|
||||
thread::sleep(time::Duration::from_millis(100));
|
||||
let mut terminal = Terminal::new();
|
||||
terminal.enable_alternate_screen();
|
||||
thread::sleep(time::Duration::from_millis(5000));
|
||||
}
|
||||
}
|
@ -6,20 +6,20 @@
|
||||
use super::super::shared::functions;
|
||||
use super::*;
|
||||
use std::io::Write;
|
||||
use Context;
|
||||
use {Context, ScreenManager};
|
||||
|
||||
use std::fmt::Display;
|
||||
use std::rc::Rc;
|
||||
|
||||
/// Struct that stores an specific platform implementation for cursor related actions.
|
||||
pub struct TerminalCursor {
|
||||
context: Rc<Context>,
|
||||
pub struct TerminalCursor<'cursor> {
|
||||
context: &'cursor ScreenManager,
|
||||
terminal_cursor: Box<ITerminalCursor>,
|
||||
}
|
||||
|
||||
impl TerminalCursor {
|
||||
impl<'cursor> TerminalCursor<'cursor> {
|
||||
/// Create new cursor instance whereon cursor related actions can be performed.
|
||||
pub fn new(context: Rc<Context>) -> TerminalCursor {
|
||||
pub fn new(context: &'cursor ScreenManager) -> TerminalCursor<'cursor> {
|
||||
#[cfg(target_os = "windows")]
|
||||
let cursor = functions::get_module::<Box<ITerminalCursor>>(
|
||||
WinApiCursor::new(context.screen_manager.clone()),
|
||||
|
@ -31,28 +31,26 @@ use std::rc::Rc;
|
||||
///! so that cursor related actions can be preformed on both unix and windows systems.
|
||||
pub trait ITerminalCursor {
|
||||
/// Goto some location (x,y) in the context.
|
||||
fn goto(&self, x: u16, y: u16);
|
||||
fn goto(&self, x: u16, y: u16, screen_manager: &ScreenManager);
|
||||
/// Get the location (x,y) of the current cusror in the context
|
||||
fn pos(&self) -> (u16, u16);
|
||||
|
||||
fn absolute_pos(&self) -> (u16, u16);
|
||||
|
||||
fn pos(&self, screen_manager: &ScreenManager) -> (u16, u16);
|
||||
fn absolute_pos(&self, screen_manager: &ScreenManager) -> (u16, u16);
|
||||
/// Move cursor n times up
|
||||
fn move_up(&self, count: u16);
|
||||
fn move_up(&self, count: u16, screen_manager: &ScreenManager);
|
||||
/// Move the cursor `n` times to the right.
|
||||
fn move_right(&self, count: u16);
|
||||
fn move_right(&self, count: u16, screen_manager: &ScreenManager);
|
||||
/// Move the cursor `n` times down.
|
||||
fn move_down(&self, count: u16);
|
||||
fn move_down(&self, count: u16, screen_manager: &ScreenManager);
|
||||
/// Move the cursor `n` times left.
|
||||
fn move_left(&self, count: u16);
|
||||
fn move_left(&self, count: u16, screen_manager: &ScreenManager);
|
||||
/// Save cursor position so that its saved position can be recalled later. Note that this position is stored program based not per instance of the cursor struct.
|
||||
fn save_position(&self);
|
||||
fn save_position(&self, screen_manager: &ScreenManager);
|
||||
/// Return to saved cursor position
|
||||
fn reset_position(&self);
|
||||
fn reset_position(&self, screen_manager: &ScreenManager);
|
||||
/// Hide the terminal cursor.
|
||||
fn hide(&self);
|
||||
fn hide(&self, screen_manager: &ScreenManager);
|
||||
/// Show the terminal cursor
|
||||
fn show(&self);
|
||||
fn show(&self, screen_manager: &ScreenManager);
|
||||
/// enable or disable the blinking of the cursor.
|
||||
fn blink(&self, blink: bool);
|
||||
fn blink(&self, blink: bool, screen_manager: &ScreenManager);
|
||||
}
|
||||
|
@ -10,65 +10,63 @@ use std::rc::Rc;
|
||||
use std::sync::Mutex;
|
||||
|
||||
/// This struct is an windows implementation for cursor related actions.
|
||||
pub struct WinApiCursor {
|
||||
screen_manager: Rc<Mutex<ScreenManager>>,
|
||||
}
|
||||
pub struct WinApiCursor;
|
||||
|
||||
impl WinApiCursor {
|
||||
pub fn new(screen_manager: Rc<Mutex<ScreenManager>>) -> Box<WinApiCursor> {
|
||||
Box::from(WinApiCursor { screen_manager })
|
||||
pub fn new() -> Box<WinApiCursor> {
|
||||
Box::from(WinApiCursor { })
|
||||
}
|
||||
}
|
||||
|
||||
impl ITerminalCursor for WinApiCursor {
|
||||
fn goto(&self, x: u16, y: u16) {
|
||||
fn goto(&self, x: u16, y: u16, screen_manager: &ScreenManager) {
|
||||
cursor::set_console_cursor_position(x as i16, y as i16, &self.screen_manager);
|
||||
}
|
||||
|
||||
fn pos(&self) -> (u16, u16) {
|
||||
fn pos(&self, screen_manager: &ScreenManager) -> (u16, u16) {
|
||||
cursor::pos(&self.screen_manager)
|
||||
}
|
||||
|
||||
fn absolute_pos(&self) -> (u16, u16)
|
||||
fn absolute_pos(&self, screen_manager: &ScreenManager) -> (u16, u16)
|
||||
{
|
||||
cursor::absolute_cursor_pos(&self.screen_manager)
|
||||
}
|
||||
|
||||
fn move_up(&self, count: u16) {
|
||||
fn move_up(&self, count: u16, screen_manager: &ScreenManager) {
|
||||
let (xpos, ypos) = self.pos();
|
||||
self.goto(xpos, ypos - count);
|
||||
}
|
||||
|
||||
fn move_right(&self, count: u16) {
|
||||
fn move_right(&self, count: u16, screen_manager: &ScreenManager) {
|
||||
let (xpos, ypos) = self.pos();
|
||||
self.goto(xpos + count, ypos);
|
||||
}
|
||||
|
||||
fn move_down(&self, count: u16) {
|
||||
fn move_down(&self, count: u16, screen_manager: &ScreenManager) {
|
||||
let (xpos, ypos) = self.pos();
|
||||
self.goto(xpos, ypos + count);
|
||||
}
|
||||
|
||||
fn move_left(&self, count: u16) {
|
||||
fn move_left(&self, count: u16, screen_manager: &ScreenManager) {
|
||||
let (xpos, ypos) = self.pos();
|
||||
self.goto(xpos - count, ypos);
|
||||
}
|
||||
|
||||
fn save_position(&self) {
|
||||
fn save_position(&self, screen_manager: &ScreenManager) {
|
||||
cursor::save_cursor_pos(&self.screen_manager);
|
||||
}
|
||||
|
||||
fn reset_position(&self) {
|
||||
fn reset_position(&self, screen_manager: &ScreenManager) {
|
||||
cursor::reset_to_saved_position(&self.screen_manager);
|
||||
}
|
||||
|
||||
fn hide(&self) {
|
||||
fn hide(&self, screen_manager: &ScreenManager) {
|
||||
cursor::cursor_visibility(false, &self.screen_manager);
|
||||
}
|
||||
|
||||
fn show(&self) {
|
||||
fn show(&self, screen_manager: &ScreenManager) {
|
||||
cursor::cursor_visibility(true, &self.screen_manager);
|
||||
}
|
||||
|
||||
fn blink(&self, blink: bool) {}
|
||||
fn blink(&self, blink: bool, screen_manager: &ScreenManager) {}
|
||||
}
|
||||
|
@ -14,6 +14,7 @@ pub mod manager;
|
||||
pub mod style;
|
||||
pub mod terminal;
|
||||
|
||||
pub use shared::Terminal::Terminal;
|
||||
pub use shared::crossterm::Crossterm;
|
||||
pub use shared::raw;
|
||||
pub use shared::screen;
|
||||
|
110
src/shared/Terminal.rs
Normal file
110
src/shared/Terminal.rs
Normal file
@ -0,0 +1,110 @@
|
||||
use {StateManager, ScreenManager};
|
||||
use super::super::state::commands::*;
|
||||
use super::raw::RawTerminal;
|
||||
use super::screen::AlternateScreen;
|
||||
use std::collections::HashMap;
|
||||
|
||||
use std::io::Result;
|
||||
|
||||
pub struct Terminal{
|
||||
raw_mode: bool,
|
||||
alternate_mode: bool,
|
||||
active_screen: ScreenManager,
|
||||
raw_terminal: Option<Box<IRawScreenCommand>>,
|
||||
alternate_screen: Option<Box<IAlternateScreenCommand>>
|
||||
}
|
||||
|
||||
impl Terminal{
|
||||
pub fn new() -> Terminal
|
||||
{
|
||||
Terminal
|
||||
{
|
||||
raw_mode: false,
|
||||
alternate_mode: false,
|
||||
active_screen: ScreenManager::new(),
|
||||
raw_terminal: None,
|
||||
alternate_screen: None,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn enable_raw_mode(&mut self) -> Result<()> {
|
||||
match self.raw_terminal
|
||||
{
|
||||
None => {
|
||||
self.raw_terminal = Some(RawTerminal::new());
|
||||
return self.enable_raw_mode();
|
||||
},
|
||||
Some(ref mut raw_terminal) => {
|
||||
raw_terminal.enable()?;
|
||||
self.raw_mode = true;
|
||||
},
|
||||
}
|
||||
|
||||
return Ok(())
|
||||
}
|
||||
|
||||
pub fn disable_raw_mode(&mut self) -> Result<()>
|
||||
{
|
||||
match self.raw_terminal
|
||||
{
|
||||
None => {
|
||||
self.raw_terminal = Some(RawTerminal::new());
|
||||
return self.disable_raw_mode();
|
||||
},
|
||||
Some(ref mut raw_terminal) => {
|
||||
raw_terminal.disable()?;
|
||||
self.raw_mode = false;
|
||||
},
|
||||
}
|
||||
|
||||
return Ok(())
|
||||
}
|
||||
|
||||
pub fn enable_alternate_screen(&mut self) -> Result<()>
|
||||
{
|
||||
match self.alternate_screen
|
||||
{
|
||||
None => {
|
||||
self.alternate_screen = Some(AlternateScreen::new());
|
||||
return self.enable_alternate_screen();
|
||||
},
|
||||
Some(ref mut alternate_screen) => {
|
||||
alternate_screen.to_alternate_screen(&mut self.active_screen)?;
|
||||
self.alternate_mode = true;
|
||||
},
|
||||
}
|
||||
|
||||
return Ok(())
|
||||
}
|
||||
|
||||
pub fn disable_alternate_screen(&mut self) -> Result<()>
|
||||
{
|
||||
match self.alternate_screen
|
||||
{
|
||||
None => {
|
||||
self.alternate_screen = Some(AlternateScreen::new());
|
||||
return self.disable_alternate_screen();
|
||||
},
|
||||
Some(ref mut alternate_screen) => {
|
||||
alternate_screen.to_main_screen(&mut self.active_screen)?;
|
||||
self.alternate_mode = false;
|
||||
},
|
||||
}
|
||||
|
||||
return Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
impl Drop for Terminal
|
||||
{
|
||||
fn drop(&mut self) {
|
||||
if let Some(ref mut screen) = self.alternate_screen
|
||||
{
|
||||
screen.to_main_screen(&mut self.active_screen);
|
||||
}
|
||||
if let Some(ref mut raw_terminal) = self.raw_terminal
|
||||
{
|
||||
raw_terminal.disable();
|
||||
}
|
||||
}
|
||||
}
|
@ -8,3 +8,4 @@ pub mod traits;
|
||||
|
||||
pub mod raw;
|
||||
pub mod screen;
|
||||
pub mod Terminal;
|
||||
|
@ -17,99 +17,39 @@ use super::super::state::commands::unix_command::EnableRawModeCommand;
|
||||
#[cfg(windows)]
|
||||
use state::commands::win_commands::EnableRawModeCommand;
|
||||
|
||||
use state::commands::IStateCommand;
|
||||
use {CommandManager, Context};
|
||||
use state::commands::IRawScreenCommand;
|
||||
use {CommandManager, Context, ScreenManager };
|
||||
|
||||
use std::io::{self, Write};
|
||||
use std::rc::Rc;
|
||||
|
||||
/// A wrapper for the raw terminal state. Which can be used to write to.
|
||||
pub struct RawTerminal {
|
||||
context: Rc<Context>,
|
||||
command_id: u16,
|
||||
}
|
||||
pub struct RawTerminal;
|
||||
|
||||
impl RawTerminal {
|
||||
pub fn new(context: &Rc<Context>) -> RawTerminal {
|
||||
let command_id = EnableRawModeCommand::new(&context.state_manager);
|
||||
pub fn new() -> Box<IRawScreenCommand> {
|
||||
Box::from(EnableRawModeCommand::new())
|
||||
}
|
||||
}
|
||||
//
|
||||
///// Trait withs contains a method for switching into raw mode.
|
||||
//pub trait IntoRawMode: Write + Sized {
|
||||
// fn into_raw_mode(&self, context: Rc<Context>) -> io::Result<RawTerminal>;
|
||||
//}
|
||||
//
|
||||
//impl<W: Write> IntoRawMode for W {
|
||||
// /// Raw mode means that input (stdin) won't be printed it will instead have to be written manually by
|
||||
// /// the program. The input isn't canonicalised or line buffered (that is, you can
|
||||
// /// read from input(stdin) one byte of a time).
|
||||
// fn into_raw_mode(&self, context: Rc<Context>) -> io::Result<RawTerminal> {
|
||||
// let raw_terminal = RawTerminal::new();
|
||||
//
|
||||
// if raw_terminal.enable()
|
||||
// {
|
||||
// return Ok(raw_terminal);
|
||||
// }
|
||||
//
|
||||
// return Err(io::Error::new(io::ErrorKind::Other, "Could not enter raw mode."))
|
||||
// }
|
||||
//}
|
||||
|
||||
RawTerminal {
|
||||
context: context.clone(),
|
||||
command_id: command_id,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn enable(&self) -> bool {
|
||||
{
|
||||
let mutex = &self.context.screen_manager;
|
||||
let mut screen = mutex.lock().unwrap();
|
||||
screen.set_is_raw_screen(true);
|
||||
}
|
||||
|
||||
CommandManager::execute(self.context.clone(), self.command_id)
|
||||
}
|
||||
|
||||
pub fn disable(&self) -> bool {
|
||||
{
|
||||
let mutex = &self.context.screen_manager;
|
||||
let mut screen = mutex.lock().unwrap();
|
||||
screen.set_is_raw_screen(false);
|
||||
}
|
||||
|
||||
CommandManager::undo(self.context.clone(), self.command_id)
|
||||
}
|
||||
}
|
||||
|
||||
/// Trait withs contains a method for switching into raw mode.
|
||||
pub trait IntoRawMode: Write + Sized {
|
||||
fn into_raw_mode(&self, context: Rc<Context>) -> io::Result<RawTerminal>;
|
||||
}
|
||||
|
||||
impl<W: Write> IntoRawMode for W {
|
||||
/// Raw mode means that input (stdin) won't be printed it will instead have to be written manually by
|
||||
/// the program. The input isn't canonicalised or line buffered (that is, you can
|
||||
/// read from input(stdin) one byte of a time).
|
||||
fn into_raw_mode(&self, context: Rc<Context>) -> io::Result<RawTerminal> {
|
||||
let command_id = EnableRawModeCommand::new(&context.state_manager);
|
||||
|
||||
{
|
||||
let mutex = &context.screen_manager;
|
||||
let mut screen = mutex.lock().unwrap();
|
||||
screen.set_is_raw_screen(true);
|
||||
}
|
||||
|
||||
let raw_terminal = RawTerminal {
|
||||
context: context.clone(),
|
||||
command_id: command_id,
|
||||
};
|
||||
|
||||
if raw_terminal.enable()
|
||||
{
|
||||
return Ok(raw_terminal);
|
||||
}
|
||||
return Err(io::Error::new(io::ErrorKind::Other, "Could not enter raw mode."))
|
||||
}
|
||||
}
|
||||
|
||||
impl Write for RawTerminal {
|
||||
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
|
||||
let mut screen = self.context.screen_manager.lock().unwrap();
|
||||
{
|
||||
screen.write(buf)
|
||||
}
|
||||
}
|
||||
|
||||
fn flush(&mut self) -> io::Result<()> {
|
||||
let mut screen = self.context.screen_manager.lock().unwrap();
|
||||
{
|
||||
screen.flush()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// If an instance of `RawTerminal` will be dropped all terminal changes that are made will be undone.
|
||||
impl Drop for RawTerminal {
|
||||
fn drop(&mut self) {
|
||||
let success = CommandManager::undo(self.context.clone(), self.command_id);
|
||||
}
|
||||
}
|
||||
|
@ -85,104 +85,43 @@
|
||||
|
||||
use shared::functions;
|
||||
use state::commands::*;
|
||||
use {CommandManager, Context};
|
||||
use {CommandManager, Context,ScreenManager};
|
||||
|
||||
use std::convert::From;
|
||||
use std::io::{self, Write};
|
||||
use std::rc::Rc;
|
||||
|
||||
pub struct AlternateScreen {
|
||||
context: Rc<Context>,
|
||||
command_id: u16,
|
||||
}
|
||||
pub struct AlternateScreen;
|
||||
|
||||
impl AlternateScreen {
|
||||
/// Get the alternate screen from the context.
|
||||
/// By calling this method the current screen will be changed to the alternate screen.
|
||||
/// And you get back an handle for that screen.
|
||||
pub fn from(context: Rc<Context>) -> Self {
|
||||
let command_id = get_to_alternate_screen_command(context.clone());
|
||||
pub fn new() -> Box<IAlternateScreenCommand> {
|
||||
let command = functions::get_module::<Box<IAlternateScreenCommand>>(
|
||||
win_commands::ToAlternateScreenBufferCommand::new(),
|
||||
shared_commands::ToAlternateScreenBufferCommand::new(),
|
||||
).unwrap();
|
||||
|
||||
let screen = AlternateScreen {
|
||||
context: context.clone(),
|
||||
command_id: command_id,
|
||||
};
|
||||
#[cfg(not(target_os = "windows"))]
|
||||
let command = shared_commands::ToAlternateScreenBufferCommand::new();
|
||||
|
||||
screen.to_alternate();
|
||||
|
||||
return screen;
|
||||
}
|
||||
|
||||
/// Change the current screen to the mainscreen.
|
||||
pub fn to_main(&self) {
|
||||
{
|
||||
let mutex = &self.context.screen_manager;
|
||||
let mut screen = mutex.lock().unwrap();
|
||||
screen.set_is_alternate_screen(false);
|
||||
}
|
||||
|
||||
CommandManager::undo(self.context.clone(), self.command_id);
|
||||
}
|
||||
|
||||
/// Change the current screen to alternate screen.
|
||||
pub fn to_alternate(&self) {
|
||||
{
|
||||
let mutex = &self.context.screen_manager;
|
||||
let mut screen = mutex.lock().unwrap();
|
||||
screen.set_is_alternate_screen(true);
|
||||
}
|
||||
CommandManager::execute(self.context.clone(), self.command_id);
|
||||
}
|
||||
}
|
||||
|
||||
impl Write for AlternateScreen {
|
||||
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
|
||||
let mut screen = self.context.screen_manager.lock().unwrap();
|
||||
{
|
||||
screen.write(buf)
|
||||
}
|
||||
}
|
||||
|
||||
fn flush(&mut self) -> io::Result<()> {
|
||||
let mut screen = self.context.screen_manager.lock().unwrap();
|
||||
{
|
||||
screen.flush()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl Drop for AlternateScreen {
|
||||
fn drop(&mut self) {
|
||||
use CommandManager;
|
||||
CommandManager::undo(self.context.clone(), self.command_id);
|
||||
command
|
||||
}
|
||||
}
|
||||
|
||||
use super::super::shared::crossterm::Crossterm;
|
||||
|
||||
impl From<Crossterm> for AlternateScreen {
|
||||
fn from(crossterm: Crossterm) -> Self {
|
||||
let command_id = get_to_alternate_screen_command(crossterm.context());
|
||||
//impl From<Crossterm> for AlternateScreen {
|
||||
// fn from(crossterm: Crossterm) -> Self {
|
||||
// let command_id = get_to_alternate_screen_command(crossterm.context());
|
||||
//
|
||||
// let screen = AlternateScreen {
|
||||
// context: crossterm.context(),
|
||||
// command_id: command_id,
|
||||
// };
|
||||
// screen.to_alternate();
|
||||
// return screen;
|
||||
// }
|
||||
//}
|
||||
|
||||
let screen = AlternateScreen {
|
||||
context: crossterm.context(),
|
||||
command_id: command_id,
|
||||
};
|
||||
screen.to_alternate();
|
||||
return screen;
|
||||
}
|
||||
}
|
||||
|
||||
// Get the alternate screen command to enable and disable alternate screen based on the current platform
|
||||
fn get_to_alternate_screen_command(context: Rc<Context>) -> u16 {
|
||||
#[cfg(target_os = "windows")]
|
||||
let command_id = functions::get_module::<u16>(
|
||||
win_commands::ToAlternateScreenBufferCommand::new(context.clone()),
|
||||
shared_commands::ToAlternateScreenBufferCommand::new(context.clone()),
|
||||
).unwrap();
|
||||
|
||||
#[cfg(not(target_os = "windows"))]
|
||||
let command_id = shared_commands::ToAlternateScreenBufferCommand::new(context.clone());
|
||||
|
||||
return command_id;
|
||||
}
|
||||
|
@ -31,3 +31,18 @@ pub trait IStateCommand {
|
||||
fn execute(&mut self) -> bool;
|
||||
fn undo(&mut self) -> bool;
|
||||
}
|
||||
|
||||
use ScreenManager;
|
||||
use std::io::Result;
|
||||
|
||||
pub trait IAlternateScreenCommand
|
||||
{
|
||||
fn to_alternate_screen(&self,screen_manager: &mut ScreenManager) -> Result<()>;
|
||||
fn to_main_screen(&self, screen_manager: &mut ScreenManager) -> Result<()>;
|
||||
}
|
||||
|
||||
pub trait IRawScreenCommand
|
||||
{
|
||||
fn enable(&mut self) -> Result<()>;
|
||||
fn disable(&mut self) -> Result<()>;
|
||||
}
|
||||
|
@ -1,11 +1,13 @@
|
||||
//! This module contains the commands that can be used for both unix and windows systems. Or else said terminals that support ansi codes.
|
||||
use super::IStateCommand;
|
||||
use super::{IStateCommand, IAlternateScreenCommand};
|
||||
use Context;
|
||||
use ScreenManager;
|
||||
|
||||
use std::rc::Rc;
|
||||
|
||||
use std::io::Result;
|
||||
pub struct EmptyCommand;
|
||||
|
||||
|
||||
impl IStateCommand for EmptyCommand {
|
||||
fn execute(&mut self) -> bool {
|
||||
return false;
|
||||
@ -17,39 +19,22 @@ impl IStateCommand for EmptyCommand {
|
||||
}
|
||||
|
||||
/// This command is used for switching to alternate screen and back to main screen.
|
||||
pub struct ToAlternateScreenBufferCommand {
|
||||
context: Rc<Context>,
|
||||
}
|
||||
pub struct ToAlternateScreenBufferCommand;
|
||||
|
||||
impl ToAlternateScreenBufferCommand {
|
||||
pub fn new(context: Rc<Context>) -> u16 {
|
||||
let mut state = context.state_manager.lock().unwrap();
|
||||
{
|
||||
let key = state.get_changes_count();
|
||||
let command = ToAlternateScreenBufferCommand {
|
||||
context: context.clone(),
|
||||
};
|
||||
|
||||
state.register_change(Box::from(command), key);
|
||||
key
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl IStateCommand for ToAlternateScreenBufferCommand {
|
||||
fn execute(&mut self) -> bool {
|
||||
let mut screen = self.context.screen_manager.lock().unwrap();
|
||||
{
|
||||
screen.write_str(csi!("?1049h"));
|
||||
return true;
|
||||
pub fn new() -> Box<ToAlternateScreenBufferCommand> {
|
||||
return Box::new(ToAlternateScreenBufferCommand {});
|
||||
}
|
||||
}
|
||||
|
||||
fn undo(&mut self) -> bool {
|
||||
let mut screen = self.context.screen_manager.lock().unwrap();
|
||||
{
|
||||
screen.write_str(csi!("?1049l"));
|
||||
return true;
|
||||
}
|
||||
impl IAlternateScreenCommand for ToAlternateScreenBufferCommand {
|
||||
fn to_alternate_screen(&self, screen_manager: &mut ScreenManager) -> Result<()> {
|
||||
screen_manager.write_str(csi!("?1049h"));
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn to_main_screen(&self, screen_manager: &mut ScreenManager) -> Result<()> {
|
||||
screen_manager.write_str(csi!("?1049l"));
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
@ -7,6 +7,7 @@ use {CommandManager, Context, StateManager};
|
||||
|
||||
const FD_STDIN: ::std::os::unix::io::RawFd = 1;
|
||||
|
||||
use std::io::Result;
|
||||
use std::rc::Rc;
|
||||
use std::sync::Mutex;
|
||||
|
||||
@ -66,32 +67,18 @@ impl IStateCommand for NoncanonicalModeCommand {
|
||||
|
||||
/// This command is used for enabling and disabling raw mode for the terminal.
|
||||
pub struct EnableRawModeCommand {
|
||||
original_mode: Option<Box<Termios>>,
|
||||
command_id: u16,
|
||||
original_mode: Result<Termios>,
|
||||
}
|
||||
|
||||
impl EnableRawModeCommand {
|
||||
pub fn new(state_manager: &Mutex<StateManager>) -> u16 {
|
||||
let mut state = state_manager.lock().unwrap();
|
||||
{
|
||||
let key = state.get_changes_count();
|
||||
let command = EnableRawModeCommand {
|
||||
original_mode: None,
|
||||
command_id: key,
|
||||
};
|
||||
|
||||
state.register_change(Box::from(command), key);
|
||||
key
|
||||
}
|
||||
pub fn new() -> EnableRawModeCommand {
|
||||
return EnableRawModeCommand { original_mode: terminal::get_terminal_mode(), }
|
||||
}
|
||||
}
|
||||
|
||||
impl IStateCommand for EnableRawModeCommand {
|
||||
impl EnableRawModeCommand {
|
||||
fn execute(&mut self) -> bool {
|
||||
let original_mode = terminal::get_terminal_mode();
|
||||
|
||||
if let Ok(original_mode) = original_mode {
|
||||
self.original_mode = Some(Box::from(original_mode));
|
||||
if let Ok(original_mode) = self.original_mode {
|
||||
let mut new_mode = original_mode;
|
||||
terminal::make_raw(&mut new_mode);
|
||||
terminal::set_terminal_mode(&new_mode);
|
||||
|
@ -1,6 +1,6 @@
|
||||
//! This module contains the commands that can be used for windows systems.
|
||||
|
||||
use super::IStateCommand;
|
||||
use super::{ IStateCommand, IAlternateScreenCommand, IRawScreenCommand};
|
||||
use {Context, StateManager};
|
||||
|
||||
use kernel::windows_kernel::{ansi_support, csbi, handle, kernel};
|
||||
@ -11,6 +11,7 @@ use winapi::um::wincon::{CHAR_INFO, COORD, ENABLE_VIRTUAL_TERMINAL_PROCESSING, S
|
||||
|
||||
use std::rc::Rc;
|
||||
use std::sync::Mutex;
|
||||
use std::io::{Result, ErrorKind, Error };
|
||||
|
||||
/// This command is used for enabling and disabling ANSI code support for windows systems,
|
||||
/// For more info check: https://docs.microsoft.com/en-us/windows/console/console-virtual-terminal-sequences.
|
||||
@ -74,116 +75,95 @@ impl IStateCommand for EnableAnsiCommand {
|
||||
#[derive(Clone, Copy)]
|
||||
pub struct EnableRawModeCommand {
|
||||
mask: DWORD,
|
||||
key: u16,
|
||||
}
|
||||
|
||||
impl EnableRawModeCommand {
|
||||
pub fn new(state_manager: &Mutex<StateManager>) -> u16 {
|
||||
pub fn new() -> EnableRawModeCommand {
|
||||
use self::wincon::{ENABLE_ECHO_INPUT, ENABLE_LINE_INPUT, ENABLE_PROCESSED_INPUT};
|
||||
|
||||
let mut state = state_manager.lock().unwrap();
|
||||
{
|
||||
let key = state.get_changes_count();
|
||||
let command = EnableRawModeCommand {
|
||||
EnableRawModeCommand {
|
||||
mask: ENABLE_LINE_INPUT | ENABLE_PROCESSED_INPUT | ENABLE_ECHO_INPUT,
|
||||
key: key,
|
||||
};
|
||||
state.register_change(Box::from(command), key);
|
||||
key
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl IStateCommand for EnableRawModeCommand {
|
||||
fn execute(&mut self) -> bool {
|
||||
let input_handle = handle::get_input_handle().unwrap();
|
||||
impl IRawScreenCommand for EnableRawModeCommand {
|
||||
fn enable(&mut self) -> Result<()> {
|
||||
let input_handle = handle::get_input_handle()?;
|
||||
|
||||
let mut dw_mode: DWORD = 0;
|
||||
if !kernel::get_console_mode(&input_handle, &mut dw_mode) {
|
||||
return false;
|
||||
return Err(Error::new(ErrorKind::Other,"Could not get console mode when enabling raw mode"))
|
||||
}
|
||||
|
||||
let new_mode = dw_mode & !self.mask;
|
||||
|
||||
if !kernel::set_console_mode(&input_handle, new_mode) {
|
||||
return false;
|
||||
return Err(Error::new(ErrorKind::Other,"Could not set console mode when enabling raw mode"))
|
||||
}
|
||||
|
||||
true
|
||||
return Ok(())
|
||||
}
|
||||
|
||||
fn undo(&mut self) -> bool {
|
||||
let output_handle = handle::get_output_handle().unwrap();
|
||||
fn disable(&mut self) -> Result<()> {
|
||||
let output_handle = handle::get_input_handle()?;
|
||||
|
||||
let mut dw_mode: DWORD = 0;
|
||||
if !kernel::get_console_mode(&output_handle, &mut dw_mode) {
|
||||
return false;
|
||||
return Err(Error::new(ErrorKind::Other,"Could not get console mode when disabling raw mode"))
|
||||
}
|
||||
|
||||
let new_mode = dw_mode | self.mask;
|
||||
|
||||
if !kernel::set_console_mode(&output_handle, new_mode) {
|
||||
return false;
|
||||
return Err(Error::new(ErrorKind::Other,"Could not set console mode when disabling raw mode"))
|
||||
}
|
||||
|
||||
true
|
||||
return Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
use ScreenManager;
|
||||
|
||||
/// This command is used for switching to alternate screen and back to main screen.
|
||||
/// check https://docs.microsoft.com/en-us/windows/console/reading-and-writing-blocks-of-characters-and-attributes for more info
|
||||
pub struct ToAlternateScreenBufferCommand {
|
||||
context: Rc<Context>,
|
||||
}
|
||||
pub struct ToAlternateScreenBufferCommand;
|
||||
|
||||
impl ToAlternateScreenBufferCommand {
|
||||
pub fn new(context: Rc<Context>) -> u16 {
|
||||
let mut state = context.state_manager.lock().unwrap();
|
||||
{
|
||||
let key = state.get_changes_count();
|
||||
let command = ToAlternateScreenBufferCommand {
|
||||
context: context.clone(),
|
||||
};
|
||||
|
||||
state.register_change(Box::from(command), key);
|
||||
key
|
||||
}
|
||||
pub fn new() -> Box<ToAlternateScreenBufferCommand>{
|
||||
return Box::from(ToAlternateScreenBufferCommand {});
|
||||
}
|
||||
}
|
||||
|
||||
impl IStateCommand for ToAlternateScreenBufferCommand {
|
||||
fn execute(&mut self) -> bool {
|
||||
impl IAlternateScreenCommand for ToAlternateScreenBufferCommand {
|
||||
fn to_alternate_screen(&self, screen_manager: &mut ScreenManager) -> Result<()>{
|
||||
use super::super::super::manager::WinApiScreenManager;
|
||||
|
||||
let handle = handle::get_output_handle().unwrap();
|
||||
let handle = handle::get_output_handle()?;
|
||||
|
||||
// create a new screen buffer to copy to.
|
||||
let new_handle = csbi::create_console_screen_buffer();
|
||||
|
||||
// Make the new screen buffer the active screen buffer.
|
||||
csbi::set_active_screen_buffer(new_handle);
|
||||
csbi::set_active_screen_buffer(new_handle)?;
|
||||
|
||||
{
|
||||
let mutex = &self.context.screen_manager;
|
||||
let mut screen = mutex.lock().unwrap();
|
||||
|
||||
let b: &mut WinApiScreenManager = match screen
|
||||
let b: &mut WinApiScreenManager = match screen_manager
|
||||
.as_any()
|
||||
.downcast_mut::<WinApiScreenManager>()
|
||||
{
|
||||
Some(b) => b,
|
||||
None => panic!(""),
|
||||
None => return Err(Error::new(ErrorKind::Other,"Invalid cast exception")),
|
||||
};
|
||||
|
||||
b.set_alternate_handle(new_handle);
|
||||
}
|
||||
true
|
||||
|
||||
Ok(())
|
||||
}
|
||||
|
||||
fn undo(&mut self) -> bool {
|
||||
let handle = handle::get_output_handle().unwrap();
|
||||
fn to_main_screen(&self, screen_manager: &mut ScreenManager) -> Result<()>{
|
||||
let handle = handle::get_output_handle()?;
|
||||
csbi::set_active_screen_buffer(handle);
|
||||
|
||||
true
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user