/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright The 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 "tool/grid_helper.h" #include #include #include #include #include // for KiROUND #include #include #include #include #include GRID_HELPER::GRID_HELPER( TOOL_MANAGER* aToolMgr, int aConstructionLayer ) : m_toolMgr( aToolMgr ), m_snapManager( m_constructionGeomPreview ) { m_maskTypes = ALL; m_enableSnap = true; m_enableSnapLine = true; m_enableGrid = true; m_snapItem = std::nullopt; KIGFX::VIEW* view = m_toolMgr->GetView(); KIGFX::RENDER_SETTINGS* settings = view->GetPainter()->GetSettings(); const KIGFX::COLOR4D constructionColour = settings->GetLayerColor( aConstructionLayer ); m_constructionGeomPreview.SetPersistentColor( constructionColour ); m_constructionGeomPreview.SetColor( constructionColour.WithAlpha( 0.7 ) ); view->Add( &m_constructionGeomPreview ); view->SetVisible( &m_constructionGeomPreview, false ); m_snapManager.SetUpdateCallback( [view, this]( bool aAnythingShown ) { const bool currentlyVisible = view->IsVisible( &m_constructionGeomPreview ); if( currentlyVisible && aAnythingShown ) { view->Update( &m_constructionGeomPreview, KIGFX::GEOMETRY ); } else { view->SetVisible( &m_constructionGeomPreview, aAnythingShown ); } } ); } GRID_HELPER::~GRID_HELPER() { KIGFX::VIEW& view = *m_toolMgr->GetView(); view.Remove( &m_constructionGeomPreview ); if( m_anchorDebug ) view.Remove( m_anchorDebug.get() ); } KIGFX::ANCHOR_DEBUG* GRID_HELPER::enableAndGetAnchorDebug() { static bool permitted = ADVANCED_CFG::GetCfg().m_EnableSnapAnchorsDebug; if( permitted && !m_anchorDebug ) { KIGFX::VIEW& view = *m_toolMgr->GetView(); m_anchorDebug = std::make_unique(); view.Add( m_anchorDebug.get() ); view.SetVisible( m_anchorDebug.get(), true ); } return m_anchorDebug.get(); } void GRID_HELPER::showConstructionGeometry( bool aShow ) { m_toolMgr->GetView()->SetVisible( &m_constructionGeomPreview, aShow ); } void GRID_HELPER::updateSnapPoint( const TYPED_POINT2I& aPoint ) { m_viewSnapPoint.SetPosition( aPoint.m_point ); m_viewSnapPoint.SetSnapTypes( aPoint.m_types ); if( m_toolMgr->GetView()->IsVisible( &m_viewSnapPoint ) ) m_toolMgr->GetView()->Update( &m_viewSnapPoint, KIGFX::GEOMETRY ); else m_toolMgr->GetView()->SetVisible( &m_viewSnapPoint, true ); } VECTOR2I GRID_HELPER::GetGrid() const { VECTOR2D size = m_toolMgr->GetView()->GetGAL()->GetGridSize(); return VECTOR2I( KiROUND( size.x ), KiROUND( size.y ) ); } VECTOR2D GRID_HELPER::GetVisibleGrid() const { return m_toolMgr->GetView()->GetGAL()->GetVisibleGridSize(); } VECTOR2I GRID_HELPER::GetOrigin() const { VECTOR2D origin = m_toolMgr->GetView()->GetGAL()->GetGridOrigin(); return VECTOR2I( origin ); } GRID_HELPER_GRIDS GRID_HELPER::GetSelectionGrid( const SELECTION& aSelection ) const { GRID_HELPER_GRIDS grid = GetItemGrid( aSelection.Front() ); // Find the largest grid of all the items and use that for( EDA_ITEM* item : aSelection ) { GRID_HELPER_GRIDS itemGrid = GetItemGrid( item ); if( GetGridSize( itemGrid ) > GetGridSize( grid ) ) grid = itemGrid; } return grid; } VECTOR2D GRID_HELPER::GetGridSize( GRID_HELPER_GRIDS aGrid ) const { return m_toolMgr->GetView()->GetGAL()->GetGridSize(); } void GRID_HELPER::SetAuxAxes( bool aEnable, const VECTOR2I& aOrigin ) { if( aEnable ) { m_auxAxis = aOrigin; m_viewAxis.SetPosition( aOrigin ); m_toolMgr->GetView()->SetVisible( &m_viewAxis, true ); } else { m_auxAxis = std::optional(); m_toolMgr->GetView()->SetVisible( &m_viewAxis, false ); } } VECTOR2I GRID_HELPER::AlignGrid( const VECTOR2I& aPoint ) const { return computeNearest( aPoint, GetGrid(), GetOrigin() ); } VECTOR2I GRID_HELPER::AlignGrid( const VECTOR2I& aPoint, const VECTOR2D& aGrid, const VECTOR2D& aOffset ) const { return computeNearest( aPoint, aGrid, aOffset ); } VECTOR2I GRID_HELPER::computeNearest( const VECTOR2I& aPoint, const VECTOR2I& aGrid, const VECTOR2I& aOffset ) const { return VECTOR2I( KiROUND( (double) ( aPoint.x - aOffset.x ) / aGrid.x ) * aGrid.x + aOffset.x, KiROUND( (double) ( aPoint.y - aOffset.y ) / aGrid.y ) * aGrid.y + aOffset.y ); } VECTOR2I GRID_HELPER::Align( const VECTOR2I& aPoint ) const { return Align( aPoint, GetGrid(), GetOrigin() ); } VECTOR2I GRID_HELPER::Align( const VECTOR2I& aPoint, const VECTOR2D& aGrid, const VECTOR2D& aOffset ) const { if( !canUseGrid() ) return aPoint; VECTOR2I nearest = AlignGrid( aPoint, aGrid, aOffset ); if( !m_auxAxis ) return nearest; if( std::abs( m_auxAxis->x - aPoint.x ) < std::abs( nearest.x - aPoint.x ) ) nearest.x = m_auxAxis->x; if( std::abs( m_auxAxis->y - aPoint.y ) < std::abs( nearest.y - aPoint.y ) ) nearest.y = m_auxAxis->y; return nearest; } bool GRID_HELPER::canUseGrid() const { return m_enableGrid && m_toolMgr->GetView()->GetGAL()->GetGridSnapping(); } std::optional GRID_HELPER::GetSnappedPoint() const { if( m_snapItem ) return m_snapItem->pos; return std::nullopt; }