diff options
Diffstat (limited to 'src/console.c')
-rw-r--r-- | src/console.c | 419 |
1 files changed, 419 insertions, 0 deletions
diff --git a/src/console.c b/src/console.c new file mode 100644 index 0000000..09d3116 --- /dev/null +++ b/src/console.c @@ -0,0 +1,419 @@ +/* +Copyright (c) 2015 + Michał Cichoń <thedmd@interia.pl> + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. +*/ + +#include "console.h" +#include <stdio.h> +#include <stdbool.h> +#include <fcntl.h> +#include <io.h> +#include <pthread.h> +#include "vtparse/vtparse.h" + +enum { READ = 0, WRITE }; + +# define BAR_IN_CAPACITY 1024 +# define BAR_OUT_CAPACITY 1024 + +static const int BarTerminalToAttibColor[8] = { + 0 /* black */, 4 /* red */, 2 /* green */, 6 /* yellow */, + 1 /* blue */, 5 /* magenta */, 3 /* cyan */, 7 /* white */ +}; + +static struct BarConsoleState { + HANDLE StdIn; + HANDLE StdOut; + + int Pipes[2]; + int OriginalStdOut; + int OriginalStdErr; + + WORD DefaultAttributes; + WORD CurrentAttributes; + + pthread_t ConsoleThread; + bool Spawned; + bool Terminate; + + vtparse_t Parser; + + char InBuffer[BAR_IN_CAPACITY]; + size_t InBufferSize; + + char OutBuffer[BAR_OUT_CAPACITY]; + size_t OutBufferSize; +} g_BarConsole; + +static inline void BarOutSetAttributes(WORD attributes) { + g_BarConsole.CurrentAttributes = attributes; + SetConsoleTextAttribute(CONSOLE_REAL_OUTPUT_HANDLE, g_BarConsole.CurrentAttributes); +} + +static inline void BarOutFlush() { + if (g_BarConsole.OutBufferSize == 0) + return; + + _write(g_BarConsole.OriginalStdOut, g_BarConsole.OutBuffer, g_BarConsole.OutBufferSize); + + g_BarConsole.OutBufferSize = 0; +} + +static inline void BarOutPut(char c) { + if (g_BarConsole.OutBufferSize >= BAR_OUT_CAPACITY) + BarOutFlush(); + + g_BarConsole.OutBuffer[g_BarConsole.OutBufferSize++] = c; +} + +static inline void BarOutPuts(const char* c) { + size_t length = strlen(c); + size_t i; + + for (i = 0; i < length; ++i) + BarOutPut(c[i]); +} + +static void BarParseCallback(struct vtparse* parser, vtparse_action_t action, unsigned char ch) { + if (action == VTPARSE_ACTION_PRINT || action == VTPARSE_ACTION_EXECUTE) + { + BarOutPut(ch); + if (action == VTPARSE_ACTION_EXECUTE) + BarOutFlush(); + } + + if (action == VTPARSE_ACTION_CSI_DISPATCH) + { + WORD attribute = g_BarConsole.CurrentAttributes; + int i; + + switch (ch) + { + case 'K': + BarConsoleEraseLine(parser->num_params > 0 ? parser->params[0] : 0); + break; + + case 'm': + for (i = 0; i < parser->num_params; ++i) { + int p = parser->params[i]; + if (p == 0) + attribute = g_BarConsole.DefaultAttributes; + //else if (p == 1) + // attribute |= FOREGROUND_INTENSITY; + else if (p == 4) + attribute |= COMMON_LVB_UNDERSCORE; + else if (p == 7) + attribute |= COMMON_LVB_REVERSE_VIDEO; + //else if (p == 21) + // attribute &= ~FOREGROUND_INTENSITY; + else if (p == 24) + attribute &= ~COMMON_LVB_UNDERSCORE; + else if (p == 27) + attribute &= ~COMMON_LVB_REVERSE_VIDEO; + else if (p >= 30 && p <= 37) + attribute = (attribute & ~0x07) | BarTerminalToAttibColor[p - 30]; + else if (p >= 40 && p <= 47) + attribute = (attribute & ~0x70) | (BarTerminalToAttibColor[p - 40] << 4); + else if (p >= 90 && p <= 97) + attribute = (attribute & ~0x07) | BarTerminalToAttibColor[p - 90] | FOREGROUND_INTENSITY; + else if (p >= 100 && p <= 107) + attribute = ((attribute & ~0x70) | (BarTerminalToAttibColor[p - 100] << 4)) | BACKGROUND_INTENSITY; + } + BarOutFlush(); + BarOutSetAttributes(attribute); + break; + } + } +} + +static void* BarConsoleThread(void* args) { + + while (!g_BarConsole.Terminate) { + + int bytes = _read(g_BarConsole.Pipes[READ], g_BarConsole.InBuffer, BAR_IN_CAPACITY); + + if (bytes > 0) + vtparse(&g_BarConsole.Parser, g_BarConsole.InBuffer, bytes); + + BarOutFlush(); + } + + return NULL; +} + +void BarConsoleInit() { + CONSOLE_SCREEN_BUFFER_INFO csbi; + memset(&g_BarConsole, 0, sizeof(g_BarConsole)); + + SetConsoleCP(CP_UTF8); + SetConsoleOutputCP(CP_UTF8); + + g_BarConsole.StdIn = GetStdHandle(STD_INPUT_HANDLE); + g_BarConsole.StdOut = GetStdHandle(STD_OUTPUT_HANDLE); + + GetConsoleScreenBufferInfo(g_BarConsole.StdOut, &csbi); + + g_BarConsole.DefaultAttributes = csbi.wAttributes; + g_BarConsole.CurrentAttributes = csbi.wAttributes; + + if (_pipe(g_BarConsole.Pipes, 65536, O_BINARY) != -1) { + g_BarConsole.OriginalStdOut = _dup(_fileno(stdout)); + g_BarConsole.OriginalStdErr = _dup(_fileno(stderr)); + + fflush(stdout); + fflush(stderr); + + dup2(g_BarConsole.Pipes[WRITE], fileno(stdout)); + dup2(g_BarConsole.Pipes[WRITE], fileno(stderr)); + + g_BarConsole.Spawned = pthread_create(&g_BarConsole.ConsoleThread, NULL, BarConsoleThread, NULL) == 0; + if (!g_BarConsole.Spawned) + BarConsoleDestroy(); + } + + vtparse_init(&g_BarConsole.Parser, BarParseCallback); +} + +void BarConsoleDestroy() { + if (g_BarConsole.Spawned) { + g_BarConsole.Terminate = true; + fputs(" ", stderr); + fputs(" ", stdout); + + fflush(stdout); + fflush(stderr); + + pthread_join(g_BarConsole.ConsoleThread, NULL); + } + + if (g_BarConsole.OriginalStdErr > 0) { + fflush(stderr); + dup2(g_BarConsole.OriginalStdErr, fileno(stderr)); + _close(g_BarConsole.OriginalStdErr); + g_BarConsole.OriginalStdErr = 0; + } + if (g_BarConsole.OriginalStdOut > 0) { + fflush(stdout); + dup2(g_BarConsole.OriginalStdOut, fileno(stdout)); + _close(g_BarConsole.OriginalStdOut); + g_BarConsole.OriginalStdOut = 0; + } + if (g_BarConsole.Pipes[0] > 0) { + _close(g_BarConsole.Pipes[0]); + g_BarConsole.Pipes[0] = 0; + } + if (g_BarConsole.Pipes[1] > 0) { + _close(g_BarConsole.Pipes[1]); + g_BarConsole.Pipes[1] = 0; + } +} + +HANDLE BarConsoleGetStdIn () { + return g_BarConsole.StdIn; +} + +HANDLE BarConsoleGetStdOut () { + return GetStdHandle(STD_OUTPUT_HANDLE);//g_BarConsole.StdOut; +} + +void BarConsoleSetTitle (const char* title) { + size_t len = MultiByteToWideChar (CP_UTF8, 0, title, -1, NULL, 0); + + TCHAR* wTitle = malloc((len + 1) * sizeof(TCHAR)); + if (NULL != wTitle) + { + MultiByteToWideChar (CP_UTF8, 0, title, -1, wTitle, len); + SetConsoleTitleW (wTitle); + + free(wTitle); + } + else + SetConsoleTitleA (title); +} + +void BarConsoleSetSize (int width, int height) { + HANDLE handle; + SMALL_RECT r; + COORD c, s; + CONSOLE_SCREEN_BUFFER_INFO csbi; + + handle = CONSOLE_REAL_OUTPUT_HANDLE; + + if (!GetConsoleScreenBufferInfo(handle, &csbi)) + return; + + s.X = csbi.srWindow.Right - csbi.srWindow.Left + 1; + s.Y = csbi.srWindow.Bottom - csbi.srWindow.Top + 1; + + if (s.X > width && s.Y > height) + return; + + c.X = width; + c.Y = height; + + if (s.X > c.X) + c.X = s.X; + if (s.Y > c.Y) + c.Y = s.Y; + + SetConsoleScreenBufferSize(handle, c); + + r.Left = 0; + r.Top = 0; + r.Right = c.X - 1; + r.Bottom = c.Y - 1; + SetConsoleWindowInfo(handle, TRUE, &r); +} + +void BarConsoleSetCursorPosition (COORD position) { + SetConsoleCursorPosition(CONSOLE_REAL_OUTPUT_HANDLE, position); +} + +COORD BarConsoleGetCursorPosition () { + CONSOLE_SCREEN_BUFFER_INFO consoleInfo; + COORD result = { -1, -1 }; + + if (GetConsoleScreenBufferInfo(CONSOLE_REAL_OUTPUT_HANDLE, &consoleInfo)) + result = consoleInfo.dwCursorPosition; + + return result; +} + +COORD BarConsoleMoveCursor (int xoffset) { + COORD position; + + position = BarConsoleGetCursorPosition(); + position.X += xoffset; + BarConsoleSetCursorPosition(position); + + return position; +} + +void BarConsoleEraseCharacter () { + TCHAR buffer[256]; + WORD buffer2[256]; + CONSOLE_SCREEN_BUFFER_INFO csbiInfo; + COORD coords; + HANDLE handle = CONSOLE_REAL_OUTPUT_HANDLE; + int length, read, write; + + if (!GetConsoleScreenBufferInfo(handle, &csbiInfo)) + return; + + length = csbiInfo.dwSize.X - csbiInfo.dwCursorPosition.X - 1; + read = csbiInfo.dwCursorPosition.X + 1; + write = csbiInfo.dwCursorPosition.X; + + while (length >= 0) { + int size = min(length, 256); + DWORD chRead = 0, chWritten = 0; + coords = csbiInfo.dwCursorPosition; + coords.X = read; + ReadConsoleOutputAttribute(handle, buffer2, size, coords, &chRead); + ReadConsoleOutputCharacter(handle, buffer, size, coords, &chRead); + + if (chRead == 0) + break; + + coords.X = write; + WriteConsoleOutputAttribute(handle, buffer2, chRead, coords, &chWritten); + WriteConsoleOutputCharacter(handle, buffer, chRead, coords, &chWritten); + + read += chRead; + write += chRead; + length -= chRead; + } +} + +void BarConsoleEraseLine (int mode) { + CONSOLE_SCREEN_BUFFER_INFO csbiInfo; + DWORD writen, length; + COORD coords; + + HANDLE handle = CONSOLE_REAL_OUTPUT_HANDLE; + + if (!GetConsoleScreenBufferInfo(handle, &csbiInfo)) + return; + + writen = 0; + coords.X = 0; + coords.Y = csbiInfo.dwCursorPosition.Y; + + switch (mode) { + default: + case 0: /* from cursor */ + coords.X = BarConsoleGetCursorPosition().X; + length = csbiInfo.dwSize.X - coords.X; + break; + + case 1: /* to cursor */ + coords.X = 0; + length = BarConsoleGetCursorPosition().X; + break; + + case 2: /* whole line */ + coords.X = 0; + length = csbiInfo.dwSize.X; + break; + } + + FillConsoleOutputCharacter(handle, ' ', length, coords, &writen); + FillConsoleOutputAttribute(handle, csbiInfo.wAttributes, csbiInfo.dwSize.X, coords, &writen); + + SetConsoleCursorPosition(handle, coords); +} + +void BarConsoleSetClipboard(const char* text) { + WCHAR* wideString; + HANDLE stringHandle; + size_t wideSize; + + wideSize = MultiByteToWideChar(CP_UTF8, 0, text, -1, NULL, 0); + wideString = calloc(1, (wideSize + 1) * sizeof(WCHAR)); + if (!wideString) + return; + + MultiByteToWideChar(CP_UTF8, 0, text, -1, wideString, wideSize); + + stringHandle = GlobalAlloc(GMEM_MOVEABLE, wideSize); + if (!stringHandle) { + free(wideString); + return; + } + + memcpy(GlobalLock(stringHandle), wideString, wideSize); + + GlobalUnlock(stringHandle); + + if (!OpenClipboard(NULL)) { + GlobalFree(stringHandle); + free(wideString); + return; + } + + EmptyClipboard(); + + SetClipboardData(CF_UNICODETEXT, stringHandle); + + CloseClipboard(); + + free(wideString); +}
\ No newline at end of file |