source: libcaca/trunk/caca/driver_ncurses.c @ 2142

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