blob: 4902d32cd1091290eaf911060e2f1c6c36791275 [file] [log] [blame]
// Copyright 2023 Google LLC
// Use of this source code is governed by a BSD-style license that can be found in the LICENSE file.
#include "modules/bentleyottmann/include/Segment.h"
#include "tests/Test.h"
using namespace bentleyottmann;
DEF_TEST(BO_SegmentBasic, reporter) {
{
Segment s = {{0, 0}, {1, 1}};
REPORTER_ASSERT(reporter, s.upper() == s.p0);
REPORTER_ASSERT(reporter, s.lower() == s.p1);
}
{
Segment s = {{1, 0}, {0, 1}};
REPORTER_ASSERT(reporter, s.upper() == s.p0);
REPORTER_ASSERT(reporter, s.lower() == s.p1);
}
{
Segment s = {{1, 1}, {0, 0}};
REPORTER_ASSERT(reporter, s.upper() == s.p1);
REPORTER_ASSERT(reporter, s.lower() == s.p0);
}
{
Segment s = {{0, 1}, {1, 0}};
REPORTER_ASSERT(reporter, s.upper() == s.p1);
REPORTER_ASSERT(reporter, s.lower() == s.p0);
}
}
static Segment swap_ends(const Segment& s) {
return {s.p1, s.p0};
}
DEF_TEST(BO_no_intersection_bounding_box, reporter) {
Segment interesting[] = {{Point::Smallest(), Point::Smallest()+ Point{10, 5}},
{Point::Largest(), Point::Largest() - Point{10, 5}},
{{-10, -5}, {10, 5}}};
// Intersection
for (auto& s0 : interesting) {
auto [l, t, r, b] = s0.bounds();
// Points in the interior of interesting rectangles
for(Point p : {Point {l + 1, t + 1},
Point {r - 1, t + 1},
Point {r - 1, b - 1},
Point {l + 1, b - 1}}) {
Segment s1 = {p, {0, 0}};
REPORTER_ASSERT(reporter, !no_intersection_by_bounding_box(s0, s1));
REPORTER_ASSERT(reporter, !no_intersection_by_bounding_box(s1, s0));
REPORTER_ASSERT(reporter,
!no_intersection_by_bounding_box(swap_ends(s0), swap_ends(s1)));
REPORTER_ASSERT(reporter,
!no_intersection_by_bounding_box(swap_ends(s0), swap_ends(s1)));
}
}
int32_t small = Point::Smallest().x,
big = Point::Largest().x;
// No Intersection
for (auto& s0 : interesting) {
auto [l, t, r, b] = s0.bounds();
Segment outside[] = {{{r, t}, {big, b}},
{{r, b}, {big, big}},
{{l, b}, {r, big}},
{{l, b}, {small, big}},
{{l, t}, {small, b}},
{{l, t}, {small, small}},
{{l, t}, {r, small}},
{{r, t}, {small, small}}};
for (auto& s1 : outside) {
REPORTER_ASSERT(reporter, no_intersection_by_bounding_box(s0, s1));
REPORTER_ASSERT(reporter, no_intersection_by_bounding_box(s1, s0));
REPORTER_ASSERT(reporter,
no_intersection_by_bounding_box(swap_ends(s0), swap_ends(s1)));
REPORTER_ASSERT(reporter,
no_intersection_by_bounding_box(swap_ends(s0), swap_ends(s1)));
}
}
}
DEF_TEST(BO_intersectBasic, reporter) {
auto checkIntersection = [reporter](Segment s0, Segment s1, Point expected) {
{
auto answer = intersect(s0, s1);
REPORTER_ASSERT(reporter, answer.has_value());
REPORTER_ASSERT(reporter, answer.value() == expected);
}
{
auto answer = intersect(s1, s0);
REPORTER_ASSERT(reporter, answer.has_value());
REPORTER_ASSERT(reporter, answer.value() == expected);
}
{
auto answer = intersect(swap_ends(s0), swap_ends(s1));
REPORTER_ASSERT(reporter, answer.has_value());
REPORTER_ASSERT(reporter, answer.value() == expected);
}
{
auto answer = intersect(swap_ends(s1), swap_ends(s0));
REPORTER_ASSERT(reporter, answer.has_value());
REPORTER_ASSERT(reporter, answer.value() == expected);
}
};
{
Segment s0 = {{-1, 0}, {1, 0}},
s1 = {{ 0, 1}, {0, -1}};
checkIntersection(s0, s1, Point{0, 0});
}
{
Segment s0 = {{-1, 0}, {5, 0}},
s1 = {{ 0, 1}, {0, -1}};
checkIntersection(s0, s1, Point{0, 0});
}
{
Segment s0 = {{5, 0}, {-1, 0}},
s1 = {{ 0, -1}, {0, 1}};
checkIntersection(s0, s1, Point{0, 0});
}
{
Segment s0 = {{-5, -5}, {5, 5}},
s1 = {{-5, 5}, {5, -5}};
checkIntersection(s0, s1, Point{0, 0});
}
// Test very close segments (x0, 0) -> (x1, 1) & (x2, 0) -> (x3, 1)
for (int32_t x0 = -10; x0 <= 10; x0++) {
for (int32_t x1 = -10; x1 <= 10; x1++) {
for (int32_t x2 = -10; x2 <= 10; x2++) {
for (int32_t x3 = -10; x3 <= 10; x3++) {
Point P0 = {x0, 0},
P1 = {x1, 1},
P2 = {x2, 0},
P3 = {x3, 1};
auto actual = intersect({P0, P1}, {P2, P3});
bool expected = (x0 < x2 && x3 < x1) || (x2 < x0 && x1 < x3);
REPORTER_ASSERT(reporter, actual.has_value() == expected);
if (actual) {
int32_t y = std::abs(x2 - x0) >= std::abs(x3 - x1);
REPORTER_ASSERT(reporter, actual.value().y == y);
}
}
}
}
}
{
Segment s0 = {{0, -100}, {0, -50}},
s1 = {{100, -100}, {-100, 100}}; // goes through (0,0)
auto answer = intersect(s0, s1);
REPORTER_ASSERT(reporter, !answer.has_value());
answer = intersect(s1, s0);
REPORTER_ASSERT(reporter, !answer.has_value());
}
{
Segment s0 = {{0, 100}, {0, 50}},
s1 = {{100, -100}, {-100, 100}}; // goes through (0,0)
auto answer = intersect(s0, s1);
REPORTER_ASSERT(reporter, !answer.has_value());
answer = intersect(s1, s0);
REPORTER_ASSERT(reporter, !answer.has_value());
}
}
DEF_TEST(BO_lessAtBasic, reporter) {
{ // Parallel lines
Segment s0 = {{-1, 1}, {-1, -1}},
s1 = {{1, 1}, {1, -1}};
REPORTER_ASSERT(reporter, less_than_at(s0, s1, -1));
REPORTER_ASSERT(reporter, !less_than_at(s1, s0, -1));
REPORTER_ASSERT(reporter, less_than_at(s0, s1, 0));
REPORTER_ASSERT(reporter, !less_than_at(s1, s0, 0));
REPORTER_ASSERT(reporter, less_than_at(s0, s1, 1));
REPORTER_ASSERT(reporter, !less_than_at(s1, s0, 1));
}
{ // Crossed lines
Segment s0 = {{-1, -1}, {1, 1}},
s1 = {{1, -1}, {-1, 1}};
REPORTER_ASSERT(reporter, less_than_at(s0, s1, -1));
REPORTER_ASSERT(reporter, !less_than_at(s1, s0, -1));
// When they are == neither is less.
REPORTER_ASSERT(reporter, !less_than_at(s0, s1, 0));
REPORTER_ASSERT(reporter, !less_than_at(s1, s0, 0));
REPORTER_ASSERT(reporter, !less_than_at(s0, s1, 1));
REPORTER_ASSERT(reporter, less_than_at(s1, s0, 1));
}
{ // Near crossing
Segment s0 = {{0, -100}, {0, 100}},
s1 = {{-3, 98}, {3, 104}};
REPORTER_ASSERT(reporter, !less_than_at(s0, s1, 98));
REPORTER_ASSERT(reporter, less_than_at(s1, s0, 98));
REPORTER_ASSERT(reporter, !less_than_at(s0, s1, 99));
REPORTER_ASSERT(reporter, less_than_at(s1, s0, 99));
REPORTER_ASSERT(reporter, !less_than_at(s0, s1, 100));
REPORTER_ASSERT(reporter, less_than_at(s1, s0, 100));
}
}
DEF_TEST(BO_compareSlopesBasic, reporter) {
{ // Both horizontal
Segment s0 = {{-1, 1}, {0, 1}},
s1 = {{-2, 1}, {1, 1}};
REPORTER_ASSERT(reporter, compare_slopes(s0, s1) == 0);
REPORTER_ASSERT(reporter, compare_slopes(s1, s0) == 0);
}
{ // One horizontal
Segment s0 = {{-1, 1}, {0, 0}},
s1 = {{-2, 1}, {1, 1}};
REPORTER_ASSERT(reporter, compare_slopes(s0, s1) == -1);
REPORTER_ASSERT(reporter, compare_slopes(s1, s0) == 1);
}
{ // One vertical
Segment s0 = {{-1, 1}, {-1, 0}}, // Vertical
s1 = {{-2, 1}, {-1, 0}},
s2 = {{2, 1}, {-1, 0}};
REPORTER_ASSERT(reporter, compare_slopes(s0, s1) == 1);
REPORTER_ASSERT(reporter, compare_slopes(s1, s0) == -1);
REPORTER_ASSERT(reporter, compare_slopes(s0, s2) == -1);
REPORTER_ASSERT(reporter, compare_slopes(s2, s0) == 1);
}
{ // Equal slope
Segment s0 = {{-2, 1}, {0, 0}},
s1 = {{-4, 2}, {0, 0}};
REPORTER_ASSERT(reporter, compare_slopes(s0, s1) == 0);
REPORTER_ASSERT(reporter, compare_slopes(s1, s0) == 0);
}
{ // Equal slope
Segment s0 = {{2, 1}, {0, 0}},
s1 = {{4, 2}, {0, 0}};
REPORTER_ASSERT(reporter, compare_slopes(s0, s1) == 0);
REPORTER_ASSERT(reporter, compare_slopes(s1, s0) == 0);
}
{
Segment s0 = {{-2, 1}, {0, 0}},
s1 = {{4, 2}, {0, 0}};
REPORTER_ASSERT(reporter, compare_slopes(s0, s1) == -1);
REPORTER_ASSERT(reporter, compare_slopes(s1, s0) == 1);
}
{
Segment s0 = {{-2, 1}, {0, 0}},
s1 = {{-3, 1}, {0, 0}};
REPORTER_ASSERT(reporter, compare_slopes(s0, s1) == 1);
REPORTER_ASSERT(reporter, compare_slopes(s1, s0) == -1);
}
}
DEF_TEST(BO_rounded_point_less_than_segment_in_x_lower, reporter) {
{ // Vertical segment
Segment s = {{3, -50}, {3, 50}};
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_lower(s, {4, 7}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_lower(s, {3, 7}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_lower(s, {2, 7}));
}
{ // Horizontal segment
Segment s = {{-10, 3}, {10, 3}};
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_lower(s, {11, 3}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_lower(s, {10, 3}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_lower(s, {4, 3}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_lower(s, {-10, 3}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_lower(s, {-11, 3}));
}
{ // Pass through {0, 0}
Segment s = {{-100, -100}, {100, 100}};
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_lower(s, {1, 0}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_lower(s, {0, 0}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_lower(s, {-1, 0}));
}
{ // Just left of {0, 0}, but still rounds to {0, 0}
Segment s = {{-100, -100}, {199, 200}};
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_lower(s, {1, 0}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_lower(s, {0, 0}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_lower(s, {-1, 0}));
}
{ // Just right of {0, 0}, but still rounds to {0, 0}
Segment s = {{-100, -100}, {201, 200}};
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_lower(s, {1, 0}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_lower(s, {0, 0}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_lower(s, {-1, 0}));
}
}
DEF_TEST(BO_rounded_point_less_than_segment_in_x_upper, reporter) {
{ // Vertical segment
Segment s = {{3, -50}, {3, 50}};
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_upper(s, {4, 7}));
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_upper(s, {3, 7}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_upper(s, {2, 7}));
}
{ // Horizontal segment
Segment s = {{-10, 3}, {10, 3}};
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_upper(s, {11, 3}));
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_upper(s, {10, 3}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_upper(s, {4, 3}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_upper(s, {-10, 3}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_upper(s, {-11, 3}));
}
{ // Pass through {0, 0}
Segment s = {{-100, -100}, {100, 100}};
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_upper(s, {1, 0}));
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_upper(s, {0, 0}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_upper(s, {-1, 0}));
}
{ // Just left of {0, 0}, but still rounds to {0, 0}
Segment s = {{-100, -100}, {199, 200}};
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_upper(s, {1, 0}));
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_upper(s, {0, 0}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_upper(s, {-1, 0}));
}
{ // Just right of {0, 0}, but still rounds to {0, 0}
Segment s = {{-100, -100}, {201, 200}};
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_upper(s, {1, 0}));
REPORTER_ASSERT(reporter, rounded_point_less_than_segment_in_x_upper(s, {0, 0}));
REPORTER_ASSERT(reporter, !rounded_point_less_than_segment_in_x_upper(s, {-1, 0}));
}
}