You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

преди 19 години
преди 19 години
преди 19 години
преди 19 години
преди 19 години
преди 19 години
преди 19 години
преди 19 години
преди 19 години
преди 19 години
преди 19 години
преди 19 години
преди 19 години
преди 19 години
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350
  1. /*
  2. * libcucul Canvas for ultrafast compositing of Unicode letters
  3. * Copyright (c) 2002-2006 Sam Hocevar <sam@zoy.org>
  4. * All Rights Reserved
  5. *
  6. * $Id$
  7. *
  8. * This library is free software; you can redistribute it and/or
  9. * modify it under the terms of the Do What The Fuck You Want To
  10. * Public License, Version 2, as published by Sam Hocevar. See
  11. * http://sam.zoy.org/wtfpl/COPYING for more details.
  12. */
  13. /*
  14. * This file contains various canvas handling functions such as character
  15. * and string drawing.
  16. */
  17. #include "config.h"
  18. #include "common.h"
  19. #if !defined(__KERNEL__)
  20. # include <stdio.h> /* BUFSIZ */
  21. # include <string.h>
  22. # include <stdlib.h>
  23. # include <stdarg.h>
  24. # if defined(HAVE_ERRNO_H)
  25. # include <errno.h>
  26. # endif
  27. # if defined(HAVE_UNISTD_H)
  28. # include <unistd.h>
  29. # endif
  30. # if defined(HAVE_SIGNAL_H)
  31. # include <signal.h>
  32. # endif
  33. # if defined(HAVE_SYS_IOCTL_H)
  34. # include <sys/ioctl.h>
  35. # endif
  36. #endif
  37. #include "cucul.h"
  38. #include "cucul_internals.h"
  39. /** \brief Print an ASCII or Unicode character.
  40. *
  41. * This function prints an ASCII or Unicode character at the given
  42. * coordinates, using the default foreground and background values.
  43. *
  44. * If the coordinates are outside the canvas boundaries, nothing is printed.
  45. * If the character value is a non-printable character or is outside the
  46. * UTF-32 range, it is replaced with a space. To print a sequence of bytes
  47. * forming an UTF-8 character instead of an UTF-32 character, use the
  48. * cucul_putstr() function instead.
  49. *
  50. * This function never fails.
  51. *
  52. * \param cv A handle to the libcucul canvas.
  53. * \param x X coordinate.
  54. * \param y Y coordinate.
  55. * \param ch The character to print.
  56. * \return This function always returns 0.
  57. */
  58. int cucul_putchar(cucul_canvas_t *cv, int x, int y, unsigned long int ch)
  59. {
  60. if(x < 0 || x >= (int)cv->width || y < 0 || y >= (int)cv->height)
  61. return 0;
  62. if(ch < 0x20)
  63. ch = 0x20;
  64. cv->chars[x + y * cv->width] = ch;
  65. cv->attr[x + y * cv->width] = (cv->bgcolor << 16) | cv->fgcolor;
  66. return 0;
  67. }
  68. /** \brief Get the Unicode character at the given coordinates.
  69. *
  70. * This function gets the ASCII or Unicode value of the character at
  71. * the given coordinates. If the value is less or equal to 127 (0x7f),
  72. * the character can be printed as ASCII. Otherise, it must be handled
  73. * as a UTF-32 value.
  74. *
  75. * If the coordinates are outside the canvas boundaries, a space (0x20)
  76. * is returned.
  77. *
  78. * This function never fails.
  79. *
  80. * \param cv A handle to the libcucul canvas.
  81. * \param x X coordinate.
  82. * \param y Y coordinate.
  83. * \param ch The requested character value.
  84. * \return The character always returns 0.
  85. */
  86. unsigned long int cucul_getchar(cucul_canvas_t *cv, int x, int y)
  87. {
  88. if(x < 0 || x >= (int)cv->width || y < 0 || y >= (int)cv->height)
  89. return (unsigned char)' ';
  90. return (unsigned long int)cv->chars[x + y * cv->width];
  91. }
  92. /** \brief Print a string.
  93. *
  94. * This function prints an UTF-8 string at the given coordinates, using the
  95. * default foreground and background values. The coordinates may be outside
  96. * the canvas boundaries (eg. a negative Y coordinate) and the string will
  97. * be cropped accordingly if it is too long.
  98. *
  99. * This function never fails.
  100. *
  101. * \param cv A handle to the libcucul canvas.
  102. * \param x X coordinate.
  103. * \param y Y coordinate.
  104. * \param s The string to print.
  105. * \return This function always returns 0.
  106. */
  107. int cucul_putstr(cucul_canvas_t *cv, int x, int y, char const *s)
  108. {
  109. uint32_t *chars, *attr;
  110. unsigned int len;
  111. if(y < 0 || y >= (int)cv->height || x >= (int)cv->width)
  112. return 0;
  113. len = _cucul_strlen_utf8(s);
  114. if(x < 0)
  115. {
  116. if(len < (unsigned int)-x)
  117. return 0;
  118. len -= -x;
  119. s = _cucul_skip_utf8(s, -x);
  120. x = 0;
  121. }
  122. chars = cv->chars + x + y * cv->width;
  123. attr = cv->attr + x + y * cv->width;
  124. if(x + len >= cv->width)
  125. len = cv->width - x;
  126. while(len)
  127. {
  128. *chars++ = cucul_utf8_to_utf32(s, NULL);
  129. *attr++ = (cv->bgcolor << 16) | cv->fgcolor;
  130. s = _cucul_skip_utf8(s, 1);
  131. len--;
  132. }
  133. return 0;
  134. }
  135. /** \brief Print a formated string.
  136. *
  137. * This function formats a string at the given coordinates, using the
  138. * default foreground and background values. The coordinates may be outside
  139. * the canvas boundaries (eg. a negative Y coordinate) and the string will
  140. * be cropped accordingly if it is too long. The syntax of the format
  141. * string is the same as for the C printf() function.
  142. *
  143. * This function never fails.
  144. *
  145. * \param cv A handle to the libcucul canvas.
  146. * \param x X coordinate.
  147. * \param y Y coordinate.
  148. * \param format The format string to print.
  149. * \param ... Arguments to the format string.
  150. * \return This function always returns 0.
  151. */
  152. int cucul_printf(cucul_canvas_t *cv, int x, int y, char const *format, ...)
  153. {
  154. char tmp[BUFSIZ];
  155. char *buf = tmp;
  156. va_list args;
  157. if(y < 0 || y >= (int)cv->height || x >= (int)cv->width)
  158. return 0;
  159. if(cv->width - x + 1 > BUFSIZ)
  160. buf = malloc(cv->width - x + 1);
  161. va_start(args, format);
  162. #if defined(HAVE_VSNPRINTF)
  163. vsnprintf(buf, cv->width - x + 1, format, args);
  164. #else
  165. vsprintf(buf, format, args);
  166. #endif
  167. buf[cv->width - x] = '\0';
  168. va_end(args);
  169. cucul_putstr(cv, x, y, buf);
  170. if(buf != tmp)
  171. free(buf);
  172. return 0;
  173. }
  174. /** \brief Clear the canvas.
  175. *
  176. * This function clears the canvas using the current background colour.
  177. *
  178. * This function never fails.
  179. *
  180. * \param cv The canvas to clear.
  181. * \return This function always returns 0.
  182. */
  183. int cucul_clear_canvas(cucul_canvas_t *cv)
  184. {
  185. uint32_t color = (cv->bgcolor << 16) | cv->fgcolor;
  186. unsigned int n;
  187. /* We could use SLsmg_cls() etc., but drawing empty lines is much faster */
  188. for(n = cv->width * cv->height; n--; )
  189. {
  190. cv->chars[n] = (uint32_t)' ';
  191. cv->attr[n] = color;
  192. }
  193. return 0;
  194. }
  195. /** \brief Blit a canvas onto another one.
  196. *
  197. * This function blits a canvas onto another one at the given coordinates.
  198. * An optional mask canvas can be used.
  199. *
  200. * If an error occurs, -1 is returned and \b errno is set accordingly:
  201. * - \c EINVAL A mask was specified but the mask size and source canvas
  202. * size do not match.
  203. *
  204. * \param dst The destination canvas.
  205. * \param x X coordinate.
  206. * \param y Y coordinate.
  207. * \param src The source canvas.
  208. * \param mask The mask canvas.
  209. * \return 0 in case of success, -1 if an error occurred.
  210. */
  211. int cucul_blit(cucul_canvas_t *dst, int x, int y,
  212. cucul_canvas_t const *src, cucul_canvas_t const *mask)
  213. {
  214. int i, j, starti, startj, endi, endj;
  215. if(mask && (src->width != mask->width || src->height != mask->height))
  216. {
  217. #if defined(HAVE_ERRNO_H)
  218. errno = EINVAL;
  219. #endif
  220. return -1;
  221. }
  222. starti = x < 0 ? -x : 0;
  223. startj = y < 0 ? -y : 0;
  224. endi = (x + src->width >= dst->width) ? dst->width - x : src->width;
  225. endj = (y + src->height >= dst->height) ? dst->height - y : src->height;
  226. if((unsigned int)starti > src->width || (unsigned int)startj > src->height
  227. || starti >= endi || startj >= endj)
  228. return 0;
  229. for(j = startj; j < endj; j++)
  230. {
  231. if(mask)
  232. {
  233. for(i = starti; i < endi; i++)
  234. {
  235. if(mask->chars[j * src->width + i] == (uint32_t)' ')
  236. continue;
  237. dst->chars[(j + y) * dst->width + (i + x)]
  238. = src->chars[j * src->width + i];
  239. dst->attr[(j + y) * dst->width + (i + x)]
  240. = src->attr[j * src->width + i];
  241. }
  242. }
  243. else
  244. {
  245. memcpy(dst->chars + (j + y) * dst->width + starti + x,
  246. src->chars + j * src->width + starti,
  247. (endi - starti) * 4);
  248. memcpy(dst->attr + (j + y) * dst->width + starti + x,
  249. src->attr + j * src->width + starti,
  250. (endi - starti) * 4);
  251. }
  252. }
  253. return 0;
  254. }
  255. /** \brief Set a canvas' new boundaries.
  256. *
  257. * This function sets new boundaries for a canvas. It can be used to crop a
  258. * canvas, to expand it or for combinations of both actions.
  259. *
  260. * If an error occurs, -1 is returned and \b errno is set accordingly:
  261. * - \c EBUSY The canvas is in use by a display driver and cannot be resized.
  262. * - \c ENOMEM Not enough memory for the requested canvas size. If this
  263. * happens, the canvas handle becomes invalid and should not be used.
  264. *
  265. * \param cv The canvas to crop.
  266. * \param x X coordinate of the top-left corner.
  267. * \param y Y coordinate of the top-left corner.
  268. * \param w The width of the cropped area.
  269. * \param h The height of the cropped area.
  270. * \return 0 in case of success, -1 if an error occurred.
  271. */
  272. int cucul_set_canvas_boundaries(cucul_canvas_t *cv, int x, int y,
  273. unsigned int w, unsigned int h)
  274. {
  275. cucul_canvas_t *new;
  276. unsigned int f, saved_f, framecount;
  277. if(cv->refcount)
  278. {
  279. #if defined(HAVE_ERRNO_H)
  280. errno = EBUSY;
  281. #endif
  282. return -1;
  283. }
  284. new = cucul_create_canvas(w, h);
  285. framecount = cucul_get_canvas_frame_count(cv);
  286. saved_f = cv->frame;
  287. for(f = 0; f < framecount; f++)
  288. {
  289. if(f)
  290. cucul_create_canvas_frame(new, framecount);
  291. cucul_set_canvas_frame(cv, f);
  292. cucul_set_canvas_frame(new, f);
  293. cucul_blit(new, -x, -y, cv, NULL);
  294. free(cv->allchars[f]);
  295. free(cv->allattr[f]);
  296. }
  297. free(cv->allchars);
  298. free(cv->allattr);
  299. memcpy(cv, new, sizeof(cucul_canvas_t));
  300. free(new);
  301. cucul_set_canvas_frame(cv, saved_f);
  302. return 0;
  303. }