mirror of
				https://github.com/tmux/tmux.git
				synced 2025-11-03 17:24:18 +00:00 
			
		
		
		
	lines (ACS or UTF-8), double or heavy (UTF-8), simple (plain ASCII) or number (the pane numbers). Lines that won't work on a non-UTF-8 terminal are translated back into ACS when they are output.
		
			
				
	
	
		
			198 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			198 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* $OpenBSD$ */
 | 
						|
 | 
						|
/*
 | 
						|
 * Copyright (c) 2010 Nicholas Marriott <nicholas.marriott@gmail.com>
 | 
						|
 *
 | 
						|
 * Permission to use, copy, modify, and distribute this software for any
 | 
						|
 * purpose with or without fee is hereby granted, provided that the above
 | 
						|
 * copyright notice and this permission notice appear in all copies.
 | 
						|
 *
 | 
						|
 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
 | 
						|
 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
 | 
						|
 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
 | 
						|
 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 | 
						|
 * WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER
 | 
						|
 * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
 | 
						|
 * OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 | 
						|
 */
 | 
						|
 | 
						|
#include <sys/types.h>
 | 
						|
 | 
						|
#include <stdlib.h>
 | 
						|
#include <string.h>
 | 
						|
 | 
						|
#include "tmux.h"
 | 
						|
 | 
						|
/* Table mapping ACS entries to UTF-8. */
 | 
						|
struct tty_acs_entry {
 | 
						|
	u_char	 	 key;
 | 
						|
	const char	*string;
 | 
						|
};
 | 
						|
static const struct tty_acs_entry tty_acs_table[] = {
 | 
						|
	{ '+', "\342\206\222" },	/* arrow pointing right */
 | 
						|
	{ ',', "\342\206\220" },	/* arrow pointing left */
 | 
						|
	{ '-', "\342\206\221" },	/* arrow pointing up */
 | 
						|
	{ '.', "\342\206\223" },	/* arrow pointing down */
 | 
						|
	{ '0', "\342\226\256" },	/* solid square block */
 | 
						|
	{ '`', "\342\227\206" },	/* diamond */
 | 
						|
	{ 'a', "\342\226\222" },	/* checker board (stipple) */
 | 
						|
	{ 'b', "\342\220\211" },
 | 
						|
	{ 'c', "\342\220\214" },
 | 
						|
	{ 'd', "\342\220\215" },
 | 
						|
	{ 'e', "\342\220\212" },
 | 
						|
	{ 'f', "\302\260" },		/* degree symbol */
 | 
						|
	{ 'g', "\302\261" },		/* plus/minus */
 | 
						|
	{ 'h', "\342\220\244" },
 | 
						|
	{ 'i', "\342\220\213" },
 | 
						|
	{ 'j', "\342\224\230" },	/* lower right corner */
 | 
						|
	{ 'k', "\342\224\220" },	/* upper right corner */
 | 
						|
	{ 'l', "\342\224\214" },	/* upper left corner */
 | 
						|
	{ 'm', "\342\224\224" },	/* lower left corner */
 | 
						|
	{ 'n', "\342\224\274" },	/* large plus or crossover */
 | 
						|
	{ 'o', "\342\216\272" },	/* scan line 1 */
 | 
						|
	{ 'p', "\342\216\273" },	/* scan line 3 */
 | 
						|
	{ 'q', "\342\224\200" },	/* horizontal line */
 | 
						|
	{ 'r', "\342\216\274" },	/* scan line 7 */
 | 
						|
	{ 's', "\342\216\275" },	/* scan line 9 */
 | 
						|
	{ 't', "\342\224\234" },	/* tee pointing right */
 | 
						|
	{ 'u', "\342\224\244" },	/* tee pointing left */
 | 
						|
	{ 'v', "\342\224\264" },	/* tee pointing up */
 | 
						|
	{ 'w', "\342\224\254" },	/* tee pointing down */
 | 
						|
	{ 'x', "\342\224\202" },	/* vertical line */
 | 
						|
	{ 'y', "\342\211\244" },	/* less-than-or-equal-to */
 | 
						|
	{ 'z', "\342\211\245" },	/* greater-than-or-equal-to */
 | 
						|
	{ '{', "\317\200" },   		/* greek pi */
 | 
						|
	{ '|', "\342\211\240" },	/* not-equal */
 | 
						|
	{ '}', "\302\243" },		/* UK pound sign */
 | 
						|
	{ '~', "\302\267" }		/* bullet */
 | 
						|
};
 | 
						|
 | 
						|
/* Table mapping UTF-8 to ACS entries. */
 | 
						|
struct tty_acs_reverse_entry {
 | 
						|
	const char	*string;
 | 
						|
	u_char		 key;
 | 
						|
};
 | 
						|
static const struct tty_acs_reverse_entry tty_acs_reverse2[] = {
 | 
						|
	{ "\302\267", '~' }
 | 
						|
};
 | 
						|
