1*47712579Snicm /* $OpenBSD: tty-acs.c,v 1.13 2023/08/08 07:19:48 nicm Exp $ */
2a9d6e51dSnicm
3a9d6e51dSnicm /*
498ca8272Snicm * Copyright (c) 2010 Nicholas Marriott <nicholas.marriott@gmail.com>
5a9d6e51dSnicm *
6a9d6e51dSnicm * Permission to use, copy, modify, and distribute this software for any
7a9d6e51dSnicm * purpose with or without fee is hereby granted, provided that the above
8a9d6e51dSnicm * copyright notice and this permission notice appear in all copies.
9a9d6e51dSnicm *
10a9d6e51dSnicm * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11a9d6e51dSnicm * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12a9d6e51dSnicm * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13a9d6e51dSnicm * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14a9d6e51dSnicm * WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER
15a9d6e51dSnicm * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
16a9d6e51dSnicm * OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17a9d6e51dSnicm */
18a9d6e51dSnicm
19a9d6e51dSnicm #include <sys/types.h>
20a9d6e51dSnicm
21a9d6e51dSnicm #include <stdlib.h>
22523d1daeSnicm #include <string.h>
23a9d6e51dSnicm
24a9d6e51dSnicm #include "tmux.h"
25a9d6e51dSnicm
26a9d6e51dSnicm /* Table mapping ACS entries to UTF-8. */
27a9d6e51dSnicm struct tty_acs_entry {
28a9d6e51dSnicm u_char key;
29a9d6e51dSnicm const char *string;
30a9d6e51dSnicm };
31413e5e52Snicm static const struct tty_acs_entry tty_acs_table[] = {
328891b56aSnicm { '+', "\342\206\222" }, /* arrow pointing right */
338891b56aSnicm { ',', "\342\206\220" }, /* arrow pointing left */
348891b56aSnicm { '-', "\342\206\221" }, /* arrow pointing up */
358891b56aSnicm { '.', "\342\206\223" }, /* arrow pointing down */
368891b56aSnicm { '0', "\342\226\256" }, /* solid square block */
378891b56aSnicm { '`', "\342\227\206" }, /* diamond */
388891b56aSnicm { 'a', "\342\226\222" }, /* checker board (stipple) */
392f3c7f2bSnicm { 'b', "\342\220\211" },
402f3c7f2bSnicm { 'c', "\342\220\214" },
412f3c7f2bSnicm { 'd', "\342\220\215" },
422f3c7f2bSnicm { 'e', "\342\220\212" },
438891b56aSnicm { 'f', "\302\260" }, /* degree symbol */
448891b56aSnicm { 'g', "\302\261" }, /* plus/minus */
452f3c7f2bSnicm { 'h', "\342\220\244" },
462f3c7f2bSnicm { 'i', "\342\220\213" },
478891b56aSnicm { 'j', "\342\224\230" }, /* lower right corner */
488891b56aSnicm { 'k', "\342\224\220" }, /* upper right corner */
498891b56aSnicm { 'l', "\342\224\214" }, /* upper left corner */
508891b56aSnicm { 'm', "\342\224\224" }, /* lower left corner */
518891b56aSnicm { 'n', "\342\224\274" }, /* large plus or crossover */
528891b56aSnicm { 'o', "\342\216\272" }, /* scan line 1 */
538891b56aSnicm { 'p', "\342\216\273" }, /* scan line 3 */
548891b56aSnicm { 'q', "\342\224\200" }, /* horizontal line */
558891b56aSnicm { 'r', "\342\216\274" }, /* scan line 7 */
568891b56aSnicm { 's', "\342\216\275" }, /* scan line 9 */
578891b56aSnicm { 't', "\342\224\234" }, /* tee pointing right */
588891b56aSnicm { 'u', "\342\224\244" }, /* tee pointing left */
598891b56aSnicm { 'v', "\342\224\264" }, /* tee pointing up */
608891b56aSnicm { 'w', "\342\224\254" }, /* tee pointing down */
618891b56aSnicm { 'x', "\342\224\202" }, /* vertical line */
628891b56aSnicm { 'y', "\342\211\244" }, /* less-than-or-equal-to */
638891b56aSnicm { 'z', "\342\211\245" }, /* greater-than-or-equal-to */
648891b56aSnicm { '{', "\317\200" }, /* greek pi */
658891b56aSnicm { '|', "\342\211\240" }, /* not-equal */
668891b56aSnicm { '}', "\302\243" }, /* UK pound sign */
678891b56aSnicm { '~', "\302\267" } /* bullet */
68a9d6e51dSnicm };
69a9d6e51dSnicm
70523d1daeSnicm /* Table mapping UTF-8 to ACS entries. */
71523d1daeSnicm struct tty_acs_reverse_entry {
72523d1daeSnicm const char *string;
73523d1daeSnicm u_char key;
74523d1daeSnicm };
75523d1daeSnicm static const struct tty_acs_reverse_entry tty_acs_reverse2[] = {
76523d1daeSnicm { "\302\267", '~' }
77523d1daeSnicm };
78523d1daeSnicm static const struct tty_acs_reverse_entry tty_acs_reverse3[] = {
79523d1daeSnicm { "\342\224\200", 'q' },
80523d1daeSnicm { "\342\224\201", 'q' },
81523d1daeSnicm { "\342\224\202", 'x' },
82523d1daeSnicm { "\342\224\203", 'x' },
83523d1daeSnicm { "\342\224\214", 'l' },
84523d1daeSnicm { "\342\224\217", 'k' },
85523d1daeSnicm { "\342\224\220", 'k' },
86523d1daeSnicm { "\342\224\223", 'l' },
87523d1daeSnicm { "\342\224\224", 'm' },
88523d1daeSnicm { "\342\224\227", 'm' },
89523d1daeSnicm { "\342\224\230", 'j' },
90523d1daeSnicm { "\342\224\233", 'j' },
91523d1daeSnicm { "\342\224\234", 't' },
92523d1daeSnicm { "\342\224\243", 't' },
93523d1daeSnicm { "\342\224\244", 'u' },
94523d1daeSnicm { "\342\224\253", 'u' },
95523d1daeSnicm { "\342\224\263", 'w' },
96523d1daeSnicm { "\342\224\264", 'v' },
97523d1daeSnicm { "\342\224\273", 'v' },
98523d1daeSnicm { "\342\224\274", 'n' },
99523d1daeSnicm { "\342\225\213", 'n' },
100523d1daeSnicm { "\342\225\220", 'q' },
101523d1daeSnicm { "\342\225\221", 'x' },
102523d1daeSnicm { "\342\225\224", 'l' },
103523d1daeSnicm { "\342\225\227", 'k' },
104523d1daeSnicm { "\342\225\232", 'm' },
105523d1daeSnicm { "\342\225\235", 'j' },
106523d1daeSnicm { "\342\225\240", 't' },
107523d1daeSnicm { "\342\225\243", 'u' },
108523d1daeSnicm { "\342\225\246", 'w' },
109523d1daeSnicm { "\342\225\251", 'v' },
110523d1daeSnicm { "\342\225\254", 'n' },
111523d1daeSnicm };
112523d1daeSnicm
113ccb627cdSnicm /* UTF-8 double borders. */
114ccb627cdSnicm static const struct utf8_data tty_acs_double_borders_list[] = {
115ccb627cdSnicm { "", 0, 0, 0 },
116ccb627cdSnicm { "\342\225\221", 0, 3, 1 }, /* U+2551 */
117ccb627cdSnicm { "\342\225\220", 0, 3, 1 }, /* U+2550 */
118ccb627cdSnicm { "\342\225\224", 0, 3, 1 }, /* U+2554 */
119ccb627cdSnicm { "\342\225\227", 0, 3, 1 }, /* U+2557 */
120ccb627cdSnicm { "\342\225\232", 0, 3, 1 }, /* U+255A */
121ccb627cdSnicm { "\342\225\235", 0, 3, 1 }, /* U+255D */
122ccb627cdSnicm { "\342\225\246", 0, 3, 1 }, /* U+2566 */
123ccb627cdSnicm { "\342\225\251", 0, 3, 1 }, /* U+2569 */
124ccb627cdSnicm { "\342\225\240", 0, 3, 1 }, /* U+2560 */
125ccb627cdSnicm { "\342\225\243", 0, 3, 1 }, /* U+2563 */
126ccb627cdSnicm { "\342\225\254", 0, 3, 1 }, /* U+256C */
127ccb627cdSnicm { "\302\267", 0, 2, 1 } /* U+00B7 */
128ccb627cdSnicm };
129ccb627cdSnicm
130ccb627cdSnicm /* UTF-8 heavy borders. */
131ccb627cdSnicm static const struct utf8_data tty_acs_heavy_borders_list[] = {
132ccb627cdSnicm { "", 0, 0, 0 },
133ccb627cdSnicm { "\342\224\203", 0, 3, 1 }, /* U+2503 */
134ccb627cdSnicm { "\342\224\201", 0, 3, 1 }, /* U+2501 */
135ccb627cdSnicm { "\342\224\217", 0, 3, 1 }, /* U+250F */
136ccb627cdSnicm { "\342\224\223", 0, 3, 1 }, /* U+2513 */
137ccb627cdSnicm { "\342\224\227", 0, 3, 1 }, /* U+2517 */
138ccb627cdSnicm { "\342\224\233", 0, 3, 1 }, /* U+251B */
139ccb627cdSnicm { "\342\224\263", 0, 3, 1 }, /* U+2533 */
140ccb627cdSnicm { "\342\224\273", 0, 3, 1 }, /* U+253B */
141ccb627cdSnicm { "\342\224\243", 0, 3, 1 }, /* U+2523 */
142ccb627cdSnicm { "\342\224\253", 0, 3, 1 }, /* U+252B */
143ccb627cdSnicm { "\342\225\213", 0, 3, 1 }, /* U+254B */
144ccb627cdSnicm { "\302\267", 0, 2, 1 } /* U+00B7 */
145ccb627cdSnicm };
146ccb627cdSnicm
147ccb627cdSnicm /* UTF-8 rounded borders. */
148ccb627cdSnicm static const struct utf8_data tty_acs_rounded_borders_list[] = {
149ccb627cdSnicm { "", 0, 0, 0 },
150ccb627cdSnicm { "\342\224\202", 0, 3, 1 }, /* U+2502 */
151ccb627cdSnicm { "\342\224\200", 0, 3, 1 }, /* U+2500 */
152ccb627cdSnicm { "\342\225\255", 0, 3, 1 }, /* U+256D */
153ccb627cdSnicm { "\342\225\256", 0, 3, 1 }, /* U+256E */
154ccb627cdSnicm { "\342\225\260", 0, 3, 1 }, /* U+2570 */
155ccb627cdSnicm { "\342\225\257", 0, 3, 1 }, /* U+256F */
156ccb627cdSnicm { "\342\224\263", 0, 3, 1 }, /* U+2533 */
157ccb627cdSnicm { "\342\224\273", 0, 3, 1 }, /* U+253B */
158*47712579Snicm { "\342\224\234", 0, 3, 1 }, /* U+2524 */
159*47712579Snicm { "\342\224\244", 0, 3, 1 }, /* U+251C */
160ccb627cdSnicm { "\342\225\213", 0, 3, 1 }, /* U+254B */
161ccb627cdSnicm { "\302\267", 0, 2, 1 } /* U+00B7 */
162ccb627cdSnicm };
163ccb627cdSnicm
164ccb627cdSnicm /* Get cell border character for double style. */
165ccb627cdSnicm const struct utf8_data *
tty_acs_double_borders(int cell_type)166ccb627cdSnicm tty_acs_double_borders(int cell_type)
167ccb627cdSnicm {
168ccb627cdSnicm return (&tty_acs_double_borders_list[cell_type]);
169ccb627cdSnicm }
170ccb627cdSnicm
171ccb627cdSnicm /* Get cell border character for heavy style. */
172ccb627cdSnicm const struct utf8_data *
tty_acs_heavy_borders(int cell_type)173ccb627cdSnicm tty_acs_heavy_borders(int cell_type)
174ccb627cdSnicm {
175ccb627cdSnicm return (&tty_acs_heavy_borders_list[cell_type]);
176ccb627cdSnicm }
177ccb627cdSnicm
178ccb627cdSnicm /* Get cell border character for rounded style. */
179ccb627cdSnicm const struct utf8_data *
tty_acs_rounded_borders(int cell_type)180ccb627cdSnicm tty_acs_rounded_borders(int cell_type)
181ccb627cdSnicm {
182ccb627cdSnicm return (&tty_acs_rounded_borders_list[cell_type]);
183ccb627cdSnicm }
184ccb627cdSnicm
185413e5e52Snicm static int
tty_acs_cmp(const void * key,const void * value)186a9d6e51dSnicm tty_acs_cmp(const void *key, const void *value)
187a9d6e51dSnicm {
188a9d6e51dSnicm const struct tty_acs_entry *entry = value;
189523d1daeSnicm int test = *(u_char *)key;
190a9d6e51dSnicm
191523d1daeSnicm return (test - entry->key);
192523d1daeSnicm }
193523d1daeSnicm
194523d1daeSnicm static int
tty_acs_reverse_cmp(const void * key,const void * value)195523d1daeSnicm tty_acs_reverse_cmp(const void *key, const void *value)
196523d1daeSnicm {
197523d1daeSnicm const struct tty_acs_reverse_entry *entry = value;
198523d1daeSnicm const char *test = key;
199523d1daeSnicm
200523d1daeSnicm return (strcmp(test, entry->string));
201a9d6e51dSnicm }
202a9d6e51dSnicm
203e5a58dfcSnicm /* Should this terminal use ACS instead of UTF-8 line drawing? */
204e5a58dfcSnicm int
tty_acs_needed(struct tty * tty)205e5a58dfcSnicm tty_acs_needed(struct tty *tty)
206e5a58dfcSnicm {
207e5a58dfcSnicm if (tty == NULL)
208e5a58dfcSnicm return (0);
209e5a58dfcSnicm
210e5a58dfcSnicm /*
211e5a58dfcSnicm * If the U8 flag is present, it marks whether a terminal supports
212e5a58dfcSnicm * UTF-8 and ACS together.
213e5a58dfcSnicm *
214e5a58dfcSnicm * If it is present and zero, we force ACS - this gives users a way to
215e5a58dfcSnicm * turn off UTF-8 line drawing.
216e5a58dfcSnicm *
217e5a58dfcSnicm * If it is nonzero, we can fall through to the default and use UTF-8
218e5a58dfcSnicm * line drawing on UTF-8 terminals.
219e5a58dfcSnicm */
220e5a58dfcSnicm if (tty_term_has(tty->term, TTYC_U8) &&
221e5a58dfcSnicm tty_term_number(tty->term, TTYC_U8) == 0)
222e5a58dfcSnicm return (1);
223e5a58dfcSnicm
2245a160f88Snicm if (tty->client->flags & CLIENT_UTF8)
225e5a58dfcSnicm return (0);
226e5a58dfcSnicm return (1);
227e5a58dfcSnicm }
228e5a58dfcSnicm
229523d1daeSnicm /* Retrieve ACS to output as UTF-8. */
230a9d6e51dSnicm const char *
tty_acs_get(struct tty * tty,u_char ch)231a9d6e51dSnicm tty_acs_get(struct tty *tty, u_char ch)
232a9d6e51dSnicm {
233523d1daeSnicm const struct tty_acs_entry *entry;
234a9d6e51dSnicm
235e5a58dfcSnicm /* Use the ACS set instead of UTF-8 if needed. */
236e5a58dfcSnicm if (tty_acs_needed(tty)) {
237a9d6e51dSnicm if (tty->term->acs[ch][0] == '\0')
238a9d6e51dSnicm return (NULL);
239a9d6e51dSnicm return (&tty->term->acs[ch][0]);
240a9d6e51dSnicm }
241a9d6e51dSnicm
242a9d6e51dSnicm /* Otherwise look up the UTF-8 translation. */
243e5a58dfcSnicm entry = bsearch(&ch, tty_acs_table, nitems(tty_acs_table),
244e5a58dfcSnicm sizeof tty_acs_table[0], tty_acs_cmp);
245a9d6e51dSnicm if (entry == NULL)
246a9d6e51dSnicm return (NULL);
247a9d6e51dSnicm return (entry->string);
248a9d6e51dSnicm }
249523d1daeSnicm
250523d1daeSnicm /* Reverse UTF-8 into ACS. */
251523d1daeSnicm int
tty_acs_reverse_get(__unused struct tty * tty,const char * s,size_t slen)252523d1daeSnicm tty_acs_reverse_get(__unused struct tty *tty, const char *s, size_t slen)
253523d1daeSnicm {
254523d1daeSnicm const struct tty_acs_reverse_entry *table, *entry;
255523d1daeSnicm u_int items;
256523d1daeSnicm
257523d1daeSnicm if (slen == 2) {
258523d1daeSnicm table = tty_acs_reverse2;
259523d1daeSnicm items = nitems(tty_acs_reverse2);
260523d1daeSnicm } else if (slen == 3) {
261523d1daeSnicm table = tty_acs_reverse3;
262523d1daeSnicm items = nitems(tty_acs_reverse3);
263523d1daeSnicm } else
264523d1daeSnicm return (-1);
265523d1daeSnicm entry = bsearch(s, table, items, sizeof table[0], tty_acs_reverse_cmp);
266523d1daeSnicm if (entry == NULL)
267523d1daeSnicm return (-1);
268523d1daeSnicm return (entry->key);
269523d1daeSnicm }
270