linux-packaging-mono/mcs/class/corlib/System/WindowsConsoleDriver.cs

631 lines
20 KiB
C#
Raw Normal View History

//
// System.WindowsConsoleDriver
//
// Author:
// Gonzalo Paniagua Javier (gonzalo@ximian.com)
//
// (C) 2005 Novell, Inc. (http://www.novell.com)
//
// Permission is hereby granted, free of charge, to any person obtaining
// a copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to
// permit persons to whom the Software is furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
//
#if MONO_FEATURE_CONSOLE
using System.Runtime.InteropServices;
using System.Text;
namespace System {
struct ConsoleCursorInfo {
public int Size;
public bool Visible;
}
#pragma warning disable 169
struct InputRecord {
public short EventType;
// This is KEY_EVENT_RECORD
public bool KeyDown;
public short RepeatCount;
public short VirtualKeyCode;
public short VirtualScanCode;
public char Character;
public int ControlKeyState;
int pad1;
bool pad2;
//
}
#pragma warning restore 169
struct CharInfo {
public char Character;
public short Attributes;
}
struct Coord {
public short X;
public short Y;
public Coord (int x, int y)
{
X = (short) x;
Y = (short) y;
}
}
struct SmallRect {
public short Left;
public short Top;
public short Right;
public short Bottom;
public SmallRect (int left, int top, int right, int bottom)
{
Left = (short) left;
Top = (short) top;
Right = (short) right;
Bottom = (short) bottom;
}
}
struct ConsoleScreenBufferInfo {
public Coord Size;
public Coord CursorPosition;
public short Attribute;
public SmallRect Window;
public Coord MaxWindowSize;
}
enum Handles {
STD_INPUT = -10,
STD_OUTPUT = -11,
STD_ERROR = -12
}
unsafe class WindowsConsoleDriver : IConsoleDriver {
IntPtr inputHandle;
IntPtr outputHandle;
short defaultAttribute;
public WindowsConsoleDriver ()
{
outputHandle = GetStdHandle (Handles.STD_OUTPUT);
inputHandle = GetStdHandle (Handles.STD_INPUT);
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
defaultAttribute = info.Attribute; // Not sure about this...
}
// FOREGROUND_BLUE 1
// FOREGROUND_GREEN 2
// FOREGROUND_RED 4
// FOREGROUND_INTENSITY 8
// BACKGROUND_BLUE 16
// BACKGROUND_GREEN 32
// BACKGROUND_RED 64
// BACKGROUND_INTENSITY 128
static ConsoleColor GetForeground (short attr)
{
attr &= 0x0F;
return (ConsoleColor) attr;
}
static ConsoleColor GetBackground (short attr)
{
attr &= 0xF0;
attr >>= 4;
return (ConsoleColor) attr;
}
static short GetAttrForeground (int attr, ConsoleColor color)
{
attr &= ~15;
return (short) (attr | (int) color);
}
static short GetAttrBackground (int attr, ConsoleColor color)
{
attr &= ~0xf0;
int c = ((int) color) << 4;
return (short) (attr | c);
}
public ConsoleColor BackgroundColor {
get {
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
return GetBackground (info.Attribute);
}
set {
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
short attr = GetAttrBackground (info.Attribute, value);
SetConsoleTextAttribute (outputHandle, attr);
}
}
public int BufferHeight {
get {
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
return info.Size.Y;
}
set { SetBufferSize (BufferWidth, value); }
}
public int BufferWidth {
get {
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
return info.Size.X;
}
set { SetBufferSize (value, BufferHeight); }
}
public bool CapsLock {
get {
short state = GetKeyState (20); // VK_CAPITAL
return ((state & 1) == 1);
}
}
public int CursorLeft {
get {
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
return info.CursorPosition.X;
}
set { SetCursorPosition (value, CursorTop); }
}
public int CursorSize {
get {
ConsoleCursorInfo info = new ConsoleCursorInfo ();
GetConsoleCursorInfo (outputHandle, out info);
return info.Size;
}
set {
if (value < 1 || value > 100)
throw new ArgumentOutOfRangeException ("value");
ConsoleCursorInfo info = new ConsoleCursorInfo ();
GetConsoleCursorInfo (outputHandle, out info);
info.Size = value;
if (!SetConsoleCursorInfo (outputHandle, ref info))
throw new Exception ("SetConsoleCursorInfo failed");
}
}
public int CursorTop {
get {
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
return info.CursorPosition.Y;
}
set { SetCursorPosition (CursorLeft, value); }
}
public bool CursorVisible {
get {
ConsoleCursorInfo info = new ConsoleCursorInfo ();
GetConsoleCursorInfo (outputHandle, out info);
return info.Visible;
}
set {
ConsoleCursorInfo info = new ConsoleCursorInfo ();
GetConsoleCursorInfo (outputHandle, out info);
if (info.Visible == value)
return;
info.Visible = value;
if (!SetConsoleCursorInfo (outputHandle, ref info))
throw new Exception ("SetConsoleCursorInfo failed");
}
}
public ConsoleColor ForegroundColor {
get {
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
return GetForeground (info.Attribute);
}
set {
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
short attr = GetAttrForeground (info.Attribute, value);
SetConsoleTextAttribute (outputHandle, attr);
}
}
public bool KeyAvailable {
get {
int eventsRead;
InputRecord record = new InputRecord ();
while (true) {
// Use GetNumberOfConsoleInputEvents and remove the while?
if (!PeekConsoleInput (inputHandle, out record, 1, out eventsRead))
throw new InvalidOperationException ("Error in PeekConsoleInput " +
Marshal.GetLastWin32Error ());
if (eventsRead == 0)
return false;
//KEY_EVENT == 1
if (record.EventType == 1 && record.KeyDown && !IsModifierKey (record.VirtualKeyCode))
return true;
if (!ReadConsoleInput (inputHandle, out record, 1, out eventsRead))
throw new InvalidOperationException ("Error in ReadConsoleInput " +
Marshal.GetLastWin32Error ());
}
}
}
public bool Initialized { // Not useful on windows, so false.
get { return false; }
}
public int LargestWindowHeight {
get {
Coord coord = GetLargestConsoleWindowSize (outputHandle);
if (coord.X == 0 && coord.Y == 0)
throw new Exception ("GetLargestConsoleWindowSize" + Marshal.GetLastWin32Error ());
return coord.Y;
}
}
public int LargestWindowWidth {
get {
Coord coord = GetLargestConsoleWindowSize (outputHandle);
if (coord.X == 0 && coord.Y == 0)
throw new Exception ("GetLargestConsoleWindowSize" + Marshal.GetLastWin32Error ());
return coord.X;
}
}
public bool NumberLock {
get {
short state = GetKeyState (144); // VK_NUMLOCK
return ((state & 1) == 1);
}
}
public string Title {
get {
StringBuilder sb = new StringBuilder (1024); // hope this is enough
if (GetConsoleTitle (sb, 1024) == 0) {
// Try the maximum
sb = new StringBuilder (26001);
if (GetConsoleTitle (sb, 26000) == 0)
throw new Exception ("Got " + Marshal.GetLastWin32Error ());
}
return sb.ToString ();
}
set {
if (value == null)
throw new ArgumentNullException ("value");
if (!SetConsoleTitle (value))
throw new Exception ("Got " + Marshal.GetLastWin32Error ());
}
}
public bool TreatControlCAsInput {
get {
int mode;
if (!GetConsoleMode (inputHandle, out mode))
throw new Exception ("Failed in GetConsoleMode: " + Marshal.GetLastWin32Error ());
// ENABLE_PROCESSED_INPUT
return ((mode & 1) == 0);
}
set {
int mode;
if (!GetConsoleMode (inputHandle, out mode))
throw new Exception ("Failed in GetConsoleMode: " + Marshal.GetLastWin32Error ());
bool cAsInput = ((mode & 1) == 0);
if (cAsInput == value)
return;
if (value)
mode &= ~1;
else
mode |= 1;
if (!SetConsoleMode (inputHandle, mode))
throw new Exception ("Failed in SetConsoleMode: " + Marshal.GetLastWin32Error ());
}
}
public int WindowHeight {
get {
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
return info.Window.Bottom - info.Window.Top + 1;
}
set { SetWindowSize (WindowWidth, value); }
}
public int WindowLeft {
get {
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
return info.Window.Left;
}
set { SetWindowPosition (value, WindowTop); }
}
public int WindowTop {
get {
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
return info.Window.Top;
}
set { SetWindowPosition (WindowLeft, value); }
}
public int WindowWidth {
get {
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
return info.Window.Right - info.Window.Left + 1;
}
set { SetWindowSize (value, WindowHeight); }
}
public void Beep (int frequency, int duration)
{
_Beep (frequency, duration);
}
public void Clear ()
{
Coord coord = new Coord ();
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
int size = info.Size.X * info.Size.Y;
int written;
FillConsoleOutputCharacter (outputHandle, ' ', size, coord, out written);
GetConsoleScreenBufferInfo (outputHandle, out info);
FillConsoleOutputAttribute (outputHandle, info.Attribute, size, coord, out written);
SetConsoleCursorPosition (outputHandle, coord);
}
public void MoveBufferArea (int sourceLeft, int sourceTop, int sourceWidth, int sourceHeight,
int targetLeft, int targetTop, Char sourceChar,
ConsoleColor sourceForeColor, ConsoleColor sourceBackColor)
{
if (sourceForeColor < 0)
throw new ArgumentException ("Cannot be less than 0.", "sourceForeColor");
if (sourceBackColor < 0)
throw new ArgumentException ("Cannot be less than 0.", "sourceBackColor");
if (sourceWidth == 0 || sourceHeight == 0)
return;
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
CharInfo [] buffer = new CharInfo [sourceWidth * sourceHeight];
Coord bsize = new Coord (sourceWidth, sourceHeight);
Coord bpos = new Coord (0, 0);
SmallRect region = new SmallRect (sourceLeft, sourceTop, sourceLeft + sourceWidth - 1, sourceTop + sourceHeight - 1);
fixed (void *ptr = &buffer [0]) {
if (!ReadConsoleOutput (outputHandle, ptr, bsize, bpos, ref region))
throw new ArgumentException (String.Empty, "Cannot read from the specified coordinates.");
}
int written;
short attr = GetAttrForeground (0, sourceForeColor);
attr = GetAttrBackground (attr, sourceBackColor);
bpos = new Coord (sourceLeft, sourceTop);
for (int i = 0; i < sourceHeight; i++, bpos.Y++) {
FillConsoleOutputCharacter (outputHandle, sourceChar, sourceWidth, bpos, out written);
FillConsoleOutputAttribute (outputHandle, attr, sourceWidth, bpos, out written);
}
bpos = new Coord (0, 0);
region = new SmallRect (targetLeft, targetTop, targetLeft + sourceWidth - 1, targetTop + sourceHeight - 1);
if (!WriteConsoleOutput (outputHandle, buffer, bsize, bpos, ref region))
throw new ArgumentException (String.Empty, "Cannot write to the specified coordinates.");
}
public void Init ()
{
}
public string ReadLine ()
{
StringBuilder builder = new StringBuilder ();
bool exit = false;
do {
ConsoleKeyInfo key = ReadKey (false);
char c = key.KeyChar;
exit = (c == '\n');
if (!exit)
builder.Append (key.KeyChar);
} while (!exit);
return builder.ToString ();
}
public ConsoleKeyInfo ReadKey (bool intercept)
{
int eventsRead;
InputRecord record = new InputRecord ();
for (;;) {
if (!ReadConsoleInput (inputHandle, out record, 1, out eventsRead))
throw new InvalidOperationException ("Error in ReadConsoleInput " +
Marshal.GetLastWin32Error ());
if (record.KeyDown && record.EventType == 1 && !IsModifierKey (record.VirtualKeyCode))
break;
}
// RIGHT_ALT_PRESSED 1
// LEFT_ALT_PRESSED 2
// RIGHT_CTRL_PRESSED 4
// LEFT_CTRL_PRESSED 8
// SHIFT_PRESSED 16
bool alt = ((record.ControlKeyState & 3) != 0);
bool ctrl = ((record.ControlKeyState & 12) != 0);
bool shift = ((record.ControlKeyState & 16) != 0);
return new ConsoleKeyInfo (record.Character, (ConsoleKey) record.VirtualKeyCode, shift, alt, ctrl);
}
public void ResetColor ()
{
SetConsoleTextAttribute (outputHandle, defaultAttribute);
}
public void SetBufferSize (int width, int height)
{
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
if (width - 1 > info.Window.Right)
throw new ArgumentOutOfRangeException ("width");
if (height - 1 > info.Window.Bottom)
throw new ArgumentOutOfRangeException ("height");
Coord coord = new Coord (width, height);
if (!SetConsoleScreenBufferSize (outputHandle, coord))
throw new ArgumentOutOfRangeException ("height/width", "Cannot be smaller than the window size.");
}
public void SetCursorPosition (int left, int top)
{
Coord coord = new Coord (left, top);
SetConsoleCursorPosition (outputHandle, coord);
}
public void SetWindowPosition (int left, int top)
{
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
SmallRect rect = info.Window;
rect.Left = (short) left;
rect.Top = (short) top;
if (!SetConsoleWindowInfo (outputHandle, true, ref rect))
throw new ArgumentOutOfRangeException ("left/top", "Windows error " + Marshal.GetLastWin32Error ());
}
public void SetWindowSize (int width, int height)
{
ConsoleScreenBufferInfo info = new ConsoleScreenBufferInfo ();
GetConsoleScreenBufferInfo (outputHandle, out info);
SmallRect rect = info.Window;
rect.Right = (short) (rect.Left + width - 1);
rect.Bottom = (short) (rect.Top + height - 1);
if (!SetConsoleWindowInfo (outputHandle, true, ref rect))
throw new ArgumentOutOfRangeException ("left/top", "Windows error " + Marshal.GetLastWin32Error ());
}
static bool IsModifierKey (short virtualKeyCode)
{
// 0x10 through 0x14 is shift/control/alt/pause/capslock
// 0x2C is print screen, 0x90 is numlock, 0x91 is scroll lock
switch (virtualKeyCode) {
case 0x10:
case 0x11:
case 0x12:
case 0x14:
case 0x90:
case 0x91:
return true;
default:
return false;
}
}
//
// Imports
//
[DllImport ("kernel32.dll", EntryPoint="GetStdHandle", SetLastError=true, CharSet=CharSet.Unicode)]
extern static IntPtr GetStdHandle (Handles handle);
[DllImport ("kernel32.dll", EntryPoint="Beep", SetLastError=true, CharSet=CharSet.Unicode)]
extern static void _Beep (int frequency, int duration);
[DllImport ("kernel32.dll", EntryPoint="GetConsoleScreenBufferInfo", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool GetConsoleScreenBufferInfo (IntPtr handle, out ConsoleScreenBufferInfo info);
[DllImport ("kernel32.dll", EntryPoint="FillConsoleOutputCharacter", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool FillConsoleOutputCharacter (IntPtr handle, char c, int size, Coord coord, out int written);
[DllImport ("kernel32.dll", EntryPoint="FillConsoleOutputAttribute", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool FillConsoleOutputAttribute (IntPtr handle, short c, int size, Coord coord, out int written);
[DllImport ("kernel32.dll", EntryPoint="SetConsoleCursorPosition", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool SetConsoleCursorPosition (IntPtr handle, Coord coord);
[DllImport ("kernel32.dll", EntryPoint="SetConsoleTextAttribute", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool SetConsoleTextAttribute (IntPtr handle, short attribute);
[DllImport ("kernel32.dll", EntryPoint="SetConsoleScreenBufferSize", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool SetConsoleScreenBufferSize (IntPtr handle, Coord newSize);
[DllImport ("kernel32.dll", EntryPoint="SetConsoleWindowInfo", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool SetConsoleWindowInfo (IntPtr handle, bool absolute, ref SmallRect rect);
[DllImport ("kernel32.dll", EntryPoint="GetConsoleTitle", SetLastError=true, CharSet=CharSet.Unicode)]
extern static int GetConsoleTitle (StringBuilder sb, int size);
[DllImport ("kernel32.dll", EntryPoint="SetConsoleTitle", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool SetConsoleTitle (string title);
[DllImport ("kernel32.dll", EntryPoint="GetConsoleCursorInfo", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool GetConsoleCursorInfo (IntPtr handle, out ConsoleCursorInfo info);
[DllImport ("kernel32.dll", EntryPoint="SetConsoleCursorInfo", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool SetConsoleCursorInfo (IntPtr handle, ref ConsoleCursorInfo info);
[DllImport ("user32.dll", EntryPoint="GetKeyState", SetLastError=true, CharSet=CharSet.Unicode)]
extern static short GetKeyState (int virtKey);
[DllImport ("kernel32.dll", EntryPoint="GetConsoleMode", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool GetConsoleMode (IntPtr handle, out int mode);
[DllImport ("kernel32.dll", EntryPoint="SetConsoleMode", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool SetConsoleMode (IntPtr handle, int mode);
[DllImport ("kernel32.dll", EntryPoint="PeekConsoleInput", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool PeekConsoleInput (IntPtr handle, out InputRecord record, int length, out int eventsRead);
[DllImport ("kernel32.dll", EntryPoint="ReadConsoleInput", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool ReadConsoleInput (IntPtr handle, out InputRecord record, int length, out int nread);
[DllImport ("kernel32.dll", EntryPoint="GetLargestConsoleWindowSize", SetLastError=true, CharSet=CharSet.Unicode)]
extern static Coord GetLargestConsoleWindowSize (IntPtr handle);
[DllImport ("kernel32.dll", EntryPoint="ReadConsoleOutput", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool ReadConsoleOutput (IntPtr handle, void *buffer, Coord bsize, Coord bpos, ref SmallRect region);
[DllImport ("kernel32.dll", EntryPoint="WriteConsoleOutput", SetLastError=true, CharSet=CharSet.Unicode)]
extern static bool WriteConsoleOutput (IntPtr handle, CharInfo [] buffer, Coord bsize, Coord bpos, ref SmallRect region);
}
}
#endif