xref: /netbsd-src/games/hack/hack.eat.c (revision 404fbe5fb94ca1e054339640cabb2801ce52dd30)
1 /*	$NetBSD: hack.eat.c,v 1.6 2003/04/02 18:36:36 jsm Exp $	*/
2 
3 /*
4  * Copyright (c) 1985, Stichting Centrum voor Wiskunde en Informatica,
5  * Amsterdam
6  * All rights reserved.
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions are
10  * met:
11  *
12  * - Redistributions of source code must retain the above copyright notice,
13  * this list of conditions and the following disclaimer.
14  *
15  * - Redistributions in binary form must reproduce the above copyright
16  * notice, this list of conditions and the following disclaimer in the
17  * documentation and/or other materials provided with the distribution.
18  *
19  * - Neither the name of the Stichting Centrum voor Wiskunde en
20  * Informatica, nor the names of its contributors may be used to endorse or
21  * promote products derived from this software without specific prior
22  * written permission.
23  *
24  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
25  * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
26  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
27  * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
28  * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
29  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
30  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
31  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
32  * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
33  * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
34  * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
35  */
36 
37 /*
38  * Copyright (c) 1982 Jay Fenlason <hack@gnu.org>
39  * All rights reserved.
40  *
41  * Redistribution and use in source and binary forms, with or without
42  * modification, are permitted provided that the following conditions
43  * are met:
44  * 1. Redistributions of source code must retain the above copyright
45  *    notice, this list of conditions and the following disclaimer.
46  * 2. Redistributions in binary form must reproduce the above copyright
47  *    notice, this list of conditions and the following disclaimer in the
48  *    documentation and/or other materials provided with the distribution.
49  * 3. The name of the author may not be used to endorse or promote products
50  *    derived from this software without specific prior written permission.
51  *
52  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
53  * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
54  * AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL
55  * THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
56  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
57  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
58  * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
59  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
60  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
61  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
62  */
63 
64 #include <sys/cdefs.h>
65 #ifndef lint
66 __RCSID("$NetBSD: hack.eat.c,v 1.6 2003/04/02 18:36:36 jsm Exp $");
67 #endif				/* not lint */
68 
69 #include "hack.h"
70 #include "extern.h"
71 char            POISONOUS[] = "ADKSVabhks";
72 
73 /* hunger texts used on bottom line (each 8 chars long) */
74 #define	SATIATED	0
75 #define NOT_HUNGRY	1
76 #define	HUNGRY		2
77 #define	WEAK		3
78 #define	FAINTING	4
79 #define FAINTED		5
80 #define STARVED		6
81 
82 const char           *const hu_stat[] = {
83 	"Satiated",
84 	"        ",
85 	"Hungry  ",
86 	"Weak    ",
87 	"Fainting",
88 	"Fainted ",
89 	"Starved "
90 };
91 
92 void
93 init_uhunger()
94 {
95 	u.uhunger = 900;
96 	u.uhs = NOT_HUNGRY;
97 }
98 
99 #define	TTSZ	SIZE(tintxts)
100 const struct {
101 	const char           *txt;
102 	int             nut;
103 }               tintxts[] = {
104 	{ "It contains first quality peaches - what a surprise!", 40 },
105 	{ "It contains salmon - not bad!", 60 },
106 	{ "It contains apple juice - perhaps not what you hoped for.", 20 },
107 	{ "It contains some nondescript substance, tasting awfully.", 500 },
108 	{ "It contains rotten meat. You vomit.", -50 },
109 	{ "It turns out to be empty.", 0 }
110 };
111 
112 static struct {
113 	struct obj     *tin;
114 	int             usedtime, reqtime;
115 }               tin;
116 
117 int
118 opentin()
119 {
120 	int             r;
121 
122 	if (!carried(tin.tin))	/* perhaps it was stolen? */
123 		return (0);	/* %% probably we should use tinoid */
124 	if (tin.usedtime++ >= 50) {
125 		pline("You give up your attempt to open the tin.");
126 		return (0);
127 	}
128 	if (tin.usedtime < tin.reqtime)
129 		return (1);	/* still busy */
130 
131 	pline("You succeed in opening the tin.");
132 	useup(tin.tin);
133 	r = rn2(2 * TTSZ);
134 	if (r < TTSZ) {
135 		pline(tintxts[r].txt);
136 		lesshungry(tintxts[r].nut);
137 		if (r == 1) {	/* SALMON */
138 			Glib = rnd(15);
139 			pline("Eating salmon made your fingers very slippery.");
140 		}
141 	} else {
142 		pline("It contains spinach - this makes you feel like Popeye!");
143 		lesshungry(600);
144 		if (u.ustr < 118)
145 			u.ustr += rnd(((u.ustr < 17) ? 19 : 118) - u.ustr);
146 		if (u.ustr > u.ustrmax)
147 			u.ustrmax = u.ustr;
148 		flags.botl = 1;
149 	}
150 	return (0);
151 }
152 
153 int
154 Meatdone()
155 {
156 	u.usym = '@';
157 	prme();
158 	return 0;
159 }
160 
161 int
162 doeat()
163 {
164 	struct obj     *otmp;
165 	struct objclass *ftmp;
166 	int tmp;
167 
168 	/* Is there some food (probably a heavy corpse) here on the ground? */
169 	if (!Levitation)
170 		for (otmp = fobj; otmp; otmp = otmp->nobj) {
171 			if (otmp->ox == u.ux && otmp->oy == u.uy &&
172 			    otmp->olet == FOOD_SYM) {
173 				pline("There %s %s here; eat %s? [ny] ",
174 				      (otmp->quan == 1) ? "is" : "are",
175 				      doname(otmp),
176 				      (otmp->quan == 1) ? "it" : "one");
177 				if (readchar() == 'y') {
178 					if (otmp->quan != 1)
179 						(void) splitobj(otmp, 1);
180 					freeobj(otmp);
181 					otmp = addinv(otmp);
182 					addtobill(otmp);
183 					goto gotit;
184 				}
185 			}
186 		}
187 	otmp = getobj("%", "eat");
188 	if (!otmp)
189 		return (0);
190 gotit:
191 	if (otmp->otyp == TIN) {
192 		if (uwep) {
193 			switch (uwep->otyp) {
194 			case CAN_OPENER:
195 				tmp = 1;
196 				break;
197 			case DAGGER:
198 			case CRYSKNIFE:
199 				tmp = 3;
200 				break;
201 			case PICK_AXE:
202 			case AXE:
203 				tmp = 6;
204 				break;
205 			default:
206 				goto no_opener;
207 			}
208 			pline("Using your %s you try to open the tin.",
209 			      aobjnam(uwep, (char *) 0));
210 		} else {
211 	no_opener:
212 			pline("It is not so easy to open this tin.");
213 			if (Glib) {
214 				pline("The tin slips out of your hands.");
215 				if (otmp->quan > 1) {
216 					struct obj     *obj;
217 
218 					obj = splitobj(otmp, 1);
219 					if (otmp == uwep)
220 						setuwep(obj);
221 				}
222 				dropx(otmp);
223 				return (1);
224 			}
225 			tmp = 10 + rn2(1 + 500 / ((int) (u.ulevel + u.ustr)));
226 		}
227 		tin.reqtime = tmp;
228 		tin.usedtime = 0;
229 		tin.tin = otmp;
230 		occupation = opentin;
231 		occtxt = "opening the tin";
232 		return (1);
233 	}
234 	ftmp = &objects[otmp->otyp];
235 	multi = -ftmp->oc_delay;
236 	if (otmp->otyp >= CORPSE && eatcorpse(otmp))
237 		goto eatx;
238 	if (!rn2(7) && otmp->otyp != FORTUNE_COOKIE) {
239 		pline("Blecch!  Rotten food!");
240 		if (!rn2(4)) {
241 			pline("You feel rather light headed.");
242 			Confusion += d(2, 4);
243 		} else if (!rn2(4) && !Blind) {
244 			pline("Everything suddenly goes dark.");
245 			Blind = d(2, 10);
246 			seeoff(0);
247 		} else if (!rn2(3)) {
248 			if (Blind)
249 				pline("The world spins and you slap against the floor.");
250 			else
251 				pline("The world spins and goes dark.");
252 			nomul(-rnd(10));
253 			nomovemsg = "You are conscious again.";
254 		}
255 		lesshungry(ftmp->nutrition / 4);
256 	} else {
257 		if (u.uhunger >= 1500) {
258 			pline("You choke over your food.");
259 			pline("You die...");
260 			killer = ftmp->oc_name;
261 			done("choked");
262 		}
263 		switch (otmp->otyp) {
264 		case FOOD_RATION:
265 			if (u.uhunger <= 200)
266 				pline("That food really hit the spot!");
267 			else if (u.uhunger <= 700)
268 				pline("That satiated your stomach!");
269 			else {
270 				pline("You're having a hard time getting all that food down.");
271 				multi -= 2;
272 			}
273 			lesshungry(ftmp->nutrition);
274 			if (multi < 0)
275 				nomovemsg = "You finished your meal.";
276 			break;
277 		case TRIPE_RATION:
278 			pline("Yak - dog food!");
279 			more_experienced(1, 0);
280 			flags.botl = 1;
281 			if (rn2(2)) {
282 				pline("You vomit.");
283 				morehungry(20);
284 				if (Sick) {
285 					Sick = 0;	/* David Neves */
286 					pline("What a relief!");
287 				}
288 			} else
289 				lesshungry(ftmp->nutrition);
290 			break;
291 		default:
292 			if (otmp->otyp >= CORPSE)
293 				pline("That %s tasted terrible!", ftmp->oc_name);
294 			else
295 				pline("That %s was delicious!", ftmp->oc_name);
296 			lesshungry(ftmp->nutrition);
297 			if (otmp->otyp == DEAD_LIZARD && (Confusion > 2))
298 				Confusion = 2;
299 			else
300 #ifdef QUEST
301 			if (otmp->otyp == CARROT && !Blind) {
302 				u.uhorizon++;
303 				setsee();
304 				pline("Your vision improves.");
305 			} else
306 #endif	/* QUEST */
307 			if (otmp->otyp == FORTUNE_COOKIE) {
308 				if (Blind) {
309 					pline("This cookie has a scrap of paper inside!");
310 					pline("What a pity, that you cannot read it!");
311 				} else
312 					outrumor();
313 			} else if (otmp->otyp == LUMP_OF_ROYAL_JELLY) {
314 				/* This stuff seems to be VERY healthy! */
315 				if (u.ustrmax < 118)
316 					u.ustrmax++;
317 				if (u.ustr < u.ustrmax)
318 					u.ustr++;
319 				u.uhp += rnd(20);
320 				if (u.uhp > u.uhpmax) {
321 					if (!rn2(17))
322 						u.uhpmax++;
323 					u.uhp = u.uhpmax;
324 				}
325 				heal_legs();
326 			}
327 			break;
328 		}
329 	}
330 eatx:
331 	if (multi < 0 && !nomovemsg) {
332 		static char     msgbuf[BUFSZ];
333 		(void) sprintf(msgbuf, "You finished eating the %s.",
334 			       ftmp->oc_name);
335 		nomovemsg = msgbuf;
336 	}
337 	useup(otmp);
338 	return (1);
339 }
340 
341 /* called in hack.main.c */
342 void
343 gethungry()
344 {
345 	--u.uhunger;
346 	if (moves % 2) {
347 		if (Regeneration)
348 			u.uhunger--;
349 		if (Hunger)
350 			u.uhunger--;
351 		/*
352 		 * a3:  if(Hunger & LEFT_RING) u.uhunger--; if(Hunger &
353 		 * RIGHT_RING) u.uhunger--; etc.
354 		 */
355 	}
356 	if (moves % 20 == 0) {	/* jimt@asgb */
357 		if (uleft)
358 			u.uhunger--;
359 		if (uright)
360 			u.uhunger--;
361 	}
362 	newuhs(TRUE);
363 }
364 
365 /* called after vomiting and after performing feats of magic */
366 void
367 morehungry(num)
368 	int num;
369 {
370 	u.uhunger -= num;
371 	newuhs(TRUE);
372 }
373 
374 /* called after eating something (and after drinking fruit juice) */
375 void
376 lesshungry(num)
377 	int num;
378 {
379 	u.uhunger += num;
380 	newuhs(FALSE);
381 }
382 
383 int
384 unfaint()
385 {
386 	u.uhs = FAINTING;
387 	flags.botl = 1;
388 	return 0;
389 }
390 
391 void
392 newuhs(incr)
393 	boolean         incr;
394 {
395 	int             newhs, h = u.uhunger;
396 
397 	newhs = (h > 1000) ? SATIATED :
398 		(h > 150) ? NOT_HUNGRY :
399 		(h > 50) ? HUNGRY :
400 		(h > 0) ? WEAK : FAINTING;
401 
402 	if (newhs == FAINTING) {
403 		if (u.uhs == FAINTED)
404 			newhs = FAINTED;
405 		if (u.uhs <= WEAK || rn2(20 - u.uhunger / 10) >= 19) {
406 			if (u.uhs != FAINTED && multi >= 0 /* %% */ ) {
407 				pline("You faint from lack of food.");
408 				nomul(-10 + (u.uhunger / 10));
409 				nomovemsg = "You regain consciousness.";
410 				afternmv = unfaint;
411 				newhs = FAINTED;
412 			}
413 		} else if (u.uhunger < -(int) (200 + 25 * u.ulevel)) {
414 			u.uhs = STARVED;
415 			flags.botl = 1;
416 			bot();
417 			pline("You die from starvation.");
418 			done("starved");
419 		}
420 	}
421 	if (newhs != u.uhs) {
422 		if (newhs >= WEAK && u.uhs < WEAK)
423 			losestr(1);	/* this may kill you -- see below */
424 		else if (newhs < WEAK && u.uhs >= WEAK && u.ustr < u.ustrmax)
425 			losestr(-1);
426 		switch (newhs) {
427 		case HUNGRY:
428 			pline((!incr) ? "You only feel hungry now." :
429 			      (u.uhunger < 145) ? "You feel hungry." :
430 			      "You are beginning to feel hungry.");
431 			break;
432 		case WEAK:
433 			pline((!incr) ? "You feel weak now." :
434 			      (u.uhunger < 45) ? "You feel weak." :
435 			      "You are beginning to feel weak.");
436 			break;
437 		}
438 		u.uhs = newhs;
439 		flags.botl = 1;
440 		if (u.uhp < 1) {
441 			pline("You die from hunger and exhaustion.");
442 			killer = "exhaustion";
443 			done("starved");
444 		}
445 	}
446 }
447 
448 #define	CORPSE_I_TO_C(otyp)	(char) ((otyp >= DEAD_ACID_BLOB)\
449 		     ?  'a' + (otyp - DEAD_ACID_BLOB)\
450 		     :	'@' + (otyp - DEAD_HUMAN))
451 int
452 poisonous(otmp)
453 	struct obj     *otmp;
454 {
455 	return (strchr(POISONOUS, CORPSE_I_TO_C(otmp->otyp)) != 0);
456 }
457 
458 /* returns 1 if some text was printed */
459 int
460 eatcorpse(otmp)
461 	struct obj     *otmp;
462 {
463 	char            let = CORPSE_I_TO_C(otmp->otyp);
464 	int             tp = 0;
465 	if (let != 'a' && moves > otmp->age + 50 + rn2(100)) {
466 		tp++;
467 		pline("Ulch -- that meat was tainted!");
468 		pline("You get very sick.");
469 		Sick = 10 + rn2(10);
470 		u.usick_cause = objects[otmp->otyp].oc_name;
471 	} else if (strchr(POISONOUS, let) && rn2(5)) {
472 		tp++;
473 		pline("Ecch -- that must have been poisonous!");
474 		if (!Poison_resistance) {
475 			losestr(rnd(4));
476 			losehp(rnd(15), "poisonous corpse");
477 		} else
478 			pline("You don't seem affected by the poison.");
479 	} else if (strchr("ELNOPQRUuxz", let) && rn2(5)) {
480 		tp++;
481 		pline("You feel sick.");
482 		losehp(rnd(8), "cadaver");
483 	}
484 	switch (let) {
485 	case 'L':
486 	case 'N':
487 	case 't':
488 		Teleportation |= INTRINSIC;
489 		break;
490 	case 'W':
491 		pluslvl();
492 		break;
493 	case 'n':
494 		u.uhp = u.uhpmax;
495 		flags.botl = 1;
496 		/* fall into next case */
497 	case '@':
498 		pline("You cannibal! You will be sorry for this!");
499 		/* not tp++; */
500 		/* fall into next case */
501 	case 'd':
502 		Aggravate_monster |= INTRINSIC;
503 		break;
504 	case 'I':
505 		if (!Invis) {
506 			Invis = 50 + rn2(100);
507 			if (!See_invisible)
508 				newsym(u.ux, u.uy);
509 		} else {
510 			Invis |= INTRINSIC;
511 			See_invisible |= INTRINSIC;
512 		}
513 		/* fall into next case */
514 	case 'y':
515 #ifdef QUEST
516 		u.uhorizon++;
517 #endif	/* QUEST */
518 		/* fall into next case */
519 	case 'B':
520 		Confusion = 50;
521 		break;
522 	case 'D':
523 		Fire_resistance |= INTRINSIC;
524 		break;
525 	case 'E':
526 		Telepat |= INTRINSIC;
527 		break;
528 	case 'F':
529 	case 'Y':
530 		Cold_resistance |= INTRINSIC;
531 		break;
532 	case 'k':
533 	case 's':
534 		Poison_resistance |= INTRINSIC;
535 		break;
536 	case 'c':
537 		pline("You turn to stone.");
538 		killer = "dead cockatrice";
539 		done("died");
540 		/* NOTREACHED */
541 	case 'a':
542 		if (Stoned) {
543 			pline("What a pity - you just destroyed a future piece of art!");
544 			tp++;
545 			Stoned = 0;
546 		}
547 		break;
548 	case 'M':
549 		pline("You cannot resist the temptation to mimic a treasure chest.");
550 		tp++;
551 		nomul(-30);
552 		afternmv = Meatdone;
553 		nomovemsg = "You now again prefer mimicking a human.";
554 		u.usym = '$';
555 		prme();
556 		break;
557 	}
558 	return (tp);
559 }
560