121505Smckusick /* 221505Smckusick * Copyright (c) 1980 Regents of the University of California. 3*33707Sbostic * All rights reserved. 4*33707Sbostic * 5*33707Sbostic * Redistribution and use in source and binary forms are permitted 6*33707Sbostic * provided that this notice is preserved and that due credit is given 7*33707Sbostic * to the University of California at Berkeley. The name of the University 8*33707Sbostic * may not be used to endorse or promote products derived from this 9*33707Sbostic * software without specific prior written permission. This software 10*33707Sbostic * is provided ``as is'' without express or implied warranty. 1121505Smckusick */ 1212572Sarnold 1321505Smckusick #ifndef lint 14*33707Sbostic static char sccsid[] = "@(#)io.c 5.3 (Berkeley) 03/10/88"; 15*33707Sbostic #endif /* not lint */ 1621505Smckusick 177872Sarnold # include <curses.h> 187872Sarnold # include <ctype.h> 1914910Sarnold # include <signal.h> 207872Sarnold # include "deck.h" 217872Sarnold # include "cribbage.h" 2212163Sarnold # include "cribcur.h" 237709Sarnold 247872Sarnold # define LINESIZE 128 257709Sarnold 2612163Sarnold # ifdef CTRL 2712163Sarnold # undef CTRL 2812163Sarnold # endif 297872Sarnold # define CTRL(X) ('X' - 'A' + 1) 307709Sarnold 3132355Sbostic # ifdef notdef /* defined in curses.h */ 327872Sarnold # define erasechar() _tty.sg_erase 337872Sarnold # define killchar() _tty.sg_kill 3432355Sbostic # endif 357872Sarnold 367709Sarnold char linebuf[ LINESIZE ]; 377709Sarnold 387709Sarnold char *rankname[ RANKS ] = { "ACE", "TWO", "THREE", "FOUR", 397872Sarnold "FIVE", "SIX", "SEVEN", "EIGHT", 407872Sarnold "NINE", "TEN", "JACK", "QUEEN", 417872Sarnold "KING" }; 427709Sarnold 437709Sarnold char *rankchar[ RANKS ] = { "A", "2", "3", "4", "5", "6", "7", 447872Sarnold "8", "9", "T", "J", "Q", "K" }; 457709Sarnold 467709Sarnold char *suitname[ SUITS ] = { "SPADES", "HEARTS", "DIAMONDS", 477872Sarnold "CLUBS" }; 487709Sarnold 497709Sarnold char *suitchar[ SUITS ] = { "S", "H", "D", "C" }; 507709Sarnold 517709Sarnold 527709Sarnold 537709Sarnold /* 547872Sarnold * msgcard: 557872Sarnold * Call msgcrd in one of two forms 567709Sarnold */ 577872Sarnold msgcard(c, brief) 587872Sarnold CARD c; 597872Sarnold BOOLEAN brief; 607709Sarnold { 617872Sarnold if (brief) 627872Sarnold return msgcrd(c, TRUE, (char *) NULL, TRUE); 637872Sarnold else 647872Sarnold return msgcrd(c, FALSE, " of ", FALSE); 657709Sarnold } 667709Sarnold 677709Sarnold 687709Sarnold 697709Sarnold /* 707872Sarnold * msgcrd: 717872Sarnold * Print the value of a card in ascii 727709Sarnold */ 737872Sarnold msgcrd(c, brfrank, mid, brfsuit) 747872Sarnold CARD c; 757872Sarnold char *mid; 767872Sarnold BOOLEAN brfrank, brfsuit; 777872Sarnold { 787872Sarnold if (c.rank == EMPTY || c.suit == EMPTY) 797872Sarnold return FALSE; 807872Sarnold if (brfrank) 817872Sarnold addmsg("%1.1s", rankchar[c.rank]); 827872Sarnold else 837872Sarnold addmsg(rankname[c.rank]); 847872Sarnold if (mid != NULL) 857872Sarnold addmsg(mid); 867872Sarnold if (brfsuit) 877872Sarnold addmsg("%1.1s", suitchar[c.suit]); 887872Sarnold else 897872Sarnold addmsg(suitname[c.suit]); 907872Sarnold return TRUE; 917872Sarnold } 927709Sarnold 937872Sarnold /* 947872Sarnold * printcard: 957872Sarnold * Print out a card. 967872Sarnold */ 978073Sarnold printcard(win, cardno, c, blank) 987872Sarnold WINDOW *win; 997872Sarnold int cardno; 1007872Sarnold CARD c; 1018073Sarnold BOOLEAN blank; 1027872Sarnold { 1038073Sarnold prcard(win, cardno * 2, cardno, c, blank); 1047872Sarnold } 1057709Sarnold 1067872Sarnold /* 1077872Sarnold * prcard: 1087872Sarnold * Print out a card on the window at the specified location 1097872Sarnold */ 1107946Sarnold prcard(win, y, x, c, blank) 1117872Sarnold WINDOW *win; 1127872Sarnold int y, x; 1137872Sarnold CARD c; 1147946Sarnold BOOLEAN blank; 1157709Sarnold { 1167872Sarnold if (c.rank == EMPTY) 1177872Sarnold return; 1187872Sarnold mvwaddstr(win, y + 0, x, "+-----+"); 1197872Sarnold mvwaddstr(win, y + 1, x, "| |"); 1207872Sarnold mvwaddstr(win, y + 2, x, "| |"); 1217872Sarnold mvwaddstr(win, y + 3, x, "| |"); 1227872Sarnold mvwaddstr(win, y + 4, x, "+-----+"); 1237946Sarnold if (!blank) { 1247946Sarnold mvwaddch(win, y + 1, x + 1, rankchar[c.rank][0]); 1257946Sarnold waddch(win, suitchar[c.suit][0]); 1267946Sarnold mvwaddch(win, y + 3, x + 4, rankchar[c.rank][0]); 1277946Sarnold waddch(win, suitchar[c.suit][0]); 1287946Sarnold } 1297709Sarnold } 1307709Sarnold 1317709Sarnold /* 1327872Sarnold * prhand: 1337872Sarnold * Print a hand of n cards 1347709Sarnold */ 1358073Sarnold prhand(h, n, win, blank) 1367872Sarnold CARD h[]; 1377872Sarnold int n; 1387872Sarnold WINDOW *win; 1398073Sarnold BOOLEAN blank; 1407709Sarnold { 1417872Sarnold register int i; 1427709Sarnold 1437872Sarnold werase(win); 1447872Sarnold for (i = 0; i < n; i++) 1458073Sarnold printcard(win, i, *h++, blank); 1467872Sarnold wrefresh(win); 1477709Sarnold } 1487709Sarnold 1497709Sarnold 1507709Sarnold 1517709Sarnold /* 1527872Sarnold * infrom: 1537872Sarnold * reads a card, supposedly in hand, accepting unambigous brief 1547872Sarnold * input, returns the index of the card found... 1557709Sarnold */ 1567872Sarnold infrom(hand, n, prompt) 1577872Sarnold CARD hand[]; 1587872Sarnold int n; 1597872Sarnold char *prompt; 1607709Sarnold { 1617872Sarnold register int i, j; 1627872Sarnold CARD crd; 1637709Sarnold 1647872Sarnold if (n < 1) { 1657872Sarnold printf("\nINFROM: %d = n < 1!!\n", n); 1667872Sarnold exit(74); 1677872Sarnold } 1687872Sarnold for (;;) { 1697947Sarnold msg(prompt); 1707872Sarnold if (incard(&crd)) { /* if card is full card */ 1717872Sarnold if (!isone(crd, hand, n)) 1727872Sarnold msg("That's not in your hand"); 1737872Sarnold else { 1747872Sarnold for (i = 0; i < n; i++) 1757872Sarnold if (hand[i].rank == crd.rank && 1767872Sarnold hand[i].suit == crd.suit) 1777872Sarnold break; 1787872Sarnold if (i >= n) { 1797872Sarnold printf("\nINFROM: isone or something messed up\n"); 1807872Sarnold exit(77); 1817872Sarnold } 1827872Sarnold return i; 1837872Sarnold } 1847872Sarnold } 1857872Sarnold else /* if not full card... */ 1867872Sarnold if (crd.rank != EMPTY) { 1877872Sarnold for (i = 0; i < n; i++) 1887872Sarnold if (hand[i].rank == crd.rank) 1897872Sarnold break; 1907872Sarnold if (i >= n) 1917872Sarnold msg("No such rank in your hand"); 1927872Sarnold else { 1937872Sarnold for (j = i + 1; j < n; j++) 1947872Sarnold if (hand[j].rank == crd.rank) 1957872Sarnold break; 1967872Sarnold if (j < n) 1977872Sarnold msg("Ambiguous rank"); 1987872Sarnold else 1997872Sarnold return i; 2007872Sarnold } 2017872Sarnold } 2027872Sarnold else 2037872Sarnold msg("Sorry, I missed that"); 2047872Sarnold } 2057872Sarnold /* NOTREACHED */ 2067709Sarnold } 2077709Sarnold 2087709Sarnold 2097709Sarnold 2107709Sarnold /* 2117872Sarnold * incard: 2127872Sarnold * Inputs a card in any format. It reads a line ending with a CR 2137872Sarnold * and then parses it. 2147709Sarnold */ 2157872Sarnold incard(crd) 2167872Sarnold CARD *crd; 2177709Sarnold { 2187872Sarnold char *getline(); 2197872Sarnold register int i; 2207872Sarnold int rnk, sut; 2217872Sarnold char *line, *p, *p1; 2227872Sarnold BOOLEAN retval; 2237709Sarnold 2247872Sarnold retval = FALSE; 2257872Sarnold rnk = sut = EMPTY; 2267872Sarnold if (!(line = getline())) 2277872Sarnold goto gotit; 2287872Sarnold p = p1 = line; 2297872Sarnold while( *p1 != ' ' && *p1 != NULL ) ++p1; 2307872Sarnold *p1++ = NULL; 2317872Sarnold if( *p == NULL ) goto gotit; 2327872Sarnold /* IMPORTANT: no real card has 2 char first name */ 2337872Sarnold if( strlen(p) == 2 ) { /* check for short form */ 2347872Sarnold rnk = EMPTY; 2357872Sarnold for( i = 0; i < RANKS; i++ ) { 2367872Sarnold if( *p == *rankchar[i] ) { 2377872Sarnold rnk = i; 2387872Sarnold break; 2397872Sarnold } 2407872Sarnold } 2417872Sarnold if( rnk == EMPTY ) goto gotit; /* it's nothing... */ 2427872Sarnold ++p; /* advance to next char */ 2437872Sarnold sut = EMPTY; 2447872Sarnold for( i = 0; i < SUITS; i++ ) { 2457872Sarnold if( *p == *suitchar[i] ) { 2467872Sarnold sut = i; 2477872Sarnold break; 2487872Sarnold } 2497872Sarnold } 2507872Sarnold if( sut != EMPTY ) retval = TRUE; 2517872Sarnold goto gotit; 2527872Sarnold } 2537872Sarnold rnk = EMPTY; 2547872Sarnold for( i = 0; i < RANKS; i++ ) { 2557872Sarnold if( !strcmp( p, rankname[i] ) || !strcmp( p, rankchar[i] ) ) { 2567872Sarnold rnk = i; 2577872Sarnold break; 2587872Sarnold } 2597872Sarnold } 2607872Sarnold if( rnk == EMPTY ) goto gotit; 2617872Sarnold p = p1; 2627872Sarnold while( *p1 != ' ' && *p1 != NULL ) ++p1; 2637872Sarnold *p1++ = NULL; 2647872Sarnold if( *p == NULL ) goto gotit; 2657872Sarnold if( !strcmp( "OF", p ) ) { 2667872Sarnold p = p1; 2677872Sarnold while( *p1 != ' ' && *p1 != NULL ) ++p1; 2687872Sarnold *p1++ = NULL; 2697872Sarnold if( *p == NULL ) goto gotit; 2707872Sarnold } 2717872Sarnold sut = EMPTY; 2727872Sarnold for( i = 0; i < SUITS; i++ ) { 2737872Sarnold if( !strcmp( p, suitname[i] ) || !strcmp( p, suitchar[i] ) ) { 2747872Sarnold sut = i; 2757872Sarnold break; 2767872Sarnold } 2777872Sarnold } 2787872Sarnold if( sut != EMPTY ) retval = TRUE; 2797709Sarnold gotit: 2807872Sarnold (*crd).rank = rnk; 2817872Sarnold (*crd).suit = sut; 2827872Sarnold return( retval ); 2837709Sarnold } 2847709Sarnold 2857709Sarnold 2867709Sarnold 2877709Sarnold /* 2887872Sarnold * getuchar: 2897872Sarnold * Reads and converts to upper case 2907709Sarnold */ 2917709Sarnold getuchar() 2927709Sarnold { 2937872Sarnold register int c; 2947709Sarnold 2957872Sarnold c = readchar(); 2967872Sarnold if (islower(c)) 2977872Sarnold c = toupper(c); 29812163Sarnold waddch(Msgwin, c); 2997872Sarnold return c; 3007709Sarnold } 3017709Sarnold 3027709Sarnold /* 3037934Sarnold * number: 3047934Sarnold * Reads in a decimal number and makes sure it is between "lo" and 3057934Sarnold * "hi" inclusive. 3067709Sarnold */ 3077934Sarnold number(lo, hi, prompt) 3087934Sarnold int lo, hi; 3097934Sarnold char *prompt; 3107709Sarnold { 3117934Sarnold char *getline(); 3127934Sarnold register char *p; 3137934Sarnold register int sum; 3147709Sarnold 3157872Sarnold sum = 0; 3167934Sarnold for (;;) { 3177934Sarnold msg(prompt); 3187934Sarnold if(!(p = getline()) || *p == NULL) { 3197934Sarnold msg(quiet ? "Not a number" : "That doesn't look like a number"); 3207934Sarnold continue; 3217934Sarnold } 3227872Sarnold sum = 0; 3237934Sarnold 3247934Sarnold if (!isdigit(*p)) 3257872Sarnold sum = lo - 1; 3267934Sarnold else 3277934Sarnold while (isdigit(*p)) { 3287934Sarnold sum = 10 * sum + (*p - '0'); 3297872Sarnold ++p; 3307934Sarnold } 3317934Sarnold 3327934Sarnold if (*p != ' ' && *p != '\t' && *p != NULL) 3337934Sarnold sum = lo - 1; 3347934Sarnold if (sum >= lo && sum <= hi) 3357934Sarnold return sum; 3367934Sarnold if (sum == lo - 1) 3377934Sarnold msg("that doesn't look like a number, try again --> "); 3387934Sarnold else 3397934Sarnold msg("%d is not between %d and %d inclusive, try again --> ", 3407934Sarnold sum, lo, hi); 3417934Sarnold } 3427709Sarnold } 3437709Sarnold 3447872Sarnold /* 3457872Sarnold * msg: 3467872Sarnold * Display a message at the top of the screen. 3477872Sarnold */ 34812157Sarnold char Msgbuf[BUFSIZ] = { '\0' }; 3497709Sarnold 3507872Sarnold int Mpos = 0; 3517709Sarnold 3527872Sarnold static int Newpos = 0; 3537872Sarnold 3547872Sarnold /* VARARGS1 */ 3557872Sarnold msg(fmt, args) 3567872Sarnold char *fmt; 3577872Sarnold int args; 3587872Sarnold { 3597872Sarnold doadd(fmt, &args); 3607872Sarnold endmsg(); 3617872Sarnold } 3627872Sarnold 3637709Sarnold /* 3647872Sarnold * addmsg: 3657872Sarnold * Add things to the current message 3667709Sarnold */ 3677872Sarnold /* VARARGS1 */ 3687872Sarnold addmsg(fmt, args) 3697872Sarnold char *fmt; 3707872Sarnold int args; 3717872Sarnold { 3727872Sarnold doadd(fmt, &args); 3737872Sarnold } 3747709Sarnold 3757872Sarnold /* 3767872Sarnold * endmsg: 37712163Sarnold * Display a new msg. 3787872Sarnold */ 37912316Sarnold 38012316Sarnold int Lineno = 0; 38112316Sarnold 3827872Sarnold endmsg() 3837709Sarnold { 38412163Sarnold register int len; 38512163Sarnold register char *mp, *omp; 38612316Sarnold static int lastline = 0; 38712163Sarnold 3887872Sarnold /* 38912316Sarnold * All messages should start with uppercase 3907872Sarnold */ 39112316Sarnold mvaddch(lastline + Y_MSG_START, SCORE_X, ' '); 3927872Sarnold if (islower(Msgbuf[0]) && Msgbuf[1] != ')') 3937872Sarnold Msgbuf[0] = toupper(Msgbuf[0]); 39412163Sarnold mp = Msgbuf; 39512163Sarnold len = strlen(mp); 39612330Sarnold if (len / MSG_X + Lineno >= MSG_Y) { 39712330Sarnold while (Lineno < MSG_Y) { 39812330Sarnold wmove(Msgwin, Lineno++, 0); 39912330Sarnold wclrtoeol(Msgwin); 40012330Sarnold } 40112316Sarnold Lineno = 0; 40212330Sarnold } 40312316Sarnold mvaddch(Lineno + Y_MSG_START, SCORE_X, '*'); 40412316Sarnold lastline = Lineno; 40512163Sarnold do { 40612316Sarnold mvwaddstr(Msgwin, Lineno, 0, mp); 40712163Sarnold if ((len = strlen(mp)) > MSG_X) { 40812163Sarnold omp = mp; 40912163Sarnold for (mp = &mp[MSG_X-1]; *mp != ' '; mp--) 41012163Sarnold continue; 41112163Sarnold while (*mp == ' ') 41212163Sarnold mp--; 41312163Sarnold mp++; 41412316Sarnold wmove(Msgwin, Lineno, mp - omp); 41512163Sarnold wclrtoeol(Msgwin); 41612163Sarnold } 41712316Sarnold if (++Lineno >= MSG_Y) 41812316Sarnold Lineno = 0; 41912163Sarnold } while (len > MSG_X); 42012163Sarnold wclrtoeol(Msgwin); 42112330Sarnold Mpos = len; 4227872Sarnold Newpos = 0; 42312163Sarnold wrefresh(Msgwin); 4247872Sarnold refresh(); 42512163Sarnold wrefresh(Msgwin); 4267872Sarnold } 4277709Sarnold 4287872Sarnold /* 4297872Sarnold * doadd: 4307872Sarnold * Perform an add onto the message buffer 4317872Sarnold */ 4327872Sarnold doadd(fmt, args) 4337872Sarnold char *fmt; 4347872Sarnold int *args; 4357872Sarnold { 4367872Sarnold static FILE junk; 4377872Sarnold 4387872Sarnold /* 4397872Sarnold * Do the printf into Msgbuf 4407872Sarnold */ 4417872Sarnold junk._flag = _IOWRT + _IOSTRG; 4427872Sarnold junk._ptr = &Msgbuf[Newpos]; 4437872Sarnold junk._cnt = 32767; 4447872Sarnold _doprnt(fmt, args, &junk); 4457872Sarnold putc('\0', &junk); 4467872Sarnold Newpos = strlen(Msgbuf); 4477709Sarnold } 4487709Sarnold 4497872Sarnold /* 45012316Sarnold * do_wait: 45112316Sarnold * Wait for the user to type ' ' before doing anything else 45212316Sarnold */ 45312316Sarnold do_wait() 45412316Sarnold { 45512316Sarnold register int line; 45612316Sarnold static char prompt[] = { '-', '-', 'M', 'o', 'r', 'e', '-', '-', '\0' }; 45712316Sarnold 45812316Sarnold if (Mpos + sizeof prompt < MSG_X) 45912316Sarnold wmove(Msgwin, Lineno > 0 ? Lineno - 1 : MSG_Y - 1, Mpos); 46012316Sarnold else { 46112316Sarnold mvwaddch(Msgwin, Lineno, 0, ' '); 46212316Sarnold wclrtoeol(Msgwin); 46312316Sarnold if (++Lineno >= MSG_Y) 46412316Sarnold Lineno = 0; 46512316Sarnold } 46612316Sarnold waddstr(Msgwin, prompt); 46712316Sarnold wrefresh(Msgwin); 46812316Sarnold wait_for(' '); 46912316Sarnold } 47012316Sarnold 47112316Sarnold /* 4727872Sarnold * wait_for 4737872Sarnold * Sit around until the guy types the right key 4747872Sarnold */ 4757872Sarnold wait_for(ch) 4767872Sarnold register char ch; 4777872Sarnold { 4787872Sarnold register char c; 4797709Sarnold 4807872Sarnold if (ch == '\n') 4817872Sarnold while ((c = readchar()) != '\n') 4827872Sarnold continue; 4837872Sarnold else 4847872Sarnold while (readchar() != ch) 4857872Sarnold continue; 4867872Sarnold } 4877709Sarnold 4887872Sarnold /* 4897872Sarnold * readchar: 4907872Sarnold * Reads and returns a character, checking for gross input errors 4917872Sarnold */ 4927872Sarnold readchar() 4937872Sarnold { 4947872Sarnold register int cnt, y, x; 4957872Sarnold auto char c; 4967709Sarnold 4977872Sarnold over: 4987872Sarnold cnt = 0; 4997872Sarnold while (read(0, &c, 1) <= 0) 5007872Sarnold if (cnt++ > 100) /* if we are getting infinite EOFs */ 5017872Sarnold bye(); /* quit the game */ 5027872Sarnold if (c == CTRL(L)) { 5037872Sarnold wrefresh(curscr); 5047872Sarnold goto over; 5057872Sarnold } 5067872Sarnold if (c == '\r') 5077872Sarnold return '\n'; 5087872Sarnold else 5097872Sarnold return c; 5107872Sarnold } 5117872Sarnold 5127872Sarnold /* 5137872Sarnold * getline: 5147872Sarnold * Reads the next line up to '\n' or EOF. Multiple spaces are 5157872Sarnold * compressed to one space; a space is inserted before a ',' 5167872Sarnold */ 5177872Sarnold char * 5187872Sarnold getline() 5197872Sarnold { 5207872Sarnold register char *sp; 5217872Sarnold register int c, oy, ox; 52212163Sarnold register WINDOW *oscr; 5237872Sarnold 52412163Sarnold oscr = stdscr; 52512163Sarnold stdscr = Msgwin; 5267872Sarnold getyx(stdscr, oy, ox); 5277872Sarnold refresh(); 5287872Sarnold /* 5297872Sarnold * loop reading in the string, and put it in a temporary buffer 5307872Sarnold */ 5317872Sarnold for (sp = linebuf; (c = readchar()) != '\n'; clrtoeol(), refresh()) { 5327872Sarnold if (c == -1) 5337872Sarnold continue; 5347872Sarnold else if (c == erasechar()) { /* process erase character */ 5357872Sarnold if (sp > linebuf) { 5367872Sarnold register int i; 5377872Sarnold 5387872Sarnold sp--; 5397872Sarnold for (i = strlen(unctrl(*sp)); i; i--) 5407872Sarnold addch('\b'); 5417872Sarnold } 5427872Sarnold continue; 5437872Sarnold } 5447872Sarnold else if (c == killchar()) { /* process kill character */ 5457872Sarnold sp = linebuf; 5467872Sarnold move(oy, ox); 5477872Sarnold continue; 5487872Sarnold } 5497872Sarnold else if (sp == linebuf && c == ' ') 5507872Sarnold continue; 5517872Sarnold if (sp >= &linebuf[LINESIZE-1] || !(isprint(c) || c == ' ')) 5527872Sarnold putchar(CTRL(G)); 5537872Sarnold else { 5547872Sarnold if (islower(c)) 5557872Sarnold c = toupper(c); 5567872Sarnold *sp++ = c; 5577872Sarnold addstr(unctrl(c)); 5587872Sarnold Mpos++; 5597872Sarnold } 5607872Sarnold } 5617872Sarnold *sp = '\0'; 56212163Sarnold stdscr = oscr; 5637872Sarnold return linebuf; 5647872Sarnold } 56514910Sarnold 56614910Sarnold /* 56714910Sarnold * bye: 56814910Sarnold * Leave the program, cleaning things up as we go. 56914910Sarnold */ 57014910Sarnold bye() 57114910Sarnold { 57214910Sarnold signal(SIGINT, SIG_IGN); 57314910Sarnold mvcur(0, COLS - 1, LINES - 1, 0); 57414910Sarnold fflush(stdout); 57514910Sarnold endwin(); 57614910Sarnold putchar('\n'); 57714910Sarnold exit(1); 57814910Sarnold } 579