mirror of
https://gitlab.com/freepascal.org/fpc/source.git
synced 2025-04-19 21:39:24 +02:00
294 lines
7.5 KiB
ObjectPascal
294 lines
7.5 KiB
ObjectPascal
{
|
|
$Id$
|
|
This file is part of the Free Pascal run time library.
|
|
Copyright (c) 1999-2000 by Florian Klaempfl
|
|
member of the Free Pascal development team
|
|
|
|
Video unit for Win32
|
|
|
|
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 Video;
|
|
interface
|
|
|
|
{$i videoh.inc}
|
|
|
|
implementation
|
|
|
|
uses
|
|
windows,dos;
|
|
|
|
{$i video.inc}
|
|
|
|
var
|
|
ConsoleInfo : TConsoleScreenBufferInfo;
|
|
ConsoleCursorInfo : TConsoleCursorInfo;
|
|
|
|
procedure SysInitVideo;
|
|
|
|
begin
|
|
ScreenColor:=true;
|
|
GetConsoleScreenBufferInfo(TextRec(Output).Handle, ConsoleInfo);
|
|
GetConsoleCursorInfo(TextRec(Output).Handle, ConsoleCursorInfo);
|
|
{
|
|
About the ConsoleCursorInfo record: There are 3 possible
|
|
structures in it that can be regarded as the 'screen':
|
|
- dwsize : contains the cols & row in current screen buffer.
|
|
- srwindow : Coordinates (relative to buffer) of upper left
|
|
& lower right corners of visible console.
|
|
- dmMaximumWindowSize : Maximal size of Screen buffer.
|
|
The first implementation of video used srWindow. After some
|
|
bug-reports, this was switched to dwMaximumWindowSize.
|
|
}
|
|
with ConsoleInfo.dwMaximumWindowSize do
|
|
begin
|
|
ScreenWidth:=X;
|
|
ScreenHeight:=Y;
|
|
end;
|
|
{ TDrawBuffer only has FVMaxWidth elements
|
|
larger values lead to crashes }
|
|
if ScreenWidth> FVMaxWidth then
|
|
ScreenWidth:=FVMaxWidth;
|
|
CursorX:=ConsoleInfo.dwCursorPosition.x;
|
|
CursorY:=ConsoleInfo.dwCursorPosition.y;
|
|
if not ConsoleCursorInfo.bvisible then
|
|
CursorLines:=0
|
|
else
|
|
CursorLines:=ConsoleCursorInfo.dwSize;
|
|
end;
|
|
|
|
|
|
procedure SysDoneVideo;
|
|
begin
|
|
SetCursorType(crUnderLine);
|
|
end;
|
|
|
|
|
|
function SysGetCapabilities: Word;
|
|
begin
|
|
SysGetCapabilities:=cpColor or cpChangeCursor;
|
|
end;
|
|
|
|
|
|
procedure SysSetCursorPos(NewCursorX, NewCursorY: Word);
|
|
var
|
|
pos : COORD;
|
|
begin
|
|
pos.x:=NewCursorX;
|
|
pos.y:=NewCursorY;
|
|
SetConsoleCursorPosition(TextRec(Output).Handle,pos);
|
|
CursorX:=pos.x;
|
|
CursorY:=pos.y;
|
|
end;
|
|
|
|
|
|
function SysGetCursorType: Word;
|
|
begin
|
|
GetConsoleCursorInfo(TextRec(Output).Handle,ConsoleCursorInfo);
|
|
if not ConsoleCursorInfo.bvisible then
|
|
SysGetCursorType:=crHidden
|
|
else
|
|
case ConsoleCursorInfo.dwSize of
|
|
1..30:
|
|
SysGetCursorType:=crUnderline;
|
|
31..70:
|
|
SysGetCursorType:=crHalfBlock;
|
|
71..100:
|
|
SysGetCursorType:=crBlock;
|
|
end;
|
|
end;
|
|
|
|
|
|
procedure SysSetCursorType(NewType: Word);
|
|
begin
|
|
GetConsoleCursorInfo(TextRec(Output).Handle,ConsoleCursorInfo);
|
|
if newType=crHidden then
|
|
ConsoleCursorInfo.bvisible:=false
|
|
else
|
|
begin
|
|
ConsoleCursorInfo.bvisible:=true;
|
|
case NewType of
|
|
crUnderline:
|
|
ConsoleCursorInfo.dwSize:=10;
|
|
|
|
crHalfBlock:
|
|
ConsoleCursorInfo.dwSize:=50;
|
|
|
|
crBlock:
|
|
ConsoleCursorInfo.dwSize:=99;
|
|
end
|
|
end;
|
|
SetConsoleCursorInfo(TextRec(Output).Handle,ConsoleCursorInfo);
|
|
end;
|
|
|
|
procedure SysClearScreen;
|
|
begin
|
|
UpdateScreen(true);
|
|
end;
|
|
|
|
{$IFDEF FPC}
|
|
function WriteConsoleOutput(hConsoleOutput:HANDLE; lpBuffer:pointer; dwBufferSize:COORD; dwBufferCoord:COORD;
|
|
var lpWriteRegion:SMALL_RECT):WINBOOL; stdcall;external 'kernel32' name 'WriteConsoleOutputA';
|
|
{$ENDIF}
|
|
|
|
procedure SysUpdateScreen(Force: Boolean);
|
|
type TmpRec = Array[0..(1024*32) - 1] of TCharInfo;
|
|
|
|
type WordRec = record
|
|
One, Two: Byte;
|
|
end; { wordrec }
|
|
|
|
var
|
|
BufSize,
|
|
BufCoord : COORD;
|
|
WriteRegion : SMALL_RECT;
|
|
LineBuf : ^TmpRec;
|
|
BufCounter : Longint;
|
|
LineCounter,
|
|
ColCounter : Longint;
|
|
smallforce : boolean;
|
|
x1,y1,x2,y2 : longint;
|
|
begin
|
|
if force then
|
|
smallforce:=true
|
|
else
|
|
begin
|
|
asm
|
|
movl VideoBuf,%esi
|
|
movl OldVideoBuf,%edi
|
|
movl VideoBufSize,%ecx
|
|
shrl $2,%ecx
|
|
repe
|
|
cmpsl
|
|
setne smallforce
|
|
end;
|
|
end;
|
|
if SmallForce then
|
|
begin
|
|
BufSize.X := ScreenWidth;
|
|
BufSize.Y := ScreenHeight;
|
|
|
|
BufCoord.X := 0;
|
|
BufCoord.Y := 0;
|
|
with WriteRegion do
|
|
begin
|
|
Top :=0;
|
|
Left :=0;
|
|
Bottom := ScreenHeight-1;
|
|
Right := ScreenWidth-1;
|
|
end;
|
|
New(LineBuf);
|
|
BufCounter := 0;
|
|
x1:=ScreenWidth+1;
|
|
x2:=-1;
|
|
y1:=ScreenHeight+1;
|
|
y2:=-1;
|
|
for LineCounter := 1 to ScreenHeight do
|
|
begin
|
|
for ColCounter := 1 to ScreenWidth do
|
|
begin
|
|
if (WordRec(VideoBuf^[BufCounter]).One<>WordRec(OldVideoBuf^[BufCounter]).One) or
|
|
(WordRec(VideoBuf^[BufCounter]).Two<>WordRec(OldVideoBuf^[BufCounter]).Two) then
|
|
begin
|
|
if ColCounter<x1 then
|
|
x1:=ColCounter;
|
|
if ColCounter>x2 then
|
|
x2:=ColCounter;
|
|
if LineCounter<y1 then
|
|
y1:=LineCounter;
|
|
if LineCounter>y2 then
|
|
y2:=LineCounter;
|
|
end;
|
|
{$ifdef VER1_0}
|
|
Word(LineBuf^[BufCounter].UniCodeChar) := WordRec(VideoBuf^[BufCounter]).One;
|
|
{$else}
|
|
LineBuf^[BufCounter].UniCodeChar := Widechar(WordRec(VideoBuf^[BufCounter]).One);
|
|
{$endif}
|
|
{ If (WordRec(VideoBuf^[BufCounter]).Two and $80)<>0 then
|
|
LineBuf^[BufCounter].Attributes := $100+WordRec(VideoBuf^[BufCounter]).Two
|
|
else }
|
|
LineBuf^[BufCounter].Attributes := WordRec(VideoBuf^[BufCounter]).Two;
|
|
|
|
Inc(BufCounter);
|
|
end; { for }
|
|
end; { for }
|
|
BufSize.X := ScreenWidth;
|
|
BufSize.Y := ScreenHeight;
|
|
|
|
with WriteRegion do
|
|
begin
|
|
if force then
|
|
begin
|
|
Top := 0;
|
|
Left :=0;
|
|
Bottom := ScreenHeight-1;
|
|
Right := ScreenWidth-1;
|
|
BufCoord.X := 0;
|
|
BufCoord.Y := 0;
|
|
end
|
|
else
|
|
begin
|
|
Top := y1-1;
|
|
Left :=x1-1;
|
|
Bottom := y2-1;
|
|
Right := x2-1;
|
|
BufCoord.X := x1-1;
|
|
BufCoord.Y := y1-1;
|
|
end;
|
|
end;
|
|
{
|
|
writeln('X1: ',x1);
|
|
writeln('Y1: ',y1);
|
|
writeln('X2: ',x2);
|
|
writeln('Y2: ',y2);
|
|
}
|
|
WriteConsoleOutput(TextRec(Output).Handle, LineBuf, BufSize, BufCoord, WriteRegion);
|
|
Dispose(LineBuf);
|
|
|
|
move(VideoBuf^,OldVideoBuf^,VideoBufSize);
|
|
end;
|
|
end;
|
|
|
|
Const
|
|
SysVideoDriver : TVideoDriver = (
|
|
InitDriver : @SysInitVideo;
|
|
DoneDriver : @SysDoneVideo;
|
|
UpdateScreen : @SysUpdateScreen;
|
|
ClearScreen : @SysClearScreen;
|
|
SetVideoMode : Nil;
|
|
GetVideoModeCount : Nil;
|
|
GetVideoModeData : Nil;
|
|
SetCursorPos : @SysSetCursorPos;
|
|
GetCursorType : @SysGetCursorType;
|
|
SetCursorType : @SysSetCursorType;
|
|
GetCapabilities : @SysGetCapabilities
|
|
|
|
);
|
|
|
|
initialization
|
|
SetVideoDriver(SysVideoDriver);
|
|
end.
|
|
{
|
|
$Log$
|
|
Revision 1.11 2003-09-17 15:06:36 peter
|
|
* stdcall patch
|
|
|
|
Revision 1.10 2002/12/15 20:22:56 peter
|
|
* fix updatescreen compare that was wrong when the last char was
|
|
different
|
|
|
|
Revision 1.9 2002/10/06 20:00:22 peter
|
|
* Use Widechar in the Windows unit
|
|
|
|
Revision 1.8 2002/09/07 16:01:29 peter
|
|
* old logs removed and tabs fixed
|
|
|
|
}
|
|
|