xref: /dragonfly/games/hack/hack.invent.c (revision 1bf4b486)
1 /* Copyright (c) Stichting Mathematisch Centrum, Amsterdam, 1985. */
2 /* hack.invent.c - version 1.0.3 */
3 /* $FreeBSD: src/games/hack/hack.invent.c,v 1.4 1999/11/16 10:26:36 marcel Exp $ */
4 /* $DragonFly: src/games/hack/hack.invent.c,v 1.4 2005/05/22 03:37:05 y0netan1 Exp $ */
5 
6 #include	"hack.h"
7 #include	<stdio.h>
8 extern struct obj *splitobj();
9 extern struct obj zeroobj;
10 extern char morc;
11 extern char quitchars[];
12 static char *xprname();
13 
14 #ifndef NOWORM
15 #include	"def.wseg.h"
16 extern struct wseg *wsegs[32];
17 #endif /* NOWORM */
18 
19 #define	NOINVSYM	'#'
20 
21 static int lastinvnr = 51;	/* 0 ... 51 */
22 static
23 assigninvlet(otmp)
24 struct obj *otmp;
25 {
26 	boolean inuse[52];
27 	int i;
28 	struct obj *obj;
29 
30 	for(i = 0; i < 52; i++) inuse[i] = FALSE;
31 	for(obj = invent; obj; obj = obj->nobj) if(obj != otmp) {
32 		i = obj->invlet;
33 		if('a' <= i && i <= 'z') inuse[i - 'a'] = TRUE; else
34 		if('A' <= i && i <= 'Z') inuse[i - 'A' + 26] = TRUE;
35 		if(i == otmp->invlet) otmp->invlet = 0;
36 	}
37 	if((i = otmp->invlet) &&
38 	    (('a' <= i && i <= 'z') || ('A' <= i && i <= 'Z')))
39 		return;
40 	for(i = lastinvnr+1; i != lastinvnr; i++) {
41 		if(i == 52) { i = -1; continue; }
42 		if(!inuse[i]) break;
43 	}
44 	otmp->invlet = (inuse[i] ? NOINVSYM :
45 			(i < 26) ? ('a'+i) : ('A'+i-26));
46 	lastinvnr = i;
47 }
48 
49 struct obj *
50 addinv(obj)
51 struct obj *obj;
52 {
53 	struct obj *otmp;
54 
55 	/* merge or attach to end of chain */
56 	if(!invent) {
57 		invent = obj;
58 		otmp = 0;
59 	} else
60 	for(otmp = invent; /* otmp */; otmp = otmp->nobj) {
61 		if(merged(otmp, obj, 0))
62 			return(otmp);
63 		if(!otmp->nobj) {
64 			otmp->nobj = obj;
65 			break;
66 		}
67 	}
68 	obj->nobj = 0;
69 
70 	if(flags.invlet_constant) {
71 		assigninvlet(obj);
72 		/*
73 		 * The ordering of the chain is nowhere significant
74 		 * so in case you prefer some other order than the
75 		 * historical one, change the code below.
76 		 */
77 		if(otmp) {	/* find proper place in chain */
78 			otmp->nobj = 0;
79 			if((invent->invlet ^ 040) > (obj->invlet ^ 040)) {
80 				obj->nobj = invent;
81 				invent = obj;
82 			} else
83 			for(otmp = invent; ; otmp = otmp->nobj) {
84 			    if(!otmp->nobj ||
85 				(otmp->nobj->invlet ^ 040) > (obj->invlet ^ 040)){
86 				obj->nobj = otmp->nobj;
87 				otmp->nobj = obj;
88 				break;
89 			    }
90 			}
91 		}
92 	}
93 
94 	return(obj);
95 }
96 
97 useup(obj)
98 struct obj *obj;
99 {
100 	if(obj->quan > 1){
101 		obj->quan--;
102 		obj->owt = weight(obj);
103 	} else {
104 		setnotworn(obj);
105 		freeinv(obj);
106 		obfree(obj, (struct obj *) 0);
107 	}
108 }
109 
110 freeinv(obj)
111 struct obj *obj;
112 {
113 	struct obj *otmp;
114 
115 	if(obj == invent)
116 		invent = invent->nobj;
117 	else {
118 		for(otmp = invent; otmp->nobj != obj; otmp = otmp->nobj)
119 			if(!otmp->nobj) panic("freeinv");
120 		otmp->nobj = obj->nobj;
121 	}
122 }
123 
124 /* destroy object in fobj chain (if unpaid, it remains on the bill) */
125 delobj(obj) struct obj *obj; {
126 	freeobj(obj);
127 	unpobj(obj);
128 	obfree(obj, (struct obj *) 0);
129 }
130 
131 /* unlink obj from chain starting with fobj */
132 freeobj(obj) struct obj *obj; {
133 	struct obj *otmp;
134 
135 	if(obj == fobj) fobj = fobj->nobj;
136 	else {
137 		for(otmp = fobj; otmp->nobj != obj; otmp = otmp->nobj)
138 			if(!otmp) panic("error in freeobj");
139 		otmp->nobj = obj->nobj;
140 	}
141 }
142 
143 /* Note: freegold throws away its argument! */
144 freegold(gold) struct gold *gold; {
145 	struct gold *gtmp;
146 
147 	if(gold == fgold) fgold = gold->ngold;
148 	else {
149 		for(gtmp = fgold; gtmp->ngold != gold; gtmp = gtmp->ngold)
150 			if(!gtmp) panic("error in freegold");
151 		gtmp->ngold = gold->ngold;
152 	}
153 	free((char *) gold);
154 }
155 
156 deltrap(trap)
157 struct trap *trap;
158 {
159 	struct trap *ttmp;
160 
161 	if(trap == ftrap)
162 		ftrap = ftrap->ntrap;
163 	else {
164 		for(ttmp = ftrap; ttmp->ntrap != trap; ttmp = ttmp->ntrap) ;
165 		ttmp->ntrap = trap->ntrap;
166 	}
167 	free((char *) trap);
168 }
169 
170 struct wseg *m_atseg;
171 
172 struct monst *
173 m_at(x,y)
174 int x,y;
175 {
176 	struct monst *mtmp;
177 #ifndef NOWORM
178 	struct wseg *wtmp;
179 #endif /* NOWORM */
180 
181 	m_atseg = 0;
182 	for(mtmp = fmon; mtmp; mtmp = mtmp->nmon){
183 		if(mtmp->mx == x && mtmp->my == y)
184 			return(mtmp);
185 #ifndef NOWORM
186 		if(mtmp->wormno){
187 		    for(wtmp = wsegs[mtmp->wormno]; wtmp; wtmp = wtmp->nseg)
188 		    if(wtmp->wx == x && wtmp->wy == y){
189 			m_atseg = wtmp;
190 			return(mtmp);
191 		    }
192 		}
193 #endif /* NOWORM */
194 	}
195 	return(0);
196 }
197 
198 struct obj *
199 o_at(x,y)
200 int x,y;
201 {
202 	struct obj *otmp;
203 
204 	for(otmp = fobj; otmp; otmp = otmp->nobj)
205 		if(otmp->ox == x && otmp->oy == y) return(otmp);
206 	return(0);
207 }
208 
209 struct obj *
210 sobj_at(n,x,y)
211 int n,x,y;
212 {
213 	struct obj *otmp;
214 
215 	for(otmp = fobj; otmp; otmp = otmp->nobj)
216 		if(otmp->ox == x && otmp->oy == y && otmp->otyp == n)
217 			return(otmp);
218 	return(0);
219 }
220 
221 carried(obj) struct obj *obj; {
222 struct obj *otmp;
223 	for(otmp = invent; otmp; otmp = otmp->nobj)
224 		if(otmp == obj) return(1);
225 	return(0);
226 }
227 
228 carrying(type)
229 int type;
230 {
231 	struct obj *otmp;
232 
233 	for(otmp = invent; otmp; otmp = otmp->nobj)
234 		if(otmp->otyp == type)
235 			return(TRUE);
236 	return(FALSE);
237 }
238 
239 struct obj *
240 o_on(id, objchn) unsigned int id; struct obj *objchn; {
241 	while(objchn) {
242 		if(objchn->o_id == id) return(objchn);
243 		objchn = objchn->nobj;
244 	}
245 	return((struct obj *) 0);
246 }
247 
248 struct trap *
249 t_at(x,y)
250 int x,y;
251 {
252 	struct trap *trap = ftrap;
253 	while(trap) {
254 		if(trap->tx == x && trap->ty == y) return(trap);
255 		trap = trap->ntrap;
256 	}
257 	return(0);
258 }
259 
260 struct gold *
261 g_at(x,y)
262 int x,y;
263 {
264 	struct gold *gold = fgold;
265 	while(gold) {
266 		if(gold->gx == x && gold->gy == y) return(gold);
267 		gold = gold->ngold;
268 	}
269 	return(0);
270 }
271 
272 /* make dummy object structure containing gold - for temporary use only */
273 struct obj *
274 mkgoldobj(q)
275 long q;
276 {
277 	struct obj *otmp;
278 
279 	otmp = newobj(0);
280 	/* should set o_id etc. but otmp will be freed soon */
281 	otmp->olet = '$';
282 	u.ugold -= q;
283 	OGOLD(otmp) = q;
284 	flags.botl = 1;
285 	return(otmp);
286 }
287 
288 /*
289  * getobj returns:
290  *	struct obj *xxx:	object to do something with.
291  *	(struct obj *) 0	error return: no object.
292  *	&zeroobj		explicitly no object (as in w-).
293  */
294 struct obj *
295 getobj(let,word)
296 char *let,*word;
297 {
298 	struct obj *otmp;
299 	char ilet,ilet1,ilet2;
300 	char buf[BUFSZ];
301 	char lets[BUFSZ];
302 	int foo = 0, foo2;
303 	char *bp = buf;
304 	xchar allowcnt = 0;	/* 0, 1 or 2 */
305 	boolean allowgold = FALSE;
306 	boolean allowall = FALSE;
307 	boolean allownone = FALSE;
308 	xchar foox = 0;
309 	long cnt;
310 
311 	if(*let == '0') let++, allowcnt = 1;
312 	if(*let == '$') let++, allowgold = TRUE;
313 	if(*let == '#') let++, allowall = TRUE;
314 	if(*let == '-') let++, allownone = TRUE;
315 	if(allownone) *bp++ = '-';
316 	if(allowgold) *bp++ = '$';
317 	if(bp > buf && bp[-1] == '-') *bp++ = ' ';
318 
319 	ilet = 'a';
320 	for(otmp = invent; otmp; otmp = otmp->nobj){
321 	    if(!*let || index(let, otmp->olet)) {
322 		bp[foo++] = flags.invlet_constant ? otmp->invlet : ilet;
323 
324 		/* ugly check: remove inappropriate things */
325 		if((!strcmp(word, "take off") &&
326 		    !(otmp->owornmask & (W_ARMOR - W_ARM2)))
327 		|| (!strcmp(word, "wear") &&
328 		    (otmp->owornmask & (W_ARMOR | W_RING)))
329 		|| (!strcmp(word, "wield") &&
330 		    (otmp->owornmask & W_WEP))) {
331 			foo--;
332 			foox++;
333 		}
334 	    }
335 	    if(ilet == 'z') ilet = 'A'; else ilet++;
336 	}
337 	bp[foo] = 0;
338 	if(foo == 0 && bp > buf && bp[-1] == ' ') *--bp = 0;
339 	(void) strcpy(lets, bp);	/* necessary since we destroy buf */
340 	if(foo > 5) {			/* compactify string */
341 		foo = foo2 = 1;
342 		ilet2 = bp[0];
343 		ilet1 = bp[1];
344 		while(ilet = bp[++foo2] = bp[++foo]){
345 			if(ilet == ilet1+1){
346 				if(ilet1 == ilet2+1)
347 					bp[foo2 - 1] = ilet1 = '-';
348 				else if(ilet2 == '-') {
349 					bp[--foo2] = ++ilet1;
350 					continue;
351 				}
352 			}
353 			ilet2 = ilet1;
354 			ilet1 = ilet;
355 		}
356 	}
357 	if(!foo && !allowall && !allowgold && !allownone) {
358 		pline("You don't have anything %sto %s.",
359 			foox ? "else " : "", word);
360 		return(0);
361 	}
362 	for(;;) {
363 		if(!buf[0])
364 			pline("What do you want to %s [*]? ", word);
365 		else
366 			pline("What do you want to %s [%s or ?*]? ",
367 				word, buf);
368 
369 		cnt = 0;
370 		ilet = readchar();
371 		while(digit(ilet) && allowcnt) {
372 			if (cnt < 100000000)
373 			    cnt = 10*cnt + (ilet - '0');
374 			else
375 			    cnt = 999999999;
376 			allowcnt = 2;	/* signal presence of cnt */
377 			ilet = readchar();
378 		}
379 		if(digit(ilet)) {
380 			pline("No count allowed with this command.");
381 			continue;
382 		}
383 		if(index(quitchars,ilet))
384 			return((struct obj *)0);
385 		if(ilet == '-') {
386 			return(allownone ? &zeroobj : (struct obj *) 0);
387 		}
388 		if(ilet == '$') {
389 			if(!allowgold){
390 				pline("You cannot %s gold.", word);
391 				continue;
392 			}
393 			if(!(allowcnt == 2 && cnt < u.ugold))
394 				cnt = u.ugold;
395 			return(mkgoldobj(cnt));
396 		}
397 		if(ilet == '?') {
398 			doinv(lets);
399 			if(!(ilet = morc)) continue;
400 			/* he typed a letter (not a space) to more() */
401 		} else if(ilet == '*') {
402 			doinv((char *) 0);
403 			if(!(ilet = morc)) continue;
404 			/* ... */
405 		}
406 		if(flags.invlet_constant) {
407 			for(otmp = invent; otmp; otmp = otmp->nobj)
408 				if(otmp->invlet == ilet) break;
409 		} else {
410 			if(ilet >= 'A' && ilet <= 'Z') ilet += 'z'-'A'+1;
411 			ilet -= 'a';
412 			for(otmp = invent; otmp && ilet;
413 					ilet--, otmp = otmp->nobj) ;
414 		}
415 		if(!otmp) {
416 			pline("You don't have that object.");
417 			continue;
418 		}
419 		if(cnt < 0 || otmp->quan < cnt) {
420 			pline("You don't have that many! [You have %u]"
421 			, otmp->quan);
422 			continue;
423 		}
424 		break;
425 	}
426 	if(!allowall && let && !index(let,otmp->olet)) {
427 		pline("That is a silly thing to %s.",word);
428 		return(0);
429 	}
430 	if(allowcnt == 2) {	/* cnt given */
431 		if(cnt == 0) return(0);
432 		if(cnt != otmp->quan) {
433 			struct obj *obj;
434 			obj = splitobj(otmp, (int) cnt);
435 			if(otmp == uwep) setuwep(obj);
436 		}
437 	}
438 	return(otmp);
439 }
440 
441 ckunpaid(otmp) struct obj *otmp; {
442 	return( otmp->unpaid );
443 }
444 
445 /* interactive version of getobj - used for Drop and Identify */
446 /* return the number of times fn was called successfully */
447 ggetobj(word, fn, max)
448 char *word;
449 int (*fn)(),  max;
450 {
451 char buf[BUFSZ];
452 char *ip;
453 char sym;
454 int oletct = 0, iletct = 0;
455 boolean allflag = FALSE;
456 char olets[20], ilets[20];
457 int (*ckfn)() = (int (*)()) 0;
458 xchar allowgold = (u.ugold && !strcmp(word, "drop")) ? 1 : 0;	/* BAH */
459 	if(!invent && !allowgold){
460 		pline("You have nothing to %s.", word);
461 		return(0);
462 	} else {
463 		struct obj *otmp = invent;
464 		int uflg = 0;
465 
466 		if(allowgold) ilets[iletct++] = '$';
467 		ilets[iletct] = 0;
468 		while(otmp) {
469 			if(!index(ilets, otmp->olet)){
470 				ilets[iletct++] = otmp->olet;
471 				ilets[iletct] = 0;
472 			}
473 			if(otmp->unpaid) uflg = 1;
474 			otmp = otmp->nobj;
475 		}
476 		ilets[iletct++] = ' ';
477 		if(uflg) ilets[iletct++] = 'u';
478 		if(invent) ilets[iletct++] = 'a';
479 		ilets[iletct] = 0;
480 	}
481 	pline("What kinds of thing do you want to %s? [%s] ",
482 		word, ilets);
483 	getlin(buf);
484 	if(buf[0] == '\033') {
485 		clrlin();
486 		return(0);
487 	}
488 	ip = buf;
489 	olets[0] = 0;
490 	while(sym = *ip++){
491 		if(sym == ' ') continue;
492 		if(sym == '$') {
493 			if(allowgold == 1)
494 				(*fn)(mkgoldobj(u.ugold));
495 			else if(!u.ugold)
496 				pline("You have no gold.");
497 			allowgold = 2;
498 		} else
499 		if(sym == 'a' || sym == 'A') allflag = TRUE; else
500 		if(sym == 'u' || sym == 'U') ckfn = ckunpaid; else
501 		if(index("!%?[()=*/\"0", sym)){
502 			if(!index(olets, sym)){
503 				olets[oletct++] = sym;
504 				olets[oletct] = 0;
505 			}
506 		}
507 		else pline("You don't have any %c's.", sym);
508 	}
509 	if(allowgold == 2 && !oletct)
510 		return(1);	/* he dropped gold (or at least tried to) */
511 	else
512 		return(askchain(invent, olets, allflag, fn, ckfn, max));
513 }
514 
515 /*
516  * Walk through the chain starting at objchn and ask for all objects
517  * with olet in olets (if nonNULL) and satisfying ckfn (if nonNULL)
518  * whether the action in question (i.e., fn) has to be performed.
519  * If allflag then no questions are asked. Max gives the max nr of
520  * objects to be treated. Return the number of objects treated.
521  */
522 askchain(objchn, olets, allflag, fn, ckfn, max)
523 struct obj *objchn;
524 char *olets;
525 int allflag;
526 int (*fn)(), (*ckfn)();
527 int max;
528 {
529 struct obj *otmp, *otmp2;
530 char sym, ilet;
531 int cnt = 0;
532 	ilet = 'a'-1;
533 	for(otmp = objchn; otmp; otmp = otmp2){
534 		if(ilet == 'z') ilet = 'A'; else ilet++;
535 		otmp2 = otmp->nobj;
536 		if(olets && *olets && !index(olets, otmp->olet)) continue;
537 		if(ckfn && !(*ckfn)(otmp)) continue;
538 		if(!allflag) {
539 			pline(xprname(otmp, ilet));
540 			addtopl(" [nyaq]? ");
541 			sym = readchar();
542 		}
543 		else	sym = 'y';
544 
545 		switch(sym){
546 		case 'a':
547 			allflag = 1;
548 		case 'y':
549 			cnt += (*fn)(otmp);
550 			if(--max == 0) goto ret;
551 		case 'n':
552 		default:
553 			break;
554 		case 'q':
555 			goto ret;
556 		}
557 	}
558 	pline(cnt ? "That was all." : "No applicable objects.");
559 ret:
560 	return(cnt);
561 }
562 
563 obj_to_let(obj)	/* should of course only be called for things in invent */
564 struct obj *obj;
565 {
566 	struct obj *otmp;
567 	char ilet;
568 
569 	if(flags.invlet_constant)
570 		return(obj->invlet);
571 	ilet = 'a';
572 	for(otmp = invent; otmp && otmp != obj; otmp = otmp->nobj)
573 		if(++ilet > 'z') ilet = 'A';
574 	return(otmp ? ilet : NOINVSYM);
575 }
576 
577 prinv(obj)
578 struct obj *obj;
579 {
580 	pline(xprname(obj, obj_to_let(obj)));
581 }
582 
583 static char *
584 xprname(obj,let)
585 struct obj *obj;
586 char let;
587 {
588 	static char li[BUFSZ];
589 
590 	(void) sprintf(li, "%c - %s.",
591 		flags.invlet_constant ? obj->invlet : let,
592 		doname(obj));
593 	return(li);
594 }
595 
596 ddoinv()
597 {
598 	doinv((char *) 0);
599 	return(0);
600 }
601 
602 /* called with 0 or "": all objects in inventory */
603 /* otherwise: all objects with (serial) letter in lets */
604 doinv(lets)
605 char *lets;
606 {
607 	struct obj *otmp;
608 	char ilet;
609 	int ct = 0;
610 	char any[BUFSZ];
611 
612 	morc = 0;		/* just to be sure */
613 
614 	if(!invent){
615 		pline("Not carrying anything.");
616 		return;
617 	}
618 
619 	cornline(0, (char *) 0);
620 	ilet = 'a';
621 	for(otmp = invent; otmp; otmp = otmp->nobj) {
622 	    if(flags.invlet_constant) ilet = otmp->invlet;
623 	    if(!lets || !*lets || index(lets, ilet)) {
624 		    cornline(1, xprname(otmp, ilet));
625 		    any[ct++] = ilet;
626 	    }
627 	    if(!flags.invlet_constant) if(++ilet > 'z') ilet = 'A';
628 	}
629 	any[ct] = 0;
630 	cornline(2, any);
631 }
632 
633 dotypeinv ()				/* free after Robert Viduya */
634 /* Changed to one type only, so he doesnt have to type cr */
635 {
636     char c, ilet;
637     char stuff[BUFSZ];
638     int stct;
639     struct obj *otmp;
640     boolean billx = inshop() && doinvbill(0);
641     boolean unpd = FALSE;
642 
643 	if (!invent && !u.ugold && !billx) {
644 	    pline ("You aren't carrying anything.");
645 	    return(0);
646 	}
647 
648 	stct = 0;
649 	if(u.ugold) stuff[stct++] = '$';
650 	stuff[stct] = 0;
651 	for(otmp = invent; otmp; otmp = otmp->nobj) {
652 	    if (!index (stuff, otmp->olet)) {
653 		stuff[stct++] = otmp->olet;
654 		stuff[stct] = 0;
655 	    }
656 	    if(otmp->unpaid)
657 		unpd = TRUE;
658 	}
659 	if(unpd) stuff[stct++] = 'u';
660 	if(billx) stuff[stct++] = 'x';
661 	stuff[stct] = 0;
662 
663 	if(stct > 1) {
664 	    pline ("What type of object [%s] do you want an inventory of? ",
665 		stuff);
666 	    c = readchar();
667 	    if(index(quitchars,c)) return(0);
668 	} else
669 	    c = stuff[0];
670 
671 	if(c == '$')
672 	    return(doprgold());
673 
674 	if(c == 'x' || c == 'X') {
675 	    if(billx)
676 		(void) doinvbill(1);
677 	    else
678 		pline("No used-up objects on the shopping bill.");
679 	    return(0);
680 	}
681 
682 	if((c == 'u' || c == 'U') && !unpd) {
683 		pline("You are not carrying any unpaid objects.");
684 		return(0);
685 	}
686 
687 	stct = 0;
688 	ilet = 'a';
689 	for (otmp = invent; otmp; otmp = otmp -> nobj) {
690 	    if(flags.invlet_constant) ilet = otmp->invlet;
691 	    if (c == otmp -> olet || (c == 'u' && otmp -> unpaid))
692 		stuff[stct++] = ilet;
693 	    if(!flags.invlet_constant) if(++ilet > 'z') ilet = 'A';
694 	}
695 	stuff[stct] = '\0';
696 	if(stct == 0)
697 		pline("You have no such objects.");
698 	else
699 		doinv (stuff);
700 
701 	return(0);
702 }
703 
704 /* look at what is here */
705 dolook() {
706     struct obj *otmp, *otmp0;
707     struct gold *gold;
708     const char *verb = Blind ? "feel" : "see";
709     int	ct = 0;
710 
711     if(!u.uswallow) {
712 	if(Blind) {
713 	    pline("You try to feel what is lying here on the floor.");
714 	    if(Levitation) {				/* ab@unido */
715 		pline("You cannot reach the floor!");
716 		return(1);
717 	    }
718 	}
719 	otmp0 = o_at(u.ux, u.uy);
720 	gold = g_at(u.ux, u.uy);
721     }
722 
723     if(u.uswallow || (!otmp0 && !gold)) {
724 	pline("You %s no objects here.", verb);
725 	return(!!Blind);
726     }
727 
728     cornline(0, "Things that are here:");
729     for(otmp = otmp0; otmp; otmp = otmp->nobj) {
730 	if(otmp->ox == u.ux && otmp->oy == u.uy) {
731 	    ct++;
732 	    cornline(1, doname(otmp));
733 	    if(Blind && otmp->otyp == DEAD_COCKATRICE && !uarmg) {
734 		pline("Touching the dead cockatrice is a fatal mistake ...");
735 		pline("You die ...");
736 		killer = "dead cockatrice";
737 		done("died");
738 	    }
739 	}
740     }
741 
742     if(gold) {
743 	char gbuf[30];
744 
745 	(void) sprintf(gbuf, "%ld gold piece%s",
746 		gold->amount, plur(gold->amount));
747 	if(!ct++)
748 	    pline("You %s here %s.", verb, gbuf);
749 	else
750 	    cornline(1, gbuf);
751     }
752 
753     if(ct == 1 && !gold) {
754 	pline("You %s here %s.", verb, doname(otmp0));
755 	cornline(3, (char *) 0);
756     }
757     if(ct > 1)
758 	cornline(2, (char *) 0);
759     return(!!Blind);
760 }
761 
762 stackobj(obj) struct obj *obj; {
763 struct obj *otmp = fobj;
764 	for(otmp = fobj; otmp; otmp = otmp->nobj) if(otmp != obj)
765 	if(otmp->ox == obj->ox && otmp->oy == obj->oy &&
766 		merged(obj,otmp,1))
767 			return;
768 }
769 
770 /* merge obj with otmp and delete obj if types agree */
771 merged(otmp,obj,lose) struct obj *otmp, *obj; {
772 	if(obj->otyp == otmp->otyp &&
773 	  obj->unpaid == otmp->unpaid &&
774 	  obj->spe == otmp->spe &&
775 	  obj->dknown == otmp->dknown &&
776 	  obj->cursed == otmp->cursed &&
777 	  (index("%*?!", obj->olet) ||
778 	    (obj->known == otmp->known &&
779 		(obj->olet == WEAPON_SYM && obj->otyp < BOOMERANG)))) {
780 		otmp->quan += obj->quan;
781 		otmp->owt += obj->owt;
782 		if(lose) freeobj(obj);
783 		obfree(obj,otmp);	/* free(obj), bill->otmp */
784 		return(1);
785 	} else	return(0);
786 }
787 
788 /*
789  * Gold is no longer displayed; in fact, when you have a lot of money,
790  * it may take a while before you have counted it all.
791  * [Bug: d$ and pickup still tell you how much it was.]
792  */
793 extern int (*occupation)();
794 static long goldcounted;
795 
796 countgold(){
797 	if((goldcounted += 100*(u.ulevel + 1)) >= u.ugold) {
798 		long eps = 0;
799 		if(!rn2(2)) eps = rnd((int) (u.ugold/100 + 1));
800 		pline("You probably have about %ld gold pieces.",
801 			u.ugold + eps);
802 		return(0);	/* done */
803 	}
804 	return(1);		/* continue */
805 }
806 
807 doprgold(){
808 	if(!u.ugold)
809 		pline("You do not carry any gold.");
810 	else if(u.ugold <= 500)
811 		pline("You are carrying %ld gold pieces.", u.ugold);
812 	else {
813 		pline("You sit down in order to count your gold pieces.");
814 		goldcounted = 500;
815 		occupation = countgold;
816 		occtxt = "counting your gold";
817 	}
818 	return(1);
819 }
820 
821 /* --- end of gold counting section --- */
822 
823 doprwep(){
824 	if(!uwep) pline("You are empty handed.");
825 	else prinv(uwep);
826 	return(0);
827 }
828 
829 doprarm(){
830 	if(!uarm && !uarmg && !uarms && !uarmh)
831 		pline("You are not wearing any armor.");
832 	else {
833 		char lets[6];
834 		int ct = 0;
835 
836 		if(uarm) lets[ct++] = obj_to_let(uarm);
837 		if(uarm2) lets[ct++] = obj_to_let(uarm2);
838 		if(uarmh) lets[ct++] = obj_to_let(uarmh);
839 		if(uarms) lets[ct++] = obj_to_let(uarms);
840 		if(uarmg) lets[ct++] = obj_to_let(uarmg);
841 		lets[ct] = 0;
842 		doinv(lets);
843 	}
844 	return(0);
845 }
846 
847 doprring(){
848 	if(!uleft && !uright)
849 		pline("You are not wearing any rings.");
850 	else {
851 		char lets[3];
852 		int ct = 0;
853 
854 		if(uleft) lets[ct++] = obj_to_let(uleft);
855 		if(uright) lets[ct++] = obj_to_let(uright);
856 		lets[ct] = 0;
857 		doinv(lets);
858 	}
859 	return(0);
860 }
861 
862 digit(c) char c; {
863 	return(c >= '0' && c <= '9');
864 }
865