ui_region.c 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  1. /*
  2. * Copyright (c) 2019 Actions Semi Co., Inc.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. */
  9. #include <ui_region.h>
  10. #ifdef __ZEPHYR__
  11. # include <linker/linker-defs.h>
  12. #else
  13. # define __ramfunc
  14. #endif
  15. /**
  16. * @brief Get the size of an region
  17. *
  18. * @param area_p pointer to an region
  19. *
  20. * @retval size of the region
  21. */
  22. int32_t ui_region_get_size(const ui_region_t * region)
  23. {
  24. return (int32_t)(region->x2 - region->x1 + 1) * (region->y2 - region->y1 + 1);
  25. }
  26. /**
  27. * @brief Set the x coordinate of an region
  28. *
  29. * @param region pointer to an region
  30. * @param x new x coordinate
  31. *
  32. * @retval N/A
  33. */
  34. void ui_region_set_x(ui_region_t * region, int16_t x)
  35. {
  36. region->x2 += x - region->x1;
  37. region->x1 = x;
  38. }
  39. /**
  40. * @brief Set the y coordinate of an region
  41. *
  42. * @param region pointer to an region
  43. * @param y new y coordinate
  44. *
  45. * @retval N/A
  46. */
  47. void ui_region_set_y(ui_region_t * region, int16_t y)
  48. {
  49. region->y2 += y - region->y1;
  50. region->y1 = y;
  51. }
  52. /**
  53. * @brief Set the position of an region (width and height will be kept)
  54. *
  55. * @param region pointer to an region
  56. * @param x new x coordinate
  57. * @param y new y coordinate
  58. *
  59. * @retval N/A
  60. */
  61. void ui_region_set_pos(ui_region_t * region, int16_t x, int16_t y)
  62. {
  63. ui_region_set_x(region, x);
  64. ui_region_set_y(region, y);
  65. }
  66. /**
  67. * @brief Set the width of an region
  68. *
  69. * @param region pointer to an region
  70. * @param w the new width of region (w == 1 makes x1 == x2)
  71. *
  72. * @retval N/A
  73. */
  74. void ui_region_set_width(ui_region_t * region, int16_t w)
  75. {
  76. region->x2 = region->x1 + w - 1;
  77. }
  78. /**
  79. * @brief Set the height of an region
  80. *
  81. * @param region pointer to an region
  82. * @param w the new height of region (h == 1 makes y1 == y2)
  83. *
  84. * @retval N/A
  85. */
  86. void ui_region_set_height(ui_region_t * region, int16_t h)
  87. {
  88. region->y2 = region->y1 + h - 1;
  89. }
  90. /**
  91. * @brief Move an region
  92. *
  93. * @param region pointer to an region
  94. * @param dx delta X to move
  95. * @param dy delta Y to move
  96. *
  97. * @retval N/A
  98. */
  99. void ui_region_move(ui_region_t * region, int16_t dx, int16_t dy)
  100. {
  101. region->x1 += dx;
  102. region->y1 += dy;
  103. region->x2 += dx;
  104. region->y2 += dy;
  105. }
  106. /**
  107. * @brief Get the common parts of two regions
  108. *
  109. * @param result pointer to an region, the result will be stored here
  110. * @param region1 pointer to the first region
  111. * @param region2 pointer to the second region
  112. *
  113. * @return false: the two area has NO common parts, res_p is invalid
  114. */
  115. bool __ramfunc ui_region_intersect(ui_region_t * result, const ui_region_t * region1, const ui_region_t * region2)
  116. {
  117. bool union_ok = true;
  118. /* Get the smaller region */
  119. result->x1 = UI_MAX(region1->x1, region2->x1);
  120. result->y1 = UI_MAX(region1->y1, region2->y1);
  121. result->x2 = UI_MIN(region1->x2, region2->x2);
  122. result->y2 = UI_MIN(region1->y2, region2->y2);
  123. /*If x1 or y1 greater then x2 or y2 then the areas union is empty*/
  124. if ((result->x1 > result->x2) || (result->y1 > result->y2)) {
  125. union_ok = false;
  126. }
  127. return union_ok;
  128. }
  129. /**
  130. * @brief merge two regions into a third which involves the other two
  131. *
  132. * @param result pointer to an region, the result will be stored here
  133. * @param region1 pointer to the first region
  134. * @param region2 pointer to the second region
  135. *
  136. * @retval N/A
  137. */
  138. void ui_region_merge(ui_region_t * result, const ui_region_t * region1, const ui_region_t * region2)
  139. {
  140. result->x1 = UI_MIN(region1->x1, region2->x1);
  141. result->y1 = UI_MIN(region1->y1, region2->y1);
  142. result->x2 = UI_MAX(region1->x2, region2->x2);
  143. result->y2 = UI_MAX(region1->y2, region2->y2);
  144. }
  145. /**
  146. * @brief Subtract the common parts of two regions
  147. *
  148. * @param result pointer to a region array (4 at maximum), the result will be stored here
  149. * @param region pointer to a region
  150. * @param exclude pointer to the region to be subtracted from 'region'
  151. *
  152. * @retval the number of result regions
  153. */
  154. int ui_region_subtract(ui_region_t * result, const ui_region_t * region, const ui_region_t * exclude)
  155. {
  156. int16_t cur_y = region->y1;
  157. int cnt = 0;
  158. if (region->x1 >= exclude->x1 && region->y1 >= exclude->y1 &&
  159. region->x2 <= exclude->x1 && region->y2 <= exclude->y1) {
  160. return 0;
  161. }
  162. if (ui_region_is_on(region, exclude) == false) {
  163. ui_region_copy(result, region);
  164. return 1;
  165. }
  166. /* the top part */
  167. if (region->y1 < exclude->y1) {
  168. result[cnt].x1 = region->x1;
  169. result[cnt].y1 = region->y1;
  170. result[cnt].x2 = region->x2;
  171. result[cnt++].y2 = exclude->y1 - 1;
  172. cur_y = exclude->y1;
  173. }
  174. /* the left part */
  175. if (region->x1 < exclude->x1) {
  176. result[cnt].x1 = region->x1;
  177. result[cnt].y1 = cur_y;
  178. result[cnt].x2 = exclude->x1 - 1;
  179. result[cnt++].y2 = UI_MIN(region->y2, exclude->y2);
  180. }
  181. /* the right part */
  182. if (region->x2 > exclude->x2) {
  183. result[cnt].x1 = exclude->x2 + 1;
  184. result[cnt].y1 = cur_y;
  185. result[cnt].x2 = region->x2;
  186. result[cnt++].y2 = UI_MIN(region->y2, exclude->y2);
  187. }
  188. /* the bottom part */
  189. if (region->y2 > exclude->y2) {
  190. result[cnt].x1 = region->x1;
  191. result[cnt].y1 = exclude->y2 + 1;
  192. result[cnt].x2 = region->x2;
  193. result[cnt++].y2 = region->y2;
  194. }
  195. return cnt;
  196. }
  197. /**
  198. * @brief Try smallest adjustment of region position to make it totally fall in the holder region
  199. *
  200. * @param region pointer to an region which should fall in 'holder'
  201. * @param holder pointer to an region which involves 'region'
  202. *
  203. * @retval N/A
  204. */
  205. void ui_region_fit_in(ui_region_t * region, const ui_region_t * holder)
  206. {
  207. int16_t dx = 0;
  208. int16_t dy = 0;
  209. if (region->x1 < holder->x1 && region->x2 < holder->x2) {
  210. dx = UI_MIN(holder->x1 - region->x1, holder->x2 - region->x2);
  211. } else if (region->x1 > holder->x1 && region->x2 > holder->x2) {
  212. dx = UI_MAX(holder->x1 - region->x1, holder->x2 - region->x2);
  213. }
  214. if (region->y1 < holder->y1 && region->y2 < holder->y2) {
  215. dy = UI_MIN(holder->y1 - region->y1, holder->y2 - region->y2);
  216. } else if (region->y1 > holder->y1 && region->y2 > holder->y2) {
  217. dy = UI_MAX(holder->y1 - region->y1, holder->y2 - region->y2);
  218. }
  219. ui_region_set_pos(region, region->x1 + dx, region->y1 + dy);
  220. }
  221. /**
  222. * @brief Check if a point is on an region
  223. *
  224. * @param region pointer to an region
  225. * @param point pointer to a point
  226. *
  227. * @return false: the point is out of the area
  228. */
  229. bool ui_region_is_point_on(const ui_region_t * region, const ui_point_t * point)
  230. {
  231. if ((point->x >= region->x1 && point->x <= region->x2) &&
  232. ((point->y >= region->y1 && point->y <= region->y2))) {
  233. return true;
  234. } else {
  235. return false;
  236. }
  237. }
  238. /**
  239. * @brief Check if two region has common parts
  240. *
  241. * @param region1 pointer to an region.
  242. * @param region2 pointer to an other region
  243. *
  244. * @return false: region1 and region2 has no common parts
  245. */
  246. bool ui_region_is_on(const ui_region_t * region1, const ui_region_t * region2)
  247. {
  248. if ((region1->x1 <= region2->x2) && (region1->x2 >= region2->x1) &&
  249. (region1->y1 <= region2->y2) && (region1->y2 >= region2->y1)) {
  250. return true;
  251. } else {
  252. return false;
  253. }
  254. }
  255. /**
  256. * @brief Check if a region is fully on an other
  257. *
  258. * @param region pointer to an region which could be in 'holder'
  259. * @param holder pointer to an region which could involve 'region'
  260. *
  261. * @return true: `region` is fully inside `holder`
  262. */
  263. bool ui_region_is_in(const ui_region_t * region, const ui_region_t * holder)
  264. {
  265. if (region->x1 >= holder->x1 && region->y1 >= holder->y1 &&
  266. region->x2 <= holder->x2 && region->y2 <= holder->y2) {
  267. return true;
  268. } else {
  269. return false;
  270. }
  271. }
  272. /**
  273. * @brief Check if an region is valid
  274. A valid region has a non negative width and height.
  275. * @param region pointer to an region.
  276. *
  277. * @return true if region is valid
  278. */
  279. bool ui_region_is_valid(const ui_region_t * region)
  280. {
  281. return ui_region_get_width(region) >= 0 && ui_region_get_height(region) >= 0;
  282. }
  283. /**
  284. * @brief Check if an region is empty
  285. *
  286. * An empty region has a zero width or height, or is invalid
  287. *
  288. * @param region pointer to an region.
  289. *
  290. * @return true if region is empty
  291. */
  292. bool ui_region_is_empty(const ui_region_t *region)
  293. {
  294. return ui_region_get_width(region) <= 0 || ui_region_get_height(region) <= 0;
  295. }