kicad/pcbnew/block.cpp

1004 lines
31 KiB
C++
Raw Normal View History

/*****************************************************************/
/* Operations sur Blocks : deplacement, rotation, effacement ... */
/*****************************************************************/
2007-05-06 16:03:28 +00:00
#include "fctsys.h"
#include "gr_basic.h"
#include "common.h"
#include "class_drawpanel.h"
#include "confirm.h"
#include "block_commande.h"
2007-05-06 16:03:28 +00:00
#include "pcbnew.h"
2009-07-30 11:04:07 +00:00
#include "wxPcbStruct.h"
2007-05-06 16:03:28 +00:00
#include "autorout.h"
#include "pcbplot.h"
#include "trigo.h"
#include "protos.h"
#define BLOCK_COLOR BROWN
/* Routines Locales */
2009-08-01 19:26:05 +00:00
static void DrawMovingBlockOutlines( WinEDA_DrawPanel* panel, wxDC* DC, bool erase );
2007-05-06 16:03:28 +00:00
/* Variables locales :*/
static bool Block_Include_Modules = TRUE;
static bool Block_Include_Tracks = TRUE;
static bool Block_Include_Zones = TRUE;
static bool Block_Include_Draw_Items = TRUE;
2007-05-06 16:03:28 +00:00
static bool Block_Include_Edges_Items = TRUE;
static bool Block_Include_PcbTextes = TRUE;
2007-05-06 16:03:28 +00:00
/************************************/
/* class WinEDA_ExecBlockCmdFrame */
/************************************/
2007-05-06 16:03:28 +00:00
class WinEDA_ExecBlockCmdFrame : public wxDialog
2007-05-06 16:03:28 +00:00
{
private:
WinEDA_BasePcbFrame* m_Parent;
wxCheckBox* m_Include_Modules;
wxCheckBox* m_Include_Tracks;
wxCheckBox* m_Include_Zones;
wxCheckBox* m_Include_Draw_Items;
wxCheckBox* m_Include_Edges_Items;
wxCheckBox* m_Include_PcbTextes;
2007-05-06 16:03:28 +00:00
public:
// Constructor and destructor
WinEDA_ExecBlockCmdFrame( WinEDA_BasePcbFrame* parent,
const wxString& title );
2007-09-01 12:00:30 +00:00
~WinEDA_ExecBlockCmdFrame()
{
}
2007-05-06 16:03:28 +00:00
private:
2009-08-01 19:26:05 +00:00
void ExecuteCommand( wxCommandEvent& event );
void Cancel( wxCommandEvent& event );
2007-05-06 16:03:28 +00:00
DECLARE_EVENT_TABLE()
2007-05-06 16:03:28 +00:00
};
BEGIN_EVENT_TABLE( WinEDA_ExecBlockCmdFrame, wxDialog )
EVT_BUTTON( wxID_OK, WinEDA_ExecBlockCmdFrame::ExecuteCommand )
EVT_BUTTON( wxID_CANCEL, WinEDA_ExecBlockCmdFrame::Cancel )
2007-05-06 16:03:28 +00:00
END_EVENT_TABLE()
/**************************************************************/
static bool InstallBlockCmdFrame( WinEDA_BasePcbFrame* parent,
const wxString& title )
2007-05-06 16:03:28 +00:00
/**************************************************************/
{
int nocmd;
wxPoint oldpos = parent->GetScreen()->m_Curseur;
2007-05-06 16:03:28 +00:00
parent->DrawPanel->m_IgnoreMouseEvents = TRUE;
WinEDA_ExecBlockCmdFrame* frame = new WinEDA_ExecBlockCmdFrame( parent, title );
2008-04-01 05:21:50 +00:00
nocmd = frame->ShowModal();
2007-10-13 06:18:44 +00:00
frame->Destroy();
2008-04-01 05:21:50 +00:00
2007-10-13 06:18:44 +00:00
parent->GetScreen()->m_Curseur = oldpos;
2008-04-01 05:21:50 +00:00
parent->DrawPanel->MouseToCursorSchema();
parent->DrawPanel->m_IgnoreMouseEvents = FALSE;
2007-05-06 16:03:28 +00:00
parent->DrawPanel->SetCursor(
parent->DrawPanel->m_PanelCursor = parent->DrawPanel->m_PanelDefaultCursor );
2007-05-06 16:03:28 +00:00
return nocmd ? FALSE : TRUE;
2007-05-06 16:03:28 +00:00
}
/******************************************************************************/
WinEDA_ExecBlockCmdFrame::WinEDA_ExecBlockCmdFrame( WinEDA_BasePcbFrame* parent,
const wxString& title ) :
wxDialog( parent, -1, title, wxPoint( -1, -1 ), wxDefaultSize,
DIALOG_STYLE )
2007-05-06 16:03:28 +00:00
/******************************************************************************/
{
wxPoint pos;
wxButton* m_button1;
wxButton* m_button2;
m_Parent = parent;
Centre();
2007-10-03 19:57:23 +00:00
this->SetSizeHints( wxDefaultSize, wxDefaultSize );
this->SetFont( wxFont( wxNORMAL_FONT->GetPointSize(), 70, 90, 90,
false, wxEmptyString ) );
/* Sizer 1 creation */
wxFlexGridSizer* fgSizer1;
fgSizer1 = new wxFlexGridSizer( 1, 1, 0, 0 );
fgSizer1->SetFlexibleDirection( wxBOTH );
fgSizer1->SetNonFlexibleGrowMode( wxFLEX_GROWMODE_SPECIFIED );
2008-04-01 05:21:50 +00:00
// Selection des options :
2009-08-01 19:26:05 +00:00
m_Include_Modules = new wxCheckBox( this, -1, _(
"Include Modules" ), wxDefaultPosition, wxDefaultSize,
0 );
m_Include_Modules->SetValue( Block_Include_Modules );
fgSizer1->Add( m_Include_Modules, 0, wxALL, 5 );
2009-08-01 19:26:05 +00:00
m_Include_Tracks = new wxCheckBox( this, -1, _(
"Include tracks" ), wxDefaultPosition, wxDefaultSize, 0 );
m_Include_Tracks->SetValue( Block_Include_Tracks );
fgSizer1->Add( m_Include_Tracks, 0, wxALL, 5 );
2009-08-01 19:26:05 +00:00
m_Include_Zones = new wxCheckBox( this, -1, _(
"Include zones" ), wxDefaultPosition, wxDefaultSize, 0 );
m_Include_Zones->SetValue( Block_Include_Zones );
fgSizer1->Add( m_Include_Zones, 0, wxALL, 5 );
m_Include_PcbTextes = new wxCheckBox( this, -1,
2009-08-01 19:26:05 +00:00
_(
"Include Text on copper layers" ), wxDefaultPosition,
wxDefaultSize, 0 );
m_Include_PcbTextes->SetValue( Block_Include_PcbTextes );
fgSizer1->Add( m_Include_PcbTextes, 0, wxALL, 5 );
2009-08-01 19:26:05 +00:00
m_Include_Draw_Items = new wxCheckBox( this, -1, _(
"Include drawings" ), wxDefaultPosition,
wxDefaultSize, 0 );
m_Include_Draw_Items->SetValue( Block_Include_Draw_Items );
fgSizer1->Add( m_Include_Draw_Items, 0, wxALL, 5 );
2009-08-01 19:26:05 +00:00
m_Include_Edges_Items = new wxCheckBox( this, -1, _(
"Include board outline layer" ), wxDefaultPosition,
wxDefaultSize, 0 );
m_Include_Edges_Items->SetValue( Block_Include_Edges_Items );
fgSizer1->Add( m_Include_Edges_Items, 0, wxALL, 5 );
/* Sizer 2 creation */
wxFlexGridSizer* fgSizer2;
fgSizer2 = new wxFlexGridSizer( 1, 2, 0, 0 );
fgSizer2->SetFlexibleDirection( wxBOTH );
fgSizer2->SetNonFlexibleGrowMode( wxFLEX_GROWMODE_SPECIFIED );
/* Creation des boutons de commande */
2009-08-01 19:26:05 +00:00
m_button2 = new wxButton( this, wxID_CANCEL, _(
"Cancel" ), wxDefaultPosition, wxDefaultSize, 0 );
fgSizer2->Add( m_button2, 0, wxALL, 5 );
2007-10-07 03:08:24 +00:00
m_button1 = new wxButton( this, wxID_OK, _( "OK" ), wxDefaultPosition, wxDefaultSize, 0 );
m_button1->SetDefault();
fgSizer2->Add( m_button1, 0, wxALL, 5 );
fgSizer1->Add( fgSizer2, 1, wxALIGN_RIGHT, 5 );
this->SetSizer( fgSizer1 );
this->Layout();
fgSizer1->Fit( this );
2007-05-06 16:03:28 +00:00
}
2007-05-06 16:03:28 +00:00
/**********************************************************************/
void WinEDA_ExecBlockCmdFrame::Cancel( wxCommandEvent& WXUNUSED (event) )
2007-05-06 16:03:28 +00:00
/**********************************************************************/
{
2007-10-07 03:08:24 +00:00
EndModal( -1 );
2007-05-06 16:03:28 +00:00
}
/*******************************************************************/
void WinEDA_ExecBlockCmdFrame::ExecuteCommand( wxCommandEvent& event )
2007-05-06 16:03:28 +00:00
/*******************************************************************/
{
Block_Include_Modules = m_Include_Modules->GetValue();
Block_Include_Tracks = m_Include_Tracks->GetValue();
Block_Include_Zones = m_Include_Zones->GetValue();
Block_Include_Draw_Items = m_Include_Draw_Items->GetValue();
Block_Include_Edges_Items = m_Include_Edges_Items->GetValue();
Block_Include_PcbTextes = m_Include_PcbTextes->GetValue();
EndModal( 0 );
2007-05-06 16:03:28 +00:00
}
/*************************************************/
int WinEDA_PcbFrame::ReturnBlockCommand( int key )
2007-05-06 16:03:28 +00:00
/*************************************************/
2007-05-06 16:03:28 +00:00
/* Return the block command (BLOCK_MOVE, BLOCK_COPY...) corresponding to
* the key (ALT, SHIFT ALT ..)
*/
2007-05-06 16:03:28 +00:00
{
int cmd = 0;
2007-05-06 16:03:28 +00:00
switch( key )
{
default:
cmd = key & 0x255;
break;
2007-05-06 16:03:28 +00:00
case 0:
cmd = BLOCK_MOVE;
break;
2007-05-06 16:03:28 +00:00
case GR_KB_SHIFT:
cmd = BLOCK_COPY;
break;
2007-05-06 16:03:28 +00:00
case GR_KB_CTRL:
cmd = BLOCK_ROTATE;
break;
2007-05-06 16:03:28 +00:00
case GR_KB_SHIFTCTRL:
cmd = BLOCK_DELETE;
break;
2007-05-06 16:03:28 +00:00
case GR_KB_ALT:
cmd = BLOCK_INVERT;
break;
2007-05-06 16:03:28 +00:00
case MOUSE_MIDDLE:
cmd = BLOCK_ZOOM;
break;
}
2007-05-06 16:03:28 +00:00
return cmd;
2007-05-06 16:03:28 +00:00
}
/*****************************************************/
void WinEDA_PcbFrame::HandleBlockPlace( wxDC* DC )
2007-05-06 16:03:28 +00:00
/*****************************************************/
/* Routine to handle the BLOCK PLACE commande */
{
bool err = FALSE;
if( DrawPanel->ManageCurseur == NULL )
{
err = TRUE;
DisplayError( this, wxT( "Error in HandleBlockPLace : ManageCurseur = NULL" ) );
}
GetScreen()->m_BlockLocate.m_State = STATE_BLOCK_STOP;
switch( GetScreen()->m_BlockLocate.m_Command )
{
2009-08-01 19:26:05 +00:00
case BLOCK_IDLE:
err = TRUE;
break;
case BLOCK_DRAG: /* Drag */
case BLOCK_MOVE: /* Move */
case BLOCK_PRESELECT_MOVE: /* Move with preselection list*/
if( DrawPanel->ManageCurseur )
DrawPanel->ManageCurseur( DrawPanel, DC, FALSE );
2009-08-01 19:26:05 +00:00
Block_Move();
GetScreen()->m_BlockLocate.ClearItemsList();
break;
case BLOCK_COPY: /* Copy */
if( DrawPanel->ManageCurseur )
DrawPanel->ManageCurseur( DrawPanel, DC, FALSE );
2009-08-01 19:26:05 +00:00
Block_Duplicate();
GetScreen()->m_BlockLocate.ClearItemsList();
break;
case BLOCK_PASTE:
break;
case BLOCK_ZOOM: // Handled by HandleBlockEnd()
default:
break;
}
GetScreen()->SetModify();
DrawPanel->ManageCurseur = NULL;
2009-08-01 19:26:05 +00:00
DrawPanel->ForceCloseManageCurseur = NULL;
GetScreen()->m_BlockLocate.m_Flags = 0;
GetScreen()->m_BlockLocate.m_State = STATE_NO_BLOCK;
GetScreen()->m_BlockLocate.m_Command = BLOCK_IDLE;
if( GetScreen()->m_BlockLocate.GetCount() )
{
DisplayError( this, wxT( "Error in HandleBlockPLace some items left in list" ) );
GetScreen()->m_BlockLocate.ClearItemsList();
}
DisplayToolMsg( wxEmptyString );
2007-05-06 16:03:28 +00:00
}
2007-05-06 16:03:28 +00:00
/**********************************************/
int WinEDA_PcbFrame::HandleBlockEnd( wxDC* DC )
2007-05-06 16:03:28 +00:00
/**********************************************/
2007-05-06 16:03:28 +00:00
/* Routine de gestion de la commande BLOCK END
* returne :
* 0 si aucun composant selectionne
* 1 sinon
* -1 si commande termin<EFBFBD>e et composants trouv<EFBFBD>s (block delete, block save)
*/
2007-05-06 16:03:28 +00:00
{
int endcommande = TRUE;
if( DrawPanel->ManageCurseur )
switch( GetScreen()->m_BlockLocate.m_Command )
{
2009-08-01 19:26:05 +00:00
case BLOCK_IDLE:
DisplayError( this, wxT( "Error in HandleBlockPLace" ) );
break;
case BLOCK_DRAG: /* Drag (not used, for future enhancements)*/
case BLOCK_MOVE: /* Move */
case BLOCK_COPY: /* Copy */
case BLOCK_PRESELECT_MOVE: /* Move with preselection list*/
GetScreen()->m_BlockLocate.m_State = STATE_BLOCK_MOVE;
endcommande = FALSE;
DrawPanel->ManageCurseur( DrawPanel, DC, FALSE );
DrawPanel->ManageCurseur = DrawMovingBlockOutlines;
DrawPanel->ManageCurseur( DrawPanel, DC, FALSE );
break;
case BLOCK_DELETE: /* Delete */
// Turn off the block rectangle now so it is not redisplayed
DrawPanel->ManageCurseur = NULL;
GetScreen()->m_BlockLocate.m_State = STATE_BLOCK_STOP;
DrawAndSizingBlockOutlines( DrawPanel, DC, FALSE );
2009-08-01 19:26:05 +00:00
Block_Delete();
break;
case BLOCK_ROTATE: /* Rotation */
// Turn off the block rectangle now so it is not redisplayed
DrawPanel->ManageCurseur = NULL;
GetScreen()->m_BlockLocate.m_State = STATE_BLOCK_STOP;
DrawAndSizingBlockOutlines( DrawPanel, DC, FALSE );
2009-08-01 19:26:05 +00:00
Block_Rotate();
break;
case BLOCK_INVERT: /* Flip */
// Turn off the block rectangle now so it is not redisplayed
DrawPanel->ManageCurseur = NULL;
GetScreen()->m_BlockLocate.m_State = STATE_BLOCK_STOP;
DrawAndSizingBlockOutlines( DrawPanel, DC, FALSE );
2009-08-01 19:26:05 +00:00
Block_Flip();
break;
case BLOCK_SAVE: /* Save (not used, for future enhancements)*/
GetScreen()->m_BlockLocate.m_State = STATE_BLOCK_STOP;
if( GetScreen()->m_BlockLocate.GetCount() )
{
DrawAndSizingBlockOutlines( DrawPanel, DC, FALSE );
2009-08-01 19:26:05 +00:00
// TODO (if useful) Save_Block( );
}
break;
case BLOCK_PASTE:
break;
case BLOCK_ZOOM: /* Window Zoom */
//Turn off the redraw block routine now so it is not displayed
// with one corner at the new center of the screen
DrawPanel->ManageCurseur = NULL;
Window_Zoom( GetScreen()->m_BlockLocate );
break;
default:
break;
}
if( endcommande == TRUE )
{
GetScreen()->m_BlockLocate.m_Flags = 0;
GetScreen()->m_BlockLocate.m_State = STATE_NO_BLOCK;
GetScreen()->m_BlockLocate.m_Command = BLOCK_IDLE;
GetScreen()->m_BlockLocate.ClearItemsList();
DrawPanel->ManageCurseur = NULL;
DrawPanel->ForceCloseManageCurseur = NULL;
DisplayToolMsg( wxEmptyString );
}
return endcommande;
2007-05-06 16:03:28 +00:00
}
2009-08-01 19:26:05 +00:00
/* Block operations: */
2009-08-01 19:26:05 +00:00
/**
* Function Block_SelectItems
* Uses GetScreen()->m_BlockLocate
* select items within the selected block.
* selected items are put in the pick list
* @param none
*/
2009-08-01 19:26:05 +00:00
void WinEDA_PcbFrame::Block_SelectItems()
2007-05-06 16:03:28 +00:00
{
2009-08-01 19:26:05 +00:00
int masque_layer;
GetScreen()->m_BlockLocate.Normalize();
2009-08-01 19:26:05 +00:00
PICKED_ITEMS_LIST* itemsList = &GetScreen()->m_BlockLocate.m_ItemsSelection;
ITEM_PICKER picker( NULL, UR_UNSPECIFIED );
/* Effacement des modules */
if( Block_Include_Modules )
{
2009-08-01 19:26:05 +00:00
for( MODULE* module = m_Pcb->m_Modules; module != NULL; module = module->Next() )
{
if( module->HitTest( GetScreen()->m_BlockLocate ) )
2008-04-01 05:21:50 +00:00
{
2009-08-01 19:26:05 +00:00
picker.m_PickedItem = module;
picker.m_PickedItemType = module->Type();
2009-08-01 19:26:05 +00:00
itemsList->PushItem( picker );
2008-04-01 05:21:50 +00:00
}
}
}
/* Remove tracks and vias */
if( Block_Include_Tracks )
{
2009-08-01 19:26:05 +00:00
for( TRACK* pt_segm = m_Pcb->m_Track; pt_segm != NULL; pt_segm = pt_segm->Next() )
{
if( pt_segm->HitTest( GetScreen()->m_BlockLocate ) )
2008-04-01 05:21:50 +00:00
{
2009-08-01 19:26:05 +00:00
/* This track is in bloc: select it */
picker.m_PickedItem = pt_segm;
picker.m_PickedItemType = pt_segm->Type();
2009-08-01 19:26:05 +00:00
itemsList->PushItem( picker );
}
}
}
2009-08-01 19:26:05 +00:00
/* Select graphic items */
masque_layer = EDGE_LAYER;
if( Block_Include_Draw_Items )
masque_layer = ALL_LAYERS;
2008-04-01 05:21:50 +00:00
if( !Block_Include_Edges_Items )
masque_layer &= ~EDGE_LAYER;
2009-08-06 15:42:09 +00:00
for( BOARD_ITEM* PtStruct = m_Pcb->m_Drawings; PtStruct != NULL; PtStruct = PtStruct->Next() )
{
2009-08-01 19:26:05 +00:00
bool select_me = false;
2007-09-01 12:00:30 +00:00
switch( PtStruct->Type() )
{
case TYPE_DRAWSEGMENT:
2008-04-01 06:07:00 +00:00
if( (g_TabOneLayerMask[PtStruct->GetLayer()] & masque_layer) == 0 )
break;
2009-08-01 19:26:05 +00:00
if( !PtStruct->HitTest( GetScreen()->m_BlockLocate ) )
break;
2009-08-01 19:26:05 +00:00
select_me = true; // This item is in bloc: select it
break;
case TYPE_TEXTE:
if( !Block_Include_PcbTextes )
break;
2009-08-01 19:26:05 +00:00
if( !PtStruct->HitTest( GetScreen()->m_BlockLocate ) )
break;
2009-08-01 19:26:05 +00:00
select_me = true; // This item is in bloc: select it
break;
case TYPE_MIRE:
2008-04-01 06:07:00 +00:00
if( (g_TabOneLayerMask[PtStruct->GetLayer()] & masque_layer) == 0 )
break;
2009-08-01 19:26:05 +00:00
if( !PtStruct->HitTest( GetScreen()->m_BlockLocate ) )
break;
2009-08-01 19:26:05 +00:00
select_me = true; // This item is in bloc: select it
break;
case TYPE_COTATION:
2008-04-01 06:07:00 +00:00
if( (g_TabOneLayerMask[PtStruct->GetLayer()] & masque_layer) == 0 )
break;
2009-08-01 19:26:05 +00:00
if( !PtStruct->HitTest( GetScreen()->m_BlockLocate ) )
break;
2009-08-01 19:26:05 +00:00
select_me = true; // This item is in bloc: select it
break;
default:
break;
}
2009-08-01 19:26:05 +00:00
if( select_me )
{
2009-08-01 19:26:05 +00:00
picker.m_PickedItem = PtStruct;
picker.m_PickedItemType = PtStruct->Type();
2009-08-01 19:26:05 +00:00
itemsList->PushItem( picker );
}
}
2009-08-06 15:42:09 +00:00
/* Zone selection */
if( Block_Include_Zones )
{
2009-08-06 15:42:09 +00:00
#if 0
/* This section can creates problems if selected:
* m_Pcb->m_Zone can have a *lot* of items (100 000 is easily possible)
* so it is not selected (and TODO: will be removed, one day)
*/
2009-08-01 19:26:05 +00:00
for( SEGZONE* pt_segm = m_Pcb->m_Zone; pt_segm != NULL; pt_segm = pt_segm->Next() )
2009-08-06 15:42:09 +00:00
{ /* Segments used in Zone filling selection */
if( pt_segm->HitTest( GetScreen()->m_BlockLocate ) )
2008-04-01 05:21:50 +00:00
{
2009-08-06 15:42:09 +00:00
picker.m_PickedItem = pt_segm;
picker.m_PickedItemType = pt_segm->Type();
2009-08-01 19:26:05 +00:00
itemsList->PushItem( picker );
}
}
2009-08-06 15:42:09 +00:00
#endif
2009-08-01 19:26:05 +00:00
for( int ii = 0; ii < m_Pcb->GetAreaCount(); ii++ )
2008-04-01 05:21:50 +00:00
{
2009-08-01 19:26:05 +00:00
if( m_Pcb->GetArea( ii )->HitTest( GetScreen()->m_BlockLocate ) )
2008-04-01 05:21:50 +00:00
{
2009-08-01 19:26:05 +00:00
BOARD_ITEM* zone_c = m_Pcb->GetArea( ii );
picker.m_PickedItem = zone_c;
picker.m_PickedItemType = zone_c->Type();
2009-08-01 19:26:05 +00:00
itemsList->PushItem( picker );
2008-04-01 05:21:50 +00:00
}
}
}
2009-08-06 15:42:09 +00:00
}
2007-05-06 16:03:28 +00:00
2009-08-01 19:26:05 +00:00
/**************************************************************************/
static void DrawMovingBlockOutlines( WinEDA_DrawPanel* panel, wxDC* DC, bool erase )
/**************************************************************************/
2009-08-01 19:26:05 +00:00
/* Retrace le contour du block de repositionnement des structures a d<>placer
*/
2007-05-06 16:03:28 +00:00
{
2009-08-01 19:26:05 +00:00
int Color;
BASE_SCREEN* screen = panel->GetScreen();
2009-08-01 19:26:05 +00:00
Color = YELLOW;
2009-08-01 19:26:05 +00:00
/* Effacement ancien cadre */
if( erase )
{
2009-08-01 19:26:05 +00:00
screen->m_BlockLocate.Draw( panel, DC, wxPoint( 0, 0 ), g_XorMode, Color );
if( screen->m_BlockLocate.m_MoveVector.x || screen->m_BlockLocate.m_MoveVector.y )
{
2009-08-01 19:26:05 +00:00
screen->m_BlockLocate.Draw( panel,
DC,
screen->m_BlockLocate.m_MoveVector,
g_XorMode,
Color );
}
}
2009-08-01 19:26:05 +00:00
if( screen->m_BlockLocate.m_State != STATE_BLOCK_STOP )
{
2009-08-01 19:26:05 +00:00
screen->m_BlockLocate.m_MoveVector.x = screen->m_Curseur.x -
screen->m_BlockLocate.GetRight();
screen->m_BlockLocate.m_MoveVector.y = screen->m_Curseur.y -
screen->m_BlockLocate.GetBottom();
}
2009-08-01 19:26:05 +00:00
screen->m_BlockLocate.Draw( panel, DC, wxPoint( 0, 0 ), g_XorMode, Color );
if( screen->m_BlockLocate.m_MoveVector.x || screen->m_BlockLocate.m_MoveVector.y )
{
2009-08-01 19:26:05 +00:00
screen->m_BlockLocate.Draw( panel,
DC,
screen->m_BlockLocate.m_MoveVector,
g_XorMode,
Color );
}
2009-08-01 19:26:05 +00:00
}
2009-08-01 19:26:05 +00:00
/************************************************/
void WinEDA_PcbFrame::Block_Delete()
/************************************************/
/*
* routine d'effacement du block deja selectionne
*/
{
if( !InstallBlockCmdFrame( this, _( "Delete Block" ) ) )
return;
Block_SelectItems();
if( GetScreen()->m_BlockLocate.GetCount() == 0 )
return;
GetScreen()->SetModify();
SetCurItem( NULL );
PICKED_ITEMS_LIST* itemsList = &GetScreen()->m_BlockLocate.m_ItemsSelection;
itemsList->m_Status = UR_DELETED;
/* unlink items and clear flags */
for( unsigned ii = 0; ii < itemsList->GetCount(); ii++ )
{
2009-08-01 19:26:05 +00:00
BOARD_ITEM* item = (BOARD_ITEM*) itemsList->GetPickedItem( ii );
itemsList->SetPickedItemStatus( UR_DELETED, ii );
switch( item->Type() )
{
2009-08-01 19:26:05 +00:00
case TYPE_MODULE:
{
MODULE* module = (MODULE*) item;
module->m_Flags = 0;
module->UnLink();
m_Pcb->m_Status_Pcb = 0;
}
break;
2009-08-01 19:26:05 +00:00
case TYPE_ZONE_CONTAINER: // a zone area
m_Pcb->Remove( item );
break;
2009-08-01 19:26:05 +00:00
case TYPE_DRAWSEGMENT: // a segment not on copper layers
case TYPE_TEXTE: // a text on a layer
case TYPE_TRACK: // a track segment (segment on a copper layer)
case TYPE_VIA: // a via (like atrack segment on a copper layer)
case TYPE_COTATION: // a dimension (graphic item)
case TYPE_MIRE: // a target (graphic item)
item->UnLink();
break;
2009-08-06 15:42:09 +00:00
// These items are deleted, but not put in undo list
case TYPE_MARKER_PCB: // a marker used to show something
case TYPE_ZONE: // a segment used to fill a zome area (segment on a copper layer)
item->UnLink();
itemsList->RemovePicker( ii );
ii--;
item->DeleteStructure();
break;
default:
2009-08-01 19:26:05 +00:00
wxMessageBox( wxT( "WinEDA_PcbFrame::Block_Delete( ) error: unexpected type" ) );
break;
}
}
2009-08-01 19:26:05 +00:00
SaveCopyInUndoList( *itemsList, UR_DELETED );
Compile_Ratsnest( NULL, TRUE );
2008-01-06 12:43:57 +00:00
DrawPanel->Refresh( TRUE );
2007-05-06 16:03:28 +00:00
}
2009-08-01 19:26:05 +00:00
/**
* Function Block_Rotate
* Rotate all items within the selected block.
* The rotation centre is the centre of the block
* @param none
*/
2009-08-01 19:26:05 +00:00
void WinEDA_PcbFrame::Block_Rotate()
2008-01-06 12:43:57 +00:00
{
2009-08-01 19:26:05 +00:00
wxPoint oldpos;
wxPoint centre; // rotation centre for the rotation transform
int rotAngle = 900; // rottaion angle in 0.1 deg.
2009-08-01 19:26:05 +00:00
if( !InstallBlockCmdFrame( this, _( "Rotate Block" ) ) )
return;
2009-08-01 19:26:05 +00:00
Block_SelectItems();
if( GetScreen()->m_BlockLocate.GetCount() == 0 )
return;
2009-08-01 19:26:05 +00:00
oldpos = GetScreen()->m_Curseur;
centre = GetScreen()->m_BlockLocate.Centre(); // This is the rotation centre
GetScreen()->SetModify();
2009-08-01 19:26:05 +00:00
PICKED_ITEMS_LIST* itemsList = &GetScreen()->m_BlockLocate.m_ItemsSelection;
itemsList->m_Status = UR_ROTATED;
for( unsigned ii = 0; ii < itemsList->GetCount(); ii++ )
{
2009-08-01 19:26:05 +00:00
BOARD_ITEM* item = (BOARD_ITEM*) itemsList->GetPickedItem( ii );
wxASSERT(item);
itemsList->SetPickedItemStatus( UR_ROTATED, ii );
item->Rotate(centre, rotAngle);
switch( item->Type() )
{
2009-08-01 19:26:05 +00:00
case TYPE_MODULE:
((MODULE*) item)->m_Flags = 0;
m_Pcb->m_Status_Pcb = 0;
2009-08-01 19:26:05 +00:00
break;
/* Move and rotate the track segments */
case TYPE_TRACK: // a track segment (segment on a copper layer)
case TYPE_VIA: // a via (like atrack segment on a copper layer)
m_Pcb->m_Status_Pcb = 0;
break;
2009-08-01 19:26:05 +00:00
case TYPE_ZONE_CONTAINER:
case TYPE_DRAWSEGMENT:
case TYPE_TEXTE:
case TYPE_MIRE:
case TYPE_COTATION:
break;
2009-08-06 15:42:09 +00:00
// This item is not put in undo list
case TYPE_ZONE: // a segment used to fill a zome area (segment on a copper layer)
itemsList->RemovePicker( ii );
ii--;
break;
2009-08-01 19:26:05 +00:00
default:
wxMessageBox( wxT( "WinEDA_PcbFrame::Block_Rotate( ) error: unexpected type" ) );
break;
}
}
2009-08-01 19:26:05 +00:00
SaveCopyInUndoList( *itemsList, UR_ROTATED, centre );
2009-08-01 19:26:05 +00:00
Compile_Ratsnest( NULL, TRUE );
DrawPanel->Refresh( TRUE );
}
2009-08-01 19:26:05 +00:00
/**
* Function Block_Flip
* Flip items within the selected block.
* The flip centre is the centre of the block
* @param none
*/
void WinEDA_PcbFrame::Block_Flip()
{
#define INVERT( pos ) (pos) = center.y - ( (pos) - center.y )
wxPoint memo;
wxPoint center; /* position de l'axe d'inversion de l'ensemble des elements */
2009-08-01 19:26:05 +00:00
Block_SelectItems();
if( GetScreen()->m_BlockLocate.GetCount() == 0 )
return;
GetScreen()->SetModify();
PICKED_ITEMS_LIST* itemsList = &GetScreen()->m_BlockLocate.m_ItemsSelection;
itemsList->m_Status = UR_FLIPPED;
2009-08-01 19:26:05 +00:00
memo = GetScreen()->m_Curseur;
/* calcul du centre d'inversion */
center = GetScreen()->m_BlockLocate.Centre();
for( unsigned ii = 0; ii < itemsList->GetCount(); ii++ )
{
2009-08-01 19:26:05 +00:00
BOARD_ITEM* item = (BOARD_ITEM*) itemsList->GetPickedItem( ii );
wxASSERT(item);
itemsList->SetPickedItemStatus( UR_FLIPPED, ii );
item->Flip(center);
switch( item->Type() )
{
2009-08-01 19:26:05 +00:00
case TYPE_MODULE:
((MODULE*) item)->m_Flags = 0;
m_Pcb->m_Status_Pcb = 0;
break;
2009-08-01 19:26:05 +00:00
/* Move and rotate the track segments */
case TYPE_TRACK: // a track segment (segment on a copper layer)
case TYPE_VIA: // a via (like atrack segment on a copper layer)
m_Pcb->m_Status_Pcb = 0;
break;
2009-08-01 19:26:05 +00:00
case TYPE_ZONE_CONTAINER:
case TYPE_DRAWSEGMENT:
case TYPE_TEXTE:
case TYPE_MIRE:
case TYPE_COTATION:
break;
2009-08-06 15:42:09 +00:00
// This item is not put in undo list
case TYPE_ZONE: // a segment used to fill a zome area (segment on a copper layer)
itemsList->RemovePicker( ii );
ii--;
break;
2009-08-01 19:26:05 +00:00
default:
2009-08-01 19:26:05 +00:00
wxMessageBox( wxT( "WinEDA_PcbFrame::Block_Flip( ) error: unexpected type" ) );
break;
}
}
2009-08-01 19:26:05 +00:00
SaveCopyInUndoList( *itemsList, UR_FLIPPED, center );
Compile_Ratsnest( NULL, TRUE );
2008-01-06 12:43:57 +00:00
DrawPanel->Refresh( TRUE );
2007-05-06 16:03:28 +00:00
}
2009-08-01 19:26:05 +00:00
/**
* Function Block_Move
* moves all tracks and segments within the selected block.
* New location is determined by the current offset from the selected block's original location.
* @param none
*/
2009-08-01 19:26:05 +00:00
void WinEDA_PcbFrame::Block_Move()
2007-05-06 16:03:28 +00:00
{
if( !InstallBlockCmdFrame( this, _( "Move Block" ) ) )
return;
2009-08-01 19:26:05 +00:00
Block_SelectItems();
if( GetScreen()->m_BlockLocate.GetCount() == 0 )
return;
2009-08-01 19:26:05 +00:00
GetScreen()->SetModify();
2009-08-01 19:26:05 +00:00
wxPoint MoveVector = GetScreen()->m_BlockLocate.m_MoveVector;
2009-08-01 19:26:05 +00:00
PICKED_ITEMS_LIST* itemsList = &GetScreen()->m_BlockLocate.m_ItemsSelection;
itemsList->m_Status = UR_MOVED;
2009-08-01 19:26:05 +00:00
for( unsigned ii = 0; ii < itemsList->GetCount(); ii++ )
{
2009-08-01 19:26:05 +00:00
BOARD_ITEM* item = (BOARD_ITEM*) itemsList->GetPickedItem( ii );
itemsList->SetPickedItemStatus( UR_MOVED, ii );
item->Move( MoveVector );
2009-08-01 19:26:05 +00:00
switch( item->Type() )
2008-04-01 05:21:50 +00:00
{
2009-08-01 19:26:05 +00:00
case TYPE_MODULE:
m_Pcb->m_Status_Pcb = 0;
((MODULE*) item)->m_Flags = 0;
break;
2009-08-01 19:26:05 +00:00
/* Move track segments */
case TYPE_TRACK: // a track segment (segment on a copper layer)
case TYPE_VIA: // a via (like a track segment on a copper layer)
m_Pcb->m_Status_Pcb = 0;
break;
2009-08-01 19:26:05 +00:00
case TYPE_ZONE_CONTAINER:
case TYPE_DRAWSEGMENT:
case TYPE_TEXTE:
case TYPE_MIRE:
case TYPE_COTATION:
break;
2009-08-06 15:42:09 +00:00
// This item is not put in undo list
case TYPE_ZONE: // a segment used to fill a zome area (segment on a copper layer)
itemsList->RemovePicker( ii );
ii--;
break;
default:
2009-08-01 19:26:05 +00:00
wxMessageBox( wxT( "WinEDA_PcbFrame::Block_Move( ) error: unexpected type" ) );
break;
}
}
2009-08-01 19:26:05 +00:00
SaveCopyInUndoList( *itemsList, UR_MOVED, MoveVector );
Compile_Ratsnest( NULL, TRUE );
2008-01-06 12:43:57 +00:00
DrawPanel->Refresh( TRUE );
2007-05-06 16:03:28 +00:00
}
2009-08-01 19:26:05 +00:00
/**
* Function Block_Duplicate
* Duplicate all items within the selected block.
* New location is determined by the current offset from the selected block's original location.
* @param none
*/
2009-08-01 19:26:05 +00:00
void WinEDA_PcbFrame::Block_Duplicate()
2007-05-06 16:03:28 +00:00
{
wxPoint MoveVector = GetScreen()->m_BlockLocate.m_MoveVector;
if( !InstallBlockCmdFrame( this, _( "Copy Block" ) ) )
return;
2009-08-01 19:26:05 +00:00
Block_SelectItems();
if( GetScreen()->m_BlockLocate.GetCount() == 0 )
return;
GetScreen()->SetModify();
2009-08-01 19:26:05 +00:00
PICKED_ITEMS_LIST* itemsList = &GetScreen()->m_BlockLocate.m_ItemsSelection;
2009-08-01 19:26:05 +00:00
PICKED_ITEMS_LIST newList;
newList.m_Status = UR_NEW;
ITEM_PICKER picker(NULL, UR_NEW);
BOARD_ITEM * newitem;
for( unsigned ii = 0; ii < itemsList->GetCount(); ii++ )
{
BOARD_ITEM* item = (BOARD_ITEM*) itemsList->GetPickedItem( ii );
newitem = NULL;
switch( item->Type() )
{
case TYPE_MODULE:
{
2009-08-01 19:26:05 +00:00
MODULE* module = (MODULE*) item;
MODULE* new_module;
m_Pcb->m_Status_Pcb = 0;
2009-08-01 19:26:05 +00:00
module->m_Flags = 0;
newitem = new_module = new MODULE( m_Pcb );
new_module->Copy( module );
new_module->m_TimeStamp = GetTimeStamp();
m_Pcb->m_Modules.PushFront( new_module );
}
2009-08-01 19:26:05 +00:00
break;
2009-08-01 19:26:05 +00:00
case TYPE_TRACK:
case TYPE_VIA:
{
2009-08-01 19:26:05 +00:00
TRACK* track = (TRACK*) item;
m_Pcb->m_Status_Pcb = 0;
TRACK* new_track = track->Copy();
newitem = new_track;
m_Pcb->m_Track.PushFront( new_track );
}
2009-08-01 19:26:05 +00:00
break;
2009-08-01 19:26:05 +00:00
case TYPE_ZONE: // a segment used to fill a zome area (segment on a copper layer)
{
2009-08-06 15:42:09 +00:00
// SEG_ZONE items are not copied or put in undo list
// they must be recreated by zone filling
}
2009-08-01 19:26:05 +00:00
break;
2008-01-06 12:43:57 +00:00
2009-08-01 19:26:05 +00:00
case TYPE_ZONE_CONTAINER:
2008-04-01 05:21:50 +00:00
{
2009-08-01 19:26:05 +00:00
ZONE_CONTAINER* new_zone = new ZONE_CONTAINER( (BOARD*) item->GetParent() );
new_zone->Copy( (ZONE_CONTAINER*) item );
new_zone->m_TimeStamp = GetTimeStamp();
newitem = new_zone;
m_Pcb->Add( new_zone );
2008-04-01 05:21:50 +00:00
}
2009-08-01 19:26:05 +00:00
break;
case TYPE_DRAWSEGMENT:
2009-08-01 19:26:05 +00:00
{
DRAWSEGMENT* new_drawsegment = new DRAWSEGMENT( m_Pcb );
new_drawsegment->Copy( (DRAWSEGMENT*) item );
m_Pcb->Add( new_drawsegment );
newitem = new_drawsegment;
}
break;
case TYPE_TEXTE:
2009-08-01 19:26:05 +00:00
{
TEXTE_PCB* new_pcbtext = new TEXTE_PCB( m_Pcb );
new_pcbtext->Copy( (TEXTE_PCB*) item );
m_Pcb->Add( new_pcbtext );
newitem = new_pcbtext;
}
break;
case TYPE_MIRE:
2009-08-01 19:26:05 +00:00
{
MIREPCB* new_mire = new MIREPCB( m_Pcb );
new_mire->Copy( (MIREPCB*) item );
m_Pcb->Add( new_mire );
newitem = new_mire;
}
break;
case TYPE_COTATION:
2009-08-01 19:26:05 +00:00
{
COTATION* new_cotation = new COTATION( m_Pcb );
new_cotation->Copy( (COTATION*) item );
m_Pcb->Add( new_cotation );
newitem = new_cotation;
}
break;
default:
2009-08-01 19:26:05 +00:00
wxMessageBox( wxT( "WinEDA_PcbFrame::Block_Duplicate( ) error: unexpected type" ) );
break;
}
2009-08-01 19:26:05 +00:00
if ( newitem )
{
newitem->Move( MoveVector );
picker.m_PickedItem = newitem;
picker.m_PickedItemType = newitem->Type();
newList.PushItem(picker);
}
}
2009-08-01 19:26:05 +00:00
if( newList.GetCount() )
SaveCopyInUndoList( newList, UR_NEW );
2009-08-01 19:26:05 +00:00
Compile_Ratsnest( NULL, TRUE );
DrawPanel->Refresh( TRUE );
2007-05-06 16:03:28 +00:00
}