]>
git.cameronkatri.com Git - bsdgames-darwin.git/blob - larn/monster.c
2 static char rcsid
[] = "$Id: monster.c,v 1.2 1993/08/02 17:20:10 mycroft Exp $";
6 * monster.c Larn is copyrighted 1986 by Noah Morgan.
8 * This file contains the following functions:
9 * ----------------------------------------------------------------------------
11 * createmonster(monstno) Function to create a monster next to the player
14 * int cgood(x,y,itm,monst) Function to check location for emptiness
17 * createitem(it,arg) Routine to place an item next to the player
20 * cast() Subroutine called by parse to cast a spell for the user
22 * speldamage(x) Function to perform spell functions cast by the player
25 * loseint() Routine to decrement your int (intelligence) if > 3
27 * isconfuse() Routine to check to see if player is confused
29 * nospell(x,monst) Routine to return 1 if a spell doesn't affect a monster
32 * fullhit(xx) Function to return full damage against a monst (aka web)
35 * direct(spnum,dam,str,arg) Routine to direct spell damage 1 square in 1 dir
39 * godirect(spnum,dam,str,delay,cshow) Function to perform missile attacks
40 * int spnum,dam,delay;
43 * ifblind(x,y) Routine to put "monster" or the monster name into lastmosnt
46 * tdirect(spnum) Routine to teleport away a monster
49 * omnidirect(sp,dam,str) Routine to damage all monsters 1 square from player
53 * dirsub(x,y) Routine to ask for direction, then modify x,y for it
56 * vxy(x,y) Routine to verify/fix (*x,*y) for being within bounds
59 * dirpoly(spnum) Routine to ask for a direction and polymorph a monst
62 * hitmonster(x,y) Function to hit a monster at the designated coordinates
65 * hitm(x,y,amt) Function to just hit a monster at a given coordinates
68 * hitplayer(x,y) Function for the monster to hit the player from (x,y)
71 * dropsomething(monst) Function to create an object when a monster dies
74 * dropgold(amount) Function to drop some gold around player
77 * something(level) Function to create a random item around player
80 * newobject(lev,i) Routine to return a randomly selected new object
83 * spattack(atckno,xx,yy) Function to process special attacks from monsters
86 * checkloss(x) Routine to subtract hp from user and flag bottomline display
89 * annihilate() Routine to annihilate monsters around player, playerx,playery
91 * newsphere(x,y,dir,lifetime) Function to create a new sphere of annihilation
92 * int x,y,dir,lifetime;
94 * rmsphere(x,y) Function to delete a sphere of annihilation from list
97 * sphboom(x,y) Function to perform the effects of a sphere detonation
100 * genmonst() Function to ask for monster and genocide from game
105 struct isave
/* used for altar reality */
107 char type
; /* 0=item, 1=monster */
108 char id
; /* item number or monster number */
109 short arg
; /* the type of item or hitpoints of monster */
113 * createmonster(monstno) Function to create a monster next to the player
116 * Enter with the monster number (1 to MAXMONST+8)
122 register int x
,y
,k
,i
;
123 if (mon
<1 || mon
>MAXMONST
+8) /* check for monster number out of bounds */
125 beep(); lprintf("\ncan't createmonst(%d)\n",(long)mon
); nap(3000); return;
127 while (monster
[mon
].genocided
&& mon
<MAXMONST
) mon
++; /* genocided? */
128 for (k
=rnd(8), i
= -8; i
<0; i
++,k
++) /* choose direction, then try all */
130 if (k
>8) k
=1; /* wraparound the diroff arrays */
131 x
= playerx
+ diroffx
[k
]; y
= playery
+ diroffy
[k
];
132 if (cgood(x
,y
,0,1)) /* if we can create here */
135 hitp
[x
][y
] = monster
[mon
].hitpoints
;
136 stealth
[x
][y
]=know
[x
][y
]=0;
139 case ROTHE
: case POLTERGEIST
: case VAMPIRE
: stealth
[x
][y
]=1;
147 * int cgood(x,y,itm,monst) Function to check location for emptiness
150 * Routine to return TRUE if a location does not have itm or monst there
151 * returns FALSE (0) otherwise
152 * Enter with itm or monst TRUE or FALSE if checking it
153 * Example: if itm==TRUE check for no item at this location
154 * if monst==TRUE check for no monster at this location
155 * This routine will return FALSE if at a wall or the dungeon exit on level 1
157 int cgood(x
,y
,itm
,monst
)
161 if ((y
>=0) && (y
<=MAXY
-1) && (x
>=0) && (x
<=MAXX
-1)) /* within bounds? */
162 if (item
[x
][y
]!=OWALL
) /* can't make anything on walls */
163 if (itm
==0 || (item
[x
][y
]==0)) /* is it free of items? */
164 if (monst
==0 || (mitem
[x
][y
]==0)) /* is it free of monsters? */
165 if ((level
!=1) || (x
!=33) || (y
!=MAXY
-1)) /* not exit to level 1 */
171 * createitem(it,arg) Routine to place an item next to the player
174 * Enter with the item number and its argument (iven[], ivenarg[])
175 * Returns no value, thus we don't know about createitem() failures.
180 register int x
,y
,k
,i
;
181 if (it
>= MAXOBJ
) return; /* no such object */
182 for (k
=rnd(8), i
= -8; i
<0; i
++,k
++) /* choose direction, then try all */
184 if (k
>8) k
=1; /* wraparound the diroff arrays */
185 x
= playerx
+ diroffx
[k
]; y
= playery
+ diroffy
[k
];
186 if (cgood(x
,y
,1,0)) /* if we can create here */
188 item
[x
][y
] = it
; know
[x
][y
]=0; iarg
[x
][y
]=arg
; return;
194 * cast() Subroutine called by parse to cast a spell for the user
196 * No arguments and no return value.
198 static char eys
[] = "\nEnter your spell: ";
201 register int i
,j
,a
,b
,d
;
203 if (c
[SPELLS
]<=0) { lprcat("\nYou don't have any spells!"); return; }
204 lprcat(eys
); --c
[SPELLS
];
205 while ((a
=getchar())=='D')
206 { seemagic(-1); cursors(); lprcat(eys
); }
207 if (a
=='\33') goto over
; /* to escape casting a spell */
208 if ((b
=getchar())=='\33') goto over
; /* to escape casting a spell */
209 if ((d
=getchar())=='\33')
210 { over
: lprcat(aborted
); c
[SPELLS
]++; return; } /* to escape casting a spell */
214 for (lprc('\n'),j
= -1,i
=0; i
<SPNUM
; i
++) /*seq search for his spell, hash?*/
215 if ((spelcode
[i
][0]==a
) && (spelcode
[i
][1]==b
) && (spelcode
[i
][2]==d
))
217 { speldamage(i
); j
= 1; i
=SPNUM
; }
219 if (j
== -1) lprcat(" Nothing Happened ");
226 * speldamage(x) Function to perform spell functions cast by the player
229 * Enter with the spell number, returns no value.
230 * Please insure that there are 2 spaces before all messages here
235 register int i
,j
,clev
;
237 register char *p
,*kn
,*pm
;
238 if (x
>=SPNUM
) return; /* no such spell */
239 if (c
[TIMESTOP
]) { lprcat(" It didn't seem to work"); return; } /* not if time stopped */
241 if ((rnd(23)==7) || (rnd(18) > c
[INTELLIGENCE
]))
242 { lprcat(" It didn't work!"); return; }
243 if (clev
*3+2 < x
) { lprcat(" Nothing happens. You seem inexperienced at this"); return; }
247 /* ----- LEVEL 1 SPELLS ----- */
249 case 0: if (c
[PROTECTIONTIME
]==0) c
[MOREDEFENSES
]+=2; /* protection field +2 */
250 c
[PROTECTIONTIME
] += 250; return;
252 case 1: i
= rnd(((clev
+1)<<1)) + clev
+ 3;
253 godirect(x
,i
,(clev
>=2)?" Your missiles hit the %s":" Your missile hit the %s",100,'+'); /* magic missile */
257 case 2: if (c
[DEXCOUNT
]==0) c
[DEXTERITY
]+=3; /* dexterity */
258 c
[DEXCOUNT
] += 400; return;
261 p
=" While the %s slept, you smashed it %d times";
262 ws
: direct(x
,fullhit(i
),p
,i
); /* sleep */ return;
264 case 4: /* charm monster */ c
[CHARMCOUNT
] += c
[CHARISMA
]<<1; return;
266 case 5: godirect(x
,rnd(10)+15+clev
," The sound damages the %s",70,'@'); /* sonic spear */
269 /* ----- LEVEL 2 SPELLS ----- */
271 case 6: i
=rnd(3)+2; p
=" While the %s is entangled, you hit %d times";
274 case 7: if (c
[STRCOUNT
]==0) c
[STREXTRA
]+=3; /* strength */
275 c
[STRCOUNT
] += 150+rnd(100); return;
277 case 8: yl
= playery
-5; /* enlightenment */
278 yh
= playery
+6; xl
= playerx
-15; xh
= playerx
+16;
279 vxy(&xl
,&yl
); vxy(&xh
,&yh
); /* check bounds */
280 for (i
=yl
; i
<=yh
; i
++) /* enlightenment */
281 for (j
=xl
; j
<=xh
; j
++) know
[j
][i
]=1;
282 draws(xl
,xh
+1,yl
,yh
+1); return;
284 case 9: raisehp(20+(clev
<<1)); return; /* healing */
286 case 10: c
[BLINDCOUNT
]=0; return; /* cure blindness */
288 case 11: createmonster(makemonst(level
+1)+8); return;
290 case 12: if (rnd(11)+7 <= c
[WISDOM
]) direct(x
,rnd(20)+20+clev
," The %s believed!",0);
291 else lprcat(" It didn't believe the illusions!");
294 case 13: /* if he has the amulet of invisibility then add more time */
295 for (j
=i
=0; i
<26; i
++)
296 if (iven
[i
]==OAMULET
) j
+= 1+ivenarg
[i
];
297 c
[INVISIBILITY
] += (j
<<7)+12; return;
299 /* ----- LEVEL 3 SPELLS ----- */
301 case 14: godirect(x
,rnd(25+clev
)+25+clev
," The fireball hits the %s",40,'*'); return; /* fireball */
303 case 15: godirect(x
,rnd(25)+20+clev
," Your cone of cold strikes the %s",60,'O'); /* cold */
306 case 16: dirpoly(x
); return; /* polymorph */
308 case 17: c
[CANCELLATION
]+= 5+clev
; return; /* cancellation */
310 case 18: c
[HASTESELF
]+= 7+clev
; return; /* haste self */
312 case 19: omnidirect(x
,30+rnd(10)," The %s gasps for air"); /* cloud kill */
315 case 20: xh
= min(playerx
+1,MAXX
-2); yh
= min(playery
+1,MAXY
-2);
316 for (i
=max(playerx
-1,1); i
<=xh
; i
++) /* vaporize rock */
317 for (j
=max(playery
-1,1); j
<=yh
; j
++)
319 kn
= &know
[i
][j
]; pm
= &mitem
[i
][j
];
320 switch(*(p
= &item
[i
][j
]))
322 case OWALL
: if (level
< MAXLEVEL
+MAXVLEVEL
-1)
326 case OSTATUE
: if (c
[HARDGAME
]<3)
328 *p
=OBOOK
; iarg
[i
][j
]=level
; *kn
=0;
332 case OTHRONE
: *pm
=GNOMEKING
; *kn
=0; *p
= OTHRONE2
;
333 hitp
[i
][j
]=monster
[GNOMEKING
].hitpoints
; break;
335 case OALTAR
: *pm
=DEMONPRINCE
; *kn
=0;
336 hitp
[i
][j
]=monster
[DEMONPRINCE
].hitpoints
; break;
340 case XORN
: ifblind(i
,j
); hitm(i
,j
,200); break; /* Xorn takes damage from vpr */
345 /* ----- LEVEL 4 SPELLS ----- */
347 case 21: direct(x
,100+clev
," The %s shrivels up",0); /* dehydration */
350 case 22: godirect(x
,rnd(25)+20+(clev
<<1)," A lightning bolt hits the %s",1,'~'); /* lightning */
353 case 23: i
=min(c
[HP
]-1,c
[HPMAX
]/2); /* drain life */
354 direct(x
,i
+i
,"",0); c
[HP
] -= i
; return;
356 case 24: if (c
[GLOBE
]==0) c
[MOREDEFENSES
] += 10;
357 c
[GLOBE
] += 200; loseint(); /* globe of invulnerability */
360 case 25: omnidirect(x
,32+clev
," The %s struggles for air in your flood!"); /* flood */
363 case 26: if (rnd(151)==63) { beep(); lprcat("\nYour heart stopped!\n"); nap(4000); died(270); return; }
364 if (c
[WISDOM
]>rnd(10)+10) direct(x
,2000," The %s's heart stopped",0); /* finger of death */
365 else lprcat(" It didn't work"); return;
367 /* ----- LEVEL 5 SPELLS ----- */
369 case 27: c
[SCAREMONST
] += rnd(10)+clev
; return; /* scare monster */
371 case 28: c
[HOLDMONST
] += rnd(10)+clev
; return; /* hold monster */
373 case 29: c
[TIMESTOP
] += rnd(20)+(clev
<<1); return; /* time stop */
375 case 30: tdirect(x
); return; /* teleport away */
377 case 31: omnidirect(x
,35+rnd(10)+clev
," The %s cringes from the flame"); /* magic fire */
380 /* ----- LEVEL 6 SPELLS ----- */
382 case 32: if ((rnd(23)==5) && (wizard
==0)) /* sphere of annihilation */
384 beep(); lprcat("\nYou have been enveloped by the zone of nothingness!\n");
385 nap(4000); died(258); return;
387 xl
=playerx
; yl
=playery
;
389 i
=dirsub(&xl
,&yl
); /* get direction of sphere */
390 newsphere(xl
,yl
,i
,rnd(20)+11); /* make a sphere */
393 case 33: genmonst(); spelknow
[33]=0; /* genocide */
397 case 34: /* summon demon */
398 if (rnd(100) > 30) { direct(x
,150," The demon strikes at the %s",0); return; }
399 if (rnd(100) > 15) { lprcat(" Nothing seems to have happened"); return; }
400 lprcat(" The demon turned on you and vanished!"); beep();
401 i
=rnd(40)+30; lastnum
=277;
402 losehp(i
); /* must say killed by a demon */ return;
404 case 35: /* walk through walls */
405 c
[WTW
] += rnd(10)+5; return;
407 case 36: /* alter reality */
409 struct isave
*save
; /* pointer to item save structure */
410 int sc
; sc
=0; /* # items saved */
411 save
= (struct isave
*)malloc(sizeof(struct isave
)*MAXX
*MAXY
*2);
412 for (j
=0; j
<MAXY
; j
++)
413 for (i
=0; i
<MAXX
; i
++) /* save all items and monsters */
416 if (xl
&& xl
!=OWALL
&& xl
!=OANNIHILATION
)
418 save
[sc
].type
=0; save
[sc
].id
=item
[i
][j
];
419 save
[sc
++].arg
=iarg
[i
][j
];
423 save
[sc
].type
=1; save
[sc
].id
=mitem
[i
][j
];
424 save
[sc
++].arg
=hitp
[i
][j
];
426 item
[i
][j
]=OWALL
; mitem
[i
][j
]=0;
427 if (wizard
) know
[i
][j
]=1; else know
[i
][j
]=0;
429 eat(1,1); if (level
==1) item
[33][MAXY
-1]=0;
430 for (j
=rnd(MAXY
-2), i
=1; i
<MAXX
-1; i
++) item
[i
][j
]=0;
431 while (sc
>0) /* put objects back in level */
434 if (save
[sc
].type
== 0)
437 for (trys
=100, i
=j
=1; --trys
>0 && item
[i
][j
]; i
=rnd(MAXX
-1), j
=rnd(MAXY
-1));
438 if (trys
) { item
[i
][j
]=save
[sc
].id
; iarg
[i
][j
]=save
[sc
].arg
; }
441 { /* put monsters back in */
443 for (trys
=100, i
=j
=1; --trys
>0 && (item
[i
][j
]==OWALL
|| mitem
[i
][j
]); i
=rnd(MAXX
-1), j
=rnd(MAXY
-1));
444 if (trys
) { mitem
[i
][j
]=save
[sc
].id
; hitp
[i
][j
]=save
[sc
].arg
; }
448 draws(0,MAXX
,0,MAXY
); if (wizard
==0) spelknow
[36]=0;
449 free((char*)save
); positionplayer(); return;
452 case 37: /* permanence */ adjtime(-99999L); spelknow
[37]=0; /* forget */
456 default: lprintf(" spell %d not available!",(long)x
); beep(); return;
461 * loseint() Routine to subtract 1 from your int (intelligence) if > 3
463 * No arguments and no return value
467 if (--c
[INTELLIGENCE
]<3) c
[INTELLIGENCE
]=3;
471 * isconfuse() Routine to check to see if player is confused
473 * This routine prints out a message saying "You can't aim your magic!"
474 * returns 0 if not confused, non-zero (time remaining confused) if confused
478 if (c
[CONFUSE
]) { lprcat(" You can't aim your magic!"); beep(); }
483 * nospell(x,monst) Routine to return 1 if a spell doesn't affect a monster
486 * Subroutine to return 1 if the spell can't affect the monster
487 * otherwise returns 0
488 * Enter with the spell number in x, and the monster number in monst.
494 if (x
>=SPNUM
|| monst
>=MAXMONST
+8 || monst
<0 || x
<0) return(0); /* bad spell or monst */
495 if ((tmp
=spelweird
[monst
-1][x
])==0) return(0);
496 cursors(); lprc('\n'); lprintf(spelmes
[tmp
],monster
[monst
].name
); return(1);
500 * fullhit(xx) Function to return full damage against a monster (aka web)
503 * Function to return hp damage to monster due to a number of full hits
504 * Enter with the number of full hits being done
510 if (xx
<0 || xx
>20) return(0); /* fullhits are out of range */
511 if (c
[LANCEDEATH
]) return(10000); /* lance of death */
512 i
= xx
* ((c
[WCLASS
]>>1)+c
[STRENGTH
]+c
[STREXTRA
]-c
[HARDGAME
]-12+c
[MOREDAM
]);
513 return( (i
>=1) ? i
: xx
);
517 * direct(spnum,dam,str,arg) Routine to direct spell damage 1 square in 1 dir
521 * Routine to ask for a direction to a spell and then hit the monster
522 * Enter with the spell number in spnum, the damage to be done in dam,
523 * lprintf format string in str, and lprintf's argument in arg.
526 direct(spnum
,dam
,str
,arg
)
532 if (spnum
<0 || spnum
>=SPNUM
|| str
==0) return; /* bad arguments */
533 if (isconfuse()) return;
536 if (item
[x
][y
]==OMIRROR
)
538 if (spnum
==3) /* sleep */
540 lprcat("You fall asleep! "); beep();
543 while (arg
-- > 0) { parse2(); nap(1000); }
546 else if (spnum
==6) /* web */
548 lprcat("You get stuck in your own web! "); beep();
554 lprintf(str
,"spell caster (thats you)",(long)arg
);
555 beep(); losehp(dam
); return;
559 { lprcat(" There wasn't anything there!"); return; }
561 if (nospell(spnum
,m
)) { lasthx
=x
; lasthy
=y
; return; }
562 lprintf(str
,lastmonst
,(long)arg
); hitm(x
,y
,dam
);
566 * godirect(spnum,dam,str,delay,cshow) Function to perform missile attacks
567 * int spnum,dam,delay;
570 * Function to hit in a direction from a missile weapon and have it keep
571 * on going in that direction until its power is exhausted
572 * Enter with the spell number in spnum, the power of the weapon in hp,
573 * lprintf format string in str, the # of milliseconds to delay between
574 * locations in delay, and the character to represent the weapon in cshow.
577 godirect(spnum
,dam
,str
,delay
,cshow
)
584 if (spnum
<0 || spnum
>=SPNUM
|| str
==0 || delay
<0) return; /* bad args */
585 if (isconfuse()) return;
586 dirsub(&dx
,&dy
); x
=dx
; y
=dy
;
587 dx
= x
-playerx
; dy
= y
-playery
; x
= playerx
; y
= playery
;
591 if ((x
> MAXX
-1) || (y
> MAXY
-1) || (x
< 0) || (y
< 0))
593 dam
=0; break; /* out of bounds */
595 if ((x
==playerx
) && (y
==playery
)) /* if energy hits player */
597 cursors(); lprcat("\nYou are hit my your own magic!"); beep();
598 lastnum
=278; losehp(dam
); return;
600 if (c
[BLINDCOUNT
]==0) /* if not blind show effect */
602 cursor(x
+1,y
+1); lprc(cshow
); nap(delay
); show1cell(x
,y
);
604 if ((m
=mitem
[x
][y
])) /* is there a monster there? */
607 if (nospell(spnum
,m
)) { lasthx
=x
; lasthy
=y
; return; }
608 cursors(); lprc('\n');
609 lprintf(str
,lastmonst
); dam
-= hitm(x
,y
,dam
);
610 show1cell(x
,y
); nap(1000); x
-= dx
; y
-= dy
;
612 else switch (*(p
= &item
[x
][y
]))
614 case OWALL
: cursors(); lprc('\n'); lprintf(str
,"wall");
615 if (dam
>=50+c
[HARDGAME
]) /* enough damage? */
616 if (level
<MAXLEVEL
+MAXVLEVEL
-1) /* not on V3 */
617 if ((x
<MAXX
-1) && (y
<MAXY
-1) && (x
) && (y
))
619 lprcat(" The wall crumbles");
624 god2
: dam
= 0; break;
626 case OCLOSEDDOOR
: cursors(); lprc('\n'); lprintf(str
,"door");
629 lprcat(" The door is blasted apart");
634 case OSTATUE
: cursors(); lprc('\n'); lprintf(str
,"statue");
638 lprcat(" The statue crumbles");
639 *p
=OBOOK
; iarg
[x
][y
]=level
;
644 case OTHRONE
: cursors(); lprc('\n'); lprintf(str
,"throne");
647 mitem
[x
][y
]=GNOMEKING
; hitp
[x
][y
]=monster
[GNOMEKING
].hitpoints
;
653 case OMIRROR
: dx
*= -1; dy
*= -1; break;
655 dam
-= 3 + (c
[HARDGAME
]>>1);
660 * ifblind(x,y) Routine to put "monster" or the monster name into lastmosnt
663 * Subroutine to copy the word "monster" into lastmonst if the player is blind
664 * Enter with the coordinates (x,y) of the monster
671 vxy(&x
,&y
); /* verify correct x,y coordinates */
672 if (c
[BLINDCOUNT
]) { lastnum
=279; p
="monster"; }
673 else { lastnum
=mitem
[x
][y
]; p
=monster
[lastnum
].name
; }
678 * tdirect(spnum) Routine to teleport away a monster
681 * Routine to ask for a direction to a spell and then teleport away monster
682 * Enter with the spell number that wants to teleport away
690 if (spnum
<0 || spnum
>=SPNUM
) return; /* bad args */
691 if (isconfuse()) return;
693 if ((m
=mitem
[x
][y
])==0)
694 { lprcat(" There wasn't anything there!"); return; }
696 if (nospell(spnum
,m
)) { lasthx
=x
; lasthy
=y
; return; }
697 fillmonst(m
); mitem
[x
][y
]=know
[x
][y
]=0;
701 * omnidirect(sp,dam,str) Routine to damage all monsters 1 square from player
705 * Routine to cast a spell and then hit the monster in all directions
706 * Enter with the spell number in sp, the damage done to wach square in dam,
707 * and the lprintf string to identify the spell in str.
710 omnidirect(spnum
,dam
,str
)
715 if (spnum
<0 || spnum
>=SPNUM
|| str
==0) return; /* bad args */
716 for (x
=playerx
-1; x
<playerx
+2; x
++)
717 for (y
=playery
-1; y
<playery
+2; y
++)
720 if (nospell(spnum
,m
) == 0)
723 cursors(); lprc('\n'); lprintf(str
,lastmonst
);
724 hitm(x
,y
,dam
); nap(800);
726 else { lasthx
=x
; lasthy
=y
; }
731 * static dirsub(x,y) Routine to ask for direction, then modify x,y for it
734 * Function to ask for a direction and modify an x,y for that direction
735 * Enter with the origination coordinates in (x,y).
736 * Returns index into diroffx[] (0-8).
743 lprcat("\nIn What Direction? ");
754 case 'j': i
++; goto out
;
757 *x
= playerx
+diroffx
[i
]; *y
= playery
+diroffy
[i
];
762 * vxy(x,y) Routine to verify/fix coordinates for being within bounds
765 * Function to verify x & y are within the bounds for a level
766 * If *x or *y is not within the absolute bounds for a level, fix them so that
767 * they are on the level.
768 * Returns TRUE if it was out of bounds, and the *x & *y in the calling
769 * routine are affected.
775 if (*x
<0) { *x
=0; flag
++; }
776 if (*y
<0) { *y
=0; flag
++; }
777 if (*x
>=MAXX
) { *x
=MAXX
-1; flag
++; }
778 if (*y
>=MAXY
) { *y
=MAXY
-1; flag
++; }
783 * dirpoly(spnum) Routine to ask for a direction and polymorph a monst
786 * Subroutine to polymorph a monster and ask for the direction its in
787 * Enter with the spell number in spmun.
794 if (spnum
<0 || spnum
>=SPNUM
) return; /* bad args */
795 if (isconfuse()) return; /* if he is confused, he can't aim his magic */
798 { lprcat(" There wasn't anything there!"); return; }
800 if (nospell(spnum
,mitem
[x
][y
])) { lasthx
=x
; lasthy
=y
; return; }
801 while ( monster
[m
= mitem
[x
][y
] = rnd(MAXMONST
+7)].genocided
);
802 hitp
[x
][y
] = monster
[m
].hitpoints
;
803 show1cell(x
,y
); /* show the new monster */
807 * hitmonster(x,y) Function to hit a monster at the designated coordinates
810 * This routine is used for a bash & slash type attack on a monster
811 * Enter with the coordinates of the monster in (x,y).
817 register int tmp
,monst
,damag
,flag
;
818 if (c
[TIMESTOP
]) return; /* not if time stopped */
819 vxy(&x
,&y
); /* verify coordinates are within range */
820 if ((monst
= mitem
[x
][y
]) == 0) return;
821 hit3flag
=1; ifblind(x
,y
);
822 tmp
= monster
[monst
].armorclass
+ c
[LEVEL
] + c
[DEXTERITY
] + c
[WCLASS
]/4 - 12;
824 if ((rnd(20) < tmp
-c
[HARDGAME
]) || (rnd(71) < 5)) /* need at least random chance to hit */
826 lprcat("\nYou hit"); flag
=1;
828 if (damag
<9999) damag
=rnd(damag
)+1;
832 lprcat("\nYou missed"); flag
=0;
834 lprcat(" the "); lprcat(lastmonst
);
835 if (flag
) /* if the monster was hit */
836 if ((monst
==RUSTMONSTER
) || (monst
==DISENCHANTRESS
) || (monst
==CUBE
))
838 if (ivenarg
[c
[WIELD
]] > -10)
840 lprintf("\nYour weapon is dulled by the %s",lastmonst
); beep();
843 if (flag
) hitm(x
,y
,damag
);
844 if (monst
== VAMPIRE
) if (hitp
[x
][y
]<25) { mitem
[x
][y
]=BAT
; know
[x
][y
]=0; }
848 * hitm(x,y,amt) Function to just hit a monster at a given coordinates
851 * Returns the number of hitpoints the monster absorbed
852 * This routine is used to specifically damage a monster at a location (x,y)
853 * Called by hitmonster(x,y)
861 vxy(&x
,&y
); /* verify coordinates are within range */
862 amt2
= amt
; /* save initial damage so we can return it */
864 if (c
[HALFDAM
]) amt
>>= 1; /* if half damage curse adjust damage points */
865 if (amt
<=0) amt2
= amt
= 1;
867 stealth
[x
][y
]=1; /* make sure hitting monst breaks stealth condition */
868 c
[HOLDMONST
]=0; /* hit a monster breaks hold monster spell */
869 switch(monst
) /* if a dragon and orb(s) of dragon slaying */
871 case WHITEDRAGON
: case REDDRAGON
: case GREENDRAGON
:
872 case BRONZEDRAGON
: case PLATINUMDRAGON
: case SILVERDRAGON
:
873 amt
*= 1+(c
[SLAYING
]<<1); break;
875 /* invincible monster fix is here */
876 if (hitp
[x
][y
] > monster
[monst
].hitpoints
)
877 hitp
[x
][y
] = monster
[monst
].hitpoints
;
878 if ((hpoints
= hitp
[x
][y
]) <= amt
)
883 lprintf("\nThe %s died!",lastmonst
);
884 raiseexperience((long)monster
[monst
].experience
);
885 amt
= monster
[monst
].gold
; if (amt
>0) dropgold(rnd(amt
)+amt
);
886 dropsomething(monst
); disappear(x
,y
); bottomline();
889 hitp
[x
][y
] = hpoints
-amt
; return(amt2
);
893 * hitplayer(x,y) Function for the monster to hit the player from (x,y)
896 * Function for the monster to hit the player with monster at location x,y
897 * Returns nothing of value.
902 register int dam
,tmp
,mster
,bias
;
903 vxy(&x
,&y
); /* verify coordinates are within range */
904 lastnum
= mster
= mitem
[x
][y
];
905 /* spirit naga's and poltergeist's do nothing if scarab of negate spirit */
906 if (c
[NEGATESPIRIT
] || c
[SPIRITPRO
]) if ((mster
==POLTERGEIST
) || (mster
==SPIRITNAGA
)) return;
907 /* if undead and cube of undead control */
908 if (c
[CUBEofUNDEAD
] || c
[UNDEADPRO
]) if ((mster
==VAMPIRE
) || (mster
==WRAITH
) || (mster
==ZOMBIE
)) return;
909 if ((know
[x
][y
]&1) == 0)
911 know
[x
][y
]=1; show1cell(x
,y
);
913 bias
= (c
[HARDGAME
]) + 1;
914 hitflag
= hit2flag
= hit3flag
= 1;
916 cursors(); ifblind(x
,y
);
917 if (c
[INVISIBILITY
]) if (rnd(33)<20)
919 lprintf("\nThe %s misses wildly",lastmonst
); return;
921 if (c
[CHARMCOUNT
]) if (rnd(30)+5*monster
[mster
].level
-c
[CHARISMA
]<30)
923 lprintf("\nThe %s is awestruck at your magnificence!",lastmonst
);
926 if (mster
==BAT
) dam
=1;
929 dam
= monster
[mster
].damage
;
930 dam
+= rnd((int)((dam
<1)?1:dam
)) + monster
[mster
].level
;
933 if (monster
[mster
].attack
>0)
934 if (((dam
+ bias
+ 8) > c
[AC
]) || (rnd((int)((c
[AC
]>0)?c
[AC
]:1))==1))
935 { if (spattack(monster
[mster
].attack
,x
,y
)) { flushall(); return; }
936 tmp
= 1; bias
-= 2; cursors(); }
937 if (((dam
+ bias
) > c
[AC
]) || (rnd((int)((c
[AC
]>0)?c
[AC
]:1))==1))
939 lprintf("\n The %s hit you ",lastmonst
); tmp
= 1;
940 if ((dam
-= c
[AC
]) < 0) dam
=0;
941 if (dam
> 0) { losehp(dam
); bottomhp(); flushall(); }
943 if (tmp
== 0) lprintf("\n The %s missed ",lastmonst
);
947 * dropsomething(monst) Function to create an object when a monster dies
950 * Function to create an object near the player when certain monsters are killed
951 * Enter with the monster number
952 * Returns nothing of value.
959 case ORC
: case NYMPH
: case ELF
: case TROGLODYTE
:
960 case TROLL
: case ROTHE
: case VIOLETFUNGI
:
961 case PLATINUMDRAGON
: case GNOMEKING
: case REDDRAGON
:
962 something(level
); return;
964 case LEPRECHAUN
: if (rnd(101)>=75) creategem();
965 if (rnd(5)==1) dropsomething(LEPRECHAUN
); return;
970 * dropgold(amount) Function to drop some gold around player
973 * Enter with the number of gold pieces to drop
974 * Returns nothing of value.
979 if (amount
> 250) createitem(OMAXGOLD
,amount
/100); else createitem(OGOLDPILE
,amount
);
983 * something(level) Function to create a random item around player
986 * Function to create an item from a designed probability around player
987 * Enter with the cave level on which something is to be dropped
988 * Returns nothing of value.
995 if (level
<0 || level
>MAXLEVEL
+MAXVLEVEL
) return; /* correct level? */
996 if (rnd(101)<8) something(level
); /* possibly more than one item */
997 j
= newobject(level
,&i
); createitem(j
,i
);
1001 * newobject(lev,i) Routine to return a randomly selected new object
1004 * Routine to return a randomly selected object to be created
1005 * Returns the object number created, and sets *i for its argument
1006 * Enter with the cave level and a pointer to the items arg
1008 static char nobjtab
[] = { 0, OSCROLL
, OSCROLL
, OSCROLL
, OSCROLL
, OPOTION
,
1009 OPOTION
, OPOTION
, OPOTION
, OGOLDPILE
, OGOLDPILE
, OGOLDPILE
, OGOLDPILE
,
1010 OBOOK
, OBOOK
, OBOOK
, OBOOK
, ODAGGER
, ODAGGER
, ODAGGER
, OLEATHER
, OLEATHER
,
1011 OLEATHER
, OREGENRING
, OPROTRING
, OENERGYRING
, ODEXRING
, OSTRRING
, OSPEAR
,
1012 OBELT
, ORING
, OSTUDLEATHER
, OSHIELD
, OFLAIL
, OCHAIN
, O2SWORD
, OPLATE
,
1016 register int lev
,*i
;
1018 register int tmp
=32,j
;
1019 if (level
<0 || level
>MAXLEVEL
+MAXVLEVEL
) return(0); /* correct level? */
1020 if (lev
>6) tmp
=37; else if (lev
>4) tmp
=35;
1021 j
= nobjtab
[tmp
=rnd(tmp
)]; /* the object type */
1024 case 1: case 2: case 3: case 4: *i
=newscroll(); break;
1025 case 5: case 6: case 7: case 8: *i
=newpotion(); break;
1026 case 9: case 10: case 11: case 12: *i
=rnd((lev
+1)*10)+lev
*10+10; break;
1027 case 13: case 14: case 15: case 16: *i
=lev
; break;
1028 case 17: case 18: case 19: if (!(*i
=newdagger())) return(0); break;
1029 case 20: case 21: case 22: if (!(*i
=newleather())) return(0); break;
1030 case 23: case 32: case 35: *i
=rund(lev
/3+1); break;
1031 case 24: case 26: *i
=rnd(lev
/4+1); break;
1032 case 25: *i
=rund(lev
/4+1); break;
1033 case 27: *i
=rnd(lev
/2+1); break;
1034 case 30: case 33: *i
=rund(lev
/2+1); break;
1035 case 28: *i
=rund(lev
/3+1); if (*i
==0) return(0); break;
1036 case 29: case 31: *i
=rund(lev
/2+1); if (*i
==0) return(0); break;
1037 case 34: *i
=newchain(); break;
1038 case 36: *i
=newplate(); break;
1039 case 37: *i
=newsword(); break;
1045 * spattack(atckno,xx,yy) Function to process special attacks from monsters
1048 * Enter with the special attack number, and the coordinates (xx,yy)
1049 * of the monster that is special attacking
1050 * Returns 1 if must do a show1cell(xx,yy) upon return, 0 otherwise
1052 * atckno monster effect
1053 * ---------------------------------------------------
1055 * 1 rust monster eat armor
1056 * 2 hell hound breathe light fire
1057 * 3 dragon breathe fire
1058 * 4 giant centipede weakening sing
1059 * 5 white dragon cold breath
1060 * 6 wraith drain level
1061 * 7 waterlord water gusher
1062 * 8 leprechaun steal gold
1063 * 9 disenchantress disenchant weapon or armor
1064 * 10 ice lizard hits with barbed tail
1065 * 11 umber hulk confusion
1066 * 12 spirit naga cast spells taken from special attacks
1067 * 13 platinum dragon psionics
1068 * 14 nymph steal objects
1072 * char rustarm[ARMORTYPES][2];
1073 * special array for maximum rust damage to armor from rustmonster
1074 * format is: { armor type , minimum attribute
1076 #define ARMORTYPES 6
1077 static char rustarm
[ARMORTYPES
][2] = { OSTUDLEATHER
,-2, ORING
,-4, OCHAIN
,-5,
1078 OSPLINT
,-6, OPLATE
,-8, OPLATEARMOR
,-9 };
1079 static char spsel
[] = { 1, 2, 3, 5, 6, 8, 9, 11, 13, 14 };
1083 register int i
,j
=0,k
,m
;
1085 if (c
[CANCELLATION
]) return(0);
1086 vxy(&xx
,&yy
); /* verify x & y coordinates */
1089 case 1: /* rust your armor, j=1 when rusting has occurred */
1091 if ((i
=c
[SHIELD
]) != -1)
1092 if (--ivenarg
[i
] < -1) ivenarg
[i
]= -1; else j
=1;
1093 if ((j
==0) && (k
!= -1))
1096 for (i
=0; i
<ARMORTYPES
; i
++)
1097 if (m
== rustarm
[i
][0]) /* find his armor in table */
1099 if (--ivenarg
[k
]< rustarm
[i
][1])
1100 ivenarg
[k
]= rustarm
[i
][1]; else j
=1;
1104 if (j
==0) /* if rusting did not occur */
1107 case OLEATHER
: p
= "\nThe %s hit you -- Your lucky you have leather on";
1109 case OSSPLATE
: p
= "\nThe %s hit you -- Your fortunate to have stainless steel armor!";
1112 else { beep(); p
= "\nThe %s hit you -- your armor feels weaker"; }
1115 case 2: i
= rnd(15)+8-c
[AC
];
1116 spout
: p
="\nThe %s breathes fire at you!";
1117 if (c
[FIRERESISTANCE
])
1118 p
="\nThe %s's flame doesn't phase you!";
1120 spout2
: if (p
) { lprintf(p
,lastmonst
); beep(); }
1124 case 3: i
= rnd(20)+25-c
[AC
]; goto spout
;
1126 case 4: if (c
[STRENGTH
]>3)
1128 p
="\nThe %s stung you! You feel weaker"; beep();
1131 else p
="\nThe %s stung you!";
1134 case 5: p
="\nThe %s blasts you with his cold breath";
1135 i
= rnd(15)+18-c
[AC
]; goto spout2
;
1137 case 6: lprintf("\nThe %s drains you of your life energy!",lastmonst
);
1138 loselevel(); beep(); return(0);
1140 case 7: p
="\nThe %s got you with a gusher!";
1141 i
= rnd(15)+25-c
[AC
]; goto spout2
;
1143 case 8: if (c
[NOTHEFT
]) return(0); /* he has a device of no theft */
1146 p
="\nThe %s hit you -- Your purse feels lighter";
1147 if (c
[GOLD
]>32767) c
[GOLD
]>>=1;
1148 else c
[GOLD
] -= rnd((int)(1+(c
[GOLD
]>>1)));
1149 if (c
[GOLD
] < 0) c
[GOLD
]=0;
1151 else p
="\nThe %s couldn't find any gold to steal";
1152 lprintf(p
,lastmonst
); disappear(xx
,yy
); beep();
1153 bottomgold(); return(1);
1155 case 9: for(j
=50; ; ) /* disenchant */
1157 i
=rund(26); m
=iven
[i
]; /* randomly select item */
1158 if (m
>0 && ivenarg
[i
]>0 && m
!=OSCROLL
&& m
!=OPOTION
)
1160 if ((ivenarg
[i
] -= 3)<0) ivenarg
[i
]=0;
1161 lprintf("\nThe %s hits you -- you feel a sense of loss",lastmonst
);
1162 srcount
=0; beep(); show3(i
); bottomline(); return(0);
1166 p
="\nThe %s nearly misses"; break;
1172 case 10: p
="\nThe %s hit you with his barbed tail";
1173 i
= rnd(25)-c
[AC
]; goto spout2
;
1175 case 11: p
="\nThe %s has confused you"; beep();
1176 c
[CONFUSE
]+= 10+rnd(10); break;
1178 case 12: /* performs any number of other special attacks */
1179 return(spattack(spsel
[rund(10)],xx
,yy
));
1181 case 13: p
="\nThe %s flattens you with his psionics!";
1182 i
= rnd(15)+30-c
[AC
]; goto spout2
;
1184 case 14: if (c
[NOTHEFT
]) return(0); /* he has device of no theft */
1185 if (emptyhanded()==1)
1187 p
="\nThe %s couldn't find anything to steal";
1190 lprintf("\nThe %s picks your pocket and takes:",lastmonst
);
1192 if (stealsomething()==0) lprcat(" nothing"); disappear(xx
,yy
);
1193 bottomline(); return(1);
1195 case 15: i
= rnd(10)+ 5-c
[AC
];
1196 spout3
: p
="\nThe %s bit you!";
1199 case 16: i
= rnd(15)+10-c
[AC
]; goto spout3
;
1201 if (p
) { lprintf(p
,lastmonst
); bottomline(); }
1206 * checkloss(x) Routine to subtract hp from user and flag bottomline display
1209 * Routine to subtract hitpoints from the user and flag the bottomline display
1210 * Enter with the number of hit points to lose
1211 * Note: if x > c[HP] this routine could kill the player!
1216 if (x
>0) { losehp(x
); bottomhp(); }
1220 * annihilate() Routine to annihilate all monsters around player (playerx,playery)
1222 * Gives player experience, but no dropped objects
1223 * Returns the experience gained from all monsters killed
1230 for (k
=0, i
=playerx
-1; i
<=playerx
+1; i
++)
1231 for (j
=playery
-1; j
<=playery
+1; j
++)
1232 if (!vxy(&i
,&j
)) /* if not out of bounds */
1233 if (*(p
= &mitem
[i
][j
])) /* if a monster there */
1236 k
+= monster
[*p
].experience
; *p
=know
[i
][j
]=0;
1240 lprintf("\nThe %s barely escapes being annihilated!",monster
[*p
].name
);
1241 hitp
[i
][j
] = (hitp
[i
][j
]>>1) + 1; /* lose half hit points*/
1245 lprcat("\nYou hear loud screams of agony!"); raiseexperience((long)k
);
1251 * newsphere(x,y,dir,lifetime) Function to create a new sphere of annihilation
1252 * int x,y,dir,lifetime;
1254 * Enter with the coordinates of the sphere in x,y
1255 * the direction (0-8 diroffx format) in dir, and the lifespan of the
1256 * sphere in lifetime (in turns)
1257 * Returns the number of spheres currently in existence
1259 newsphere(x
,y
,dir
,life
)
1264 if (((sp
=(struct sphere
*)malloc(sizeof(struct sphere
)))) == 0)
1265 return(c
[SPHCAST
]); /* can't malloc, therefore failure */
1266 if (dir
>=9) dir
=0; /* no movement if direction not found */
1267 if (level
==0) vxy(&x
,&y
); /* don't go out of bounds */
1270 if (x
<1) x
=1; if (x
>=MAXX
-1) x
=MAXX
-2;
1271 if (y
<1) y
=1; if (y
>=MAXY
-1) y
=MAXY
-2;
1273 if ((m
=mitem
[x
][y
]) >= DEMONLORD
+4) /* demons dispel spheres */
1275 know
[x
][y
]=1; show1cell(x
,y
); /* show the demon (ha ha) */
1276 cursors(); lprintf("\nThe %s dispels the sphere!",monster
[m
].name
);
1277 beep(); rmsphere(x
,y
); /* remove any spheres that are here */
1280 if (m
==DISENCHANTRESS
) /* disenchantress cancels spheres */
1282 cursors(); lprintf("\nThe %s causes cancellation of the sphere!",monster
[m
].name
); beep();
1283 boom
: sphboom(x
,y
); /* blow up stuff around sphere */
1284 rmsphere(x
,y
); /* remove any spheres that are here */
1287 if (c
[CANCELLATION
]) /* cancellation cancels spheres */
1289 cursors(); lprcat("\nAs the cancellation takes effect, you hear a great earth shaking blast!"); beep();
1292 if (item
[x
][y
]==OANNIHILATION
) /* collision of spheres detonates spheres */
1294 cursors(); lprcat("\nTwo spheres of annihilation collide! You hear a great earth shaking blast!"); beep();
1298 if (playerx
==x
&& playery
==y
) /* collision of sphere and player! */
1301 lprcat("\nYou have been enveloped by the zone of nothingness!\n");
1302 beep(); rmsphere(x
,y
); /* remove any spheres that are here */
1303 nap(4000); died(258);
1305 item
[x
][y
]=OANNIHILATION
; mitem
[x
][y
]=0; know
[x
][y
]=1;
1306 show1cell(x
,y
); /* show the new sphere */
1307 sp
->x
=x
; sp
->y
=y
; sp
->lev
=level
; sp
->dir
=dir
; sp
->lifetime
=life
; sp
->p
=0;
1308 if (spheres
==0) spheres
=sp
; /* if first node in the sphere list */
1309 else /* add sphere to beginning of linked list */
1311 sp
->p
= spheres
; spheres
= sp
;
1313 return(++c
[SPHCAST
]); /* one more sphere in the world */
1317 * rmsphere(x,y) Function to delete a sphere of annihilation from list
1320 * Enter with the coordinates of the sphere (on current level)
1321 * Returns the number of spheres currently in existence
1326 register struct sphere
*sp
,*sp2
=0;
1327 for (sp
=spheres
; sp
; sp2
=sp
,sp
=sp
->p
)
1328 if (level
==sp
->lev
) /* is sphere on this level? */
1329 if ((x
==sp
->x
) && (y
==sp
->y
)) /* locate sphere at this location */
1331 item
[x
][y
]=mitem
[x
][y
]=0; know
[x
][y
]=1;
1332 show1cell(x
,y
); /* show the now missing sphere */
1334 if (sp
==spheres
) { sp2
=sp
; spheres
=sp
->p
; free((char*)sp2
); }
1336 { sp2
->p
= sp
->p
; free((char*)sp
); }
1339 return(c
[SPHCAST
]); /* return number of spheres in the world */
1343 * sphboom(x,y) Function to perform the effects of a sphere detonation
1346 * Enter with the coordinates of the blast, Returns no value
1352 if (c
[HOLDMONST
]) c
[HOLDMONST
]=1;
1353 if (c
[CANCELLATION
]) c
[CANCELLATION
]=1;
1354 for (j
=max(1,x
-2); j
<min(x
+3,MAXX
-1); j
++)
1355 for (i
=max(1,y
-2); i
<min(y
+3,MAXY
-1); i
++)
1357 item
[j
][i
]=mitem
[j
][i
]=0;
1359 if (playerx
==j
&& playery
==i
)
1362 lprcat("\nYou were too close to the sphere!");
1364 died(283); /* player killed in explosion */
1370 * genmonst() Function to ask for monster and genocide from game
1372 * This is done by setting a flag in the monster[] structure
1377 cursors(); lprcat("\nGenocide what monster? ");
1378 for (i
=0; (!isalpha(i
)) && (i
!=' '); i
=getchar());
1380 for (j
=0; j
<MAXMONST
; j
++) /* search for the monster type */
1381 if (monstnamelist
[j
]==i
) /* have we found it? */
1383 monster
[j
].genocided
=1; /* genocided from game */
1384 lprintf(" There will be no more %s's",monster
[j
].name
);
1385 /* now wipe out monsters on this level */
1386 newcavelevel(level
); draws(0,MAXX
,0,MAXY
); bot_linex();
1389 lprcat(" You sense failure!");