1 /**
2 * @file lv_area.c
3 *
4 */
5
6 /*********************
7 * INCLUDES
8 *********************/
9 #include "../lv_conf_internal.h"
10
11 #include "lv_area.h"
12 #include "lv_math.h"
13
14 /*********************
15 * DEFINES
16 *********************/
17
18 /**********************
19 * TYPEDEFS
20 **********************/
21
22 /**********************
23 * STATIC PROTOTYPES
24 **********************/
25
26 static bool lv_point_within_circle(const lv_area_t * area, const lv_point_t * p);
27
28 /**********************
29 * STATIC VARIABLES
30 **********************/
31
32 /**********************
33 * MACROS
34 **********************/
35
36 /**********************
37 * GLOBAL FUNCTIONS
38 **********************/
39
40 /**
41 * Initialize an area
42 * @param area_p pointer to an area
43 * @param x1 left coordinate of the area
44 * @param y1 top coordinate of the area
45 * @param x2 right coordinate of the area
46 * @param y2 bottom coordinate of the area
47 */
lv_area_set(lv_area_t * area_p,lv_coord_t x1,lv_coord_t y1,lv_coord_t x2,lv_coord_t y2)48 void lv_area_set(lv_area_t * area_p, lv_coord_t x1, lv_coord_t y1, lv_coord_t x2, lv_coord_t y2)
49 {
50 area_p->x1 = x1;
51 area_p->y1 = y1;
52 area_p->x2 = x2;
53 area_p->y2 = y2;
54 }
55
56 /**
57 * Set the width of an area
58 * @param area_p pointer to an area
59 * @param w the new width of the area (w == 1 makes x1 == x2)
60 */
lv_area_set_width(lv_area_t * area_p,lv_coord_t w)61 void lv_area_set_width(lv_area_t * area_p, lv_coord_t w)
62 {
63 area_p->x2 = area_p->x1 + w - 1;
64 }
65
66 /**
67 * Set the height of an area
68 * @param area_p pointer to an area
69 * @param h the new height of the area (h == 1 makes y1 == y2)
70 */
lv_area_set_height(lv_area_t * area_p,lv_coord_t h)71 void lv_area_set_height(lv_area_t * area_p, lv_coord_t h)
72 {
73 area_p->y2 = area_p->y1 + h - 1;
74 }
75
76 /**
77 * Set the position of an area (width and height will be kept)
78 * @param area_p pointer to an area
79 * @param x the new x coordinate of the area
80 * @param y the new y coordinate of the area
81 */
_lv_area_set_pos(lv_area_t * area_p,lv_coord_t x,lv_coord_t y)82 void _lv_area_set_pos(lv_area_t * area_p, lv_coord_t x, lv_coord_t y)
83 {
84 lv_coord_t w = lv_area_get_width(area_p);
85 lv_coord_t h = lv_area_get_height(area_p);
86 area_p->x1 = x;
87 area_p->y1 = y;
88 lv_area_set_width(area_p, w);
89 lv_area_set_height(area_p, h);
90 }
91
92 /**
93 * Return with area of an area (x * y)
94 * @param area_p pointer to an area
95 * @return size of area
96 */
lv_area_get_size(const lv_area_t * area_p)97 uint32_t lv_area_get_size(const lv_area_t * area_p)
98 {
99 uint32_t size;
100
101 size = (uint32_t)(area_p->x2 - area_p->x1 + 1) * (area_p->y2 - area_p->y1 + 1);
102
103 return size;
104 }
105
lv_area_increase(lv_area_t * area,lv_coord_t w_extra,lv_coord_t h_extra)106 void lv_area_increase(lv_area_t * area, lv_coord_t w_extra, lv_coord_t h_extra)
107 {
108 area->x1 -= w_extra;
109 area->x2 += w_extra;
110 area->y1 -= h_extra;
111 area->y2 += h_extra;
112 }
113
lv_area_move(lv_area_t * area,lv_coord_t x_ofs,lv_coord_t y_ofs)114 void lv_area_move(lv_area_t * area, lv_coord_t x_ofs, lv_coord_t y_ofs)
115 {
116 area->x1 += x_ofs;
117 area->x2 += x_ofs;
118 area->y1 += y_ofs;
119 area->y2 += y_ofs;
120 }
121
122 /**
123 * Get the common parts of two areas
124 * @param res_p pointer to an area, the result will be stored here
125 * @param a1_p pointer to the first area
126 * @param a2_p pointer to the second area
127 * @return false: the two area has NO common parts, res_p is invalid
128 */
_lv_area_intersect(lv_area_t * res_p,const lv_area_t * a1_p,const lv_area_t * a2_p)129 bool _lv_area_intersect(lv_area_t * res_p, const lv_area_t * a1_p, const lv_area_t * a2_p)
130 {
131 /*Get the smaller area from 'a1_p' and 'a2_p'*/
132 res_p->x1 = LV_MAX(a1_p->x1, a2_p->x1);
133 res_p->y1 = LV_MAX(a1_p->y1, a2_p->y1);
134 res_p->x2 = LV_MIN(a1_p->x2, a2_p->x2);
135 res_p->y2 = LV_MIN(a1_p->y2, a2_p->y2);
136
137 /*If x1 or y1 greater than x2 or y2 then the areas union is empty*/
138 bool union_ok = true;
139 if((res_p->x1 > res_p->x2) || (res_p->y1 > res_p->y2)) {
140 union_ok = false;
141 }
142
143 return union_ok;
144 }
145
146 /**
147 * Get resulting sub areas after removing the common parts of two areas from the first area
148 * @param res_p pointer to an array of areas with a count of 4, the resulting areas will be stored here
149 * @param a1_p pointer to the first area
150 * @param a2_p pointer to the second area
151 * @return number of results or -1 if no intersect
152 */
_lv_area_diff(lv_area_t * res_p,const lv_area_t * a1_p,const lv_area_t * a2_p)153 int8_t _lv_area_diff(lv_area_t * res_p, const lv_area_t * a1_p, const lv_area_t * a2_p)
154 {
155 /*Areas have no common parts*/
156 if(!_lv_area_is_on(a1_p, a2_p)) return -1;
157
158 /*No remaining areas after removing common parts*/
159 if(_lv_area_is_in(a1_p, a2_p, 0)) return 0;
160
161 /*Result counter*/
162 int8_t res_c = 0;
163
164 /*Get required information*/
165 lv_area_t n;
166 lv_coord_t a1_w = lv_area_get_width(a1_p) - 1;
167 lv_coord_t a1_h = lv_area_get_height(a1_p) - 1;
168
169 /*Compute top rectangle*/
170 lv_coord_t th = a2_p->y1 - a1_p->y1;
171 if(th > 0) {
172 n.x1 = a1_p->x1;
173 n.y1 = a1_p->y1;
174 n.x2 = a1_p->x2;
175 n.y2 = a1_p->y1 + th;
176 res_p[res_c++] = n;
177 }
178
179 /*Compute the bottom rectangle*/
180 lv_coord_t bh = a1_h - (a2_p->y2 - a1_p->y1);
181 if(bh > 0 && a2_p->y2 < a1_p->y2) {
182 n.x1 = a1_p->x1;
183 n.y1 = a2_p->y2;
184 n.x2 = a1_p->x2;
185 n.y2 = a2_p->y2 + bh;
186 res_p[res_c++] = n;
187 }
188
189 /*Compute side height*/
190 lv_coord_t y1 = a2_p->y1 > a1_p->y1 ? a2_p->y1 : a1_p->y1;
191 lv_coord_t y2 = a2_p->y2 < a1_p->y2 ? a2_p->y2 : a1_p->y2;
192 lv_coord_t sh = y2 - y1;
193
194 /*Compute the left rectangle*/
195 lv_coord_t lw = a2_p->x1 - a1_p->x1;
196 if(lw > 0 && sh > 0) {
197 n.x1 = a1_p->x1;
198 n.y1 = y1;
199 n.x2 = a1_p->x1 + lw;
200 n.y2 = y1 + sh;
201 res_p[res_c++] = n;
202 }
203
204 /*Compute the right rectangle*/
205 lv_coord_t rw = a1_w - (a2_p->x2 - a1_p->x1);
206 if(rw > 0) {
207 n.x1 = a2_p->x2;
208 n.y1 = y1;
209 n.x2 = a2_p->x2 + rw;
210 n.y2 = y1 + sh;
211 res_p[res_c++] = n;
212 }
213
214 //Return number of results
215 return res_c;
216 }
217
218 /**
219 * Join two areas into a third which involves the other two
220 * @param res_p pointer to an area, the result will be stored here
221 * @param a1_p pointer to the first area
222 * @param a2_p pointer to the second area
223 */
_lv_area_join(lv_area_t * a_res_p,const lv_area_t * a1_p,const lv_area_t * a2_p)224 void _lv_area_join(lv_area_t * a_res_p, const lv_area_t * a1_p, const lv_area_t * a2_p)
225 {
226 a_res_p->x1 = LV_MIN(a1_p->x1, a2_p->x1);
227 a_res_p->y1 = LV_MIN(a1_p->y1, a2_p->y1);
228 a_res_p->x2 = LV_MAX(a1_p->x2, a2_p->x2);
229 a_res_p->y2 = LV_MAX(a1_p->y2, a2_p->y2);
230 }
231
232 /**
233 * Check if a point is on an area
234 * @param a_p pointer to an area
235 * @param p_p pointer to a point
236 * @param radius radius of area (e.g. for rounded rectangle)
237 * @return false:the point is out of the area
238 */
_lv_area_is_point_on(const lv_area_t * a_p,const lv_point_t * p_p,lv_coord_t radius)239 bool _lv_area_is_point_on(const lv_area_t * a_p, const lv_point_t * p_p, lv_coord_t radius)
240 {
241 /*First check the basic area*/
242 bool is_on_rect = false;
243 if((p_p->x >= a_p->x1 && p_p->x <= a_p->x2) && ((p_p->y >= a_p->y1 && p_p->y <= a_p->y2))) {
244 is_on_rect = true;
245 }
246 if(!is_on_rect)
247 return false;
248 /*Now handle potential rounded rectangles*/
249 if(radius <= 0) {
250 /*No radius, it is within the rectangle*/
251 return true;
252 }
253 lv_coord_t w = lv_area_get_width(a_p) / 2;
254 lv_coord_t h = lv_area_get_height(a_p) / 2;
255 lv_coord_t max_radius = LV_MIN(w, h);
256 if(radius > max_radius)
257 radius = max_radius;
258
259 /*Check if it's in one of the corners*/
260 lv_area_t corner_area;
261 /*Top left*/
262 corner_area.x1 = a_p->x1;
263 corner_area.x2 = a_p->x1 + radius;
264 corner_area.y1 = a_p->y1;
265 corner_area.y2 = a_p->y1 + radius;
266 if(_lv_area_is_point_on(&corner_area, p_p, 0)) {
267 corner_area.x2 += radius;
268 corner_area.y2 += radius;
269 return lv_point_within_circle(&corner_area, p_p);
270 }
271 /*Bottom left*/
272 corner_area.y1 = a_p->y2 - radius;
273 corner_area.y2 = a_p->y2;
274 if(_lv_area_is_point_on(&corner_area, p_p, 0)) {
275 corner_area.x2 += radius;
276 corner_area.y1 -= radius;
277 return lv_point_within_circle(&corner_area, p_p);
278 }
279 /*Bottom right*/
280 corner_area.x1 = a_p->x2 - radius;
281 corner_area.x2 = a_p->x2;
282 if(_lv_area_is_point_on(&corner_area, p_p, 0)) {
283 corner_area.x1 -= radius;
284 corner_area.y1 -= radius;
285 return lv_point_within_circle(&corner_area, p_p);
286 }
287 /*Top right*/
288 corner_area.y1 = a_p->y1;
289 corner_area.y2 = a_p->y1 + radius;
290 if(_lv_area_is_point_on(&corner_area, p_p, 0)) {
291 corner_area.x1 -= radius;
292 corner_area.y2 += radius;
293 return lv_point_within_circle(&corner_area, p_p);
294 }
295 /*Not within corners*/
296 return true;
297 }
298
299 /**
300 * Check if two area has common parts
301 * @param a1_p pointer to an area.
302 * @param a2_p pointer to an other area
303 * @return false: a1_p and a2_p has no common parts
304 */
_lv_area_is_on(const lv_area_t * a1_p,const lv_area_t * a2_p)305 bool _lv_area_is_on(const lv_area_t * a1_p, const lv_area_t * a2_p)
306 {
307 if((a1_p->x1 <= a2_p->x2) && (a1_p->x2 >= a2_p->x1) && (a1_p->y1 <= a2_p->y2) && (a1_p->y2 >= a2_p->y1)) {
308 return true;
309 }
310 else {
311 return false;
312 }
313 }
314
315 /**
316 * Check if an area is fully on an other
317 * @param ain_p pointer to an area which could be in 'aholder_p'
318 * @param aholder_p pointer to an area which could involve 'ain_p'
319 * @param radius radius of `aholder_p` (e.g. for rounded rectangle)
320 * @return true: `ain_p` is fully inside `aholder_p`
321 */
_lv_area_is_in(const lv_area_t * ain_p,const lv_area_t * aholder_p,lv_coord_t radius)322 bool _lv_area_is_in(const lv_area_t * ain_p, const lv_area_t * aholder_p, lv_coord_t radius)
323 {
324 bool is_in = false;
325
326 if(ain_p->x1 >= aholder_p->x1 && ain_p->y1 >= aholder_p->y1 && ain_p->x2 <= aholder_p->x2 &&
327 ain_p->y2 <= aholder_p->y2) {
328 is_in = true;
329 }
330
331 if(!is_in) return false;
332 if(radius == 0) return true;
333
334 /*Check if the corner points are inside the radius or not*/
335 lv_point_t p;
336
337 p.x = ain_p->x1;
338 p.y = ain_p->y1;
339 if(_lv_area_is_point_on(aholder_p, &p, radius) == false) return false;
340
341 p.x = ain_p->x2;
342 p.y = ain_p->y1;
343 if(_lv_area_is_point_on(aholder_p, &p, radius) == false) return false;
344
345 p.x = ain_p->x1;
346 p.y = ain_p->y2;
347 if(_lv_area_is_point_on(aholder_p, &p, radius) == false) return false;
348
349 p.x = ain_p->x2;
350 p.y = ain_p->y2;
351 if(_lv_area_is_point_on(aholder_p, &p, radius) == false) return false;
352
353 return true;
354 }
355
356 /**
357 * Check if an area is fully out of an other
358 * @param aout_p pointer to an area which could be in 'aholder_p'
359 * @param aholder_p pointer to an area which could involve 'ain_p'
360 * @param radius radius of `aholder_p` (e.g. for rounded rectangle)
361 * @return true: `aout_p` is fully outside `aholder_p`
362 */
_lv_area_is_out(const lv_area_t * aout_p,const lv_area_t * aholder_p,lv_coord_t radius)363 bool _lv_area_is_out(const lv_area_t * aout_p, const lv_area_t * aholder_p, lv_coord_t radius)
364 {
365 if(aout_p->x2 < aholder_p->x1 || aout_p->y2 < aholder_p->y1 || aout_p->x1 > aholder_p->x2 ||
366 aout_p->y1 > aholder_p->y2) {
367 return true;
368 }
369
370 if(radius == 0) return false;
371
372 /*Check if the corner points are outside the radius or not*/
373 lv_point_t p;
374
375 p.x = aout_p->x1;
376 p.y = aout_p->y1;
377 if(_lv_area_is_point_on(aholder_p, &p, radius)) return false;
378
379 p.x = aout_p->x2;
380 p.y = aout_p->y1;
381 if(_lv_area_is_point_on(aholder_p, &p, radius)) return false;
382
383 p.x = aout_p->x1;
384 p.y = aout_p->y2;
385 if(_lv_area_is_point_on(aholder_p, &p, radius)) return false;
386
387 p.x = aout_p->x2;
388 p.y = aout_p->y2;
389 if(_lv_area_is_point_on(aholder_p, &p, radius)) return false;
390
391 return true;
392 }
393
_lv_area_is_equal(const lv_area_t * a,const lv_area_t * b)394 bool _lv_area_is_equal(const lv_area_t * a, const lv_area_t * b)
395 {
396 return a->x1 == b->x1 && a->x2 == b->x2 && a->y1 == b->y1 && a->y2 == b->y2;
397 }
398
399 /**
400 * Align an area to an other
401 * @param base an are where the other will be aligned
402 * @param to_align the area to align
403 * @param align `LV_ALIGN_...`
404 * @param res x/y coordinates where `to_align` align area should be placed
405 */
lv_area_align(const lv_area_t * base,lv_area_t * to_align,lv_align_t align,lv_coord_t ofs_x,lv_coord_t ofs_y)406 void lv_area_align(const lv_area_t * base, lv_area_t * to_align, lv_align_t align, lv_coord_t ofs_x, lv_coord_t ofs_y)
407 {
408
409 lv_coord_t x;
410 lv_coord_t y;
411 switch(align) {
412 case LV_ALIGN_CENTER:
413 x = lv_area_get_width(base) / 2 - lv_area_get_width(to_align) / 2;
414 y = lv_area_get_height(base) / 2 - lv_area_get_height(to_align) / 2;
415 break;
416
417 case LV_ALIGN_TOP_LEFT:
418 x = 0;
419 y = 0;
420 break;
421 case LV_ALIGN_TOP_MID:
422 x = lv_area_get_width(base) / 2 - lv_area_get_width(to_align) / 2;
423 y = 0;
424 break;
425
426 case LV_ALIGN_TOP_RIGHT:
427 x = lv_area_get_width(base) - lv_area_get_width(to_align);
428 y = 0;
429 break;
430
431 case LV_ALIGN_BOTTOM_LEFT:
432 x = 0;
433 y = lv_area_get_height(base) - lv_area_get_height(to_align);
434 break;
435 case LV_ALIGN_BOTTOM_MID:
436 x = lv_area_get_width(base) / 2 - lv_area_get_width(to_align) / 2;
437 y = lv_area_get_height(base) - lv_area_get_height(to_align);
438 break;
439
440 case LV_ALIGN_BOTTOM_RIGHT:
441 x = lv_area_get_width(base) - lv_area_get_width(to_align);
442 y = lv_area_get_height(base) - lv_area_get_height(to_align);
443 break;
444
445 case LV_ALIGN_LEFT_MID:
446 x = 0;
447 y = lv_area_get_height(base) / 2 - lv_area_get_height(to_align) / 2;
448 break;
449
450 case LV_ALIGN_RIGHT_MID:
451 x = lv_area_get_width(base) - lv_area_get_width(to_align);
452 y = lv_area_get_height(base) / 2 - lv_area_get_height(to_align) / 2;
453 break;
454
455 case LV_ALIGN_OUT_TOP_LEFT:
456 x = 0;
457 y = -lv_area_get_height(to_align);
458 break;
459
460 case LV_ALIGN_OUT_TOP_MID:
461 x = lv_area_get_width(base) / 2 - lv_area_get_width(to_align) / 2;
462 y = -lv_area_get_height(to_align);
463 break;
464
465 case LV_ALIGN_OUT_TOP_RIGHT:
466 x = lv_area_get_width(base) - lv_area_get_width(to_align);
467 y = -lv_area_get_height(to_align);
468 break;
469
470 case LV_ALIGN_OUT_BOTTOM_LEFT:
471 x = 0;
472 y = lv_area_get_height(base);
473 break;
474
475 case LV_ALIGN_OUT_BOTTOM_MID:
476 x = lv_area_get_width(base) / 2 - lv_area_get_width(to_align) / 2;
477 y = lv_area_get_height(base);
478 break;
479
480 case LV_ALIGN_OUT_BOTTOM_RIGHT:
481 x = lv_area_get_width(base) - lv_area_get_width(to_align);
482 y = lv_area_get_height(base);
483 break;
484
485 case LV_ALIGN_OUT_LEFT_TOP:
486 x = -lv_area_get_width(to_align);
487 y = 0;
488 break;
489
490 case LV_ALIGN_OUT_LEFT_MID:
491 x = -lv_area_get_width(to_align);
492 y = lv_area_get_height(base) / 2 - lv_area_get_height(to_align) / 2;
493 break;
494
495 case LV_ALIGN_OUT_LEFT_BOTTOM:
496 x = -lv_area_get_width(to_align);
497 y = lv_area_get_height(base) - lv_area_get_height(to_align);
498 break;
499
500 case LV_ALIGN_OUT_RIGHT_TOP:
501 x = lv_area_get_width(base);
502 y = 0;
503 break;
504
505 case LV_ALIGN_OUT_RIGHT_MID:
506 x = lv_area_get_width(base);
507 y = lv_area_get_height(base) / 2 - lv_area_get_height(to_align) / 2;
508 break;
509
510 case LV_ALIGN_OUT_RIGHT_BOTTOM:
511 x = lv_area_get_width(base);
512 y = lv_area_get_height(base) - lv_area_get_height(to_align);
513 break;
514 default:
515 x = 0;
516 y = 0;
517 break;
518 }
519
520 x += base->x1;
521 y += base->y1;
522
523 lv_coord_t w = lv_area_get_width(to_align);
524 lv_coord_t h = lv_area_get_height(to_align);
525 to_align->x1 = x + ofs_x;
526 to_align->y1 = y + ofs_y;
527 to_align->x2 = to_align->x1 + w - 1;
528 to_align->y2 = to_align->y1 + h - 1;
529 }
530
531 #define _LV_TRANSFORM_TRIGO_SHIFT 10
lv_point_transform(lv_point_t * p,int32_t angle,int32_t zoom,const lv_point_t * pivot)532 void lv_point_transform(lv_point_t * p, int32_t angle, int32_t zoom, const lv_point_t * pivot)
533 {
534 if(angle == 0 && zoom == 256) {
535 return;
536 }
537
538 p->x -= pivot->x;
539 p->y -= pivot->y;
540
541 if(angle == 0) {
542 p->x = (((int32_t)(p->x) * zoom) >> 8) + pivot->x;
543 p->y = (((int32_t)(p->y) * zoom) >> 8) + pivot->y;
544 return;
545 }
546
547 static int32_t angle_prev = INT32_MIN;
548 static int32_t sinma;
549 static int32_t cosma;
550 if(angle_prev != angle) {
551 int32_t angle_limited = angle;
552 if(angle_limited > 3600) angle_limited -= 3600;
553 if(angle_limited < 0) angle_limited += 3600;
554
555 int32_t angle_low = angle_limited / 10;
556 int32_t angle_high = angle_low + 1;
557 int32_t angle_rem = angle_limited - (angle_low * 10);
558
559 int32_t s1 = lv_trigo_sin(angle_low);
560 int32_t s2 = lv_trigo_sin(angle_high);
561
562 int32_t c1 = lv_trigo_sin(angle_low + 90);
563 int32_t c2 = lv_trigo_sin(angle_high + 90);
564
565 sinma = (s1 * (10 - angle_rem) + s2 * angle_rem) / 10;
566 cosma = (c1 * (10 - angle_rem) + c2 * angle_rem) / 10;
567 sinma = sinma >> (LV_TRIGO_SHIFT - _LV_TRANSFORM_TRIGO_SHIFT);
568 cosma = cosma >> (LV_TRIGO_SHIFT - _LV_TRANSFORM_TRIGO_SHIFT);
569 angle_prev = angle;
570 }
571 int32_t x = p->x;
572 int32_t y = p->y;
573 if(zoom == 256) {
574 p->x = ((cosma * x - sinma * y) >> _LV_TRANSFORM_TRIGO_SHIFT) + pivot->x;
575 p->y = ((sinma * x + cosma * y) >> _LV_TRANSFORM_TRIGO_SHIFT) + pivot->y;
576 }
577 else {
578 p->x = (((cosma * x - sinma * y) * zoom) >> (_LV_TRANSFORM_TRIGO_SHIFT + 8)) + pivot->x;
579 p->y = (((sinma * x + cosma * y) * zoom) >> (_LV_TRANSFORM_TRIGO_SHIFT + 8)) + pivot->y;
580 }
581 }
582
583 /**********************
584 * STATIC FUNCTIONS
585 **********************/
586
lv_point_within_circle(const lv_area_t * area,const lv_point_t * p)587 static bool lv_point_within_circle(const lv_area_t * area, const lv_point_t * p)
588 {
589 lv_coord_t r = (area->x2 - area->x1) / 2;
590
591 /*Circle center*/
592 lv_coord_t cx = area->x1 + r;
593 lv_coord_t cy = area->y1 + r;
594
595 /*Simplify the code by moving everything to (0, 0)*/
596 lv_coord_t px = p->x - cx;
597 lv_coord_t py = p->y - cy;
598
599 uint32_t r_sqrd = r * r;
600 uint32_t dist = (px * px) + (py * py);
601
602 if(dist <= r_sqrd)
603 return true;
604 else
605 return false;
606 }
607