/* Interface to the terminal controlling library. Copyright (C) 2005-2019 Free Software Foundation, Inc. Written by: Roland Illig , 2005. Andrew Borodin , 2009. This file is part of the Midnight Commander. The Midnight Commander is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. The Midnight Commander is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ /** \file tty.c * \brief Source: %interface to the terminal controlling library */ #include #include #include #include #include /* memset() */ #include /* exit() */ #ifdef HAVE_SYS_IOCTL_H #include #endif #include "lib/global.h" #include "lib/strutil.h" #include "tty.h" #include "tty-internal.h" #include "mouse.h" /* use_mouse_p */ #include "win.h" /*** global variables ****************************************************************************/ int mc_tty_frm[MC_TTY_FRM_MAX]; /*** file scope macro definitions ****************************************************************/ /*** file scope type declarations ****************************************************************/ /*** file scope variables ************************************************************************/ static SIG_ATOMIC_VOLATILE_T got_interrupt = 0; /*** file scope functions ************************************************************************/ /* --------------------------------------------------------------------------------------------- */ static void sigintr_handler (int signo) { (void) &signo; got_interrupt = 1; } /* --------------------------------------------------------------------------------------------- */ /*** public functions ****************************************************************************/ /* --------------------------------------------------------------------------------------------- */ /** * Check terminal type. If $TERM is not set or value is empty, mc finishes with EXIT_FAILURE. * * @param force_xterm Set forced the XTerm type * * @return true if @param force_xterm is true or value of $TERM is one of term*, konsole* * rxvt*, Eterm or dtterm */ gboolean tty_check_term (gboolean force_xterm) { const char *termvalue; const char *xdisplay; termvalue = getenv ("TERM"); if (termvalue == NULL || *termvalue == '\0') { fputs (_("The TERM environment variable is unset!\n"), stderr); exit (EXIT_FAILURE); } xdisplay = getenv ("DISPLAY"); if (xdisplay != NULL && *xdisplay == '\0') xdisplay = NULL; return force_xterm || strncmp (termvalue, "xterm", 5) == 0 || strncmp (termvalue, "konsole", 7) == 0 || strncmp (termvalue, "rxvt", 4) == 0 || strcmp (termvalue, "Eterm") == 0 || strcmp (termvalue, "dtterm") == 0 || (strncmp (termvalue, "screen", 6) == 0 && xdisplay != NULL); } /* --------------------------------------------------------------------------------------------- */ extern void tty_start_interrupt_key (void) { struct sigaction act; memset (&act, 0, sizeof (act)); act.sa_handler = sigintr_handler; sigemptyset (&act.sa_mask); #ifdef SA_RESTART act.sa_flags = SA_RESTART; #endif /* SA_RESTART */ sigaction (SIGINT, &act, NULL); } /* --------------------------------------------------------------------------------------------- */ extern void tty_enable_interrupt_key (void) { struct sigaction act; memset (&act, 0, sizeof (act)); act.sa_handler = sigintr_handler; sigemptyset (&act.sa_mask); sigaction (SIGINT, &act, NULL); got_interrupt = 0; } /* --------------------------------------------------------------------------------------------- */ extern void tty_disable_interrupt_key (void) { struct sigaction act; memset (&act, 0, sizeof (act)); act.sa_handler = SIG_IGN; sigemptyset (&act.sa_mask); sigaction (SIGINT, &act, NULL); } /* --------------------------------------------------------------------------------------------- */ extern gboolean tty_got_interrupt (void) { gboolean rv; rv = (got_interrupt != 0); got_interrupt = 0; return rv; } /* --------------------------------------------------------------------------------------------- */ void tty_print_one_hline (gboolean single) { tty_print_alt_char (ACS_HLINE, single); } /* --------------------------------------------------------------------------------------------- */ void tty_print_one_vline (gboolean single) { tty_print_alt_char (ACS_VLINE, single); } /* --------------------------------------------------------------------------------------------- */ void tty_draw_box (int y, int x, int ys, int xs, gboolean single) { int y2, x2; if (ys <= 0 || xs <= 0) return; ys--; xs--; y2 = y + ys; x2 = x + xs; tty_draw_vline (y, x, mc_tty_frm[single ? MC_TTY_FRM_VERT : MC_TTY_FRM_DVERT], ys); tty_draw_vline (y, x2, mc_tty_frm[single ? MC_TTY_FRM_VERT : MC_TTY_FRM_DVERT], ys); tty_draw_hline (y, x, mc_tty_frm[single ? MC_TTY_FRM_HORIZ : MC_TTY_FRM_DHORIZ], xs); tty_draw_hline (y2, x, mc_tty_frm[single ? MC_TTY_FRM_HORIZ : MC_TTY_FRM_DHORIZ], xs); tty_gotoyx (y, x); tty_print_alt_char (ACS_ULCORNER, single); tty_gotoyx (y2, x); tty_print_alt_char (ACS_LLCORNER, single); tty_gotoyx (y, x2); tty_print_alt_char (ACS_URCORNER, single); tty_gotoyx (y2, x2); tty_print_alt_char (ACS_LRCORNER, single); } /* --------------------------------------------------------------------------------------------- */ char * mc_tty_normalize_from_utf8 (const char *str) { GIConv conv; GString *buffer; const char *_system_codepage = str_detect_termencoding (); if (str_isutf8 (_system_codepage)) return g_strdup (str); conv = g_iconv_open (_system_codepage, "UTF-8"); if (conv == INVALID_CONV) return g_strdup (str); buffer = g_string_new (""); if (str_convert (conv, str, buffer) == ESTR_FAILURE) { g_string_free (buffer, TRUE); str_close_conv (conv); return g_strdup (str); } str_close_conv (conv); return g_string_free (buffer, FALSE); } /* --------------------------------------------------------------------------------------------- */ /** Resize given terminal using TIOCSWINSZ, return ioctl() result */ int tty_resize (int fd) { #if defined TIOCSWINSZ struct winsize tty_size; tty_size.ws_row = LINES; tty_size.ws_col = COLS; tty_size.ws_xpixel = tty_size.ws_ypixel = 0; return ioctl (fd, TIOCSWINSZ, &tty_size); #else return 0; #endif } /* --------------------------------------------------------------------------------------------- */ void tty_init_xterm_support (gboolean is_xterm) { const char *termvalue; termvalue = getenv ("TERM"); /* Check mouse and ca capabilities */ /* terminfo/termcap structures have been already initialized, in slang_init() or/and init_curses() */ /* Check terminfo at first, then check termcap */ xmouse_seq = tty_tgetstr ("kmous"); if (xmouse_seq == NULL) xmouse_seq = tty_tgetstr ("Km"); smcup = tty_tgetstr ("smcup"); if (smcup == NULL) smcup = tty_tgetstr ("ti"); rmcup = tty_tgetstr ("rmcup"); if (rmcup == NULL) rmcup = tty_tgetstr ("te"); if (strcmp (termvalue, "cygwin") == 0) { is_xterm = TRUE; use_mouse_p = MOUSE_DISABLED; } if (is_xterm) { /* Default to the standard xterm sequence */ if (xmouse_seq == NULL) xmouse_seq = ESC_STR "[M"; /* Enable mouse unless explicitly disabled by --nomouse */ if (use_mouse_p != MOUSE_DISABLED) { if (mc_global.tty.old_mouse) use_mouse_p = MOUSE_XTERM_NORMAL_TRACKING; else { /* FIXME: this dirty hack to set supported type of tracking the mouse */ const char *color_term = getenv ("COLORTERM"); if (strncmp (termvalue, "rxvt", 4) == 0 || (color_term != NULL && strncmp (color_term, "rxvt", 4) == 0) || strcmp (termvalue, "Eterm") == 0) use_mouse_p = MOUSE_XTERM_NORMAL_TRACKING; else use_mouse_p = MOUSE_XTERM_BUTTON_EVENT_TRACKING; } } } /* No termcap for SGR extended mouse (yet), hardcode it for now */ if (xmouse_seq != NULL) xmouse_extended_seq = ESC_STR "[<"; } /* --------------------------------------------------------------------------------------------- */