source: libcaca/trunk/caca/driver/ncurses.c @ 4148

Last change on this file since 4148 was 4146, checked in by Sam Hocevar, 11 years ago

Fix an alignment issue in caca_get_canvas_chars() and caca_get_canvas_attrs().
Fixes #41.

  • Property svn:keywords set to Id
File size: 21.1 KB
Line 
1/*
2 *  libcaca       Colour ASCII-Art library
3 *  Copyright (c) 2002-2009 Sam Hocevar <sam@hocevar.net>
4 *                2007 Ben Wiley Sittler <bsittler@gmail.com>
5 *                All Rights Reserved
6 *
7 *  $Id: ncurses.c 4146 2009-12-18 21:50:37Z sam $
8 *
9 *  This library is free software. It comes without any warranty, to
10 *  the extent permitted by applicable law. You can redistribute it
11 *  and/or modify it under the terms of the Do What The Fuck You Want
12 *  To Public License, Version 2, as published by Sam Hocevar. See
13 *  http://sam.zoy.org/wtfpl/COPYING for more details.
14 */
15
16/*
17 *  This file contains the libcaca Ncurses input and output driver
18 */
19
20#include "config.h"
21
22#if defined USE_NCURSES
23
24#if defined HAVE_NCURSESW_NCURSES_H
25#   include <ncursesw/ncurses.h>
26#elif defined HAVE_NCURSES_NCURSES_H
27#   include <ncurses/ncurses.h>
28#elif defined HAVE_NCURSES_H
29#   include <ncurses.h>
30#else
31#   include <curses.h>
32#endif
33
34#include <stdlib.h>
35#include <string.h>
36
37#if defined HAVE_UNISTD_H
38#   include <unistd.h>
39#endif
40#if defined HAVE_SIGNAL_H
41#   include <signal.h>
42#endif
43#if defined HAVE_SYS_IOCTL_H
44#   include <sys/ioctl.h>
45#endif
46#if defined HAVE_LOCALE_H
47#   include <locale.h>
48#endif
49#if defined HAVE_TERMIOS_H
50#   include <termios.h>
51#endif
52
53#include "caca.h"
54#include "caca_internals.h"
55
56/*
57 * Emulation for missing ACS_* in older curses
58 */
59
60#ifndef ACS_BLOCK
61#define ACS_BLOCK '#'
62#endif
63
64#ifndef ACS_BOARD
65#define ACS_BOARD '#'
66#endif
67
68#ifndef ACS_BTEE
69#define ACS_BTEE '+'
70#endif
71
72#ifndef ACS_BULLET
73#define ACS_BULLET '.'
74#endif
75
76#ifndef ACS_CKBOARD
77#define ACS_CKBOARD ':'
78#endif
79
80#ifndef ACS_DARROW
81#define ACS_DARROW 'v'
82#endif
83
84#ifndef ACS_DEGREE
85#define ACS_DEGREE '\''
86#endif
87
88#ifndef ACS_DIAMOND
89#define ACS_DIAMOND '+'
90#endif
91
92#ifndef ACS_GEQUAL
93#define ACS_GEQUAL '>'
94#endif
95
96#ifndef ACS_HLINE
97#define ACS_HLINE '-'
98#endif
99
100#ifndef ACS_LANTERN
101#define ACS_LANTERN '#'
102#endif
103
104#ifndef ACS_LARROW
105#define ACS_LARROW '<'
106#endif
107
108#ifndef ACS_LEQUAL
109#define ACS_LEQUAL '<'
110#endif
111
112#ifndef ACS_LLCORNER
113#define ACS_LLCORNER '+'
114#endif
115
116#ifndef ACS_LRCORNER
117#define ACS_LRCORNER '+'
118#endif
119
120#ifndef ACS_LTEE
121#define ACS_LTEE '+'
122#endif
123
124#ifndef ACS_NEQUAL
125#define ACS_NEQUAL '!'
126#endif
127
128#ifndef ACS_PI
129#define ACS_PI '*'
130#endif
131
132#ifndef ACS_STERLING
133#define ACS_STERLING 'f'
134#endif
135
136#ifndef ACS_PLMINUS
137#define ACS_PLMINUS '#'
138#endif
139
140#ifndef ACS_PLUS
141#define ACS_PLUS '+'
142#endif
143
144#ifndef ACS_RARROW
145#define ACS_RARROW '>'
146#endif
147
148#ifndef ACS_RTEE
149#define ACS_RTEE '+'
150#endif
151
152#ifndef ACS_S1
153#define ACS_S1 '-'
154#endif
155
156#ifndef ACS_S3
157#define ACS_S3 '-'
158#endif
159
160#ifndef ACS_S7
161#define ACS_S7 '-'
162#endif
163
164#ifndef ACS_S9
165#define ACS_S9 '-'
166#endif
167
168#ifndef ACS_TTEE
169#define ACS_TTEE '+'
170#endif
171
172#ifndef ACS_UARROW
173#define ACS_UARROW '^'
174#endif
175
176#ifndef ACS_ULCORNER
177#define ACS_ULCORNER '+'
178#endif
179
180#ifndef ACS_URCORNER
181#define ACS_URCORNER '+'
182#endif
183
184#ifndef ACS_VLINE
185#define ACS_VLINE '|'
186#endif
187
188/*
189 * Local functions
190 */
191
192#if defined HAVE_SIGNAL
193static RETSIGTYPE sigwinch_handler(int);
194static caca_display_t *sigwinch_d; /* FIXME: we ought to get rid of this */
195#endif
196#if defined HAVE_GETENV && defined HAVE_PUTENV
197static void ncurses_install_terminal(caca_display_t *);
198static void ncurses_uninstall_terminal(caca_display_t *);
199#endif
200static void ncurses_write_utf32(uint32_t);
201
202struct driver_private
203{
204    int attr[16*16];
205    mmask_t oldmask;
206    char *term;
207};
208
209static int ncurses_init_graphics(caca_display_t *dp)
210{
211    static int curses_colors[] =
212    {
213        /* Standard curses colours */
214        COLOR_BLACK,
215        COLOR_BLUE,
216        COLOR_GREEN,
217        COLOR_CYAN,
218        COLOR_RED,
219        COLOR_MAGENTA,
220        COLOR_YELLOW,
221        COLOR_WHITE,
222        /* Extra values for xterm-16color */
223        COLOR_BLACK + 8,
224        COLOR_BLUE + 8,
225        COLOR_GREEN + 8,
226        COLOR_CYAN + 8,
227        COLOR_RED + 8,
228        COLOR_MAGENTA + 8,
229        COLOR_YELLOW + 8,
230        COLOR_WHITE + 8
231    };
232
233    mmask_t newmask;
234    int fg, bg, max;
235
236    dp->drv.p = malloc(sizeof(struct driver_private));
237
238#if defined HAVE_GETENV && defined HAVE_PUTENV
239    ncurses_install_terminal(dp);
240#endif
241
242#if defined HAVE_SIGNAL
243    sigwinch_d = dp;
244    signal(SIGWINCH, sigwinch_handler);
245#endif
246
247#if defined HAVE_LOCALE_H
248    setlocale(LC_ALL, "");
249#endif
250
251    _caca_set_term_title("caca for ncurses");
252
253    initscr();
254    keypad(stdscr, TRUE);
255    nonl();
256    raw();
257    noecho();
258    nodelay(stdscr, TRUE);
259    curs_set(0);
260
261    /* Activate mouse */
262    newmask = REPORT_MOUSE_POSITION | ALL_MOUSE_EVENTS;
263    mousemask(newmask, &dp->drv.p->oldmask);
264    mouseinterval(-1); /* No click emulation */
265
266    /* Set the escape delay to a ridiculously low value */
267    ESCDELAY = 10;
268
269    /* Activate colour */
270    start_color();
271
272    /* If COLORS == 16, it means the terminal supports full bright colours
273     * using setab and setaf (will use \e[90m \e[91m etc. for colours >= 8),
274     * we can build 16*16 colour pairs.
275     * If COLORS == 8, it means the terminal does not know about bright
276     * colours and we need to get them through A_BOLD and A_BLINK (\e[1m
277     * and \e[5m). We can only build 8*8 colour pairs. */
278    max = COLORS >= 16 ? 16 : 8;
279
280    for(bg = 0; bg < max; bg++)
281        for(fg = 0; fg < max; fg++)
282        {
283            /* Use ((max + 7 - fg) % max) instead of fg so that colour 0
284             * is light gray on black. Some terminals don't like this
285             * colour pair to be redefined. */
286            int col = ((max + 7 - fg) % max) + max * bg;
287            init_pair(col, curses_colors[fg], curses_colors[bg]);
288            dp->drv.p->attr[fg + 16 * bg] = COLOR_PAIR(col);
289
290            if(max == 8)
291            {
292                /* Bright fg on simple bg */
293                dp->drv.p->attr[fg + 8 + 16 * bg] = A_BOLD | COLOR_PAIR(col);
294                /* Simple fg on bright bg */
295                dp->drv.p->attr[fg + 16 * (bg + 8)] = A_BLINK
296                                                    | COLOR_PAIR(col);
297                /* Bright fg on bright bg */
298                dp->drv.p->attr[fg + 8 + 16 * (bg + 8)] = A_BLINK | A_BOLD
299                                                        | COLOR_PAIR(col);
300            }
301        }
302
303    caca_add_dirty_rect(dp->cv, 0, 0, dp->cv->width, dp->cv->height);
304    dp->resize.allow = 1;
305    caca_set_canvas_size(dp->cv, COLS, LINES);
306    dp->resize.allow = 0;
307
308    return 0;
309}
310
311static int ncurses_end_graphics(caca_display_t *dp)
312{
313    _caca_set_term_title("");
314    mousemask(dp->drv.p->oldmask, NULL);
315    curs_set(1);
316    noraw();
317    endwin();
318
319#if defined HAVE_GETENV && defined HAVE_PUTENV
320    ncurses_uninstall_terminal(dp);
321#endif
322
323    free(dp->drv.p);
324
325    return 0;
326}
327
328static int ncurses_set_display_title(caca_display_t *dp, char const *title)
329{
330    _caca_set_term_title(title);
331
332    return 0;
333}
334
335static int ncurses_get_display_width(caca_display_t const *dp)
336{
337    /* Fallback to a 6x10 font */
338    return caca_get_canvas_width(dp->cv) * 6;
339}
340
341static int ncurses_get_display_height(caca_display_t const *dp)
342{
343    /* Fallback to a 6x10 font */
344    return caca_get_canvas_height(dp->cv) * 10;
345}
346
347static void ncurses_display(caca_display_t *dp)
348{
349    int x, y, i;
350
351    for(i = 0; i < caca_get_dirty_rect_count(dp->cv); i++)
352    {
353        uint32_t const *cvchars, *cvattrs;
354        int dx, dy, dw, dh;
355
356        caca_get_dirty_rect(dp->cv, i, &dx, &dy, &dw, &dh);
357
358        cvchars = caca_get_canvas_chars(dp->cv) + dx + dy * dp->cv->width;
359        cvattrs = caca_get_canvas_attrs(dp->cv) + dx + dy * dp->cv->width;
360
361        for(y = dy; y < dy + dh; y++)
362        {
363            move(y, dx);
364            for(x = dx; x < dx + dw; x++)
365            {
366                (void)attrset(dp->drv.p->attr[caca_attr_to_ansi(*cvattrs++)]);
367                ncurses_write_utf32(*cvchars++);
368            }
369
370            cvchars += dp->cv->width - dw;
371            cvattrs += dp->cv->width - dw;
372        }
373    }
374
375    x = caca_wherex(dp->cv);
376    y = caca_wherey(dp->cv);
377    move(y, x);
378
379    refresh();
380}
381
382static void ncurses_handle_resize(caca_display_t *dp)
383{
384    struct winsize size;
385
386#if defined HAVE_SYS_IOCTL_H
387    if(ioctl(fileno(stdout), TIOCGWINSZ, &size) == 0)
388    {
389        dp->resize.w = size.ws_col;
390        dp->resize.h = size.ws_row;
391#if defined HAVE_RESIZE_TERM
392        resize_term(dp->resize.h, dp->resize.w);
393#else
394        resizeterm(dp->resize.h, dp->resize.w);
395#endif
396        wrefresh(curscr);
397        return;
398    }
399#endif
400
401    /* Fallback */
402    dp->resize.w = caca_get_canvas_width(dp->cv);
403    dp->resize.h = caca_get_canvas_height(dp->cv);
404}
405
406static int ncurses_get_event(caca_display_t *dp, caca_privevent_t *ev)
407{
408    int intkey;
409
410    intkey = getch();
411    if(intkey == ERR)
412    {
413        ev->type = CACA_EVENT_NONE;
414        return 0;
415    }
416
417    if(intkey < 0x7f)
418    {
419        ev->type = CACA_EVENT_KEY_PRESS;
420        ev->data.key.ch = intkey;
421        ev->data.key.utf32 = intkey;
422        ev->data.key.utf8[0] = intkey;
423        ev->data.key.utf8[1] = '\0';
424        return 1;
425    }
426
427    /* If the key was UTF-8, parse the whole sequence */
428    if(intkey >= 0x80 && intkey < 0x100)
429    {
430        int keys[7]; /* Necessary for ungetch(); */
431        char utf8[7];
432        uint32_t utf32;
433        size_t i, bytes = 0;
434
435        keys[0] = intkey;
436        utf8[0] = intkey;
437
438        for(i = 1; i < 6; i++)
439        {
440            keys[i] = getch();
441            utf8[i] = (unsigned char)keys[i];
442        }
443
444        utf8[i] = '\0';
445        utf32 = caca_utf8_to_utf32(utf8, &bytes);
446
447        while(i > bytes)
448            ungetch(keys[--i]);
449
450        if(bytes)
451        {
452            ev->type = CACA_EVENT_KEY_PRESS;
453            ev->data.key.ch = 0;
454            ev->data.key.utf32 = utf32;
455            strcpy(ev->data.key.utf8, utf8);
456            return 1;
457        }
458    }
459
460    if(intkey == KEY_MOUSE)
461    {
462        MEVENT mevent;
463        getmouse(&mevent);
464
465        switch(mevent.bstate)
466        {
467#define PRESS(x) ev->data.mouse.button = x; \
468                 ev->type = CACA_EVENT_MOUSE_PRESS; _push_event(dp, ev)
469#define RELEASE(x) ev->data.mouse.button = x; \
470                   ev->type = CACA_EVENT_MOUSE_RELEASE; _push_event(dp, ev)
471#define CLICK(x) PRESS(x); RELEASE(x)
472            case BUTTON1_PRESSED: PRESS(1); break;
473            case BUTTON1_RELEASED: RELEASE(1); break;
474            case BUTTON1_CLICKED: CLICK(1); break;
475            case BUTTON1_DOUBLE_CLICKED: CLICK(1); CLICK(1); break;
476            case BUTTON1_TRIPLE_CLICKED: CLICK(1); CLICK(1); CLICK(1); break;
477            case BUTTON1_RESERVED_EVENT: break;
478
479            case BUTTON2_PRESSED: PRESS(2); break;
480            case BUTTON2_RELEASED: RELEASE(2); break;
481            case BUTTON2_CLICKED: CLICK(2); break;
482            case BUTTON2_DOUBLE_CLICKED: CLICK(2); CLICK(2); break;
483            case BUTTON2_TRIPLE_CLICKED: CLICK(2); CLICK(2); CLICK(2); break;
484            case BUTTON2_RESERVED_EVENT: break;
485
486            case BUTTON3_PRESSED: PRESS(3); break;
487            case BUTTON3_RELEASED: RELEASE(3); break;
488            case BUTTON3_CLICKED: CLICK(3); break;
489            case BUTTON3_DOUBLE_CLICKED: CLICK(3); CLICK(3); break;
490            case BUTTON3_TRIPLE_CLICKED: CLICK(3); CLICK(3); CLICK(3); break;
491            case BUTTON3_RESERVED_EVENT: break;
492
493            case BUTTON4_PRESSED: PRESS(4); break;
494            case BUTTON4_RELEASED: RELEASE(4); break;
495            case BUTTON4_CLICKED: CLICK(4); break;
496            case BUTTON4_DOUBLE_CLICKED: CLICK(4); CLICK(4); break;
497            case BUTTON4_TRIPLE_CLICKED: CLICK(4); CLICK(4); CLICK(4); break;
498            case BUTTON4_RESERVED_EVENT: break;
499
500            default:
501                break;
502#undef PRESS
503#undef RELEASE
504#undef CLICK
505        }
506
507        if(dp->mouse.x == mevent.x && dp->mouse.y == mevent.y)
508            return _pop_event(dp, ev);
509
510        dp->mouse.x = mevent.x;
511        dp->mouse.y = mevent.y;
512
513        ev->type = CACA_EVENT_MOUSE_MOTION;
514        ev->data.mouse.x = dp->mouse.x;
515        ev->data.mouse.y = dp->mouse.y;
516        return 1;
517    }
518
519    switch(intkey)
520    {
521        case KEY_UP: ev->data.key.ch = CACA_KEY_UP; break;
522        case KEY_DOWN: ev->data.key.ch = CACA_KEY_DOWN; break;
523        case KEY_LEFT: ev->data.key.ch = CACA_KEY_LEFT; break;
524        case KEY_RIGHT: ev->data.key.ch = CACA_KEY_RIGHT; break;
525
526        case KEY_IC: ev->data.key.ch = CACA_KEY_INSERT; break;
527        case KEY_DC: ev->data.key.ch = CACA_KEY_DELETE; break;
528        case 0x7f:
529        case KEY_BACKSPACE: ev->data.key.ch = CACA_KEY_BACKSPACE; break;
530        case KEY_HOME: ev->data.key.ch = CACA_KEY_HOME; break;
531        case KEY_END: ev->data.key.ch = CACA_KEY_END; break;
532        case KEY_PPAGE: ev->data.key.ch = CACA_KEY_PAGEUP; break;
533        case KEY_NPAGE: ev->data.key.ch = CACA_KEY_PAGEDOWN; break;
534
535        case KEY_F(1): ev->data.key.ch = CACA_KEY_F1; break;
536        case KEY_F(2): ev->data.key.ch = CACA_KEY_F2; break;
537        case KEY_F(3): ev->data.key.ch = CACA_KEY_F3; break;
538        case KEY_F(4): ev->data.key.ch = CACA_KEY_F4; break;
539        case KEY_F(5): ev->data.key.ch = CACA_KEY_F5; break;
540        case KEY_F(6): ev->data.key.ch = CACA_KEY_F6; break;
541        case KEY_F(7): ev->data.key.ch = CACA_KEY_F7; break;
542        case KEY_F(8): ev->data.key.ch = CACA_KEY_F8; break;
543        case KEY_F(9): ev->data.key.ch = CACA_KEY_F9; break;
544        case KEY_F(10): ev->data.key.ch = CACA_KEY_F10; break;
545        case KEY_F(11): ev->data.key.ch = CACA_KEY_F11; break;
546        case KEY_F(12): ev->data.key.ch = CACA_KEY_F12; break;
547
548        default:
549            /* Unknown key */
550            ev->type = CACA_EVENT_NONE; return 0;
551    }
552
553    ev->type = CACA_EVENT_KEY_PRESS;
554    ev->data.key.utf32 = 0;
555    ev->data.key.utf8[0] = '\0';
556    return 1;
557}
558
559static void ncurses_set_cursor(caca_display_t *dp, int flags)
560{
561    curs_set(flags ? 2 : 0);
562}
563
564/*
565 * XXX: following functions are local
566 */
567
568#if defined HAVE_SIGNAL
569static RETSIGTYPE sigwinch_handler(int sig)
570{
571    sigwinch_d->resize.resized = 1;
572
573    signal(SIGWINCH, sigwinch_handler);
574}
575#endif
576
577#if defined HAVE_GETENV && defined HAVE_PUTENV
578static void ncurses_install_terminal(caca_display_t *dp)
579{
580    char *term, *colorterm;
581
582    dp->drv.p->term = NULL;
583
584    term = getenv("TERM");
585    colorterm = getenv("COLORTERM");
586
587    if(!term || strcmp(term, "xterm"))
588        return;
589
590    /* If we are using gnome-terminal, it's really a 16 colour terminal.
591     * Ditto if we are using xfce4-terminal, or Konsole. */
592    if((colorterm && (!strcmp(colorterm, "gnome-terminal")
593                       || !strcmp(colorterm, "Terminal")))
594         || getenv("KONSOLE_DCOP_SESSION"))
595    {
596        SCREEN *screen;
597        screen = newterm("xterm-16color", stdout, stdin);
598        if(screen == NULL)
599            return;
600        endwin();
601        (void)putenv("TERM=xterm-16color");
602        dp->drv.p->term = strdup(term);
603        return;
604    }
605}
606
607static void ncurses_uninstall_terminal(caca_display_t *dp)
608{
609    /* Needs to be persistent because we use putenv() */
610    static char termenv[1024];
611
612    if(!dp->drv.p->term)
613        return;
614
615    snprintf(termenv, 1023, "TERM=%s", dp->drv.p->term);
616    free(dp->drv.p->term);
617    (void)putenv(termenv);
618}
619#endif
620
621static void ncurses_write_utf32(uint32_t ch)
622{
623#if defined HAVE_NCURSESW_NCURSES_H
624    char buf[10];
625    int bytes;
626#endif
627
628    if(ch == CACA_MAGIC_FULLWIDTH)
629        return;
630
631#if defined HAVE_NCURSESW_NCURSES_H
632    bytes = caca_utf32_to_utf8(buf, ch);
633    buf[bytes] = '\0';
634    addstr(buf);
635#else
636    if(ch < 0x80)
637    {
638        addch(ch);
639    }
640    else
641    {
642        chtype cch;
643        chtype cch2;
644
645        cch = '?';
646        cch2 = ' ';
647        if ((ch > 0x0000ff00) && (ch < 0x0000ff5f))
648        {
649            cch = ch - 0x0000ff00 + ' ';
650        }
651        switch (ch)
652        {
653        case 0x000000a0: /* <nbsp> */
654        case 0x00003000: /*   */
655            cch = ' ';
656            break;
657        case 0x000000a3: /* £ */
658            cch = ACS_STERLING;
659            break;
660        case 0x000000b0: /* ° */
661            cch = ACS_DEGREE;
662            break;
663        case 0x000000b1: /* ± */
664            cch = ACS_PLMINUS;
665            break;
666        case 0x000000b7: /* · */
667        case 0x00002219: /* ∙ */
668        case 0x000030fb: /* ・ */
669            cch = ACS_BULLET;
670            break;
671        case 0x000003c0: /* π */
672            cch = ACS_PI;
673            break;
674        case 0x00002018: /* ‘ */
675        case 0x00002019: /* ’ */
676            cch = '\'';
677            break;
678        case 0x0000201c: /* “ */
679        case 0x0000201d: /* ” */
680            cch = '"';
681            break;
682        case 0x00002190: /* ← */
683            cch = ACS_LARROW;
684            break;
685        case 0x00002191: /* ↑ */
686            cch = ACS_UARROW;
687            break;
688        case 0x00002192: /* → */
689            cch = ACS_RARROW;
690            break;
691        case 0x00002193: /* ↓ */
692            cch = ACS_DARROW;
693            break;
694        case 0x00002260: /* ≠ */
695            cch = ACS_NEQUAL;
696            break;
697        case 0x00002261: /* ≡ */
698            cch = '=';
699            break;
700        case 0x00002264: /* ≤ */
701            cch = ACS_LEQUAL;
702            break;
703        case 0x00002265: /* ≥ */
704            cch = ACS_GEQUAL;
705            break;
706        case 0x000023ba: /* ⎺ */
707            cch = ACS_S1;
708            cch2 = cch;
709            break;
710        case 0x000023bb: /* ⎻ */
711            cch = ACS_S3;
712            cch2 = cch;
713            break;
714        case 0x000023bc: /* ⎼ */
715            cch = ACS_S7;
716            cch2 = cch;
717            break;
718        case 0x000023bd: /* ⎽ */
719            cch = ACS_S9;
720            cch2 = cch;
721            break;
722        case 0x00002500: /* ─ */
723        case 0x00002550: /* ═ */
724            cch = ACS_HLINE;
725            cch2 = cch;
726            break;
727        case 0x00002502: /* │ */
728        case 0x00002551: /* ║ */
729            cch = ACS_VLINE;
730            break;
731        case 0x0000250c: /* ┌ */
732        case 0x00002552: /* ╒ */
733        case 0x00002553: /* ╓ */
734        case 0x00002554: /* ╔ */
735            cch = ACS_ULCORNER;
736            cch2 = ACS_HLINE;
737            break;
738        case 0x00002510: /* ┐ */
739        case 0x00002555: /* ╕ */
740        case 0x00002556: /* ╖ */
741        case 0x00002557: /* ╗ */
742            cch = ACS_URCORNER;
743            break;
744        case 0x00002514: /* └ */
745        case 0x00002558: /* ╘ */
746        case 0x00002559: /* ╙ */
747        case 0x0000255a: /* ╚ */
748            cch = ACS_LLCORNER;
749            cch2 = ACS_HLINE;
750            break;
751        case 0x00002518: /* ┘ */
752        case 0x0000255b: /* ╛ */
753        case 0x0000255c: /* ╜ */
754        case 0x0000255d: /* ╝ */
755            cch = ACS_LRCORNER;
756            break;
757        case 0x0000251c: /* ├ */
758        case 0x0000255e: /* ╞ */
759        case 0x0000255f: /* ╟ */
760        case 0x00002560: /* ╠ */
761            cch = ACS_LTEE;
762            cch2 = ACS_HLINE;
763            break;
764        case 0x00002524: /* ┤ */
765        case 0x00002561: /* ╡ */
766        case 0x00002562: /* ╢ */
767        case 0x00002563: /* ╣ */
768            cch = ACS_RTEE;
769            break;
770        case 0x0000252c: /* ┬ */
771        case 0x00002564: /* ╤ */
772        case 0x00002565: /* ╥ */
773        case 0x00002566: /* ╦ */
774            cch = ACS_TTEE;
775            cch2 = ACS_HLINE;
776            break;
777        case 0x00002534: /* ┴ */
778        case 0x00002567: /* ╧ */
779        case 0x00002568: /* ╨ */
780        case 0x00002569: /* ╩ */
781            cch = ACS_BTEE;
782            cch2 = ACS_HLINE;
783            break;
784        case 0x0000253c: /* ┼ */
785        case 0x0000256a: /* ╪ */
786        case 0x0000256b: /* ╫ */
787        case 0x0000256c: /* ╬ */
788            cch = ACS_PLUS;
789            cch2 = ACS_HLINE;
790            break;
791        case 0x00002591: /* ░ */
792            cch = ACS_BOARD;
793            cch2 = cch;
794            break;
795        case 0x00002592: /* ▒ */
796        case 0x00002593: /* ▓ */
797            cch = ACS_CKBOARD;
798            cch2 = cch;
799            break;
800        case 0x00002580: /* ▀ */
801        case 0x00002584: /* ▄ */
802        case 0x00002588: /* █ */
803        case 0x0000258c: /* ▌ */
804        case 0x00002590: /* ▐ */
805        case 0x000025a0: /* ■ */
806        case 0x000025ac: /* ▬ */
807        case 0x000025ae: /* ▮ */
808            cch = ACS_BLOCK;
809            cch2 = cch;
810            break;
811        case 0x000025c6: /* ◆ */
812        case 0x00002666: /* ♦ */
813            cch = ACS_DIAMOND;
814            break;
815        case 0x00002022: /* • */
816        case 0x000025cb: /* ○ */
817        case 0x000025cf: /* ● */
818        case 0x00002603: /* ☃ */
819        case 0x0000263c: /* ☼ */
820            cch = ACS_LANTERN;
821            break;
822        case 0x0000301c: /* 〜 */
823            cch = '~';
824            break;
825        }
826        addch(cch);
827        if(caca_utf32_is_fullwidth(ch))
828        {
829            addch(cch2);
830        }
831    }
832#endif
833}
834
835/*
836 * Driver initialisation
837 */
838
839int ncurses_install(caca_display_t *dp)
840{
841    dp->drv.id = CACA_DRIVER_NCURSES;
842    dp->drv.driver = "ncurses";
843
844    dp->drv.init_graphics = ncurses_init_graphics;
845    dp->drv.end_graphics = ncurses_end_graphics;
846    dp->drv.set_display_title = ncurses_set_display_title;
847    dp->drv.get_display_width = ncurses_get_display_width;
848    dp->drv.get_display_height = ncurses_get_display_height;
849    dp->drv.display = ncurses_display;
850    dp->drv.handle_resize = ncurses_handle_resize;
851    dp->drv.get_event = ncurses_get_event;
852    dp->drv.set_mouse = NULL;
853    dp->drv.set_cursor = ncurses_set_cursor;
854
855    return 0;
856}
857
858#endif /* USE_NCURSES */
859
Note: See TracBrowser for help on using the repository browser.