724 lines
24 KiB
C++
724 lines
24 KiB
C++
/*
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
*
|
|
* Copyright (C) 2014-2019 CERN
|
|
* @author Maciej Suminski <maciej.suminski@cern.ch>
|
|
*
|
|
* 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 "footprint_editor_tools.h"
|
|
#include "kicad_clipboard.h"
|
|
#include "selection_tool.h"
|
|
#include "pcb_actions.h"
|
|
#include <core/optional.h>
|
|
#include <tool/tool_manager.h>
|
|
#include <class_draw_panel_gal.h>
|
|
#include <view/view_controls.h>
|
|
#include <view/view_group.h>
|
|
#include <pcb_painter.h>
|
|
#include <origin_viewitem.h>
|
|
#include <status_popup.h>
|
|
#include <footprint_edit_frame.h>
|
|
#include <kicad_plugin.h>
|
|
#include <pcbnew_id.h>
|
|
#include <collectors.h>
|
|
#include <confirm.h>
|
|
#include <dialogs/dialog_enum_pads.h>
|
|
#include <hotkeys.h>
|
|
#include <bitmaps.h>
|
|
#include <pcb_edit_frame.h>
|
|
#include <class_board.h>
|
|
#include <class_module.h>
|
|
#include <class_edge_mod.h>
|
|
#include <board_commit.h>
|
|
|
|
#include <tools/tool_event_utils.h>
|
|
|
|
#include <functional>
|
|
using namespace std::placeholders;
|
|
#include <wx/defs.h>
|
|
|
|
|
|
TOOL_ACTION PCB_ACTIONS::newFootprint( "pcbnew.ModuleEditor.newFootprint",
|
|
AS_GLOBAL, TOOL_ACTION::LegacyHotKey( HK_NEW ),
|
|
_( "New Footprint..." ), _( "Create a new, empty footprint" ),
|
|
new_footprint_xpm );
|
|
|
|
TOOL_ACTION PCB_ACTIONS::createFootprint( "pcbnew.ModuleEditor.createFootprint",
|
|
AS_GLOBAL, 0,
|
|
_( "Create Footprint..." ), _( "Create a new footprint using the Footprint Wizard" ),
|
|
module_wizard_xpm );
|
|
|
|
TOOL_ACTION PCB_ACTIONS::saveToBoard( "pcbnew.ModuleEditor.saveToBoard",
|
|
AS_GLOBAL, 0,
|
|
_( "Save to Board" ), _( "Update footprint on board" ),
|
|
save_fp_to_board_xpm );
|
|
|
|
TOOL_ACTION PCB_ACTIONS::saveToLibrary( "pcbnew.ModuleEditor.saveToLibrary",
|
|
AS_GLOBAL, 0,
|
|
_( "Save to Library" ), _( "Save changes to library" ),
|
|
save_xpm );
|
|
|
|
// Module editor tools
|
|
TOOL_ACTION PCB_ACTIONS::footprintProperties( "pcbnew.ModuleEditor.footprintProperties",
|
|
AS_GLOBAL, 0,
|
|
_( "Footprint Properties..." ), "",
|
|
module_options_xpm );
|
|
|
|
TOOL_ACTION PCB_ACTIONS::deleteFootprint( "pcbnew.ModuleEditor.deleteFootprint",
|
|
AS_GLOBAL, 0,
|
|
_( "Delete Footprint from Library" ), "",
|
|
delete_xpm );
|
|
|
|
TOOL_ACTION PCB_ACTIONS::placePad( "pcbnew.ModuleEditor.placePad",
|
|
AS_GLOBAL, 0,
|
|
_( "Add Pad" ), _( "Add a pad" ),
|
|
pad_xpm, AF_ACTIVATE );
|
|
|
|
TOOL_ACTION PCB_ACTIONS::createPadFromShapes( "pcbnew.ModuleEditor.createPadFromShapes",
|
|
AS_CONTEXT, 0,
|
|
_( "Create Pad from Selected Shapes" ),
|
|
_( "Creates a custom-shaped pads from a set of selected shapes" ),
|
|
primitives_to_custom_pad_xpm );
|
|
|
|
TOOL_ACTION PCB_ACTIONS::explodePadToShapes( "pcbnew.ModuleEditor.explodePadToShapes",
|
|
AS_CONTEXT, 0,
|
|
_( "Explode Pad to Graphic Shapes" ),
|
|
_( "Converts a custom-shaped pads to a set of graphical shapes" ),
|
|
custom_pad_to_primitives_xpm );
|
|
|
|
TOOL_ACTION PCB_ACTIONS::enumeratePads( "pcbnew.ModuleEditor.enumeratePads",
|
|
AS_GLOBAL, 0,
|
|
_( "Renumber Pads..." ), _( "Renumber pads by clicking on them in the desired order" ),
|
|
pad_enumerate_xpm, AF_ACTIVATE );
|
|
|
|
TOOL_ACTION PCB_ACTIONS::defaultPadProperties( "pcbnew.ModuleEditor.defaultPadProperties",
|
|
AS_GLOBAL, 0,
|
|
_( "Default Pad Properties..." ), _( "Edit the pad properties used when creating new pads" ),
|
|
options_pad_xpm );
|
|
|
|
|
|
MODULE_EDITOR_TOOLS::MODULE_EDITOR_TOOLS() :
|
|
PCB_TOOL_BASE( "pcbnew.ModuleEditor" ),
|
|
m_frame( nullptr )
|
|
{
|
|
}
|
|
|
|
|
|
MODULE_EDITOR_TOOLS::~MODULE_EDITOR_TOOLS()
|
|
{
|
|
}
|
|
|
|
|
|
void MODULE_EDITOR_TOOLS::Reset( RESET_REASON aReason )
|
|
{
|
|
m_frame = getEditFrame<FOOTPRINT_EDIT_FRAME>();
|
|
}
|
|
|
|
|
|
bool MODULE_EDITOR_TOOLS::Init()
|
|
{
|
|
// Build a context menu for the footprint tree
|
|
//
|
|
CONDITIONAL_MENU& ctxMenu = m_menu.GetMenu();
|
|
|
|
auto libSelectedCondition = [ this ] ( const SELECTION& aSel ) {
|
|
LIB_ID sel = m_frame->GetTreeFPID();
|
|
return !sel.GetLibNickname().empty() && sel.GetLibItemName().empty();
|
|
};
|
|
auto fpSelectedCondition = [ this ] ( const SELECTION& aSel ) {
|
|
LIB_ID sel = m_frame->GetTreeFPID();
|
|
return !sel.GetLibNickname().empty() && !sel.GetLibItemName().empty();
|
|
};
|
|
|
|
ctxMenu.AddItem( ACTIONS::newLibrary, SELECTION_CONDITIONS::ShowAlways );
|
|
ctxMenu.AddItem( ACTIONS::addLibrary, SELECTION_CONDITIONS::ShowAlways );
|
|
ctxMenu.AddItem( ACTIONS::save, libSelectedCondition );
|
|
ctxMenu.AddItem( ACTIONS::saveAs, libSelectedCondition );
|
|
ctxMenu.AddItem( ACTIONS::revert, libSelectedCondition );
|
|
|
|
ctxMenu.AddSeparator( SELECTION_CONDITIONS::ShowAlways );
|
|
ctxMenu.AddItem( PCB_ACTIONS::newFootprint, SELECTION_CONDITIONS::ShowAlways );
|
|
#ifdef KICAD_SCRIPTING
|
|
ctxMenu.AddItem( PCB_ACTIONS::createFootprint, SELECTION_CONDITIONS::ShowAlways );
|
|
#endif
|
|
ctxMenu.AddItem( ID_MODEDIT_EDIT_MODULE,
|
|
_( "Edit Footprint" ), _( "Show selected footprint on editor canvas" ),
|
|
edit_xpm, fpSelectedCondition );
|
|
|
|
ctxMenu.AddSeparator( SELECTION_CONDITIONS::ShowAlways );
|
|
ctxMenu.AddItem( ACTIONS::save, fpSelectedCondition );
|
|
ctxMenu.AddItem( ACTIONS::saveCopyAs, fpSelectedCondition );
|
|
ctxMenu.AddItem( PCB_ACTIONS::deleteFootprint, fpSelectedCondition );
|
|
ctxMenu.AddItem( ACTIONS::revert, fpSelectedCondition );
|
|
|
|
ctxMenu.AddSeparator( SELECTION_CONDITIONS::ShowAlways );
|
|
ctxMenu.AddItem( ID_MODEDIT_CUT_PART, _( "Cut Footprint" ), "",
|
|
cut_xpm, fpSelectedCondition );
|
|
ctxMenu.AddItem( ID_MODEDIT_COPY_PART, _( "Copy Footprint" ), "",
|
|
copy_xpm, fpSelectedCondition );
|
|
ctxMenu.AddItem( ID_MODEDIT_PASTE_PART, _( "Paste Footprint" ), "",
|
|
paste_xpm, SELECTION_CONDITIONS::ShowAlways );
|
|
|
|
ctxMenu.AddSeparator( fpSelectedCondition );
|
|
ctxMenu.AddItem( ID_MODEDIT_IMPORT_PART, _( "Import Footprint..." ), "",
|
|
import_module_xpm, SELECTION_CONDITIONS::ShowAlways );
|
|
ctxMenu.AddItem( ID_MODEDIT_EXPORT_PART, _( "Export Footprint..." ), "",
|
|
export_module_xpm, fpSelectedCondition );
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
int MODULE_EDITOR_TOOLS::NewFootprint( const TOOL_EVENT& aEvent )
|
|
{
|
|
wxCommandEvent evt( wxEVT_NULL, ID_MODEDIT_NEW_MODULE );
|
|
getEditFrame<FOOTPRINT_EDIT_FRAME>()->Process_Special_Functions( evt );
|
|
return 0;
|
|
}
|
|
|
|
|
|
int MODULE_EDITOR_TOOLS::CreateFootprint( const TOOL_EVENT& aEvent )
|
|
{
|
|
wxCommandEvent evt( wxEVT_NULL, ID_MODEDIT_NEW_MODULE_FROM_WIZARD );
|
|
getEditFrame<FOOTPRINT_EDIT_FRAME>()->Process_Special_Functions( evt );
|
|
return 0;
|
|
}
|
|
|
|
|
|
int MODULE_EDITOR_TOOLS::Save( const TOOL_EVENT& aEvent )
|
|
{
|
|
wxCommandEvent evt( wxEVT_NULL, ID_MODEDIT_SAVE );
|
|
getEditFrame<FOOTPRINT_EDIT_FRAME>()->Process_Special_Functions( evt );
|
|
return 0;
|
|
}
|
|
|
|
|
|
int MODULE_EDITOR_TOOLS::SaveAs( const TOOL_EVENT& aEvent )
|
|
{
|
|
wxCommandEvent evt( wxEVT_NULL, ID_MODEDIT_SAVE_AS );
|
|
getEditFrame<FOOTPRINT_EDIT_FRAME>()->Process_Special_Functions( evt );
|
|
return 0;
|
|
}
|
|
|
|
|
|
int MODULE_EDITOR_TOOLS::Revert( const TOOL_EVENT& aEvent )
|
|
{
|
|
getEditFrame<FOOTPRINT_EDIT_FRAME>()->RevertFootprint();
|
|
return 0;
|
|
}
|
|
|
|
|
|
int MODULE_EDITOR_TOOLS::Delete( const TOOL_EVENT& aEvent )
|
|
{
|
|
FOOTPRINT_EDIT_FRAME* frame = getEditFrame<FOOTPRINT_EDIT_FRAME>();
|
|
|
|
if( frame->DeleteModuleFromLibrary( frame->GetTargetFPID(), true ) )
|
|
{
|
|
if( frame->GetTargetFPID() == frame->GetLoadedFPID() )
|
|
frame->Clear_Pcb( false );
|
|
|
|
frame->SyncLibraryTree( true );
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int MODULE_EDITOR_TOOLS::Properties( const TOOL_EVENT& aEvent )
|
|
{
|
|
MODULE* footprint = m_frame->GetBoard()->GetFirstModule();
|
|
|
|
if( footprint )
|
|
{
|
|
getEditFrame<FOOTPRINT_EDIT_FRAME>()->OnEditItemRequest( footprint );
|
|
m_frame->GetGalCanvas()->Refresh();
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
int MODULE_EDITOR_TOOLS::DefaultPadProperties( const TOOL_EVENT& aEvent )
|
|
{
|
|
getEditFrame<FOOTPRINT_EDIT_FRAME>()->InstallPadOptionsFrame( nullptr );
|
|
return 0;
|
|
}
|
|
|
|
|
|
int MODULE_EDITOR_TOOLS::PlacePad( const TOOL_EVENT& aEvent )
|
|
{
|
|
struct PAD_PLACER : public INTERACTIVE_PLACER_BASE
|
|
{
|
|
std::unique_ptr<BOARD_ITEM> CreateItem() override
|
|
{
|
|
D_PAD* pad = new D_PAD( m_board->GetFirstModule() );
|
|
m_frame->Import_Pad_Settings( pad, false ); // use the global settings for pad
|
|
pad->IncrementPadName( true, true );
|
|
return std::unique_ptr<BOARD_ITEM>( pad );
|
|
}
|
|
|
|
bool PlaceItem( BOARD_ITEM *aItem, BOARD_COMMIT& aCommit ) override
|
|
{
|
|
D_PAD* pad = dynamic_cast<D_PAD*>( aItem );
|
|
|
|
if( pad )
|
|
{
|
|
m_frame->Export_Pad_Settings( pad );
|
|
pad->SetLocalCoord();
|
|
aCommit.Add( aItem );
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
};
|
|
|
|
PAD_PLACER placer;
|
|
|
|
m_frame->SetToolID( ID_MODEDIT_PAD_TOOL, wxCURSOR_PENCIL, _( "Add pads" ) );
|
|
|
|
wxASSERT( board()->GetFirstModule() );
|
|
|
|
doInteractiveItemPlacement( &placer, _( "Place pad" ), IPO_REPEAT | IPO_SINGLE_CLICK | IPO_ROTATE | IPO_FLIP );
|
|
|
|
m_frame->SetNoToolSelected();
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int MODULE_EDITOR_TOOLS::EnumeratePads( const TOOL_EVENT& aEvent )
|
|
{
|
|
if( !board()->GetFirstModule() || !board()->GetFirstModule()->Pads().empty() )
|
|
return 0;
|
|
|
|
DIALOG_ENUM_PADS settingsDlg( m_frame );
|
|
|
|
if( settingsDlg.ShowModal() != wxID_OK )
|
|
return 0;
|
|
|
|
Activate();
|
|
|
|
GENERAL_COLLECTOR collector;
|
|
const KICAD_T types[] = { PCB_PAD_T, EOT };
|
|
|
|
GENERAL_COLLECTORS_GUIDE guide = m_frame->GetCollectorsGuide();
|
|
guide.SetIgnoreMTextsMarkedNoShow( true );
|
|
guide.SetIgnoreMTextsOnBack( true );
|
|
guide.SetIgnoreMTextsOnFront( true );
|
|
guide.SetIgnoreModulesVals( true );
|
|
guide.SetIgnoreModulesRefs( true );
|
|
|
|
int seqPadNum = settingsDlg.GetStartNumber();
|
|
wxString padPrefix = settingsDlg.GetPrefix();
|
|
std::deque<int> storedPadNumbers;
|
|
|
|
m_frame->SetToolID( ID_MODEDIT_PAD_TOOL, wxCURSOR_HAND,
|
|
_( "Click on successive pads to renumber them" ) );
|
|
|
|
m_toolMgr->RunAction( PCB_ACTIONS::selectionClear, true );
|
|
getViewControls()->ShowCursor( true );
|
|
|
|
KIGFX::VIEW* view = m_toolMgr->GetView();
|
|
VECTOR2I oldCursorPos; // store the previous mouse cursor position, during mouse drag
|
|
std::list<D_PAD*> selectedPads;
|
|
BOARD_COMMIT commit( m_frame );
|
|
std::map<wxString, std::pair<int, wxString>> oldNames;
|
|
bool isFirstPoint = true; // used to be sure oldCursorPos will be initialized at least once.
|
|
|
|
STATUS_TEXT_POPUP statusPopup( m_frame );
|
|
statusPopup.SetText( wxString::Format(
|
|
_( "Click on pad %s%d\nPress Escape to cancel or double-click to commit" ),
|
|
padPrefix.c_str(), seqPadNum ) );
|
|
statusPopup.Popup();
|
|
statusPopup.Move( wxGetMousePosition() + wxPoint( 20, 20 ) );
|
|
|
|
while( OPT_TOOL_EVENT evt = Wait() )
|
|
{
|
|
if( evt->IsDrag( BUT_LEFT ) || evt->IsClick( BUT_LEFT ) )
|
|
{
|
|
selectedPads.clear();
|
|
VECTOR2I cursorPos = getViewControls()->GetCursorPosition();
|
|
|
|
// 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*IU_PER_MM ) + 1;
|
|
const wxPoint line_step( ( cursorPos - oldCursorPos ) / segments );
|
|
|
|
collector.Empty();
|
|
|
|
for( int j = 0; j < segments; ++j )
|
|
{
|
|
wxPoint testpoint( cursorPos.x - j * line_step.x,
|
|
cursorPos.y - j * line_step.y );
|
|
collector.Collect( board(), types, testpoint, guide );
|
|
|
|
for( int i = 0; i < collector.GetCount(); ++i )
|
|
{
|
|
selectedPads.push_back( static_cast<D_PAD*>( collector[i] ) );
|
|
}
|
|
}
|
|
|
|
selectedPads.unique();
|
|
|
|
for( D_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++;
|
|
|
|
wxString newName = wxString::Format( wxT( "%s%d" ), padPrefix.c_str(), newval );
|
|
oldNames[newName] = { newval, pad->GetName() };
|
|
pad->SetName( newName );
|
|
pad->SetSelected();
|
|
getView()->Update( pad );
|
|
|
|
// Ensure the popup text shows the correct next value
|
|
if( storedPadNumbers.size() > 0 )
|
|
newval = storedPadNumbers.front();
|
|
else
|
|
newval = seqPadNum;
|
|
|
|
statusPopup.SetText( wxString::Format(
|
|
_( "Click on pad %s%d\nPress Escape to cancel or double-click to commit" ),
|
|
padPrefix.c_str(), newval ) );
|
|
}
|
|
|
|
// ..or restore the old name if it was enumerated and clicked again
|
|
else if( pad->IsSelected() && evt->IsClick( BUT_LEFT ) )
|
|
{
|
|
auto it = oldNames.find( pad->GetName() );
|
|
wxASSERT( it != oldNames.end() );
|
|
|
|
if( it != oldNames.end() )
|
|
{
|
|
storedPadNumbers.push_back( it->second.first );
|
|
pad->SetName( it->second.second );
|
|
oldNames.erase( it );
|
|
|
|
statusPopup.SetText( wxString::Format(
|
|
_( "Click on pad %s%d\nPress Escape to cancel or double-click to commit" ),
|
|
padPrefix.c_str(), storedPadNumbers.front() ) );
|
|
}
|
|
|
|
pad->ClearSelected();
|
|
getView()->Update( pad );
|
|
}
|
|
}
|
|
}
|
|
|
|
else if( ( evt->IsKeyPressed() && evt->KeyCode() == WXK_RETURN ) ||
|
|
evt->IsDblClick( BUT_LEFT ) )
|
|
{
|
|
commit.Push( _( "Renumber pads" ) );
|
|
break;
|
|
}
|
|
|
|
// This is a cancel-current-action (ie: <esc>).
|
|
// Note that this must go before IsCancelInteractive() as it also checks IsCancel().
|
|
else if( evt->IsCancel() )
|
|
{
|
|
// Clear current selection list to avoid selection of deleted items
|
|
m_toolMgr->RunAction( PCB_ACTIONS::selectionClear, true );
|
|
|
|
commit.Revert();
|
|
break;
|
|
}
|
|
|
|
// Now that cancel-current-action has been handled, check for cancel-tool.
|
|
else if( TOOL_EVT_UTILS::IsCancelInteractive( *evt ) )
|
|
{
|
|
commit.Push( _( "Renumber pads" ) );
|
|
break;
|
|
}
|
|
|
|
else if( evt->IsClick( BUT_RIGHT ) )
|
|
{
|
|
m_menu.ShowContextMenu( selection() );
|
|
}
|
|
|
|
// Prepare the next loop by updating the old cursor mouse position
|
|
// to this last mouse cursor position
|
|
oldCursorPos = getViewControls()->GetCursorPosition();
|
|
statusPopup.Move( wxGetMousePosition() + wxPoint( 20, 20 ) );
|
|
}
|
|
|
|
for( auto p : board()->GetFirstModule()->Pads() )
|
|
{
|
|
p->ClearSelected();
|
|
view->Update( p );
|
|
}
|
|
|
|
statusPopup.Hide();
|
|
m_frame->SetNoToolSelected();
|
|
m_frame->GetGalCanvas()->SetCursor( wxCURSOR_ARROW );
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int MODULE_EDITOR_TOOLS::ExplodePadToShapes( const TOOL_EVENT& aEvent )
|
|
{
|
|
SELECTION& selection = m_toolMgr->GetTool<SELECTION_TOOL>()->GetSelection();
|
|
BOARD_COMMIT commit( m_frame );
|
|
|
|
if( selection.Size() != 1 )
|
|
return 0;
|
|
|
|
if( selection[0]->Type() != PCB_PAD_T )
|
|
return 0;
|
|
|
|
auto pad = static_cast<D_PAD*>( selection[0] );
|
|
|
|
if( pad->GetShape() != PAD_SHAPE_CUSTOM )
|
|
return 0;
|
|
|
|
commit.Modify( pad );
|
|
|
|
wxPoint anchor = pad->GetPosition();
|
|
|
|
for( auto prim : pad->GetPrimitives() )
|
|
{
|
|
auto ds = new EDGE_MODULE( board()->GetFirstModule() );
|
|
|
|
prim.ExportTo( ds ); // ExportTo exports to a DRAWSEGMENT
|
|
// Fix an arbitray draw layer for this EDGE_MODULE
|
|
ds->SetLayer( Dwgs_User ); //pad->GetLayer() );
|
|
ds->Move( anchor );
|
|
|
|
commit.Add( ds );
|
|
}
|
|
|
|
pad->SetShape( pad->GetAnchorPadShape() );
|
|
// Cleanup the pad primitives data, because the initial pad was a custom
|
|
// shaped pad, and it contains primitives, that does not exist in non custom pads,
|
|
// and can create issues later:
|
|
if( pad->GetShape() != PAD_SHAPE_CUSTOM ) // should be always the case
|
|
{
|
|
pad->DeletePrimitivesList();
|
|
}
|
|
|
|
commit.Push( _("Explode pad to shapes") );
|
|
|
|
m_toolMgr->RunAction( PCB_ACTIONS::selectionClear, true );
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int MODULE_EDITOR_TOOLS::CreatePadFromShapes( const TOOL_EVENT& aEvent )
|
|
{
|
|
SELECTION& selection = m_toolMgr->GetTool<SELECTION_TOOL>()->GetSelection();
|
|
|
|
std::unique_ptr<D_PAD> pad( new D_PAD( board()->GetFirstModule() ) );
|
|
D_PAD *refPad = nullptr;
|
|
bool multipleRefPadsFound = false;
|
|
bool illegalItemsFound = false;
|
|
|
|
std::vector<PAD_CS_PRIMITIVE> shapes;
|
|
|
|
BOARD_COMMIT commit( m_frame );
|
|
|
|
for( auto item : selection )
|
|
{
|
|
switch( item->Type() )
|
|
{
|
|
case PCB_PAD_T:
|
|
{
|
|
if( refPad )
|
|
multipleRefPadsFound = true;
|
|
|
|
refPad = static_cast<D_PAD*>( item );
|
|
break;
|
|
}
|
|
|
|
case PCB_MODULE_EDGE_T:
|
|
{
|
|
auto em = static_cast<EDGE_MODULE*> ( item );
|
|
|
|
PAD_CS_PRIMITIVE shape( em->GetShape() );
|
|
shape.m_Start = em->GetStart();
|
|
shape.m_End = em->GetEnd();
|
|
shape.m_Radius = em->GetRadius();
|
|
shape.m_Thickness = em->GetWidth();
|
|
shape.m_ArcAngle = em->GetAngle();
|
|
shape.m_Ctrl1 = em->GetBezControl1();
|
|
shape.m_Ctrl2 = em->GetBezControl2();
|
|
shape.m_Poly = em->BuildPolyPointsList();
|
|
|
|
shapes.push_back(shape);
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
illegalItemsFound = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if( refPad && selection.Size() == 1 )
|
|
{
|
|
// don't convert a pad into itself...
|
|
return 0;
|
|
}
|
|
|
|
if( multipleRefPadsFound )
|
|
{
|
|
DisplayErrorMessage( m_frame, _( "Cannot convert items to a custom-shaped pad:\n"
|
|
"selection contains more than one reference pad." ) );
|
|
return 0;
|
|
}
|
|
|
|
if( illegalItemsFound )
|
|
{
|
|
DisplayErrorMessage( m_frame, _( "Cannot convert items to a custom-shaped pad:\n"
|
|
"selection contains unsupported items.\n"
|
|
"Only graphical lines, circles, arcs and polygons "
|
|
"are allowed." ) );
|
|
return 0;
|
|
}
|
|
|
|
if( refPad )
|
|
{
|
|
pad.reset( static_cast<D_PAD*>( refPad->Clone() ) );
|
|
|
|
if( refPad->GetShape() == PAD_SHAPE_RECT )
|
|
pad->SetAnchorPadShape( PAD_SHAPE_RECT );
|
|
|
|
// ignore the pad orientation and offset for the moment. Makes more trouble than it's worth.
|
|
pad->SetOrientation( 0 );
|
|
pad->SetOffset( wxPoint( 0, 0 ) );
|
|
}
|
|
else
|
|
{
|
|
// Create a default pad anchor:
|
|
pad->SetAnchorPadShape( PAD_SHAPE_CIRCLE );
|
|
pad->SetAttribute( PAD_ATTRIB_SMD );
|
|
pad->SetLayerSet( D_PAD::SMDMask() );
|
|
int radius = Millimeter2iu( 0.2 );
|
|
pad->SetSize ( wxSize( radius, radius ) );
|
|
pad->IncrementPadName( true, true );
|
|
pad->SetOrientation( 0 );
|
|
}
|
|
|
|
pad->SetShape ( PAD_SHAPE_CUSTOM );
|
|
|
|
OPT<VECTOR2I> anchor;
|
|
VECTOR2I tmp;
|
|
|
|
if( refPad )
|
|
{
|
|
anchor = VECTOR2I( pad->GetPosition() );
|
|
}
|
|
else if( pad->GetBestAnchorPosition( tmp ) )
|
|
{
|
|
anchor = tmp;
|
|
}
|
|
|
|
if( !anchor )
|
|
{
|
|
DisplayErrorMessage( m_frame, _( "Cannot convert items to a custom-shaped pad:\n"
|
|
"unable to determine the anchor point position.\n"
|
|
"Consider adding a small anchor pad to the selection "
|
|
"and try again.") );
|
|
return 0;
|
|
}
|
|
|
|
|
|
// relocate the shapes, they are relative to the anchor pad position
|
|
for( auto& shape : shapes )
|
|
{
|
|
shape.Move( wxPoint( -anchor->x, -anchor->y ) );
|
|
}
|
|
|
|
|
|
pad->SetPosition( wxPoint( anchor->x, anchor->y ) );
|
|
pad->AddPrimitives( shapes );
|
|
pad->ClearFlags();
|
|
|
|
bool result = pad->MergePrimitivesAsPolygon();
|
|
|
|
if( !result )
|
|
{
|
|
DisplayErrorMessage( m_frame, _( "Cannot convert items to a custom-shaped pad:\n"
|
|
"selected items do not form a single solid shape.") );
|
|
return 0;
|
|
}
|
|
|
|
auto padPtr = pad.release();
|
|
|
|
commit.Add( padPtr );
|
|
for ( auto item : selection )
|
|
{
|
|
commit.Remove( item );
|
|
}
|
|
|
|
m_toolMgr->RunAction( PCB_ACTIONS::selectionClear, true );
|
|
commit.Push(_("Create Pad from Selected Shapes") );
|
|
m_toolMgr->RunAction( PCB_ACTIONS::selectItem, true, padPtr );
|
|
|
|
return 0;
|
|
}
|
|
|
|
void MODULE_EDITOR_TOOLS::setTransitions()
|
|
{
|
|
Go( &MODULE_EDITOR_TOOLS::NewFootprint, PCB_ACTIONS::newFootprint.MakeEvent() );
|
|
Go( &MODULE_EDITOR_TOOLS::CreateFootprint, PCB_ACTIONS::createFootprint.MakeEvent() );
|
|
Go( &MODULE_EDITOR_TOOLS::Save, ACTIONS::save.MakeEvent() );
|
|
Go( &MODULE_EDITOR_TOOLS::Save, PCB_ACTIONS::saveToBoard.MakeEvent() );
|
|
Go( &MODULE_EDITOR_TOOLS::Save, PCB_ACTIONS::saveToLibrary.MakeEvent() );
|
|
Go( &MODULE_EDITOR_TOOLS::SaveAs, ACTIONS::saveAs.MakeEvent() );
|
|
Go( &MODULE_EDITOR_TOOLS::SaveAs, ACTIONS::saveCopyAs.MakeEvent() );
|
|
Go( &MODULE_EDITOR_TOOLS::Revert, ACTIONS::revert.MakeEvent() );
|
|
Go( &MODULE_EDITOR_TOOLS::Delete, PCB_ACTIONS::deleteFootprint.MakeEvent() );
|
|
|
|
Go( &MODULE_EDITOR_TOOLS::Properties, PCB_ACTIONS::footprintProperties.MakeEvent() );
|
|
Go( &MODULE_EDITOR_TOOLS::DefaultPadProperties, PCB_ACTIONS::defaultPadProperties.MakeEvent() );
|
|
|
|
Go( &MODULE_EDITOR_TOOLS::PlacePad, PCB_ACTIONS::placePad.MakeEvent() );
|
|
Go( &MODULE_EDITOR_TOOLS::CreatePadFromShapes, PCB_ACTIONS::createPadFromShapes.MakeEvent() );
|
|
Go( &MODULE_EDITOR_TOOLS::ExplodePadToShapes, PCB_ACTIONS::explodePadToShapes.MakeEvent() );
|
|
Go( &MODULE_EDITOR_TOOLS::EnumeratePads, PCB_ACTIONS::enumeratePads.MakeEvent() );
|
|
}
|