48 return (sincos[0] * p[0]) + (sincos[1] * p[1]);
53 return (sincos[1] * p[0]) - (sincos[0] * p[1]);
77static float is_left(
const float p0[2],
const float p1[2],
const float p2[2])
79 return (p1[0] - p0[0]) * (p2[1] - p0[1]) - (p2[0] - p0[0]) * (p1[1] - p0[1]);
93 const int maxmax = points_num - 1;
100 float xmin = points[0][0];
101 for (
i = 1;
i <= maxmax;
i++) {
102 if (points[
i][0] != xmin) {
108 if (minmax == maxmax) {
109 r_points[++
top] = minmin;
110 if (points[minmax][1] != points[minmin][1]) {
112 r_points[++
top] = minmax;
120 xmax = points[maxmax][0];
121 for (
i = maxmax - 1;
i >= 0;
i--) {
122 if (points[
i][0] != xmax) {
129 r_points[++
top] = minmin;
131 while (++
i <= maxmin) {
133 if ((
i < maxmin) && (
is_left(points[minmin], points[maxmin], points[
i]) >= 0)) {
139 if (
is_left(points[r_points[
top - 1]], points[r_points[
top]], points[
i]) > 0.0f) {
149 if (maxmax != maxmin) {
150 r_points[++
top] = maxmax;
155 while (--
i >= minmax) {
157 if ((
i > minmax) && (
is_left(points[maxmax], points[minmax], points[
i]) >= 0)) {
163 if (
is_left(points[r_points[
top - 1]], points[r_points[
top]], points[
i]) > 0.0f) {
169 if (points[
i][0] == points[r_points[0]][0] && points[
i][1] == points[r_points[0]][1]) {
177 if (minmax != minmin && r_points[0] != minmin) {
178 r_points[++
top] = minmin;
188 if (points_num < 2) {
189 if (points_num == 1) {
197 for (
int i = 0;
i < points_num;
i++) {
202 std::sort(points_map, points_map + points_num, [points](
const int &a_index,
const int &b_index) {
203 const float *a = points[a_index];
204 const float *
b = points[b_index];
221 for (
int i = 0;
i < points_num;
i++) {
228 for (
int i = 0;
i < points_hull_num;
i++) {
229 r_points[
i] = points_map[r_points[
i]];
236 return points_hull_num;
245#if defined(USE_BRUTE_FORCE_ASSERT) && !defined(NDEBUG)
246static float2 convexhull_aabb_fit_hull_2d_brute_force(
const float (*points_hull)[2],
250 float2 sincos_best = {0.0f, 1.0f};
252 for (
int i = 0;
i < points_hull_num;
i++) {
253 const int i_next = (
i + 1) % points_hull_num;
255 float dvec_length = 0.0f;
257 float2(points_hull[i_next]) -
float2(points_hull[
i]), dvec_length);
258 if (
UNLIKELY(dvec_length == 0.0f)) {
265 for (
int j = 0; j < points_hull_num; j++) {
277 if (area_test > area_best) {
282 if (area_test < area_best) {
283 area_best = area_test;
284 sincos_best = sincos;
311 if (sincos[0] < 0.0f) {
312 if (sincos[1] < 0.0f) {
316 else if ((sincos[0] == -1.0f) && (sincos[1] == 0.0f)) {
326 if (sincos[1] < 0.0f) {
330 else if ((sincos[0] == 0.0f) && (sincos[1] == 1.0f)) {
418 prev_p = &iter->
next;
429 const int i_curr = hstep.
index;
432 float dir_length = 0.0f;
434 hstep.
index = i_next;
435 if (
LIKELY(dir_length != 0.0f)) {
449 const int points_hull_num)
451 const int points_hull_num_minus_1 = points_hull_num - 1;
456 for (
int axis = 0; axis < 2; axis++) {
457 range[axis][0] = points_hull[0][axis];
458 range[axis][1] = points_hull[0][axis];
464 for (
int i = 1;
i < points_hull_num;
i++) {
465 const float *p = points_hull[
i];
466 for (
int axis = 0; axis < 2; axis++) {
467 if (range[axis][0] < p[axis]) {
468 range[axis][0] = p[axis];
471 if (range[axis][1] > p[axis]) {
472 range[axis][1] = p[axis];
483 for (
int axis = 0; axis < 2; axis++) {
484 for (
int i = 0;
i < 2;
i++) {
486 int i_curr = i_orig, i_prev;
490 while ((i_prev = (i_curr + points_hull_num_minus_1) % points_hull_num) != i_orig) {
491 float dir_length = 0.0f;
493 float2(points_hull[i_curr]) -
float2(points_hull[i_prev]), dir_length);
494 if (
LIKELY(dir_length != 0.0f)) {
496 if (
math::abs(sincos_test[axis]) > 0.5f) {
500 if (
LIKELY(sincos_test_canonical[0] != 1.0f)) {
519 for (
int axis = 0; axis < 2; axis++) {
520 for (
int i = 0;
i < 2;
i++) {
534#ifdef USE_ANGLE_ITER_ORDER_ASSERT
543#ifdef USE_ANGLE_ITER_ORDER_ASSERT
562template<
int Axis,
int AxisSign>
564 const int points_hull_num,
574 const int index_init = *index_p;
575 int index_best = index_init;
576 float value_init = (Axis == 0) ?
sincos_rotate_cw_x(sincos, points_hull[index_best]) :
578 float value_best = value_init;
581 const int index_test = (index_init +
count) % points_hull_num;
582 const float value_test = (Axis == 0) ?
sincos_rotate_cw_x(sincos, points_hull[index_test]) :
584 if ((AxisSign == -1) ? (value_test > value_best) : (value_test < value_best)) {
587 value_best = value_test;
588 index_best = index_test;
591 *index_p = index_best;
598 float2 sincos_best = {0.0f, 1.0f};
599 int index_best = INT_MAX;
615 points_hull, points_hull_num, hstep->angle.sincos_canonical, &bounds_index[0].
min),
617 points_hull, points_hull_num, hstep->angle.sincos_canonical, &bounds_index[0].
max)},
619 points_hull, points_hull_num, hstep->angle.sincos_canonical, &bounds_index[1].
min),
621 points_hull, points_hull_num, hstep->angle.sincos_canonical, &bounds_index[1].
max)},
624 const float area_test = (bounds_test[0].
max - bounds_test[0].
min) *
625 (bounds_test[1].
max - bounds_test[1].
min);
627 if (area_test < area_best ||
630 ((area_test == area_best) && (hstep->angle.index < index_best)))
632 area_best = area_test;
633 sincos_best = hstep->angle.sincos;
634 index_best = hstep->angle.index;
640 const float angle = (area_best !=
FLT_MAX) ? atan2(sincos_best[0], sincos_best[1]) : 0.0f;
642#if defined(USE_BRUTE_FORCE_ASSERT) && !defined(NDEBUG)
645 const float2 sincos_test = convexhull_aabb_fit_hull_2d_brute_force(points_hull,
647 if (sincos_best != sincos_test) {
665 if (points_hull_num > 1) {
667 for (
int j = 0; j < points_hull_num; j++) {
668 copy_v2_v2(points_hull[j], points[index_map[j]]);
void BLI_kdtree_nd_ insert(KDTree *tree, int index, const float co[KD_DIMS]) ATTR_NONNULL(1
MINLINE void copy_v2_v2(float r[2], const float a[2])
static double angle(const Eigen::Vector3d &v1, const Eigen::Vector3d &v2)
Read Guarded memory(de)allocation.
static int hull_angle_canonical_cmp(const AngleCanonical &a, const AngleCanonical &b)
static HullAngleIter convexhull_2d_angle_iter_init(const float(*points_hull)[2], const int points_hull_num)
static float convexhull_aabb_fit_hull_2d(const float(*points_hull)[2], int points_hull_num)
static void hull_angle_insert_ordered(HullAngleIter &hiter, HullAngleStep *insert)
static float convexhull_2d_compute_extent_on_axis(const float(*points_hull)[2], const int points_hull_num, const float2 &sincos, int *index_p)
static float2 sincos_canonical(const float2 &sincos)
static int convexhull_2d_sorted(const float(*points)[2], const int points_num, int r_points[])
float BLI_convexhull_aabb_fit_points_2d(const float(*points)[2], int points_num)
int BLI_convexhull_2d(const float(*points)[2], const int points_num, int r_points[])
static float sincos_rotate_cw_x(const float2 &sincos, const float2 &p)
static void convexhull_2d_angle_iter_step(HullAngleIter &hiter)
static float sincos_rotate_cw_y(const float2 &sincos, const float2 &p)
static bool convexhull_2d_angle_iter_step_on_axis(const HullAngleIter &hiter, HullAngleStep &hstep)
static float is_left(const float p0[2], const float p1[2], const float p2[2])
void * MEM_malloc_arrayN(size_t len, size_t size, const char *str)
void MEM_freeN(void *vmemh)
QuaternionBase< T > normalize_and_get_length(const QuaternionBase< T > &q, T &out_length)
T min(const T &a, const T &b)
T max(const T &a, const T &b)
VecBase< float, 2 > float2
const float(* points_hull)[2]
HullAngleStep * axis_ordered