#include #include "nvim/globals.h" #include "nvim/memory.h" #include "nvim/os/fs.h" #include "nvim/os/input.h" #include "nvim/os/os.h" #include "nvim/os/os_win_console.h" #include "os/os_win_console.c.generated.h" static char origTitle[256] = { 0 }; static HWND hWnd = NULL; static HICON hOrigIconSmall = NULL; static HICON hOrigIcon = NULL; /// Re-enable normal Ctrl-C processing after detached startup. /// /// On Windows, UV_PROCESS_DETACHED implies CREATE_NEW_PROCESS_GROUP, which /// disables Ctrl-C handling for the new process. Restore the default behavior /// once the embedded server has a console so terminal jobs inherit it. void os_enable_ctrl_c(void) { SetConsoleCtrlHandler(NULL, false); } int os_open_conin_fd(void) { const HANDLE conin_handle = CreateFile("CONIN$", GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, (LPSECURITY_ATTRIBUTES)NULL, OPEN_EXISTING, 0, (HANDLE)NULL); assert(conin_handle != INVALID_HANDLE_VALUE); int conin_fd = _open_osfhandle((intptr_t)conin_handle, _O_RDONLY); assert(conin_fd != -1); return conin_fd; } void os_clear_hwnd(void) { hWnd = NULL; } void os_replace_stdin_to_conin(void) { close(STDIN_FILENO); const int conin_fd = os_open_conin_fd(); assert(conin_fd == STDIN_FILENO); } void os_replace_stdout_and_stderr_to_conout(void) { const HANDLE conout_handle = CreateFile("CONOUT$", GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, (LPSECURITY_ATTRIBUTES)NULL, OPEN_EXISTING, 0, (HANDLE)NULL); assert(conout_handle != INVALID_HANDLE_VALUE); close(STDOUT_FILENO); const int conout_fd = _open_osfhandle((intptr_t)conout_handle, 0); assert(conout_fd == STDOUT_FILENO); close(STDERR_FILENO); const int conerr_fd = _open_osfhandle((intptr_t)conout_handle, 0); assert(conerr_fd == STDERR_FILENO); } /// Detach from the current console and switch stdio to a hidden private one. /// /// Used when an embedded server must outlive its parent console, while keeping /// CONIN$/CONOUT$ and ConPTY functional for :terminal and stdio writes. void os_swap_to_hidden_console(void) { FreeConsole(); AllocConsole(); ShowWindow(GetConsoleWindow(), SW_HIDE); os_enable_ctrl_c(); os_replace_stdin_to_conin(); os_replace_stdout_and_stderr_to_conout(); } /// Resets Windows console icon if we got an original one on startup. void os_icon_reset(void) { if (hWnd == NULL) { return; } if (hOrigIconSmall) { SendMessage(hWnd, WM_SETICON, (WPARAM)ICON_SMALL, (LPARAM)hOrigIconSmall); } if (hOrigIcon) { SendMessage(hWnd, WM_SETICON, (WPARAM)ICON_BIG, (LPARAM)hOrigIcon); } } /// Sets Nvim logo as Windows console icon. /// /// Saves the original icon so it can be restored at exit. void os_icon_init(void) { if ((hWnd = GetConsoleWindow()) == NULL) { return; } char *vimruntime = os_getenv("VIMRUNTIME"); if (vimruntime != NULL) { snprintf(NameBuff, MAXPATHL, "%s/neovim.ico", vimruntime); if (!os_path_exists(NameBuff)) { WLOG("neovim.ico not found: %s", NameBuff); } else { HICON hVimIcon = LoadImage(NULL, NameBuff, IMAGE_ICON, 64, 64, LR_LOADFROMFILE | LR_LOADMAP3DCOLORS); hOrigIconSmall = (HICON)SendMessage(hWnd, WM_SETICON, (WPARAM)ICON_SMALL, (LPARAM)hVimIcon); hOrigIcon = (HICON)SendMessage(hWnd, WM_SETICON, (WPARAM)ICON_BIG, (LPARAM)hVimIcon); } xfree(vimruntime); } } /// Saves the original Windows console title. void os_title_save(void) { GetConsoleTitle(origTitle, sizeof(origTitle)); } /// Resets the original Windows console title. void os_title_reset(void) { SetConsoleTitle(origTitle); } #if !defined(ENABLE_VIRTUAL_TERMINAL_PROCESSING) # define ENABLE_VIRTUAL_TERMINAL_PROCESSING 0x0004 #endif /// Guesses the terminal-type. Calls SetConsoleMode() and uv_set_vterm_state() /// if appropriate. /// /// @param[in,out] term Name of the guessed terminal, statically-allocated /// @param out_fd stdout file descriptor void os_tty_guess_term(const char **term, int out_fd) { bool conemu_ansi = strequal(os_getenv_noalloc("ConEmuANSI"), "ON"); bool vtp = false; HANDLE handle = (HANDLE)_get_osfhandle(out_fd); DWORD dwMode; if (handle != INVALID_HANDLE_VALUE && GetConsoleMode(handle, &dwMode)) { dwMode |= ENABLE_VIRTUAL_TERMINAL_PROCESSING; if (SetConsoleMode(handle, dwMode)) { vtp = true; } } if (*term == NULL) { if (vtp) { *term = "vtpcon"; } else if (conemu_ansi) { *term = "conemu"; } else { *term = "win32con"; } } if (conemu_ansi) { uv_tty_set_vterm_state(UV_TTY_SUPPORTED); } }