/* * src/cgol.c * Curses Game of Life * * Copyright (C) 2011 Patrick "P. J." McDermott * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include #include #include #include #include #include void usage(const char *invocation); void init_curses(); void init_game(); void fini_game(); void fini_curses(); static int width, height; static double seedprob; static int genrate, gens; static bool *grid_cur, *grid_next; int main(int argc, char **argv) { int opt; width = 0; height = 0; seedprob = 0.5; genrate = 1; /* TODO: Units. */ gens = 0; while ((opt = getopt(argc, argv, "w:h:s:r:n:")) != -1) { switch (opt) { case 'w': width = atoi(optarg); break; case 'h': height = atoi(optarg); break; case 's': seedprob = atof(optarg); break; case 'r': genrate = atoi(optarg); break; case 'n': gens = atoi(optarg); break; default: usage(argv[0]); return 1; } } /* Sanity. Let's check it. */ if (seedprob <= 0 || seedprob > 1) { fprintf(stderr, "Error: Ensure 0 < SEEDPROB <= 1\n"); return 2; } init_curses(); init_game(); getch(); fini_game(); fini_curses(); return 0; } void init_curses() { int maxwidth, maxheight; initscr(); noecho(); raw(); curs_set(0); /* Leave room for spaces between cells, but allow for one more cell on odd- * width terminals. */ maxwidth = (COLS - 2) / 2 + COLS % 1; maxheight = (LINES - 2); if (width == 0 || width > maxwidth) { width = maxwidth; } if (height == 0 || height > maxheight) { height = maxheight; } } void init_game() { size_t size; int i, j, n; double r; n = width * height; /* Allocate and initialize the grids. */ size = sizeof(bool) * n; grid_cur = malloc(size); memset(grid_cur, 0, size); grid_next = malloc(size); memset(grid_next, 0, size); srand(time(NULL)); /* Generate the seed pattern. */ for (i = 0; i < n; ++i) { r = rand() / (double) RAND_MAX; if (r > seedprob) { /* Dead. */ grid_cur[i] = false; } else { /* Alive. */ grid_cur[i] = true; } } /* Print the grid. */ for (i = 0; i < height; ++i) { for (j = 0; j < width; ++j) { mvprintw(1 + i, 1 + j * 2, "%c", grid_cur[i * width + j] ? 'o' : ' '); } } } void fini_game() { free(grid_cur); free(grid_next); } void fini_curses() { endwin(); } void usage(const char *invocation) { printf("Usage: %s [-w WIDTH] [-h HEIGHT] [-s SEEDPROB] [-r RATE] " "[-n GENERATIONS]", invocation); }