xref: /csrg-svn/games/cribbage/io.c (revision 7934)
17872Sarnold # include	<curses.h>
27872Sarnold # include	<ctype.h>
37872Sarnold # include	<unctrl.h>
47872Sarnold # include	"deck.h"
57872Sarnold # include	"cribbage.h"
67709Sarnold 
77872Sarnold # define	LINESIZE		128
87709Sarnold 
97872Sarnold # define	CTRL(X)			('X' - 'A' + 1)
107709Sarnold 
117872Sarnold # ifndef	attron
127872Sarnold #	define	erasechar()	_tty.sg_erase
137872Sarnold #	define	killchar()	_tty.sg_kill
147872Sarnold # endif		attron
157872Sarnold 
167709Sarnold char		linebuf[ LINESIZE ];
177709Sarnold 
187709Sarnold char		*rankname[ RANKS ]	= { "ACE", "TWO", "THREE", "FOUR",
197872Sarnold 					    "FIVE", "SIX", "SEVEN", "EIGHT",
207872Sarnold 					    "NINE", "TEN", "JACK", "QUEEN",
217872Sarnold 					    "KING" };
227709Sarnold 
237709Sarnold char            *rankchar[ RANKS ]      = { "A", "2", "3", "4", "5", "6", "7",
247872Sarnold 					    "8", "9", "T", "J", "Q", "K" };
257709Sarnold 
267709Sarnold char            *suitname[ SUITS ]      = { "SPADES", "HEARTS", "DIAMONDS",
277872Sarnold 					    "CLUBS" };
287709Sarnold 
297709Sarnold char            *suitchar[ SUITS ]      = { "S", "H", "D", "C" };
307709Sarnold 
317709Sarnold 
327709Sarnold 
337709Sarnold /*
347872Sarnold  * msgcard:
357872Sarnold  *	Call msgcrd in one of two forms
367709Sarnold  */
377872Sarnold msgcard(c, brief)
387872Sarnold CARD		c;
397872Sarnold BOOLEAN		brief;
407709Sarnold {
417872Sarnold 	if (brief)
427872Sarnold 		return msgcrd(c, TRUE, (char *) NULL, TRUE);
437872Sarnold 	else
447872Sarnold 		return msgcrd(c, FALSE, " of ", FALSE);
457709Sarnold }
467709Sarnold 
477709Sarnold 
487709Sarnold 
497709Sarnold /*
507872Sarnold  * msgcrd:
517872Sarnold  *	Print the value of a card in ascii
527709Sarnold  */
537872Sarnold msgcrd(c, brfrank, mid, brfsuit)
547872Sarnold CARD		c;
557872Sarnold char		*mid;
567872Sarnold BOOLEAN		brfrank,  brfsuit;
577872Sarnold {
587872Sarnold 	if (c.rank == EMPTY || c.suit == EMPTY)
597872Sarnold 	    return FALSE;
607872Sarnold 	if (brfrank)
617872Sarnold 	    addmsg("%1.1s", rankchar[c.rank]);
627872Sarnold 	else
637872Sarnold 	    addmsg(rankname[c.rank]);
647872Sarnold 	if (mid != NULL)
657872Sarnold 	    addmsg(mid);
667872Sarnold 	if (brfsuit)
677872Sarnold 	    addmsg("%1.1s", suitchar[c.suit]);
687872Sarnold 	else
697872Sarnold 	    addmsg(suitname[c.suit]);
707872Sarnold 	return TRUE;
717872Sarnold }
727709Sarnold 
737872Sarnold /*
747872Sarnold  * printcard:
757872Sarnold  *	Print out a card.
767872Sarnold  */
777872Sarnold printcard(win, cardno, c)
787872Sarnold WINDOW		*win;
797872Sarnold int		cardno;
807872Sarnold CARD		c;
817872Sarnold {
827872Sarnold 	prcard(win, cardno * 2, cardno, c);
837872Sarnold }
847709Sarnold 
857872Sarnold /*
867872Sarnold  * prcard:
877872Sarnold  *	Print out a card on the window at the specified location
887872Sarnold  */
897872Sarnold prcard(win, y, x, c)
907872Sarnold WINDOW		*win;
917872Sarnold int		y, x;
927872Sarnold CARD		c;
937709Sarnold {
947872Sarnold 	if (c.rank == EMPTY)
957872Sarnold 	    return;
967872Sarnold 	mvwaddstr(win, y + 0, x, "+-----+");
977872Sarnold 	mvwaddstr(win, y + 1, x, "|     |");
987872Sarnold 	mvwaddstr(win, y + 2, x, "|     |");
997872Sarnold 	mvwaddstr(win, y + 3, x, "|     |");
1007872Sarnold 	mvwaddstr(win, y + 4, x, "+-----+");
1017872Sarnold 	mvwaddch(win, y + 1, x + 1, rankchar[c.rank][0]);
1027872Sarnold 	waddch(win, suitchar[c.suit][0]);
1037872Sarnold 	mvwaddch(win, y + 3, x + 4, rankchar[c.rank][0]);
1047872Sarnold 	waddch(win, suitchar[c.suit][0]);
1057709Sarnold }
1067709Sarnold 
1077709Sarnold /*
1087872Sarnold  * prhand:
1097872Sarnold  *	Print a hand of n cards
1107709Sarnold  */
1117872Sarnold prhand(h, n, win)
1127872Sarnold CARD		h[];
1137872Sarnold int		n;
1147872Sarnold WINDOW		*win;
1157709Sarnold {
1167872Sarnold 	register int	i;
1177709Sarnold 
1187872Sarnold 	werase(win);
1197872Sarnold 	for (i = 0; i < n; i++)
1207872Sarnold 	    printcard(win, i, h[i]);
1217872Sarnold 	wrefresh(win);
1227709Sarnold }
1237709Sarnold 
1247709Sarnold 
1257709Sarnold 
1267709Sarnold /*
1277872Sarnold  * infrom:
1287872Sarnold  *	reads a card, supposedly in hand, accepting unambigous brief
1297872Sarnold  *	input, returns the index of the card found...
1307709Sarnold  */
1317872Sarnold infrom(hand, n, prompt)
1327872Sarnold CARD		hand[];
1337872Sarnold int		n;
1347872Sarnold char		*prompt;
1357709Sarnold {
1367872Sarnold 	register int           i, j;
1377872Sarnold 	CARD                    crd;
1387709Sarnold 
1397872Sarnold 	if (n < 1) {
1407872Sarnold 	    printf("\nINFROM: %d = n < 1!!\n", n);
1417872Sarnold 	    exit(74);
1427872Sarnold 	}
1437872Sarnold 	for (;;) {
1447872Sarnold 	    if (incard(&crd)) {			/* if card is full card */
1457872Sarnold 		if (!isone(crd, hand, n))
1467872Sarnold 		    msg("That's not in your hand");
1477872Sarnold 		else {
1487872Sarnold 		    for (i = 0; i < n; i++)
1497872Sarnold 			if (hand[i].rank == crd.rank &&
1507872Sarnold 			    hand[i].suit == crd.suit)
1517872Sarnold 				break;
1527872Sarnold 		    if (i >= n) {
1537872Sarnold 			printf("\nINFROM: isone or something messed up\n");
1547872Sarnold 			exit(77);
1557872Sarnold 		    }
1567872Sarnold 		    return i;
1577872Sarnold 		}
1587872Sarnold 	    }
1597872Sarnold 	    else				/* if not full card... */
1607872Sarnold 		if (crd.rank != EMPTY) {
1617872Sarnold 		    for (i = 0; i < n; i++)
1627872Sarnold 			if (hand[i].rank == crd.rank)
1637872Sarnold 				break;
1647872Sarnold 		    if (i >= n)
1657872Sarnold 			msg("No such rank in your hand");
1667872Sarnold 		    else {
1677872Sarnold 			for (j = i + 1; j < n; j++)
1687872Sarnold 			    if (hand[j].rank == crd.rank)
1697872Sarnold 				break;
1707872Sarnold 			if (j < n)
1717872Sarnold 			    msg("Ambiguous rank");
1727872Sarnold 			else
1737872Sarnold 			    return i;
1747872Sarnold 		    }
1757872Sarnold 		}
1767872Sarnold 		else
1777872Sarnold 		    msg("Sorry, I missed that");
1787872Sarnold 	    msg(prompt);
1797872Sarnold 	}
1807872Sarnold 	/* NOTREACHED */
1817709Sarnold }
1827709Sarnold 
1837709Sarnold 
1847709Sarnold 
1857709Sarnold /*
1867872Sarnold  * incard:
1877872Sarnold  *	Inputs a card in any format.  It reads a line ending with a CR
1887872Sarnold  *	and then parses it.
1897709Sarnold  */
1907872Sarnold incard(crd)
1917872Sarnold CARD		*crd;
1927709Sarnold {
1937872Sarnold 	char		*getline();
1947872Sarnold 	register int	i;
1957872Sarnold 	int		rnk, sut;
1967872Sarnold 	char		*line, *p, *p1;
1977872Sarnold 	BOOLEAN		retval;
1987709Sarnold 
1997872Sarnold 	retval = FALSE;
2007872Sarnold 	rnk = sut = EMPTY;
2017872Sarnold 	if (!(line = getline()))
2027872Sarnold 		goto gotit;
2037872Sarnold 	p = p1 = line;
2047872Sarnold 	while(  *p1 != ' '  &&  *p1 != NULL  )  ++p1;
2057872Sarnold 	*p1++ = NULL;
2067872Sarnold 	if(  *p == NULL  )  goto  gotit;
2077872Sarnold 			/* IMPORTANT: no real card has 2 char first name */
2087872Sarnold 	if(  strlen(p) == 2  )  {               /* check for short form */
2097872Sarnold 	    rnk = EMPTY;
2107872Sarnold 	    for( i = 0; i < RANKS; i++ )  {
2117872Sarnold 		if(  *p == *rankchar[i]  )  {
2127872Sarnold 		    rnk = i;
2137872Sarnold 		    break;
2147872Sarnold 		}
2157872Sarnold 	    }
2167872Sarnold 	    if(  rnk == EMPTY  )  goto  gotit;     /* it's nothing... */
2177872Sarnold 	    ++p;                                /* advance to next char */
2187872Sarnold 	    sut = EMPTY;
2197872Sarnold 	    for( i = 0; i < SUITS; i++ )  {
2207872Sarnold 		if(  *p == *suitchar[i]  )  {
2217872Sarnold 		    sut = i;
2227872Sarnold 		    break;
2237872Sarnold 		}
2247872Sarnold 	    }
2257872Sarnold 	    if(  sut != EMPTY  )  retval = TRUE;
2267872Sarnold 	    goto  gotit;
2277872Sarnold 	}
2287872Sarnold 	rnk = EMPTY;
2297872Sarnold 	for( i = 0; i < RANKS; i++ )  {
2307872Sarnold 	    if(  !strcmp( p, rankname[i] )  ||  !strcmp( p, rankchar[i] )  )  {
2317872Sarnold 		rnk = i;
2327872Sarnold 		break;
2337872Sarnold 	    }
2347872Sarnold 	}
2357872Sarnold 	if(  rnk == EMPTY  )  goto  gotit;
2367872Sarnold 	p = p1;
2377872Sarnold 	while(  *p1 != ' '  &&  *p1 != NULL  )  ++p1;
2387872Sarnold 	*p1++ = NULL;
2397872Sarnold 	if(  *p == NULL  )  goto  gotit;
2407872Sarnold 	if(  !strcmp( "OF", p )  )  {
2417872Sarnold 	    p = p1;
2427872Sarnold 	    while(  *p1 != ' '  &&  *p1 != NULL  )  ++p1;
2437872Sarnold 	    *p1++ = NULL;
2447872Sarnold 	    if(  *p == NULL  )  goto  gotit;
2457872Sarnold 	}
2467872Sarnold 	sut = EMPTY;
2477872Sarnold 	for( i = 0; i < SUITS; i++ )  {
2487872Sarnold 	    if(  !strcmp( p, suitname[i] )  ||  !strcmp( p, suitchar[i] )  )  {
2497872Sarnold 		sut = i;
2507872Sarnold 		break;
2517872Sarnold 	    }
2527872Sarnold 	}
2537872Sarnold 	if(  sut != EMPTY  )  retval = TRUE;
2547709Sarnold gotit:
2557872Sarnold 	(*crd).rank = rnk;
2567872Sarnold 	(*crd).suit = sut;
2577872Sarnold 	return( retval );
2587709Sarnold }
2597709Sarnold 
2607709Sarnold 
2617709Sarnold 
2627709Sarnold /*
2637872Sarnold  * getuchar:
2647872Sarnold  *	Reads and converts to upper case
2657709Sarnold  */
2667709Sarnold getuchar()
2677709Sarnold {
2687872Sarnold 	register int		c;
2697709Sarnold 
2707872Sarnold 	c = readchar();
2717872Sarnold 	if (islower(c))
2727872Sarnold 	    c = toupper(c);
2737872Sarnold 	addch(c);
2747872Sarnold 	return c;
2757709Sarnold }
2767709Sarnold 
2777709Sarnold /*
278*7934Sarnold  * number:
279*7934Sarnold  *	Reads in a decimal number and makes sure it is between "lo" and
280*7934Sarnold  *	"hi" inclusive.
2817709Sarnold  */
282*7934Sarnold number(lo, hi, prompt)
283*7934Sarnold int		lo, hi;
284*7934Sarnold char		*prompt;
2857709Sarnold {
286*7934Sarnold 	char			*getline();
287*7934Sarnold 	register char		*p;
288*7934Sarnold 	register int		sum;
2897709Sarnold 
2907872Sarnold 	sum = 0;
291*7934Sarnold 	for (;;) {
292*7934Sarnold 	    msg(prompt);
293*7934Sarnold 	    if(!(p = getline()) || *p == NULL) {
294*7934Sarnold 		msg(quiet ? "Not a number" : "That doesn't look like a number");
295*7934Sarnold 		continue;
296*7934Sarnold 	    }
2977872Sarnold 	    sum = 0;
298*7934Sarnold 
299*7934Sarnold 	    if (!isdigit(*p))
3007872Sarnold 		sum = lo - 1;
301*7934Sarnold 	    else
302*7934Sarnold 		while (isdigit(*p)) {
303*7934Sarnold 		    sum = 10 * sum + (*p - '0');
3047872Sarnold 		    ++p;
305*7934Sarnold 		}
306*7934Sarnold 
307*7934Sarnold 	    if (*p != ' ' && *p != '\t' && *p != NULL)
308*7934Sarnold 		sum = lo - 1;
309*7934Sarnold 	    if (sum >= lo && sum <= hi)
310*7934Sarnold 		return sum;
311*7934Sarnold 	    if (sum == lo - 1)
312*7934Sarnold 		msg("that doesn't look like a number, try again --> ");
313*7934Sarnold 	    else
314*7934Sarnold 		msg("%d is not between %d and %d inclusive, try again --> ",
315*7934Sarnold 								sum, lo, hi);
316*7934Sarnold 	}
3177709Sarnold }
3187709Sarnold 
3197872Sarnold /*
3207872Sarnold  * msg:
3217872Sarnold  *	Display a message at the top of the screen.
3227872Sarnold  */
3237872Sarnold char		Msgbuf[BUFSIZ] = "";
3247709Sarnold 
3257872Sarnold int		Mpos = 0;
3267709Sarnold 
3277872Sarnold static int	Newpos = 0;
3287872Sarnold 
3297872Sarnold /* VARARGS1 */
3307872Sarnold msg(fmt, args)
3317872Sarnold char	*fmt;
3327872Sarnold int	args;
3337872Sarnold {
3347872Sarnold     /*
3357872Sarnold      * if the string is "", just clear the line
3367872Sarnold      */
3377872Sarnold     if (*fmt == '\0') {
3387872Sarnold 	move(LINES - 1, 0);
3397872Sarnold 	clrtoeol();
3407872Sarnold 	Mpos = 0;
3417872Sarnold 	Hasread = TRUE;
3427872Sarnold 	return;
3437872Sarnold     }
3447872Sarnold     /*
3457872Sarnold      * otherwise add to the message and flush it out
3467872Sarnold      */
3477872Sarnold     doadd(fmt, &args);
3487872Sarnold     endmsg();
3497872Sarnold }
3507872Sarnold 
3517709Sarnold /*
3527872Sarnold  * addmsg:
3537872Sarnold  *	Add things to the current message
3547709Sarnold  */
3557872Sarnold /* VARARGS1 */
3567872Sarnold addmsg(fmt, args)
3577872Sarnold char	*fmt;
3587872Sarnold int	args;
3597872Sarnold {
3607872Sarnold     doadd(fmt, &args);
3617872Sarnold }
3627709Sarnold 
3637872Sarnold /*
3647872Sarnold  * endmsg:
3657872Sarnold  *	Display a new msg (giving him a chance to see the previous one
3667872Sarnold  *	if it is up there with the --More--)
3677872Sarnold  */
3687872Sarnold endmsg()
3697709Sarnold {
3707872Sarnold     if (!Hasread) {
3717872Sarnold 	move(LINES - 1, Mpos);
3727872Sarnold 	addstr("--More--");
3737872Sarnold 	refresh();
3747872Sarnold 	wait_for(' ');
3757872Sarnold     }
3767872Sarnold     /*
3777872Sarnold      * All messages should start with uppercase, except ones that
3787872Sarnold      * start with a pack addressing character
3797872Sarnold      */
3807872Sarnold     if (islower(Msgbuf[0]) && Msgbuf[1] != ')')
3817872Sarnold 	Msgbuf[0] = toupper(Msgbuf[0]);
3827872Sarnold     mvaddstr(LINES - 1, 0, Msgbuf);
3837872Sarnold     clrtoeol();
3847872Sarnold     Mpos = Newpos;
3857872Sarnold     Newpos = 0;
3867872Sarnold     refresh();
3877872Sarnold     Hasread = FALSE;
3887872Sarnold }
3897709Sarnold 
3907872Sarnold /*
3917872Sarnold  * doadd:
3927872Sarnold  *	Perform an add onto the message buffer
3937872Sarnold  */
3947872Sarnold doadd(fmt, args)
3957872Sarnold char	*fmt;
3967872Sarnold int	*args;
3977872Sarnold {
3987872Sarnold     static FILE	junk;
3997872Sarnold 
4007872Sarnold     /*
4017872Sarnold      * Do the printf into Msgbuf
4027872Sarnold      */
4037872Sarnold     junk._flag = _IOWRT + _IOSTRG;
4047872Sarnold     junk._ptr = &Msgbuf[Newpos];
4057872Sarnold     junk._cnt = 32767;
4067872Sarnold     _doprnt(fmt, args, &junk);
4077872Sarnold     putc('\0', &junk);
4087872Sarnold     Newpos = strlen(Msgbuf);
4097709Sarnold }
4107709Sarnold 
4117872Sarnold /*
4127872Sarnold  * wait_for
4137872Sarnold  *	Sit around until the guy types the right key
4147872Sarnold  */
4157872Sarnold wait_for(ch)
4167872Sarnold register char	ch;
4177872Sarnold {
4187872Sarnold     register char	c;
4197709Sarnold 
4207872Sarnold     if (ch == '\n')
4217872Sarnold 	while ((c = readchar()) != '\n')
4227872Sarnold 	    continue;
4237872Sarnold     else
4247872Sarnold 	while (readchar() != ch)
4257872Sarnold 	    continue;
4267872Sarnold }
4277709Sarnold 
4287872Sarnold /*
4297872Sarnold  * readchar:
4307872Sarnold  *	Reads and returns a character, checking for gross input errors
4317872Sarnold  */
4327872Sarnold readchar()
4337872Sarnold {
4347872Sarnold     register int	cnt, y, x;
4357872Sarnold     auto char		c;
4367709Sarnold 
4377872Sarnold over:
4387872Sarnold     cnt = 0;
4397872Sarnold     while (read(0, &c, 1) <= 0)
4407872Sarnold 	if (cnt++ > 100)	/* if we are getting infinite EOFs */
4417872Sarnold 	    bye();		/* quit the game */
4427872Sarnold     if (c == CTRL(L)) {
4437872Sarnold 	wrefresh(curscr);
4447872Sarnold 	goto over;
4457872Sarnold     }
4467872Sarnold     Hasread = TRUE;
4477872Sarnold     if (c == '\r')
4487872Sarnold 	return '\n';
4497872Sarnold     else
4507872Sarnold 	return c;
4517872Sarnold }
4527872Sarnold 
4537872Sarnold /*
4547872Sarnold  * getline:
4557872Sarnold  *      Reads the next line up to '\n' or EOF.  Multiple spaces are
4567872Sarnold  *	compressed to one space; a space is inserted before a ','
4577872Sarnold  */
4587872Sarnold char *
4597872Sarnold getline()
4607872Sarnold {
4617872Sarnold     register char	*sp;
4627872Sarnold     register int	c, oy, ox;
4637872Sarnold 
4647872Sarnold     getyx(stdscr, oy, ox);
4657872Sarnold     refresh();
4667872Sarnold     /*
4677872Sarnold      * loop reading in the string, and put it in a temporary buffer
4687872Sarnold      */
4697872Sarnold     for (sp = linebuf; (c = readchar()) != '\n'; clrtoeol(), refresh()) {
4707872Sarnold 	if (c == -1)
4717872Sarnold 	    continue;
4727872Sarnold 	else if (c == erasechar()) {	/* process erase character */
4737872Sarnold 	    if (sp > linebuf) {
4747872Sarnold 		register int i;
4757872Sarnold 
4767872Sarnold 		sp--;
4777872Sarnold 		for (i = strlen(unctrl(*sp)); i; i--)
4787872Sarnold 		    addch('\b');
4797872Sarnold 	    }
4807872Sarnold 	    continue;
4817872Sarnold 	}
4827872Sarnold 	else if (c == killchar()) {	/* process kill character */
4837872Sarnold 	    sp = linebuf;
4847872Sarnold 	    move(oy, ox);
4857872Sarnold 	    continue;
4867872Sarnold 	}
4877872Sarnold 	else if (sp == linebuf && c == ' ')
4887872Sarnold 	    continue;
4897872Sarnold 	if (sp >= &linebuf[LINESIZE-1] || !(isprint(c) || c == ' '))
4907872Sarnold 	    putchar(CTRL(G));
4917872Sarnold 	else {
4927872Sarnold 	    if (islower(c))
4937872Sarnold 		c = toupper(c);
4947872Sarnold 	    *sp++ = c;
4957872Sarnold 	    addstr(unctrl(c));
4967872Sarnold /*###366 [cc] Mpos undefined %%%*/
4977872Sarnold 	    Mpos++;
4987872Sarnold 	}
4997872Sarnold     }
5007872Sarnold     *sp = '\0';
5017872Sarnold     return linebuf;
5027872Sarnold }
503