Files
Catacomb3D/ID_US_1.C
2026-03-12 19:22:23 +01:00

1286 lines
31 KiB
C

/* Catacomb 3-D Source Code
* Copyright (C) 1993-2014 Flat Rock Software
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* 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. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
//
// ID Engine
// ID_US_1.c - User Manager - General routines
// v1.1d1
// By Jason Blochowiak
// Hacked up for Catacomb 3D
//
//
// This module handles dealing with user input & feedback
//
// Depends on: Input Mgr, View Mgr, some variables from the Sound, Caching,
// and Refresh Mgrs, Memory Mgr for background save/restore
//
// Globals:
// ingame - Flag set by game indicating if a game is in progress
// abortgame - Flag set if the current game should be aborted (if a load
// game fails)
// loadedgame - Flag set if a game was loaded
// abortprogram - Normally nil, this points to a terminal error message
// if the program needs to abort
// restartgame - Normally set to gd_Continue, this is set to one of the
// difficulty levels if a new game should be started
// PrintX, PrintY - Where the User Mgr will print (global coords)
// WindowX,WindowY,WindowW,WindowH - The dimensions of the current
// window
//
#include "ID_HEADS.H"
#pragma hdrstop
#pragma warn -pia
// Special imports
extern boolean showscorebox;
#ifdef KEEN
extern boolean oldshooting;
extern ScanCode firescan;
#else
ScanCode firescan;
#endif
// Global variables
char *abortprogram;
boolean NoWait,
HighScoresDirty;
word PrintX,PrintY;
word WindowX,WindowY,WindowW,WindowH;
// Internal variables
#define ConfigVersion 1
static char *ParmStrings[] = {"TEDLEVEL","NOWAIT"},
*ParmStrings2[] = {"COMP","NOCOMP"};
static boolean US_Started;
boolean Button0,Button1,
CursorBad;
int CursorX,CursorY;
void (*USL_MeasureString)(char far *,word *,word *) = VW_MeasurePropString,
(*USL_DrawString)(char far *) = VWB_DrawPropString;
boolean (*USL_SaveGame)(int),(*USL_LoadGame)(int);
void (*USL_ResetGame)(void);
SaveGame Games[MaxSaveGames];
HighScore Scores[MaxScores] =
{
{"Sir Lancelot",500,3},
{"",0},
{"",0},
{"",0},
{"",0},
{"",0},
{"",0},
};
// Internal routines
// Public routines
///////////////////////////////////////////////////////////////////////////
//
// USL_HardError() - Handles the Abort/Retry/Fail sort of errors passed
// from DOS.
//
///////////////////////////////////////////////////////////////////////////
#pragma warn -par
#pragma warn -rch
int
USL_HardError(word errval,int ax,int bp,int si)
{
#define IGNORE 0
#define RETRY 1
#define ABORT 2
extern void ShutdownId(void);
static char buf[32];
static WindowRec wr;
int di;
char c,*s,*t;
di = _DI;
if (ax < 0)
s = "Device Error";
else
{
if ((di & 0x00ff) == 0)
s = "Drive ~ is Write Protected";
else
s = "Error on Drive ~";
for (t = buf;*s;s++,t++) // Can't use sprintf()
if ((*t = *s) == '~')
*t = (ax & 0x00ff) + 'A';
*t = '\0';
s = buf;
}
c = peekb(0x40,0x49); // Get the current screen mode
if ((c < 4) || (c == 7))
goto oh_kill_me;
// DEBUG - handle screen cleanup
US_SaveWindow(&wr);
US_CenterWindow(30,3);
US_CPrint(s);
US_CPrint("(R)etry or (A)bort?");
VW_UpdateScreen();
IN_ClearKeysDown();
asm sti // Let the keyboard interrupts come through
while (true)
{
switch (IN_WaitForASCII())
{
case key_Escape:
case 'a':
case 'A':
goto oh_kill_me;
break;
case key_Return:
case key_Space:
case 'r':
case 'R':
US_ClearWindow();
VW_UpdateScreen();
US_RestoreWindow(&wr);
return(RETRY);
break;
}
}
oh_kill_me:
abortprogram = s;
ShutdownId();
fprintf(stderr,"Terminal Error: %s\n",s);
if (tedlevel)
fprintf(stderr,"You launched from TED. I suggest that you reboot...\n");
return(ABORT);
#undef IGNORE
#undef RETRY
#undef ABORT
}
#pragma warn +par
#pragma warn +rch
///////////////////////////////////////////////////////////////////////////
//
// USL_GiveSaveName() - Returns a pointer to a static buffer that contains
// the filename to use for the specified save game
//
///////////////////////////////////////////////////////////////////////////
char *
USL_GiveSaveName(word game)
{
static char name[] = "SAVEGAMx."EXTENSION;
name[7] = '0' + game;
return(name);
}
///////////////////////////////////////////////////////////////////////////
//
// US_SetLoadSaveHooks() - Sets the routines that the User Mgr calls after
// reading or writing the save game headers
//
///////////////////////////////////////////////////////////////////////////
void
US_SetLoadSaveHooks(boolean (*load)(int),boolean (*save)(int),void (*reset)(void))
{
USL_LoadGame = load;
USL_SaveGame = save;
USL_ResetGame = reset;
}
///////////////////////////////////////////////////////////////////////////
//
// USL_ReadConfig() - Reads the configuration file, if present, and sets
// things up accordingly. If it's not present, uses defaults. This file
// includes the high scores.
//
///////////////////////////////////////////////////////////////////////////
static void
USL_ReadConfig(void)
{
boolean gotit;
char sig[sizeof(EXTENSION)];
word version;
int file;
SDMode sd;
SMMode sm;
ControlType ctl;
if ((file = open("CONFIG."EXTENSION,O_BINARY | O_RDONLY)) != -1)
{
read(file,sig,sizeof(EXTENSION));
read(file,&version,sizeof(version));
if (strcmp(sig,EXTENSION) || (version != ConfigVersion))
{
close(file);
goto rcfailed;
}
read(file,Scores,sizeof(HighScore) * MaxScores);
read(file,&sd,sizeof(sd));
read(file,&sm,sizeof(sm));
read(file,&ctl,sizeof(ctl));
read(file,&(KbdDefs[0]),sizeof(KbdDefs[0]));
read(file,&showscorebox,sizeof(showscorebox));
read(file,&compatability,sizeof(compatability));
#ifdef KEEN
read(file,&oldshooting,sizeof(oldshooting));
read(file,&firescan,sizeof(firescan));
#endif
close(file);
HighScoresDirty = false;
gotit = true;
}
else
{
rcfailed:
sd = sdm_Off;
sm = smm_Off;
ctl = ctrl_Keyboard;
showscorebox = true;
#ifdef KEEN
oldshooting = false;
#endif
gotit = false;
HighScoresDirty = true;
}
SD_Default(gotit,sd,sm);
IN_Default(gotit,ctl);
}
///////////////////////////////////////////////////////////////////////////
//
// USL_WriteConfig() - Writes out the current configuration, including the
// high scores.
//
///////////////////////////////////////////////////////////////////////////
static void
USL_WriteConfig(void)
{
word version;
int file;
version = ConfigVersion;
file = open("CONFIG."EXTENSION,O_CREAT | O_BINARY | O_WRONLY,
S_IREAD | S_IWRITE | S_IFREG);
if (file != -1)
{
write(file,EXTENSION,sizeof(EXTENSION));
write(file,&version,sizeof(version));
write(file,Scores,sizeof(HighScore) * MaxScores);
write(file,&SoundMode,sizeof(SoundMode));
write(file,&MusicMode,sizeof(MusicMode));
if // Hack
(
(Controls[0] == ctrl_Joystick1)
|| (Controls[0] == ctrl_Joystick2)
)
Controls[0] = ctrl_Keyboard;
write(file,&(Controls[0]),sizeof(Controls[0]));
write(file,&(KbdDefs[0]),sizeof(KbdDefs[0]));
write(file,&showscorebox,sizeof(showscorebox));
write(file,&compatability,sizeof(compatability));
#ifdef KEEN
write(file,&oldshooting,sizeof(oldshooting));
write(file,&firescan,sizeof(firescan));
#endif
close(file);
}
}
///////////////////////////////////////////////////////////////////////////
//
// USL_CheckSavedGames() - Checks to see which saved games are present
// & valid
//
///////////////////////////////////////////////////////////////////////////
static void
USL_CheckSavedGames(void)
{
boolean ok;
char *filename;
word i;
int file;
SaveGame *game;
USL_SaveGame = 0;
USL_LoadGame = 0;
for (i = 0,game = Games;i < MaxSaveGames;i++,game++)
{
filename = USL_GiveSaveName(i);
ok = false;
if ((file = open(filename,O_BINARY | O_RDONLY)) != -1)
{
if
(
(read(file,game,sizeof(*game)) == sizeof(*game))
&& (!strcmp(game->signature,EXTENSION))
&& (game->oldtest == &PrintX)
)
ok = true;
close(file);
}
if (ok)
game->present = true;
else
{
strcpy(game->signature,EXTENSION);
game->present = false;
strcpy(game->name,"Empty");
}
}
}
///////////////////////////////////////////////////////////////////////////
//
// US_Startup() - Starts the User Mgr
//
///////////////////////////////////////////////////////////////////////////
void
US_Startup(void)
{
int i;
if (US_Started)
return;
harderr(USL_HardError); // Install the fatal error handler
US_InitRndT(true); // Initialize the random number generator
USL_ReadConfig(); // Read config file
for (i = 1;i < _argc;i++)
{
switch (US_CheckParm(_argv[i],ParmStrings2))
{
case 0:
if (grmode == EGAGR)
compatability = true;
break;
case 1:
compatability = false;
break;
}
}
US_Started = true;
}
///////////////////////////////////////////////////////////////////////////
//
// US_Setup() - Does the disk access part of the User Mgr's startup
//
///////////////////////////////////////////////////////////////////////////
void
US_Setup(void)
{
USL_CheckSavedGames(); // Check which saved games are present
}
///////////////////////////////////////////////////////////////////////////
//
// US_Shutdown() - Shuts down the User Mgr
//
///////////////////////////////////////////////////////////////////////////
void
US_Shutdown(void)
{
if (!US_Started)
return;
if (!abortprogram)
USL_WriteConfig();
US_Started = false;
}
///////////////////////////////////////////////////////////////////////////
//
// US_CheckParm() - checks to see if a string matches one of a set of
// strings. The check is case insensitive. The routine returns the
// index of the string that matched, or -1 if no matches were found
//
///////////////////////////////////////////////////////////////////////////
int
US_CheckParm(char *parm,char **strings)
{
char cp,cs,
*p,*s;
int i;
while (!isalpha(*parm)) // Skip non-alphas
parm++;
for (i = 0;*strings && **strings;i++)
{
for (s = *strings++,p = parm,cs = cp = 0;cs == cp;)
{
cs = *s++;
if (!cs)
return(i);
cp = *p++;
if (isupper(cs))
cs = tolower(cs);
if (isupper(cp))
cp = tolower(cp);
}
}
return(-1);
}
///////////////////////////////////////////////////////////////////////////
//
// USL_ScreenDraw() - Draws a chunk of the text screen (called only by
// US_TextScreen())
//
///////////////////////////////////////////////////////////////////////////
static void
USL_ScreenDraw(word x,word y,char *s,byte attr)
{
byte far *screen,far *oscreen;
screen = MK_FP(0xb800,(x * 2) + (y * 80 * 2));
oscreen = (&introscn + 7) + ((x - 1) * 2) + (y * 80 * 2) + 1;
while (*s)
{
*screen++ = *s++;
if (attr != 0xff)
{
*screen++ = (attr & 0x8f) | (*oscreen & 0x70);
oscreen += 2;
}
else
screen++;
}
}
///////////////////////////////////////////////////////////////////////////
//
// USL_ClearTextScreen() - Makes sure the screen is in text mode, clears it,
// and moves the cursor to the leftmost column of the bottom line
//
///////////////////////////////////////////////////////////////////////////
static void
USL_ClearTextScreen(void)
{
// Set to 80x25 color text mode
_AL = 3; // Mode 3
_AH = 0x00;
geninterrupt(0x10);
// Use BIOS to move the cursor to the bottom of the screen
_AH = 0x0f;
geninterrupt(0x10); // Get current video mode into _BH
_DL = 0; // Lefthand side of the screen
_DH = 24; // Bottom row
_AH = 0x02;
geninterrupt(0x10);
}
///////////////////////////////////////////////////////////////////////////
//
// US_TextScreen() - Puts up the startup text screen
// Note: These are the only User Manager functions that can be safely called
// before the User Mgr has been started up
//
///////////////////////////////////////////////////////////////////////////
void
US_TextScreen(void)
{
word i,n;
USL_ClearTextScreen();
_fmemcpy(MK_FP(0xb800,0),7 + &introscn,80 * 25 * 2);
// Check for TED launching here
for (i = 1;i < _argc;i++)
{
n = US_CheckParm(_argv[i],ParmStrings);
if (n == 0)
{
tedlevelnum = atoi(_argv[i + 1]);
if (tedlevelnum >= 0)
{
tedlevel = true;
return;
}
else
break;
}
else if (n == 1)
{
NoWait = true;
return;
}
}
}
///////////////////////////////////////////////////////////////////////////
//
// USL_Show() - Changes the appearance of one of the fields on the text
// screen. Possibly adds a checkmark in front of it and highlights it
//
///////////////////////////////////////////////////////////////////////////
static void
USL_Show(word x,word y,word w,boolean show,boolean hilight)
{
byte far *screen,far *oscreen;
screen = MK_FP(0xb800,((x - 1) * 2) + (y * 80 * 2));
oscreen = (&introscn + 7) + ((x - 1) * 2) + (y * 80 * 2) - 1;
*screen++ = show? 251 : ' '; // Checkmark char or space
// *screen = 0x48;
// *screen = (*oscreen & 0xf0) | 8;
oscreen += 2;
if (show && hilight)
{
for (w++;w--;screen += 2,oscreen += 2)
*screen = (*oscreen & 0xf0) | 0x0f;
}
}
///////////////////////////////////////////////////////////////////////////
//
// USL_ShowMem() - Right justifies a longword in one of the memory fields on
// the text screen
//
///////////////////////////////////////////////////////////////////////////
static void
USL_ShowMem(word x,word y,long mem)
{
char buf[16];
word i;
for (i = strlen(ltoa(mem,buf,10));i < 5;i++)
USL_ScreenDraw(x++,y," ",0xff);
USL_ScreenDraw(x,y,buf,0xff);
}
///////////////////////////////////////////////////////////////////////////
//
// US_UpdateTextScreen() - Called after the ID libraries are started up.
// Displays what hardware is present.
//
///////////////////////////////////////////////////////////////////////////
void
US_UpdateTextScreen(void)
{
boolean b;
longword totalmem;
// Show video card info
b = (grmode == CGAGR);
USL_Show(21,7,4,(videocard >= CGAcard) && (videocard <= VGAcard),b);
b = (grmode == EGAGR);
USL_Show(21,8,4,(videocard >= EGAcard) && (videocard <= VGAcard),b);
b = (grmode == VGAGR);
USL_Show(21,9,4,videocard == VGAcard,b);
if (compatability)
USL_ScreenDraw(5,10,"SVGA Compatibility Mode Enabled.",0x4f);
// Show input device info
USL_Show(60,7,8,true,true);
USL_Show(60,8,11,JoysPresent[0],true);
USL_Show(60,9,11,JoysPresent[1],true);
USL_Show(60,10,5,MousePresent,true);
// Show sound hardware info
USL_Show(21,14,11,true,SoundMode == sdm_PC);
b = (SoundMode == sdm_AdLib) || (MusicMode == smm_AdLib);
USL_Show(21,15,14,AdLibPresent,b);
if (b && AdLibPresent) // Hack because of two lines
{
byte far *screen,far *oscreen;
word x,y,w;
x = 21;
y = 16;
w = 14;
screen = MK_FP(0xb800,(x * 2) + (y * 80 * 2) - 1);
oscreen = (&introscn + 7) + (x * 2) + (y * 80 * 2) - 1;
oscreen += 2;
for (w++;w--;screen += 2,oscreen += 2)
*screen = (*oscreen & 0xf0) | 0x0f;
}
// Show memory available/used
USL_ShowMem(63,15,mminfo.mainmem / 1024);
USL_Show(53,15,23,true,true);
USL_ShowMem(63,16,mminfo.EMSmem / 1024);
USL_Show(53,16,23,mminfo.EMSmem? true : false,true);
USL_ShowMem(63,17,mminfo.XMSmem / 1024);
USL_Show(53,17,23,mminfo.XMSmem? true : false,true);
totalmem = mminfo.mainmem + mminfo.EMSmem + mminfo.XMSmem;
USL_ShowMem(63,18,totalmem / 1024);
USL_Show(53,18,23,true,true); // DEBUG
USL_ScreenDraw(52,18," ",0xff);
// Change Initializing... to Loading...
USL_ScreenDraw(27,22," Loading... ",0x9c);
}
///////////////////////////////////////////////////////////////////////////
//
// US_FinishTextScreen() - After the main program has finished its initial
// loading, this routine waits for a keypress and then clears the screen
//
///////////////////////////////////////////////////////////////////////////
void
US_FinishTextScreen(void)
{
static byte colors[] = {4,6,13,15,15,15,15,15,15};
boolean up;
int i,c;
// Change Loading... to Press a Key
if (!(tedlevel || NoWait))
{
// IN_ClearKeysDown();
for (i = 0,up = true;!IN_UserInput(4,true);)
{
c = colors[i];
if (up)
{
if (++i == 9)
i = 8,up = false;
}
else
{
if (--i < 0)
i = 1,up = true;
}
// USL_ScreenDraw(29,22," Ready - Press a Key ",0x00 + c);
}
}
// else
// USL_ScreenDraw(29,22," Ready - Press a Key ",0x9a);
// IN_ClearKeysDown();
USL_ClearTextScreen();
}
// Window/Printing routines
///////////////////////////////////////////////////////////////////////////
//
// US_SetPrintRoutines() - Sets the routines used to measure and print
// from within the User Mgr. Primarily provided to allow switching
// between masked and non-masked fonts
//
///////////////////////////////////////////////////////////////////////////
void
US_SetPrintRoutines(void (*measure)(char far *,word *,word *),void (*print)(char far *))
{
USL_MeasureString = measure;
USL_DrawString = print;
}
///////////////////////////////////////////////////////////////////////////
//
// US_Print() - Prints a string in the current window. Newlines are
// supported.
//
///////////////////////////////////////////////////////////////////////////
void
US_Print(char *s)
{
char c,*se;
word w,h;
while (*s)
{
se = s;
while ((c = *se) && (c != '\n'))
se++;
*se = '\0';
USL_MeasureString(s,&w,&h);
px = PrintX;
py = PrintY;
USL_DrawString(s);
s = se;
if (c)
{
*se = c;
s++;
PrintX = WindowX;
PrintY += h;
}
else
PrintX += w;
}
}
///////////////////////////////////////////////////////////////////////////
//
// US_PrintUnsigned() - Prints an unsigned long
//
///////////////////////////////////////////////////////////////////////////
void
US_PrintUnsigned(longword n)
{
char buffer[32];
US_Print(ultoa(n,buffer,10));
}
///////////////////////////////////////////////////////////////////////////
//
// US_PrintSigned() - Prints a signed long
//
///////////////////////////////////////////////////////////////////////////
void
US_PrintSigned(long n)
{
char buffer[32];
US_Print(ltoa(n,buffer,10));
}
///////////////////////////////////////////////////////////////////////////
//
// USL_PrintInCenter() - Prints a string in the center of the given rect
//
///////////////////////////////////////////////////////////////////////////
void
USL_PrintInCenter(char *s,Rect r)
{
word w,h,
rw,rh;
USL_MeasureString(s,&w,&h);
rw = r.lr.x - r.ul.x;
rh = r.lr.y - r.ul.y;
px = r.ul.x + ((rw - w) / 2);
py = r.ul.y + ((rh - h) / 2);
USL_DrawString(s);
}
///////////////////////////////////////////////////////////////////////////
//
// US_PrintCentered() - Prints a string centered in the current window.
//
///////////////////////////////////////////////////////////////////////////
void
US_PrintCentered(char *s)
{
Rect r;
r.ul.x = WindowX;
r.ul.y = WindowY;
r.lr.x = r.ul.x + WindowW;
r.lr.y = r.ul.y + WindowH;
USL_PrintInCenter(s,r);
}
///////////////////////////////////////////////////////////////////////////
//
// US_CPrintLine() - Prints a string centered on the current line and
// advances to the next line. Newlines are not supported.
//
///////////////////////////////////////////////////////////////////////////
void
US_CPrintLine(char *s)
{
word w,h;
USL_MeasureString(s,&w,&h);
if (w > WindowW)
Quit("US_CPrintLine() - String exceeds width");
px = WindowX + ((WindowW - w) / 2);
py = PrintY;
USL_DrawString(s);
PrintY += h;
}
///////////////////////////////////////////////////////////////////////////
//
// US_CPrint() - Prints a string in the current window. Newlines are
// supported.
//
///////////////////////////////////////////////////////////////////////////
void
US_CPrint(char *s)
{
char c,*se;
while (*s)
{
se = s;
while ((c = *se) && (c != '\n'))
se++;
*se = '\0';
US_CPrintLine(s);
s = se;
if (c)
{
*se = c;
s++;
}
}
}
///////////////////////////////////////////////////////////////////////////
//
// US_ClearWindow() - Clears the current window to white and homes the
// cursor
//
///////////////////////////////////////////////////////////////////////////
void
US_ClearWindow(void)
{
VWB_Bar(WindowX,WindowY,WindowW,WindowH,WHITE);
PrintX = WindowX;
PrintY = WindowY;
}
///////////////////////////////////////////////////////////////////////////
//
// US_DrawWindow() - Draws a frame and sets the current window parms
//
///////////////////////////////////////////////////////////////////////////
void
US_DrawWindow(word x,word y,word w,word h)
{
word i,
sx,sy,sw,sh;
WindowX = x * 8;
WindowY = y * 8;
WindowW = w * 8;
WindowH = h * 8;
PrintX = WindowX;
PrintY = WindowY;
sx = (x - 1) * 8;
sy = (y - 1) * 8;
sw = (w + 1) * 8;
sh = (h + 1) * 8;
US_ClearWindow();
VWB_DrawTile8M(sx,sy,0),VWB_DrawTile8M(sx,sy + sh,6);
for (i = sx + 8;i <= sx + sw - 8;i += 8)
VWB_DrawTile8M(i,sy,1),VWB_DrawTile8M(i,sy + sh,7);
VWB_DrawTile8M(i,sy,2),VWB_DrawTile8M(i,sy + sh,8);
for (i = sy + 8;i <= sy + sh - 8;i += 8)
VWB_DrawTile8M(sx,i,3),VWB_DrawTile8M(sx + sw,i,5);
}
///////////////////////////////////////////////////////////////////////////
//
// US_CenterWindow() - Generates a window of a given width & height in the
// middle of the screen
//
///////////////////////////////////////////////////////////////////////////
void
US_CenterWindow(word w,word h)
{
US_DrawWindow(((MaxX / 8) - w) / 2,((MaxY / 8) - h) / 2,w,h);
}
///////////////////////////////////////////////////////////////////////////
//
// US_CenterSaveWindow() - Generates a window of a given width & height in
// the middle of the screen, saving the background
//
///////////////////////////////////////////////////////////////////////////
void
US_CenterSaveWindow(word w,word h,memptr *save)
{
word x,y,
screen;
x = ((MaxX / 8) - w) / 2;
y = ((MaxY / 8) - h) / 2;
MM_GetPtr(save,(w * h) * CHARWIDTH);
screen = bufferofs + panadjust + ylookup[y] + (x * CHARWIDTH);
VW_ScreenToMem(screen,*save,w * CHARWIDTH,h);
US_DrawWindow(((MaxX / 8) - w) / 2,((MaxY / 8) - h) / 2,w,h);
}
///////////////////////////////////////////////////////////////////////////
//
// US_RestoreSaveWindow() - Restores the background of the size of the
// current window from the memory specified by save
//
///////////////////////////////////////////////////////////////////////////
void
US_RestoreSaveWindow(memptr *save)
{
word screen;
screen = bufferofs + panadjust + ylookup[WindowY] + (WindowX * CHARWIDTH);
VW_MemToScreen(*save,screen,WindowW * CHARWIDTH,WindowH);
MM_FreePtr(save);
}
///////////////////////////////////////////////////////////////////////////
//
// US_SaveWindow() - Saves the current window parms into a record for
// later restoration
//
///////////////////////////////////////////////////////////////////////////
void
US_SaveWindow(WindowRec *win)
{
win->x = WindowX;
win->y = WindowY;
win->w = WindowW;
win->h = WindowH;
win->px = PrintX;
win->py = PrintY;
}
///////////////////////////////////////////////////////////////////////////
//
// US_RestoreWindow() - Sets the current window parms to those held in the
// record
//
///////////////////////////////////////////////////////////////////////////
void
US_RestoreWindow(WindowRec *win)
{
WindowX = win->x;
WindowY = win->y;
WindowW = win->w;
WindowH = win->h;
PrintX = win->px;
PrintY = win->py;
}
// Cursor routines
#if 0
///////////////////////////////////////////////////////////////////////////
//
// US_StartCursor() - Sets up the cursor for User Mgr use
//
///////////////////////////////////////////////////////////////////////////
void
US_StartCursor(void)
{
CursorInfo info;
VW_SetCursor(CURSORARROWSPR);
CursorX = MaxX / 2;
CursorY = MaxY / 2;
VW_MoveCursor(CursorX,CursorY);
VW_ShowCursor();
IN_ReadCursor(&info); // Dispose of any accumulated movement
}
///////////////////////////////////////////////////////////////////////////
//
// US_ShutCursor() - Cleans up after US_StartCursor()
//
///////////////////////////////////////////////////////////////////////////
void
US_ShutCursor(void)
{
VW_HideCursor();
}
///////////////////////////////////////////////////////////////////////////
//
// US_UpdateCursor() - Gets the new cursor position & button states from
// the Input Mgr and tells the View Mgr where the cursor is
//
///////////////////////////////////////////////////////////////////////////
boolean
US_UpdateCursor(void)
{
CursorInfo info;
IN_ReadCursor(&info);
if (info.x || info.y || CursorBad)
{
CursorX += info.x;
if (CursorX >= MaxX)
CursorX = MaxX - 1;
else if (CursorX < 0)
CursorX = 0;
CursorY += info.y;
if (CursorY >= MaxY)
CursorY = MaxY - 1;
else if (CursorY < 0)
CursorY = 0;
VW_MoveCursor(CursorX,CursorY);
CursorBad = false;
}
Button0 = info.button0;
Button1 = info.button1;
return(Button0 || Button1);
}
#endif
// Input routines
///////////////////////////////////////////////////////////////////////////
//
// USL_XORICursor() - XORs the I-bar text cursor. Used by US_LineInput()
//
///////////////////////////////////////////////////////////////////////////
static void
USL_XORICursor(int x,int y,char *s,word cursor)
{
char buf[MaxString];
word w,h;
strcpy(buf,s);
buf[cursor] = '\0';
USL_MeasureString(buf,&w,&h);
px = x + w - 1;
py = y;
USL_DrawString("\x80");
}
///////////////////////////////////////////////////////////////////////////
//
// US_LineInput() - Gets a line of user input at (x,y), the string defaults
// to whatever is pointed at by def. Input is restricted to maxchars
// chars or maxwidth pixels wide. If the user hits escape (and escok is
// true), nothing is copied into buf, and false is returned. If the
// user hits return, the current string is copied into buf, and true is
// returned
//
///////////////////////////////////////////////////////////////////////////
boolean
US_LineInput(int x,int y,char *buf,char *def,boolean escok,
int maxchars,int maxwidth)
{
boolean redraw,
cursorvis,cursormoved,
done,result;
ScanCode sc;
char c,
s[MaxString],olds[MaxString];
word i,
cursor,
w,h,
len;
longword lasttime;
VW_HideCursor();
if (def)
strcpy(s,def);
else
*s = '\0';
*olds = '\0';
cursor = strlen(s);
cursormoved = redraw = true;
cursorvis = done = false;
lasttime = TimeCount;
LastASCII = key_None;
LastScan = sc_None;
while (!done)
{
if (cursorvis)
USL_XORICursor(x,y,s,cursor);
asm pushf
asm cli
sc = LastScan;
LastScan = sc_None;
c = LastASCII;
LastASCII = key_None;
asm popf
switch (sc)
{
case sc_LeftArrow:
if (cursor)
cursor--;
c = key_None;
cursormoved = true;
break;
case sc_RightArrow:
if (s[cursor])
cursor++;
c = key_None;
cursormoved = true;
break;
case sc_Home:
cursor = 0;
c = key_None;
cursormoved = true;
break;
case sc_End:
cursor = strlen(s);
c = key_None;
cursormoved = true;
break;
case sc_Return:
strcpy(buf,s);
done = true;
result = true;
c = key_None;
break;
case sc_Escape:
if (escok)
{
done = true;
result = false;
}
c = key_None;
break;
case sc_BackSpace:
if (cursor)
{
strcpy(s + cursor - 1,s + cursor);
cursor--;
redraw = true;
}
c = key_None;
cursormoved = true;
break;
case sc_Delete:
if (s[cursor])
{
strcpy(s + cursor,s + cursor + 1);
redraw = true;
}
c = key_None;
cursormoved = true;
break;
case 0x4c: // Keypad 5
case sc_UpArrow:
case sc_DownArrow:
case sc_PgUp:
case sc_PgDn:
case sc_Insert:
c = key_None;
break;
}
if (c)
{
len = strlen(s);
USL_MeasureString(s,&w,&h);
if
(
isprint(c)
&& (len < MaxString - 1)
&& ((!maxchars) || (len < maxchars))
&& ((!maxwidth) || (w < maxwidth))
)
{
for (i = len + 1;i > cursor;i--)
s[i] = s[i - 1];
s[cursor++] = c;
redraw = true;
}
}
if (redraw)
{
px = x;
py = y;
USL_DrawString(olds);
strcpy(olds,s);
px = x;
py = y;
USL_DrawString(s);
redraw = false;
}
if (cursormoved)
{
cursorvis = false;
lasttime = TimeCount - TickBase;
cursormoved = false;
}
if (TimeCount - lasttime > TickBase / 2)
{
lasttime = TimeCount;
cursorvis ^= true;
}
if (cursorvis)
USL_XORICursor(x,y,s,cursor);
VW_UpdateScreen();
}
if (cursorvis)
USL_XORICursor(x,y,s,cursor);
if (!result)
{
px = x;
py = y;
USL_DrawString(olds);
}
VW_ShowCursor();
VW_UpdateScreen();
IN_ClearKeysDown();
return(result);
}