home *** CD-ROM | disk | FTP | other *** search
- /* LIFE.C The much implemented game of Life invented by John Conway */
-
- #include <stdio.h>
- #include "life.h"
-
- char World[ROWS][COLS],Create_mode,Quit_flag;
- int Population,Generation,Currow,Curcol;
-
- main() { /*& Main sequence, cycle 'till two ESC chars */
-
- while(1)
- {
- scr_clr();
-
- instruct();
-
- if(setup())
- break;
-
- do {
- Generation++;
- cycle();
- screen();
- }
- while (Population && !Quit_flag);
-
- }
-
- scr_rowcol(ROWS,27);
- puts("goodbye");
- scr_curson();
- }
-
- instruct() /*& Print instructions, one screen's worth */
- {
- puts(" The game of Life by John Conroy\n\n");
- puts(" If started with a number, a random pattern starts the game.\n");
- puts(" Otherwise, move the cursor with the four arrow keys to create life.\n\n");
- puts(" DEL changes cursor movement to mean that cells are deleted\n");
- puts(" INS flips back to create mode.\n\n");
- puts(" The '+' key will toggle the game on or off.\n");
- puts(" when off, 'S' saves the current pattern, 'L' loads previous save.\n\n");
- puts(" Hit ESC to bail out.\n\n");
- puts(" Enter starting number of cells or hit CR");
- scr_curson();
- scr_curson();
- }
-
- setup() {
- /*& Put in initial cells, by drawing or random placement */
- int rnumber;
- int i,row,col,seed,rnum;
- char ch;
-
- /*$ Initialize globals */
-
- Quit_flag=Population=Generation=Currow=Curcol=0;
- Create_mode=1;
- for(row=0;row<ROWS;row++)
- for(col=0;col<COLS;col++)
- World[row][col]=0;
-
- rnumber=0;
- while (1) {
- while ((ch=get_key()) == 0) seed++;
- if (ch < '0' || ch > '9') break;
- putch(ch);
- rnumber*=10;
- rnumber+=ch-'0';
- }
- if(ch==ESC)
- return(1);
-
- scr_cursoff();
- scr_clr();
- scr_rowcol(ROWS,20); /*$ Print Population message */
- printf("Generation Population ");
-
- srand(seed); /*$ Initilize random number generator */
-
- for (i=0; i < rnumber; i++) {
- rnum=rand();
- row=rnum%ROWS;
- col=(rnum/ROWS)%COLS;
- World[row][col]='X'; /*$ Put in a cell */
- scr_rowcol(row,col);
- putch(2);
- }
- if (rnumber == 0) create(1);
- return(0);
- }
-
-
- screen() /*& Update the screen and set the world back to x's */
- {
- int row,col;
- char cell;
-
- Population=0;
- for (row=0; row < ROWS; row++) {
- for (col=0; col < COLS; col++) {
- cell=World[row][col];
- /*$ stay alive if 3 neighbors, born if next to 2 or 3 */
-
- if (cell && (cell == 3 || cell == 'X'+2 || cell == 'X'+3)) {
- Population++;
- if (cell < 'X') {
- scr_rowcol(row,col);
- putch(2);
- }
- cell='X';
- }
- else {
- if (cell >= 'X') {
- scr_rowcol(row,col);
- putch(' ');
- }
- cell=0;
- }
- World[row][col]=cell;
- }
- }
- scr_rowcol(ROWS,31);
- printf("%4d",Generation);
- scr_rowcol(ROWS,51);
- printf("%4d",Population);
- }
-
-
- create(suspend)
- /*& If interrupted, define cells, otherwise update most of cell pattern */
- char suspend; {
- unsigned char ch,wait;
-
- scr_rowcol(Currow,Curcol);
-
- while ((ch=get_key()) || suspend) {
-
- scr_curson();
- switch (ch) {
- case 'l':
- case 'L': load();
- continue;
- case 's':
- case 'S': save();
- continue;
- case UP_CHAR: Currow=Currow ? Currow-1: ROWS-1;
- break;
- case DOWN_CHAR: Currow=Currow == ROWS-1 ? 0: Currow+1;
- break;
- case LEFT_CHAR: Curcol=Curcol ? Curcol-1: COLS-1;
- break;
- case RIGHT_CHAR:Curcol=Curcol == COLS-1 ? 0: Curcol+1;
- break;
- case BOL_CHAR: Curcol=0;
- break;
- case EOL_CHAR: Curcol=COLS-1;
- break;
- case '+': suspend=!suspend;
- continue;
- case Ins_char: Create_mode=1;
- continue;
- case Del_char: Create_mode=0;
- continue;
- case ESC: Quit_flag=1; /* flag for stop */
- return;
- default: continue;
- }
-
-
- World[Currow][Curcol]= Create_mode ? 'X': 0;
- scr_rowcol(Currow,Curcol);
- if (Create_mode) {
- putch(2);
- Population++;
- }
- else {
- wait=30;
- while (wait--) {
- putch(1);
- scr_rowcol(Currow,Curcol);
- }
- putch(' ');
- }
- scr_rowcol(Currow,Curcol);
- }
- scr_cursoff();
- }
-
- cycle() /*& Cycle to the next Generation */
- {
- register int row,col;
-
- create(0);
- /*$ Take care of left and right column first */
- for (row=0; row < ROWS; row++) {
- if (World[row][0] >= 'X') add8(row,0);
- if (World[row][COLS-1] >= 'X') add8(row,COLS-1);
- }
-
- /*$ Take care of top and bottom line */
- for (col=1; col < COLS-1;col++) {
- if (World[0][col] >= 'X') add8(0,col);
- if (World[ROWS-1][col] >= 'X') add8(ROWS-1,col);
- }
-
- /*$ fill in the box, ignoring border conditions */
- for (row=1; row < ROWS-1; row++) {
- for (col=1; col < COLS-1; col++) {
- if (World[row][col] >= 'X' ) {
- World[row-1][col-1]++;
- World[row-1][col]++;
- World[row-1][col+1]++;
- World[row][col-1]++;
- World[row][col+1]++;
- World[row+1][col-1]++;
- World[row+1][col]++;
- World[row+1][col+1]++;
- }
- }
- }
- }
-
-
- add8(row,col) /*& Fixup cells on the border of screen */
- int row,col; {
- int rrow,ccol,rr,cc;
-
- for (rr=row-1; rr <= row+1; rr++) {
- for (cc=col-1; cc <= col+1; cc++) {
- rrow=rr != -1 ? rr : ROWS-1;
- ccol=cc != -1 ? cc : COLS-1;
- if (rrow >= ROWS) rrow=0;
- if (ccol >= COLS) ccol=0;
- World[rrow][ccol]++;
- }
- }
- World[row][col]--;
- }
-
- save() /*& Save current pattern in life.dat */
- {
- FILE *fptr;
- int r,c;
-
- if(fptr=fopen("life.dat","wb"))
- {
- fwrite(&World[0][0],ROWS,COLS,fptr);
- fclose(fptr);
- }
- #if TEXT
- if(fptr=fopen("life.txt","w"))
- {
- for(r=0;r<ROWS;r++)
- {
- for(c=0;c<COLS;c++)
- fputc(World[r][c]?'X':' ',fptr);
- fputc('\n',fptr);
- }
- }
- #endif
- }
-
- load() /*& Load life.dat pattern, replacing current pattern */
- {
- FILE *fptr;
- int row,col;
-
- if(fptr=fopen("life.dat","rb"))
- {
- fread(&World[0][0],ROWS,COLS,fptr);
- fclose(fptr);
- /*$ now put world to screen */
- for(row=0;row<ROWS;row++)
- for(col=0;col<COLS;col++)
- if(World[row][col]=='X')
- {
- scr_rowcol(row,col);
- putch(2);
- }
- }
-
- }
-