2020-05-02 02:21:35 -07:00
|
|
|
#include "shared-module/vectorio/__init__.h"
|
|
|
|
#include "shared-bindings/vectorio/Polygon.h"
|
|
|
|
#include "shared-module/displayio/area.h"
|
|
|
|
|
|
|
|
#include "py/runtime.h"
|
2020-05-12 21:55:18 -07:00
|
|
|
#include "py/gc.h"
|
|
|
|
|
2020-05-02 02:21:35 -07:00
|
|
|
#include "stdlib.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
|
|
|
|
#define VECTORIO_POLYGON_DEBUG(...) (void)0
|
2021-08-07 17:47:57 -07:00
|
|
|
// #define VECTORIO_POLYGON_DEBUG(...) mp_printf(&mp_plat_print, __VA_ARGS__)
|
2020-05-02 02:21:35 -07:00
|
|
|
|
|
|
|
|
2020-05-12 21:55:18 -07:00
|
|
|
// Converts a list of points tuples to a flat list of ints for speedier internal use.
|
|
|
|
// Also validates the points.
|
|
|
|
static void _clobber_points_list(vectorio_polygon_t *self, mp_obj_t points_tuple_list) {
|
|
|
|
size_t len = 0;
|
|
|
|
mp_obj_t *items;
|
|
|
|
mp_obj_list_get(points_tuple_list, &len, &items);
|
2020-07-19 12:27:35 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG(" self.len: %d, len: %d, ", self->len, len);
|
2020-05-12 21:55:18 -07:00
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
if (len < 3) {
|
2021-09-10 09:34:20 +01:00
|
|
|
mp_raise_TypeError(translate("Polygon needs at least 3 points"));
|
2020-05-12 21:55:18 -07:00
|
|
|
}
|
|
|
|
|
2021-03-15 19:27:36 +05:30
|
|
|
if (self->len < 2 * len) {
|
|
|
|
if (self->points_list != NULL) {
|
2020-07-19 12:27:35 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG("free(%d), ", sizeof(self->points_list));
|
2021-03-15 19:27:36 +05:30
|
|
|
gc_free(self->points_list);
|
2020-05-12 21:55:18 -07:00
|
|
|
}
|
2021-08-07 17:47:57 -07:00
|
|
|
self->points_list = gc_alloc(2 * len * sizeof(uint16_t), false, false);
|
|
|
|
VECTORIO_POLYGON_DEBUG("alloc(%p, %d)", self->points_list, 2 * len * sizeof(uint16_t));
|
2020-05-12 21:55:18 -07:00
|
|
|
}
|
2021-03-15 19:27:36 +05:30
|
|
|
self->len = 2 * len;
|
2020-05-12 21:55:18 -07:00
|
|
|
|
2021-08-07 17:47:57 -07:00
|
|
|
for (uint16_t i = 0; i < len; ++i) {
|
2020-05-12 21:55:18 -07:00
|
|
|
size_t tuple_len = 0;
|
|
|
|
mp_obj_t *tuple_items;
|
|
|
|
mp_obj_tuple_get(items[i], &tuple_len, &tuple_items);
|
|
|
|
|
|
|
|
if (tuple_len != 2) {
|
|
|
|
mp_raise_ValueError_varg(translate("%q must be a tuple of length 2"), MP_QSTR_point);
|
|
|
|
}
|
2021-08-07 17:47:57 -07:00
|
|
|
mp_int_t x;
|
|
|
|
mp_int_t y;
|
|
|
|
if (!mp_obj_get_int_maybe(tuple_items[ 0 ], &x)
|
|
|
|
|| !mp_obj_get_int_maybe(tuple_items[ 1 ], &y)
|
|
|
|
|| x < SHRT_MIN || x > SHRT_MAX || y < SHRT_MIN || y > SHRT_MAX
|
2021-03-15 19:27:36 +05:30
|
|
|
) {
|
|
|
|
gc_free(self->points_list);
|
2020-05-12 21:55:18 -07:00
|
|
|
self->points_list = NULL;
|
|
|
|
mp_raise_ValueError_varg(translate("unsupported %q type"), MP_QSTR_point);
|
2021-08-07 17:47:57 -07:00
|
|
|
self->len = 0;
|
2020-05-12 21:55:18 -07:00
|
|
|
}
|
2021-08-07 17:47:57 -07:00
|
|
|
self->points_list[2 * i ] = (int16_t)x;
|
|
|
|
self->points_list[2 * i + 1] = (int16_t)y;
|
2020-05-12 21:55:18 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2020-05-02 02:21:35 -07:00
|
|
|
void common_hal_vectorio_polygon_construct(vectorio_polygon_t *self, mp_obj_t points_list) {
|
2020-07-19 12:27:35 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG("%p polygon_construct: ", self);
|
2020-05-12 21:55:18 -07:00
|
|
|
self->points_list = NULL;
|
|
|
|
self->len = 0;
|
2020-05-02 02:21:35 -07:00
|
|
|
self->on_dirty.obj = NULL;
|
2021-03-15 19:27:36 +05:30
|
|
|
_clobber_points_list(self, points_list);
|
2020-07-19 12:27:35 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG("\n");
|
2020-05-02 02:21:35 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
mp_obj_t common_hal_vectorio_polygon_get_points(vectorio_polygon_t *self) {
|
2020-07-19 12:27:35 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG("%p common_hal_vectorio_polygon_get_points {len: %d, points_list: %p}\n", self, self->len, self->points_list);
|
2021-08-07 17:47:57 -07:00
|
|
|
mp_obj_list_t *list = MP_OBJ_TO_PTR(mp_obj_new_list(0, NULL));
|
|
|
|
|
|
|
|
VECTORIO_POLYGON_DEBUG(" >points\n");
|
|
|
|
for (uint16_t i = 0; i < self->len; i += 2) {
|
|
|
|
VECTORIO_POLYGON_DEBUG(" (%4d, %4d)\n", self->points_list[i], self->points_list[i + 1]);
|
|
|
|
|
|
|
|
mp_obj_tuple_t *pair = MP_OBJ_TO_PTR(mp_obj_new_tuple(2, NULL));
|
2021-08-07 19:40:07 -07:00
|
|
|
pair->items[0] = mp_obj_new_int((mp_int_t)self->points_list[i ]);
|
|
|
|
pair->items[1] = mp_obj_new_int((mp_int_t)self->points_list[i + 1]);
|
2020-07-19 12:27:35 -07:00
|
|
|
|
|
|
|
mp_obj_list_append(
|
|
|
|
list,
|
2021-08-07 17:47:57 -07:00
|
|
|
pair
|
2021-03-15 19:27:36 +05:30
|
|
|
);
|
2020-07-19 12:27:35 -07:00
|
|
|
}
|
2021-08-07 17:47:57 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG(" <points\n");
|
|
|
|
return MP_OBJ_FROM_PTR(list);
|
2020-05-02 02:21:35 -07:00
|
|
|
}
|
|
|
|
void common_hal_vectorio_polygon_set_points(vectorio_polygon_t *self, mp_obj_t points_list) {
|
2020-07-19 12:27:35 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG("%p common_hal_vectorio_polygon_set_points: ", self);
|
2021-03-15 19:27:36 +05:30
|
|
|
_clobber_points_list(self, points_list);
|
2020-05-02 02:21:35 -07:00
|
|
|
if (self->on_dirty.obj != NULL) {
|
|
|
|
self->on_dirty.event(self->on_dirty.obj);
|
|
|
|
}
|
2020-07-19 12:27:35 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG("\n");
|
2020-05-02 02:21:35 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void common_hal_vectorio_polygon_set_on_dirty(vectorio_polygon_t *self, vectorio_event_t notification) {
|
2021-03-15 19:27:36 +05:30
|
|
|
if (self->on_dirty.obj != NULL) {
|
2020-05-02 02:21:35 -07:00
|
|
|
mp_raise_TypeError(translate("polygon can only be registered in one parent"));
|
|
|
|
}
|
|
|
|
self->on_dirty = notification;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void common_hal_vectorio_polygon_get_area(void *polygon, displayio_area_t *area) {
|
|
|
|
vectorio_polygon_t *self = polygon;
|
2021-08-07 17:47:57 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG("%p common_hal_vectorio_polygon_get_area\n");
|
2020-05-02 02:21:35 -07:00
|
|
|
|
|
|
|
area->x1 = SHRT_MAX;
|
|
|
|
area->y1 = SHRT_MAX;
|
|
|
|
area->x2 = SHRT_MIN;
|
|
|
|
area->y2 = SHRT_MIN;
|
2021-08-07 17:47:57 -07:00
|
|
|
for (uint16_t i = 0; i < self->len; ++i) {
|
|
|
|
int16_t x = self->points_list[i];
|
2020-05-02 02:21:35 -07:00
|
|
|
++i;
|
2021-08-07 17:47:57 -07:00
|
|
|
int16_t y = self->points_list[i];
|
2021-08-01 13:01:57 -07:00
|
|
|
if (x < area->x1) {
|
2021-08-07 17:47:57 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG(" x1: %d\n", x);
|
2021-08-01 13:01:57 -07:00
|
|
|
area->x1 = x;
|
2021-03-15 19:27:36 +05:30
|
|
|
}
|
2021-08-01 13:01:57 -07:00
|
|
|
if (y < area->y1) {
|
2021-08-07 17:47:57 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG(" y1: %d\n", y);
|
2021-08-01 13:01:57 -07:00
|
|
|
area->y1 = y;
|
2021-03-15 19:27:36 +05:30
|
|
|
}
|
2021-08-01 13:01:57 -07:00
|
|
|
if (x > area->x2) {
|
2021-08-07 17:47:57 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG(" x2: %d\n", x);
|
2021-08-01 13:01:57 -07:00
|
|
|
area->x2 = x;
|
2021-03-15 19:27:36 +05:30
|
|
|
}
|
2021-08-01 13:01:57 -07:00
|
|
|
if (y > area->y2) {
|
2021-08-07 17:47:57 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG(" y2: %d\n", y);
|
2021-08-01 13:01:57 -07:00
|
|
|
area->y2 = y;
|
2021-03-15 19:27:36 +05:30
|
|
|
}
|
2020-05-02 02:21:35 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// <0 if the point is to the left of the line vector
|
|
|
|
// 0 if the point is on the line
|
|
|
|
// >0 if the point is to the right of the line vector
|
2021-08-07 17:47:57 -07:00
|
|
|
__attribute__((always_inline)) static inline int line_side(int16_t x1, int16_t y1, int16_t x2, int16_t y2, int16_t px, int16_t py) {
|
2020-05-02 02:21:35 -07:00
|
|
|
return (px - x1) * (y2 - y1)
|
2021-03-15 19:27:36 +05:30
|
|
|
- (py - y1) * (x2 - x1);
|
2020-05-02 02:21:35 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t common_hal_vectorio_polygon_get_pixel(void *obj, int16_t x, int16_t y) {
|
|
|
|
VECTORIO_POLYGON_DEBUG("%p polygon get_pixel %d, %d\n", obj, x, y);
|
|
|
|
vectorio_polygon_t *self = obj;
|
|
|
|
|
2020-05-12 21:55:18 -07:00
|
|
|
if (self->len == 0) {
|
2020-05-02 02:21:35 -07:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-08-07 17:47:57 -07:00
|
|
|
int16_t winding_number = 0;
|
|
|
|
int16_t x1 = self->points_list[0];
|
|
|
|
int16_t y1 = self->points_list[1];
|
|
|
|
for (uint16_t i = 2; i <= self->len + 1; ++i) {
|
2020-05-02 02:21:35 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG(" {(%3d, %3d),", x1, y1);
|
2021-08-07 17:47:57 -07:00
|
|
|
int16_t x2 = self->points_list[i % self->len];
|
2020-05-02 02:21:35 -07:00
|
|
|
++i;
|
2021-08-07 17:47:57 -07:00
|
|
|
int16_t y2 = self->points_list[i % self->len];
|
2020-05-02 02:21:35 -07:00
|
|
|
VECTORIO_POLYGON_DEBUG(" (%3d, %3d)}\n", x2, y2);
|
2021-03-15 19:27:36 +05:30
|
|
|
if (y1 <= y) {
|
|
|
|
if (y2 > y && line_side(x1, y1, x2, y2, x, y) < 0) {
|
use top-left heuristic for vectorio.Polygon
this flips the bottom-right style to top-left which is at least
kind of normal. A 2x2 square at (0,0) would be defined like
(0,0), (3,0), (3,3), (0,3)
Which seems kind of surprising but at least less bonkers than
that square being defined at (1,1), which is the current behavior.
2020-07-28 22:58:28 -07:00
|
|
|
// Wind up, point is to the left of the edge vector
|
2020-05-02 02:21:35 -07:00
|
|
|
++winding_number;
|
|
|
|
VECTORIO_POLYGON_DEBUG(" wind:%2d winding_number:%2d\n", 1, winding_number);
|
|
|
|
}
|
2021-03-15 19:27:36 +05:30
|
|
|
} else if (y2 <= y && line_side(x1, y1, x2, y2, x, y) > 0) {
|
use top-left heuristic for vectorio.Polygon
this flips the bottom-right style to top-left which is at least
kind of normal. A 2x2 square at (0,0) would be defined like
(0,0), (3,0), (3,3), (0,3)
Which seems kind of surprising but at least less bonkers than
that square being defined at (1,1), which is the current behavior.
2020-07-28 22:58:28 -07:00
|
|
|
// Wind down, point is to the right of the edge vector
|
2020-05-02 02:21:35 -07:00
|
|
|
--winding_number;
|
2020-05-30 10:44:13 +01:00
|
|
|
VECTORIO_POLYGON_DEBUG(" wind:%2d winding_number:%2d\n", -1, winding_number);
|
2020-05-02 02:21:35 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
x1 = x2;
|
|
|
|
y1 = y2;
|
|
|
|
}
|
|
|
|
return winding_number == 0 ? 0 : 1;
|
|
|
|
}
|
2021-08-01 13:01:57 -07:00
|
|
|
|
|
|
|
mp_obj_t common_hal_vectorio_polygon_get_draw_protocol(void *polygon) {
|
|
|
|
vectorio_polygon_t *self = polygon;
|
|
|
|
return self->draw_protocol_instance;
|
|
|
|
}
|