nuke/nuke.c

461 lines
13 KiB
C
Raw Normal View History

2019-01-02 20:30:11 -08:00
/**
* ASCII explosion from squeamish ossifrage at:
* https://codegolf.stackexchange.com/questions/24462/display-the-explosion-of-a-star-in-ascii-art/24554#24554
*/
2019-01-03 01:51:33 -08:00
#include "colors.h"
#include "list.h"
#include "globe.h"
2019-01-02 20:30:11 -08:00
#include <curses.h>
2019-01-03 01:51:33 -08:00
#include <ftw.h>
2019-01-02 20:30:11 -08:00
#include <math.h>
#include <poll.h>
2019-01-03 01:51:33 -08:00
#include <signal.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
2019-01-02 20:30:11 -08:00
2019-01-03 04:38:27 -08:00
#define NUM_FRAMES 100
2019-01-02 20:30:11 -08:00
#define NUM_BLOBS 800
#define PERSPECTIVE 50.0
#define ESCDELAY 10
typedef struct {
double x,y,z;
} spaceblob;
double prng() {
static long long s=1;
s = s * 1488248101 + 981577151;
return ((s % 65536) - 32768) / 32768.0;
}
static void sighandler(int sig) {
(void)sig;
curs_set(1);
endwin();
exit(EXIT_FAILURE);
}
2019-01-03 01:51:33 -08:00
int color_ramp1[] = {1,1,1,2,2,2,3,3,3,1,1, -1};
int bold_ramp1[] = {1,1,0,1,0,0,1,0,0,0,0, -1};
int color_ramp2[] = {1,1,1,1,2,2,2,2,2,3,3,3,3,3,1,1,1,1,1,1, -1};
int bold_ramp2[] = {1,1,0,0,1,1,0,0,0,1,1,0,0,0,0,0,0,0,0,0, -1};
void draw_frame(int i, int midx, int midy, spaceblob* blobs) {
int rows,cols;
getmaxyx(stdscr,rows,cols);
int maxx,minx,maxy,miny;
minx = -midx;
maxx = cols+minx-1;
miny = -midy;
maxy = rows+miny-1;
chtype line[cols+1];
for (int y=miny; y<=maxy; y++) {
int row = y+midy;
for (int x=minx; x<=maxx; x++) {
int col = x+midx;
/* Show expanding star in next 7 frames */
if (i<8) {
double r = sqrt(x*x + 4*y*y);
if (r < i*2)
mvaddch(row,col,'@');
continue;
}
/* Otherwise show blast wave */
double r = sqrt(x*x + 4*y*y) * (0.5 + (prng()/3.0)*cos(16.*atan2(y*2.+0.01,x+0.01))*.3);
int v = i - r - 7;
if (v<0) {
if (i<19) {
int attr = 0;//color_ramp1[i-8];
line[col] = "%@W#H=+~-:."[i-8] | attr;
} else {
line[col] = ' ';
}
} else if (v<20) {
int attr = 0;//color_ramp2[v];
line[col] = " .:!HIOMW#%$&@08O=+-"[v] | attr;
} else {
line[col] = ' ';
}
}
if (i >= 8) {
line[cols+1] = '\0';
mvaddchstr(row,0,line);
2019-01-02 20:30:11 -08:00
}
}
2019-01-03 01:51:33 -08:00
/* Add blobs with perspective effect */
if (i>6) {
int i0 = i-6;
for (int j=0; j<NUM_BLOBS; j++) {
double bx = blobs[j].x * i0;
double by = blobs[j].y * i0;
double bz = blobs[j].z * i0;
if (bz<5-PERSPECTIVE || bz>PERSPECTIVE) continue;
int x = midx + bx * PERSPECTIVE / (bz+PERSPECTIVE);
int y = midy + by * PERSPECTIVE / (bz+PERSPECTIVE);
if (x>=0 && x<cols && y>=0 && y<rows) {
int row = y, col = x;
mvaddch(row,col,(bz>40) ? '.' : (bz>-20) ? 'o' : '@');
2019-01-02 20:30:11 -08:00
}
}
}
}
2019-01-03 01:51:33 -08:00
int unlink_cb(const char *fpath, const struct stat *sb, int typeflag, struct FTW *ftwbuf)
{
int rv = remove(fpath);
if (rv)
perror(fpath);
return rv;
}
int rmrf(char *path)
{
return nftw(path, unlink_cb, 64, FTW_DEPTH | FTW_PHYS);
}
int main(int argc, char *argv[])
{
2019-01-03 04:38:27 -08:00
if (argc > 1 && strcmp(argv[1], "--help") == 0) {
printf("nuke: Nuke files from orbit.\nUsage: nuke [files...]\n");
return EXIT_SUCCESS;
}
char *failure_message = "nothing was deleted";
2019-01-03 04:38:27 -08:00
2019-01-03 01:51:33 -08:00
int rows,cols;
int delay=1E5;
2019-01-02 20:30:11 -08:00
spaceblob *blobs;
2019-01-03 01:51:33 -08:00
int capacity = 32, num_files= 0;
char **files = malloc(sizeof(char*)*capacity);
for (int i = 1; i < argc; i++) {
add_files(argv[i], &files, &num_files, &capacity);
}
if (argc == 1) {
add_file("<testing grounds>", &files, &num_files, &capacity);
}
2019-01-02 20:30:11 -08:00
/* Initialize ncurses and get window dimensions */
//initscr();
char* term_type = getenv("TERM");
if (term_type == NULL || *term_type == '\0') {
term_type = "unknown";
}
FILE* term_in = fopen("/dev/tty", "r");
if (term_in == NULL) {
perror("fopen(/dev/tty)");
exit(EXIT_FAILURE);
}
SCREEN* main_screen = newterm(term_type, stdout, term_in);
set_term(main_screen);
signal(SIGINT, sighandler);
start_color();
2019-01-03 01:51:33 -08:00
init_pair(WHITE, COLOR_WHITE, COLOR_BLACK);
init_pair(YELLOW, COLOR_YELLOW, COLOR_BLACK);
init_pair(RED, COLOR_RED, COLOR_BLACK);
init_pair(BLUE, COLOR_BLUE, COLOR_BLACK);
init_pair(GREEN, COLOR_GREEN, COLOR_BLACK);
init_pair(BLACK, COLOR_BLACK, COLOR_BLACK);
for (int i = 0; color_ramp1[i] >= 0; i++)
color_ramp1[i] = COLOR_PAIR(color_ramp1[i]) | (bold_ramp1[i] ? A_BOLD : 0);
for (int i = 0; color_ramp2[i] >= 0; i++)
color_ramp2[i] = COLOR_PAIR(color_ramp2[i]) | (bold_ramp2[i] ? A_BOLD : 0);
const int BLACK_ON_RED = 7, BLACK_ON_GREEN = 8;
2019-01-02 20:30:11 -08:00
init_pair(BLACK_ON_RED, COLOR_BLACK, COLOR_RED);
2019-01-03 01:51:33 -08:00
init_pair(BLACK_ON_GREEN, COLOR_BLACK, COLOR_GREEN);
2019-01-02 20:30:11 -08:00
getmaxyx(stdscr,rows,cols);
/* Generate random blob coordinates */
blobs = (spaceblob*)malloc(NUM_BLOBS * sizeof(spaceblob));
2019-01-03 01:51:33 -08:00
for (int i=0; i<NUM_BLOBS; i++) {
double bx,by,bz,br;
2019-01-02 20:30:11 -08:00
bx = prng();
by = prng();
bz = prng();
br = sqrt(bx*bx + by*by + bz*bz);
blobs[i].x = (bx / br) * (1.3 + 0.2 * prng());
blobs[i].y = (0.5 * by / br) * (1.3 + 0.2 * prng());;
blobs[i].z = (bz / br) * (1.3 + 0.2 * prng());;
}
2019-01-03 01:51:33 -08:00
curs_set(0); /* hide text cursor */
noecho();
2019-01-03 04:38:27 -08:00
keypad(stdscr, 1);
2019-01-02 20:30:11 -08:00
2019-01-03 01:51:33 -08:00
// First frame
int max_line = 0;
for (int f = 0; f < num_files; f++) {
int len = strlen(files[f]);
if (len > max_line) max_line = len;
}
2019-01-02 20:30:11 -08:00
2019-01-03 04:38:27 -08:00
erase();
int padwidth = max_line < cols*3/4 ? max_line : cols*3/4;
int padheight = num_files < rows*3/4 ? num_files : rows*3/4;
int padframer = rows/2-padheight/2-1;
int padframec = cols/2-padwidth/2-1;
WINDOW *padframe = newwin(padheight+2, padwidth+2, padframer, padframec);
if (!padframe) {
failure_message = "Couldn't create pad frame";
goto exit_failure;
2019-01-03 01:51:33 -08:00
}
2019-01-03 04:38:27 -08:00
wattron(padframe, COLOR_PAIR(YELLOW) | A_BOLD);
box(padframe, 0, 0);
WINDOW *pad = newpad(padheight, max_line);
if (!pad) {
failure_message = "Couldn't create pad";
goto exit_failure;
}
2019-01-03 01:51:33 -08:00
/* Display confirmation */
attron(COLOR_PAIR(BLACK_ON_RED));
char *description = num_files == 1 ? " TARGET: " : " TARGETS: ";
2019-01-03 04:38:27 -08:00
mvprintw(padframer-2, cols/2 - strlen(description)/2, "%s", description);
attron(COLOR_PAIR(BLACK_ON_RED) | A_BLINK);
const char confirm[] = " FIRE NUKE? y/n ";
mvprintw(padframer+padheight+3, cols/2 - strlen(confirm)/2, "%s", confirm);
attroff(COLOR_PAIR(BLACK_ON_RED) | A_BLINK);
2019-01-03 01:51:33 -08:00
refresh();
2019-01-02 20:30:11 -08:00
2019-01-03 04:38:27 -08:00
//immedok(pad, 1);
int scrollx = 0, scrolly = 0;
2019-01-02 20:30:11 -08:00
2019-01-03 01:51:33 -08:00
timeout(10);
2019-01-03 04:38:27 -08:00
while (1) {
// Draw scroll bar
if (padheight < num_files) {
mvaddch(padframer, padframec+padwidth+2, ACS_UARROW);
mvaddch(padframer+padheight+1, padframec+padwidth+2, ACS_DARROW);
for (int i = 1; i <= padheight; i++)
mvaddch(padframer+i, padframec+padwidth+2, ACS_VLINE);
for (int i = scrolly*padheight/num_files; i <= scrolly*padheight/num_files+padheight*padheight/num_files; i++)
mvaddch(padframer+1+i, padframec+padwidth+2, ACS_CKBOARD);
2019-01-02 20:30:11 -08:00
}
2019-01-03 04:38:27 -08:00
box(padframe, 0, 0);
if (wrefresh(padframe) == ERR) goto exit_failure;
// Redo this each time so that pad doesn't have to hold everything in memory
werase(pad);
for (int f = scrolly; f < scrolly+padheight; f++) {
mvwprintw(pad,f-scrolly,0,"%s",files[f]);
}
if (prefresh(pad, 0, scrollx, padframer+1, padframec+1, padframer+padheight, padframec+padwidth) == ERR)
2019-01-03 01:51:33 -08:00
goto exit_failure;
2019-01-03 04:38:27 -08:00
//prefresh(pad, scrolly, scrollx, 1, 1, 1+padheight, 1+padwidth);
//prefresh(pad, scrolly, scrollx, rows/2-padheight/2, cols/2-padwidth/2, rows/2+padheight/2, cols/2+padwidth/2);
refresh();
get_input:
switch(getch()) {
case 'y':
goto do_the_nuking;
case 'n': case 'q': case 27:
goto exit_failure;
case KEY_DOWN: case 'j': case ' ':
scrolly++;
break;
case KEY_UP: case 'k':
scrolly--;
break;
case KEY_RIGHT: case 'l':
scrollx++;
break;
case KEY_LEFT: case 'h':
scrollx--;
break;
case 'J': case KEY_NPAGE:
scrolly += num_files/20+1;
break;
case 'K': case KEY_PPAGE:
scrolly -= num_files/20+1;
break;
case KEY_SRIGHT: case 'L':
scrollx += 10;
break;
case KEY_SLEFT: case 'H':
scrollx -= 10;
break;
case KEY_HOME: case 'g':
scrolly = 0;
break;
case KEY_END: case 'G':
scrolly = num_files-padheight;
break;
default:
goto get_input;
}
if (scrolly < 0) scrolly = 0;
if (scrolly > num_files-padheight) scrolly = num_files-padheight;
if (scrollx < 0) scrollx = 0;
if (scrollx > max_line-padwidth) scrollx = max_line-padwidth;
2019-01-03 01:51:33 -08:00
}
2019-01-02 20:30:11 -08:00
2019-01-03 04:38:27 -08:00
do_the_nuking: ;
double zoom = .8; // percent of viewport globe will fill
2019-01-02 20:30:11 -08:00
2019-01-03 04:38:27 -08:00
const double targeting_time = 2.5;
const double anticipation = 0.5;
const double firing_time = 1.0;
const double nuking_time = .5;
2019-01-03 05:23:55 -08:00
const double aftermath = 100.0;
2019-01-03 01:51:33 -08:00
int f = 0;
int targetr, targetc;
2019-01-03 04:38:27 -08:00
double t = 0.0;
for (; t < targeting_time + anticipation + firing_time; t += 1./30.) {
2019-01-02 20:30:11 -08:00
erase();
2019-01-03 04:38:27 -08:00
draw_stars(t);
draw_globe(t, zoom);
if (get_target_pos(t, zoom, &targetr, &targetc)) {
// If visible draw '*' on globe
attron(COLOR_PAIR(YELLOW));
mvaddch(targetr,targetc, '*' | A_BOLD);
attroff(COLOR_PAIR(YELLOW));
}
2019-01-03 05:23:55 -08:00
draw_clouds(t, zoom);
2019-01-03 04:38:27 -08:00
// Draw crosshair
double wobble = t > targeting_time ? 0. : targeting_time - t;
int r = targetr, c = targetc;
if (wobble > .01) {
r = mix(r, rows*hillnoise(784,5.*t), wobble);
c = mix(c, cols*hillnoise(-784,5.*t), wobble);
}
int attr = COLOR_PAIR(RED);
attron(attr);
mvaddch(r-1,c-2,ACS_ULCORNER | A_BOLD);
mvaddch(r-1,c-1,ACS_HLINE | A_BOLD);
mvaddch(r-1,c+1,ACS_HLINE | A_BOLD);
mvaddch(r-1,c+2,ACS_URCORNER | A_BOLD);
mvaddch(r+1,c+2,ACS_LRCORNER | A_BOLD);
mvaddch(r+1,c+1,ACS_HLINE | A_BOLD);
mvaddch(r+1,c-1,ACS_HLINE | A_BOLD);
mvaddch(r+1,c-2,ACS_LLCORNER | A_BOLD);
mvaddch(r,c,'X' | A_BOLD);
attroff(attr);
// Draw label
if (get_target_pos(t, zoom, &targetr, &targetc)) {
attron(COLOR_PAIR(RED));
mvaddch(targetr-1,targetc+1,'/');
attroff(COLOR_PAIR(RED));
attron(COLOR_PAIR(BLACK_ON_RED));
mvprintw(targetr-2,targetc+2,"%s", argc > 1 ? argv[f+1] : "testing grounds");
2019-01-03 04:38:27 -08:00
attroff(COLOR_PAIR(BLACK_ON_RED));
if (argc > 2)
f = (f + 1) % (argc - 1);
}
// Draw nuke
if (t > targeting_time + anticipation) {
double k = (t-targeting_time-anticipation)/firing_time;
int nuker = mix(rows, targetr, k), nukec = mix(cols/2, targetc, k);
attron(COLOR_PAIR(WHITE) | A_BOLD);
2019-01-03 05:49:28 -08:00
mvaddch(nuker, nukec, "A^' "[(int)(k*4.)]);
2019-01-03 04:38:27 -08:00
attroff(COLOR_PAIR(WHITE) | A_BOLD);
2019-01-03 05:35:38 -08:00
// Draw nuke label
attron(COLOR_PAIR(RED));
2019-01-03 05:49:28 -08:00
mvaddch(nuker-1,nukec-1,'\\');
2019-01-03 05:35:38 -08:00
attroff(COLOR_PAIR(RED));
attron(COLOR_PAIR(BLACK_ON_RED));
2019-01-03 05:49:28 -08:00
mvprintw(nuker-2,nukec-2-5,"%0.3f", (targeting_time + anticipation + firing_time)-t);
2019-01-03 05:35:38 -08:00
attroff(COLOR_PAIR(BLACK_ON_RED));
2019-01-03 04:38:27 -08:00
}
2019-01-02 20:30:11 -08:00
refresh();
2019-01-03 04:38:27 -08:00
// Check for input
2019-01-03 01:51:33 -08:00
int ch = getch();
if (ch == 'q' || ch == 27) {
2019-01-03 04:38:27 -08:00
if (t <= targeting_time + anticipation) {
goto exit_failure;
} else {
goto exit_success;
}
2019-01-02 20:30:11 -08:00
}
2019-01-03 01:51:33 -08:00
usleep(33333); // 30 FPS
}
2019-01-03 04:38:27 -08:00
t += nuking_time;
2019-01-03 01:51:33 -08:00
for (int i=1; i<NUM_FRAMES; i++) {
draw_frame(i, targetc, targetr, blobs);
refresh();
2019-01-02 20:30:11 -08:00
/* Quit early? */
int ch = getch();
if (ch == 'q' || ch == 27) {
goto exit_success;
}
2019-01-03 01:51:33 -08:00
usleep(33333); // 30 FPS
}
2019-01-02 20:30:11 -08:00
2019-01-03 04:38:27 -08:00
for (; t < targeting_time + anticipation + firing_time + nuking_time + aftermath; t += 1./30.) {
2019-01-03 01:51:33 -08:00
erase();
2019-01-03 04:38:27 -08:00
draw_stars(t);
draw_globe(t, zoom);
// Draw crater
if (get_target_pos(t, zoom, &targetr, &targetc)) {
attron(COLOR_PAIR(BLACK) | A_BOLD);
2019-01-03 05:23:55 -08:00
for (int i = 31; i >= 0; i--) {
double rad = 4./31.*i;
double a = i*2*M_PI*GOLDEN_RATIO;
int r = targetr+(int)(rad*sin(a)/2), c = targetc+(int)(rad*cos(a));
if ((r-rows/2)*(r-rows/2) + (c/2-cols/4)*(c/2-cols/4) <= (zoom*rows/2)*(zoom*rows/2))
mvaddch(r, c, " #*."[i*4/31]);
}
attroff(COLOR_PAIR(BLACK) | A_BOLD);
/*
for (double =target_lat
2019-01-03 04:38:27 -08:00
mvprintw(targetr-1,targetc-1,"****");
mvprintw(targetr,targetc-2, "** **");
mvprintw(targetr+1,targetc-1,"****");
2019-01-03 05:23:55 -08:00
*/
}
draw_clouds(t, zoom);
if (get_target_pos(t, zoom, &targetr, &targetc)) {
// Draw label
attron(COLOR_PAIR(GREEN));
mvaddch(targetr-1,targetc+1,'/');
attroff(COLOR_PAIR(GREEN));
attron(COLOR_PAIR(BLACK_ON_GREEN));
mvprintw(targetr-2,targetc+2," DELETED ");
attroff(COLOR_PAIR(BLACK_ON_GREEN));
2019-01-03 04:38:27 -08:00
}
2019-01-03 05:29:25 -08:00
2019-01-03 05:35:38 -08:00
attron(COLOR_PAIR(BLACK_ON_GREEN) | A_BLINK);
2019-01-03 05:29:25 -08:00
const char *any_key = " PRESS ANY KEY TO EXIT ";
2019-01-03 05:35:38 -08:00
mvprintw(rows-(1.-zoom)*rows/4, cols/2-strlen(any_key)/2, any_key);
attroff(COLOR_PAIR(BLACK_ON_GREEN) | A_BLINK);
2019-01-03 05:29:25 -08:00
2019-01-03 01:51:33 -08:00
refresh();
2019-01-03 05:23:55 -08:00
if (getch() != -1)
2019-01-03 01:51:33 -08:00
goto exit_success;
usleep(33333); // 30 FPS
2019-01-02 20:30:11 -08:00
}
exit_success:
curs_set(1); /* unhide cursor */
endwin(); /* Exit ncurses */
2019-01-03 01:51:33 -08:00
for (int i = 1; i < argc; i++) {
if (rmrf(argv[i]) == 0) {
printf("deleted %s\n", argv[i]);
} else {
printf("unable to delete %s\n", argv[i]);
return EXIT_FAILURE;
}
2019-01-02 20:30:11 -08:00
}
return EXIT_SUCCESS;
exit_failure:
curs_set(1); /* unhide cursor */
endwin(); /* Exit ncurses */
2019-01-03 04:38:27 -08:00
printf("%s\n",failure_message);
2019-01-02 20:30:11 -08:00
return EXIT_FAILURE;
}