lab@qubix.UUCP (Larry Bickford) (07/05/85)
/* battleships by Judah S. Kaminetsky */ /* Vax 4.2bsd improvements by Tom Truscott (rti-sel!trt) */ /* Improvements on Version 1 by Larry Bickford (qubix!lab): /* register efficiency * horizontal expansion (more readable) * catching SIGINT gracefully * exit(0) rather than exit(1) * 4.2bsd correction in flash() * other various fixes * Known bug: when battleship exits, your cursor is left at some * screwball place on the screen. Un-nice. * Note: Judah has introduced Version 2, which could still use some of * the above changes, even on SysV. Someone should blend the two. */ /* qubix!lab: next line is for 4.2 compilation */ /* cc -O -o bs bs.c -lcurses -ltermcap */ #define MAX_X 12 #define MAX_Y 12 #define NO 1 #define YES 0 #include <curses.h> /* rti-sel!trt: hack to support old-fashioned curses */ #ifndef A_REVERSE #define wattron(w, foo) wstandout(w) #define wattroff(w, foo) wstandend(w) #endif long r;/* for random number generator*/ int myscore = 0; int hisscore = 0; int hismoves = 1; int mymoves = 1; int nextx, nexty;/* next square to attack if adjacent to previous hit*/ int lastx, lasty; /* last move made */ /* qubix!lab: SIGINT catcher */ getout() { endwin(); exit(0); } main() { register WINDOW *mywin, /*player's screen*/ *myscrwin, /*players's score window*/ *hisscrwin, /*computers's score window*/ *hidwin, /* computers hidden screen */ *showwin, /*computer's displayed screen*/ *recwin; /*computers record of my screen */ register int t; time(&r);/*Seed the random number generator*/ srand((int)(r&0177777L)); initscr(); nonl(); noecho(); cbreak(); signal(2, getout); err("Battleships System Test - comments to hounx!juda"); mywin = newwin(MAX_Y, MAX_X << 1, 5, 15); recwin = newwin(MAX_Y, MAX_X, 5, 0); werase(mywin); werase(recwin); wattron(mywin,A_REVERSE); label_x(mywin); wrefresh(mywin); label_y(mywin); wrefresh(mywin); wattroff(mywin,A_REVERSE); /* qubix!lab: deemed unnecessary; need for labels later removed wattron(recwin,A_REVERSE); label_x(recwin); label_y(recwin); wattroff(recwin,A_REVERSE); */ set(mywin,'A',5); wrefresh(mywin); set(mywin,'B',4); wrefresh(mywin); set(mywin,'S',3); wrefresh(mywin); set(mywin,'D',3); wrefresh(mywin); set(mywin,'P',2); wrefresh(mywin); clrtop(); hidwin = newwin(MAX_Y, MAX_X, 5, 40); showwin = newwin(MAX_Y, MAX_X << 1, 5, 55); myscrwin = newwin(4, MAX_X+13, 17, 55); hisscrwin = newwin(4, MAX_X+13, 17, 15); werase(hidwin); werase(showwin); /** qubix!lab: deemed unnecessary; need for labels later removed ** wattron(hidwin,A_REVERSE); label_x(hidwin); wrefresh(hidwin); label_y(hidwin); wrefresh(hidwin); wattroff(hidwin,A_REVERSE); /**/ wattron(showwin,A_REVERSE); label_x(showwin); label_y(showwin); wrefresh(showwin); wattroff(showwin,A_REVERSE); setup(hidwin,'A',5); setup(hidwin,'B',4); setup(hidwin,'S',3); setup(hidwin,'D',3); setup(hidwin,'P',2); clrtop(); for(t = (MAX_X-2)*(MAX_Y-2); --t >= 0; ) { myttack(hidwin,showwin); myrecord(myscrwin); if(myscore>16) { msg("YOU WIN!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!"); wrefresh(hidwin); wrefresh(recwin); endwin(); exit(0); } hisattack(mywin,recwin); hisrecord(hisscrwin); if(hisscore>16) { msg("YOU LOSE!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!"); wrefresh(hidwin); wrefresh(recwin); endwin(); exit(0); } } endwin(); } /*************************************************************/ /* Draw x axis labels */ label_x(win) register WINDOW *win; { register int x,y; register int ch; for(y=0;y<MAX_Y;y+=MAX_Y-1) { for(x=1,ch=0;x<MAX_X-1;x++,ch++) { /* qubix!lab: horiz. expand */ wmove(win,y,x << 1); waddch(win,ch+'0'); } } touchwin(win); } /*****************************************************************/ /* Draw y axis labels */ label_y(win) register WINDOW *win; { register int x,y; /* in columns 1 and MAX_X */ for(x=0;x<MAX_X;x+=MAX_X-1) { for(y=0;y<MAX_Y;y++) { /* qubix!lab: horiz. expand */ wmove(win,y,x << 1); waddch(win,y+'/');/*start with char 0*/ } } touchwin(win); } /************************************************************************/ /* Place ship of length len, represented by character ch at coordinates * y, x */ /* direction up, down, right, left from start */ place(win,len,ch,init_y,init_x,dir) register WINDOW *win; register int len; /*length*/ char ch; register int init_y,init_x; register int dir; /*direction*/ { register int i; char c; register int dir_y, dir_x; register int x, y; dir_x=xdir(dir); dir_y=ydir(dir); /* avoid collisions with existing characters*/ for(i=0, x=init_x, y=init_y; i<len; i++, x=x+dir_x, y=y+dir_y) { if(x < 1 || y < 1 || y >= MAX_Y-1) return(1); if(dir > 0) if(x >= MAX_X - 1) return(1); /* qubix!lab: horiz expand */ else if(x >= (MAX_X-1)<<1) return(1); wmove(win,y,x); c=winch(win); if(c!=' ') { return(1); } } /* place characters if no collision */ for(i=0, x=init_x, y=init_y; i<len; i++, x=x+dir_x, y=y+dir_y) { wmove(win,y,x); waddch(win,ch); } return(0); } /*********************************************************************/ /* Determine x direction increment or decrement */ /* neg values are input by user => expand to 2X */ xdir(dir) register int dir; { register int xdir; switch(dir){ case 0 :/*up*/ xdir=0; break; case 1 :/*down*/ case -1 :/*down*/ xdir=0; break; case 2 :/*right*/ xdir=1; break; case -2 :/*right*/ xdir=2; break; case 3 :/*left*/ xdir=(-1); break; case -3 :/*left*/ xdir=(-2); break; default: xdir=1; break; } return(xdir); } /*********************************************************************/ /* Determine y direction increment or decrement */ ydir(dir) register int dir; { register int ydir; switch(dir){ case 0 :/*up*/ ydir=(-1); break; case -1: case 1 :/*down*/ ydir=1; break; case -2: case 2 :/*right*/ ydir=0; break; case -3: case 3 :/*left*/ ydir=0; break; default: ydir=0; break; } return(ydir); } /***********************************************************************/ /* generate random number between 0 and high (arg1), retrun random number */ random(high) register int high; { /* qubix!lab: without right-shift, and high = 10, was returning * values ending alternately 00 or 10 (binary). Un-random. */ return( ((rand() >> 2) % high) +1); } /********************************************************************/ /* place ship for computer*/ setup(win,ship,length) register WINDOW *win; char ship; register int length; { register int y , x; register int dir; for(;;) { msg("The computer is now placing its ships in a random manner"); x=random(MAX_X-2); y=random(MAX_X-2); dir=random(3); if(place(win,length,ship,y,x,dir)!=1) { return(0); } } } /********************************************************************/ /* get x coordinate from user*/ get_x() { register WINDOW *xwin; register int i, c; char ch; xwin = newwin(1, COLS, 1, 0);/*line 1*/ werase(xwin); wprintw(xwin,"Enter x (horizontal) coordinate [0-9] (q to exit):\t"); touchwin(stdscr); wrefresh(xwin); c=getch(); out(c); waddch(xwin,c); touchwin(stdscr); wrefresh(xwin); return(c-'0'+1); } /********************************************************************/ /* get y coordinate from user*/ get_y() { register WINDOW *ywin; register int i, c; char ch; ywin = newwin(1, COLS, 2, 0);/*line 2*/ werase(ywin); wprintw(ywin,"Enter y (vertical) coordinate [0-9] (q to exit):\t"); touchwin(stdscr); wrefresh(ywin); c=getch(); out(c); waddch(ywin,c); touchwin(stdscr); wrefresh(ywin); return(c-'0'+1); } /*********************************************************************/ /* get dir from user*/ get_dir() { register WINDOW *dirwin; register int i, c; char ch; dirwin = newwin(1, COLS, 3, 0);/*line 3*/ werase(dirwin); wprintw(dirwin, "Enter Direction, 0 is up, 1 is down, 2 is right, 3 is left: (q to exit)"); touchwin(stdscr); wrefresh(dirwin); c=getch(); out(c); waddch(dirwin,c); touchwin(stdscr); wrefresh(dirwin); /* get the NEGATIVE value */ return(c-'0'); } /*********************************************************************/ /* place ship for player*/ set(win,ship,length) register WINDOW *win; char ship; register int length; { register int y , x; register int dir; char msg_str[80]; for(;;) { sprintf(msg_str, "PLACE SHIP %c, length %d avoiding collisions",ship,length); msg(msg_str); if(verify(x = get_x(), 0, MAX_X-2)==1) { continue; } if(verify(y = get_y(), 0, MAX_Y-2)==1) { continue; } if(verify( dir = get_dir(), 0, 3) == 1) { continue; } /* qubix!lab: horiz expand; also note negative */ if(place(win, length, ship, y, x<<1, -dir) != 1) { touchwin(win); return(0); } } } /***********************************************************************/ msg(str) register char *str; { register WINDOW *msgwin; msgwin = newwin(1, COLS, 0,0); werase(msgwin); wattron(msgwin, A_REVERSE); wprintw(msgwin, "%s", str); touchwin(stdscr); wrefresh(msgwin); return(0); } /***********************************************************************/ verify(c,low,high) register int c; register int low; register int high; { char msgstr[80]; if(c < low || c > high) { sprintf(msgstr,"%c is out of legal range %c to %c",c,low,high); return(1); } else { return(0); } } /*********************************************************************/ /*player attack computer - goes to attack coordinates on hidwin*/ /*and copies ch (+attributes?) to showwin - touchwin and refresh showwin*/ /* allows duplicate moves but does not score duplicaate hits*/ myttack(hidwin,showwin) register WINDOW *hidwin, *showwin; { register int y , x; register int hit; char c; char d; char msg_str[80]; msg("ATTACK!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!"); while(verify(x=get_x(),0,MAX_X-2)==1) { } while(verify(y=get_y(),0,MAX_Y-2)==1) { } wmove(hidwin,y,x); c=winch(hidwin); if(c!=' ' && c != '.') { hit=YES; } wmove(showwin,y,x << 1); d=winch(showwin); if(hit == YES && (d==' ' || d=='.')) /* first hit - not repeat */ { flash(); myscore++; } wattron(showwin,A_REVERSE); /* qubix!lab: record where I've been */ if(c == ' ') c = '.'; waddch(showwin,c); touchwin(showwin); wrefresh(showwin); lastx = x; lasty = y; } /***************************************************************************/ /* clear top of screen */ clrtop() { register WINDOW *topwin; register int i, c; char ch; topwin = newwin(5, COLS, 0, 0);/*top 5 lines*/ werase(topwin); touchwin(topwin); wrefresh(topwin); } /*********************************************************************/ /*computer attack player - goes to attack coordinates on recwin*/ /* checks recwin - blank means haven't attacked here yet */ /* if so go to coord on mywin - attack & record result on recwin */ hisattack(mywin,recwin) register WINDOW *mywin; register WINDOW *recwin; { register int y , x; char c; char mark='+'; clrtop(); for(;;) { if(nextshot(recwin)==YES) { x=nextx; y=nexty; } else /* if(nextshot(recwin)==NO) */ { x=random(MAX_X-2); y=random(MAX_Y-2); } lastx = x; lasty = y; wmove(recwin,y,x);/*check for repeat move*/ c=winch(recwin); if(c!=' ')/*repeat move */ { continue; } else { /* qubix!lab: horiz expand */ wmove(mywin,y,x << 1); c=winch(mywin); if(c!=' ')/*hit*/ { flash(); mark=c;/*mark recwin with ship character*/ hisscore++; } else c = '.'; wattron(mywin,A_REVERSE); waddch(mywin,c); touchwin(mywin); wrefresh(mywin); waddch(recwin,mark); /*mark square as tried already and result + for blank or char for ship */ touchwin(recwin); return(YES); } } return(NO); } /***********************************************************************/ err(str) register char *str; { register WINDOW *errwin; /* rti-sel!trt: fix for window manager */ errwin=newwin(1,COLS,LINES-1,0); if (errwin == NULL) { fprintf(stderr, "cannot create error window\n"); return; } werase(errwin); wattron(errwin,A_REVERSE); wprintw(errwin,"%s",str); touchwin(stdscr); wrefresh(errwin); return(0); } /***********************************************************************/ out(c) char c; { if(c=='q') { endwin(); exit(1); } } /*********************************************************************/ myrecord(win) register WINDOW *win; { werase(win); wprintw(win, "Try %d,%d: ",lastx-1,lasty-1); wprintw(win,"hit %d ",myscore); wprintw(win,"move %d\n",mymoves++); touchwin(win); wrefresh(win); } /*********************************************************************/ hisrecord(win) register WINDOW *win; { werase(win); wprintw(win, "Try %d,%d: ",lastx-1,lasty-1); wprintw(win,"hit %d ",hisscore); wprintw(win,"move %d\n",hismoves++); touchwin(win); wrefresh(win); } /************************************************************************/ /* check win for previous hit and choose next guess at adjacent square */ nextshot(win) register WINDOW *win; { register int y; register int x; register int ax; register int by; char c; for(by = 1; by < MAX_Y-1; by++) { for(ax = 1;ax < MAX_X-1; ax++) { wmove(win,by,ax); c=winch(win); if(c!=' '&&c!='+') /* space was hit previously */ { /* eliminate need for hidden labels */ y=by; /*right*/ if((x=ax+1) < MAX_X-1 && check(win,y,x)==YES ) { return(YES); } /*left*/ if((x=ax-1) > 0 && check(win,y,x)==YES ) { return(YES); } x=ax; /*1 square down */ if((y=by+1) < MAX_Y - 1 && check(win,y,x)==YES ) { return(YES); } x=ax; /*up*/ if((y=by-1) > 0 && check(win,y,x)==YES ) { return(YES); } } } } return(NO);/* no untried squares adjacent to hits */ } /*******************************************************************/ /* check y,x on win, return YES if blank - not shot at yet return NO if not blank */ check(win,y,x) register WINDOW *win; register int y; register int x; { char c; wmove(win,y,x); c=winch(win); if(c==' ') /* adjacent to previous hit & blank */ { nextx=x; nexty=y; return(YES); } else { return(NO); } } #ifndef A_REVERSE /* need to define some other routines */ cbreak() { crmode(); } flash() { extern int _putchar(); /* qubix!lab: no ';' because of ways curses.h defines it! */ if (VB) _puts(VB) else putchar('\07'); } #endif -- The Ice Floe of Larry Bickford {amd,decwrl,sun,idi,ittvax}!qubix!lab You can't settle the issue until you've settled how to settle the issue.