2007-08-20 01:20:48 +00:00
|
|
|
/****************************************************************/
|
2009-11-12 15:43:38 +00:00
|
|
|
/* Routines for automatic displacement and rotation of modules. */
|
2007-08-20 01:20:48 +00:00
|
|
|
/****************************************************************/
|
2007-06-05 12:10:51 +00:00
|
|
|
|
2010-11-26 17:47:35 +00:00
|
|
|
#include <algorithm>
|
|
|
|
|
2007-06-05 12:10:51 +00:00
|
|
|
#include "fctsys.h"
|
2009-02-04 15:25:03 +00:00
|
|
|
#include "class_drawpanel.h"
|
|
|
|
#include "confirm.h"
|
|
|
|
#include "kicad_string.h"
|
2007-06-05 12:10:51 +00:00
|
|
|
#include "pcbnew.h"
|
2009-07-30 11:04:07 +00:00
|
|
|
#include "wxPcbStruct.h"
|
2011-09-23 13:57:12 +00:00
|
|
|
#include "kicad_device_context.h"
|
|
|
|
|
2007-06-05 12:10:51 +00:00
|
|
|
#include "autorout.h"
|
|
|
|
#include "cell.h"
|
2009-09-22 12:27:57 +00:00
|
|
|
#include "pcbnew_id.h"
|
2011-09-23 13:57:12 +00:00
|
|
|
#include "class_board.h"
|
|
|
|
#include "class_module.h"
|
2007-06-05 12:10:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
typedef enum {
|
2007-08-20 01:20:48 +00:00
|
|
|
FIXE_MODULE,
|
|
|
|
FREE_MODULE,
|
|
|
|
FIXE_ALL_MODULES,
|
|
|
|
FREE_ALL_MODULES
|
|
|
|
} SelectFixeFct;
|
2007-06-05 12:10:51 +00:00
|
|
|
|
|
|
|
|
2010-11-26 17:47:35 +00:00
|
|
|
static bool sortModulesbySize( MODULE* ref, MODULE* compare );
|
2007-06-05 12:10:51 +00:00
|
|
|
|
|
|
|
|
2009-11-12 15:43:38 +00:00
|
|
|
wxString ModulesMaskSelection = wxT( "*" );
|
2007-06-05 12:10:51 +00:00
|
|
|
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2009-10-23 15:28:45 +00:00
|
|
|
/* Called on events (popup menus) relative to automove and autoplace footprints
|
2007-08-20 01:20:48 +00:00
|
|
|
*/
|
2011-03-01 19:26:17 +00:00
|
|
|
void PCB_EDIT_FRAME::AutoPlace( wxCommandEvent& event )
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
int id = event.GetId();
|
|
|
|
bool on_state;
|
|
|
|
|
|
|
|
if( m_HToolBar == NULL )
|
|
|
|
return;
|
|
|
|
|
2011-01-30 22:22:38 +00:00
|
|
|
INSTALL_UNBUFFERED_DC( dc, DrawPanel );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2009-10-23 15:28:45 +00:00
|
|
|
switch( id )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
2010-08-17 10:41:44 +00:00
|
|
|
case ID_TOOLBARH_PCB_MODE_MODULE:
|
|
|
|
on_state = m_HToolBar->GetToolState( ID_TOOLBARH_PCB_MODE_MODULE );
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2007-08-20 01:20:48 +00:00
|
|
|
if( on_state )
|
|
|
|
{
|
2011-09-07 19:41:04 +00:00
|
|
|
m_HToolBar->ToggleTool( ID_TOOLBARH_PCB_MODE_TRACKS, false );
|
2010-08-17 10:41:44 +00:00
|
|
|
m_HTOOL_current_state = ID_TOOLBARH_PCB_MODE_MODULE;
|
2007-08-20 01:20:48 +00:00
|
|
|
}
|
|
|
|
else
|
2011-09-07 19:41:04 +00:00
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
m_HTOOL_current_state = 0;
|
2011-09-07 19:41:04 +00:00
|
|
|
}
|
|
|
|
|
2010-08-12 05:57:14 +00:00
|
|
|
return;
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2010-08-17 10:41:44 +00:00
|
|
|
case ID_TOOLBARH_PCB_MODE_TRACKS:
|
|
|
|
on_state = m_HToolBar->GetToolState( ID_TOOLBARH_PCB_MODE_TRACKS );
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2007-08-20 01:20:48 +00:00
|
|
|
if( on_state )
|
|
|
|
{
|
2011-09-07 19:41:04 +00:00
|
|
|
m_HToolBar->ToggleTool( ID_TOOLBARH_PCB_MODE_MODULE, false );
|
2010-08-17 10:41:44 +00:00
|
|
|
m_HTOOL_current_state = ID_TOOLBARH_PCB_MODE_TRACKS;
|
2007-08-20 01:20:48 +00:00
|
|
|
}
|
|
|
|
else
|
2011-09-07 19:41:04 +00:00
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
m_HTOOL_current_state = 0;
|
2011-09-07 19:41:04 +00:00
|
|
|
}
|
|
|
|
|
2010-08-12 05:57:14 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOROUTE_SELECT_LAYERS:
|
|
|
|
return;
|
2007-08-20 01:20:48 +00:00
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOPLACE_FIXE_MODULE:
|
2011-03-09 14:30:39 +00:00
|
|
|
LockModule( (MODULE*) GetScreen()->GetCurItem(), true );
|
2010-08-12 05:57:14 +00:00
|
|
|
return;
|
2007-08-20 01:20:48 +00:00
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOPLACE_FREE_MODULE:
|
2011-09-07 19:41:04 +00:00
|
|
|
LockModule( (MODULE*) GetScreen()->GetCurItem(), false );
|
2010-08-12 05:57:14 +00:00
|
|
|
return;
|
2007-08-20 01:20:48 +00:00
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOPLACE_FREE_ALL_MODULES:
|
2011-09-07 19:41:04 +00:00
|
|
|
LockModule( NULL, false );
|
2010-08-12 05:57:14 +00:00
|
|
|
return;
|
2007-08-20 01:20:48 +00:00
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOPLACE_FIXE_ALL_MODULES:
|
2011-03-09 14:30:39 +00:00
|
|
|
LockModule( NULL, true );
|
2010-08-12 05:57:14 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
case ID_POPUP_CANCEL_CURRENT_COMMAND:
|
2011-02-11 20:48:13 +00:00
|
|
|
if( DrawPanel->IsMouseCaptured() )
|
2010-08-12 05:57:14 +00:00
|
|
|
{
|
2011-02-11 20:48:13 +00:00
|
|
|
DrawPanel->m_endMouseCaptureCallback( DrawPanel, &dc );
|
2010-08-12 05:57:14 +00:00
|
|
|
}
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2007-08-20 01:20:48 +00:00
|
|
|
break;
|
|
|
|
|
2010-08-12 05:57:14 +00:00
|
|
|
default: // Abort a current command (if any)
|
2011-02-24 20:22:12 +00:00
|
|
|
DrawPanel->EndMouseCapture( ID_NO_TOOL_SELECTED, DrawPanel->GetDefaultCursor() );
|
2010-08-12 05:57:14 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Erase ratsnest if needed */
|
|
|
|
if( GetBoard()->IsElementVisible(RATSNEST_VISIBLE) )
|
|
|
|
DrawGeneralRatsnest( &dc );
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2010-08-12 05:57:14 +00:00
|
|
|
GetBoard()->m_Status_Pcb |= DO_NOT_SHOW_GENERAL_RASTNEST;
|
|
|
|
|
|
|
|
switch( id )
|
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
case ID_POPUP_PCB_AUTOPLACE_CURRENT_MODULE:
|
2011-09-07 19:41:04 +00:00
|
|
|
AutoPlaceModule( (MODULE*) GetScreen()->GetCurItem(), PLACE_1_MODULE, &dc );
|
2007-08-20 01:20:48 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOPLACE_ALL_MODULES:
|
|
|
|
AutoPlaceModule( NULL, PLACE_ALL, &dc );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOPLACE_NEW_MODULES:
|
|
|
|
AutoPlaceModule( NULL, PLACE_OUT_OF_BOARD, &dc );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOPLACE_NEXT_MODULE:
|
|
|
|
AutoPlaceModule( NULL, PLACE_INCREMENTAL, &dc );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOMOVE_ALL_MODULES:
|
2011-09-07 19:41:04 +00:00
|
|
|
AutoMoveModulesOnPcb( false );
|
2007-08-20 01:20:48 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOMOVE_NEW_MODULES:
|
2011-03-09 14:30:39 +00:00
|
|
|
AutoMoveModulesOnPcb( true );
|
2007-08-20 01:20:48 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOROUTE_ALL_MODULES:
|
|
|
|
Autoroute( &dc, ROUTE_ALL );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOROUTE_MODULE:
|
|
|
|
Autoroute( &dc, ROUTE_MODULE );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOROUTE_PAD:
|
|
|
|
Autoroute( &dc, ROUTE_PAD );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOROUTE_NET:
|
|
|
|
Autoroute( &dc, ROUTE_NET );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_AUTOROUTE_RESET_UNROUTED:
|
|
|
|
Reset_Noroutable( &dc );
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2011-03-09 14:30:39 +00:00
|
|
|
wxMessageBox( wxT( "AutoPlace command error" ) );
|
2007-08-20 01:20:48 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2009-01-05 05:21:35 +00:00
|
|
|
GetBoard()->m_Status_Pcb &= ~DO_NOT_SHOW_GENERAL_RASTNEST;
|
2009-06-11 14:26:17 +00:00
|
|
|
Compile_Ratsnest( &dc, true );
|
2007-06-05 12:10:51 +00:00
|
|
|
}
|
|
|
|
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2010-08-11 18:17:12 +00:00
|
|
|
/* Function to move components in a rectangular area format 4 / 3,
|
|
|
|
* starting from the mouse cursor
|
|
|
|
* The components with the FIXED status set are not moved
|
2007-08-20 01:20:48 +00:00
|
|
|
*/
|
2011-03-01 19:26:17 +00:00
|
|
|
void PCB_EDIT_FRAME::AutoMoveModulesOnPcb( bool PlaceModulesHorsPcb )
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2010-11-26 17:47:35 +00:00
|
|
|
std::vector <MODULE*> moduleList;
|
2007-08-20 01:20:48 +00:00
|
|
|
wxPoint start, current;
|
|
|
|
int Ymax_size, Xsize_allowed;
|
2009-11-12 15:43:38 +00:00
|
|
|
int pas_grille = (int) GetScreen()->GetGridSize().x;
|
2010-11-26 17:47:35 +00:00
|
|
|
bool edgesExists;
|
|
|
|
double surface;
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2009-01-05 05:21:35 +00:00
|
|
|
if( GetBoard()->m_Modules == NULL )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
2009-11-12 15:43:38 +00:00
|
|
|
DisplayError( this, _( "No modules found!" ) );
|
|
|
|
return;
|
2007-08-20 01:20:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Confirmation */
|
2009-11-12 15:43:38 +00:00
|
|
|
if( !IsOK( this, _( "Move modules?" ) ) )
|
2007-08-20 01:20:48 +00:00
|
|
|
return;
|
|
|
|
|
2011-02-25 16:23:24 +00:00
|
|
|
edgesExists = GetBoard()->ComputeBoundingBox( true );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2010-11-26 17:47:35 +00:00
|
|
|
if( PlaceModulesHorsPcb && !edgesExists )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
|
|
|
DisplayError( this,
|
2010-06-19 10:58:50 +00:00
|
|
|
_( "Could not automatically place modules. No board outlines detected." ) );
|
2007-08-20 01:20:48 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-11-26 17:47:35 +00:00
|
|
|
// Build sorted footprints list (sort by decreasing size )
|
|
|
|
MODULE* Module = GetBoard()->m_Modules;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2009-11-12 15:43:38 +00:00
|
|
|
for( ; Module != NULL; Module = Module->Next() )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
2011-09-07 19:41:04 +00:00
|
|
|
Module->CalculateBoundingBox();
|
2010-11-26 17:47:35 +00:00
|
|
|
moduleList.push_back(Module);
|
2007-08-20 01:20:48 +00:00
|
|
|
}
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2010-11-26 17:47:35 +00:00
|
|
|
sort( moduleList.begin(), moduleList.end(), sortModulesbySize );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2010-11-26 17:47:35 +00:00
|
|
|
/* to move modules outside the board, the cursor is placed below
|
|
|
|
* the current board, to avoid placing components in board area.
|
2007-08-20 01:20:48 +00:00
|
|
|
*/
|
2010-11-26 17:47:35 +00:00
|
|
|
if( PlaceModulesHorsPcb && edgesExists )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
2011-02-11 20:48:13 +00:00
|
|
|
if( GetScreen()->GetCrossHairPosition().y < (GetBoard()->m_BoundaryBox.GetBottom() + 2000) )
|
2011-02-16 19:16:38 +00:00
|
|
|
{
|
|
|
|
wxPoint pos = GetScreen()->GetCrossHairPosition();
|
|
|
|
pos.y = GetBoard()->m_BoundaryBox.GetBottom() + 2000;
|
|
|
|
GetScreen()->SetCrossHairPosition( pos );
|
|
|
|
}
|
2007-08-20 01:20:48 +00:00
|
|
|
}
|
|
|
|
|
2010-11-26 17:47:35 +00:00
|
|
|
/* calculate the area needed by footprints */
|
2007-08-20 01:20:48 +00:00
|
|
|
surface = 0.0;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2010-11-26 17:47:35 +00:00
|
|
|
for( unsigned ii = 0; ii < moduleList.size(); ii++ )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
2010-11-26 17:47:35 +00:00
|
|
|
Module = moduleList[ii];
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2010-11-26 17:47:35 +00:00
|
|
|
if( PlaceModulesHorsPcb && edgesExists )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
2010-12-20 17:44:25 +00:00
|
|
|
if( GetBoard()->m_BoundaryBox.Contains( Module->m_Pos ) )
|
2007-08-20 01:20:48 +00:00
|
|
|
continue;
|
|
|
|
}
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2007-08-20 01:20:48 +00:00
|
|
|
surface += Module->m_Surface;
|
|
|
|
}
|
|
|
|
|
2009-11-12 15:43:38 +00:00
|
|
|
Xsize_allowed = (int) ( sqrt( surface ) * 4.0 / 3.0 );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2011-02-11 20:48:13 +00:00
|
|
|
start = current = GetScreen()->GetCrossHairPosition();
|
2007-08-20 01:20:48 +00:00
|
|
|
Ymax_size = 0;
|
|
|
|
|
2010-11-26 17:47:35 +00:00
|
|
|
for( unsigned ii = 0; ii < moduleList.size(); ii++ )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
2010-11-26 17:47:35 +00:00
|
|
|
Module = moduleList[ii];
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2007-09-09 02:27:03 +00:00
|
|
|
if( Module->IsLocked() )
|
2007-08-20 01:20:48 +00:00
|
|
|
continue;
|
|
|
|
|
2010-11-26 17:47:35 +00:00
|
|
|
if( PlaceModulesHorsPcb && edgesExists )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
2010-12-20 17:44:25 +00:00
|
|
|
if( GetBoard()->m_BoundaryBox.Contains( Module->m_Pos ) )
|
2007-08-20 01:20:48 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( current.x > (Xsize_allowed + start.x) )
|
|
|
|
{
|
|
|
|
current.x = start.x;
|
|
|
|
current.y += Ymax_size + pas_grille;
|
|
|
|
Ymax_size = 0;
|
|
|
|
}
|
|
|
|
|
2011-02-11 20:48:13 +00:00
|
|
|
GetScreen()->SetCrossHairPosition( current + Module->m_Pos -
|
2011-09-01 21:39:38 +00:00
|
|
|
Module->m_BoundaryBox.GetPosition() );
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2011-09-01 21:39:38 +00:00
|
|
|
Ymax_size = MAX( Ymax_size, Module->m_BoundaryBox.GetHeight() );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2011-09-16 14:13:02 +00:00
|
|
|
PlaceModule( Module, NULL, true );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2011-09-01 21:39:38 +00:00
|
|
|
current.x += Module->m_BoundaryBox.GetWidth() + pas_grille;
|
2007-08-20 01:20:48 +00:00
|
|
|
}
|
|
|
|
|
2009-10-23 15:28:45 +00:00
|
|
|
DrawPanel->Refresh();
|
2007-06-05 12:10:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-09-07 19:41:04 +00:00
|
|
|
/* Set or reset (true or false) Lock attribute of aModule or all modules if aModule == NULL
|
2007-08-20 01:20:48 +00:00
|
|
|
*/
|
2011-03-09 14:30:39 +00:00
|
|
|
void PCB_EDIT_FRAME::LockModule( MODULE* aModule, bool aLocked )
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2011-03-09 14:30:39 +00:00
|
|
|
if( aModule )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
2011-03-09 14:30:39 +00:00
|
|
|
aModule->SetLocked( aLocked );
|
2008-04-01 05:21:50 +00:00
|
|
|
|
2011-03-09 14:30:39 +00:00
|
|
|
aModule->DisplayInfo( this );
|
2010-02-19 13:23:58 +00:00
|
|
|
OnModify();
|
2007-08-20 01:20:48 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-03-09 14:30:39 +00:00
|
|
|
aModule = GetBoard()->m_Modules;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2011-03-09 14:30:39 +00:00
|
|
|
for( ; aModule != NULL; aModule = aModule->Next() )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
2011-09-07 19:41:04 +00:00
|
|
|
if( WildCompareString( ModulesMaskSelection, aModule->m_Reference->m_Text ) )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
2011-03-09 14:30:39 +00:00
|
|
|
aModule->SetLocked( aLocked );
|
2010-02-19 13:23:58 +00:00
|
|
|
OnModify();
|
2007-08-20 01:20:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-06-05 12:10:51 +00:00
|
|
|
}
|
|
|
|
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2010-11-26 17:47:35 +00:00
|
|
|
static bool sortModulesbySize( MODULE* ref, MODULE* compare )
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2010-11-26 17:47:35 +00:00
|
|
|
return compare->m_Surface < ref->m_Surface;
|
2007-06-05 12:10:51 +00:00
|
|
|
}
|