xref: /openbsd-src/games/hack/hack.trap.c (revision aed906e4b20d9afbda31247cdb6acf6f29da8819)
1*aed906e4Smestre /*	$OpenBSD: hack.trap.c,v 1.9 2016/01/09 18:33:15 mestre Exp $	*/
2d0b779f3Sniklas 
3df930be7Sderaadt /*
4d25013f2Scamield  * Copyright (c) 1985, Stichting Centrum voor Wiskunde en Informatica,
5d25013f2Scamield  * Amsterdam
6d25013f2Scamield  * All rights reserved.
7d25013f2Scamield  *
8d25013f2Scamield  * Redistribution and use in source and binary forms, with or without
9d25013f2Scamield  * modification, are permitted provided that the following conditions are
10d25013f2Scamield  * met:
11d25013f2Scamield  *
12d25013f2Scamield  * - Redistributions of source code must retain the above copyright notice,
13d25013f2Scamield  * this list of conditions and the following disclaimer.
14d25013f2Scamield  *
15d25013f2Scamield  * - Redistributions in binary form must reproduce the above copyright
16d25013f2Scamield  * notice, this list of conditions and the following disclaimer in the
17d25013f2Scamield  * documentation and/or other materials provided with the distribution.
18d25013f2Scamield  *
19d25013f2Scamield  * - Neither the name of the Stichting Centrum voor Wiskunde en
20d25013f2Scamield  * Informatica, nor the names of its contributors may be used to endorse or
21d25013f2Scamield  * promote products derived from this software without specific prior
22d25013f2Scamield  * written permission.
23d25013f2Scamield  *
24d25013f2Scamield  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
25d25013f2Scamield  * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
26d25013f2Scamield  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
27d25013f2Scamield  * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
28d25013f2Scamield  * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
29d25013f2Scamield  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
30d25013f2Scamield  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
31d25013f2Scamield  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
32d25013f2Scamield  * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
33d25013f2Scamield  * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
34d25013f2Scamield  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
35d25013f2Scamield  */
36d25013f2Scamield 
37d25013f2Scamield /*
38d25013f2Scamield  * Copyright (c) 1982 Jay Fenlason <hack@gnu.org>
39d25013f2Scamield  * All rights reserved.
40d25013f2Scamield  *
41d25013f2Scamield  * Redistribution and use in source and binary forms, with or without
42d25013f2Scamield  * modification, are permitted provided that the following conditions
43d25013f2Scamield  * are met:
44d25013f2Scamield  * 1. Redistributions of source code must retain the above copyright
45d25013f2Scamield  *    notice, this list of conditions and the following disclaimer.
46d25013f2Scamield  * 2. Redistributions in binary form must reproduce the above copyright
47d25013f2Scamield  *    notice, this list of conditions and the following disclaimer in the
48d25013f2Scamield  *    documentation and/or other materials provided with the distribution.
49d25013f2Scamield  * 3. The name of the author may not be used to endorse or promote products
50d25013f2Scamield  *    derived from this software without specific prior written permission.
51d25013f2Scamield  *
52d25013f2Scamield  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
53d25013f2Scamield  * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
54d25013f2Scamield  * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL
55d25013f2Scamield  * THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
56d25013f2Scamield  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
57d25013f2Scamield  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
58d25013f2Scamield  * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
59d25013f2Scamield  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
60d25013f2Scamield  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
61d25013f2Scamield  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
62df930be7Sderaadt  */
63df930be7Sderaadt 
644a5fbbc4Spjanzen #include <ctype.h>
654a5fbbc4Spjanzen #include <stdlib.h>
66*aed906e4Smestre 
67df930be7Sderaadt #include "hack.h"
68df930be7Sderaadt 
69df930be7Sderaadt char vowels[] = "aeiou";
70df930be7Sderaadt 
71df930be7Sderaadt char *traps[] = {
72df930be7Sderaadt 	" bear trap",
73df930be7Sderaadt 	"n arrow trap",
74df930be7Sderaadt 	" dart trap",
75df930be7Sderaadt 	" trapdoor",
76df930be7Sderaadt 	" teleportation trap",
77df930be7Sderaadt 	" pit",
78df930be7Sderaadt 	" sleeping gas trap",
79df930be7Sderaadt 	" piercer",
80df930be7Sderaadt 	" mimic"
81df930be7Sderaadt };
82df930be7Sderaadt 
834a5fbbc4Spjanzen static void vtele(void);
844a5fbbc4Spjanzen static void teleds(int, int);
854a5fbbc4Spjanzen static int  teleok(int, int);
864a5fbbc4Spjanzen 
87df930be7Sderaadt struct trap *
maketrap(int x,int y,int typ)884a5fbbc4Spjanzen maketrap(int x, int y, int typ)
89df930be7Sderaadt {
904a5fbbc4Spjanzen 	struct trap *ttmp;
91df930be7Sderaadt 
92df930be7Sderaadt 	ttmp = newtrap();
93df930be7Sderaadt 	ttmp->ttyp = typ;
94df930be7Sderaadt 	ttmp->tseen = 0;
95df930be7Sderaadt 	ttmp->once = 0;
96df930be7Sderaadt 	ttmp->tx = x;
97df930be7Sderaadt 	ttmp->ty = y;
98df930be7Sderaadt 	ttmp->ntrap = ftrap;
99df930be7Sderaadt 	ftrap = ttmp;
100df930be7Sderaadt 	return(ttmp);
101df930be7Sderaadt }
102df930be7Sderaadt 
1034a5fbbc4Spjanzen void
dotrap(struct trap * trap)1044a5fbbc4Spjanzen dotrap(struct trap *trap)
1054a5fbbc4Spjanzen {
1064a5fbbc4Spjanzen 	int ttype = trap->ttyp;
107df930be7Sderaadt 
108df930be7Sderaadt 	nomul(0);
109df930be7Sderaadt 	if(trap->tseen && !rn2(5) && ttype != PIT)
110df930be7Sderaadt 		pline("You escape a%s.", traps[ttype]);
111df930be7Sderaadt 	else {
112df930be7Sderaadt 		trap->tseen = 1;
113df930be7Sderaadt 		switch(ttype) {
114df930be7Sderaadt 		case SLP_GAS_TRAP:
115df930be7Sderaadt 			pline("A cloud of gas puts you to sleep!");
116df930be7Sderaadt 			nomul(-rnd(25));
117df930be7Sderaadt 			break;
118df930be7Sderaadt 		case BEAR_TRAP:
119df930be7Sderaadt 			if(Levitation) {
120df930be7Sderaadt 				pline("You float over a bear trap.");
121df930be7Sderaadt 				break;
122df930be7Sderaadt 			}
123df930be7Sderaadt 			u.utrap = 4 + rn2(4);
124df930be7Sderaadt 			u.utraptype = TT_BEARTRAP;
125df930be7Sderaadt 			pline("A bear trap closes on your foot!");
126df930be7Sderaadt 			break;
127df930be7Sderaadt 		case PIERC:
128df930be7Sderaadt 			deltrap(trap);
129df930be7Sderaadt 			if(makemon(PM_PIERCER,u.ux,u.uy)) {
130df930be7Sderaadt 			  pline("A piercer suddenly drops from the ceiling!");
131df930be7Sderaadt 			  if(uarmh)
132df930be7Sderaadt 				pline("Its blow glances off your helmet.");
133df930be7Sderaadt 			  else
134df930be7Sderaadt 				(void) thitu(3,d(4,6),"falling piercer");
135df930be7Sderaadt 			}
136df930be7Sderaadt 			break;
137df930be7Sderaadt 		case ARROW_TRAP:
138df930be7Sderaadt 			pline("An arrow shoots out at you!");
139df930be7Sderaadt 			if(!thitu(8,rnd(6),"arrow")){
140df930be7Sderaadt 				mksobj_at(ARROW, u.ux, u.uy);
141df930be7Sderaadt 				fobj->quan = 1;
142df930be7Sderaadt 			}
143df930be7Sderaadt 			break;
144df930be7Sderaadt 		case TRAPDOOR:
145df930be7Sderaadt 			if(!xdnstair) {
146df930be7Sderaadt pline("A trap door in the ceiling opens and a rock falls on your head!");
147df930be7Sderaadt if(uarmh) pline("Fortunately, you are wearing a helmet!");
148df930be7Sderaadt 			    losehp(uarmh ? 2 : d(2,10),"falling rock");
149df930be7Sderaadt 			    mksobj_at(ROCK, u.ux, u.uy);
150df930be7Sderaadt 			    fobj->quan = 1;
151df930be7Sderaadt 			    stackobj(fobj);
152df930be7Sderaadt 			    if(Invisible) newsym(u.ux, u.uy);
153df930be7Sderaadt 			} else {
1544a5fbbc4Spjanzen 			    int newlevel = dlevel + 1;
155df930be7Sderaadt 				while(!rn2(4) && newlevel < 29)
156df930be7Sderaadt 					newlevel++;
157df930be7Sderaadt 				pline("A trap door opens up under you!");
158df930be7Sderaadt 				if(Levitation || u.ustuck) {
159df930be7Sderaadt  				pline("For some reason you don't fall in.");
160df930be7Sderaadt 					break;
161df930be7Sderaadt 				}
162df930be7Sderaadt 
163df930be7Sderaadt 				goto_level(newlevel, FALSE);
164df930be7Sderaadt 			}
165df930be7Sderaadt 			break;
166df930be7Sderaadt 		case DART_TRAP:
167df930be7Sderaadt 			pline("A little dart shoots out at you!");
168df930be7Sderaadt 			if(thitu(7,rnd(3),"little dart")) {
169df930be7Sderaadt 			    if(!rn2(6))
170df930be7Sderaadt 				poisoned("dart","poison dart");
171df930be7Sderaadt 			} else {
172df930be7Sderaadt 				mksobj_at(DART, u.ux, u.uy);
173df930be7Sderaadt 				fobj->quan = 1;
174df930be7Sderaadt 			}
175df930be7Sderaadt 			break;
176df930be7Sderaadt 		case TELEP_TRAP:
177df930be7Sderaadt 			if(trap->once) {
178df930be7Sderaadt 				deltrap(trap);
179df930be7Sderaadt 				newsym(u.ux,u.uy);
180df930be7Sderaadt 				vtele();
181df930be7Sderaadt 			} else {
182df930be7Sderaadt 				newsym(u.ux,u.uy);
183df930be7Sderaadt 				tele();
184df930be7Sderaadt 			}
185df930be7Sderaadt 			break;
186df930be7Sderaadt 		case PIT:
187df930be7Sderaadt 			if(Levitation) {
188df930be7Sderaadt 				pline("A pit opens up under you!");
189df930be7Sderaadt 				pline("You don't fall in!");
190df930be7Sderaadt 				break;
191df930be7Sderaadt 			}
192df930be7Sderaadt 			pline("You fall into a pit!");
193df930be7Sderaadt 			u.utrap = rn1(6,2);
194df930be7Sderaadt 			u.utraptype = TT_PIT;
195df930be7Sderaadt 			losehp(rnd(6),"fall into a pit");
196df930be7Sderaadt 			selftouch("Falling, you");
197df930be7Sderaadt 			break;
198df930be7Sderaadt 		default:
199df930be7Sderaadt 			impossible("You hit a trap of type %u", trap->ttyp);
200df930be7Sderaadt 		}
201df930be7Sderaadt 	}
202df930be7Sderaadt }
203df930be7Sderaadt 
2044a5fbbc4Spjanzen int
mintrap(struct monst * mtmp)2054a5fbbc4Spjanzen mintrap(struct monst *mtmp)
2064a5fbbc4Spjanzen {
2074a5fbbc4Spjanzen 	struct trap *trap = t_at(mtmp->mx, mtmp->my);
2084a5fbbc4Spjanzen 	int wasintrap = mtmp->mtrapped;
209df930be7Sderaadt 
210df930be7Sderaadt 	if(!trap) {
211df930be7Sderaadt 		mtmp->mtrapped = 0;	/* perhaps teleported? */
212df930be7Sderaadt 	} else if(wasintrap) {
213df930be7Sderaadt 		if(!rn2(40)) mtmp->mtrapped = 0;
214df930be7Sderaadt 	} else {
2154a5fbbc4Spjanzen 	    int tt = trap->ttyp;
216df930be7Sderaadt 	    int in_sight = cansee(mtmp->mx,mtmp->my);
217df930be7Sderaadt 	    extern char mlarge[];
218df930be7Sderaadt 
219df930be7Sderaadt 	    if(mtmp->mtrapseen & (1 << tt)) {
220df930be7Sderaadt 		/* he has been in such a trap - perhaps he escapes */
221df930be7Sderaadt 		if(rn2(4)) return(0);
222df930be7Sderaadt 	    }
223df930be7Sderaadt 	    mtmp->mtrapseen |= (1 << tt);
224df930be7Sderaadt 	    switch (tt) {
225df930be7Sderaadt 		case BEAR_TRAP:
226180acc8fSmillert 			if(strchr(mlarge, mtmp->data->mlet)) {
227df930be7Sderaadt 				if(in_sight)
228df930be7Sderaadt 				  pline("%s is caught in a bear trap!",
229df930be7Sderaadt 					Monnam(mtmp));
230df930be7Sderaadt 				else
231df930be7Sderaadt 				  if(mtmp->data->mlet == 'o')
232df930be7Sderaadt 			    pline("You hear the roaring of an angry bear!");
233df930be7Sderaadt 				mtmp->mtrapped = 1;
234df930be7Sderaadt 			}
235df930be7Sderaadt 			break;
236df930be7Sderaadt 		case PIT:
237df930be7Sderaadt 			/* there should be a mtmp/data -> floating */
238180acc8fSmillert 			if(!strchr("EywBfk'& ", mtmp->data->mlet)) { /* ab */
239df930be7Sderaadt 				mtmp->mtrapped = 1;
240df930be7Sderaadt 				if(in_sight)
241df930be7Sderaadt 				  pline("%s falls in a pit!", Monnam(mtmp));
242df930be7Sderaadt 			}
243df930be7Sderaadt 			break;
244df930be7Sderaadt 		case SLP_GAS_TRAP:
245df930be7Sderaadt 			if(!mtmp->msleep && !mtmp->mfroz) {
246df930be7Sderaadt 				mtmp->msleep = 1;
247df930be7Sderaadt 				if(in_sight)
248df930be7Sderaadt 				  pline("%s suddenly falls asleep!",
249df930be7Sderaadt 					Monnam(mtmp));
250df930be7Sderaadt 			}
251df930be7Sderaadt 			break;
252df930be7Sderaadt 		case TELEP_TRAP:
253df930be7Sderaadt 			rloc(mtmp);
254df930be7Sderaadt 			if(in_sight && !cansee(mtmp->mx,mtmp->my))
255df930be7Sderaadt 				pline("%s suddenly disappears!",
256df930be7Sderaadt 					Monnam(mtmp));
257df930be7Sderaadt 			break;
258df930be7Sderaadt 		case ARROW_TRAP:
259df930be7Sderaadt 			if(in_sight) {
260df930be7Sderaadt 				pline("%s is hit by an arrow!",
261df930be7Sderaadt 					Monnam(mtmp));
262df930be7Sderaadt 			}
263df930be7Sderaadt 			mtmp->mhp -= 3;
264df930be7Sderaadt 			break;
265df930be7Sderaadt 		case DART_TRAP:
266df930be7Sderaadt 			if(in_sight) {
267df930be7Sderaadt 				pline("%s is hit by a dart!",
268df930be7Sderaadt 					Monnam(mtmp));
269df930be7Sderaadt 			}
270df930be7Sderaadt 			mtmp->mhp -= 2;
271df930be7Sderaadt 			/* not mondied here !! */
272df930be7Sderaadt 			break;
273df930be7Sderaadt 		case TRAPDOOR:
274df930be7Sderaadt 			if(!xdnstair) {
275df930be7Sderaadt 				mtmp->mhp -= 10;
276df930be7Sderaadt 				if(in_sight)
277df930be7Sderaadt pline("A trap door in the ceiling opens and a rock hits %s!", monnam(mtmp));
278df930be7Sderaadt 				break;
279df930be7Sderaadt 			}
280df930be7Sderaadt 			if(mtmp->data->mlet != 'w'){
281df930be7Sderaadt 				fall_down(mtmp);
282df930be7Sderaadt 				if(in_sight)
283df930be7Sderaadt 		pline("Suddenly, %s disappears out of sight.", monnam(mtmp));
284df930be7Sderaadt 				return(2);	/* no longer on this level */
285df930be7Sderaadt 			}
286df930be7Sderaadt 			break;
287df930be7Sderaadt 		case PIERC:
288df930be7Sderaadt 			break;
289df930be7Sderaadt 		default:
290df930be7Sderaadt 			impossible("Some monster encountered a strange trap.");
291df930be7Sderaadt 	    }
292df930be7Sderaadt 	}
293df930be7Sderaadt 	return(mtmp->mtrapped);
294df930be7Sderaadt }
295df930be7Sderaadt 
2964a5fbbc4Spjanzen void
selftouch(char * arg)2974a5fbbc4Spjanzen selftouch(char *arg)
2984a5fbbc4Spjanzen {
299df930be7Sderaadt 	if(uwep && uwep->otyp == DEAD_COCKATRICE){
300df930be7Sderaadt 		pline("%s touch the dead cockatrice.", arg);
301df930be7Sderaadt 		pline("You turn to stone.");
302df930be7Sderaadt 		killer = objects[uwep->otyp].oc_name;
303df930be7Sderaadt 		done("died");
304df930be7Sderaadt 	}
305df930be7Sderaadt }
306df930be7Sderaadt 
3074a5fbbc4Spjanzen void
float_up(void)308*aed906e4Smestre float_up(void)
3094a5fbbc4Spjanzen {
310df930be7Sderaadt 	if(u.utrap) {
311df930be7Sderaadt 		if(u.utraptype == TT_PIT) {
312df930be7Sderaadt 			u.utrap = 0;
313df930be7Sderaadt 			pline("You float up, out of the pit!");
314df930be7Sderaadt 		} else {
315df930be7Sderaadt 			pline("You float up, only your leg is still stuck.");
316df930be7Sderaadt 		}
317df930be7Sderaadt 	} else
318df930be7Sderaadt 		pline("You start to float in the air!");
319df930be7Sderaadt }
320df930be7Sderaadt 
3214a5fbbc4Spjanzen int
float_down(void)322*aed906e4Smestre float_down(void)
3234a5fbbc4Spjanzen {
3244a5fbbc4Spjanzen 	struct trap *trap;
3254a5fbbc4Spjanzen 
326df930be7Sderaadt 	pline("You float gently to the ground.");
3274a5fbbc4Spjanzen 	if ((trap = t_at(u.ux,u.uy)))
328df930be7Sderaadt 		switch(trap->ttyp) {
329df930be7Sderaadt 		case PIERC:
330df930be7Sderaadt 			break;
331df930be7Sderaadt 		case TRAPDOOR:
332df930be7Sderaadt 			if(!xdnstair || u.ustuck) break;
333df930be7Sderaadt 			/* fall into next case */
334df930be7Sderaadt 		default:
335df930be7Sderaadt 			dotrap(trap);
336df930be7Sderaadt 	}
337df930be7Sderaadt 	pickup(1);
3384a5fbbc4Spjanzen 	return(0);	/* XXX value needed in hack.potion.c */
339df930be7Sderaadt }
340df930be7Sderaadt 
3414a5fbbc4Spjanzen static void
vtele(void)342*aed906e4Smestre vtele(void)
3434a5fbbc4Spjanzen {
3444a5fbbc4Spjanzen 	struct mkroom *croom;
3454a5fbbc4Spjanzen 
346df930be7Sderaadt 	for(croom = &rooms[0]; croom->hx >= 0; croom++)
347df930be7Sderaadt 	    if(croom->rtype == VAULT) {
3484a5fbbc4Spjanzen 		int x,y;
349df930be7Sderaadt 
350df930be7Sderaadt 		x = rn2(2) ? croom->lx : croom->hx;
351df930be7Sderaadt 		y = rn2(2) ? croom->ly : croom->hy;
352df930be7Sderaadt 		if(teleok(x,y)) {
353df930be7Sderaadt 		    teleds(x,y);
354df930be7Sderaadt 		    return;
355df930be7Sderaadt 		}
356df930be7Sderaadt 	    }
357df930be7Sderaadt 	tele();
358df930be7Sderaadt }
359df930be7Sderaadt 
3604a5fbbc4Spjanzen void
tele(void)361*aed906e4Smestre tele(void)
3624a5fbbc4Spjanzen {
363df930be7Sderaadt 	coord cc;
3644a5fbbc4Spjanzen 	int nux,nuy;
365df930be7Sderaadt 
366df930be7Sderaadt 	if(Teleport_control) {
367df930be7Sderaadt 		pline("To what position do you want to be teleported?");
368df930be7Sderaadt 		cc = getpos(1, "the desired position"); /* 1: force valid */
369df930be7Sderaadt 		/* possible extensions: introduce a small error if
370df930be7Sderaadt 		   magic power is low; allow transfer to solid rock */
371df930be7Sderaadt 		if(teleok(cc.x, cc.y)){
372df930be7Sderaadt 			teleds(cc.x, cc.y);
373df930be7Sderaadt 			return;
374df930be7Sderaadt 		}
375df930be7Sderaadt 		pline("Sorry ...");
376df930be7Sderaadt 	}
377df930be7Sderaadt 	do {
378df930be7Sderaadt 		nux = rnd(COLNO-1);
379df930be7Sderaadt 		nuy = rn2(ROWNO);
380df930be7Sderaadt 	} while(!teleok(nux, nuy));
381df930be7Sderaadt 	teleds(nux, nuy);
382df930be7Sderaadt }
383df930be7Sderaadt 
3844a5fbbc4Spjanzen static void
teleds(int nux,int nuy)3854a5fbbc4Spjanzen teleds(int nux, int nuy)
386df930be7Sderaadt {
387df930be7Sderaadt 	if(Punished) unplacebc();
388df930be7Sderaadt 	unsee();
389df930be7Sderaadt 	u.utrap = 0;
390df930be7Sderaadt 	u.ustuck = 0;
391df930be7Sderaadt 	u.ux = nux;
392df930be7Sderaadt 	u.uy = nuy;
393df930be7Sderaadt 	setsee();
394df930be7Sderaadt 	if(Punished) placebc(1);
395df930be7Sderaadt 	if(u.uswallow){
396df930be7Sderaadt 		u.uswldtim = u.uswallow = 0;
397df930be7Sderaadt 		docrt();
398df930be7Sderaadt 	}
399df930be7Sderaadt 	nomul(0);
400df930be7Sderaadt 	if(levl[nux][nuy].typ == POOL && !Levitation)
401df930be7Sderaadt 		drown();
402df930be7Sderaadt 	(void) inshop();
403df930be7Sderaadt 	pickup(1);
404df930be7Sderaadt 	if(!Blind) read_engr_at(u.ux,u.uy);
405df930be7Sderaadt }
406df930be7Sderaadt 
4074a5fbbc4Spjanzen static int
teleok(int x,int y)4084a5fbbc4Spjanzen teleok(int x, int y)
4094a5fbbc4Spjanzen {	/* might throw him into a POOL */
410df930be7Sderaadt 	return( isok(x,y) && !IS_ROCK(levl[x][y].typ) && !m_at(x,y) &&
411df930be7Sderaadt 		!sobj_at(ENORMOUS_ROCK,x,y) && !t_at(x,y)
412df930be7Sderaadt 	);
413df930be7Sderaadt 	/* Note: gold is permitted (because of vaults) */
414df930be7Sderaadt }
415df930be7Sderaadt 
4164a5fbbc4Spjanzen int
dotele(void)417*aed906e4Smestre dotele(void)
4184a5fbbc4Spjanzen {
419df930be7Sderaadt 	extern char pl_character[];
420df930be7Sderaadt 
421df930be7Sderaadt 	if(
422df930be7Sderaadt #ifdef WIZARD
423df930be7Sderaadt 	   !wizard &&
42454da88e4Spjanzen #endif /* WIZARD */
425df930be7Sderaadt 		      (!Teleportation || u.ulevel < 6 ||
426df930be7Sderaadt 			(pl_character[0] != 'W' && u.ulevel < 10))) {
427df930be7Sderaadt 		pline("You are not able to teleport at will.");
428df930be7Sderaadt 		return(0);
429df930be7Sderaadt 	}
430df930be7Sderaadt 	if(u.uhunger <= 100 || u.ustr < 6) {
431df930be7Sderaadt 		pline("You miss the strength for a teleport spell.");
432df930be7Sderaadt 		return(1);
433df930be7Sderaadt 	}
434df930be7Sderaadt 	tele();
435df930be7Sderaadt 	morehungry(100);
436df930be7Sderaadt 	return(1);
437df930be7Sderaadt }
438df930be7Sderaadt 
4394a5fbbc4Spjanzen void
placebc(int attach)4404a5fbbc4Spjanzen placebc(int attach)
4414a5fbbc4Spjanzen {
442df930be7Sderaadt 	if(!uchain || !uball){
443df930be7Sderaadt 		impossible("Where are your chain and ball??");
444df930be7Sderaadt 		return;
445df930be7Sderaadt 	}
446df930be7Sderaadt 	uball->ox = uchain->ox = u.ux;
447df930be7Sderaadt 	uball->oy = uchain->oy = u.uy;
448df930be7Sderaadt 	if(attach){
449df930be7Sderaadt 		uchain->nobj = fobj;
450df930be7Sderaadt 		fobj = uchain;
451df930be7Sderaadt 		if(!carried(uball)){
452df930be7Sderaadt 			uball->nobj = fobj;
453df930be7Sderaadt 			fobj = uball;
454df930be7Sderaadt 		}
455df930be7Sderaadt 	}
456df930be7Sderaadt }
457df930be7Sderaadt 
4584a5fbbc4Spjanzen void
unplacebc(void)459*aed906e4Smestre unplacebc(void)
4604a5fbbc4Spjanzen {
461df930be7Sderaadt 	if(!carried(uball)){
462df930be7Sderaadt 		freeobj(uball);
463df930be7Sderaadt 		unpobj(uball);
464df930be7Sderaadt 	}
465df930be7Sderaadt 	freeobj(uchain);
466df930be7Sderaadt 	unpobj(uchain);
467df930be7Sderaadt }
468df930be7Sderaadt 
4694a5fbbc4Spjanzen void
level_tele(void)470*aed906e4Smestre level_tele(void)
4714a5fbbc4Spjanzen {
4724a5fbbc4Spjanzen 	int newlevel;
4734a5fbbc4Spjanzen 
474df930be7Sderaadt 	if(Teleport_control) {
475df930be7Sderaadt 	    char buf[BUFSZ];
476df930be7Sderaadt 
477df930be7Sderaadt 	    do {
478df930be7Sderaadt 	      pline("To what level do you want to teleport? [type a number] ");
479df930be7Sderaadt 	      getlin(buf);
480cb40a3d3Smmcc 	    } while(!isdigit((unsigned char)buf[0]) &&
481cb40a3d3Smmcc 	        (buf[0] != '-' || !isdigit((unsigned char)buf[1])));
482df930be7Sderaadt 	    newlevel = atoi(buf);
483df930be7Sderaadt 	} else {
484df930be7Sderaadt 	    newlevel  = 5 + rn2(20);	/* 5 - 24 */
4854a5fbbc4Spjanzen 	    if(dlevel == newlevel) {
4864a5fbbc4Spjanzen 		if(!xdnstair)
4874a5fbbc4Spjanzen 			newlevel--;
4884a5fbbc4Spjanzen 		else
4894a5fbbc4Spjanzen 			newlevel++;
4904a5fbbc4Spjanzen 	    }
491df930be7Sderaadt 	}
492df930be7Sderaadt 	if(newlevel >= 30) {
493df930be7Sderaadt 	    if(newlevel > MAXLEVEL) newlevel = MAXLEVEL;
494df930be7Sderaadt 	    pline("You arrive at the center of the earth ...");
495df930be7Sderaadt 	    pline("Unfortunately it is here that hell is located.");
496df930be7Sderaadt 	    if(Fire_resistance) {
497df930be7Sderaadt 		pline("But the fire doesn't seem to harm you.");
498df930be7Sderaadt 	    } else {
499df930be7Sderaadt 		pline("You burn to a crisp.");
500df930be7Sderaadt 		dlevel = maxdlevel = newlevel;
501df930be7Sderaadt 		killer = "visit to the hell";
502df930be7Sderaadt 		done("burned");
503df930be7Sderaadt 	    }
504df930be7Sderaadt 	}
505df930be7Sderaadt 	if(newlevel < 0) {
506df930be7Sderaadt 	    newlevel = 0;
507df930be7Sderaadt 	    pline("You are now high above the clouds ...");
508df930be7Sderaadt 	    if(Levitation) {
509df930be7Sderaadt 		pline("You float gently down to earth.");
510df930be7Sderaadt 		done("escaped");
511df930be7Sderaadt 	    }
512df930be7Sderaadt 	    pline("Unfortunately, you don't know how to fly.");
513df930be7Sderaadt 	    pline("You fall down a few thousand feet and break your neck.");
514df930be7Sderaadt 	    dlevel = 0;
515df930be7Sderaadt 	    killer = "fall";
516df930be7Sderaadt 	    done("died");
517df930be7Sderaadt 	}
518df930be7Sderaadt 
519df930be7Sderaadt 	goto_level(newlevel, FALSE); /* calls done("escaped") if newlevel==0 */
520df930be7Sderaadt }
521df930be7Sderaadt 
5224a5fbbc4Spjanzen void
drown(void)523*aed906e4Smestre drown(void)
524df930be7Sderaadt {
525df930be7Sderaadt 	pline("You fall into a pool!");
526df930be7Sderaadt 	pline("You can't swim!");
527df930be7Sderaadt 	if(rn2(3) < u.uluck+2) {
528df930be7Sderaadt 		/* most scrolls become unreadable */
5294a5fbbc4Spjanzen 		struct obj *obj;
530df930be7Sderaadt 
531df930be7Sderaadt 		for(obj = invent; obj; obj = obj->nobj)
532df930be7Sderaadt 			if(obj->olet == SCROLL_SYM && rn2(12) > u.uluck)
533df930be7Sderaadt 				obj->otyp = SCR_BLANK_PAPER;
534df930be7Sderaadt 		/* we should perhaps merge these scrolls ? */
535df930be7Sderaadt 
536df930be7Sderaadt 		pline("You attempt a teleport spell.");	/* utcsri!carroll */
537df930be7Sderaadt 		(void) dotele();
5384a5fbbc4Spjanzen 		if(levl[(int)u.ux][(int)u.uy].typ != POOL)
5394a5fbbc4Spjanzen 			return;
540df930be7Sderaadt 	}
541df930be7Sderaadt 	pline("You drown ...");
542df930be7Sderaadt 	killer = "pool of water";
543df930be7Sderaadt 	done("drowned");
544df930be7Sderaadt }
545