static const struct tty_acs_reverse_entry tty_acs_reverse3[] = {
 | 
						|
	{ "\342\224\200", 'q' },
 | 
						|
	{ "\342\224\201", 'q' },
 | 
						|
	{ "\342\224\202", 'x' },
 | 
						|
	{ "\342\224\203", 'x' },
 | 
						|
	{ "\342\224\214", 'l' },
 | 
						|
	{ "\342\224\217", 'k' },
 | 
						|
	{ "\342\224\220", 'k' },
 | 
						|
	{ "\342\224\223", 'l' },
 | 
						|
	{ "\342\224\224", 'm' },
 | 
						|
	{ "\342\224\227", 'm' },
 | 
						|
	{ "\342\224\230", 'j' },
 | 
						|
	{ "\342\224\233", 'j' },
 | 
						|
	{ "\342\224\234", 't' },
 | 
						|
	{ "\342\224\243", 't' },
 | 
						|
	{ "\342\224\244", 'u' },
 | 
						|
	{ "\342\224\253", 'u' },
 | 
						|
	{ "\342\224\263", 'w' },
 | 
						|
	{ "\342\224\264", 'v' },
 | 
						|
	{ "\342\224\273", 'v' },
 | 
						|
	{ "\342\224\274", 'n' },
 | 
						|
	{ "\342\225\213", 'n' },
 | 
						|
	{ "\342\225\220", 'q' },
 | 
						|
	{ "\342\225\221", 'x' },
 | 
						|
	{ "\342\225\224", 'l' },
 | 
						|
	{ "\342\225\227", 'k' },
 | 
						|
	{ "\342\225\232", 'm' },
 | 
						|
	{ "\342\225\235", 'j' },
 | 
						|
	{ "\342\225\240", 't' },
 | 
						|
	{ "\342\225\243", 'u' },
 | 
						|
	{ "\342\225\246", 'w' },
 | 
						|
	{ "\342\225\251", 'v' },
 | 
						|
	{ "\342\225\254", 'n' },
 | 
						|
};
 | 
						|
 | 
						|
static int
 | 
						|
tty_acs_cmp(const void *key, const void *value)
 | 
						|
{
 | 
						|
	const struct tty_acs_entry	*entry = value;
 | 
						|
	int				 test = *(u_char *)key;
 | 
						|
 | 
						|
	return (test - entry->key);
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
tty_acs_reverse_cmp(const void *key, const void *value)
 | 
						|
{
 | 
						|
	const struct tty_acs_reverse_entry	*entry = value;
 | 
						|
	const char				*test = key;
 | 
						|
 | 
						|
	return (strcmp(test, entry->string));
 | 
						|
}
 | 
						|
 | 
						|
/* Should this terminal use ACS instead of UTF-8 line drawing? */
 | 
						|
int
 | 
						|
tty_acs_needed(struct tty *tty)
 | 
						|
{
 | 
						|
	if (tty == NULL)
 | 
						|
		return (0);
 | 
						|
 | 
						|
	/*
 | 
						|
	 * If the U8 flag is present, it marks whether a terminal supports
 | 
						|
	 * UTF-8 and ACS together.
 | 
						|
	 *
 | 
						|
	 * If it is present and zero, we force ACS - this gives users a way to
 | 
						|
	 * turn off UTF-8 line drawing.
 | 
						|
	 *
 | 
						|
	 * If it is nonzero, we can fall through to the default and use UTF-8
 | 
						|
	 * line drawing on UTF-8 terminals.
 | 
						|
	 */
 | 
						|
	if (tty_term_has(tty->term, TTYC_U8) &&
 | 
						|
	    tty_term_number(tty->term, TTYC_U8) == 0)
 | 
						|
		return (1);
 | 
						|
 | 
						|
	if (tty->client->flags & CLIENT_UTF8)
 | 
						|
		return (0);
 | 
						|
	return (1);
 | 
						|
}
 | 
						|
 | 
						|
/* Retrieve ACS to output as UTF-8. */
 | 
						|
const char *
 | 
						|
tty_acs_get(struct tty *tty, u_char ch)
 | 
						|
{
 | 
						|
	const struct tty_acs_entry	*entry;
 | 
						|
 | 
						|
	/* Use the ACS set instead of UTF-8 if needed. */
 | 
						|
	if (tty_acs_needed(tty)) {
 | 
						|
		if (tty->term->acs[ch][0] == '\0')
 | 
						|
			return (NULL);
 | 
						|
		return (&tty->term->acs[ch][0]);
 | 
						|
	}
 | 
						|
 | 
						|
	/* Otherwise look up the UTF-8 translation. */
 | 
						|
	entry = bsearch(&ch, tty_acs_table, nitems(tty_acs_table),
 | 
						|
	    sizeof tty_acs_table[0], tty_acs_cmp);
 | 
						|
	if (entry == NULL)
 | 
						|
		return (NULL);
 | 
						|
	return (entry->string);
 | 
						|
}
 | 
						|
 | 
						|
/* Reverse UTF-8 into ACS. */
 | 
						|
int
 | 
						|
tty_acs_reverse_get(__unused struct tty *tty, const char *s, size_t slen)
 | 
						|
{
 | 
						|
	const struct tty_acs_reverse_entry	*table, *entry;
 | 
						|
	u_int					 items;
 | 
						|
 | 
						|
	if (slen == 2) {
 | 
						|
		table = tty_acs_reverse2;
 | 
						|
		items = nitems(tty_acs_reverse2);
 | 
						|
	} else if (slen == 3) {
 | 
						|
		table = tty_acs_reverse3;
 | 
						|
		items = nitems(tty_acs_reverse3);
 | 
						|
	} else
 | 
						|
		return (-1);
 | 
						|
	entry = bsearch(s, table, items, sizeof table[0], tty_acs_reverse_cmp);
 | 
						|
	if (entry == NULL)
 | 
						|
		return (-1);
 | 
						|
	return (entry->key);
 | 
						|
}
 |