481 lines
11 KiB
C
481 lines
11 KiB
C
/* $NetBSD: attributes.c,v 1.30 2018/11/24 01:04:18 uwe Exp $ */
|
|
|
|
/*-
|
|
* Copyright (c) 1999 The NetBSD Foundation, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This code is derived from software contributed to The NetBSD Foundation
|
|
* by Julian Coleman.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include <sys/cdefs.h>
|
|
#ifndef lint
|
|
__RCSID("$NetBSD: attributes.c,v 1.30 2018/11/24 01:04:18 uwe Exp $");
|
|
#endif /* not lint */
|
|
|
|
#include "curses.h"
|
|
#include "curses_private.h"
|
|
|
|
static int __wattr_off(WINDOW *, attr_t);
|
|
static int __wattr_on(WINDOW *, attr_t);
|
|
static void __wcolor_set(WINDOW *, attr_t);
|
|
|
|
|
|
#ifndef _CURSES_USE_MACROS
|
|
/*
|
|
* attr_get --
|
|
* Get wide attributes and color pair from stdscr
|
|
*/
|
|
/* ARGSUSED */
|
|
int
|
|
attr_get(attr_t *attr, short *pair, void *opts)
|
|
{
|
|
return wattr_get(stdscr, attr, pair, opts);
|
|
}
|
|
|
|
/*
|
|
* attr_on --
|
|
* Test and set wide attributes on stdscr
|
|
*/
|
|
/* ARGSUSED */
|
|
int
|
|
attr_on(attr_t attr, void *opts)
|
|
{
|
|
return wattr_on(stdscr, attr, opts);
|
|
}
|
|
|
|
/*
|
|
* attr_off --
|
|
* Test and unset wide attributes on stdscr
|
|
*/
|
|
/* ARGSUSED */
|
|
int
|
|
attr_off(attr_t attr, void *opts)
|
|
{
|
|
return wattr_off(stdscr, attr, opts);
|
|
}
|
|
|
|
/*
|
|
* attr_set --
|
|
* Set wide attributes and color pair on stdscr
|
|
*/
|
|
/* ARGSUSED */
|
|
int
|
|
attr_set(attr_t attr, short pair, void *opts)
|
|
{
|
|
return wattr_set(stdscr, attr, pair, opts);
|
|
}
|
|
|
|
/*
|
|
* color_set --
|
|
* Set color pair on stdscr
|
|
*/
|
|
/* ARGSUSED */
|
|
int
|
|
color_set(short pair, void *opts)
|
|
{
|
|
return wcolor_set(stdscr, pair, opts);
|
|
}
|
|
|
|
/*
|
|
* attron --
|
|
* Test and set attributes on stdscr
|
|
*/
|
|
int
|
|
attron(int attr)
|
|
{
|
|
return wattron(stdscr, attr);
|
|
}
|
|
|
|
/*
|
|
* attroff --
|
|
* Test and unset attributes on stdscr.
|
|
*/
|
|
int
|
|
attroff(int attr)
|
|
{
|
|
return wattroff(stdscr, attr);
|
|
}
|
|
|
|
/*
|
|
* attrset --
|
|
* Set specific attribute modes.
|
|
* Unset others. On stdscr.
|
|
*/
|
|
int
|
|
attrset(int attr)
|
|
{
|
|
return wattrset(stdscr, attr);
|
|
}
|
|
#endif /* _CURSES_USE_MACROS */
|
|
|
|
/*
|
|
* wattr_get --
|
|
* Get wide attributes and colour pair from window
|
|
* Note that attributes also includes colour.
|
|
*/
|
|
/* ARGSUSED */
|
|
int
|
|
wattr_get(WINDOW *win, attr_t *attr, short *pair, void *opts)
|
|
{
|
|
#ifdef DEBUG
|
|
__CTRACE(__CTRACE_ATTR, "wattr_get: win %p\n", win);
|
|
#endif
|
|
if (attr != NULL) {
|
|
*attr = win->wattr;
|
|
#ifdef HAVE_WCHAR
|
|
*attr &= WA_ATTRIBUTES;
|
|
#endif
|
|
}
|
|
|
|
if (pair != NULL)
|
|
*pair = PAIR_NUMBER(win->wattr);
|
|
return OK;
|
|
}
|
|
|
|
/*
|
|
* wattr_on --
|
|
* Test and set wide attributes on window
|
|
*/
|
|
int
|
|
wattr_on(WINDOW *win, attr_t attr, void *opts)
|
|
{
|
|
if (__predict_false(opts != NULL))
|
|
return ERR;
|
|
|
|
return __wattr_on(win, attr);
|
|
}
|
|
|
|
/*
|
|
* wattr_off --
|
|
* Test and unset wide attributes on window
|
|
*
|
|
* Note that the 'me' sequence unsets all attributes. We handle
|
|
* which attributes should really be set in refresh.c:makech().
|
|
*/
|
|
int
|
|
wattr_off(WINDOW *win, attr_t attr, void *opts)
|
|
{
|
|
if (__predict_false(opts != NULL))
|
|
return ERR;
|
|
|
|
return __wattr_off(win, attr);
|
|
}
|
|
|
|
|
|
/*
|
|
* wattr_set --
|
|
* Set wide attributes and color pair on window
|
|
*/
|
|
int
|
|
wattr_set(WINDOW *win, attr_t attr, short pair, void *opts)
|
|
{
|
|
#ifdef DEBUG
|
|
__CTRACE(__CTRACE_ATTR, "wattr_set: win %p, attr %08x, pair %d\n",
|
|
win, attr, pair);
|
|
#endif
|
|
if (__predict_false(opts != NULL))
|
|
return ERR;
|
|
|
|
/*
|
|
* This overwrites any colour setting from the attributes
|
|
* and is compatible with ncurses.
|
|
*/
|
|
attr = (attr & ~__COLOR) | COLOR_PAIR(pair);
|
|
|
|
__wattr_off(win, WA_ATTRIBUTES);
|
|
__wattr_on(win, attr);
|
|
return OK;
|
|
}
|
|
|
|
/*
|
|
* wcolor_set --
|
|
* Set color pair on window
|
|
*/
|
|
/* ARGSUSED */
|
|
int
|
|
wcolor_set(WINDOW *win, short pair, void *opts)
|
|
{
|
|
#ifdef DEBUG
|
|
__CTRACE(__CTRACE_COLOR, "wolor_set: win %p, pair %d\n", win, pair);
|
|
#endif
|
|
__wcolor_set(win, (attr_t) COLOR_PAIR(pair));
|
|
return OK;
|
|
}
|
|
|
|
/*
|
|
* getattrs --
|
|
* Get window attributes.
|
|
*/
|
|
chtype
|
|
getattrs(WINDOW *win)
|
|
{
|
|
#ifdef DEBUG
|
|
__CTRACE(__CTRACE_ATTR, "getattrs: win %p\n", win);
|
|
#endif
|
|
return((chtype) win->wattr);
|
|
}
|
|
|
|
/*
|
|
* wattron --
|
|
* Test and set attributes.
|
|
*/
|
|
int
|
|
wattron(WINDOW *win, int attr)
|
|
{
|
|
#ifdef DEBUG
|
|
__CTRACE(__CTRACE_ATTR, "wattron: win %p, attr %08x\n", win, attr);
|
|
#endif
|
|
return __wattr_on(win, (attr_t) attr);
|
|
}
|
|
|
|
/*
|
|
* wattroff --
|
|
* Test and unset attributes.
|
|
*/
|
|
int
|
|
wattroff(WINDOW *win, int attr)
|
|
{
|
|
#ifdef DEBUG
|
|
__CTRACE(__CTRACE_ATTR, "wattroff: win %p, attr %08x\n", win, attr);
|
|
#endif
|
|
return __wattr_off(win, (attr_t) attr);
|
|
}
|
|
|
|
/*
|
|
* wattrset --
|
|
* Set specific attribute modes.
|
|
* Unset others.
|
|
*/
|
|
int
|
|
wattrset(WINDOW *win, int attr)
|
|
{
|
|
#ifdef DEBUG
|
|
__CTRACE(__CTRACE_ATTR, "wattrset: win %p, attr %08x\n", win, attr);
|
|
#endif
|
|
__wattr_off(win, __ATTRIBUTES);
|
|
__wattr_on(win, (attr_t) attr);
|
|
return OK;
|
|
}
|
|
|
|
/*
|
|
* termattrs --
|
|
* Get terminal attributes
|
|
*/
|
|
chtype
|
|
termattrs(void)
|
|
{
|
|
chtype ch = 0;
|
|
|
|
#ifdef DEBUG
|
|
__CTRACE(__CTRACE_ATTR, "termattrs\n");
|
|
#endif
|
|
if (exit_attribute_mode != NULL) {
|
|
#ifdef DEBUG
|
|
__CTRACE(__CTRACE_ATTR, "termattrs: have exit attribute mode\n");
|
|
#endif
|
|
if (enter_blink_mode != NULL)
|
|
ch |= __BLINK;
|
|
if (enter_bold_mode != NULL)
|
|
ch |= __BOLD;
|
|
if (enter_dim_mode != NULL)
|
|
ch |= __DIM;
|
|
if (enter_secure_mode != NULL)
|
|
ch |= __BLANK;
|
|
if (enter_protected_mode != NULL)
|
|
ch |= __PROTECT;
|
|
if (enter_reverse_mode != NULL)
|
|
ch |= __REVERSE;
|
|
}
|
|
if (enter_standout_mode != NULL && exit_standout_mode != NULL)
|
|
ch |= __STANDOUT;
|
|
if (enter_underline_mode != NULL && exit_underline_mode != NULL)
|
|
ch |= __UNDERSCORE;
|
|
if (enter_alt_charset_mode != NULL && exit_alt_charset_mode != NULL)
|
|
ch |= __ALTCHARSET;
|
|
|
|
return ch;
|
|
}
|
|
|
|
/*
|
|
* term_attrs --
|
|
* Get terminal wide attributes
|
|
*/
|
|
attr_t
|
|
term_attrs(void)
|
|
{
|
|
attr_t attr = 0;
|
|
|
|
#ifdef DEBUG
|
|
__CTRACE(__CTRACE_ATTR, "term_attrs\n");
|
|
#endif
|
|
if (exit_attribute_mode != NULL) {
|
|
if (enter_blink_mode != NULL)
|
|
attr |= __BLINK;
|
|
if (enter_bold_mode != NULL)
|
|
attr |= __BOLD;
|
|
if (enter_dim_mode != NULL)
|
|
attr |= __DIM;
|
|
if (enter_secure_mode != NULL)
|
|
attr |= __BLANK;
|
|
if (enter_protected_mode != NULL)
|
|
attr |= __PROTECT;
|
|
if (enter_reverse_mode != NULL)
|
|
attr |= __REVERSE;
|
|
#ifdef HAVE_WCHAR
|
|
if (enter_low_hl_mode != NULL)
|
|
attr |= WA_LOW;
|
|
if (enter_top_hl_mode != NULL)
|
|
attr |= WA_TOP;
|
|
if (enter_left_hl_mode != NULL)
|
|
attr |= WA_LEFT;
|
|
if (enter_right_hl_mode != NULL)
|
|
attr |= WA_RIGHT;
|
|
if (enter_horizontal_hl_mode != NULL)
|
|
attr |= WA_HORIZONTAL;
|
|
if (enter_vertical_hl_mode != NULL)
|
|
attr |= WA_VERTICAL;
|
|
#endif /* HAVE_WCHAR */
|
|
}
|
|
if (enter_standout_mode != NULL && exit_standout_mode != NULL)
|
|
attr |= __STANDOUT;
|
|
if (enter_underline_mode != NULL && exit_underline_mode != NULL)
|
|
attr |= __UNDERSCORE;
|
|
if (enter_alt_charset_mode != NULL && exit_alt_charset_mode != NULL)
|
|
attr |= __ALTCHARSET;
|
|
|
|
return attr;
|
|
}
|
|
|
|
|
|
static int
|
|
__wattr_on(WINDOW *win, attr_t attr)
|
|
{
|
|
const TERMINAL *t = win->screen->term;
|
|
|
|
#ifdef DEBUG
|
|
__CTRACE(__CTRACE_ATTR, "wattr_on: win %p, attr %08x\n", win, attr);
|
|
#endif
|
|
/* If can enter modes, set the relevent attribute bits. */
|
|
if (t_exit_attribute_mode(t) != NULL) {
|
|
if (attr & __BLINK && t_enter_blink_mode(t) != NULL)
|
|
win->wattr |= __BLINK;
|
|
if (attr & __BOLD && t_enter_bold_mode(t) != NULL)
|
|
win->wattr |= __BOLD;
|
|
if (attr & __DIM && t_enter_dim_mode(t) != NULL)
|
|
win->wattr |= __DIM;
|
|
if (attr & __BLANK && t_enter_secure_mode(t) != NULL)
|
|
win->wattr |= __BLANK;
|
|
if (attr & __PROTECT && t_enter_protected_mode(t) != NULL)
|
|
win->wattr |= __PROTECT;
|
|
if (attr & __REVERSE && t_enter_reverse_mode(t) != NULL)
|
|
win->wattr |= __REVERSE;
|
|
#ifdef HAVE_WCHAR
|
|
if (attr & WA_LOW && t_enter_low_hl_mode(t) != NULL)
|
|
win->wattr |= WA_LOW;
|
|
if (attr & WA_TOP && t_enter_top_hl_mode(t) != NULL)
|
|
win->wattr |= WA_TOP;
|
|
if (attr & WA_LEFT && t_enter_left_hl_mode(t) != NULL)
|
|
win->wattr |= WA_LEFT;
|
|
if (attr & WA_RIGHT && t_enter_right_hl_mode(t) != NULL)
|
|
win->wattr |= WA_RIGHT;
|
|
if (attr & WA_HORIZONTAL && t_enter_horizontal_hl_mode(t) != NULL)
|
|
win->wattr |= WA_HORIZONTAL;
|
|
if (attr & WA_VERTICAL && t_enter_vertical_hl_mode(t) != NULL)
|
|
win->wattr |= WA_VERTICAL;
|
|
#endif /* HAVE_WCHAR */
|
|
}
|
|
if (attr & __STANDOUT && t_enter_standout_mode(t) != NULL &&
|
|
t_exit_standout_mode(t) != NULL)
|
|
wstandout(win);
|
|
if (attr & __UNDERSCORE && t_enter_underline_mode(t) != NULL &&
|
|
t_exit_underline_mode(t) != NULL)
|
|
wunderscore(win);
|
|
if (attr & __COLOR)
|
|
__wcolor_set(win, attr);
|
|
return OK;
|
|
}
|
|
|
|
|
|
static int
|
|
__wattr_off(WINDOW *win, attr_t attr)
|
|
{
|
|
const TERMINAL *t = win->screen->term;
|
|
|
|
#ifdef DEBUG
|
|
__CTRACE(__CTRACE_ATTR, "wattr_off: win %p, attr %08x\n", win, attr);
|
|
#endif
|
|
/* If can do exit modes, unset the relevent attribute bits. */
|
|
if (t_exit_attribute_mode(t) != NULL) {
|
|
if (attr & __BLINK)
|
|
win->wattr &= ~__BLINK;
|
|
if (attr & __BOLD)
|
|
win->wattr &= ~__BOLD;
|
|
if (attr & __DIM)
|
|
win->wattr &= ~__DIM;
|
|
if (attr & __BLANK)
|
|
win->wattr &= ~__BLANK;
|
|
if (attr & __PROTECT)
|
|
win->wattr &= ~__PROTECT;
|
|
if (attr & __REVERSE)
|
|
win->wattr &= ~__REVERSE;
|
|
#ifdef HAVE_WCHAR
|
|
if (attr & WA_LOW)
|
|
win->wattr &= ~WA_LOW;
|
|
if (attr & WA_TOP)
|
|
win->wattr &= ~WA_TOP;
|
|
if (attr & WA_LEFT)
|
|
win->wattr &= ~WA_LEFT;
|
|
if (attr & WA_RIGHT)
|
|
win->wattr &= ~WA_RIGHT;
|
|
if (attr & WA_HORIZONTAL)
|
|
win->wattr &= ~WA_HORIZONTAL;
|
|
if (attr & WA_VERTICAL)
|
|
win->wattr &= ~WA_VERTICAL;
|
|
#endif /* HAVE_WCHAR */
|
|
}
|
|
if (attr & __STANDOUT)
|
|
wstandend(win);
|
|
if (attr & __UNDERSCORE)
|
|
wunderend(win);
|
|
if (attr & __COLOR) {
|
|
if (max_colors != 0)
|
|
win->wattr &= ~__COLOR;
|
|
}
|
|
return OK;
|
|
}
|
|
|
|
|
|
static void
|
|
__wcolor_set(WINDOW *win, attr_t attr)
|
|
{
|
|
const TERMINAL *t = win->screen->term;
|
|
|
|
/* If another color pair is set, turn that off first. */
|
|
win->wattr &= ~__COLOR;
|
|
/* If can do color video, set the color pair bits. */
|
|
if (t_max_colors(t) != 0 && attr & __COLOR)
|
|
win->wattr |= attr & __COLOR;
|
|
}
|