MD_AnsiCons.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378
  1. #include <windows.h>
  2. #include <stdarg.h>
  3. #include <stdio.h>
  4. #include <errno.h>
  5. #include <io.h>
  6. #ifndef FOREGROUND_MASK
  7. # define FOREGROUND_MASK (FOREGROUND_RED|FOREGROUND_BLUE|FOREGROUND_GREEN)
  8. #endif
  9. #ifndef BACKGROUND_MASK
  10. # define BACKGROUND_MASK (BACKGROUND_RED|BACKGROUND_BLUE|BACKGROUND_GREEN)
  11. #endif
  12. int ansi_write(FILE* fp, const char* buf, int len) {
  13. static int first = 1;
  14. static int state = 0;
  15. static int n;
  16. static int v[6];
  17. static COORD saved_coord = { 0, 0 };
  18. int type;
  19. HANDLE handle = INVALID_HANDLE_VALUE;
  20. static WORD attr;
  21. DWORD written, csize;
  22. CONSOLE_CURSOR_INFO cci;
  23. CONSOLE_SCREEN_BUFFER_INFO csbi;
  24. COORD coord;
  25. const char* ptr = buf;
  26. int w, h;
  27. if (fp == stdout) {
  28. type = 0;
  29. }
  30. else if (fp == stderr) {
  31. type = 1;
  32. }
  33. else {
  34. type = 0;
  35. }
  36. handle = (HANDLE)_get_osfhandle(fileno(fp));
  37. GetConsoleScreenBufferInfo(handle, &csbi);
  38. attr = csbi.wAttributes;
  39. int z = 0;
  40. for (z = 0; z < len; z++) {
  41. if (state == 0) {
  42. if (ptr[z] == '\033') {
  43. state = 1;
  44. continue;
  45. }
  46. else {
  47. fputc(ptr[z], fp);
  48. }
  49. }
  50. else if (state == 1) {
  51. if (ptr[z] == '[') {
  52. state = 2;
  53. }
  54. else {
  55. state = 0;
  56. }
  57. } else if (state == 2) {
  58. n = 0;
  59. for (int j = 0; j < 6; j++) {
  60. v[j] = 0;
  61. }
  62. state = 3;
  63. }
  64. if (state == 3) {
  65. if (ptr[z] == ';') {
  66. if (n < 6) {
  67. n++;
  68. }
  69. continue;
  70. } else if (isdigit(ptr[z])) {
  71. if (n == 0) {
  72. n = 1;
  73. }
  74. v[n-1] = v[n-1] * 10 + (ptr[z] - '0');
  75. continue;
  76. }
  77. else {
  78. state = 4;
  79. }
  80. }
  81. if (state == 4) {
  82. state = 0;
  83. switch (ptr[z]) {
  84. case 'h':
  85. for (int i = 0; i < n; i++) {
  86. switch (v[i]) {
  87. case 3:
  88. GetConsoleScreenBufferInfo(handle, &csbi);
  89. w = csbi.dwSize.X;
  90. h = csbi.srWindow.Bottom - csbi.srWindow.Top + 1;
  91. csize = w * (h + 1);
  92. coord.X = 0;
  93. coord.Y = csbi.srWindow.Top;
  94. FillConsoleOutputCharacter(handle, ' ', csize, coord, &written);
  95. FillConsoleOutputAttribute(handle, csbi.wAttributes, csize, coord, &written);
  96. SetConsoleCursorPosition(handle, csbi.dwCursorPosition);
  97. csbi.dwSize.X = 132;
  98. SetConsoleScreenBufferSize(handle, csbi.dwSize);
  99. csbi.srWindow.Right = csbi.srWindow.Left + 131;
  100. SetConsoleWindowInfo(handle, TRUE, &csbi.srWindow);
  101. break;
  102. case 5:
  103. attr =
  104. ((attr & FOREGROUND_MASK) << 4) |
  105. ((attr & BACKGROUND_MASK) >> 4);
  106. SetConsoleTextAttribute(handle, attr);
  107. break;
  108. case 9:
  109. break;
  110. case 25:
  111. GetConsoleCursorInfo(handle, &cci);
  112. cci.bVisible = TRUE;
  113. SetConsoleCursorInfo(handle, &cci);
  114. break;
  115. case 47:
  116. coord.X = 0;
  117. coord.Y = 0;
  118. SetConsoleCursorPosition(handle, coord);
  119. break;
  120. default:
  121. break;
  122. }
  123. }
  124. break;
  125. case 'l':
  126. for (int i = 0; i < n; i++) {
  127. switch (v[i]) {
  128. case 3:
  129. GetConsoleScreenBufferInfo(handle, &csbi);
  130. w = csbi.dwSize.X;
  131. h = csbi.srWindow.Bottom - csbi.srWindow.Top + 1;
  132. csize = w * (h + 1);
  133. coord.X = 0;
  134. coord.Y = csbi.srWindow.Top;
  135. FillConsoleOutputCharacter(handle, ' ', csize, coord, &written);
  136. FillConsoleOutputAttribute(handle, csbi.wAttributes, csize, coord, &written);
  137. SetConsoleCursorPosition(handle, csbi.dwCursorPosition);
  138. csbi.srWindow.Right = csbi.srWindow.Left + 79;
  139. SetConsoleWindowInfo(handle, TRUE, &csbi.srWindow);
  140. csbi.dwSize.X = 80;
  141. SetConsoleScreenBufferSize(handle, csbi.dwSize);
  142. break;
  143. case 5:
  144. attr =
  145. ((attr & FOREGROUND_MASK) << 4) |
  146. ((attr & BACKGROUND_MASK) >> 4);
  147. SetConsoleTextAttribute(handle, attr);
  148. break;
  149. case 25:
  150. GetConsoleCursorInfo(handle, &cci);
  151. cci.bVisible = FALSE;
  152. SetConsoleCursorInfo(handle, &cci);
  153. break;
  154. default:
  155. break;
  156. }
  157. }
  158. break;
  159. case 'm':
  160. for (int i = 0; i < n; i++) {
  161. if (v[i] == -1 || v[i] == 0) {
  162. attr &= ~(FOREGROUND_INTENSITY);
  163. attr |= FOREGROUND_MASK;
  164. attr &= ~(BACKGROUND_MASK);
  165. }
  166. else if (v[i] == 1)
  167. attr |= FOREGROUND_INTENSITY;
  168. else if (v[i] == 4)
  169. attr |= FOREGROUND_INTENSITY;
  170. else if (v[i] == 5)
  171. attr |= FOREGROUND_INTENSITY;
  172. else if (v[i] == 7)
  173. attr =
  174. ((attr & FOREGROUND_MASK) << 4) |
  175. ((attr & BACKGROUND_MASK) >> 4);
  176. else if (v[i] == 10)
  177. ; // symbol on
  178. else if (v[i] == 11)
  179. ; // symbol off
  180. else if (v[i] == 22)
  181. attr &= ~FOREGROUND_INTENSITY;
  182. else if (v[i] == 24)
  183. attr &= ~FOREGROUND_INTENSITY;
  184. else if (v[i] == 25)
  185. attr &= ~FOREGROUND_INTENSITY;
  186. else if (v[i] == 27)
  187. attr =
  188. ((attr & FOREGROUND_MASK) << 4) |
  189. ((attr & BACKGROUND_MASK) >> 4);
  190. else if (v[i] >= 30 && v[i] <= 37) {
  191. attr = (attr & (BACKGROUND_MASK | FOREGROUND_INTENSITY));
  192. if ((v[i] - 30) & 1)
  193. attr |= FOREGROUND_RED;
  194. if ((v[i] - 30) & 2)
  195. attr |= FOREGROUND_GREEN;
  196. if ((v[i] - 30) & 4)
  197. attr |= FOREGROUND_BLUE;
  198. }
  199. //else if (v[i] == 39)
  200. //attr = (~attr & BACKGROUND_MASK);
  201. else if (v[i] >= 40 && v[i] <= 47) {
  202. attr = (attr & (FOREGROUND_MASK | BACKGROUND_INTENSITY));
  203. if ((v[i] - 40) & 1)
  204. attr |= BACKGROUND_RED;
  205. if ((v[i] - 40) & 2)
  206. attr |= BACKGROUND_GREEN;
  207. if ((v[i] - 40) & 4)
  208. attr |= BACKGROUND_BLUE;
  209. }
  210. else if (v[i] >= 90 && v[i] <= 97) {
  211. attr = (attr & BACKGROUND_MASK) | FOREGROUND_INTENSITY;
  212. if ((v[i] - 90) & 1)
  213. attr |= FOREGROUND_RED;
  214. if ((v[i] - 90) & 2)
  215. attr |= FOREGROUND_GREEN;
  216. if ((v[i] - 90) & 4)
  217. attr |= FOREGROUND_BLUE;
  218. }
  219. else if (v[i] >= 100 && v[i] <= 107) {
  220. attr = (attr & FOREGROUND_MASK) | BACKGROUND_INTENSITY;
  221. if ((v[i] - 100) & 1)
  222. attr |= BACKGROUND_RED;
  223. if ((v[i] - 100) & 2)
  224. attr |= BACKGROUND_GREEN;
  225. if ((v[i] - 100) & 4)
  226. attr |= BACKGROUND_BLUE;
  227. }
  228. //else if (v[i] == 49)
  229. //attr = (~attr & FOREGROUND_MASK);
  230. }
  231. SetConsoleTextAttribute(handle, attr);
  232. break;
  233. case 'K':
  234. GetConsoleScreenBufferInfo(handle, &csbi);
  235. coord = csbi.dwCursorPosition;
  236. switch (v[0]) {
  237. default:
  238. case 0:
  239. csize = csbi.dwSize.X - coord.X;
  240. break;
  241. case 1:
  242. csize = coord.X;
  243. coord.X = 0;
  244. break;
  245. case 2:
  246. csize = csbi.dwSize.X;
  247. coord.X = 0;
  248. break;
  249. }
  250. FillConsoleOutputCharacter(handle, ' ', csize, coord, &written);
  251. FillConsoleOutputAttribute(handle, attr, csize, coord, &written);
  252. SetConsoleCursorPosition(handle, csbi.dwCursorPosition);
  253. break;
  254. case 'J':
  255. GetConsoleScreenBufferInfo(handle, &csbi);
  256. w = csbi.dwSize.X;
  257. h = csbi.srWindow.Bottom - csbi.srWindow.Top + 1;
  258. coord = csbi.dwCursorPosition;
  259. switch (v[0]) {
  260. default:
  261. case 0:
  262. csize = w * (h - coord.Y) - coord.X;
  263. coord.X = 0;
  264. break;
  265. case 1:
  266. csize = w * coord.Y + coord.X;
  267. coord.X = 0;
  268. coord.Y = csbi.srWindow.Top;
  269. break;
  270. case 2:
  271. csize = w * (h + 1);
  272. coord.X = 0;
  273. coord.Y = csbi.srWindow.Top;
  274. break;
  275. }
  276. FillConsoleOutputCharacter(handle, ' ', csize, coord, &written);
  277. FillConsoleOutputAttribute(handle, attr, csize, coord, &written);
  278. SetConsoleCursorPosition(handle, csbi.dwCursorPosition);
  279. break;
  280. case 'H':
  281. GetConsoleScreenBufferInfo(handle, &csbi);
  282. coord = csbi.dwCursorPosition;
  283. if (v[0] != -1) {
  284. if (v[1] != -1) {
  285. coord.Y = csbi.srWindow.Top + v[0] - 1;
  286. coord.X = v[1] - 1;
  287. }
  288. else
  289. coord.X = v[0] - 1;
  290. }
  291. else {
  292. coord.X = 0;
  293. coord.Y = csbi.srWindow.Top;
  294. }
  295. if (coord.X < csbi.srWindow.Left)
  296. coord.X = csbi.srWindow.Left;
  297. else if (coord.X > csbi.srWindow.Right)
  298. coord.X = csbi.srWindow.Right;
  299. if (coord.Y < csbi.srWindow.Top)
  300. coord.Y = csbi.srWindow.Top;
  301. else if (coord.Y > csbi.srWindow.Bottom)
  302. coord.Y = csbi.srWindow.Bottom;
  303. SetConsoleCursorPosition(handle, coord);
  304. break;
  305. case 'A': // Move up
  306. GetConsoleScreenBufferInfo(handle, &csbi);
  307. coord = csbi.dwCursorPosition;
  308. if (n > 0) {
  309. coord.Y = coord.Y - v[0];
  310. }
  311. else {
  312. coord.Y = coord.Y - 1;
  313. }
  314. if (coord.Y < csbi.srWindow.Top) coord.Y = csbi.srWindow.Top;
  315. SetConsoleCursorPosition(handle, coord);
  316. SetConsoleTextAttribute(handle, attr);
  317. break;
  318. case 'B': // Move down
  319. GetConsoleScreenBufferInfo(handle, &csbi);
  320. coord = csbi.dwCursorPosition;
  321. if (n > 0) {
  322. coord.Y = coord.Y + v[0];
  323. }
  324. else {
  325. coord.Y = coord.Y + 1;
  326. }
  327. if (coord.Y > csbi.srWindow.Bottom) coord.Y = csbi.srWindow.Bottom;
  328. SetConsoleCursorPosition(handle, coord);
  329. SetConsoleTextAttribute(handle, attr);
  330. break;
  331. case 'C': // Move forward / right
  332. GetConsoleScreenBufferInfo(handle, &csbi);
  333. coord = csbi.dwCursorPosition;
  334. if (n > 0) {
  335. coord.X = coord.X + v[0];
  336. }
  337. else {
  338. coord.X = coord.X + 1;
  339. }
  340. if (coord.X > csbi.srWindow.Right) coord.X = csbi.srWindow.Right;
  341. SetConsoleCursorPosition(handle, coord);
  342. SetConsoleTextAttribute(handle, attr);
  343. break;
  344. case 'D': // Move backward / left
  345. GetConsoleScreenBufferInfo(handle, &csbi);
  346. coord = csbi.dwCursorPosition;
  347. if (n > 0) {
  348. coord.X = coord.X - v[0];
  349. }
  350. else {
  351. coord.X = coord.X - 1;
  352. }
  353. if (coord.X < csbi.srWindow.Left) coord.X = csbi.srWindow.Left;
  354. SetConsoleCursorPosition(handle, coord);
  355. SetConsoleTextAttribute(handle, attr);
  356. break;
  357. case 's':
  358. GetConsoleScreenBufferInfo(handle, &csbi);
  359. saved_coord = csbi.dwCursorPosition;
  360. break;
  361. case 'u':
  362. SetConsoleCursorPosition(handle, saved_coord);
  363. break;
  364. default:
  365. break;
  366. }
  367. }
  368. }
  369. return z;
  370. }