fpc/rtl/amiga/crt.pp
1998-03-25 11:18:12 +00:00

628 lines
16 KiB
ObjectPascal

{
$Id$
This file is part of the Free Pascal run time library.
Copyright (c) 1997 by Nils Sjoholm
member of the Amiga RTL development team.
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
const
{ screen modes }
bw40 = 0;
co40 = 1;
bw80 = 2;
co80 = 3;
mono = 7;
font8x8 = 256;
{ screen color, fore- and background }
black = 0;
blue = 1;
green = 2;
cyan = 3;
red = 4;
magenta = 5;
brown = 6;
lightgray = 7;
{ only foreground }
darkgray = 8;
lightblue = 9;
lightgreen = 10;
lightcyan = 11;
lightred = 12;
lightmagenta = 13;
yellow = 14;
white = 15;
{ blink flag }
blink = $80;
var
{ for compatibility }
checkbreak,checkeof,checksnow : boolean;
{ works in another way than in TP }
{ true: cursor is set with direct port access }
{ false: cursor is set with a bios call }
directvideo : boolean;
lastmode : word; { screen mode}
textattr : byte; { current text attribute }
windmin : word; { upper right corner of the CRT window }
windmax : word; { lower left corner of the CRT window }
function keypressed : boolean;
function readkey : char;
procedure gotoxy(x,y : integer);
procedure window(left,top,right,bottom : byte);
procedure clrscr;
procedure textcolor(color : byte);
procedure textbackground(color : byte);
procedure assigncrt(var f : text);
function wherex : integer;
function wherey : integer;
procedure delline;
procedure delline(line : byte);
procedure clreol;
procedure insline;
procedure cursoron;
procedure cursoroff;
procedure cursorbig;
procedure lowvideo;
procedure highvideo;
procedure nosound;
procedure sound(hz : word);
procedure delay(ms : longint);
procedure textmode(mode : integer);
procedure normvideo;
implementation
Type
{$PACKRECORDS 4}
{ returned by Info(), must be on a 4 byte boundary }
pInfoData = ^tInfoData;
tInfoData = record
id_NumSoftErrors : Longint; { number of soft errors on disk }
id_UnitNumber : Longint; { Which unit disk is (was) mounted on }
id_DiskState : Longint; { See defines below }
id_NumBlocks : Longint; { Number of blocks on disk }
id_NumBlocksUsed : Longint; { Number of block in use }
id_BytesPerBlock : Longint;
id_DiskType : Longint; { Disk Type code }
id_VolumeNode : Longint; { BCPL pointer to volume node }
id_InUse : Longint; { Flag, zero if not in use }
end;
{ * List Node Structure. Each member in a list starts with a Node * }
pNode = ^tNode;
tNode = Record
ln_Succ, { * Pointer to next (successor) * }
ln_Pred : pNode; { * Pointer to previous (predecessor) * }
ln_Type : Byte;
ln_Pri : Shortint; { * Priority, for sorting * }
ln_Name : PChar; { * ID string, null terminated * }
End; { * Note: Integer aligned * }
{$PACKRECORDS NORMAL}
{ normal, full featured list }
pList = ^tList;
tList = record
lh_Head : pNode;
lh_Tail : pNode;
lh_TailPred : pNode;
lh_Type : Byte;
l_pad : Byte;
end;
pMsgPort = ^tMsgPort;
tMsgPort = record
mp_Node : tNode;
mp_Flags : Byte;
mp_SigBit : Byte; { signal bit number }
mp_SigTask : Pointer; { task to be signalled (TaskPtr) }
mp_MsgList : tList; { message linked list }
end;
pMessage = ^tMessage;
tMessage = record
mn_Node : tNode;
mn_ReplyPort : pMsgPort; { message reply port }
mn_Length : Word; { message len in bytes }
end;
pIOStdReq = ^tIOStdReq;
tIOStdReq = record
io_Message : tMessage;
io_Device : Pointer; { device node pointer }
io_Unit : Pointer; { unit (driver private)}
io_Command : Word; { device command }
io_Flags : Byte;
io_Error : Shortint; { error or warning num }
io_Actual : Longint; { actual number of bytes transferred }
io_Length : Longint; { requested number bytes transferred}
io_Data : Pointer; { points to data area }
io_Offset : Longint; { offset for block structured devices }
end;
pIntuiMessage = ^tIntuiMessage;
tIntuiMessage = record
ExecMessage : tMessage;
Class_ : Longint;
Code : Word;
Qualifier : Word;
IAddress : Pointer;
MouseX,
MouseY : Word;
Seconds,
Micros : Longint;
IDCMPWindow : Pointer;
SpecialLink : pIntuiMessage;
end;
pWindow = ^tWindow;
tWindow = record
NextWindow : pWindow; { for the linked list in a screen }
LeftEdge,
TopEdge : Integer; { screen dimensions of window }
Width,
Height : Integer; { screen dimensions of window }
MouseY,
MouseX : Integer; { relative to upper-left of window }
MinWidth,
MinHeight : Integer; { minimum sizes }
MaxWidth,
MaxHeight : Word; { maximum sizes }
Flags : Longint; { see below for defines }
MenuStrip : Pointer; { the strip of Menu headers }
Title : PChar; { the title text for this window }
FirstRequest : Pointer; { all active Requesters }
DMRequest : Pointer; { double-click Requester }
ReqCount : Integer; { count of reqs blocking Window }
WScreen : Pointer; { this Window's Screen }
RPort : Pointer; { this Window's very own RastPort }
BorderLeft,
BorderTop,
BorderRight,
BorderBottom : Shortint;
BorderRPort : Pointer;
FirstGadget : Pointer;
Parent,
Descendant : pWindow;
Pointer_ : Pointer; { sprite data }
PtrHeight : Shortint; { sprite height (not including sprite padding) }
PtrWidth : Shortint; { sprite width (must be less than or equal to 16) }
XOffset,
YOffset : Shortint; { sprite offsets }
IDCMPFlags : Longint; { User-selected flags }
UserPort,
WindowPort : pMsgPort;
MessageKey : pIntuiMessage;
DetailPen,
BlockPen : Byte; { for bar/border/gadget rendering }
CheckMark : Pointer;
ScreenTitle : PChar; { if non-null, Screen title when Window is active }
GZZMouseX : Integer;
GZZMouseY : Integer;
GZZWidth : Integer;
GZZHeight : Word;
ExtData : Pointer;
UserData : Pointer; { general-purpose pointer to User data extension }
WLayer : Pointer;
IFont : Pointer;
MoreFlags : Longint;
end;
pConUnit = ^tConUnit;
tConUnit = record
cu_MP : tMsgPort;
cu_Window : Pointer; { (WindowPtr) intuition window bound to this unit }
cu_XCP : Integer; { character position }
cu_YCP : Integer;
cu_XMax : Integer; { max character position }
cu_YMax : Integer;
cu_XRSize : Integer; { character raster size }
cu_YRSize : Integer;
cu_XROrigin : Integer; { raster origin }
cu_YROrigin : Integer;
cu_XRExtant : Integer; { raster maxima }
cu_YRExtant : Integer;
cu_XMinShrink : Integer; { smallest area intact from resize process }
cu_YMinShrink : Integer;
cu_XCCP : Integer; { cursor position }
cu_YCCP : Integer;
cu_KeyMapStruct : Pointer;
cu_TabStops : Array [0..80-1] of Word;
cu_Mask : Shortint;
cu_FgPen : Shortint;
cu_BgPen : Shortint;
cu_AOLPen : Shortint;
cu_DrawMode : Shortint;
cu_AreaPtSz : Shortint;
cu_AreaPtrn : Pointer; { cursor area pattern }
cu_Minterms : Array [0..7] of Byte; { console minterms }
cu_Font : Pointer; { (TextFontPtr) }
cu_AlgoStyle : Byte;
cu_TxFlags : Byte;
cu_TxHeight : Word;
cu_TxWidth : Word;
cu_TxBaseline : Word;
cu_TxSpacing : Word;
cu_Modes : Array [0..(22+7) div 8 - 1] of Byte;
cu_RawEvents : Array [0..($15+7) div 8 - 1] of Byte;
end;
const
CD_CURRX = 1;
CD_CURRY = 2;
CD_MAXX = 3;
CD_MAXY = 4;
function AllocVec( size, reqm : Longint ): Pointer; Assembler;
asm
MOVE.L A6,-(A7)
MOVE.L _ExecBase,A6
MOVE.L size,d0
MOVE.L reqm,d1
JSR -684(A6)
MOVE.L (A7)+,A6
end;
function DoPkt(ID : pMsgPort;
Action, Param1, Param2,
Param3, Param4, Param5 : Longint) : Longint; Assembler;
asm
MOVEM.L d2/d3/d4/d5/d6/d7/a6,-(A7)
MOVE.L _DOSBase,A6
MOVE.L ID,d1
MOVE.L Action,d2
MOVE.L Param1,d3
MOVE.L Param2,d4
MOVE.L Param3,d5
MOVE.L Param4,d6
MOVE.L Param5,d7
JSR -240(A6)
MOVEM.L (A7)+,d2/d3/d4/d5/d6/d7/a6
end;
procedure FreeVec( memory : Pointer ); Assembler;
asm
MOVE.L A6,-(A7)
MOVE.L _ExecBase,A6
MOVE.L memory,a1
JSR -690(A6)
MOVE.L (A7)+,A6
end;
function GetConsoleTask : pMsgPort; Assembler;
asm
MOVE.L A6,-(A7)
MOVE.L _DOSBase,A6
JSR -510(A6)
MOVE.L (A7)+,A6
end;
function GetMsg(port : pMsgPort): pMessage; Assembler;
asm
MOVE.L A6,-(A7)
MOVE.L _ExecBase,A6
MOVE.L port,a0
JSR -372(A6)
MOVE.L (A7)+,A6
end;
function ModifyIDCMP(window : pWindow;
IDCMPFlags : Longint) : Boolean; Assembler;
asm
MOVE.L A6,-(A7)
MOVE.L _IntuitionBase,A6
MOVE.L window,a0
MOVE.L IDCMPFlags,d0
JSR -150(A6)
MOVE.L (A7)+,A6
TST.L d0
SNE d0
end;
procedure ReplyMsg(mess : pMessage); Assembler;
asm
MOVE.L A6,-(A7)
MOVE.L _ExecBase,A6
MOVE.L mess,a1
JSR -378(A6)
MOVE.L (A7)+,A6
end;
function WaitPort(port : pMsgPort): pMessage; Assembler;
asm
MOVE.L A6,-(A7)
MOVE.L _ExecBase,A6
MOVE.L port,a0
JSR -384(A6)
MOVE.L (A7)+,A6
end;
procedure Delay_(ticks : Integer); Assembler;
asm
MOVE.L A6,-(A7)
MOVE.L _DOSBase,A6
MOVE.L ticks,d1
JSR -198(A6)
MOVE.L (A7)+,A6
end;
function OpenInfo : pInfoData;
var
port : pMsgPort;
info : pInfoData;
bptr, d4, d5, d6, d7 : Longint;
begin
info := pInfoData(AllocVec(SizeOf(tInfoData), 1));
if info <> nil then begin
port := GetConsoleTask;
bptr := Longint(info) shr 2;
if port <> nil then begin
if DoPkt(port, $19, bptr, d4, d5, d6, d7) <> 0 then info := pInfoData(bptr shl 2)
else port := nil;
end;
if port = nil then begin
FreeVec(info);
info := nil;
end;
end;
OpenInfo := info;
end;
procedure CloseInfo(var info : pInfoData);
begin
if info <> nil then begin
FreeVec(info);
info := nil;
end;
end;
function ConData(modus : byte) : integer;
var
info : pInfoData;
theunit : pConUnit;
pos : Longint;
begin
pos := 1;
info := OpenInfo;
if info <> nil then begin
theunit := pConUnit((pIoStdReq(info^.id_InUse))^.io_Unit);
case modus of
CD_CURRX : pos := theunit^.cu_XCP;
CD_CURRY : pos := theunit^.cu_YCP;
CD_MAXX : pos := theunit^.cu_XMax;
CD_MAXY : pos := theunit^.cu_YMax;
end;
CloseInfo(info);
end;
ConData := pos + 1;
end;
function wherex : integer;
begin
wherex := ConData(CD_CURRX);
end;
function wherey : integer;
begin
wherey := ConData(CD_CURRY);
end;
function maxx : integer;
begin
maxx := ConData(CD_MAXX);
end;
function maxy : integer;
begin
maxy := ConData(CD_MAXY);
end;
procedure gotoxy(x, y : integer);
var
mx, my : integer;
begin
mx := maxx;
my := maxy;
if x < 1 then x := wherex
else if x > mx then x := mx;
if y < 1 then y := wherey
else if y > my then y := my;
Write($9b, y, ';', x, 'H');
end;
procedure cursoroff;
begin
Write($9b,'0 p');
end;
procedure cursoron;
begin
Write($9b,'1 p');
end;
procedure clrscr;
begin
Write(Chr($0c));
end;
function ReadKey : char;
const
IDCMP_VANILLAKEY = $00200000;
IDCMP_RAWKEY = $00000400;
var
info : pInfoData;
win : pWindow;
imsg : pIntuiMessage;
msg : pMessage;
key : char;
idcmp, vanil : longint;
begin
key := #0;
info := OpenInfo;
if info <> nil then begin
win := pWindow(pConUnit((pIoStdReq(info^.id_InUse))^.io_Unit)^.cu_Window);
idcmp := win^.IDCMPFlags;
vanil := IDCMP_VANILLAKEY or IDCMP_RAWKEY;
ModifyIDCMP(win, (idcmp or vanil));
repeat
msg := WaitPort(win^.UserPort);
imsg := pIntuiMessage(GetMsg(win^.UserPort));
if (imsg^.Class_ = IDCMP_VANILLAKEY) or (imsg^.Class_ = IDCMP_RAWKEY) then key := char(imsg^.Code);
ReplyMsg(pMessage(imsg));
until key <> char(0);
repeat
msg := GetMsg(win^.UserPort);
if msg <> nil then ReplyMsg(msg);
until msg = nil;
ModifyIDCMP(win, idcmp);
CloseInfo(info);
end;
ReadKey := key;
end;
procedure textcolor(fgpen : byte);
begin
Write($9b, '3', fgpen, 'm');
end;
procedure textbackground(bgpen : byte);
begin
Write($9b, '4', bgpen, 'm');
end;
function keypressed : boolean;
begin
keypressed := true;
end;
procedure window(left,top,right,bottom : byte);
begin
end;
procedure assigncrt(var f : text);
begin
end;
procedure delline;
begin
Write($9b,'X');
end;
procedure delline(line : byte);
begin
Write($9b,'X');
end;
procedure clreol;
begin
Write($9b,'K');
end;
procedure insline;
begin
Write($9b,'1 L');
end;
procedure cursorbig;
begin
end;
procedure lowvideo;
begin
end;
procedure highvideo;
begin
end;
procedure nosound;
begin
end;
procedure sound(hz : word);
begin
end;
{ MsDos have 1000 ticks per second
and Amiga only 50, so we have to
do some calcs here.
The min value this procedure will
handle is 20, (less you will get 0)
this will be 1 tick in Amiga. If
you want to use amigados delay just
use Delay_. }
procedure delay(ms : longint);
var
dummy : integer;
begin
dummy := trunc((real(ms) / 1000.0) * 50.0);
Delay_(dummy);
end;
procedure textmode(mode : integer);
begin
end;
procedure normvideo;
begin
end;
end.