/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2017-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 "pad_tool.h" #include "pcb_painter.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using KIGFX::PCB_RENDER_SETTINGS; PAD_TOOL::PAD_TOOL() : PCB_TOOL_BASE( "pcbnew.PadTool" ), m_previousHighContrastMode( HIGH_CONTRAST_MODE::NORMAL ), m_editPad( niluuid ) {} PAD_TOOL::~PAD_TOOL() {} void PAD_TOOL::Reset( RESET_REASON aReason ) { if( aReason == MODEL_RELOAD ) m_lastPadNumber = wxT( "1" ); if( board() && board()->GetItem( m_editPad ) == DELETED_BOARD_ITEM::GetInstance() ) { PCB_DISPLAY_OPTIONS opts = frame()->GetDisplayOptions(); if( m_previousHighContrastMode != opts.m_ContrastModeDisplay ) { opts.m_ContrastModeDisplay = m_previousHighContrastMode; frame()->SetDisplayOptions( opts ); } frame()->GetInfoBar()->Dismiss(); m_editPad = niluuid; } } bool PAD_TOOL::Init() { PCB_SELECTION_TOOL* selTool = m_toolMgr->GetTool(); if( selTool ) { // Add context menu entries that are displayed when selection tool is active CONDITIONAL_MENU& menu = selTool->GetToolMenu().GetMenu(); SELECTION_CONDITION padSel = SELECTION_CONDITIONS::HasType( PCB_PAD_T ); SELECTION_CONDITION singlePadSel = SELECTION_CONDITIONS::Count( 1 ) && SELECTION_CONDITIONS::OnlyTypes( { PCB_PAD_T } ); auto explodeCondition = [&]( const SELECTION& aSel ) { return m_editPad == niluuid && aSel.Size() == 1 && aSel[0]->Type() == PCB_PAD_T; }; auto recombineCondition = [&]( const SELECTION& aSel ) { return m_editPad != niluuid; }; menu.AddSeparator( 400 ); if( m_isFootprintEditor ) { menu.AddItem( PCB_ACTIONS::enumeratePads, SELECTION_CONDITIONS::ShowAlways, 400 ); menu.AddItem( PCB_ACTIONS::recombinePad, recombineCondition, 400 ); menu.AddItem( PCB_ACTIONS::explodePad, explodeCondition, 400 ); } menu.AddItem( PCB_ACTIONS::copyPadSettings, singlePadSel, 400 ); menu.AddItem( PCB_ACTIONS::applyPadSettings, padSel, 400 ); menu.AddItem( PCB_ACTIONS::pushPadSettings, singlePadSel, 400 ); } auto& ctxMenu = m_menu.GetMenu(); // cancel current tool goes in main context menu at the top if present ctxMenu.AddItem( ACTIONS::cancelInteractive, SELECTION_CONDITIONS::ShowAlways, 1 ); ctxMenu.AddSeparator( 1 ); ctxMenu.AddItem( PCB_ACTIONS::rotateCcw, SELECTION_CONDITIONS::ShowAlways ); ctxMenu.AddItem( PCB_ACTIONS::rotateCw, SELECTION_CONDITIONS::ShowAlways ); ctxMenu.AddItem( PCB_ACTIONS::flip, SELECTION_CONDITIONS::ShowAlways ); ctxMenu.AddItem( PCB_ACTIONS::mirrorH, SELECTION_CONDITIONS::ShowAlways ); ctxMenu.AddItem( PCB_ACTIONS::mirrorV, SELECTION_CONDITIONS::ShowAlways ); ctxMenu.AddItem( PCB_ACTIONS::properties, SELECTION_CONDITIONS::ShowAlways ); // Finally, add the standard zoom/grid items getEditFrame()->AddStandardSubMenus( m_menu ); return true; } int PAD_TOOL::pastePadProperties( const TOOL_EVENT& aEvent ) { PCB_SELECTION_TOOL* selTool = m_toolMgr->GetTool(); const PCB_SELECTION& selection = selTool->GetSelection(); const PAD* masterPad = frame()->GetDesignSettings().m_Pad_Master.get(); BOARD_COMMIT commit( frame() ); // for every selected pad, paste global settings for( EDA_ITEM* item : selection ) { if( item->Type() == PCB_PAD_T ) { commit.Modify( item ); static_cast( *item ).ImportSettingsFrom( *masterPad ); } } commit.Push( _( "Paste Pad Properties" ) ); m_toolMgr->ProcessEvent( EVENTS::SelectedItemsModified ); frame()->Refresh(); return 0; } int PAD_TOOL::copyPadSettings( const TOOL_EVENT& aEvent ) { PCB_SELECTION_TOOL* selTool = m_toolMgr->GetTool(); const PCB_SELECTION& selection = selTool->GetSelection(); // can only copy from a single pad if( selection.Size() == 1 ) { EDA_ITEM* item = selection[0]; if( item->Type() == PCB_PAD_T ) { const PAD& selPad = static_cast( *item ); frame()->GetDesignSettings().m_Pad_Master->ImportSettingsFrom( selPad ); } } return 0; } static void doPushPadProperties( BOARD& board, const PAD& aSrcPad, BOARD_COMMIT& commit, bool aSameFootprints, bool aPadShapeFilter, bool aPadOrientFilter, bool aPadLayerFilter, bool aPadTypeFilter ) { const FOOTPRINT* refFootprint = aSrcPad.GetParentFootprint(); EDA_ANGLE srcPadAngle = aSrcPad.GetOrientation() - refFootprint->GetOrientation(); for( FOOTPRINT* footprint : board.Footprints() ) { if( !aSameFootprints && ( footprint != refFootprint ) ) continue; if( footprint->GetFPID() != refFootprint->GetFPID() ) continue; for( PAD* pad : footprint->Pads() ) { if( aPadShapeFilter && ( pad->GetShape() != aSrcPad.GetShape() ) ) continue; EDA_ANGLE padAngle = pad->GetOrientation() - footprint->GetOrientation(); if( aPadOrientFilter && ( padAngle != srcPadAngle ) ) continue; if( aPadLayerFilter && ( pad->GetLayerSet() != aSrcPad.GetLayerSet() ) ) continue; if( aPadTypeFilter && ( pad->GetAttribute() != aSrcPad.GetAttribute() ) ) continue; // Special-case for aperture pads if( aPadTypeFilter && pad->GetAttribute() == PAD_ATTRIB::CONN ) { if( pad->IsAperturePad() != aSrcPad.IsAperturePad() ) continue; } commit.Modify( pad ); // Apply source pad settings to this pad pad->ImportSettingsFrom( aSrcPad ); } } } int PAD_TOOL::pushPadSettings( const TOOL_EVENT& aEvent ) { PCB_SELECTION_TOOL* selTool = m_toolMgr->GetTool(); const PCB_SELECTION& selection = selTool->GetSelection(); if( selection.Size() == 1 && selection[0]->Type() == PCB_PAD_T ) { PAD* srcPad = static_cast( selection[0] ); if( FOOTPRINT* footprint = srcPad->GetParentFootprint() ) { frame()->SetMsgPanel( footprint ); DIALOG_PUSH_PAD_PROPERTIES dlg( frame() ); int dialogRet = dlg.ShowModal(); if( dialogRet == wxID_CANCEL ) return 0; const bool edit_Same_Modules = (dialogRet == 1); BOARD_COMMIT commit( frame() ); doPushPadProperties( *getModel(), *srcPad, commit, edit_Same_Modules, DIALOG_PUSH_PAD_PROPERTIES::m_Pad_Shape_Filter, DIALOG_PUSH_PAD_PROPERTIES::m_Pad_Orient_Filter, DIALOG_PUSH_PAD_PROPERTIES::m_Pad_Layer_Filter, DIALOG_PUSH_PAD_PROPERTIES::m_Pad_Type_Filter ); commit.Push( _( "Push Pad Settings" ) ); m_toolMgr->ProcessEvent( EVENTS::SelectedItemsModified ); frame()->Refresh(); } } return 0; } /** * @brief Prompts the user for parameters for sequential pad numbering * * @param aFrame The parent window for the dialog * @return The parameters, or nullopt if no parameters, e.g. user cancelled the dialog */ static std::optional GetSequentialPadNumberingParams( wxWindow* aFrame ) { // Persistent settings for the pad enumeration dialog. static SEQUENTIAL_PAD_ENUMERATION_PARAMS s_lastUsedParams; DIALOG_ENUM_PADS settingsDlg( aFrame, s_lastUsedParams ); if( settingsDlg.ShowModal() != wxID_OK ) return std::nullopt; return s_lastUsedParams; } int PAD_TOOL::EnumeratePads( const TOOL_EVENT& aEvent ) { if( !m_isFootprintEditor ) return 0; if( !board()->GetFirstFootprint() || board()->GetFirstFootprint()->Pads().empty() ) return 0; GENERAL_COLLECTOR collector; GENERAL_COLLECTORS_GUIDE guide = frame()->GetCollectorsGuide(); guide.SetIgnoreMTextsMarkedNoShow( true ); guide.SetIgnoreMTextsOnBack( true ); guide.SetIgnoreMTextsOnFront( true ); guide.SetIgnoreModulesVals( true ); guide.SetIgnoreModulesRefs( true ); const std::optional params = GetSequentialPadNumberingParams( frame() ); // Cancelled or otherwise failed to get any useful parameters if( !params ) return 0; int seqPadNum = params->m_start_number; std::deque storedPadNumbers; std::map> oldNumbers; m_toolMgr->RunAction( PCB_ACTIONS::selectionClear ); frame()->PushTool( aEvent ); VECTOR2I oldCursorPos; // store the previous mouse cursor position, during mouse drag std::list selectedPads; BOARD_COMMIT commit( frame() ); bool isFirstPoint = true; // make sure oldCursorPos is initialized at least once PADS pads = board()->GetFirstFootprint()->Pads(); MAGNETIC_SETTINGS mag_settings; mag_settings.graphics = false; mag_settings.tracks = MAGNETIC_OPTIONS::NO_EFFECT; mag_settings.pads = MAGNETIC_OPTIONS::CAPTURE_ALWAYS; PCB_GRID_HELPER grid( m_toolMgr, &mag_settings ); grid.SetSnap( true ); grid.SetUseGrid( false ); auto setCursor = [&]() { canvas()->SetCurrentCursor( KICURSOR::BULLSEYE ); }; Activate(); // Must be done after Activate() so that it gets set into the correct context getViewControls()->ShowCursor( true ); getViewControls()->ForceCursorPosition( false ); // Set initial cursor setCursor(); STATUS_TEXT_POPUP statusPopup( frame() ); // Callable lambda to construct the pad number string for the given value const auto constructPadNumber = [&]( int aValue ) { return wxString::Format( wxT( "%s%d" ), params->m_prefix.value_or( "" ), aValue ); }; // Callable lambda to set the popup text for the given pad value const auto setPopupTextForValue = [&]( int aValue ) { const wxString msg = _( "Click on pad %s\n" "Press to cancel all; double-click to finish" ); statusPopup.SetText( wxString::Format( msg, constructPadNumber( aValue ) ) ); }; setPopupTextForValue( seqPadNum ); statusPopup.Popup(); statusPopup.Move( KIPLATFORM::UI::GetMousePosition() + wxPoint( 20, 20 ) ); canvas()->SetStatusPopup( statusPopup.GetPanel() ); while( TOOL_EVENT* evt = Wait() ) { setCursor(); VECTOR2I cursorPos = grid.AlignToNearestPad( getViewControls()->GetMousePosition(), pads ); getViewControls()->ForceCursorPosition( true, cursorPos ); if( evt->IsCancelInteractive() ) { m_toolMgr->RunAction( PCB_ACTIONS::selectionClear ); commit.Revert(); frame()->PopTool( aEvent ); break; } else if( evt->IsActivate() ) { commit.Push( _( "Renumber pads" ) ); frame()->PopTool( aEvent ); break; } else if( evt->IsDrag( BUT_LEFT ) || evt->IsClick( BUT_LEFT ) ) { selectedPads.clear(); // Be sure the old cursor mouse position was initialized: if( isFirstPoint ) { oldCursorPos = cursorPos; isFirstPoint = false; } // wxWidgets deliver mouse move events not frequently enough, resulting in skipping // pads if the user moves cursor too fast. To solve it, create a line that approximates // the mouse move and search pads that are on the line. int distance = ( cursorPos - oldCursorPos ).EuclideanNorm(); // Search will be made every 0.1 mm: int segments = distance / int( 0.1 * pcbIUScale.IU_PER_MM ) + 1; const VECTOR2I line_step( ( cursorPos - oldCursorPos ) / segments ); collector.Empty(); for( int j = 0; j < segments; ++j ) { VECTOR2I testpoint( cursorPos.x - j * line_step.x, cursorPos.y - j * line_step.y ); collector.Collect( board(), { PCB_PAD_T }, testpoint, guide ); for( int i = 0; i < collector.GetCount(); ++i ) selectedPads.push_back( static_cast( collector[i] ) ); } selectedPads.unique(); for( PAD* pad : selectedPads ) { // If pad was not selected, then enumerate it if( !pad->IsSelected() ) { commit.Modify( pad ); // Rename pad and store the old name int newval; if( storedPadNumbers.size() > 0 ) { newval = storedPadNumbers.front(); storedPadNumbers.pop_front(); } else { newval = seqPadNum; seqPadNum += params->m_step; } const wxString newNumber = constructPadNumber( newval ); oldNumbers[newNumber] = { newval, pad->GetNumber() }; pad->SetNumber( newNumber ); SetLastPadNumber( newNumber ); pad->SetSelected(); getView()->Update( pad ); // Ensure the popup text shows the correct next value if( storedPadNumbers.size() > 0 ) newval = storedPadNumbers.front(); else newval = seqPadNum; setPopupTextForValue( newval ); } // ... or restore the old name if it was enumerated and clicked again else if( pad->IsSelected() && evt->IsClick( BUT_LEFT ) ) { auto it = oldNumbers.find( pad->GetNumber() ); wxASSERT( it != oldNumbers.end() ); if( it != oldNumbers.end() ) { storedPadNumbers.push_back( it->second.first ); pad->SetNumber( it->second.second ); SetLastPadNumber( it->second.second ); oldNumbers.erase( it ); const int newval = storedPadNumbers.front(); setPopupTextForValue( newval ); } pad->ClearSelected(); getView()->Update( pad ); } } } else if( evt->IsDblClick( BUT_LEFT ) ) { commit.Push( _( "Renumber pads" ) ); frame()->PopTool( aEvent ); break; } else if( evt->IsClick( BUT_RIGHT ) ) { m_menu.ShowContextMenu( selection() ); } else { evt->SetPassEvent(); } // Prepare the next loop by updating the old cursor mouse position // to this last mouse cursor position oldCursorPos = getViewControls()->GetCursorPosition(); statusPopup.Move( KIPLATFORM::UI::GetMousePosition() + wxPoint( 20, 20 ) ); } for( PAD* p : board()->GetFirstFootprint()->Pads() ) { p->ClearSelected(); getView()->Update( p ); } canvas()->SetStatusPopup( nullptr ); statusPopup.Hide(); canvas()->SetCurrentCursor( KICURSOR::ARROW ); getViewControls()->ForceCursorPosition( false ); return 0; } int PAD_TOOL::PlacePad( const TOOL_EVENT& aEvent ) { // When creating a new pad (in FP editor) we can use a new pad number // or the last entered pad number // neednewPadNumber = true to create a new pad number, false to use the last // entered pad number static bool neednewPadNumber; if( !m_isFootprintEditor ) return 0; if( !board()->GetFirstFootprint() ) return 0; struct PAD_PLACER : public INTERACTIVE_PLACER_BASE { PAD_PLACER( PAD_TOOL* aPadTool ) { neednewPadNumber = true; // Use a new pad number when creatin a pad by default m_padTool = aPadTool; } virtual ~PAD_PLACER() { } std::unique_ptr CreateItem() override { PAD* pad = new PAD( m_board->GetFirstFootprint() ); PAD* master = m_frame->GetDesignSettings().m_Pad_Master.get(); pad->ImportSettingsFrom( *master ); // If the footprint type and master pad type directly conflict then make some // adjustments. Otherwise assume the user set what they wanted. // Note also a HEATSINK pad (thermal via) is allowed in SMD footprint if( ( m_board->GetFirstFootprint()->GetAttributes() & FP_SMD ) && master->GetAttribute() == PAD_ATTRIB::PTH ) { if( pad->GetProperty() != PAD_PROP::HEATSINK ) { pad->SetAttribute( PAD_ATTRIB::SMD ); pad->SetShape( PAD_SHAPE::ROUNDRECT ); pad->SetSizeX( 1.5 * pad->GetSizeY() ); pad->SetLayerSet( PAD::SMDMask() ); } } else if( ( m_board->GetFirstFootprint()->GetAttributes() & FP_THROUGH_HOLE ) && master->GetAttribute() == PAD_ATTRIB::SMD ) { pad->SetAttribute( PAD_ATTRIB::PTH ); pad->SetShape( PAD_SHAPE::CIRCLE ); pad->SetSize( VECTOR2I( pad->GetSizeX(), pad->GetSizeX() ) ); // Gives an acceptable drill size: it cannot be 0, but from pad master // it is currently 0, therefore change it: pad->SetDrillShape( PAD_DRILL_SHAPE_CIRCLE ); int hole_size = pad->GetSizeX() / 2; pad->SetDrillSize( VECTOR2I( hole_size, hole_size ) ); pad->SetLayerSet( PAD::PTHMask() ); } if( pad->CanHaveNumber() ) { wxString padNumber = m_padTool->GetLastPadNumber(); // Use the last entered pad number when recreating a pad without using the // previously created pad, and a new number when creating a really new pad if( neednewPadNumber ) padNumber = m_board->GetFirstFootprint()->GetNextPadNumber( padNumber ); pad->SetNumber( padNumber ); m_padTool->SetLastPadNumber( padNumber ); // If a pad is recreated and the previously created was not placed, use // the last entered pad number neednewPadNumber = false; } return std::unique_ptr( pad ); } bool PlaceItem( BOARD_ITEM *aItem, BOARD_COMMIT& aCommit ) override { PAD* pad = dynamic_cast( aItem ); // We are using this pad number. // therefore use a new pad number for a newly created pad neednewPadNumber = true; if( pad ) { m_frame->GetDesignSettings().m_Pad_Master->ImportSettingsFrom( *pad ); aCommit.Add( aItem ); return true; } return false; } PAD_TOOL* m_padTool; }; PAD_PLACER placer( this ); doInteractiveItemPlacement( aEvent, &placer, _( "Place pad" ), IPO_REPEAT | IPO_SINGLE_CLICK | IPO_ROTATE | IPO_FLIP ); return 0; } int PAD_TOOL::EditPad( const TOOL_EVENT& aEvent ) { if( !m_isFootprintEditor ) return 0; Activate(); KIGFX::PCB_PAINTER* painter = static_cast( view()->GetPainter() ); PCB_RENDER_SETTINGS* settings = painter->GetSettings(); PCB_SELECTION& selection = m_toolMgr->GetTool()->GetSelection(); if( m_editPad != niluuid ) { PAD* pad = dynamic_cast( frame()->GetItem( m_editPad ) ); if( pad ) { BOARD_COMMIT commit( frame() ); commit.Modify( pad->GetParentFootprint() ); std::vector shapes = RecombinePad( pad, false ); for( PCB_SHAPE* shape : shapes ) commit.Remove( shape ); commit.Push( _( "Edit Pad" ) ); } m_editPad = niluuid; } else if( selection.Size() == 1 && selection[0]->Type() == PCB_PAD_T ) { PCB_LAYER_ID layer; PAD* pad = static_cast( selection[0] ); BOARD_COMMIT commit( frame() ); commit.Modify( pad->GetParentFootprint() ); explodePad( pad, &layer ); commit.Push( _( "Edit Pad" ) ); m_toolMgr->RunAction( PCB_ACTIONS::selectionClear ); frame()->SetActiveLayer( layer ); settings->m_PadEditModePad = pad; enterPadEditMode(); } if( m_editPad == niluuid ) { settings->m_PadEditModePad = nullptr; exitPadEditMode(); } return 0; } int PAD_TOOL::OnUndoRedo( const TOOL_EVENT& aEvent ) { PAD* flaggedPad = nullptr; KIID flaggedPadId = niluuid; for( FOOTPRINT* fp : board()->Footprints() ) { for( PAD* pad : fp->Pads() ) { if( pad->IsEntered() ) { flaggedPad = pad; flaggedPadId = pad->m_Uuid; break; } } } if( flaggedPadId != m_editPad ) { KIGFX::PCB_PAINTER* painter = static_cast( view()->GetPainter() ); PCB_RENDER_SETTINGS* settings = painter->GetSettings(); m_editPad = flaggedPadId; settings->m_PadEditModePad = flaggedPad; if( flaggedPad ) enterPadEditMode(); else exitPadEditMode(); } return 0; } void PAD_TOOL::enterPadEditMode() { PCB_DISPLAY_OPTIONS opts = frame()->GetDisplayOptions(); WX_INFOBAR* infoBar = frame()->GetInfoBar(); wxString msg; canvas()->GetView()->UpdateAllItemsConditionally( KIGFX::REPAINT, [&]( KIGFX::VIEW_ITEM* aItem ) -> bool { return dynamic_cast( aItem ) != nullptr; } ); m_previousHighContrastMode = opts.m_ContrastModeDisplay; if( opts.m_ContrastModeDisplay == HIGH_CONTRAST_MODE::NORMAL ) { opts.m_ContrastModeDisplay = HIGH_CONTRAST_MODE::DIMMED; frame()->SetDisplayOptions( opts ); } if( PCB_ACTIONS::explodePad.GetHotKey() == PCB_ACTIONS::recombinePad.GetHotKey() ) { msg.Printf( _( "Pad Edit Mode. Press %s again to exit." ), KeyNameFromKeyCode( PCB_ACTIONS::recombinePad.GetHotKey() ) ); } else { msg.Printf( _( "Pad Edit Mode. Press %s to exit." ), KeyNameFromKeyCode( PCB_ACTIONS::recombinePad.GetHotKey() ) ); } infoBar->RemoveAllButtons(); infoBar->ShowMessage( msg, wxICON_INFORMATION ); } void PAD_TOOL::exitPadEditMode() { PCB_DISPLAY_OPTIONS opts = frame()->GetDisplayOptions(); if( m_previousHighContrastMode != opts.m_ContrastModeDisplay ) { opts.m_ContrastModeDisplay = m_previousHighContrastMode; frame()->SetDisplayOptions( opts ); } // Note: KIGFX::REPAINT isn't enough for things that go from invisible to visible as // they won't be found in the view layer's itemset for re-painting. canvas()->GetView()->UpdateAllItemsConditionally( KIGFX::ALL, [&]( KIGFX::VIEW_ITEM* aItem ) -> bool { return dynamic_cast( aItem ) != nullptr; } ); // Refresh now (otherwise there's an uncomfortably long pause while the infoBar // closes before refresh). canvas()->ForceRefresh(); frame()->GetInfoBar()->Dismiss(); } void PAD_TOOL::explodePad( PAD* aPad, PCB_LAYER_ID* aLayer ) { if( aPad->IsOnLayer( F_Cu ) ) *aLayer = F_Cu; else if( aPad->IsOnLayer( B_Cu ) ) *aLayer = B_Cu; else *aLayer = *aPad->GetLayerSet().UIOrder(); if( aPad->GetShape() == PAD_SHAPE::CUSTOM ) { for( const std::shared_ptr& primitive : aPad->GetPrimitives() ) { PCB_SHAPE* shape = static_cast( primitive->Duplicate() ); shape->SetParent( board()->GetFirstFootprint() ); shape->Rotate( VECTOR2I( 0, 0 ), aPad->GetOrientation() ); shape->Move( aPad->ShapePos() ); shape->SetLayer( *aLayer ); if( shape->IsProxyItem() && shape->GetShape() == SHAPE_T::SEGMENT ) { if( aPad->GetThermalSpokeWidth() ) shape->SetWidth( aPad->GetThermalSpokeWidth() ); else shape->SetWidth( pcbIUScale.mmToIU( ZONE_THERMAL_RELIEF_COPPER_WIDTH_MM ) ); } board()->GetFirstFootprint()->Add( shape ); frame()->GetCanvas()->GetView()->Add( shape ); } aPad->SetShape( aPad->GetAnchorPadShape() ); aPad->DeletePrimitivesList(); } aPad->SetFlags( ENTERED ); m_editPad = aPad->m_Uuid; } std::vector PAD_TOOL::RecombinePad( PAD* aPad, bool aIsDryRun ) { int maxError = board()->GetDesignSettings().m_MaxError; FOOTPRINT* footprint = aPad->GetParentFootprint(); // Don't leave an object in the point editor that might no longer exist after recombining. m_toolMgr->RunAction( PCB_ACTIONS::selectionClear ); for( BOARD_ITEM* item : footprint->GraphicalItems() ) item->ClearFlags( SKIP_STRUCT ); auto findNext = [&]( PCB_LAYER_ID aLayer ) -> PCB_SHAPE* { SHAPE_POLY_SET padPoly; aPad->TransformShapeToPolygon( padPoly, aLayer, 0, maxError, ERROR_INSIDE ); for( BOARD_ITEM* item : footprint->GraphicalItems() ) { PCB_SHAPE* shape = dynamic_cast( item ); if( !shape || ( shape->GetFlags() & SKIP_STRUCT ) ) continue; if( shape->GetLayer() != aLayer ) continue; if( shape->IsProxyItem() ) // Pad number (and net name) box return shape; SHAPE_POLY_SET drawPoly; shape->TransformShapeToPolygon( drawPoly, aLayer, 0, maxError, ERROR_INSIDE ); drawPoly.BooleanIntersection( padPoly, SHAPE_POLY_SET::PM_FAST ); if( !drawPoly.IsEmpty() ) return shape; } return nullptr; }; auto findMatching = [&]( PCB_SHAPE* aShape ) -> std::vector { std::vector matching; for( BOARD_ITEM* item : footprint->GraphicalItems() ) { PCB_SHAPE* other = dynamic_cast( item ); if( !other || ( other->GetFlags() & SKIP_STRUCT ) ) continue; if( aPad->GetLayerSet().test( other->GetLayer() ) && aShape->Compare( other ) == 0 ) { matching.push_back( other ); } } return matching; }; PCB_LAYER_ID layer; std::vector mergedShapes; if( aPad->IsOnLayer( F_Cu ) ) layer = F_Cu; else if( aPad->IsOnLayer( B_Cu ) ) layer = B_Cu; else layer = *aPad->GetLayerSet().UIOrder(); // If there are intersecting items to combine, we need to first make sure the pad is a // custom-shape pad. if( !aIsDryRun && findNext( layer ) && aPad->GetShape() != PAD_SHAPE::CUSTOM ) { if( aPad->GetShape() == PAD_SHAPE::CIRCLE || aPad->GetShape() == PAD_SHAPE::RECTANGLE ) { // Use the existing pad as an anchor aPad->SetAnchorPadShape( aPad->GetShape() ); aPad->SetShape( PAD_SHAPE::CUSTOM ); } else { // Create a new circular anchor and convert existing pad to a polygon primitive SHAPE_POLY_SET existingOutline; aPad->TransformShapeToPolygon( existingOutline, layer, 0, maxError, ERROR_INSIDE ); int minExtent = std::min( aPad->GetSize().x, aPad->GetSize().y ); aPad->SetAnchorPadShape( PAD_SHAPE::CIRCLE ); aPad->SetSize( VECTOR2I( minExtent, minExtent ) ); aPad->SetShape( PAD_SHAPE::CUSTOM ); PCB_SHAPE* shape = new PCB_SHAPE( nullptr, SHAPE_T::POLY ); shape->SetFilled( true ); shape->SetStroke( STROKE_PARAMS( 0, LINE_STYLE::SOLID ) ); shape->SetPolyShape( existingOutline ); shape->Rotate( VECTOR2I( 0, 0 ), - aPad->GetOrientation() ); shape->Move( - aPad->ShapePos() ); aPad->AddPrimitive( shape ); } } while( PCB_SHAPE* fpShape = findNext( layer ) ) { fpShape->SetFlags( SKIP_STRUCT ); mergedShapes.push_back( fpShape ); if( !aIsDryRun ) { PCB_SHAPE* primitive = static_cast( fpShape->Duplicate() ); primitive->SetParent( nullptr ); primitive->Move( - aPad->ShapePos() ); primitive->Rotate( VECTOR2I( 0, 0 ), - aPad->GetOrientation() ); aPad->AddPrimitive( primitive ); } // See if there are other shapes that match and mark them for delete. (KiCad won't // produce these, but old footprints from other vendors have them.) for( PCB_SHAPE* other : findMatching( fpShape ) ) { other->SetFlags( SKIP_STRUCT ); mergedShapes.push_back( other ); } } for( BOARD_ITEM* item : footprint->GraphicalItems() ) item->ClearFlags( SKIP_STRUCT ); if( !aIsDryRun ) aPad->ClearFlags( ENTERED ); return mergedShapes; } void PAD_TOOL::setTransitions() { Go( &PAD_TOOL::pastePadProperties, PCB_ACTIONS::applyPadSettings.MakeEvent() ); Go( &PAD_TOOL::copyPadSettings, PCB_ACTIONS::copyPadSettings.MakeEvent() ); Go( &PAD_TOOL::pushPadSettings, PCB_ACTIONS::pushPadSettings.MakeEvent() ); Go( &PAD_TOOL::PlacePad, PCB_ACTIONS::placePad.MakeEvent() ); Go( &PAD_TOOL::EnumeratePads, PCB_ACTIONS::enumeratePads.MakeEvent() ); Go( &PAD_TOOL::EditPad, PCB_ACTIONS::explodePad.MakeEvent() ); Go( &PAD_TOOL::EditPad, PCB_ACTIONS::recombinePad.MakeEvent() ); Go( &PAD_TOOL::OnUndoRedo, EVENTS::UndoRedoPostEvent ); }