source: libcaca/trunk/test/demo.c @ 1753

Last change on this file since 1753 was 1753, checked in by Jean-Yves Lamoureux, 15 years ago
  • Added error checks on canvas / display creations
  • Property svn:keywords set to Id
File size: 16.4 KB
Line 
1/*
2 *  demo          demo for libcaca
3 *  Copyright (c) 2003 Sam Hocevar <sam@zoy.org>
4 *                All Rights Reserved
5 *
6 *  $Id: demo.c 1753 2007-02-22 15:31:39Z jylam $
7 *
8 *  This program is free software. It comes without any warranty, to
9 *  the extent permitted by applicable law. You can redistribute it
10 *  and/or modify it under the terms of the Do What The Fuck You Want
11 *  To Public License, Version 2, as published by Sam Hocevar. See
12 *  http://sam.zoy.org/wtfpl/COPYING for more details.
13 */
14
15#include "config.h"
16#include "common.h"
17#if !defined(__KERNEL__)
18#   include <math.h>
19#   include <string.h>
20#   include <stdio.h>
21#endif
22#include "caca.h"
23
24static void display_menu(void);
25
26static void demo_all(void);
27
28static void demo_dots(void);
29static void demo_lines(void);
30static void demo_boxes(void);
31static void demo_ellipses(void);
32static void demo_triangles(void);
33/*static void demo_sprites(void);*/
34static void demo_render(void);
35
36int bounds = 0;
37int outline = 0;
38int dithering = 0;
39#if 0
40cucul_sprite_t *sprite = NULL;
41#endif
42
43cucul_canvas_t *cv;
44caca_display_t *dp;
45
46int main(int argc, char **argv)
47{
48    void (*demo)(void) = NULL;
49    int quit = 0;
50
51    cv = cucul_create_canvas(32, 16);
52    if(cv == NULL)
53    {
54        printf("Failed to create canvas\n");
55        return 1;
56    }
57
58    dp = caca_create_display(cv);
59    if(dp == NULL)
60    {
61        printf("Failed to create display\n");
62        return 1;
63    }
64
65    caca_set_display_time(dp, 40000);
66
67    /* Initialize data */
68#if 0
69    sprite = cucul_load_sprite(DATADIR "/caca.txt");
70    if(!sprite)
71        sprite = cucul_load_sprite("caca.txt");
72    if(!sprite)
73        sprite = cucul_load_sprite("examples/caca.txt");
74#endif
75
76    /* Disable cursor */
77    caca_set_mouse(dp, 0);
78
79    /* Main menu */
80    display_menu();
81    caca_refresh_display(dp);
82
83    /* Go ! */
84    while(!quit)
85    {
86        caca_event_t ev;
87        int menu = 0, mouse = 0, xmouse = 0, ymouse = 0;
88
89        while(caca_get_event(dp, CACA_EVENT_ANY, &ev, 0))
90        {
91            if(demo && (ev.type & CACA_EVENT_KEY_PRESS))
92            {
93                menu = 1;
94                demo = NULL;
95            }
96            else if(ev.type & CACA_EVENT_KEY_PRESS)
97            {
98                switch(ev.data.key.ch)
99                {
100                case 'q':
101                case 'Q':
102                    demo = NULL;
103                    quit = 1;
104                    break;
105                case 'o':
106                case 'O':
107                    outline = (outline + 1) % 3;
108                    display_menu();
109                    break;
110                case 'b':
111                case 'B':
112                    bounds = (bounds + 1) % 2;
113                    display_menu();
114                    break;
115#if 0
116                case 'd':
117                case 'D':
118                    dithering = (dithering + 1) % 5;
119                    cucul_set_feature(cv, dithering);
120                    display_menu();
121                    break;
122#endif
123                case 'f':
124                case 'F':
125                    demo = demo_all;
126                    break;
127                case '1':
128                    demo = demo_dots;
129                    break;
130                case '2':
131                    demo = demo_lines;
132                    break;
133                case '3':
134                    demo = demo_boxes;
135                    break;
136                case '4':
137                    demo = demo_triangles;
138                    break;
139                case '5':
140                    demo = demo_ellipses;
141                    break;
142#if 0
143                case 's':
144                case 'S':
145                    if(sprite)
146                        demo = demo_sprites;
147                    break;
148#endif
149                case 'r':
150                case 'R':
151                    demo = demo_render;
152                    break;
153                }
154
155                if(demo)
156                {
157                    cucul_set_color_ansi(cv, CUCUL_LIGHTGRAY, CUCUL_BLACK);
158                    cucul_clear_canvas(cv);
159                }
160            }
161            else if(ev.type & CACA_EVENT_MOUSE_MOTION)
162            {
163                mouse = 1;
164                xmouse = ev.data.mouse.x;
165                ymouse = ev.data.mouse.y;
166            }
167            else if(ev.type & CACA_EVENT_RESIZE)
168            {
169                mouse = 1; /* old hack */
170            }
171        }
172
173        if(menu || (mouse && !demo))
174        {
175            display_menu();
176            if(mouse && !demo)
177            {
178                cucul_set_color_ansi(cv, CUCUL_RED, CUCUL_BLACK);
179                cucul_put_str(cv, xmouse, ymouse,     ".");
180                cucul_put_str(cv, xmouse, ymouse + 1, "|\\");
181            }
182            caca_refresh_display(dp);
183            mouse = menu = 0;
184        }
185
186        if(demo)
187        {
188            demo();
189
190            cucul_set_color_ansi(cv, CUCUL_LIGHTGRAY, CUCUL_BLACK);
191            cucul_draw_thin_box(cv, 1, 1, cucul_get_canvas_width(cv) - 2,
192                                          cucul_get_canvas_height(cv) - 2);
193            cucul_printf(cv, 4, 1, "[%i.%i fps]----",
194                         1000000 / caca_get_display_time(dp),
195                         (10000000 / caca_get_display_time(dp)) % 10);
196            caca_refresh_display(dp);
197        }
198    }
199
200    /* Clean up */
201#if 0
202    cucul_free_sprite(sprite);
203#endif
204    caca_free_display(dp);
205    cucul_free_canvas(cv);
206
207    return 0;
208}
209
210static void display_menu(void)
211{
212    int xo = cucul_get_canvas_width(cv) - 2;
213    int yo = cucul_get_canvas_height(cv) - 2;
214
215    cucul_set_color_ansi(cv, CUCUL_LIGHTGRAY, CUCUL_BLACK);
216    cucul_clear_canvas(cv);
217    cucul_draw_thin_box(cv, 1, 1, xo, yo);
218
219    cucul_put_str(cv, (xo - strlen("libcaca demo")) / 2, 3, "libcaca demo");
220    cucul_put_str(cv, (xo - strlen("==============")) / 2, 4, "==============");
221
222    cucul_put_str(cv, 4, 6, "demos:");
223    cucul_put_str(cv, 4, 7, "'f': full");
224    cucul_put_str(cv, 4, 8, "'1': dots");
225    cucul_put_str(cv, 4, 9, "'2': lines");
226    cucul_put_str(cv, 4, 10, "'3': boxes");
227    cucul_put_str(cv, 4, 11, "'4': triangles");
228    cucul_put_str(cv, 4, 12, "'5': ellipses");
229    cucul_put_str(cv, 4, 13, "'c': colour");
230    cucul_put_str(cv, 4, 14, "'r': render");
231#if 0
232    if(sprite)
233        cucul_put_str(cv, 4, 15, "'s': sprites");
234#endif
235
236    cucul_put_str(cv, 4, 16, "settings:");
237    cucul_printf(cv, 4, 17, "'o': outline: %s",
238                 outline == 0 ? "none" : outline == 1 ? "solid" : "thin");
239    cucul_printf(cv, 4, 18, "'b': drawing boundaries: %s",
240                 bounds == 0 ? "screen" : "infinite");
241    //cucul_printf(cv, 4, 19, "'d': dithering (%s)",
242    //             cucul_get_feature_name(dithering));
243
244    cucul_put_str(cv, 4, yo - 2, "'q': quit");
245
246    caca_refresh_display(dp);
247}
248
249static void demo_all(void)
250{
251    static int i = 0;
252
253    int j, xo, yo, xa, ya, xb, yb, xc, yc;
254
255    i++;
256
257    cucul_set_color_ansi(cv, CUCUL_LIGHTGRAY, CUCUL_BLACK);
258    cucul_clear_canvas(cv);
259
260    /* Draw the sun */
261    cucul_set_color_ansi(cv, CUCUL_YELLOW, CUCUL_BLACK);
262    xo = cucul_get_canvas_width(cv) / 4;
263    yo = cucul_get_canvas_height(cv) / 4 + 5 * sin(0.03*i);
264
265    for(j = 0; j < 16; j++)
266    {
267        xa = xo - (30 + sin(0.03*i) * 8) * sin(0.03*i + M_PI*j/8);
268        ya = yo + (15 + sin(0.03*i) * 4) * cos(0.03*i + M_PI*j/8);
269        cucul_draw_thin_line(cv, xo, yo, xa, ya);
270    }
271
272    j = 15 + sin(0.03*i) * 8;
273    cucul_set_color_ansi(cv, CUCUL_WHITE, CUCUL_BLACK);
274    cucul_fill_ellipse(cv, xo, yo, j, j / 2, '#');
275    cucul_set_color_ansi(cv, CUCUL_YELLOW, CUCUL_BLACK);
276    cucul_draw_ellipse(cv, xo, yo, j, j / 2, '#');
277
278    /* Draw the pyramid */
279    xo = cucul_get_canvas_width(cv) * 5 / 8;
280    yo = 2;
281
282    xa = cucul_get_canvas_width(cv) / 8 + sin(0.03*i) * 5;
283    ya = cucul_get_canvas_height(cv) / 2 + cos(0.03*i) * 5;
284
285    xb = cucul_get_canvas_width(cv) - 10 - cos(0.02*i) * 10;
286    yb = cucul_get_canvas_height(cv) * 3 / 4 - 5 + sin(0.02*i) * 5;
287
288    xc = cucul_get_canvas_width(cv) / 4 - sin(0.02*i) * 5;
289    yc = cucul_get_canvas_height(cv) * 3 / 4 + cos(0.02*i) * 5;
290
291    cucul_set_color_ansi(cv, CUCUL_GREEN, CUCUL_BLACK);
292    cucul_fill_triangle(cv, xo, yo, xb, yb, xa, ya, '%');
293    cucul_set_color_ansi(cv, CUCUL_YELLOW, CUCUL_BLACK);
294    cucul_draw_thin_triangle(cv, xo, yo, xb, yb, xa, ya);
295
296    cucul_set_color_ansi(cv, CUCUL_RED, CUCUL_BLACK);
297    cucul_fill_triangle(cv, xa, ya, xb, yb, xc, yc, '#');
298    cucul_set_color_ansi(cv, CUCUL_YELLOW, CUCUL_BLACK);
299    cucul_draw_thin_triangle(cv, xa, ya, xb, yb, xc, yc);
300
301    cucul_set_color_ansi(cv, CUCUL_BLUE, CUCUL_BLACK);
302    cucul_fill_triangle(cv, xo, yo, xb, yb, xc, yc, '%');
303    cucul_set_color_ansi(cv, CUCUL_YELLOW, CUCUL_BLACK);
304    cucul_draw_thin_triangle(cv, xo, yo, xb, yb, xc, yc);
305
306    /* Draw a background triangle */
307    xa = 2;
308    ya = 2;
309
310    xb = cucul_get_canvas_width(cv) - 3;
311    yb = cucul_get_canvas_height(cv) / 2;
312
313    xc = cucul_get_canvas_width(cv) / 3;
314    yc = cucul_get_canvas_height(cv) - 3;
315
316    cucul_set_color_ansi(cv, CUCUL_CYAN, CUCUL_BLACK);
317    cucul_draw_thin_triangle(cv, xa, ya, xb, yb, xc, yc);
318
319    xo = cucul_get_canvas_width(cv) / 2 + cos(0.027*i) * cucul_get_canvas_width(cv) / 3;
320    yo = cucul_get_canvas_height(cv) / 2 - sin(0.027*i) * cucul_get_canvas_height(cv) / 2;
321
322    cucul_draw_thin_line(cv, xa, ya, xo, yo);
323    cucul_draw_thin_line(cv, xb, yb, xo, yo);
324    cucul_draw_thin_line(cv, xc, yc, xo, yo);
325
326    /* Draw a sprite on the pyramid */
327#if 0
328    cucul_draw_sprite(cv, xo, yo, sprite, 0);
329#endif
330
331    /* Draw a trail behind the foreground sprite */
332    for(j = i - 60; j < i; j++)
333    {
334        int delta = cucul_rand(-5, 6);
335        cucul_set_color_ansi(cv, cucul_rand(0, 16), cucul_rand(0, 16));
336        cucul_put_char(cv, cucul_get_canvas_width(cv) / 2
337                    + cos(0.02*j) * (delta + cucul_get_canvas_width(cv) / 4),
338                   cucul_get_canvas_height(cv) / 2
339                    + sin(0.02*j) * (delta + cucul_get_canvas_height(cv) / 3),
340                   '#');
341    }
342
343    /* Draw foreground sprite */
344#if 0
345    cucul_draw_sprite(cv, cucul_get_canvas_width(cv) / 2 + cos(0.02*i) * cucul_get_canvas_width(cv) / 4,
346                   cucul_get_canvas_height(cv) / 2 + sin(0.02*i) * cucul_get_canvas_height(cv) / 3,
347                   sprite, 0);
348#endif
349}
350
351static void demo_dots(void)
352{
353    int xmax = cucul_get_canvas_width(cv);
354    int ymax = cucul_get_canvas_height(cv);
355    int i;
356    static char chars[10] =
357    {
358        '+', '-', '*', '#', 'X', '@', '%', '$', 'M', 'W'
359    };
360
361    for(i = 1000; i--;)
362    {
363        /* Putpixel */
364        cucul_set_color_ansi(cv, cucul_rand(0, 16), cucul_rand(0, 16));
365        cucul_put_char(cv, cucul_rand(0, xmax), cucul_rand(0, ymax),
366                      chars[cucul_rand(0, 9)]);
367    }
368}
369
370static void demo_lines(void)
371{
372    int w = cucul_get_canvas_width(cv);
373    int h = cucul_get_canvas_height(cv);
374    int xa, ya, xb, yb;
375
376    if(bounds)
377    {
378        xa = cucul_rand(- w, 2 * w); ya = cucul_rand(- h, 2 * h);
379        xb = cucul_rand(- w, 2 * w); yb = cucul_rand(- h, 2 * h);
380    }
381    else
382    {
383        xa = cucul_rand(0, w); ya = cucul_rand(0, h);
384        xb = cucul_rand(0, w); yb = cucul_rand(0, h);
385    }
386
387    cucul_set_color_ansi(cv, cucul_rand(0, 16), CUCUL_BLACK);
388    if(outline > 1)
389        cucul_draw_thin_line(cv, xa, ya, xb, yb);
390    else
391        cucul_draw_line(cv, xa, ya, xb, yb, '#');
392}
393
394static void demo_boxes(void)
395{
396    int w = cucul_get_canvas_width(cv);
397    int h = cucul_get_canvas_height(cv);
398    int xa, ya, xb, yb;
399
400    if(bounds)
401    {
402        xa = cucul_rand(- w, 2 * w); ya = cucul_rand(- h, 2 * h);
403        xb = cucul_rand(- w, 2 * w); yb = cucul_rand(- h, 2 * h);
404    }
405    else
406    {
407        xa = cucul_rand(0, w); ya = cucul_rand(0, h);
408        xb = cucul_rand(0, w); yb = cucul_rand(0, h);
409    }
410
411    cucul_set_color_ansi(cv, cucul_rand(0, 16), cucul_rand(0, 16));
412    cucul_fill_box(cv, xa, ya, xb, yb, '#');
413
414    cucul_set_color_ansi(cv, cucul_rand(0, 16), CUCUL_BLACK);
415    if(outline == 2)
416        cucul_draw_thin_box(cv, xa, ya, xb, yb);
417    else if(outline == 1)
418        cucul_draw_box(cv, xa, ya, xb, yb, '#');
419}
420
421static void demo_ellipses(void)
422{
423    int w = cucul_get_canvas_width(cv);
424    int h = cucul_get_canvas_height(cv);
425    int x, y, a, b;
426
427    if(bounds)
428    {
429        x = cucul_rand(- w, 2 * w); y = cucul_rand(- h, 2 * h);
430        a = cucul_rand(0, w); b = cucul_rand(0, h);
431    }
432    else
433    {
434        do
435        {
436            x = cucul_rand(0, w); y = cucul_rand(0, h);
437            a = cucul_rand(0, w); b = cucul_rand(0, h);
438
439        } while(x - a < 0 || x + a >= w || y - b < 0 || y + b >= h);
440    }
441
442    cucul_set_color_ansi(cv, cucul_rand(0, 16), cucul_rand(0, 16));
443    cucul_fill_ellipse(cv, x, y, a, b, '#');
444
445    cucul_set_color_ansi(cv, cucul_rand(0, 16), CUCUL_BLACK);
446    if(outline == 2)
447        cucul_draw_thin_ellipse(cv, x, y, a, b);
448    else if(outline == 1)
449        cucul_draw_ellipse(cv, x, y, a, b, '#');
450}
451
452static void demo_triangles(void)
453{
454    int w = cucul_get_canvas_width(cv);
455    int h = cucul_get_canvas_height(cv);
456    int xa, ya, xb, yb, xc, yc;
457
458    if(bounds)
459    {
460        xa = cucul_rand(- w, 2 * w); ya = cucul_rand(- h, 2 * h);
461        xb = cucul_rand(- w, 2 * w); yb = cucul_rand(- h, 2 * h);
462        xc = cucul_rand(- w, 2 * w); yc = cucul_rand(- h, 2 * h);
463    }
464    else
465    {
466
467        xa = cucul_rand(0, w); ya = cucul_rand(0, h);
468        xb = cucul_rand(0, w); yb = cucul_rand(0, h);
469        xc = cucul_rand(0, w); yc = cucul_rand(0, h);
470    }
471
472    cucul_set_color_ansi(cv, cucul_rand(0, 16), cucul_rand(0, 16));
473    cucul_fill_triangle(cv, xa, ya, xb, yb, xc, yc, '#');
474
475    cucul_set_color_ansi(cv, cucul_rand(0, 16), CUCUL_BLACK);
476    if(outline == 2)
477        cucul_draw_thin_triangle(cv, xa, ya, xb, yb, xc, yc);
478    else if(outline == 1)
479        cucul_draw_triangle(cv, xa, ya, xb, yb, xc, yc, '#');
480}
481#if 0
482static void demo_sprites(void)
483{
484
485    cucul_draw_sprite(cv, cucul_rand(0, cucul_get_canvas_width(cv)),
486                      cucul_rand(0, cucul_get_canvas_height(cv)), sprite, 0);
487
488}
489#endif
490#if 0
491static void demo_render(void)
492{
493    cucul_dither_t *dither;
494    //short buffer[256*256];
495    //short *dest = buffer;
496    int buffer[256*256];
497    int *dest = buffer;
498    int x, y, z;
499    static int i = 0;
500
501    i = (i + 1) % 512;
502    z = i < 256 ? i : 511 - i;
503
504    for(x = 0; x < 256; x++)
505        for(y = 0; y < 256; y++)
506    {
507        //*dest++ = ((x >> 3) << 11) | ((y >> 2) << 5) | ((z >> 3));
508        *dest++ = (x << 16) | (y << 8) | (z);
509    }
510    cucul_set_dither_invert(dither, 1);
511    //dither = cucul_create_dither(16, 256, 256, 2 * 256, 0xf800, 0x07e0, 0x001f, 0x0000);
512    dither = cucul_create_dither(32, 256, 256, 4 * 256, 0x00ff0000, 0x0000ff00, 0x000000ff, 0xff000000);
513    cucul_dither_bitmap(cv, 0, 0, cucul_get_canvas_width(cv), cucul_get_canvas_height(cv),
514                     dither, buffer);
515    cucul_free_dither(dither);
516}
517#endif
518
519static void draw_circle(int *buffer, int xo, int yo, int r, int mask, int val);
520
521static void demo_render(void)
522{
523    cucul_dither_t *dither;
524    int buffer[256*256];
525    int *dest;
526    int x, y, z, xo, yo;
527    static int i = 0;
528
529    i++;
530
531    dest = buffer;
532    for(x = 0; x < 256; x++)
533        for(y = 0; y < 256; y++)
534    {
535        *dest++ = 0xff000000;
536    }
537
538    /* red */
539    xo = 128 + 48 * sin(0.02 * i);
540    yo = 128 + 48 * cos(0.03 * i);
541    for(z = 0; z < 240; z++)
542        draw_circle(buffer, xo, yo, z, 0x00ff0000, 200 << 16);
543
544    /* green */
545    xo = 128 + 48 * sin(2 + 0.06 * i);
546    yo = 128 + 48 * cos(2 + 0.05 * i);
547    for(z = 0; z < 240; z++)
548        draw_circle(buffer, xo, yo, z, 0x0000ff00, 200 << 8);
549
550    /* blue */
551    xo = 128 + 48 * sin(1 + 0.04 * i);
552    yo = 128 + 48 * cos(1 + 0.03 * i);
553    for(z = 0; z < 240; z++)
554        draw_circle(buffer, xo, yo, z, 0x000000ff, 200);
555
556    dither = cucul_create_dither(32, 256, 256, 4 * 256, 0x00ff0000, 0x0000ff00, 0x000000ff, 0xff000000);
557 cucul_set_dither_invert(dither, 1);
558    cucul_dither_bitmap(cv, 0, 0, cucul_get_canvas_width(cv), cucul_get_canvas_height(cv), dither, (char *)buffer);
559    cucul_free_dither(dither);
560}
561
562static void draw_circle(int *buffer, int x, int y, int r, int mask, int val)
563{
564    int t, dx, dy;
565
566#define POINT(X,Y) \
567    buffer[(X) + 256 * (Y)] = 0xff000000 | (buffer[(X) + 256 * (Y)] & ~mask) | val;
568
569    for(t = 0, dx = 0, dy = r; dx <= dy; dx++)
570    {
571        POINT(x - dx / 3, y - dy / 3);
572        POINT(x + dx / 3, y - dy / 3);
573        POINT(x - dx / 3, y + dy / 3);
574        POINT(x + dx / 3, y + dy / 3);
575
576        POINT(x - dy / 3, y - dx / 3);
577        POINT(x + dy / 3, y - dx / 3);
578        POINT(x - dy / 3, y + dx / 3);
579        POINT(x + dy / 3, y + dx / 3);
580
581        t += t > 0 ? dx - dy-- : dx;
582    }
583}
584
Note: See TracBrowser for help on using the repository browser.