ncurses 5.7 - patch 20090510
[ncurses.git] / ncurses / tinfo / lib_acs.c
1 /****************************************************************************
2  * Copyright (c) 1998-2008,2009 Free Software Foundation, Inc.              *
3  *                                                                          *
4  * Permission is hereby granted, free of charge, to any person obtaining a  *
5  * copy of this software and associated documentation files (the            *
6  * "Software"), to deal in the Software without restriction, including      *
7  * without limitation the rights to use, copy, modify, merge, publish,      *
8  * distribute, distribute with modifications, sublicense, and/or sell       *
9  * copies of the Software, and to permit persons to whom the Software is    *
10  * furnished to do so, subject to the following conditions:                 *
11  *                                                                          *
12  * The above copyright notice and this permission notice shall be included  *
13  * in all copies or substantial portions of the Software.                   *
14  *                                                                          *
15  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS  *
16  * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF               *
17  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.   *
18  * IN NO EVENT SHALL THE ABOVE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,   *
19  * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR    *
20  * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR    *
21  * THE USE OR OTHER DEALINGS IN THE SOFTWARE.                               *
22  *                                                                          *
23  * Except as contained in this notice, the name(s) of the above copyright   *
24  * holders shall not be used in advertising or otherwise to promote the     *
25  * sale, use or other dealings in this Software without prior written       *
26  * authorization.                                                           *
27  ****************************************************************************/
28
29 /****************************************************************************
30  *  Author: Zeyd M. Ben-Halim <zmbenhal@netcom.com> 1992,1995               *
31  *     and: Eric S. Raymond <esr@snark.thyrsus.com>                         *
32  *     and: Thomas E. Dickey                        1996-on                 *
33  ****************************************************************************/
34
35 #include <curses.priv.h>
36 #include <term.h>               /* ena_acs, acs_chars */
37
38 #ifndef CUR
39 #define CUR SP_TERMTYPE 
40 #endif
41
42 MODULE_ID("$Id: lib_acs.c,v 1.39 2009/05/10 00:48:29 tom Exp $")
43
44 #if BROKEN_LINKER || USE_REENTRANT
45 #define MyBuffer _nc_prescreen.real_acs_map
46 NCURSES_EXPORT(chtype *)
47 NCURSES_PUBLIC_VAR(acs_map) (void)
48 {
49     if (MyBuffer == 0)
50         MyBuffer = typeCalloc(chtype, ACS_LEN);
51     return MyBuffer;
52 }
53 #undef MyBuffer
54 #else
55 NCURSES_EXPORT_VAR (chtype) acs_map[ACS_LEN] =
56 {
57     0
58 };
59 #endif
60
61 NCURSES_EXPORT(void)
62 _nc_init_acs(void)
63 {
64 #if NCURSES_SP_FUNCS
65     SCREEN *sp = CURRENT_SCREEN;
66 #endif
67     chtype *fake_map = acs_map;
68     chtype *real_map = SP != 0 ? SP->_acs_map : fake_map;
69     int j;
70
71     T(("initializing ACS map"));
72
73     /*
74      * If we're using this from curses (rather than terminfo), we are storing
75      * the mapping information in the SCREEN struct so we can decide how to
76      * render it.
77      */
78     if (real_map != fake_map) {
79         for (j = 1; j < ACS_LEN; ++j) {
80             real_map[j] = 0;
81             fake_map[j] = A_ALTCHARSET | j;
82             if (SP)
83                 SP->_screen_acs_map[j] = FALSE;
84         }
85     } else {
86         for (j = 1; j < ACS_LEN; ++j) {
87             real_map[j] = 0;
88         }
89     }
90
91     /*
92      * Initializations for a UNIX-like multi-terminal environment.  Use
93      * ASCII chars and count on the terminfo description to do better.
94      */
95     real_map['l'] = '+';        /* should be upper left corner */
96     real_map['m'] = '+';        /* should be lower left corner */
97     real_map['k'] = '+';        /* should be upper right corner */
98     real_map['j'] = '+';        /* should be lower right corner */
99     real_map['u'] = '+';        /* should be tee pointing left */
100     real_map['t'] = '+';        /* should be tee pointing right */
101     real_map['v'] = '+';        /* should be tee pointing up */
102     real_map['w'] = '+';        /* should be tee pointing down */
103     real_map['q'] = '-';        /* should be horizontal line */
104     real_map['x'] = '|';        /* should be vertical line */
105     real_map['n'] = '+';        /* should be large plus or crossover */
106     real_map['o'] = '~';        /* should be scan line 1 */
107     real_map['s'] = '_';        /* should be scan line 9 */
108     real_map['`'] = '+';        /* should be diamond */
109     real_map['a'] = ':';        /* should be checker board (stipple) */
110     real_map['f'] = '\'';       /* should be degree symbol */
111     real_map['g'] = '#';        /* should be plus/minus */
112     real_map['~'] = 'o';        /* should be bullet */
113     real_map[','] = '<';        /* should be arrow pointing left */
114     real_map['+'] = '>';        /* should be arrow pointing right */
115     real_map['.'] = 'v';        /* should be arrow pointing down */
116     real_map['-'] = '^';        /* should be arrow pointing up */
117     real_map['h'] = '#';        /* should be board of squares */
118     real_map['i'] = '#';        /* should be lantern symbol */
119     real_map['0'] = '#';        /* should be solid square block */
120     /* these defaults were invented for ncurses */
121     real_map['p'] = '-';        /* should be scan line 3 */
122     real_map['r'] = '-';        /* should be scan line 7 */
123     real_map['y'] = '<';        /* should be less-than-or-equal-to */
124     real_map['z'] = '>';        /* should be greater-than-or-equal-to */
125     real_map['{'] = '*';        /* should be greek pi */
126     real_map['|'] = '!';        /* should be not-equal */
127     real_map['}'] = 'f';        /* should be pound-sterling symbol */
128
129     if (ena_acs != NULL) {
130         TPUTS_TRACE("ena_acs");
131         putp(ena_acs);
132     }
133 #if NCURSES_EXT_FUNCS
134     /*
135      * Linux console "supports" the "PC ROM" character set by the coincidence
136      * that smpch/rmpch and smacs/rmacs have the same values.  ncurses has
137      * no codepage support (see SCO Merge for an example).  Outside of the
138      * values defined in acsc, there are no definitions for the "PC ROM"
139      * character set (assumed by some applications to be codepage 437), but we
140      * allow those applications to use those codepoints.
141      *
142      * test/blue.c uses this feature.
143      */
144 #define PCH_KLUDGE(a,b) (a != 0 && b != 0 && !strcmp(a,b))
145     if (PCH_KLUDGE(enter_pc_charset_mode, enter_alt_charset_mode) &&
146         PCH_KLUDGE(exit_pc_charset_mode, exit_alt_charset_mode)) {
147         size_t i;
148         for (i = 1; i < ACS_LEN; ++i) {
149             if (real_map[i] == 0) {
150                 real_map[i] = i;
151                 if (real_map != fake_map) {
152                     if (SP != 0)
153                         SP->_screen_acs_map[i] = TRUE;
154                 }
155             }
156         }
157     }
158 #endif
159
160     if (acs_chars != NULL) {
161         size_t i = 0;
162         size_t length = strlen(acs_chars);
163
164         while (i + 1 < length) {
165             if (acs_chars[i] != 0 && UChar(acs_chars[i]) < ACS_LEN) {
166                 real_map[UChar(acs_chars[i])] = UChar(acs_chars[i + 1]) | A_ALTCHARSET;
167                 if (SP != 0)
168                     SP->_screen_acs_map[UChar(acs_chars[i])] = TRUE;
169             }
170             i += 2;
171         }
172     }
173 #ifdef TRACE
174     /* Show the equivalent mapping, noting if it does not match the
175      * given attribute, whether by re-ordering or duplication.
176      */
177     if (USE_TRACEF(TRACE_CALLS)) {
178         size_t n, m;
179         char show[ACS_LEN * 2 + 1];
180         for (n = 1, m = 0; n < ACS_LEN; n++) {
181             if (real_map[n] != 0) {
182                 show[m++] = (char) n;
183                 show[m++] = (char) ChCharOf(real_map[n]);
184             }
185         }
186         show[m] = 0;
187         if (acs_chars == NULL || strcmp(acs_chars, show))
188             _tracef("%s acs_chars %s",
189                     (acs_chars == NULL) ? "NULL" : "READ",
190                     _nc_visbuf(acs_chars));
191         _tracef("%s acs_chars %s",
192                 (acs_chars == NULL)
193                 ? "NULL"
194                 : (strcmp(acs_chars, show)
195                    ? "DIFF"
196                    : "SAME"),
197                 _nc_visbuf(show));
198         _nc_unlock_global(tracef);
199     }
200 #endif /* TRACE */
201 }