| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906 | {    This file is part of the Free Pascal run time library.    Copyright (c) 1999-2000 by the Free Pascal development team.    Borland Pascal 7 Compatible CRT Unit - win32 implentation    See the file COPYING.FPC, included in this distribution,    for details about the copyright.    This program is distributed in the hope that it will be useful,    but WITHOUT ANY WARRANTY; without even the implied warranty of    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. **********************************************************************}unit crt;interface{$i crth.inc}procedure Window32(X1,Y1,X2,Y2: DWord);procedure GotoXY32(X,Y: DWord);function WhereX32: DWord;function WhereY32: DWord;implementationuses  windows;var    SaveCursorSize: Longint;{  definition of textrec is in textrec.inc}{$i textrec.inc}{****************************************************************************                           Low level Routines****************************************************************************}procedure TurnMouseOff;var Mode: DWORD;begin  if GetConsoleMode(GetStdHandle(STD_INPUT_HANDLE), @Mode) then begin { Turn the mouse-cursor off }    Mode := Mode AND cardinal(NOT enable_processed_input)      AND cardinal(NOT enable_mouse_input);    SetConsoleMode(GetStdHandle(STD_INPUT_HANDLE), Mode);  end; { if }end; { proc. TurnMouseOff }function GetScreenHeight : DWord;var  ConsoleInfo: TConsoleScreenBufferinfo;begin  if (not GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), ConsoleInfo)) then begin{$ifdef SYSTEMDEBUG}    Writeln(stderr,'GetScreenHeight failed GetLastError returns ',GetLastError);    Halt(1);{$endif SYSTEMDEBUG}    // ts: this is really silly assumption; imho better: issue a halt    GetScreenHeight:=25;  end else    GetScreenHeight := ConsoleInfo.dwSize.Y;end; { func. GetScreenHeight }function GetScreenWidth : DWord;var  ConsoleInfo: TConsoleScreenBufferInfo;begin  if (not GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), ConsoleInfo)) then begin{$ifdef SYSTEMDEBUG}    Writeln(stderr,'GetScreenWidth failed GetLastError returns ',GetLastError);    Halt(1);{$endif SYSTEMDEBUG}    // ts: this is really silly assumption; imho better: issue a halt    GetScreenWidth:=80;  end else    GetScreenWidth := ConsoleInfo.dwSize.X;end; { func. GetScreenWidth }procedure GetScreenCursor(var x : DWord; var y : DWord);var  ConsoleInfo : TConsoleScreenBufferInfo;begin  FillChar(ConsoleInfo, SizeOf(ConsoleInfo), 0);  GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), ConsoleInfo);  X := ConsoleInfo.dwCursorPosition.X + 1;  Y := ConsoleInfo.dwCursorPosition.Y + 1;end;procedure SetScreenCursor(x,y : DWord);var  CurInfo: TCoord;begin  FillChar(Curinfo, SizeOf(Curinfo), 0);  CurInfo.X := X - 1;  CurInfo.Y := Y - 1;  SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE), CurInfo);end;{****************************************************************************                             Public Crt Functions****************************************************************************}procedure TextMode (Mode: word);beginend;Procedure TextColor(Color: Byte);{ Switch foregroundcolor }Begin  TextAttr:=(Color and $8f) or (TextAttr and $70);End;Procedure TextBackground(Color: Byte);{ Switch backgroundcolor }Begin  TextAttr:=((Color shl 4) and ($f0 and not Blink)) or (TextAttr and ($0f OR Blink) );End;Procedure HighVideo;{ Set highlighted output. }Begin  TextColor(TextAttr Or $08);End;Procedure LowVideo;{ Set normal output }Begin  TextColor(TextAttr And $77);End;Procedure NormVideo;{ Set normal back and foregroundcolors. }Begin  TextColor(7);  TextBackGround(0);End;Procedure GotoXY(X: tcrtcoord; Y: tcrtcoord);begin  GotoXY32(X,Y);end;Procedure GotoXY32(X: DWord; Y: DWord);{ Go to coordinates X,Y in the current window. }Begin  If (X > 0) and (X <= (WindMaxX - WindMinX + 1)) and    (Y > 0) and (Y <= (WindMaxY - WindMinY + 1)) Then Begin    Inc(X, WindMinX - 1);    Inc(Y, WindMinY - 1);    SetScreenCursor(x,y);  End;End;Procedure Window(X1, Y1, X2, Y2: Byte);begin  Window32(X1,Y1,X2,Y2);end;Procedure Window32(X1, Y1, X2, Y2: DWord);{  Set screen window to the specified coordinates.}Begin  if (X1 > X2) or (X2 > GetScreenWidth) or    (Y1 > Y2) or (Y2 > GetScreenHeight) then    exit;  WindMinY := Y1;  WindMaxY := Y2;  WindMinX := X1;  WindMaxX := X2;  WindMin:=((Y1-1) Shl 8)+(X1-1);  WindMax:=((Y2-1) Shl 8)+(X2-1);  GotoXY(1, 1);End;procedure ClrScr;var  DestCoor: TCoord;  numChars, x : DWord;begin  DestCoor.X := WindMinX - 1;  DestCoor.Y := WindMinY - 1;  numChars := (WindMaxX - WindMinX + 1);  repeat    FillConsoleOutputAttribute(GetStdHandle(STD_OUTPUT_HANDLE), TextAttr,      numChars, DestCoor, x);    FillConsoleOutputCharacter(GetStdHandle(STD_OUTPUT_HANDLE), #32,      numChars, DestCoor, x);    inc(DestCoor.Y);  until DWord(DestCoor.Y)=WindMaxY;  GotoXY(1, 1);end; { proc. ClrScr }procedure ClrEol;{  Clear from current position to end of line.}var  Temp: DWord;  CharInfo: Char;  Coord: TCoord;  X,Y: DWord;begin  GetScreenCursor(x, y);  CharInfo := #32;  Coord.X := X - 1;  Coord.Y := Y - 1;  FillConsoleOutputCharacter(GetStdHandle(STD_OUTPUT_HANDLE), CharInfo, WindMaxX - X + 1,    Coord, @Temp);  FillConsoleOutputAttribute(GetStdHandle(STD_OUTPUT_HANDLE), TextAttr, WindMaxX - X + 1,    Coord, @Temp);end;Function WhereX: tcrtcoord;begin  WhereX:=WhereX32 mod 256;end;Function WhereX32: DWord;{  Return current X-position of cursor.}var  x,y : DWord;Begin  GetScreenCursor(x, y);  WhereX32:= x - WindMinX +1;End;Function WhereY: tcrtcoord;begin  WhereY:=WhereY32 mod 256;end;Function WhereY32: DWord;{  Return current Y-position of cursor.}var  x, y : DWord;Begin  GetScreenCursor(x, y);  WhereY32:= y - WindMinY + 1;End;{*************************************************************************                            KeyBoard*************************************************************************}var   ScanCode : char;   SpecialKey : boolean;   DoingNumChars: Boolean;   DoingNumCode: Byte;Function RemapScanCode (ScanCode: byte; CtrlKeyState: byte; keycode:longint): byte;  { Several remappings of scancodes are necessary to comply with what    we get with MSDOS. Special Windows keys, as Alt-Tab, Ctrl-Esc etc.    are excluded }var  AltKey, CtrlKey, ShiftKey: boolean;const  {    Keypad key scancodes:      Ctrl Norm      $77  $47 - Home      $8D  $48 - Up arrow      $84  $49 - PgUp      $8E  $4A - -      $73  $4B - Left Arrow      $8F  $4C - 5      $74  $4D - Right arrow      $4E  $4E - +      $75  $4F - End      $91  $50 - Down arrow      $76  $51 - PgDn      $92  $52 - Ins      $93  $53 - Del  }  CtrlKeypadKeys: array[$47..$53] of byte =    ($77, $8D, $84, $8E, $73, $8F, $74, $4E, $75, $91, $76, $92, $93);begin  AltKey := ((CtrlKeyState AND            (RIGHT_ALT_PRESSED OR LEFT_ALT_PRESSED)) > 0);  CtrlKey := ((CtrlKeyState AND            (RIGHT_CTRL_PRESSED OR LEFT_CTRL_PRESSED)) > 0);  ShiftKey := ((CtrlKeyState AND SHIFT_PRESSED) > 0);  if AltKey then   begin    case ScanCode of    // Digits, -, =    $02..$0D: inc(ScanCode, $76);    // Function keys    $3B..$44: inc(Scancode, $2D);    $57..$58: inc(Scancode, $34);    // Extended cursor block keys    $47..$49, $4B, $4D, $4F..$53:              inc(Scancode, $50);    // Other keys    $1C:      Scancode := $A6;   // Enter    $35:      Scancode := $A4;   // / (keypad and normal!)    end   end  else if CtrlKey then    case Scancode of    // Tab key    $0F:      Scancode := $94;    // Function keys    $3B..$44: inc(Scancode, $23);    $57..$58: inc(Scancode, $32);    // Keypad keys    $35:      Scancode := $95;   // \    $37:      Scancode := $96;   // *    $47..$53: Scancode := CtrlKeypadKeys[Scancode];    //Enter on Numpad    $1C:    begin      Scancode := $0A;      SpecialKey := False;    end;    end  else if ShiftKey then    case Scancode of    // Function keys    $3B..$44: inc(Scancode, $19);    $57..$58: inc(Scancode, $30);    //Enter on Numpad    $1C:    begin      Scancode := $0D;      SpecialKey := False;    end;    end  else    case Scancode of      // Function keys      $57..$58: inc(Scancode, $2E); // F11 and F12      //Enter on NumPad      $1C:        begin          Scancode := $0D;          SpecialKey := False;        end;  end;  RemapScanCode := ScanCode;end;function KeyPressed : boolean;var  nevents,nread : dword;  buf : TINPUTRECORD;  AltKey: Boolean;  c : longint;begin  KeyPressed := FALSE;  if ScanCode <> #0 then    KeyPressed := TRUE  else   begin     GetNumberOfConsoleInputEvents(TextRec(input).Handle,nevents);     while nevents>0 do       begin          ReadConsoleInputA(TextRec(input).Handle,buf,1,nread);          if buf.EventType = KEY_EVENT then            if buf.Event.KeyEvent.bKeyDown then              begin                 { Alt key is VK_MENU }                 { Capslock key is VK_CAPITAL }                 AltKey := ((Buf.Event.KeyEvent.dwControlKeyState AND                            (RIGHT_ALT_PRESSED OR LEFT_ALT_PRESSED)) > 0);                 if not(Buf.Event.KeyEvent.wVirtualKeyCode in [VK_SHIFT, VK_MENU, VK_CONTROL,                                                      VK_CAPITAL, VK_NUMLOCK,                                                      VK_SCROLL]) then                   begin                      keypressed:=true;                      if (ord(buf.Event.KeyEvent.AsciiChar) = 0) or                         (buf.Event.KeyEvent.dwControlKeyState and (LEFT_ALT_PRESSED or ENHANCED_KEY) > 0) then                        begin                           SpecialKey := TRUE;                           ScanCode := Chr(RemapScanCode(Buf.Event.KeyEvent.wVirtualScanCode, Buf.Event.KeyEvent.dwControlKeyState,                                           Buf.Event.KeyEvent.wVirtualKeyCode));                        end                      else                        begin                           { Map shift-tab }                           if (buf.Event.KeyEvent.AsciiChar=#9) and                              (buf.Event.KeyEvent.dwControlKeyState and SHIFT_PRESSED > 0) then                            begin                              SpecialKey := TRUE;                              ScanCode := #15;                            end                           else                            begin                              SpecialKey := FALSE;                              ScanCode := Chr(Ord(buf.Event.KeyEvent.AsciiChar));                            end;                        end;                      if AltKey then                        begin                           case Buf.Event.KeyEvent.wVirtualScanCode of                             71 : c:=7;                             72 : c:=8;                             73 : c:=9;                             75 : c:=4;                             76 : c:=5;                             77 : c:=6;                             79 : c:=1;                             80 : c:=2;                             81 : c:=3;                             82 : c:=0;                           else                             break;                           end;                           DoingNumChars := true;                           DoingNumCode := Byte((DoingNumCode * 10) + c);                           Keypressed := false;                           Specialkey := false;                           ScanCode := #0;                        end                      else                        break;                   end;              end             else              begin                if (Buf.Event.KeyEvent.wVirtualKeyCode in [VK_MENU]) then               if DoingNumChars then                 if DoingNumCode > 0 then                   begin                      ScanCode := Chr(DoingNumCode);                      Keypressed := true;                      DoingNumChars := false;                      DoingNumCode := 0;                      break                   end; { if }              end;          { if we got a key then we can exit }          if keypressed then            exit;          GetNumberOfConsoleInputEvents(TextRec(input).Handle,nevents);       end;   end;end;function ReadKey: char;begin  while (not KeyPressed) do    Sleep(1);  if SpecialKey then begin    ReadKey := #0;    SpecialKey := FALSE;  end else begin    ReadKey := ScanCode;    ScanCode := #0;  end;end;{*************************************************************************                                   Delay*************************************************************************}procedure Delay(MS: Word);begin  Sleep(ms);end; { proc. Delay }procedure sound(hz : word);begin  MessageBeep(0); { lame ;-) }end;procedure nosound;beginend;{****************************************************************************                          HighLevel Crt Functions****************************************************************************}procedure removeline(y : DWord);var  ClipRect: TSmallRect;  SrcRect: TSmallRect;  DestCoor: TCoord;  CharInfo: TCharInfo;begin  CharInfo.UnicodeChar := #32;  CharInfo.Attributes := TextAttr;  Y := (WindMinY - 1) + (Y - 1) + 1;  SrcRect.Top := Y;  SrcRect.Left := WindMinX - 1;  SrcRect.Right := WindMaxX - 1;  SrcRect.Bottom := WindMaxY - 1;  DestCoor.X := WindMinX - 1;  DestCoor.Y := Y - 1;  ClipRect := SrcRect;  cliprect.top := destcoor.y;  ScrollConsoleScreenBuffer(GetStdHandle(STD_OUTPUT_HANDLE), SrcRect, ClipRect,    DestCoor, CharInfo);end; { proc. RemoveLine }procedure delline;begin  removeline(wherey);end; { proc. DelLine }procedure insline;var  ClipRect: TSmallRect;  SrcRect: TSmallRect;  DestCoor: TCoord;  CharInfo: TCharInfo;  X,Y: DWord;begin  GetScreenCursor(X, Y);  CharInfo.UnicodeChar := #32;  CharInfo.Attributes := TextAttr;  SrcRect.Top := Y - 1;  SrcRect.Left := WindMinX - 1;  SrcRect.Right := WindMaxX - 1;  SrcRect.Bottom := WindMaxY - 1 + 1;  DestCoor.X := WindMinX - 1;  DestCoor.Y := Y;  ClipRect := SrcRect;  ClipRect.Bottom := WindMaxY - 1;  ScrollConsoleScreenBuffer(GetStdHandle(STD_OUTPUT_HANDLE), SrcRect, ClipRect,    DestCoor, CharInfo);end; { proc. InsLine }{****************************************************************************                             Extra Crt Functions****************************************************************************}procedure cursoron;var CursorInfo: TConsoleCursorInfo;begin  GetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), CursorInfo);  CursorInfo.dwSize := SaveCursorSize;  CursorInfo.bVisible := true;  SetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), CursorInfo);end;procedure cursoroff;var CursorInfo: TConsoleCursorInfo;begin  GetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), CursorInfo);  CursorInfo.bVisible := false;  SetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), CursorInfo);end;procedure cursorbig;var CursorInfo: TConsoleCursorInfo;begin  GetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), CursorInfo);  CursorInfo.dwSize := 93;  CursorInfo.bVisible := true;  SetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), CursorInfo);end;{*****************************************************************************                          Read and Write routines*****************************************************************************}var  CurrX, CurrY : DWord;procedure WriteChar(c : char);var    WritePos: Coord;                       { Upper-left cell to write from }    numWritten : DWord;    WinAttr : word;begin  Case C of    #10 : begin      Inc(CurrY);    end;    #13 : begin      CurrX := WindMinX;    end; { if }    #08 : begin      if CurrX > WindMinX then Dec(CurrX);    end; { ^H }    #07 : begin      //MessagBeep(0);    end; { ^G }    else begin      WritePos.X := currX - 1;      WritePos.Y := currY - 1;      WriteConsoleOutputCharacter(GetStdhandle(STD_OUTPUT_HANDLE),        @c, 1, writePos, numWritten);      WinAttr:=TextAttr;      WriteConsoleOutputAttribute(GetStdhandle(STD_OUTPUT_HANDLE),        @WinAttr, 1, writePos, numWritten);      Inc(CurrX);    end; { else }  end; { case }  if CurrX > WindMaxX then begin    CurrX := WindMinX;    Inc(CurrY);  end; { if }  While CurrY > WindMaxY do begin    RemoveLine(1);    Dec(CurrY);  end; { while }end;procedure WriteStr(const s: string);var  WritePos: Coord; { Upper-left cell to write from }  numWritten : DWord;  WinAttr : word;  i: integer;begin  WritePos.X:=currX-2;  WritePos.Y:=currY-1;  WinAttr:=TextAttr;  for i:=1 to Length(s) do    begin      Inc(WritePos.X);      WriteConsoleOutputCharacter(GetStdhandle(STD_OUTPUT_HANDLE), @s[i], 1, writePos, numWritten);      WriteConsoleOutputAttribute(GetStdhandle(STD_OUTPUT_HANDLE),@WinAttr, 1, writePos, numWritten);      Inc(CurrX);      if CurrX>WindMaxX then        begin          CurrX:=WindMinX;          Inc(CurrY);          While CurrY>WindMaxY do            begin              RemoveLine(1);              Dec(CurrY);            end;          WritePos.X:=currX-2;          WritePos.Y:=currY-1;        end;    end;end;Function CrtWrite(var f : textrec) : integer;var  i : longint;  s : string;  OldConsoleOutputCP : Word;begin  OldConsoleOutputCP:=GetConsoleOutputCP;  SetConsoleOutputCP(GetACP);  GetScreenCursor(CurrX, CurrY);  s:='';  for i:=0 to f.bufpos-1 do    if f.buffer[i] in [#7,#8,#10,#13] then // special chars directly.      begin        if s<>'' then          begin            WriteStr(s); 	    s:='';          end;        WriteChar(f.buffer[i]);      end    else      s:=s+f.buffer[i];  if s<>'' then    WriteStr(s);  SetScreenCursor(CurrX, CurrY);  SetConsoleOutputCP(OldConsoleOutputCP);  f.bufpos:=0;  CrtWrite:=0;end;Function CrtRead(Var F: TextRec): Integer;  procedure BackSpace;  begin    if (f.bufpos>0) and (f.bufpos=f.bufend) then begin      WriteChar(#8);      WriteChar(' ');      WriteChar(#8);      dec(f.bufpos);      dec(f.bufend);    end;  end;var  ch : Char;  OldConsoleOutputCP : Word;Begin  OldConsoleOutputCP:=GetConsoleOutputCP;  SetConsoleOutputCP(GetACP);  GetScreenCursor(CurrX,CurrY);  f.bufpos:=0;  f.bufend:=0;  repeat    if f.bufpos>f.bufend then      f.bufend:=f.bufpos;      SetScreenCursor(CurrX,CurrY);      ch:=readkey;      case ch of        #0 : case readkey of          #71 : while f.bufpos>0 do begin            dec(f.bufpos);            WriteChar(#8);          end;          #75 : if f.bufpos>0 then begin            dec(f.bufpos);            WriteChar(#8);          end;          #77 : if f.bufpos<f.bufend then begin            WriteChar(f.bufptr^[f.bufpos]);            inc(f.bufpos);          end;          #79 : while f.bufpos<f.bufend do begin            WriteChar(f.bufptr^[f.bufpos]);            inc(f.bufpos);          end;          #28: begin                    // numpad enter                WriteChar(#13);                WriteChar(#10);                f.bufptr^[f.bufend]:=#13;                f.bufptr^[f.bufend+1]:=#10;                inc(f.bufend,2);                break;               end;          #53: begin                 ch:='/';                 if f.bufpos<f.bufsize-2 then begin                    f.buffer[f.bufpos]:=ch;                    inc(f.bufpos);                    WriteChar(ch);                 end;               end;        end;        ^S,      #8 : BackSpace;      ^Y,      #27 : begin        while f.bufpos<f.bufend do begin         WriteChar(f.bufptr^[f.bufpos]);         inc(f.bufpos);        end;        while f.bufend>0 do          BackSpace;      end;      #13 : begin        WriteChar(#13);        WriteChar(#10);        f.bufptr^[f.bufend]:=#13;        f.bufptr^[f.bufend+1]:=#10;        inc(f.bufend,2);        break;      end;      #26 : if CheckEOF then begin        f.bufptr^[f.bufend]:=#26;        inc(f.bufend);        break;      end;      else begin        if f.bufpos<f.bufsize-2 then begin          f.bufptr^[f.bufpos]:=ch;          inc(f.bufpos);          WriteChar(ch);        end;      end;      end;  until false;  SetConsoleOutputCP(OldConsoleOutputCP);	  f.bufpos:=0;  SetScreenCursor(CurrX, CurrY);  CrtRead:=0;End;Function CrtReturn(Var F:TextRec):Integer;Begin  CrtReturn:=0;end;Function CrtClose(Var F: TextRec): Integer;Begin  F.Mode:=fmClosed;  CrtClose:=0;End;Function CrtOpen(Var F: TextRec): Integer;Begin  If F.Mode=fmOutput Then begin    TextRec(F).InOutFunc:=@CrtWrite;    TextRec(F).FlushFunc:=@CrtWrite;  end Else begin    F.Mode:=fmInput;    TextRec(F).InOutFunc:=@CrtRead;    TextRec(F).FlushFunc:=@CrtReturn;  end;  TextRec(F).CloseFunc:=@CrtClose;  CrtOpen:=0;End;procedure AssignCrt(var F: Text);begin  Assign(F,'');  TextRec(F).OpenFunc:=@CrtOpen;end;var  CursorInfo  : TConsoleCursorInfo;  ConsoleInfo : TConsoleScreenBufferinfo;// tsbegin  { Initialize the output handles }  LastMode := 3;  SetActiveWindow(0);  {--------------------- Get the cursor size and such -----------------------}  FillChar(CursorInfo, SizeOf(CursorInfo), 00);  GetConsoleCursorInfo(GetStdHandle(STD_OUTPUT_HANDLE), CursorInfo);  SaveCursorSize := CursorInfo.dwSize;  {------------------ Get the current cursor position and attr --------------}  FillChar(ConsoleInfo, SizeOf(ConsoleInfo), 0);  GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), ConsoleInfo);  TextAttr := ConsoleInfo.wAttributes;  { Not required, the dos crt does also not touch the mouse }  {TurnMouseOff;}  WindMinX := (ConsoleInfo.srWindow.Left) + 1;  WindMinY := (ConsoleInfo.srWindow.Top) + 1;  WindMaxX := (ConsoleInfo.srWindow.Right) + 1;  WindMaxY := (ConsoleInfo.srWindow.Bottom) + 1;  WindMax:=((WindMaxY-1) Shl 8)+(WindMaxX-1);  DoingNumChars := false;  DoingNumCode := 0;  { Redirect the standard output }  AssignCrt(Output);  Rewrite(Output);  TextRec(Output).Handle:= GetStdHandle(STD_OUTPUT_HANDLE);  AssignCrt(Input);  Reset(Input);  TextRec(Input).Handle:= GetStdHandle(STD_INPUT_HANDLE);end. { unit Crt }
 |