2018-06-07 15:32:19 +02:00
|
|
|
/**
|
|
|
|
* @file lv_draw_triangle.c
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*********************
|
|
|
|
* INCLUDES
|
|
|
|
*********************/
|
|
|
|
#include "lv_draw_triangle.h"
|
|
|
|
#include "../lv_misc/lv_math.h"
|
|
|
|
|
|
|
|
/*********************
|
|
|
|
* DEFINES
|
|
|
|
*********************/
|
2019-05-31 07:51:18 +02:00
|
|
|
|
2018-06-07 15:32:19 +02:00
|
|
|
/**********************
|
|
|
|
* TYPEDEFS
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
/**********************
|
|
|
|
* STATIC PROTOTYPES
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
/**********************
|
|
|
|
* STATIC VARIABLES
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
/**********************
|
|
|
|
* MACROS
|
|
|
|
**********************/
|
|
|
|
|
|
|
|
/**********************
|
|
|
|
* GLOBAL FUNCTIONS
|
|
|
|
**********************/
|
2019-03-18 16:16:52 +01:00
|
|
|
|
2018-06-07 15:32:19 +02:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param points pointer to an array with 3 points
|
2019-10-16 20:38:45 +02:00
|
|
|
* @param clip the triangle will be drawn only in this mask
|
2019-03-18 16:16:52 +01:00
|
|
|
* @param style style for of the triangle
|
2019-06-02 20:35:33 +02:00
|
|
|
* @param opa_scale scale down all opacities by the factor (0..255)
|
2018-06-07 15:32:19 +02:00
|
|
|
*/
|
2019-10-16 20:38:45 +02:00
|
|
|
void lv_draw_triangle(const lv_point_t * points, const lv_area_t * clip, const lv_style_t * style, lv_opa_t opa_scale)
|
2019-04-17 15:14:09 +02:00
|
|
|
{
|
2019-10-16 20:38:45 +02:00
|
|
|
|
|
|
|
lv_area_t clip_tri;
|
|
|
|
clip_tri.x1 = LV_MATH_MIN(LV_MATH_MIN(points[0].x, points[1].x), points[2].x);
|
|
|
|
clip_tri.x2 = LV_MATH_MAX(LV_MATH_MAX(points[0].x, points[1].x), points[2].x);
|
|
|
|
clip_tri.y1 = LV_MATH_MIN(LV_MATH_MIN(points[0].y, points[1].y), points[2].y);
|
|
|
|
clip_tri.y2 = LV_MATH_MAX(LV_MATH_MAX(points[0].y, points[1].y), points[2].y);
|
|
|
|
|
|
|
|
bool is_common;
|
|
|
|
is_common = lv_area_intersect(&clip_tri, &clip_tri, clip);
|
|
|
|
if(!is_common) return;
|
|
|
|
|
2019-10-10 15:36:53 +02:00
|
|
|
lv_draw_mask_line_side_t side_right;
|
|
|
|
lv_draw_mask_line_side_t side_left;
|
|
|
|
lv_draw_mask_line_side_t side_bottom;
|
|
|
|
|
|
|
|
/*Find the lowest point*/
|
|
|
|
lv_coord_t y_min = points[0].y;
|
|
|
|
int8_t y_min_i = 0;
|
|
|
|
int8_t i;
|
|
|
|
|
|
|
|
for(i = 1; i < 3; i++) {
|
|
|
|
if(points[i].y < y_min) {
|
|
|
|
y_min = points[i].y;
|
|
|
|
y_min_i = i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int8_t i_left;
|
|
|
|
int8_t i_right;
|
|
|
|
|
|
|
|
switch(y_min_i) {
|
|
|
|
case 0:
|
|
|
|
i_left = 1;
|
|
|
|
i_right = 2;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
i_left = 0;
|
|
|
|
i_right = 2;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
i_left = 0;
|
|
|
|
i_right = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(points[i_right].x < points[y_min_i].x) {
|
|
|
|
int8_t tmp;
|
|
|
|
tmp = i_right;
|
|
|
|
i_right = i_left;
|
|
|
|
i_left = tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(points[y_min_i].y == points[i_right].y) side_right = LV_DRAW_MASK_LINE_SIDE_BOTTOM;
|
|
|
|
else side_right = LV_DRAW_MASK_LINE_SIDE_LEFT;
|
|
|
|
|
|
|
|
if(points[y_min_i].y == points[i_left].y) side_left = LV_DRAW_MASK_LINE_SIDE_BOTTOM;
|
|
|
|
else side_left = LV_DRAW_MASK_LINE_SIDE_RIGHT;
|
|
|
|
|
|
|
|
side_bottom = LV_DRAW_MASK_LINE_SIDE_TOP;
|
|
|
|
|
|
|
|
lv_draw_mask_param_t p_right;
|
|
|
|
lv_draw_mask_param_t p_left;
|
|
|
|
lv_draw_mask_param_t p_bottom;
|
|
|
|
|
|
|
|
lv_draw_mask_line_points_init(&p_right, points[y_min_i].x, points[y_min_i].y,
|
|
|
|
points[i_right].x, points[i_right].y, side_right);
|
|
|
|
|
|
|
|
lv_draw_mask_line_points_init(&p_left, points[y_min_i].x, points[y_min_i].y,
|
|
|
|
points[i_left].x, points[i_left].y, side_left);
|
|
|
|
|
|
|
|
lv_draw_mask_line_points_init(&p_bottom, points[i_left].x, points[i_left].y,
|
|
|
|
points[i_right].x, points[i_right].y, side_bottom);
|
|
|
|
|
|
|
|
|
|
|
|
int16_t id_right = lv_draw_mask_add(&p_right, NULL);
|
|
|
|
int16_t id_left = lv_draw_mask_add(&p_left, NULL);
|
|
|
|
int16_t id_bottom = lv_draw_mask_add(&p_bottom, NULL);
|
|
|
|
|
|
|
|
|
2019-10-16 20:38:45 +02:00
|
|
|
lv_draw_rect(&clip_tri, &clip_tri, style, opa_scale);
|
2019-10-10 15:36:53 +02:00
|
|
|
|
|
|
|
lv_draw_mask_remove_id(id_right);
|
|
|
|
lv_draw_mask_remove_id(id_left);
|
|
|
|
lv_draw_mask_remove_id(id_bottom);
|
2019-06-04 15:20:22 +02:00
|
|
|
|
2019-04-17 15:14:09 +02:00
|
|
|
}
|
|
|
|
|
2019-06-02 20:35:33 +02:00
|
|
|
/**
|
|
|
|
* Draw a polygon from triangles. Only convex polygons are supported
|
|
|
|
* @param points an array of points
|
|
|
|
* @param point_cnt number of points
|
|
|
|
* @param mask polygon will be drawn only in this mask
|
|
|
|
* @param style style of the polygon
|
|
|
|
* @param opa_scale scale down all opacities by the factor (0..255)
|
|
|
|
*/
|
|
|
|
void lv_draw_polygon(const lv_point_t * points, uint32_t point_cnt, const lv_area_t * mask, const lv_style_t * style,
|
2019-06-06 06:05:40 +02:00
|
|
|
lv_opa_t opa_scale)
|
2019-06-02 20:35:33 +02:00
|
|
|
{
|
2019-10-16 20:38:45 +02:00
|
|
|
if(point_cnt < 3) return;
|
|
|
|
if(points == NULL) return;
|
|
|
|
|
|
|
|
uint32_t i;
|
|
|
|
lv_point_t tri[3];
|
|
|
|
tri[0].x = points[0].x;
|
|
|
|
tri[0].y = points[0].y;
|
|
|
|
for(i = 0; i < point_cnt - 1; i++) {
|
|
|
|
tri[1].x = points[i].x;
|
|
|
|
tri[1].y = points[i].y;
|
|
|
|
tri[2].x = points[i + 1].x;
|
|
|
|
tri[2].y = points[i + 1].y;
|
|
|
|
lv_draw_triangle(tri, mask, style, opa_scale);
|
|
|
|
}
|
2019-06-02 20:35:33 +02:00
|
|
|
}
|
|
|
|
|
2019-04-17 15:14:09 +02:00
|
|
|
/**********************
|
|
|
|
* STATIC FUNCTIONS
|
|
|
|
**********************/
|