epdpaint.cpp 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  1. /**
  2. * @filename : epdpaint.cpp
  3. * @brief : Paint tools
  4. * @author : Yehui from Waveshare
  5. *
  6. * Copyright (C) Waveshare September 9 2017
  7. *
  8. * Permission is hereby granted, free of charge, to any person obtaining a copy
  9. * of this software and associated documnetation files (the "Software"), to deal
  10. * in the Software without restriction, including without limitation the rights
  11. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  12. * copies of the Software, and to permit persons to whom the Software is
  13. * furished to do so, subject to the following conditions:
  14. *
  15. * The above copyright notice and this permission notice shall be included in
  16. * all copies or substantial portions of the Software.
  17. *
  18. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  19. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  20. * FITNESS OR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  21. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  22. * LIABILITY WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  23. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  24. * THE SOFTWARE.
  25. */
  26. #include <avr/pgmspace.h>
  27. #include "epdpaint.h"
  28. Paint::Paint(unsigned char* image, int width, int height) {
  29. this->rotate = ROTATE_0;
  30. this->image = image;
  31. /* 1 byte = 8 pixels, so the width should be the multiple of 8 */
  32. this->width = width % 8 ? width + 8 - (width % 8) : width;
  33. this->height = height;
  34. }
  35. Paint::~Paint() {
  36. }
  37. /**
  38. * @brief: clear the image
  39. */
  40. void Paint::Clear(int colored) {
  41. for (int x = 0; x < this->width; x++) {
  42. for (int y = 0; y < this->height; y++) {
  43. DrawAbsolutePixel(x, y, colored);
  44. }
  45. }
  46. }
  47. /**
  48. * @brief: this draws a pixel by absolute coordinates.
  49. * this function won't be affected by the rotate parameter.
  50. */
  51. void Paint::DrawAbsolutePixel(int x, int y, int colored) {
  52. if (x < 0 || x >= this->width || y < 0 || y >= this->height) {
  53. return;
  54. }
  55. if (IF_INVERT_COLOR) {
  56. if (colored) {
  57. image[(x + y * this->width) / 8] |= 0x80 >> (x % 8);
  58. } else {
  59. image[(x + y * this->width) / 8] &= ~(0x80 >> (x % 8));
  60. }
  61. } else {
  62. if (colored) {
  63. image[(x + y * this->width) / 8] &= ~(0x80 >> (x % 8));
  64. } else {
  65. image[(x + y * this->width) / 8] |= 0x80 >> (x % 8);
  66. }
  67. }
  68. }
  69. /**
  70. * @brief: Getters and Setters
  71. */
  72. unsigned char* Paint::GetImage(void) {
  73. return this->image;
  74. }
  75. int Paint::GetWidth(void) {
  76. return this->width;
  77. }
  78. void Paint::SetWidth(int width) {
  79. this->width = width % 8 ? width + 8 - (width % 8) : width;
  80. }
  81. int Paint::GetHeight(void) {
  82. return this->height;
  83. }
  84. void Paint::SetHeight(int height) {
  85. this->height = height;
  86. }
  87. int Paint::GetRotate(void) {
  88. return this->rotate;
  89. }
  90. void Paint::SetRotate(int rotate){
  91. this->rotate = rotate;
  92. }
  93. /**
  94. * @brief: this draws a pixel by the coordinates
  95. */
  96. void Paint::DrawPixel(int x, int y, int colored) {
  97. int point_temp;
  98. if (this->rotate == ROTATE_0) {
  99. if(x < 0 || x >= this->width || y < 0 || y >= this->height) {
  100. return;
  101. }
  102. DrawAbsolutePixel(x, y, colored);
  103. } else if (this->rotate == ROTATE_90) {
  104. if(x < 0 || x >= this->height || y < 0 || y >= this->width) {
  105. return;
  106. }
  107. point_temp = x;
  108. x = this->width - y;
  109. y = point_temp;
  110. DrawAbsolutePixel(x, y, colored);
  111. } else if (this->rotate == ROTATE_180) {
  112. if(x < 0 || x >= this->width || y < 0 || y >= this->height) {
  113. return;
  114. }
  115. x = this->width - x;
  116. y = this->height - y;
  117. DrawAbsolutePixel(x, y, colored);
  118. } else if (this->rotate == ROTATE_270) {
  119. if(x < 0 || x >= this->height || y < 0 || y >= this->width) {
  120. return;
  121. }
  122. point_temp = x;
  123. x = y;
  124. y = this->height - point_temp;
  125. DrawAbsolutePixel(x, y, colored);
  126. }
  127. }
  128. /**
  129. * @brief: this draws a charactor on the frame buffer but not refresh
  130. */
  131. void Paint::DrawCharAt(int x, int y, char ascii_char, sFONT* font, int colored) {
  132. int i, j;
  133. unsigned int char_offset = (ascii_char - ' ') * font->Height * (font->Width / 8 + (font->Width % 8 ? 1 : 0));
  134. const unsigned char* ptr = &font->table[char_offset];
  135. for (j = 0; j < font->Height; j++) {
  136. for (i = 0; i < font->Width; i++) {
  137. if (pgm_read_byte(ptr) & (0x80 >> (i % 8))) {
  138. DrawPixel(x + i, y + j, colored);
  139. }
  140. if (i % 8 == 7) {
  141. ptr++;
  142. }
  143. }
  144. if (font->Width % 8 != 0) {
  145. ptr++;
  146. }
  147. }
  148. }
  149. /**
  150. * @brief: this displays a string on the frame buffer but not refresh
  151. */
  152. void Paint::DrawStringAt(int x, int y, const char* text, sFONT* font, int colored) {
  153. const char* p_text = text;
  154. unsigned int counter = 0;
  155. int refcolumn = x;
  156. /* Send the string character by character on EPD */
  157. while (*p_text != 0) {
  158. /* Display one character on EPD */
  159. DrawCharAt(refcolumn, y, *p_text, font, colored);
  160. /* Decrement the column position by 16 */
  161. refcolumn += font->Width;
  162. /* Point on the next character */
  163. p_text++;
  164. counter++;
  165. }
  166. }
  167. /**
  168. * @brief: this draws a line on the frame buffer
  169. */
  170. void Paint::DrawLine(int x0, int y0, int x1, int y1, int colored) {
  171. /* Bresenham algorithm */
  172. int dx = x1 - x0 >= 0 ? x1 - x0 : x0 - x1;
  173. int sx = x0 < x1 ? 1 : -1;
  174. int dy = y1 - y0 <= 0 ? y1 - y0 : y0 - y1;
  175. int sy = y0 < y1 ? 1 : -1;
  176. int err = dx + dy;
  177. while((x0 != x1) && (y0 != y1)) {
  178. DrawPixel(x0, y0 , colored);
  179. if (2 * err >= dy) {
  180. err += dy;
  181. x0 += sx;
  182. }
  183. if (2 * err <= dx) {
  184. err += dx;
  185. y0 += sy;
  186. }
  187. }
  188. }
  189. /**
  190. * @brief: this draws a horizontal line on the frame buffer
  191. */
  192. void Paint::DrawHorizontalLine(int x, int y, int line_width, int colored) {
  193. int i;
  194. for (i = x; i < x + line_width; i++) {
  195. DrawPixel(i, y, colored);
  196. }
  197. }
  198. /**
  199. * @brief: this draws a vertical line on the frame buffer
  200. */
  201. void Paint::DrawVerticalLine(int x, int y, int line_height, int colored) {
  202. int i;
  203. for (i = y; i < y + line_height; i++) {
  204. DrawPixel(x, i, colored);
  205. }
  206. }
  207. /**
  208. * @brief: this draws a rectangle
  209. */
  210. void Paint::DrawRectangle(int x0, int y0, int x1, int y1, int colored) {
  211. int min_x, min_y, max_x, max_y;
  212. min_x = x1 > x0 ? x0 : x1;
  213. max_x = x1 > x0 ? x1 : x0;
  214. min_y = y1 > y0 ? y0 : y1;
  215. max_y = y1 > y0 ? y1 : y0;
  216. DrawHorizontalLine(min_x, min_y, max_x - min_x + 1, colored);
  217. DrawHorizontalLine(min_x, max_y, max_x - min_x + 1, colored);
  218. DrawVerticalLine(min_x, min_y, max_y - min_y + 1, colored);
  219. DrawVerticalLine(max_x, min_y, max_y - min_y + 1, colored);
  220. }
  221. /**
  222. * @brief: this draws a filled rectangle
  223. */
  224. void Paint::DrawFilledRectangle(int x0, int y0, int x1, int y1, int colored) {
  225. int min_x, min_y, max_x, max_y;
  226. int i;
  227. min_x = x1 > x0 ? x0 : x1;
  228. max_x = x1 > x0 ? x1 : x0;
  229. min_y = y1 > y0 ? y0 : y1;
  230. max_y = y1 > y0 ? y1 : y0;
  231. for (i = min_x; i <= max_x; i++) {
  232. DrawVerticalLine(i, min_y, max_y - min_y + 1, colored);
  233. }
  234. }
  235. /**
  236. * @brief: this draws a circle
  237. */
  238. void Paint::DrawCircle(int x, int y, int radius, int colored) {
  239. /* Bresenham algorithm */
  240. int x_pos = -radius;
  241. int y_pos = 0;
  242. int err = 2 - 2 * radius;
  243. int e2;
  244. do {
  245. DrawPixel(x - x_pos, y + y_pos, colored);
  246. DrawPixel(x + x_pos, y + y_pos, colored);
  247. DrawPixel(x + x_pos, y - y_pos, colored);
  248. DrawPixel(x - x_pos, y - y_pos, colored);
  249. e2 = err;
  250. if (e2 <= y_pos) {
  251. err += ++y_pos * 2 + 1;
  252. if(-x_pos == y_pos && e2 <= x_pos) {
  253. e2 = 0;
  254. }
  255. }
  256. if (e2 > x_pos) {
  257. err += ++x_pos * 2 + 1;
  258. }
  259. } while (x_pos <= 0);
  260. }
  261. /**
  262. * @brief: this draws a filled circle
  263. */
  264. void Paint::DrawFilledCircle(int x, int y, int radius, int colored) {
  265. /* Bresenham algorithm */
  266. int x_pos = -radius;
  267. int y_pos = 0;
  268. int err = 2 - 2 * radius;
  269. int e2;
  270. do {
  271. DrawPixel(x - x_pos, y + y_pos, colored);
  272. DrawPixel(x + x_pos, y + y_pos, colored);
  273. DrawPixel(x + x_pos, y - y_pos, colored);
  274. DrawPixel(x - x_pos, y - y_pos, colored);
  275. DrawHorizontalLine(x + x_pos, y + y_pos, 2 * (-x_pos) + 1, colored);
  276. DrawHorizontalLine(x + x_pos, y - y_pos, 2 * (-x_pos) + 1, colored);
  277. e2 = err;
  278. if (e2 <= y_pos) {
  279. err += ++y_pos * 2 + 1;
  280. if(-x_pos == y_pos && e2 <= x_pos) {
  281. e2 = 0;
  282. }
  283. }
  284. if(e2 > x_pos) {
  285. err += ++x_pos * 2 + 1;
  286. }
  287. } while(x_pos <= 0);
  288. }
  289. /* END OF FILE */