2019-02-03 06:19:25 +03:00
|
|
|
/* $NetBSD: main.c,v 1.24 2019/02/03 03:19:25 mrg Exp $ */
|
1995-03-24 06:58:08 +03:00
|
|
|
|
1994-10-22 00:19:39 +03:00
|
|
|
/*
|
|
|
|
* Phantasia 3.3.2 -- Interterminal fantasy game
|
|
|
|
*
|
|
|
|
* Edward A. Estes
|
|
|
|
* AT&T, March 12, 1986
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* DISCLAIMER:
|
|
|
|
*
|
|
|
|
* This game is distributed for free as is. It is not guaranteed to work
|
|
|
|
* in every conceivable environment. It is not even guaranteed to work
|
|
|
|
* in ANY environment.
|
|
|
|
*
|
|
|
|
* This game is distributed without notice of copyright, therefore it
|
|
|
|
* may be used in any manner the recipient sees fit. However, the
|
|
|
|
* author assumes no responsibility for maintaining or revising this
|
|
|
|
* game, in its original form, or any derivitives thereof.
|
|
|
|
*
|
|
|
|
* The author shall not be responsible for any loss, cost, or damage,
|
|
|
|
* including consequential damage, caused by reliance on this material.
|
|
|
|
*
|
|
|
|
* The author makes no warranties, express or implied, including warranties
|
|
|
|
* of merchantability or fitness for a particular purpose or use.
|
|
|
|
*
|
|
|
|
* AT&T is in no way connected with this game.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
2009-08-31 12:27:16 +04:00
|
|
|
#include <sys/stat.h>
|
2009-05-26 04:27:23 +04:00
|
|
|
#include <err.h>
|
2009-08-31 12:27:16 +04:00
|
|
|
#include <math.h>
|
1994-10-22 00:19:39 +03:00
|
|
|
#include <pwd.h>
|
2009-08-31 12:27:16 +04:00
|
|
|
#include <setjmp.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include "macros.h"
|
|
|
|
#include "phantdefs.h"
|
|
|
|
#include "phantstruct.h"
|
|
|
|
#include "phantglobs.h"
|
|
|
|
#include "pathnames.h"
|
|
|
|
|
|
|
|
#undef bool
|
|
|
|
#include <curses.h>
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The program allocates as much file space as it needs to store characters,
|
|
|
|
* so the possibility exists for the character file to grow without bound.
|
|
|
|
* The file is purged upon normal entry to try to avoid that problem.
|
|
|
|
* A similar problem exists for energy voids. To alleviate the problem here,
|
|
|
|
* the void file is cleared with every new king, and a limit is placed
|
|
|
|
* on the size of the energy void file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Put one line of text into the file 'motd' for announcements, etc.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The scoreboard file is updated when someone dies, and keeps track
|
|
|
|
* of the highest character to date for that login.
|
|
|
|
* Being purged from the character file does not cause the scoreboard
|
|
|
|
* to be updated.
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* main.c Main routines for Phantasia
|
|
|
|
*/
|
|
|
|
|
2009-08-12 12:21:41 +04:00
|
|
|
static void genchar(int);
|
|
|
|
static void initialstate(void);
|
|
|
|
static void neatstuff(void);
|
|
|
|
static void playinit(void);
|
|
|
|
static void procmain(void);
|
|
|
|
static long recallplayer(void);
|
|
|
|
static long rollnewplayer(void);
|
|
|
|
static void titlelist(void);
|
|
|
|
|
2003-05-08 17:03:49 +04:00
|
|
|
int main(int, char **);
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
int
|
2009-05-26 03:08:45 +04:00
|
|
|
main(int argc, char **argv)
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
bool noheader = FALSE; /* set if don't want header */
|
|
|
|
bool headeronly = FALSE; /* set if only want header */
|
|
|
|
bool examine = FALSE; /* set if examine a character */
|
|
|
|
time_t seconds; /* for time of day */
|
|
|
|
double dtemp; /* for temporary calculations */
|
|
|
|
|
|
|
|
initialstate(); /* init globals */
|
|
|
|
|
|
|
|
/* process arguments */
|
|
|
|
while (--argc && (*++argv)[0] == '-')
|
|
|
|
switch ((*argv)[1]) {
|
|
|
|
case 's': /* short */
|
|
|
|
noheader = TRUE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'H': /* Header */
|
|
|
|
headeronly = TRUE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'a': /* all users */
|
|
|
|
activelist();
|
|
|
|
cleanup(TRUE);
|
2019-02-03 06:19:25 +03:00
|
|
|
__unreachable();
|
1997-10-13 06:18:06 +04:00
|
|
|
/* NOTREACHED */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
case 'p': /* purge old players */
|
|
|
|
purgeoldplayers();
|
|
|
|
cleanup(TRUE);
|
2019-02-03 06:19:25 +03:00
|
|
|
__unreachable();
|
1997-10-13 06:18:06 +04:00
|
|
|
/* NOTREACHED */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
case 'S': /* set 'Wizard' */
|
|
|
|
Wizard = !getuid();
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
case 'x': /* examine */
|
|
|
|
examine = TRUE;
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
case 'm': /* monsters */
|
|
|
|
monstlist();
|
|
|
|
cleanup(TRUE);
|
2019-02-03 06:19:25 +03:00
|
|
|
__unreachable();
|
1997-10-13 06:18:06 +04:00
|
|
|
/* NOTREACHED */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
case 'b': /* scoreboard */
|
|
|
|
scorelist();
|
|
|
|
cleanup(TRUE);
|
2019-02-03 06:19:25 +03:00
|
|
|
__unreachable();
|
1997-10-13 06:18:06 +04:00
|
|
|
/* NOTREACHED */
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if (!isatty(0)) /* don't let non-tty's play */
|
|
|
|
cleanup(TRUE);
|
|
|
|
/* NOTREACHED */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
playinit(); /* set up to catch signals, init curses */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if (examine) {
|
|
|
|
changestats(FALSE);
|
|
|
|
cleanup(TRUE);
|
2019-02-03 06:19:25 +03:00
|
|
|
__unreachable();
|
1997-10-13 06:18:06 +04:00
|
|
|
/* NOTREACHED */
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
if (!noheader) {
|
|
|
|
titlelist();
|
|
|
|
purgeoldplayers(); /* clean up old characters */
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
2019-02-03 06:19:25 +03:00
|
|
|
if (headeronly) {
|
1997-10-13 06:18:06 +04:00
|
|
|
cleanup(TRUE);
|
2019-02-03 06:19:25 +03:00
|
|
|
__unreachable();
|
|
|
|
/* NOTREACHED */
|
|
|
|
}
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
do
|
|
|
|
/* get the player structure filled */
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
Fileloc = -1L;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
mvaddstr(22, 17, "Do you have a character to run [Q = Quit] ? ");
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
switch (getanswer("NYQ", FALSE)) {
|
|
|
|
case 'Y':
|
|
|
|
Fileloc = recallplayer();
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
case 'Q':
|
|
|
|
cleanup(TRUE);
|
2019-02-03 06:19:25 +03:00
|
|
|
__unreachable();
|
1997-10-13 06:18:06 +04:00
|
|
|
/* NOTREACHED */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
default:
|
|
|
|
Fileloc = rollnewplayer();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
clear();
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
while (Fileloc < 0L);
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Player.p_level > 5.0)
|
|
|
|
/* low level players have long timeout */
|
|
|
|
Timeout = TRUE;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
/* update some important player statistics */
|
2009-05-26 04:30:16 +04:00
|
|
|
strlcpy(Player.p_login, Login, sizeof(Player.p_login));
|
1997-10-13 06:18:06 +04:00
|
|
|
time(&seconds);
|
|
|
|
Player.p_lastused = localtime(&seconds)->tm_yday;
|
|
|
|
Player.p_status = S_PLAYING;
|
|
|
|
writerecord(&Player, Fileloc);
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
Statptr = &Stattable[Player.p_type]; /* initialize pointer */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
/* catch interrupts */
|
1994-10-22 00:19:39 +03:00
|
|
|
#ifdef BSD41
|
1997-10-13 06:18:06 +04:00
|
|
|
sigset(SIGINT, interrupt);
|
1994-10-22 00:19:39 +03:00
|
|
|
#endif
|
|
|
|
#ifdef BSD42
|
1997-10-13 06:18:06 +04:00
|
|
|
signal(SIGINT, interrupt);
|
1994-10-22 00:19:39 +03:00
|
|
|
#endif
|
|
|
|
#ifdef SYS3
|
1997-10-13 06:18:06 +04:00
|
|
|
signal(SIGINT, interrupt);
|
1994-10-22 00:19:39 +03:00
|
|
|
#endif
|
|
|
|
#ifdef SYS5
|
1997-10-13 06:18:06 +04:00
|
|
|
signal(SIGINT, interrupt);
|
1994-10-22 00:19:39 +03:00
|
|
|
#endif
|
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
altercoordinates(Player.p_x, Player.p_y, A_FORCED); /* set some flags */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
clear();
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
for (;;)
|
|
|
|
/* loop forever, processing input */
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
adjuststats(); /* cleanup stats */
|
|
|
|
|
|
|
|
if (Throne && Player.p_crowns == 0 && Player.p_specialtype != SC_KING)
|
|
|
|
/* not allowed on throne -- move */
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
mvaddstr(5, 0, "You're not allowed in the Lord's Chamber without a crown.\n");
|
|
|
|
altercoordinates(0.0, 0.0, A_NEAR);
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
checktampered();/* check for energy voids, etc. */
|
|
|
|
|
|
|
|
if (Player.p_status != S_CLOAKED
|
|
|
|
/* not cloaked */
|
|
|
|
&& (dtemp = fabs(Player.p_x)) == fabs(Player.p_y)
|
|
|
|
/* |x| = |y| */
|
|
|
|
&& !Throne)
|
|
|
|
/* not on throne */
|
|
|
|
{
|
|
|
|
dtemp = sqrt(dtemp / 100.0);
|
|
|
|
if (floor(dtemp) == dtemp)
|
|
|
|
/* |x| / 100 == n*n; at a trading post */
|
|
|
|
{
|
|
|
|
tradingpost();
|
|
|
|
clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
checkbattle(); /* check for player to player battle */
|
|
|
|
neatstuff(); /* gurus, medics, etc. */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1998-08-30 13:19:36 +04:00
|
|
|
if (Player.p_status == S_CLOAKED) {
|
1997-10-13 06:18:06 +04:00
|
|
|
/* costs 3 mana per turn to be cloaked */
|
|
|
|
if (Player.p_mana > 3.0)
|
|
|
|
Player.p_mana -= 3.0;
|
|
|
|
else
|
|
|
|
/* ran out of mana, uncloak */
|
|
|
|
{
|
|
|
|
Player.p_status = S_PLAYING;
|
|
|
|
Changed = TRUE;
|
|
|
|
}
|
1998-08-30 13:19:36 +04:00
|
|
|
}
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Player.p_status != S_PLAYING && Player.p_status != S_CLOAKED)
|
|
|
|
/* change status back to S_PLAYING */
|
|
|
|
{
|
|
|
|
Player.p_status = S_PLAYING;
|
|
|
|
Changed = TRUE;
|
|
|
|
}
|
|
|
|
if (Changed)
|
|
|
|
/* update file only if important stuff has changed */
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
writerecord(&Player, Fileloc);
|
|
|
|
Changed = FALSE;
|
|
|
|
continue;
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
readmessage(); /* read message, if any */
|
|
|
|
|
|
|
|
displaystats(); /* print statistics */
|
|
|
|
|
|
|
|
move(6, 0);
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Throne)
|
|
|
|
/* maybe make king, print prompt, etc. */
|
|
|
|
throneroom();
|
|
|
|
|
|
|
|
/* print status line */
|
|
|
|
addstr("1:Move 2:Players 3:Talk 4:Stats 5:Quit ");
|
|
|
|
if (Player.p_level >= MEL_CLOAK && Player.p_magiclvl >= ML_CLOAK)
|
|
|
|
addstr("6:Cloak ");
|
|
|
|
if (Player.p_level >= MEL_TELEPORT && Player.p_magiclvl >= ML_TELEPORT)
|
|
|
|
addstr("7:Teleport ");
|
|
|
|
if (Player.p_specialtype >= SC_COUNCIL || Wizard)
|
|
|
|
addstr("8:Intervene ");
|
|
|
|
|
|
|
|
procmain(); /* process input */
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-12 12:21:41 +04:00
|
|
|
static void
|
2009-05-26 03:08:45 +04:00
|
|
|
initialstate(void)
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
2004-12-09 08:15:59 +03:00
|
|
|
struct stat sb;
|
2009-05-26 04:27:23 +04:00
|
|
|
struct passwd *pw;
|
2004-12-09 08:15:59 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
Beyond = FALSE;
|
|
|
|
Marsh = FALSE;
|
|
|
|
Throne = FALSE;
|
|
|
|
Changed = FALSE;
|
|
|
|
Wizard = FALSE;
|
|
|
|
Timeout = FALSE;
|
|
|
|
Users = 0;
|
|
|
|
Windows = FALSE;
|
|
|
|
Echo = TRUE;
|
|
|
|
|
|
|
|
/* setup login name */
|
2009-05-26 04:27:23 +04:00
|
|
|
if ((Login = getlogin()) == NULL) {
|
|
|
|
pw = getpwuid(getuid());
|
|
|
|
if (pw == NULL) {
|
|
|
|
errx(1, "Who are you?");
|
|
|
|
}
|
|
|
|
Login = pw->pw_name;
|
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
|
|
|
|
/* open some files */
|
|
|
|
if ((Playersfp = fopen(_PATH_PEOPLE, "r+")) == NULL)
|
|
|
|
error(_PATH_PEOPLE);
|
|
|
|
/* NOTREACHED */
|
1999-09-13 21:15:42 +04:00
|
|
|
if (fileno(Playersfp) < 3)
|
|
|
|
exit(1);
|
1997-10-13 06:18:06 +04:00
|
|
|
|
|
|
|
if ((Monstfp = fopen(_PATH_MONST, "r+")) == NULL)
|
|
|
|
error(_PATH_MONST);
|
|
|
|
/* NOTREACHED */
|
|
|
|
|
|
|
|
if ((Messagefp = fopen(_PATH_MESS, "r")) == NULL)
|
|
|
|
error(_PATH_MESS);
|
|
|
|
/* NOTREACHED */
|
|
|
|
|
|
|
|
if ((Energyvoidfp = fopen(_PATH_VOID, "r+")) == NULL)
|
|
|
|
error(_PATH_VOID);
|
2004-12-09 08:15:59 +03:00
|
|
|
if (fstat(fileno(Energyvoidfp), &sb) == -1)
|
|
|
|
error("stat");
|
|
|
|
if (sb.st_size == 0) {
|
|
|
|
/* initialize grail to new location */
|
|
|
|
Enrgyvoid.ev_active = TRUE;
|
|
|
|
Enrgyvoid.ev_x = ROLL(-1.0e6, 2.0e6);
|
|
|
|
Enrgyvoid.ev_y = ROLL(-1.0e6, 2.0e6);
|
|
|
|
writevoid(&Enrgyvoid, 0L);
|
|
|
|
}
|
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
/* NOTREACHED */
|
|
|
|
|
|
|
|
srandom((unsigned) time(NULL)); /* prime random numbers */
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
|
2009-08-12 12:21:41 +04:00
|
|
|
static long
|
2009-05-26 03:08:45 +04:00
|
|
|
rollnewplayer(void)
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
int chartype; /* character type */
|
|
|
|
int ch; /* input */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
initplayer(&Player); /* initialize player structure */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
clear();
|
|
|
|
mvaddstr(4, 21, "Which type of character do you want:");
|
|
|
|
mvaddstr(8, 4,
|
|
|
|
"1:Magic User 2:Fighter 3:Elf 4:Dwarf 5:Halfling 6:Experimento ");
|
|
|
|
if (Wizard) {
|
|
|
|
addstr("7:Super ? ");
|
|
|
|
chartype = getanswer("1234567", FALSE);
|
|
|
|
} else {
|
|
|
|
addstr("? ");
|
|
|
|
chartype = getanswer("123456", FALSE);
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
do {
|
|
|
|
genchar(chartype); /* roll up a character */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
/* print out results */
|
|
|
|
mvprintw(12, 14,
|
|
|
|
"Strength : %2.0f Quickness: %2.0f Mana : %2.0f\n",
|
|
|
|
Player.p_strength, Player.p_quickness, Player.p_mana);
|
|
|
|
mvprintw(13, 14,
|
|
|
|
"Energy Level: %2.0f Brains : %2.0f Magic Level: %2.0f\n",
|
|
|
|
Player.p_energy, Player.p_brains, Player.p_magiclvl);
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Player.p_type == C_EXPER || Player.p_type == C_SUPER)
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
mvaddstr(14, 14, "Type '1' to keep >");
|
|
|
|
ch = getanswer(" ", TRUE);
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
while (ch != '1');
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Player.p_type == C_EXPER || Player.p_type == C_SUPER)
|
|
|
|
/* get coordinates for experimento */
|
|
|
|
for (;;) {
|
|
|
|
mvaddstr(16, 0, "Enter the X Y coordinates of your experimento ? ");
|
|
|
|
getstring(Databuf, SZ_DATABUF);
|
|
|
|
sscanf(Databuf, "%lf %lf", &Player.p_x, &Player.p_y);
|
|
|
|
|
|
|
|
if (fabs(Player.p_x) > D_EXPER || fabs(Player.p_y) > D_EXPER)
|
|
|
|
mvaddstr(17, 0, "Invalid coordinates. Try again.\n");
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
for (;;)
|
|
|
|
/* name the new character */
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
mvprintw(18, 0,
|
|
|
|
"Give your character a name [up to %d characters] ? ", SZ_NAME - 1);
|
|
|
|
getstring(Player.p_name, SZ_NAME);
|
|
|
|
truncstring(Player.p_name); /* remove trailing blanks */
|
|
|
|
|
|
|
|
if (Player.p_name[0] == '\0')
|
|
|
|
/* no null names */
|
|
|
|
mvaddstr(19, 0, "Invalid name.");
|
|
|
|
else
|
|
|
|
if (findname(Player.p_name, &Other) >= 0L)
|
|
|
|
/* cannot have duplicate names */
|
|
|
|
mvaddstr(19, 0, "Name already in use.");
|
|
|
|
else
|
|
|
|
/* name is acceptable */
|
|
|
|
break;
|
|
|
|
|
|
|
|
addstr(" Pick another.\n");
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
/* get a password for character */
|
|
|
|
Echo = FALSE;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
do {
|
|
|
|
mvaddstr(20, 0, "Give your character a password [up to 8 characters] ? ");
|
|
|
|
getstring(Player.p_password, SZ_PASSWORD);
|
1999-08-18 05:39:16 +04:00
|
|
|
mvaddstr(21, 0, "Enter again to verify: ");
|
1997-10-13 06:18:06 +04:00
|
|
|
getstring(Databuf, SZ_PASSWORD);
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
while (strcmp(Player.p_password, Databuf) != 0);
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
Echo = TRUE;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
return (allocrecord());
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
|
2009-08-12 12:21:41 +04:00
|
|
|
static void
|
2009-05-26 03:08:45 +04:00
|
|
|
procmain(void)
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
int ch; /* input */
|
|
|
|
double x; /* desired new x coordinate */
|
|
|
|
double y; /* desired new y coordinate */
|
|
|
|
double temp; /* for temporary calculations */
|
|
|
|
FILE *fp; /* for opening files */
|
|
|
|
int loop; /* a loop counter */
|
|
|
|
bool hasmoved = FALSE; /* set if player has moved */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
ch = inputoption();
|
|
|
|
mvaddstr(4, 0, "\n\n"); /* clear status area */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
move(7, 0);
|
|
|
|
clrtobot(); /* clear data on bottom area of screen */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Player.p_specialtype == SC_VALAR && (ch == '1' || ch == '7'))
|
|
|
|
/* valar cannot move */
|
|
|
|
ch = ' ';
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
switch (ch) {
|
1994-10-22 00:19:39 +03:00
|
|
|
case 'K': /* move up/north */
|
|
|
|
case 'N':
|
1997-10-13 06:18:06 +04:00
|
|
|
x = Player.p_x;
|
|
|
|
y = Player.p_y + MAXMOVE();
|
|
|
|
hasmoved = TRUE;
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case 'J': /* move down/south */
|
|
|
|
case 'S':
|
1997-10-13 06:18:06 +04:00
|
|
|
x = Player.p_x;
|
|
|
|
y = Player.p_y - MAXMOVE();
|
|
|
|
hasmoved = TRUE;
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case 'L': /* move right/east */
|
|
|
|
case 'E':
|
1997-10-13 06:18:06 +04:00
|
|
|
x = Player.p_x + MAXMOVE();
|
|
|
|
y = Player.p_y;
|
|
|
|
hasmoved = TRUE;
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case 'H': /* move left/west */
|
|
|
|
case 'W':
|
1997-10-13 06:18:06 +04:00
|
|
|
x = Player.p_x - MAXMOVE();
|
|
|
|
y = Player.p_y;
|
|
|
|
hasmoved = TRUE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default: /* rest */
|
|
|
|
Player.p_energy += (Player.p_maxenergy + Player.p_shield) / 15.0
|
|
|
|
+ Player.p_level / 3.0 + 2.0;
|
|
|
|
Player.p_energy =
|
|
|
|
MIN(Player.p_energy, Player.p_maxenergy + Player.p_shield);
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Player.p_status != S_CLOAKED)
|
|
|
|
/* cannot find mana if cloaked */
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
Player.p_mana += (Circle + Player.p_level) / 4.0;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if (drandom() < 0.2 && Player.p_status == S_PLAYING && !Throne)
|
|
|
|
/* wandering monster */
|
|
|
|
encounter(-1);
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case 'X': /* change/examine a character */
|
1997-10-13 06:18:06 +04:00
|
|
|
changestats(TRUE);
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case '1': /* move */
|
1997-10-13 06:18:06 +04:00
|
|
|
for (loop = 3; loop; --loop) {
|
|
|
|
mvaddstr(4, 0, "X Y Coordinates ? ");
|
|
|
|
getstring(Databuf, SZ_DATABUF);
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if (sscanf(Databuf, "%lf %lf", &x, &y) != 2)
|
|
|
|
mvaddstr(5, 0, "Try again\n");
|
|
|
|
else
|
|
|
|
if (distance(Player.p_x, x, Player.p_y, y) > MAXMOVE())
|
|
|
|
ILLMOVE();
|
|
|
|
else {
|
|
|
|
hasmoved = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case '2': /* players */
|
1997-10-13 06:18:06 +04:00
|
|
|
userlist(TRUE);
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case '3': /* message */
|
1997-10-13 06:18:06 +04:00
|
|
|
mvaddstr(4, 0, "Message ? ");
|
|
|
|
getstring(Databuf, SZ_DATABUF);
|
|
|
|
/* we open the file for writing to erase any data which is
|
|
|
|
* already there */
|
|
|
|
fp = fopen(_PATH_MESS, "w");
|
|
|
|
if (Databuf[0] != '\0')
|
|
|
|
fprintf(fp, "%s: %s", Player.p_name, Databuf);
|
|
|
|
fclose(fp);
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case '4': /* stats */
|
1997-10-13 06:18:06 +04:00
|
|
|
allstatslist();
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case '5': /* good-bye */
|
1997-10-13 06:18:06 +04:00
|
|
|
leavegame();
|
2019-02-03 06:19:25 +03:00
|
|
|
__unreachable();
|
1997-10-13 06:18:06 +04:00
|
|
|
/* NOTREACHED */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case '6': /* cloak */
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Player.p_level < MEL_CLOAK || Player.p_magiclvl < ML_CLOAK)
|
|
|
|
ILLCMD();
|
|
|
|
else
|
|
|
|
if (Player.p_status == S_CLOAKED)
|
|
|
|
Player.p_status = S_PLAYING;
|
1994-10-22 00:19:39 +03:00
|
|
|
else
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Player.p_mana < MM_CLOAK)
|
|
|
|
mvaddstr(5, 0, "No mana left.\n");
|
|
|
|
else {
|
|
|
|
Changed = TRUE;
|
|
|
|
Player.p_mana -= MM_CLOAK;
|
|
|
|
Player.p_status = S_CLOAKED;
|
|
|
|
}
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
case '7': /* teleport */
|
|
|
|
/*
|
|
|
|
* conditions for teleport
|
|
|
|
* - 20 per (level plus magic level)
|
|
|
|
* - OR council of the wise or valar or ex-valar
|
|
|
|
* - OR transport from throne
|
|
|
|
* transports from throne cost no mana
|
|
|
|
*/
|
|
|
|
if (Player.p_level < MEL_TELEPORT || Player.p_magiclvl < ML_TELEPORT)
|
|
|
|
ILLCMD();
|
|
|
|
else
|
|
|
|
for (loop = 3; loop; --loop) {
|
|
|
|
mvaddstr(4, 0, "X Y Coordinates ? ");
|
|
|
|
getstring(Databuf, SZ_DATABUF);
|
|
|
|
|
|
|
|
if (sscanf(Databuf, "%lf %lf", &x, &y) == 2) {
|
|
|
|
temp = distance(Player.p_x, x, Player.p_y, y);
|
|
|
|
if (!Throne
|
|
|
|
/* can transport anywhere from throne */
|
|
|
|
&& Player.p_specialtype <= SC_COUNCIL
|
|
|
|
/* council, valar can transport
|
|
|
|
* anywhere */
|
|
|
|
&& temp > (Player.p_level + Player.p_magiclvl) * 20.0)
|
|
|
|
/* can only move 20 per exp.
|
|
|
|
* level + mag. level */
|
|
|
|
ILLMOVE();
|
|
|
|
else {
|
|
|
|
temp = (temp / 75.0 + 1.0) * 20.0; /* mana used */
|
|
|
|
|
|
|
|
if (!Throne && temp > Player.p_mana)
|
|
|
|
mvaddstr(5, 0, "Not enough power for that distance.\n");
|
|
|
|
else {
|
|
|
|
if (!Throne)
|
|
|
|
Player.p_mana -= temp;
|
|
|
|
hasmoved = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case 'C':
|
|
|
|
case '9': /* monster */
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Throne)
|
|
|
|
/* no monsters while on throne */
|
|
|
|
mvaddstr(5, 0, "No monsters in the chamber!\n");
|
|
|
|
else
|
|
|
|
if (Player.p_specialtype != SC_VALAR)
|
|
|
|
/* the valar cannot call monsters */
|
|
|
|
{
|
|
|
|
Player.p_sin += 1e-6;
|
|
|
|
encounter(-1);
|
|
|
|
}
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case '0': /* decree */
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Wizard || (Player.p_specialtype == SC_KING && Throne))
|
|
|
|
/* kings must be on throne to decree */
|
|
|
|
dotampered();
|
|
|
|
else
|
|
|
|
ILLCMD();
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case '8': /* intervention */
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Wizard || Player.p_specialtype >= SC_COUNCIL)
|
|
|
|
dotampered();
|
|
|
|
else
|
|
|
|
ILLCMD();
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if (hasmoved)
|
|
|
|
/* player has moved -- alter coordinates, and do random
|
|
|
|
* monster */
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
altercoordinates(x, y, A_SPECIFIC);
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if (drandom() < 0.2 && Player.p_status == S_PLAYING && !Throne)
|
|
|
|
encounter(-1);
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-12 12:21:41 +04:00
|
|
|
static void
|
2009-05-26 03:08:45 +04:00
|
|
|
titlelist(void)
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
FILE *fp; /* used for opening various files */
|
|
|
|
bool councilfound = FALSE; /* set if we find a member of the
|
|
|
|
* council */
|
|
|
|
bool kingfound = FALSE; /* set if we find a king */
|
|
|
|
double hiexp, nxtexp; /* used for finding the two highest players */
|
|
|
|
double hilvl, nxtlvl; /* used for finding the two highest players */
|
|
|
|
char hiname[21], nxtname[21]; /* used for finding the two
|
|
|
|
* highest players */
|
|
|
|
|
|
|
|
nxtexp = 0;
|
|
|
|
mvaddstr(0, 14,
|
|
|
|
"W e l c o m e t o P h a n t a s i a (vers. 3.3.2)!");
|
|
|
|
|
|
|
|
/* print message of the day */
|
|
|
|
if ((fp = fopen(_PATH_MOTD, "r")) != NULL
|
|
|
|
&& fgets(Databuf, SZ_DATABUF, fp) != NULL) {
|
|
|
|
mvaddstr(2, 40 - strlen(Databuf) / 2, Databuf);
|
|
|
|
fclose(fp);
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
/* search for king */
|
1999-09-09 01:57:16 +04:00
|
|
|
fseek(Playersfp, 0L, SEEK_SET);
|
1997-10-13 06:18:06 +04:00
|
|
|
while (fread((char *) &Other, SZ_PLAYERSTRUCT, 1, Playersfp) == 1)
|
|
|
|
if (Other.p_specialtype == SC_KING &&
|
|
|
|
Other.p_status != S_NOTUSED)
|
|
|
|
/* found the king */
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
2009-05-26 03:14:33 +04:00
|
|
|
snprintf(Databuf, SZ_DATABUF,
|
|
|
|
"The present ruler is %s Level:%.0f",
|
1997-10-13 06:18:06 +04:00
|
|
|
Other.p_name, Other.p_level);
|
|
|
|
mvaddstr(4, 40 - strlen(Databuf) / 2, Databuf);
|
|
|
|
kingfound = TRUE;
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
if (!kingfound)
|
|
|
|
mvaddstr(4, 24, "There is no ruler at this time.");
|
|
|
|
|
|
|
|
/* search for valar */
|
1999-09-09 01:57:16 +04:00
|
|
|
fseek(Playersfp, 0L, SEEK_SET);
|
1997-10-13 06:18:06 +04:00
|
|
|
while (fread((char *) &Other, SZ_PLAYERSTRUCT, 1, Playersfp) == 1)
|
|
|
|
if (Other.p_specialtype == SC_VALAR && Other.p_status != S_NOTUSED)
|
|
|
|
/* found the valar */
|
|
|
|
{
|
2009-05-26 03:14:33 +04:00
|
|
|
snprintf(Databuf, SZ_DATABUF,
|
|
|
|
"The Valar is %s Login: %s",
|
|
|
|
Other.p_name, Other.p_login);
|
1997-10-13 06:18:06 +04:00
|
|
|
mvaddstr(6, 40 - strlen(Databuf) / 2, Databuf);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* search for council of the wise */
|
1999-09-09 01:57:16 +04:00
|
|
|
fseek(Playersfp, 0L, SEEK_SET);
|
1997-10-13 06:18:06 +04:00
|
|
|
Lines = 10;
|
|
|
|
while (fread((char *) &Other, SZ_PLAYERSTRUCT, 1, Playersfp) == 1)
|
|
|
|
if (Other.p_specialtype == SC_COUNCIL && Other.p_status != S_NOTUSED)
|
|
|
|
/* found a member of the council */
|
|
|
|
{
|
|
|
|
if (!councilfound) {
|
|
|
|
mvaddstr(8, 30, "Council of the Wise:");
|
|
|
|
councilfound = TRUE;
|
|
|
|
}
|
|
|
|
/* This assumes a finite (<=5) number of C.O.W.: */
|
2009-05-26 03:14:33 +04:00
|
|
|
snprintf(Databuf, SZ_DATABUF,
|
|
|
|
"%s Login: %s", Other.p_name, Other.p_login);
|
1997-10-13 06:18:06 +04:00
|
|
|
mvaddstr(Lines++, 40 - strlen(Databuf) / 2, Databuf);
|
|
|
|
}
|
|
|
|
/* search for the two highest players */
|
|
|
|
nxtname[0] = hiname[0] = '\0';
|
|
|
|
hiexp = 0.0;
|
|
|
|
nxtlvl = hilvl = 0;
|
|
|
|
|
1999-09-09 01:57:16 +04:00
|
|
|
fseek(Playersfp, 0L, SEEK_SET);
|
1997-10-13 06:18:06 +04:00
|
|
|
while (fread((char *) &Other, SZ_PLAYERSTRUCT, 1, Playersfp) == 1)
|
|
|
|
if (Other.p_experience > hiexp && Other.p_specialtype <= SC_KING && Other.p_status != S_NOTUSED)
|
|
|
|
/* highest found so far */
|
|
|
|
{
|
|
|
|
nxtexp = hiexp;
|
|
|
|
hiexp = Other.p_experience;
|
|
|
|
nxtlvl = hilvl;
|
|
|
|
hilvl = Other.p_level;
|
|
|
|
strcpy(nxtname, hiname);
|
|
|
|
strcpy(hiname, Other.p_name);
|
|
|
|
} else
|
|
|
|
if (Other.p_experience > nxtexp
|
|
|
|
&& Other.p_specialtype <= SC_KING
|
|
|
|
&& Other.p_status != S_NOTUSED)
|
|
|
|
/* next highest found so far */
|
|
|
|
{
|
|
|
|
nxtexp = Other.p_experience;
|
|
|
|
nxtlvl = Other.p_level;
|
|
|
|
strcpy(nxtname, Other.p_name);
|
|
|
|
}
|
|
|
|
mvaddstr(15, 28, "Highest characters are:");
|
2009-05-26 03:14:33 +04:00
|
|
|
snprintf(Databuf, SZ_DATABUF,
|
|
|
|
"%s Level:%.0f and %s Level:%.0f",
|
1997-10-13 06:18:06 +04:00
|
|
|
hiname, hilvl, nxtname, nxtlvl);
|
|
|
|
mvaddstr(17, 40 - strlen(Databuf) / 2, Databuf);
|
|
|
|
|
|
|
|
/* print last to die */
|
|
|
|
if ((fp = fopen(_PATH_LASTDEAD, "r")) != NULL
|
|
|
|
&& fgets(Databuf, SZ_DATABUF, fp) != NULL) {
|
|
|
|
mvaddstr(19, 25, "The last character to die was:");
|
|
|
|
mvaddstr(20, 40 - strlen(Databuf) / 2, Databuf);
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
2006-05-14 02:29:53 +04:00
|
|
|
if (fp)
|
|
|
|
fclose(fp);
|
1997-10-13 06:18:06 +04:00
|
|
|
refresh();
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
|
2009-08-12 12:21:41 +04:00
|
|
|
static long
|
2009-05-26 03:08:45 +04:00
|
|
|
recallplayer(void)
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
long loc = 0L; /* location in player file */
|
|
|
|
int loop; /* loop counter */
|
|
|
|
int ch; /* input */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
clear();
|
|
|
|
mvprintw(10, 0, "What was your character's name ? ");
|
|
|
|
getstring(Databuf, SZ_NAME);
|
|
|
|
truncstring(Databuf);
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if ((loc = findname(Databuf, &Player)) >= 0L)
|
|
|
|
/* found character */
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
Echo = FALSE;
|
|
|
|
|
|
|
|
for (loop = 0; loop < 2; ++loop) {
|
|
|
|
/* prompt for password */
|
|
|
|
mvaddstr(11, 0, "Password ? ");
|
|
|
|
getstring(Databuf, SZ_PASSWORD);
|
|
|
|
if (strcmp(Databuf, Player.p_password) == 0)
|
|
|
|
/* password good */
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
Echo = TRUE;
|
|
|
|
|
|
|
|
if (Player.p_status != S_OFF)
|
|
|
|
/* player did not exit normally last
|
|
|
|
* time */
|
|
|
|
{
|
|
|
|
clear();
|
|
|
|
addstr("Your character did not exit normally last time.\n");
|
|
|
|
addstr("If you think you have good cause to have your character saved,\n");
|
|
|
|
printw("you may quit and mail your reason to 'root'.\n");
|
|
|
|
addstr("Otherwise, continuing spells certain death.\n");
|
|
|
|
addstr("Do you want to quit ? ");
|
|
|
|
ch = getanswer("YN", FALSE);
|
|
|
|
if (ch == 'Y') {
|
|
|
|
Player.p_status = S_HUNGUP;
|
|
|
|
writerecord(&Player, loc);
|
|
|
|
cleanup(TRUE);
|
|
|
|
/* NOTREACHED */
|
|
|
|
}
|
|
|
|
death("Stupidity");
|
|
|
|
/* NOTREACHED */
|
|
|
|
}
|
|
|
|
return (loc);
|
|
|
|
} else
|
|
|
|
mvaddstr(12, 0, "No good.\n");
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
Echo = TRUE;
|
|
|
|
} else
|
|
|
|
mvaddstr(11, 0, "Not found.\n");
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
more(13);
|
|
|
|
return (-1L);
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
|
2009-08-12 12:21:41 +04:00
|
|
|
static void
|
2009-05-26 03:08:45 +04:00
|
|
|
neatstuff(void)
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
double temp; /* for temporary calculations */
|
|
|
|
int ch; /* input */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
switch ((int) ROLL(0.0, 100.0)) {
|
1994-10-22 00:19:39 +03:00
|
|
|
case 1:
|
|
|
|
case 2:
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Player.p_poison > 0.0) {
|
|
|
|
mvaddstr(4, 0, "You've found a medic! How much will you offer to be cured ? ");
|
|
|
|
temp = floor(infloat());
|
|
|
|
if (temp < 0.0 || temp > Player.p_gold)
|
|
|
|
/* negative gold, or more than available */
|
|
|
|
{
|
|
|
|
mvaddstr(6, 0, "He was not amused, and made you worse.\n");
|
|
|
|
Player.p_poison += 1.0;
|
|
|
|
} else
|
|
|
|
if (drandom() / 2.0 > (temp + 1.0) / MAX(Player.p_gold, 1))
|
|
|
|
/* medic wants 1/2 of available gold */
|
|
|
|
mvaddstr(5, 0, "Sorry, he wasn't interested.\n");
|
|
|
|
else {
|
|
|
|
mvaddstr(5, 0, "He accepted.");
|
|
|
|
Player.p_poison = MAX(0.0, Player.p_poison - 1.0);
|
|
|
|
Player.p_gold -= temp;
|
|
|
|
}
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case 3:
|
1997-10-13 06:18:06 +04:00
|
|
|
mvaddstr(4, 0, "You've been caught raping and pillaging!\n");
|
|
|
|
Player.p_experience += 4000.0;
|
|
|
|
Player.p_sin += 0.5;
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case 4:
|
1997-10-13 06:18:06 +04:00
|
|
|
temp = ROLL(10.0, 75.0);
|
|
|
|
mvprintw(4, 0, "You've found %.0f gold pieces, want them ? ", temp);
|
|
|
|
ch = getanswer("NY", FALSE);
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
if (ch == 'Y')
|
|
|
|
collecttaxes(temp, 0.0);
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case 5:
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Player.p_sin > 1.0) {
|
|
|
|
mvaddstr(4, 0, "You've found a Holy Orb!\n");
|
|
|
|
Player.p_sin -= 0.25;
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case 6:
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Player.p_poison < 1.0) {
|
|
|
|
mvaddstr(4, 0, "You've been hit with a plague!\n");
|
|
|
|
Player.p_poison += 1.0;
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case 7:
|
1997-10-13 06:18:06 +04:00
|
|
|
mvaddstr(4, 0, "You've found some holy water.\n");
|
|
|
|
++Player.p_holywater;
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case 8:
|
1997-10-13 06:18:06 +04:00
|
|
|
mvaddstr(4, 0, "You've met a Guru. . .");
|
|
|
|
if (drandom() * Player.p_sin > 1.0)
|
|
|
|
addstr("You disgusted him with your sins!\n");
|
|
|
|
else
|
|
|
|
if (Player.p_poison > 0.0) {
|
|
|
|
addstr("He looked kindly upon you, and cured you.\n");
|
|
|
|
Player.p_poison = 0.0;
|
|
|
|
} else {
|
|
|
|
addstr("He rewarded you for your virtue.\n");
|
|
|
|
Player.p_mana += 50.0;
|
|
|
|
Player.p_shield += 2.0;
|
|
|
|
}
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case 9:
|
1997-10-13 06:18:06 +04:00
|
|
|
mvaddstr(4, 0, "You've found an amulet.\n");
|
|
|
|
++Player.p_amulets;
|
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
case 10:
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Player.p_blindness) {
|
|
|
|
mvaddstr(4, 0, "You've regained your sight!\n");
|
|
|
|
Player.p_blindness = FALSE;
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
default: /* deal with poison */
|
|
|
|
if (Player.p_poison > 0.0) {
|
|
|
|
temp = Player.p_poison * Statptr->c_weakness
|
|
|
|
* Player.p_maxenergy / 600.0;
|
|
|
|
if (Player.p_energy > Player.p_maxenergy / 10.0
|
|
|
|
&& temp + 5.0 < Player.p_energy)
|
|
|
|
Player.p_energy -= temp;
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
break;
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-12 12:21:41 +04:00
|
|
|
static void
|
2009-05-26 03:08:45 +04:00
|
|
|
genchar(int type)
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
int subscript; /* used for subscripting into Stattable */
|
Add use of `const' where appropriate to the games.
This merges in all such remaining changes from the Linux port of the
NetBSD games, except in hunt (where substantial changes from OpenBSD
need to be looked at).
Some such changes were previously covered in PRs bin/6041, bin/6146,
bin/6148, bin/6150, bin/6151, bin/6580, bin/6660, bin/7993, bin/7994,
bin/8039, bin/8057 and bin/8093.
1999-09-09 01:17:44 +04:00
|
|
|
const struct charstats *statptr; /* for pointing into Stattable */
|
1997-10-13 06:18:06 +04:00
|
|
|
|
|
|
|
subscript = type - '1';
|
|
|
|
|
|
|
|
if (subscript < C_MAGIC || subscript > C_EXPER)
|
|
|
|
if (subscript != C_SUPER || !Wizard)
|
|
|
|
/* fighter is default */
|
|
|
|
subscript = C_FIGHTER;
|
|
|
|
|
|
|
|
statptr = &Stattable[subscript];
|
|
|
|
|
|
|
|
Player.p_quickness =
|
|
|
|
ROLL(statptr->c_quickness.base, statptr->c_quickness.interval);
|
|
|
|
Player.p_strength =
|
|
|
|
ROLL(statptr->c_strength.base, statptr->c_strength.interval);
|
|
|
|
Player.p_mana =
|
|
|
|
ROLL(statptr->c_mana.base, statptr->c_mana.interval);
|
|
|
|
Player.p_maxenergy =
|
|
|
|
Player.p_energy =
|
|
|
|
ROLL(statptr->c_energy.base, statptr->c_energy.interval);
|
|
|
|
Player.p_brains =
|
|
|
|
ROLL(statptr->c_brains.base, statptr->c_brains.interval);
|
|
|
|
Player.p_magiclvl =
|
|
|
|
ROLL(statptr->c_magiclvl.base, statptr->c_magiclvl.interval);
|
|
|
|
|
|
|
|
Player.p_type = subscript;
|
|
|
|
|
|
|
|
if (Player.p_type == C_HALFLING)
|
|
|
|
/* give halfling some experience */
|
|
|
|
Player.p_experience = ROLL(600.0, 200.0);
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
|
2009-08-12 12:21:41 +04:00
|
|
|
static void
|
2009-05-26 03:08:45 +04:00
|
|
|
playinit(void)
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
/* catch/ingnore signals */
|
1994-10-22 00:19:39 +03:00
|
|
|
|
|
|
|
#ifdef BSD41
|
1997-10-13 06:18:06 +04:00
|
|
|
sigignore(SIGQUIT);
|
|
|
|
sigignore(SIGALRM);
|
|
|
|
sigignore(SIGTERM);
|
|
|
|
sigignore(SIGTSTP);
|
|
|
|
sigignore(SIGTTIN);
|
|
|
|
sigignore(SIGTTOU);
|
|
|
|
sighold(SIGINT);
|
|
|
|
sigset(SIGHUP, ill_sig);
|
|
|
|
sigset(SIGTRAP, ill_sig);
|
|
|
|
sigset(SIGIOT, ill_sig);
|
|
|
|
sigset(SIGEMT, ill_sig);
|
|
|
|
sigset(SIGFPE, ill_sig);
|
|
|
|
sigset(SIGBUS, ill_sig);
|
|
|
|
sigset(SIGSEGV, ill_sig);
|
|
|
|
sigset(SIGSYS, ill_sig);
|
|
|
|
sigset(SIGPIPE, ill_sig);
|
1994-10-22 00:19:39 +03:00
|
|
|
#endif
|
|
|
|
#ifdef BSD42
|
1997-10-13 06:18:06 +04:00
|
|
|
signal(SIGQUIT, ill_sig);
|
|
|
|
signal(SIGALRM, SIG_IGN);
|
|
|
|
signal(SIGTERM, SIG_IGN);
|
|
|
|
signal(SIGTSTP, SIG_IGN);
|
|
|
|
signal(SIGTTIN, SIG_IGN);
|
|
|
|
signal(SIGTTOU, SIG_IGN);
|
|
|
|
signal(SIGINT, ill_sig);
|
|
|
|
signal(SIGHUP, SIG_DFL);
|
|
|
|
signal(SIGTRAP, ill_sig);
|
|
|
|
signal(SIGIOT, ill_sig);
|
|
|
|
signal(SIGEMT, ill_sig);
|
|
|
|
signal(SIGFPE, ill_sig);
|
|
|
|
signal(SIGBUS, ill_sig);
|
|
|
|
signal(SIGSEGV, ill_sig);
|
|
|
|
signal(SIGSYS, ill_sig);
|
|
|
|
signal(SIGPIPE, ill_sig);
|
1994-10-22 00:19:39 +03:00
|
|
|
#endif
|
|
|
|
#ifdef SYS3
|
1997-10-13 06:18:06 +04:00
|
|
|
signal(SIGINT, SIG_IGN);
|
|
|
|
signal(SIGQUIT, SIG_IGN);
|
|
|
|
signal(SIGTERM, SIG_IGN);
|
|
|
|
signal(SIGALRM, SIG_IGN);
|
|
|
|
signal(SIGHUP, ill_sig);
|
|
|
|
signal(SIGTRAP, ill_sig);
|
|
|
|
signal(SIGIOT, ill_sig);
|
|
|
|
signal(SIGEMT, ill_sig);
|
|
|
|
signal(SIGFPE, ill_sig);
|
|
|
|
signal(SIGBUS, ill_sig);
|
|
|
|
signal(SIGSEGV, ill_sig);
|
|
|
|
signal(SIGSYS, ill_sig);
|
|
|
|
signal(SIGPIPE, ill_sig);
|
1994-10-22 00:19:39 +03:00
|
|
|
#endif
|
|
|
|
#ifdef SYS5
|
1997-10-13 06:18:06 +04:00
|
|
|
signal(SIGINT, SIG_IGN);
|
|
|
|
signal(SIGQUIT, SIG_IGN);
|
|
|
|
signal(SIGTERM, SIG_IGN);
|
|
|
|
signal(SIGALRM, SIG_IGN);
|
|
|
|
signal(SIGHUP, ill_sig);
|
|
|
|
signal(SIGTRAP, ill_sig);
|
|
|
|
signal(SIGIOT, ill_sig);
|
|
|
|
signal(SIGEMT, ill_sig);
|
|
|
|
signal(SIGFPE, ill_sig);
|
|
|
|
signal(SIGBUS, ill_sig);
|
|
|
|
signal(SIGSEGV, ill_sig);
|
|
|
|
signal(SIGSYS, ill_sig);
|
|
|
|
signal(SIGPIPE, ill_sig);
|
1994-10-22 00:19:39 +03:00
|
|
|
#endif
|
|
|
|
|
2008-08-08 20:10:47 +04:00
|
|
|
if (!initscr()) { /* turn on curses */
|
|
|
|
fprintf(stderr, "couldn't initialize screen\n");
|
|
|
|
exit (0);
|
|
|
|
}
|
1997-10-13 06:18:06 +04:00
|
|
|
noecho(); /* do not echo input */
|
2001-12-06 15:15:37 +03:00
|
|
|
cbreak(); /* do not process erase, kill */
|
1997-10-13 06:18:06 +04:00
|
|
|
clear();
|
|
|
|
refresh();
|
|
|
|
Windows = TRUE; /* mark the state */
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
|
|
|
|
1997-10-13 06:18:06 +04:00
|
|
|
void
|
2009-05-26 03:08:45 +04:00
|
|
|
cleanup(int doexit)
|
1994-10-22 00:19:39 +03:00
|
|
|
{
|
1997-10-13 06:18:06 +04:00
|
|
|
if (Windows) {
|
|
|
|
move(LINES - 2, 0);
|
|
|
|
refresh();
|
2001-12-06 15:15:37 +03:00
|
|
|
nocbreak();
|
1997-10-13 06:18:06 +04:00
|
|
|
endwin();
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|
2004-01-01 19:03:46 +03:00
|
|
|
if (Playersfp)
|
|
|
|
fclose(Playersfp);
|
|
|
|
if (Monstfp)
|
|
|
|
fclose(Monstfp);
|
|
|
|
if (Messagefp)
|
|
|
|
fclose(Messagefp);
|
|
|
|
if (Energyvoidfp)
|
|
|
|
fclose(Energyvoidfp);
|
1997-10-13 06:18:06 +04:00
|
|
|
|
|
|
|
if (doexit)
|
|
|
|
exit(0);
|
|
|
|
/* NOTREACHED */
|
1994-10-22 00:19:39 +03:00
|
|
|
}
|