/* * demo demo for libcaca * Copyright (c) 2003-2010 Sam Hocevar * All Rights Reserved * * This program is free software. It comes without any warranty, to * the extent permitted by applicable law. You can redistribute it * and/or modify it under the terms of the Do What The Fuck You Want * To Public License, Version 2, as published by Sam Hocevar. See * http://sam.zoy.org/wtfpl/COPYING for more details. */ #include "config.h" #if !defined(__KERNEL__) # include # include # include #endif #ifndef M_PI # define M_PI 3.14159265358979323846 #endif #include "caca.h" static void display_menu(void); static void demo_all(void); static void demo_dots(void); static void demo_lines(void); static void demo_boxes(void); static void demo_ellipses(void); static void demo_triangles(void); /*static void demo_sprites(void);*/ static void demo_render(void); int bounds = 0; int outline = 0; int dithering = 0; #if 0 caca_sprite_t *sprite = NULL; #endif caca_canvas_t *cv; caca_display_t *dp; int main(int argc, char **argv) { void (*demo)(void) = NULL; int quit = 0; cv = caca_create_canvas(80, 24); if(cv == NULL) { printf("Failed to create canvas\n"); return 1; } dp = caca_create_display(cv); if(dp == NULL) { printf("Failed to create display\n"); return 1; } caca_set_display_time(dp, 40000); /* Initialize data */ #if 0 sprite = caca_load_sprite("caca.txt"); if(!sprite) sprite = caca_load_sprite("examples/caca.txt"); #endif /* Disable cursor */ caca_set_mouse(dp, 0); /* Main menu */ display_menu(); caca_refresh_display(dp); /* Go ! */ while(!quit) { caca_event_t ev; int menu = 0, mouse = 0, xmouse = 0, ymouse = 0; while(caca_get_event(dp, CACA_EVENT_ANY, &ev, 0)) { if(demo && (caca_get_event_type(&ev) & CACA_EVENT_KEY_PRESS)) { menu = 1; demo = NULL; } else if(caca_get_event_type(&ev) & CACA_EVENT_KEY_PRESS) { switch(caca_get_event_key_ch(&ev)) { case 'q': case 'Q': case CACA_KEY_ESCAPE: demo = NULL; quit = 1; break; case 'o': case 'O': outline = (outline + 1) % 3; display_menu(); caca_refresh_display(dp); break; case 'b': case 'B': bounds = (bounds + 1) % 2; display_menu(); caca_refresh_display(dp); break; #if 0 case 'd': case 'D': dithering = (dithering + 1) % 5; caca_set_feature(cv, dithering); display_menu(); caca_refresh_display(dp); break; #endif case 'f': case 'F': demo = demo_all; break; case '1': demo = demo_dots; break; case '2': demo = demo_lines; break; case '3': demo = demo_boxes; break; case '4': demo = demo_triangles; break; case '5': demo = demo_ellipses; break; #if 0 case 's': case 'S': if(sprite) demo = demo_sprites; break; #endif case 'r': case 'R': demo = demo_render; break; } if(demo) { caca_set_color_ansi(cv, CACA_LIGHTGRAY, CACA_BLACK); caca_clear_canvas(cv); } } else if(caca_get_event_type(&ev) & CACA_EVENT_MOUSE_MOTION) { mouse = 1; xmouse = caca_get_event_mouse_x(&ev); ymouse = caca_get_event_mouse_y(&ev); } else if(caca_get_event_type(&ev) & CACA_EVENT_RESIZE) { mouse = 1; /* old hack */ } } if(menu || (mouse && !demo)) { display_menu(); if(mouse && !demo) { caca_set_color_ansi(cv, CACA_RED, CACA_BLACK); caca_put_str(cv, xmouse, ymouse, "."); caca_put_str(cv, xmouse, ymouse + 1, "|\\"); } caca_refresh_display(dp); mouse = menu = 0; } else if(demo) { demo(); caca_set_color_ansi(cv, CACA_LIGHTGRAY, CACA_BLACK); caca_draw_thin_box(cv, 1, 1, caca_get_canvas_width(cv) - 2, caca_get_canvas_height(cv) - 2); caca_printf(cv, 4, 1, "[%i.%i fps]----", 1000000 / caca_get_display_time(dp), (10000000 / caca_get_display_time(dp)) % 10); caca_refresh_display(dp); } } /* Clean up */ #if 0 caca_free_sprite(sprite); #endif caca_free_display(dp); caca_free_canvas(cv); return 0; } static void display_menu(void) { int xo = caca_get_canvas_width(cv) - 2; int yo = caca_get_canvas_height(cv) - 2; caca_set_color_ansi(cv, CACA_LIGHTGRAY, CACA_BLACK); caca_clear_canvas(cv); caca_draw_thin_box(cv, 1, 1, xo, yo); caca_put_str(cv, (xo - strlen("libcaca demo")) / 2, 3, "libcaca demo"); caca_put_str(cv, (xo - strlen("==============")) / 2, 4, "=============="); caca_put_str(cv, 4, 6, "demos:"); caca_put_str(cv, 4, 7, "'f': full"); caca_put_str(cv, 4, 8, "'1': dots"); caca_put_str(cv, 4, 9, "'2': lines"); caca_put_str(cv, 4, 10, "'3': boxes"); caca_put_str(cv, 4, 11, "'4': triangles"); caca_put_str(cv, 4, 12, "'5': ellipses"); caca_put_str(cv, 4, 13, "'c': colour"); caca_put_str(cv, 4, 14, "'r': render"); #if 0 if(sprite) caca_put_str(cv, 4, 15, "'s': sprites"); #endif caca_put_str(cv, 4, 16, "settings:"); caca_printf(cv, 4, 17, "'o': outline: %s", outline == 0 ? "none" : outline == 1 ? "solid" : "thin"); caca_printf(cv, 4, 18, "'b': drawing boundaries: %s", bounds == 0 ? "screen" : "infinite"); //caca_printf(cv, 4, 19, "'d': dithering (%s)", // caca_get_feature_name(dithering)); caca_put_str(cv, 4, yo - 2, "'q': quit"); //caca_refresh_display(dp); } static void demo_all(void) { static int i = 0; int j, xo, yo, xa, ya, xb, yb, xc, yc; i++; caca_set_color_ansi(cv, CACA_LIGHTGRAY, CACA_BLACK); caca_clear_canvas(cv); /* Draw the sun */ caca_set_color_ansi(cv, CACA_YELLOW, CACA_BLACK); xo = caca_get_canvas_width(cv) / 4; yo = caca_get_canvas_height(cv) / 4 + 5 * sin(0.03*i); for(j = 0; j < 16; j++) { xa = xo - (30 + sin(0.03*i) * 8) * sin(0.03*i + M_PI*j/8); ya = yo + (15 + sin(0.03*i) * 4) * cos(0.03*i + M_PI*j/8); caca_draw_thin_line(cv, xo, yo, xa, ya); } j = 15 + sin(0.03*i) * 8; caca_set_color_ansi(cv, CACA_WHITE, CACA_BLACK); caca_fill_ellipse(cv, xo, yo, j, j / 2, '#'); caca_set_color_ansi(cv, CACA_YELLOW, CACA_BLACK); caca_draw_ellipse(cv, xo, yo, j, j / 2, '#'); /* Draw the pyramid */ xo = caca_get_canvas_width(cv) * 5 / 8; yo = 2; xa = caca_get_canvas_width(cv) / 8 + sin(0.03*i) * 5; ya = caca_get_canvas_height(cv) / 2 + cos(0.03*i) * 5; xb = caca_get_canvas_width(cv) - 10 - cos(0.02*i) * 10; yb = caca_get_canvas_height(cv) * 3 / 4 - 5 + sin(0.02*i) * 5; xc = caca_get_canvas_width(cv) / 4 - sin(0.02*i) * 5; yc = caca_get_canvas_height(cv) * 3 / 4 + cos(0.02*i) * 5; caca_set_color_ansi(cv, CACA_GREEN, CACA_BLACK); caca_fill_triangle(cv, xo, yo, xb, yb, xa, ya, '%'); caca_set_color_ansi(cv, CACA_YELLOW, CACA_BLACK); caca_draw_thin_triangle(cv, xo, yo, xb, yb, xa, ya); caca_set_color_ansi(cv, CACA_RED, CACA_BLACK); caca_fill_triangle(cv, xa, ya, xb, yb, xc, yc, '#'); caca_set_color_ansi(cv, CACA_YELLOW, CACA_BLACK); caca_draw_thin_triangle(cv, xa, ya, xb, yb, xc, yc); caca_set_color_ansi(cv, CACA_BLUE, CACA_BLACK); caca_fill_triangle(cv, xo, yo, xb, yb, xc, yc, '%'); caca_set_color_ansi(cv, CACA_YELLOW, CACA_BLACK); caca_draw_thin_triangle(cv, xo, yo, xb, yb, xc, yc); /* Draw a background triangle */ xa = 2; ya = 2; xb = caca_get_canvas_width(cv) - 3; yb = caca_get_canvas_height(cv) / 2; xc = caca_get_canvas_width(cv) / 3; yc = caca_get_canvas_height(cv) - 3; caca_set_color_ansi(cv, CACA_CYAN, CACA_BLACK); caca_draw_thin_triangle(cv, xa, ya, xb, yb, xc, yc); xo = caca_get_canvas_width(cv) / 2 + cos(0.027*i) * caca_get_canvas_width(cv) / 3; yo = caca_get_canvas_height(cv) / 2 - sin(0.027*i) * caca_get_canvas_height(cv) / 2; caca_draw_thin_line(cv, xa, ya, xo, yo); caca_draw_thin_line(cv, xb, yb, xo, yo); caca_draw_thin_line(cv, xc, yc, xo, yo); /* Draw a sprite on the pyramid */ #if 0 caca_draw_sprite(cv, xo, yo, sprite, 0); #endif /* Draw a trail behind the foreground sprite */ for(j = i - 60; j < i; j++) { int delta = caca_rand(-5, 6); caca_set_color_ansi(cv, caca_rand(0, 16), caca_rand(0, 16)); caca_put_char(cv, caca_get_canvas_width(cv) / 2 + cos(0.02*j) * (delta + caca_get_canvas_width(cv) / 4), caca_get_canvas_height(cv) / 2 + sin(0.02*j) * (delta + caca_get_canvas_height(cv) / 3), '#'); } /* Draw foreground sprite */ #if 0 caca_draw_sprite(cv, caca_get_canvas_width(cv) / 2 + cos(0.02*i) * caca_get_canvas_width(cv) / 4, caca_get_canvas_height(cv) / 2 + sin(0.02*i) * caca_get_canvas_height(cv) / 3, sprite, 0); #endif } static void demo_dots(void) { int xmax = caca_get_canvas_width(cv); int ymax = caca_get_canvas_height(cv); int i; static char chars[10] = { '+', '-', '*', '#', 'X', '@', '%', '$', 'M', 'W' }; for(i = 1000; i--;) { /* Putpixel */ caca_set_color_ansi(cv, caca_rand(0, 16), caca_rand(0, 16)); caca_put_char(cv, caca_rand(0, xmax), caca_rand(0, ymax), chars[caca_rand(0, 9)]); } } static void demo_lines(void) { int w = caca_get_canvas_width(cv); int h = caca_get_canvas_height(cv); int xa, ya, xb, yb; if(bounds) { xa = caca_rand(- w, 2 * w); ya = caca_rand(- h, 2 * h); xb = caca_rand(- w, 2 * w); yb = caca_rand(- h, 2 * h); } else { xa = caca_rand(0, w); ya = caca_rand(0, h); xb = caca_rand(0, w); yb = caca_rand(0, h); } caca_set_color_ansi(cv, caca_rand(0, 16), CACA_BLACK); if(outline > 1) caca_draw_thin_line(cv, xa, ya, xb, yb); else caca_draw_line(cv, xa, ya, xb, yb, '#'); } static void demo_boxes(void) { int w = caca_get_canvas_width(cv); int h = caca_get_canvas_height(cv); int xa, ya, xb, yb; if(bounds) { xa = caca_rand(- w, 2 * w); ya = caca_rand(- h, 2 * h); xb = caca_rand(- w, 2 * w); yb = caca_rand(- h, 2 * h); } else { xa = caca_rand(0, w); ya = caca_rand(0, h); xb = caca_rand(0, w); yb = caca_rand(0, h); } caca_set_color_ansi(cv, caca_rand(0, 16), caca_rand(0, 16)); caca_fill_box(cv, xa, ya, xb, yb, '#'); caca_set_color_ansi(cv, caca_rand(0, 16), CACA_BLACK); if(outline == 2) caca_draw_thin_box(cv, xa, ya, xb, yb); else if(outline == 1) caca_draw_box(cv, xa, ya, xb, yb, '#'); } static void demo_ellipses(void) { int w = caca_get_canvas_width(cv); int h = caca_get_canvas_height(cv); int x, y, a, b; if(bounds) { x = caca_rand(- w, 2 * w); y = caca_rand(- h, 2 * h); a = caca_rand(0, w); b = caca_rand(0, h); } else { do { x = caca_rand(0, w); y = caca_rand(0, h); a = caca_rand(0, w); b = caca_rand(0, h); } while(x - a < 0 || x + a >= w || y - b < 0 || y + b >= h); } caca_set_color_ansi(cv, caca_rand(0, 16), caca_rand(0, 16)); caca_fill_ellipse(cv, x, y, a, b, '#'); caca_set_color_ansi(cv, caca_rand(0, 16), CACA_BLACK); if(outline == 2) caca_draw_thin_ellipse(cv, x, y, a, b); else if(outline == 1) caca_draw_ellipse(cv, x, y, a, b, '#'); } static void demo_triangles(void) { int w = caca_get_canvas_width(cv); int h = caca_get_canvas_height(cv); int xa, ya, xb, yb, xc, yc; if(bounds) { xa = caca_rand(- w, 2 * w); ya = caca_rand(- h, 2 * h); xb = caca_rand(- w, 2 * w); yb = caca_rand(- h, 2 * h); xc = caca_rand(- w, 2 * w); yc = caca_rand(- h, 2 * h); } else { xa = caca_rand(0, w); ya = caca_rand(0, h); xb = caca_rand(0, w); yb = caca_rand(0, h); xc = caca_rand(0, w); yc = caca_rand(0, h); } caca_set_color_ansi(cv, caca_rand(0, 16), caca_rand(0, 16)); caca_fill_triangle(cv, xa, ya, xb, yb, xc, yc, '#'); caca_set_color_ansi(cv, caca_rand(0, 16), CACA_BLACK); if(outline == 2) caca_draw_thin_triangle(cv, xa, ya, xb, yb, xc, yc); else if(outline == 1) caca_draw_triangle(cv, xa, ya, xb, yb, xc, yc, '#'); } #if 0 static void demo_sprites(void) { caca_draw_sprite(cv, caca_rand(0, caca_get_canvas_width(cv)), caca_rand(0, caca_get_canvas_height(cv)), sprite, 0); } #endif #if 0 static void demo_render(void) { caca_dither_t *dither; //short buffer[256*256]; //short *dest = buffer; int buffer[256*256]; int *dest = buffer; int x, y, z; static int i = 0; i = (i + 1) % 512; z = i < 256 ? i : 511 - i; for(x = 0; x < 256; x++) for(y = 0; y < 256; y++) { //*dest++ = ((x >> 3) << 11) | ((y >> 2) << 5) | ((z >> 3)); *dest++ = (x << 16) | (y << 8) | (z); } caca_set_dither_invert(dither, 1); //dither = caca_create_dither(16, 256, 256, 2 * 256, 0xf800, 0x07e0, 0x001f, 0x0000); dither = caca_create_dither(32, 256, 256, 4 * 256, 0x00ff0000, 0x0000ff00, 0x000000ff, 0xff000000); caca_dither_bitmap(cv, 0, 0, caca_get_canvas_width(cv), caca_get_canvas_height(cv), dither, buffer); caca_free_dither(dither); } #endif static void draw_circle(int *buffer, int xo, int yo, int r, int mask, int val); static void demo_render(void) { caca_dither_t *dither; int buffer[256*256]; int *dest; int x, y, z, xo, yo; static int i = 0; i++; dest = buffer; for(x = 0; x < 256; x++) for(y = 0; y < 256; y++) { *dest++ = 0xff000000; } /* red */ xo = 128 + 48 * sin(0.02 * i); yo = 128 + 48 * cos(0.03 * i); for(z = 0; z < 240; z++) draw_circle(buffer, xo, yo, z, 0x00ff0000, 200 << 16); /* green */ xo = 128 + 48 * sin(2 + 0.06 * i); yo = 128 + 48 * cos(2 + 0.05 * i); for(z = 0; z < 240; z++) draw_circle(buffer, xo, yo, z, 0x0000ff00, 200 << 8); /* blue */ xo = 128 + 48 * sin(1 + 0.04 * i); yo = 128 + 48 * cos(1 + 0.03 * i); for(z = 0; z < 240; z++) draw_circle(buffer, xo, yo, z, 0x000000ff, 200); dither = caca_create_dither(32, 256, 256, 4 * 256, 0x00ff0000, 0x0000ff00, 0x000000ff, 0xff000000); caca_set_dither_gamma(dither, -1.0); caca_dither_bitmap(cv, 0, 0, caca_get_canvas_width(cv), caca_get_canvas_height(cv), dither, (char *)buffer); caca_free_dither(dither); } static void draw_circle(int *buffer, int x, int y, int r, int mask, int val) { int t, dx, dy; #define POINT(X,Y) \ buffer[(X) + 256 * (Y)] = 0xff000000 | (buffer[(X) + 256 * (Y)] & ~mask) | val; for(t = 0, dx = 0, dy = r; dx <= dy; dx++) { POINT(x - dx / 3, y - dy / 3); POINT(x + dx / 3, y - dy / 3); POINT(x - dx / 3, y + dy / 3); POINT(x + dx / 3, y + dy / 3); POINT(x - dy / 3, y - dx / 3); POINT(x + dy / 3, y - dx / 3); POINT(x - dy / 3, y + dx / 3); POINT(x + dy / 3, y + dx / 3); t += t > 0 ? dx - dy-- : dx; } }