7
mirror of https://gitlab.com/kicad/code/kicad.git synced 2024-11-24 00:34:47 +00:00
kicad/common/preview_items/construction_geom.cpp
Seth Hillbrand 9dfcb6a362 Cleanup ViewGetLayers()
Old style c-array replaced with vector returns
2024-11-21 13:18:36 -08:00

164 lines
5.7 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 2
* 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.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, you may find one here:
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
* or you may search the http://www.gnu.org website for the version 2 license,
* or you may write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
*/
#include "preview_items/construction_geom.h"
#include <layer_ids.h>
#include <gal/graphics_abstraction_layer.h>
#include <geometry/shape_utils.h>
#include <preview_items/item_drawing_utils.h>
#include <view/view.h>
using namespace KIGFX;
CONSTRUCTION_GEOM::CONSTRUCTION_GEOM() :
EDA_ITEM( nullptr, NOT_USED ), // Never added to a BOARD/SCHEMATIC so it needs no type
m_color( COLOR4D::WHITE ), m_persistentColor( COLOR4D::WHITE )
{
}
void CONSTRUCTION_GEOM::AddDrawable( const DRAWABLE& aItem, bool aPersistent )
{
m_drawables.push_back( { aItem, aPersistent } );
}
void CONSTRUCTION_GEOM::ClearDrawables()
{
m_drawables.clear();
}
const BOX2I CONSTRUCTION_GEOM::ViewBBox() const
{
// We could be a bit more circumspect here, but much of the time the
// enxtended lines go across the whole screen anyway
BOX2I bbox;
bbox.SetMaximum();
return bbox;
}
void CONSTRUCTION_GEOM::ViewDraw( int aLayer, VIEW* aView ) const
{
GAL& gal = *aView->GetGAL();
gal.SetIsFill( false );
gal.SetIsStroke( true );
gal.SetLineWidth( 1 );
const BOX2I viewport = BOX2ISafe( aView->GetViewport() );
// Prevents extremely short snap lines from inhibiting drawing
// These can happen due to rounding in intersections, etc.
// (usually it is length 1 IU)
const int minSnapLineLength = 10;
const bool haveSnapLine = m_snapLine && m_snapLine->Length() >= minSnapLineLength;
// Avoid fighting with the snap line
const auto drawLineIfNotAlsoSnapLine = [&]( const SEG& aLine )
{
if( !haveSnapLine || !aLine.ApproxCollinear( *m_snapLine, 1 ) )
{
gal.DrawLine( aLine.A, aLine.B );
}
};
// Draw all the items
for( const DRAWABLE_INFO& drawable : m_drawables )
{
gal.SetStrokeColor( drawable.IsPersistent ? m_persistentColor : m_color );
std::visit(
[&]( const auto& visited )
{
using ItemType = std::decay_t<decltype( visited )>;
if constexpr( std::is_same_v<ItemType, LINE> )
{
// Extend the segment to the viewport boundary
std::optional<SEG> segToBoundary =
KIGEOM::ClipLineToBox( visited, viewport );
if( segToBoundary )
{
drawLineIfNotAlsoSnapLine( *segToBoundary );
}
}
else if constexpr( std::is_same_v<ItemType, HALF_LINE> )
{
// Extend the ray to the viewport boundary
std::optional<SEG> segToBoundary =
KIGEOM::ClipHalfLineToBox( visited, viewport );
if( segToBoundary )
{
drawLineIfNotAlsoSnapLine( *segToBoundary );
}
}
else if constexpr( std::is_same_v<ItemType, SEG> )
{
drawLineIfNotAlsoSnapLine( visited );
}
else if constexpr( std::is_same_v<ItemType, CIRCLE> )
{
gal.DrawCircle( visited.Center, visited.Radius );
}
else if constexpr( std::is_same_v<ItemType, SHAPE_ARC> )
{
gal.DrawArc( visited.GetCenter(), visited.GetRadius(),
visited.GetStartAngle(), visited.GetCentralAngle() );
}
else if constexpr( std::is_same_v<ItemType, VECTOR2I> )
{
KIGFX::DrawCross( gal, visited, aView->ToWorld( 16 ) );
}
},
drawable.Item );
}
if( haveSnapLine )
{
gal.SetStrokeColor( m_persistentColor );
const int dashSizeBasis = aView->ToWorld( 12 );
const int snapOriginMarkerSize = aView->ToWorld( 16 );
// Avoid clash with the snap marker if very close
const int omitStartMarkerIfWithinLength = aView->ToWorld( 8 );
// The line itself
KIGFX::DrawDashedLine( gal, *m_snapLine, dashSizeBasis );
// The line origin marker if the line is long enough
if( m_snapLine->A.Distance( m_snapLine->B ) > omitStartMarkerIfWithinLength )
{
KIGFX::DrawCross( gal, m_snapLine->A, snapOriginMarkerSize );
gal.DrawCircle( m_snapLine->A, snapOriginMarkerSize / 2 );
}
}
}
std::vector<int> CONSTRUCTION_GEOM::ViewGetLayers() const
{
std::vector<int> layers{ LAYER_GP_OVERLAY };
return layers;
}