kicad-source/qa/tests/libs/kimath/geometry/test_vector_utils.cpp
John Beard 5925f9c374 Break out some geom functions from dimension text adjustment
Put these in a separate header - they don't need to be available
whenever VECTOR2 is, but they are generic, reusable functions that
can have the corner cases defined and tested.
2024-09-11 22:08:36 +01:00

106 lines
3.4 KiB
C++

/*
* This program source code file is part of KiCad, a free EDA CAD application.
*
* Copyright (C) 2024 KiCad Developers, see AUTHORS.TXT for contributors.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 3
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
*/
#include <qa_utils/wx_utils/unit_test_utils.h>
#include <geometry/vector_utils.h>
#include <geometry/seg.h>
BOOST_AUTO_TEST_SUITE( VectorUtils )
BOOST_AUTO_TEST_CASE( PointIsInDirection )
{
const VECTOR2I p0( 1000, 1000 );
const VECTOR2I ne( 100, 100 );
const VECTOR2I n( 0, 100 );
const VECTOR2I s( 0, -100 );
// To the east, so not directly inline with ne from p0
const VECTOR2I p1_east = p0 + VECTOR2I( 1000, 0 );
const VECTOR2I p1_west = p0 - VECTOR2I( 1000, 0 );
BOOST_TEST( KIGEOM::PointIsInDirection( p1_east, ne, p0 ) );
BOOST_TEST( !KIGEOM::PointIsInDirection( p1_west, ne, p0 ) );
// Test the perpendicular corner case
// Points on both sides are not in the direction
BOOST_TEST( !KIGEOM::PointIsInDirection( p1_east, n, p0 ) );
BOOST_TEST( !KIGEOM::PointIsInDirection( p1_west, n, p0 ) );
// And they're also not in the opposite direction
BOOST_TEST( !KIGEOM::PointIsInDirection( p1_east, s, p0 ) );
BOOST_TEST( !KIGEOM::PointIsInDirection( p1_west, s, p0 ) );
}
BOOST_AUTO_TEST_CASE( ProjectsOntoSeg )
{
const SEG seg( VECTOR2I( 0, 0 ), VECTOR2I( 100, 0 ) );
BOOST_TEST( KIGEOM::PointProjectsOntoSegment( VECTOR2I( 50, 1000 ), seg ) );
BOOST_TEST( !KIGEOM::PointProjectsOntoSegment( VECTOR2I( 150, 1000 ), seg ) );
}
BOOST_AUTO_TEST_CASE( LengthRatio )
{
const SEG seg( VECTOR2I( 0, 0 ), VECTOR2I( 100, 0 ) );
BOOST_TEST( KIGEOM::GetLengthRatioFromStart( VECTOR2I( 0, 0 ), seg ) == 0.0 );
BOOST_TEST( KIGEOM::GetLengthRatioFromStart( VECTOR2I( 100, 0 ), seg ) == 1.0 );
BOOST_TEST( KIGEOM::GetLengthRatioFromStart( VECTOR2I( 50, 0 ), seg ) == 0.5 );
// Points not on the segment also work
BOOST_CHECK_CLOSE( KIGEOM::GetLengthRatioFromStart( VECTOR2I( 0, 100 ), seg ), 1.0, 0.0001 );
BOOST_CHECK_CLOSE( KIGEOM::GetLengthRatioFromStart( VECTOR2I( 0, 50 ), seg ), 0.5, 0.0001 );
}
BOOST_AUTO_TEST_CASE( NearestEndpoint )
{
struct PtCase
{
VECTOR2I Point;
bool ExpectedStart;
};
const SEG seg( VECTOR2I( 0, 0 ), VECTOR2I( 100, 0 ) );
const std::vector<PtCase> cases{
{ { -100, 0 }, true },
{ { 0, 0 }, true },
{ { 0, 50 }, true },
// Make sure the tie breaks predictably
// Equidistant points -> start
{ { 50, 0 }, true },
{ { 50, 50 }, true },
{ { 50, -50 }, true },
// End points
{ { 200, 0 }, false },
{ { 100, 0 }, false },
{ { 100, 50 }, false },
};
for( const PtCase& pc : cases )
{
BOOST_TEST_INFO( "Point: " << pc.Point );
BOOST_TEST( KIGEOM::GetNearestEndpoint( seg, pc.Point )
== ( pc.ExpectedStart ? seg.A : seg.B ) );
}
}
BOOST_AUTO_TEST_SUITE_END()