/* $NetBSD: testpat.c,v 1.5 2021/05/02 12:50:46 rillig Exp $ */ /*- * Copyright (c) 2016 Nathanial Sloss * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include __RCSID("$NetBSD: testpat.c,v 1.5 2021/05/02 12:50:46 rillig Exp $"); #include #include #include #include #include #include #include #include #include #include static int colour_list[6] = { COLOR_YELLOW, COLOR_CYAN, COLOR_GREEN, COLOR_MAGENTA, COLOR_RED, COLOR_BLUE, }; static int numcolours = (int)__arraycount(colour_list); int main(int argc, char *argv[]) { int i, col, colour, line, x_limit, y_limit, colourOK, spacing; int xpos, ypos, spacing_residual, spacing_start, spacing_end; int grid_x, grid_y, **circle_pos; size_t ncpos; float grid_unit; const char *title = "NetBSD"; float coord_x, circle_int; float a_axis, b_axis; if (!initscr()) { errx(EXIT_FAILURE, "Unknown terminal type"); } curs_set(0); if (argc > 2) { endwin(); fprintf(stderr, "Usage: %s ", getprogname()); return EXIT_FAILURE; } if (argc == 2) { title = argv[1]; if (strlen(title) >= (size_t)COLS) { endwin(); errx(EXIT_FAILURE, "Title string is longer than display cols"); } } colourOK = has_colors(); if (COLS < 13 || LINES < 13) { endwin(); errx(EXIT_FAILURE, "Terminal size must be at least 72x25."); } if (colourOK) { start_color(); init_pair(0, COLOR_WHITE, COLOR_BLACK); init_pair(1, COLOR_WHITE, COLOR_RED); init_pair(2, COLOR_WHITE, COLOR_GREEN); init_pair(3, COLOR_WHITE, COLOR_YELLOW); init_pair(4, COLOR_WHITE, COLOR_BLUE); init_pair(5, COLOR_WHITE, COLOR_MAGENTA); init_pair(6, COLOR_WHITE, COLOR_CYAN); init_pair(7, COLOR_BLACK, COLOR_WHITE); attrset(COLOR_PAIR(0)); } x_limit = (COLS - 1) / 2; x_limit = x_limit * 2; y_limit = (LINES - 2) / 2; y_limit = y_limit * 2; spacing = 2 * y_limit / numcolours; spacing_residual = ((2 * y_limit) % numcolours) / 2; a_axis = y_limit / 2; b_axis = y_limit; grid_unit = b_axis / 13; grid_y = grid_unit; grid_x = grid_unit * 2; ncpos = y_limit * sizeof(*circle_pos) + y_limit * 2 * sizeof(**circle_pos); circle_pos = malloc(ncpos); if (circle_pos == NULL) { endwin(); errx(EXIT_FAILURE, "Can't allocate circle positions"); } for (i = 0; i < y_limit; i++) { circle_pos[i] = (void *)&circle_pos[y_limit + i * 2]; circle_pos[i][0] = circle_pos[i][1] = -1; } for (i = 0; i < y_limit; i++) { /* Draw an ellipse (looks more circular.) */ circle_int = (i - a_axis) / a_axis; circle_int = 1 - powf(circle_int, 2); circle_int = circle_int * powf(b_axis, 2); #if 0 /* Draw a circle, commented out as elipse looks better.*/ circle_int = powf(a_axis, 2) - powf(i - a_axis, 2); #endif coord_x = sqrtf(circle_int); circle_pos[i][0] = (-coord_x + ((float)x_limit / 2)); circle_pos[i][1] = (coord_x + ((float)x_limit / 2)); } clear(); attron(A_ALTCHARSET); move(0, 0); /* Draw a grid. */ for (line = 1; line < y_limit; line += grid_y) { for (col = 1; col < x_limit; col = col + grid_x) { xpos = col; while ((xpos < col + grid_x - 1) && (xpos < x_limit)) { mvaddch(line + grid_y - 1, xpos, 113 | A_ALTCHARSET); xpos++; } if (xpos < x_limit) mvaddch(line + grid_y - 1, xpos, 110 | A_ALTCHARSET); } ypos = line; while (ypos < line + grid_y - 1) { for (col = grid_x - 1; col < x_limit; col += grid_x) { mvaddch(ypos, col + 1, 120 | A_ALTCHARSET); } ypos++; } } for (line = 1; line < y_limit; line += grid_y) { mvaddch(line + grid_y - 1, 0, 116 | A_ALTCHARSET); mvaddch(line + grid_y - 1, x_limit, 117 | A_ALTCHARSET); ypos = line; while (ypos < line + grid_y - 1) { mvaddch(ypos, 0, 120 | A_ALTCHARSET); mvaddch(ypos, x_limit, 120 | A_ALTCHARSET); ypos++; } } for (col = 1; col < x_limit; col += grid_x) { mvaddch(0, col + grid_x - 1, 119 | A_ALTCHARSET); mvaddch(y_limit, col + grid_x - 1, 118 | A_ALTCHARSET); xpos = col; while ((xpos < col + grid_x - 1) && (xpos < x_limit)) { mvaddch(0, xpos, 113 | A_ALTCHARSET); mvaddch(y_limit, xpos, 113 | A_ALTCHARSET); xpos++; } } mvaddch(0, 0, 108 | A_ALTCHARSET); mvaddch(0, x_limit, 107 | A_ALTCHARSET); mvaddch(y_limit, 0, 109 | A_ALTCHARSET); mvaddch(y_limit, x_limit, 106 | A_ALTCHARSET); /* Draw a white circle. */ for (i = 1; i < y_limit; i++) { for (col = circle_pos[i][0]; col <= circle_pos[i][1]; col++) { mvaddch(i, col, 32 | A_REVERSE); } } /* Add title segment. */ for (i = roundf(1 * grid_unit); i < roundf(2 * grid_unit); i++) { if (colourOK) attrset(COLOR_PAIR(COLOR_BLACK)); for (col = roundf((4 * grid_unit * 2) + circle_pos[y_limit / 2][0]); col <= roundf((9 * grid_unit * 2) + circle_pos[y_limit / 2][0]); col++) mvaddch(i, col, ' '); } i = roundf(1.4 * grid_unit); col = y_limit - (strlen(title) / 2) + circle_pos[y_limit / 2][0]; mvprintw(i, col, "%s", title); /* Add black segments at top. */ for (line = roundf(2 * grid_unit); line < 4 * grid_unit; line++) { if (colourOK) attrset(COLOR_PAIR(COLOR_BLACK)); for (col = 0; col <= roundf((3.5 * grid_unit * 2)); col++) { xpos = col + circle_pos[y_limit / 2][0]; if (xpos >= circle_pos[line][0] && xpos <= circle_pos[line][1]) mvaddch(line, xpos, ' '); } for (col = roundf((9.5 * grid_unit * 2)); col < roundf((13 * grid_unit * 2)); col++) { xpos = col + circle_pos[y_limit / 2][0]; if (xpos >= circle_pos[line][0] && xpos <= circle_pos[line][1]) mvaddch(line, xpos, ' '); } } /* Add black and white squares close to top. */ int gap = (circle_pos[(int)(5 * grid_unit)][1] - circle_pos[(int)(5 * grid_unit)][0]) / 13; for (i = roundf(3 * grid_unit); i < roundf(4 * grid_unit); i++) { for (xpos = 0; xpos <= x_limit; xpos += 2 * gap) { if (colourOK) attrset(COLOR_PAIR(COLOR_BLACK)); for (col = xpos; col < xpos + gap; col++) { if (col >= circle_pos[i][0] && col <= circle_pos[i][1]) mvaddch(i, col, ' '); } if (colourOK) attrset(COLOR_PAIR(COLOR_WHITE)); for (col = xpos + gap ; col < xpos + (2 * gap); col++) { if (col >= circle_pos[i][0] && col <= circle_pos[i][1]) mvaddch(i, col, ' '); } } } /* Add colour bars. */ for (i = 0; i < numcolours; i++) { colour = colour_list[i]; if (colourOK) attrset(COLOR_PAIR(colour)); if (i == 0) spacing_start = 0; else spacing_start = (spacing * i) + spacing_residual; if (i == numcolours - 1) spacing_end = circle_pos[y_limit / 2][1]; else spacing_end = (spacing * (i + 1)) + spacing_residual; for (line = roundf(4 * grid_unit); line < (y_limit / 2); line++) { for (col = spacing_start; col < spacing_end; col++) { xpos = col + circle_pos[y_limit / 2][0]; if (xpos >= circle_pos[line][0] && xpos <= circle_pos[line][1]) mvprintw(line, xpos, " "); } } } /* Add black segment under centre line. */ for (line = y_limit / 2; line < (9.5 * grid_unit); line++) { if (colourOK) attrset(COLOR_PAIR(COLOR_BLACK)); for (col = circle_pos[line][0]; col <= circle_pos[line][1]; col++) mvaddch(line, col, ' '); for (col = roundf((1.5 * grid_unit * 2)); col < roundf((4.3 * grid_unit * 2)); col++) { xpos = col + circle_pos[y_limit / 2][0]; if (xpos >= circle_pos[line][0] && xpos < circle_pos[line][1]) mvaddch(line, xpos, 120 | A_ALTCHARSET); } for (col = roundf((4.3 * grid_unit * 2)); col < roundf((7.6 * grid_unit * 2)); col++) { xpos = col + circle_pos[y_limit / 2][0]; if (xpos >= circle_pos[line][0] && xpos < circle_pos[line][1]) mvaddch(line, xpos, '|'); } for (col = roundf((7.6 * grid_unit * 2)); col < roundf((11.5 * grid_unit * 2)); col++) { xpos = col + circle_pos[y_limit / 2][0]; if (xpos >= circle_pos[line][0] && xpos < circle_pos[line][1]) mvaddch(line, xpos, 97 | A_ALTCHARSET); } } /* Add black segment close to bottom. */ for (line = roundf(9.5 * grid_unit); line <= (10.5 * grid_unit); line++) { if (colourOK) attrset(COLOR_PAIR(COLOR_BLACK)); for (col = roundf((0 * grid_unit * 2)); col < roundf((4 * grid_unit * 2)); col++) { xpos = col + circle_pos[y_limit / 2][0]; if (xpos >= circle_pos[line][0] && xpos < circle_pos[line][1]) mvaddch(line, xpos, ' '); } for (col = roundf((4 * grid_unit * 2)); col < roundf((6.5 * grid_unit * 2)); col++) { xpos = col + circle_pos[y_limit / 2][0]; if (xpos >= circle_pos[line][0] && xpos < circle_pos[line][1]) mvaddch(line, xpos, 97 | A_ALTCHARSET); } if (colourOK) attrset(COLOR_PAIR(COLOR_WHITE)); for (col = roundf((6.5 * grid_unit * 2)); col < roundf((9 * grid_unit * 2)); col++) { xpos = col + circle_pos[y_limit / 2][0]; if (xpos >= circle_pos[line][0] && xpos < circle_pos[line][1]) mvaddch(line, xpos, 97 | A_ALTCHARSET); } for (col = roundf((9 * grid_unit * 2)); col < roundf((13 * grid_unit * 2)); col++) { xpos = col + circle_pos[y_limit / 2][0]; if (xpos >= circle_pos[line][0] && xpos < circle_pos[line][1]) mvaddch(line, xpos, ' '); } } /* Add name segment close to bottom. */ for (line = roundf(10.5 * grid_unit); line < (12 * grid_unit); line++) { if (colourOK) attrset(COLOR_PAIR(COLOR_BLACK)); for (col = roundf(3.5 * grid_unit * 2); col <= roundf(9.5 * grid_unit * 2); col++) { xpos = col + circle_pos[y_limit / 2][0]; if (xpos >= circle_pos[line][0] && xpos < circle_pos[line][1]) mvaddch(line, xpos, ' '); } if (colourOK) attrset(COLOR_PAIR(COLOR_WHITE)); for (col = roundf(0 * grid_unit * 2); col <= roundf(3.5 * grid_unit * 2); col++) { xpos = col + circle_pos[y_limit / 2][0]; if (xpos >= circle_pos[line][0] && xpos < circle_pos[line][1]) mvaddch(line, xpos, ' '); } for (col = roundf(9.5 * grid_unit * 2); col <= roundf(13 * grid_unit * 2); col++) { xpos = col + circle_pos[y_limit / 2][0]; if (xpos >= circle_pos[line][0] && xpos < circle_pos[line][1]) mvaddch(line, xpos, ' '); } } /* Add yellow segment at bottom. */ for (line = 12 * grid_unit; line < y_limit; line++) { if (colourOK) attrset(COLOR_PAIR(COLOR_YELLOW)); for (col = circle_pos[line][0]; col <= circle_pos[line][1]; col++) mvaddch(line, col, ' '); if (colourOK) attrset(COLOR_PAIR(COLOR_RED)); for (col = roundf((6 * grid_unit * 2)); col < roundf((7 * grid_unit * 2)); col++) { xpos = col + circle_pos[y_limit / 2][0]; if (xpos >= circle_pos[line][0] && xpos < circle_pos[line][1]) mvaddch(line, xpos, ' '); } } if (colourOK) attrset(COLOR_PAIR(COLOR_BLACK)); for (line = 6 * grid_unit; line <= (7 * grid_unit) + 1; line++) { if (colourOK) attrset(COLOR_PAIR(COLOR_BLACK)); col = x_limit / 2; if (line != a_axis) { mvaddch(line, col - 1, ' '); mvaddch(line, col, 120 | A_ALTCHARSET); mvaddch(line, col + 1, ' '); } } line = y_limit / 2; for (col = 1; col < x_limit; col = col + grid_x) { xpos = col; while (xpos < col + grid_x - 1) { if (xpos >= circle_pos[line][0] && xpos < circle_pos[line][1]) mvaddch(line, xpos, 113 | A_ALTCHARSET); xpos++; } if (xpos >= circle_pos[line][0] && xpos < circle_pos[line][1]) mvaddch(line, xpos, 110 | A_ALTCHARSET); } line = y_limit / 2; col = x_limit / 2; mvaddch(line, col, 110 | A_ALTCHARSET); mvaddch(line, col - 1, 113 | A_ALTCHARSET); mvaddch(line, col + 1, 113 | A_ALTCHARSET); refresh(); getch(); endwin(); return EXIT_SUCCESS; }