xref: /netbsd-src/games/hack/hack.eat.c (revision 10ad5ffa714ce1a679dcc9dd8159648df2d67b5a)
1 /*	$NetBSD: hack.eat.c,v 1.8 2009/06/07 20:13:18 dholland 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.8 2009/06/07 20:13:18 dholland 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(void)
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(void)
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(void)
155 {
156 	u.usym = '@';
157 	prme();
158 	return 0;
159 }
160 
161 int
162 doeat(void)
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) snprintf(msgbuf, sizeof(msgbuf),
334 			       "You finished eating the %s.",
335 			       ftmp->oc_name);
336 		nomovemsg = msgbuf;
337 	}
338 	useup(otmp);
339 	return (1);
340 }
341 
342 /* called in hack.main.c */
343 void
344 gethungry(void)
345 {
346 	--u.uhunger;
347 	if (moves % 2) {
348 		if (Regeneration)
349 			u.uhunger--;
350 		if (Hunger)
351 			u.uhunger--;
352 		/*
353 		 * a3:  if(Hunger & LEFT_RING) u.uhunger--; if(Hunger &
354 		 * RIGHT_RING) u.uhunger--; etc.
355 		 */
356 	}
357 	if (moves % 20 == 0) {	/* jimt@asgb */
358 		if (uleft)
359 			u.uhunger--;
360 		if (uright)
361 			u.uhunger--;
362 	}
363 	newuhs(TRUE);
364 }
365 
366 /* called after vomiting and after performing feats of magic */
367 void
368 morehungry(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(int num)
377 {
378 	u.uhunger += num;
379 	newuhs(FALSE);
380 }
381 
382 int
383 unfaint(void)
384 {
385 	u.uhs = FAINTING;
386 	flags.botl = 1;
387 	return 0;
388 }
389 
390 void
391 newuhs(boolean incr)
392 {
393 	int             newhs, h = u.uhunger;
394 
395 	newhs = (h > 1000) ? SATIATED :
396 		(h > 150) ? NOT_HUNGRY :
397 		(h > 50) ? HUNGRY :
398 		(h > 0) ? WEAK : FAINTING;
399 
400 	if (newhs == FAINTING) {
401 		if (u.uhs == FAINTED)
402 			newhs = FAINTED;
403 		if (u.uhs <= WEAK || rn2(20 - u.uhunger / 10) >= 19) {
404 			if (u.uhs != FAINTED && multi >= 0 /* %% */ ) {
405 				pline("You faint from lack of food.");
406 				nomul(-10 + (u.uhunger / 10));
407 				nomovemsg = "You regain consciousness.";
408 				afternmv = unfaint;
409 				newhs = FAINTED;
410 			}
411 		} else if (u.uhunger < -(int) (200 + 25 * u.ulevel)) {
412 			u.uhs = STARVED;
413 			flags.botl = 1;
414 			bot();
415 			pline("You die from starvation.");
416 			done("starved");
417 		}
418 	}
419 	if (newhs != u.uhs) {
420 		if (newhs >= WEAK && u.uhs < WEAK)
421 			losestr(1);	/* this may kill you -- see below */
422 		else if (newhs < WEAK && u.uhs >= WEAK && u.ustr < u.ustrmax)
423 			losestr(-1);
424 		switch (newhs) {
425 		case HUNGRY:
426 			pline((!incr) ? "You only feel hungry now." :
427 			      (u.uhunger < 145) ? "You feel hungry." :
428 			      "You are beginning to feel hungry.");
429 			break;
430 		case WEAK:
431 			pline((!incr) ? "You feel weak now." :
432 			      (u.uhunger < 45) ? "You feel weak." :
433 			      "You are beginning to feel weak.");
434 			break;
435 		}
436 		u.uhs = newhs;
437 		flags.botl = 1;
438 		if (u.uhp < 1) {
439 			pline("You die from hunger and exhaustion.");
440 			killer = "exhaustion";
441 			done("starved");
442 		}
443 	}
444 }
445 
446 #define	CORPSE_I_TO_C(otyp)	(char) ((otyp >= DEAD_ACID_BLOB)\
447 		     ?  'a' + (otyp - DEAD_ACID_BLOB)\
448 		     :	'@' + (otyp - DEAD_HUMAN))
449 int
450 poisonous(struct obj *otmp)
451 {
452 	return (strchr(POISONOUS, CORPSE_I_TO_C(otmp->otyp)) != 0);
453 }
454 
455 /* returns 1 if some text was printed */
456 int
457 eatcorpse(struct obj *otmp)
458 {
459 	char            let = CORPSE_I_TO_C(otmp->otyp);
460 	int             tp = 0;
461 	if (let != 'a' && moves > otmp->age + 50 + rn2(100)) {
462 		tp++;
463 		pline("Ulch -- that meat was tainted!");
464 		pline("You get very sick.");
465 		Sick = 10 + rn2(10);
466 		u.usick_cause = objects[otmp->otyp].oc_name;
467 	} else if (strchr(POISONOUS, let) && rn2(5)) {
468 		tp++;
469 		pline("Ecch -- that must have been poisonous!");
470 		if (!Poison_resistance) {
471 			losestr(rnd(4));
472 			losehp(rnd(15), "poisonous corpse");
473 		} else
474 			pline("You don't seem affected by the poison.");
475 	} else if (strchr("ELNOPQRUuxz", let) && rn2(5)) {
476 		tp++;
477 		pline("You feel sick.");
478 		losehp(rnd(8), "cadaver");
479 	}
480 	switch (let) {
481 	case 'L':
482 	case 'N':
483 	case 't':
484 		Teleportation |= INTRINSIC;
485 		break;
486 	case 'W':
487 		pluslvl();
488 		break;
489 	case 'n':
490 		u.uhp = u.uhpmax;
491 		flags.botl = 1;
492 		/* fall into next case */
493 	case '@':
494 		pline("You cannibal! You will be sorry for this!");
495 		/* not tp++; */
496 		/* fall into next case */
497 	case 'd':
498 		Aggravate_monster |= INTRINSIC;
499 		break;
500 	case 'I':
501 		if (!Invis) {
502 			Invis = 50 + rn2(100);
503 			if (!See_invisible)
504 				newsym(u.ux, u.uy);
505 		} else {
506 			Invis |= INTRINSIC;
507 			See_invisible |= INTRINSIC;
508 		}
509 		/* fall into next case */
510 	case 'y':
511 #ifdef QUEST
512 		u.uhorizon++;
513 #endif	/* QUEST */
514 		/* fall into next case */
515 	case 'B':
516 		Confusion = 50;
517 		break;
518 	case 'D':
519 		Fire_resistance |= INTRINSIC;
520 		break;
521 	case 'E':
522 		Telepat |= INTRINSIC;
523 		break;
524 	case 'F':
525 	case 'Y':
526 		Cold_resistance |= INTRINSIC;
527 		break;
528 	case 'k':
529 	case 's':
530 		Poison_resistance |= INTRINSIC;
531 		break;
532 	case 'c':
533 		pline("You turn to stone.");
534 		killer = "dead cockatrice";
535 		done("died");
536 		/* NOTREACHED */
537 	case 'a':
538 		if (Stoned) {
539 			pline("What a pity - you just destroyed a future piece of art!");
540 			tp++;
541 			Stoned = 0;
542 		}
543 		break;
544 	case 'M':
545 		pline("You cannot resist the temptation to mimic a treasure chest.");
546 		tp++;
547 		nomul(-30);
548 		afternmv = Meatdone;
549 		nomovemsg = "You now again prefer mimicking a human.";
550 		u.usym = '$';
551 		prme();
552 		break;
553 	}
554 	return (tp);
555 }
556