942 lines
28 KiB
C++
942 lines
28 KiB
C++
/*
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
*
|
|
* Copyright (C) 2004 Jean-Pierre Charras, jaen-pierre.charras@gipsa-lab.inpg.com
|
|
* Copyright (C) 2011 Wayne Stambaugh <stambaughw@verizon.net>
|
|
* Copyright (C) 1992-2011 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
|
|
*/
|
|
|
|
/**
|
|
* @file pcbnew/block.cpp
|
|
*/
|
|
|
|
|
|
#include "fctsys.h"
|
|
#include "class_drawpanel.h"
|
|
#include "confirm.h"
|
|
#include "block_commande.h"
|
|
#include "pcbcommon.h"
|
|
#include "wxPcbStruct.h"
|
|
#include "trigo.h"
|
|
|
|
#include "class_board.h"
|
|
#include "class_track.h"
|
|
#include "class_drawsegment.h"
|
|
#include "class_pcb_text.h"
|
|
#include "class_mire.h"
|
|
#include "class_module.h"
|
|
#include "class_dimension.h"
|
|
#include "class_zone.h"
|
|
|
|
#include "dialog_block_options_base.h"
|
|
|
|
#include "pcbnew.h"
|
|
#include "protos.h"
|
|
|
|
#define BLOCK_OUTLINE_COLOR YELLOW
|
|
|
|
/**
|
|
* Function drawPickedItems
|
|
* draws items currently selected in a block
|
|
* @param aPanel = Current draw panel
|
|
* @param aDC = Current device context
|
|
* @param aOffset = Drawing offset
|
|
**/
|
|
static void drawPickedItems( EDA_DRAW_PANEL* aPanel, wxDC* aDC, wxPoint aOffset );
|
|
|
|
/**
|
|
* Function drawMovingBlock
|
|
* handles drawing of a moving block
|
|
* @param aPanel = Current draw panel
|
|
* @param aDC = Current device context
|
|
* @param aPosition The cursor position in logical (drawing) units.
|
|
* @param aErase = Erase block at current position
|
|
**/
|
|
static void drawMovingBlock( EDA_DRAW_PANEL* aPanel, wxDC* aDC, const wxPoint& aPosition,
|
|
bool aErase );
|
|
|
|
|
|
static bool blockIncludeModules = true;
|
|
static bool blockIncludeLockedModules = true;
|
|
static bool blockIncludeTracks = true;
|
|
static bool blockIncludeZones = true;
|
|
static bool blockIncludeItemsOnTechLayers = true;
|
|
static bool blockIncludeBoardOutlineLayer = true;
|
|
static bool blockIncludePcbTexts = true;
|
|
static bool blockDrawItems = true;
|
|
static bool blockIncludeItemsOnInvisibleLayers = false;
|
|
|
|
|
|
/************************************/
|
|
/* class DIALOG_BLOCK_OPTIONS */
|
|
/************************************/
|
|
|
|
class DIALOG_BLOCK_OPTIONS : public DIALOG_BLOCK_OPTIONS_BASE
|
|
{
|
|
private:
|
|
PCB_BASE_FRAME* m_Parent;
|
|
|
|
public:
|
|
|
|
DIALOG_BLOCK_OPTIONS( PCB_BASE_FRAME* parent, const wxString& title );
|
|
~DIALOG_BLOCK_OPTIONS()
|
|
{
|
|
}
|
|
|
|
|
|
private:
|
|
void ExecuteCommand( wxCommandEvent& event );
|
|
void OnCancel( wxCommandEvent& event )
|
|
{
|
|
EndModal( wxID_CANCEL );
|
|
}
|
|
void checkBoxClicked( wxCommandEvent& aEvent )
|
|
{
|
|
if( m_Include_Modules->GetValue() )
|
|
m_IncludeLockedModules->Enable();
|
|
else
|
|
m_IncludeLockedModules->Disable();
|
|
}
|
|
};
|
|
|
|
|
|
static bool InstallBlockCmdFrame( PCB_BASE_FRAME* parent, const wxString& title )
|
|
{
|
|
wxPoint oldpos = parent->GetScreen()->GetCrossHairPosition();
|
|
|
|
parent->GetCanvas()->SetIgnoreMouseEvents( true );
|
|
DIALOG_BLOCK_OPTIONS * dlg = new DIALOG_BLOCK_OPTIONS( parent, title );
|
|
|
|
int cmd = dlg->ShowModal();
|
|
dlg->Destroy();
|
|
|
|
parent->GetScreen()->SetCrossHairPosition( oldpos );
|
|
parent->GetCanvas()->MoveCursorToCrossHair();
|
|
parent->GetCanvas()->SetIgnoreMouseEvents( false );
|
|
|
|
return cmd == wxID_OK;
|
|
}
|
|
|
|
|
|
DIALOG_BLOCK_OPTIONS::DIALOG_BLOCK_OPTIONS( PCB_BASE_FRAME* aParent, const wxString& aTitle ) :
|
|
DIALOG_BLOCK_OPTIONS_BASE( aParent, -1, aTitle )
|
|
{
|
|
m_Parent = aParent;
|
|
|
|
m_Include_Modules->SetValue( blockIncludeModules );
|
|
m_IncludeLockedModules->SetValue( blockIncludeLockedModules );
|
|
|
|
if( m_Include_Modules->GetValue() )
|
|
m_IncludeLockedModules->Enable();
|
|
else
|
|
m_IncludeLockedModules->Disable();
|
|
|
|
m_Include_Tracks->SetValue( blockIncludeTracks );
|
|
m_Include_Zones->SetValue( blockIncludeZones );
|
|
m_Include_Draw_Items->SetValue( blockIncludeItemsOnTechLayers );
|
|
m_Include_Edges_Items->SetValue( blockIncludeBoardOutlineLayer );
|
|
m_Include_PcbTextes->SetValue( blockIncludePcbTexts );
|
|
m_DrawBlockItems->SetValue( blockDrawItems );
|
|
m_checkBoxIncludeInvisible->SetValue( blockIncludeItemsOnInvisibleLayers );
|
|
m_sdbSizer1OK->SetDefault();
|
|
SetFocus();
|
|
GetSizer()->SetSizeHints( this );
|
|
Centre();
|
|
}
|
|
|
|
|
|
void DIALOG_BLOCK_OPTIONS::ExecuteCommand( wxCommandEvent& event )
|
|
{
|
|
blockIncludeModules = m_Include_Modules->GetValue();
|
|
blockIncludeLockedModules = m_IncludeLockedModules->GetValue();
|
|
blockIncludeTracks = m_Include_Tracks->GetValue();
|
|
blockIncludeZones = m_Include_Zones->GetValue();
|
|
blockIncludeItemsOnTechLayers = m_Include_Draw_Items->GetValue();
|
|
blockIncludeBoardOutlineLayer = m_Include_Edges_Items->GetValue();
|
|
blockIncludePcbTexts = m_Include_PcbTextes->GetValue();
|
|
blockDrawItems = m_DrawBlockItems->GetValue();
|
|
blockIncludeItemsOnInvisibleLayers = m_checkBoxIncludeInvisible->GetValue();
|
|
|
|
EndModal( wxID_OK );
|
|
}
|
|
|
|
|
|
int PCB_EDIT_FRAME::ReturnBlockCommand( int aKey )
|
|
{
|
|
int cmd = 0;
|
|
|
|
switch( aKey )
|
|
{
|
|
default:
|
|
cmd = aKey & 0x255;
|
|
break;
|
|
|
|
case 0:
|
|
cmd = BLOCK_MOVE;
|
|
break;
|
|
|
|
case GR_KB_SHIFT:
|
|
cmd = BLOCK_COPY;
|
|
break;
|
|
|
|
case GR_KB_CTRL:
|
|
cmd = BLOCK_ROTATE;
|
|
break;
|
|
|
|
case GR_KB_SHIFTCTRL:
|
|
cmd = BLOCK_DELETE;
|
|
break;
|
|
|
|
case GR_KB_ALT:
|
|
cmd = BLOCK_FLIP;
|
|
break;
|
|
|
|
case MOUSE_MIDDLE:
|
|
cmd = BLOCK_ZOOM;
|
|
break;
|
|
}
|
|
|
|
return cmd;
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::HandleBlockPlace( wxDC* DC )
|
|
{
|
|
if( !m_canvas->IsMouseCaptured() )
|
|
{
|
|
DisplayError( this, wxT( "Error in HandleBlockPLace : m_mouseCaptureCallback = NULL" ) );
|
|
}
|
|
|
|
GetScreen()->m_BlockLocate.m_State = STATE_BLOCK_STOP;
|
|
|
|
switch( GetScreen()->m_BlockLocate.m_Command )
|
|
{
|
|
case BLOCK_IDLE:
|
|
break;
|
|
|
|
case BLOCK_DRAG: /* Drag */
|
|
case BLOCK_MOVE: /* Move */
|
|
case BLOCK_PRESELECT_MOVE: /* Move with preselection list*/
|
|
if( m_canvas->IsMouseCaptured() )
|
|
m_canvas->CallMouseCapture( DC, wxDefaultPosition, false );
|
|
|
|
Block_Move();
|
|
GetScreen()->m_BlockLocate.ClearItemsList();
|
|
break;
|
|
|
|
case BLOCK_COPY: /* Copy */
|
|
if( m_canvas->IsMouseCaptured() )
|
|
m_canvas->CallMouseCapture( DC, wxDefaultPosition, false );
|
|
|
|
Block_Duplicate();
|
|
GetScreen()->m_BlockLocate.ClearItemsList();
|
|
break;
|
|
|
|
case BLOCK_PASTE:
|
|
break;
|
|
|
|
case BLOCK_ZOOM: // Handled by HandleBlockEnd()
|
|
default:
|
|
break;
|
|
}
|
|
|
|
OnModify();
|
|
|
|
m_canvas->EndMouseCapture( GetToolId(), m_canvas->GetCurrentCursor(), wxEmptyString, false );
|
|
GetScreen()->ClearBlockCommand();
|
|
|
|
if( GetScreen()->m_BlockLocate.GetCount() )
|
|
{
|
|
DisplayError( this, wxT( "Error in HandleBlockPLace some items left in list" ) );
|
|
GetScreen()->m_BlockLocate.ClearItemsList();
|
|
}
|
|
}
|
|
|
|
|
|
bool PCB_EDIT_FRAME::HandleBlockEnd( wxDC* DC )
|
|
{
|
|
bool nextcmd = false; // Will be set to true if a block place is needed
|
|
bool cancelCmd = false;
|
|
|
|
// If coming here after cancel block, clean up and exit
|
|
if( GetScreen()->m_BlockLocate.m_State == STATE_NO_BLOCK )
|
|
{
|
|
m_canvas->EndMouseCapture( GetToolId(), m_canvas->GetCurrentCursor(), wxEmptyString,
|
|
false );
|
|
GetScreen()->ClearBlockCommand();
|
|
return false;
|
|
}
|
|
|
|
// Show dialog if there are no selected items and we're not zooming
|
|
if( !GetScreen()->m_BlockLocate.GetCount()
|
|
&& GetScreen()->m_BlockLocate.m_Command != BLOCK_ZOOM )
|
|
{
|
|
if( InstallBlockCmdFrame( this, _( "Block Operation" ) ) == false )
|
|
{
|
|
cancelCmd = true;
|
|
|
|
// undraw block outline
|
|
m_canvas->CallMouseCapture( DC, wxDefaultPosition, false );
|
|
}
|
|
else
|
|
{
|
|
DrawAndSizingBlockOutlines( m_canvas, DC, wxDefaultPosition, false );
|
|
Block_SelectItems();
|
|
|
|
// Exit if no items found
|
|
if( !GetScreen()->m_BlockLocate.GetCount() )
|
|
cancelCmd = true;
|
|
}
|
|
}
|
|
|
|
if( !cancelCmd && m_canvas->IsMouseCaptured() )
|
|
{
|
|
switch( GetScreen()->m_BlockLocate.m_Command )
|
|
{
|
|
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;
|
|
nextcmd = true;
|
|
m_canvas->SetMouseCaptureCallback( drawMovingBlock );
|
|
m_canvas->CallMouseCapture( DC, wxDefaultPosition, false );
|
|
break;
|
|
|
|
case BLOCK_DELETE: /* Delete */
|
|
m_canvas->SetMouseCaptureCallback( NULL );
|
|
GetScreen()->m_BlockLocate.m_State = STATE_BLOCK_STOP;
|
|
Block_Delete();
|
|
break;
|
|
|
|
case BLOCK_ROTATE: /* Rotation */
|
|
m_canvas->SetMouseCaptureCallback( NULL );
|
|
GetScreen()->m_BlockLocate.m_State = STATE_BLOCK_STOP;
|
|
Block_Rotate();
|
|
break;
|
|
|
|
case BLOCK_FLIP: /* Flip */
|
|
m_canvas->SetMouseCaptureCallback( NULL );
|
|
GetScreen()->m_BlockLocate.m_State = STATE_BLOCK_STOP;
|
|
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() )
|
|
{
|
|
// @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
|
|
m_canvas->SetMouseCaptureCallback( NULL );
|
|
Window_Zoom( GetScreen()->m_BlockLocate );
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
if( ! nextcmd )
|
|
{
|
|
GetScreen()->ClearBlockCommand();
|
|
m_canvas->EndMouseCapture( GetToolId(), m_canvas->GetCurrentCursor(), wxEmptyString,
|
|
false );
|
|
}
|
|
|
|
return nextcmd;
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::Block_SelectItems()
|
|
{
|
|
int layerMask;
|
|
|
|
GetScreen()->m_BlockLocate.Normalize();
|
|
|
|
PICKED_ITEMS_LIST* itemsList = &GetScreen()->m_BlockLocate.m_ItemsSelection;
|
|
ITEM_PICKER picker( NULL, UR_UNSPECIFIED );
|
|
|
|
// Add modules
|
|
if( blockIncludeModules )
|
|
{
|
|
for( MODULE* module = m_Pcb->m_Modules; module != NULL; module = module->Next() )
|
|
{
|
|
int layer = module->GetLayer();
|
|
|
|
if( module->HitTest( GetScreen()->m_BlockLocate )
|
|
&& ( !module->IsLocked() || blockIncludeLockedModules ) )
|
|
{
|
|
if( blockIncludeItemsOnInvisibleLayers || m_Pcb->IsModuleLayerVisible( layer ) )
|
|
{
|
|
picker.m_PickedItem = module;
|
|
picker.m_PickedItemType = module->Type();
|
|
itemsList->PushItem( picker );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Add tracks and vias
|
|
if( blockIncludeTracks )
|
|
{
|
|
for( TRACK* pt_segm = m_Pcb->m_Track; pt_segm != NULL; pt_segm = pt_segm->Next() )
|
|
{
|
|
if( pt_segm->HitTest( GetScreen()->m_BlockLocate ) )
|
|
{
|
|
if( blockIncludeItemsOnInvisibleLayers
|
|
|| m_Pcb->IsLayerVisible( pt_segm->GetLayer() ) )
|
|
{
|
|
picker.m_PickedItem = pt_segm;
|
|
picker.m_PickedItemType = pt_segm->Type();
|
|
itemsList->PushItem( picker );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Add graphic items
|
|
layerMask = EDGE_LAYER;
|
|
|
|
if( blockIncludeItemsOnTechLayers )
|
|
layerMask = ALL_LAYERS;
|
|
|
|
if( !blockIncludeBoardOutlineLayer )
|
|
layerMask &= ~EDGE_LAYER;
|
|
|
|
for( BOARD_ITEM* PtStruct = m_Pcb->m_Drawings; PtStruct != NULL; PtStruct = PtStruct->Next() )
|
|
{
|
|
if( !m_Pcb->IsLayerVisible( PtStruct->GetLayer() ) && ! blockIncludeItemsOnInvisibleLayers)
|
|
continue;
|
|
bool select_me = false;
|
|
switch( PtStruct->Type() )
|
|
{
|
|
case PCB_LINE_T:
|
|
if( (GetLayerMask( PtStruct->GetLayer() ) & layerMask) == 0 )
|
|
break;
|
|
|
|
if( !PtStruct->HitTest( GetScreen()->m_BlockLocate ) )
|
|
break;
|
|
|
|
select_me = true; // This item is in bloc: select it
|
|
break;
|
|
|
|
case PCB_TEXT_T:
|
|
if( !blockIncludePcbTexts )
|
|
break;
|
|
|
|
if( !PtStruct->HitTest( GetScreen()->m_BlockLocate ) )
|
|
break;
|
|
|
|
select_me = true; // This item is in bloc: select it
|
|
break;
|
|
|
|
case PCB_TARGET_T:
|
|
if( ( GetLayerMask( PtStruct->GetLayer() ) & layerMask ) == 0 )
|
|
break;
|
|
|
|
if( !PtStruct->HitTest( GetScreen()->m_BlockLocate ) )
|
|
break;
|
|
|
|
select_me = true; // This item is in bloc: select it
|
|
break;
|
|
|
|
case PCB_DIMENSION_T:
|
|
if( ( GetLayerMask( PtStruct->GetLayer() ) & layerMask ) == 0 )
|
|
break;
|
|
|
|
if( !PtStruct->HitTest( GetScreen()->m_BlockLocate ) )
|
|
break;
|
|
|
|
select_me = true; // This item is in bloc: select it
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if( select_me )
|
|
{
|
|
picker.m_PickedItem = PtStruct;
|
|
picker.m_PickedItemType = PtStruct->Type();
|
|
itemsList->PushItem( picker );
|
|
}
|
|
}
|
|
|
|
// Add zones
|
|
if( blockIncludeZones )
|
|
{
|
|
for( int ii = 0; ii < m_Pcb->GetAreaCount(); ii++ )
|
|
{
|
|
ZONE_CONTAINER* area = m_Pcb->GetArea( ii );
|
|
|
|
if( area->HitTest( GetScreen()->m_BlockLocate ) )
|
|
{
|
|
if( blockIncludeItemsOnInvisibleLayers
|
|
|| m_Pcb->IsLayerVisible( area->GetLayer() ) )
|
|
{
|
|
BOARD_ITEM* zone_c = (BOARD_ITEM*) area;
|
|
picker.m_PickedItem = zone_c;
|
|
picker.m_PickedItemType = zone_c->Type();
|
|
itemsList->PushItem( picker );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static void drawPickedItems( EDA_DRAW_PANEL* aPanel, wxDC* aDC, wxPoint aOffset )
|
|
{
|
|
PICKED_ITEMS_LIST* itemsList = &aPanel->GetScreen()->m_BlockLocate.m_ItemsSelection;
|
|
PCB_BASE_FRAME* frame = (PCB_BASE_FRAME*) aPanel->GetParent();
|
|
|
|
g_Offset_Module = -aOffset;
|
|
|
|
for( unsigned ii = 0; ii < itemsList->GetCount(); ii++ )
|
|
{
|
|
BOARD_ITEM* item = (BOARD_ITEM*) itemsList->GetPickedItem( ii );
|
|
|
|
switch( item->Type() )
|
|
{
|
|
case PCB_MODULE_T:
|
|
frame->GetBoard()->m_Status_Pcb &= ~RATSNEST_ITEM_LOCAL_OK;
|
|
DrawModuleOutlines( aPanel, aDC, (MODULE*) item );
|
|
break;
|
|
|
|
case PCB_LINE_T:
|
|
case PCB_TEXT_T:
|
|
case PCB_TRACE_T:
|
|
case PCB_VIA_T:
|
|
case PCB_TARGET_T:
|
|
case PCB_DIMENSION_T: // Currently markers are not affected by block commands
|
|
case PCB_MARKER_T:
|
|
item->Draw( aPanel, aDC, GR_XOR, aOffset );
|
|
break;
|
|
|
|
case PCB_ZONE_AREA_T:
|
|
item->Draw( aPanel, aDC, GR_XOR, aOffset );
|
|
((ZONE_CONTAINER*) item)->DrawFilledArea( aPanel, aDC, GR_XOR, aOffset );
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
g_Offset_Module = wxPoint( 0, 0 );
|
|
}
|
|
|
|
|
|
static void drawMovingBlock( EDA_DRAW_PANEL* aPanel, wxDC* aDC, const wxPoint& aPosition,
|
|
bool aErase )
|
|
{
|
|
BASE_SCREEN* screen = aPanel->GetScreen();
|
|
|
|
if( aErase )
|
|
{
|
|
if( screen->m_BlockLocate.m_MoveVector.x || screen->m_BlockLocate.m_MoveVector.y )
|
|
{
|
|
screen->m_BlockLocate.Draw( aPanel, aDC, screen->m_BlockLocate.m_MoveVector,
|
|
GR_XOR, BLOCK_OUTLINE_COLOR );
|
|
|
|
if( blockDrawItems )
|
|
drawPickedItems( aPanel, aDC, screen->m_BlockLocate.m_MoveVector );
|
|
}
|
|
}
|
|
|
|
if( screen->m_BlockLocate.m_State != STATE_BLOCK_STOP )
|
|
{
|
|
screen->m_BlockLocate.m_MoveVector = screen->GetCrossHairPosition() -
|
|
screen->m_BlockLocate.m_BlockLastCursorPosition;
|
|
}
|
|
|
|
if( screen->m_BlockLocate.m_MoveVector.x || screen->m_BlockLocate.m_MoveVector.y )
|
|
{
|
|
screen->m_BlockLocate.Draw( aPanel, aDC, screen->m_BlockLocate.m_MoveVector,
|
|
GR_XOR, BLOCK_OUTLINE_COLOR );
|
|
|
|
if( blockDrawItems )
|
|
drawPickedItems( aPanel, aDC, screen->m_BlockLocate.m_MoveVector );
|
|
}
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::Block_Delete()
|
|
{
|
|
OnModify();
|
|
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++ )
|
|
{
|
|
BOARD_ITEM* item = (BOARD_ITEM*) itemsList->GetPickedItem( ii );
|
|
itemsList->SetPickedItemStatus( UR_DELETED, ii );
|
|
|
|
switch( item->Type() )
|
|
{
|
|
case PCB_MODULE_T:
|
|
{
|
|
MODULE* module = (MODULE*) item;
|
|
module->ClearFlags();
|
|
module->UnLink();
|
|
m_Pcb->m_Status_Pcb = 0;
|
|
}
|
|
break;
|
|
|
|
case PCB_ZONE_AREA_T: // a zone area
|
|
m_Pcb->Remove( item );
|
|
break;
|
|
|
|
case PCB_LINE_T: // a segment not on copper layers
|
|
case PCB_TEXT_T: // a text on a layer
|
|
case PCB_TRACE_T: // a track segment (segment on a copper layer)
|
|
case PCB_VIA_T: // a via (like track segment on a copper layer)
|
|
case PCB_DIMENSION_T: // a dimension (graphic item)
|
|
case PCB_TARGET_T: // a target (graphic item)
|
|
item->UnLink();
|
|
break;
|
|
|
|
// These items are deleted, but not put in undo list
|
|
case PCB_MARKER_T: // a marker used to show something
|
|
case PCB_ZONE_T: // SEG_ZONE items are now deprecated
|
|
item->UnLink();
|
|
itemsList->RemovePicker( ii );
|
|
ii--;
|
|
item->DeleteStructure();
|
|
break;
|
|
|
|
default:
|
|
wxMessageBox( wxT( "PCB_EDIT_FRAME::Block_Delete( ) error: unexpected type" ) );
|
|
break;
|
|
}
|
|
}
|
|
|
|
SaveCopyInUndoList( *itemsList, UR_DELETED );
|
|
|
|
Compile_Ratsnest( NULL, true );
|
|
m_canvas->Refresh( true );
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::Block_Rotate()
|
|
{
|
|
wxPoint oldpos;
|
|
wxPoint centre; // rotation cent-re for the rotation transform
|
|
int rotAngle = 900; // rotation angle in 0.1 deg.
|
|
|
|
oldpos = GetScreen()->GetCrossHairPosition();
|
|
centre = GetScreen()->m_BlockLocate.Centre();
|
|
|
|
OnModify();
|
|
|
|
PICKED_ITEMS_LIST* itemsList = &GetScreen()->m_BlockLocate.m_ItemsSelection;
|
|
itemsList->m_Status = UR_ROTATED;
|
|
|
|
for( unsigned ii = 0; ii < itemsList->GetCount(); ii++ )
|
|
{
|
|
BOARD_ITEM* item = (BOARD_ITEM*) itemsList->GetPickedItem( ii );
|
|
wxASSERT( item );
|
|
itemsList->SetPickedItemStatus( UR_ROTATED, ii );
|
|
item->Rotate( centre, rotAngle );
|
|
|
|
switch( item->Type() )
|
|
{
|
|
case PCB_MODULE_T:
|
|
( (MODULE*) item )->ClearFlags();
|
|
m_Pcb->m_Status_Pcb = 0;
|
|
break;
|
|
|
|
/* Move and rotate the track segments */
|
|
case PCB_TRACE_T: // a track segment (segment on a copper layer)
|
|
case PCB_VIA_T: // a via (like track segment on a copper layer)
|
|
m_Pcb->m_Status_Pcb = 0;
|
|
break;
|
|
|
|
case PCB_ZONE_AREA_T:
|
|
case PCB_LINE_T:
|
|
case PCB_TEXT_T:
|
|
case PCB_TARGET_T:
|
|
case PCB_DIMENSION_T:
|
|
break;
|
|
|
|
// This item is not put in undo list
|
|
case PCB_ZONE_T: // SEG_ZONE items are now deprecated
|
|
itemsList->RemovePicker( ii );
|
|
ii--;
|
|
break;
|
|
|
|
default:
|
|
wxMessageBox( wxT( "PCB_EDIT_FRAME::Block_Rotate( ) error: unexpected type" ) );
|
|
break;
|
|
}
|
|
}
|
|
|
|
SaveCopyInUndoList( *itemsList, UR_ROTATED, centre );
|
|
|
|
Compile_Ratsnest( NULL, true );
|
|
m_canvas->Refresh( true );
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::Block_Flip()
|
|
{
|
|
#define INVERT( pos ) (pos) = center.y - ( (pos) - center.y )
|
|
wxPoint memo;
|
|
wxPoint center; /* Position of the axis for inversion of all elements */
|
|
|
|
OnModify();
|
|
|
|
PICKED_ITEMS_LIST* itemsList = &GetScreen()->m_BlockLocate.m_ItemsSelection;
|
|
itemsList->m_Status = UR_FLIPPED;
|
|
|
|
memo = GetScreen()->GetCrossHairPosition();
|
|
|
|
center = GetScreen()->m_BlockLocate.Centre();
|
|
|
|
for( unsigned ii = 0; ii < itemsList->GetCount(); ii++ )
|
|
{
|
|
BOARD_ITEM* item = (BOARD_ITEM*) itemsList->GetPickedItem( ii );
|
|
wxASSERT( item );
|
|
itemsList->SetPickedItemStatus( UR_FLIPPED, ii );
|
|
item->Flip( center );
|
|
|
|
switch( item->Type() )
|
|
{
|
|
case PCB_MODULE_T:
|
|
item->ClearFlags();
|
|
m_Pcb->m_Status_Pcb = 0;
|
|
break;
|
|
|
|
/* Move and rotate the track segments */
|
|
case PCB_TRACE_T: // a track segment (segment on a copper layer)
|
|
case PCB_VIA_T: // a via (like track segment on a copper layer)
|
|
m_Pcb->m_Status_Pcb = 0;
|
|
break;
|
|
|
|
case PCB_ZONE_AREA_T:
|
|
case PCB_LINE_T:
|
|
case PCB_TEXT_T:
|
|
case PCB_TARGET_T:
|
|
case PCB_DIMENSION_T:
|
|
break;
|
|
|
|
// This item is not put in undo list
|
|
case PCB_ZONE_T: // SEG_ZONE items are now deprecated
|
|
itemsList->RemovePicker( ii );
|
|
ii--;
|
|
break;
|
|
|
|
|
|
default:
|
|
wxMessageBox( wxT( "PCB_EDIT_FRAME::Block_Flip( ) error: unexpected type" ) );
|
|
break;
|
|
}
|
|
}
|
|
|
|
SaveCopyInUndoList( *itemsList, UR_FLIPPED, center );
|
|
Compile_Ratsnest( NULL, true );
|
|
m_canvas->Refresh( true );
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::Block_Move()
|
|
{
|
|
OnModify();
|
|
|
|
wxPoint MoveVector = GetScreen()->m_BlockLocate.m_MoveVector;
|
|
|
|
PICKED_ITEMS_LIST* itemsList = &GetScreen()->m_BlockLocate.m_ItemsSelection;
|
|
itemsList->m_Status = UR_MOVED;
|
|
|
|
for( unsigned ii = 0; ii < itemsList->GetCount(); ii++ )
|
|
{
|
|
BOARD_ITEM* item = (BOARD_ITEM*) itemsList->GetPickedItem( ii );
|
|
itemsList->SetPickedItemStatus( UR_MOVED, ii );
|
|
item->Move( MoveVector );
|
|
|
|
switch( item->Type() )
|
|
{
|
|
case PCB_MODULE_T:
|
|
m_Pcb->m_Status_Pcb = 0;
|
|
item->ClearFlags();
|
|
break;
|
|
|
|
/* Move track segments */
|
|
case PCB_TRACE_T: // a track segment (segment on a copper layer)
|
|
case PCB_VIA_T: // a via (like a track segment on a copper layer)
|
|
m_Pcb->m_Status_Pcb = 0;
|
|
break;
|
|
|
|
case PCB_ZONE_AREA_T:
|
|
case PCB_LINE_T:
|
|
case PCB_TEXT_T:
|
|
case PCB_TARGET_T:
|
|
case PCB_DIMENSION_T:
|
|
break;
|
|
|
|
// This item is not put in undo list
|
|
case PCB_ZONE_T: // SEG_ZONE items are now deprecated
|
|
itemsList->RemovePicker( ii );
|
|
ii--;
|
|
break;
|
|
|
|
default:
|
|
wxMessageBox( wxT( "PCB_EDIT_FRAME::Block_Move( ) error: unexpected type" ) );
|
|
break;
|
|
}
|
|
}
|
|
|
|
SaveCopyInUndoList( *itemsList, UR_MOVED, MoveVector );
|
|
|
|
Compile_Ratsnest( NULL, true );
|
|
m_canvas->Refresh( true );
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::Block_Duplicate()
|
|
{
|
|
wxPoint MoveVector = GetScreen()->m_BlockLocate.m_MoveVector;
|
|
|
|
OnModify();
|
|
|
|
PICKED_ITEMS_LIST* itemsList = &GetScreen()->m_BlockLocate.m_ItemsSelection;
|
|
|
|
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 PCB_MODULE_T:
|
|
{
|
|
MODULE* module = (MODULE*) item;
|
|
MODULE* new_module;
|
|
m_Pcb->m_Status_Pcb = 0;
|
|
module->ClearFlags();
|
|
newitem = new_module = new MODULE( m_Pcb );
|
|
new_module->Copy( module );
|
|
new_module->SetTimeStamp( GetNewTimeStamp() );
|
|
m_Pcb->m_Modules.PushFront( new_module );
|
|
}
|
|
break;
|
|
|
|
case PCB_TRACE_T:
|
|
case PCB_VIA_T:
|
|
{
|
|
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 );
|
|
}
|
|
break;
|
|
|
|
case PCB_ZONE_T: // SEG_ZONE items are now deprecated
|
|
break;
|
|
|
|
case PCB_ZONE_AREA_T:
|
|
{
|
|
ZONE_CONTAINER* new_zone = new ZONE_CONTAINER( (BOARD*) item->GetParent() );
|
|
new_zone->Copy( (ZONE_CONTAINER*) item );
|
|
new_zone->SetTimeStamp( GetNewTimeStamp() );
|
|
newitem = new_zone;
|
|
m_Pcb->Add( new_zone );
|
|
}
|
|
break;
|
|
|
|
case PCB_LINE_T:
|
|
{
|
|
DRAWSEGMENT* new_drawsegment = new DRAWSEGMENT( m_Pcb );
|
|
new_drawsegment->Copy( (DRAWSEGMENT*) item );
|
|
m_Pcb->Add( new_drawsegment );
|
|
newitem = new_drawsegment;
|
|
}
|
|
break;
|
|
|
|
case PCB_TEXT_T:
|
|
{
|
|
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 PCB_TARGET_T:
|
|
{
|
|
PCB_TARGET* target = new PCB_TARGET( m_Pcb );
|
|
target->Copy( (PCB_TARGET*) item );
|
|
m_Pcb->Add( target );
|
|
newitem = target;
|
|
}
|
|
break;
|
|
|
|
case PCB_DIMENSION_T:
|
|
{
|
|
DIMENSION* new_cotation = new DIMENSION( m_Pcb );
|
|
new_cotation->Copy( (DIMENSION*) item );
|
|
m_Pcb->Add( new_cotation );
|
|
newitem = new_cotation;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
wxMessageBox( wxT( "PCB_EDIT_FRAME::Block_Duplicate( ) error: unexpected type" ) );
|
|
break;
|
|
}
|
|
|
|
if( newitem )
|
|
{
|
|
newitem->Move( MoveVector );
|
|
picker.m_PickedItem = newitem;
|
|
picker.m_PickedItemType = newitem->Type();
|
|
newList.PushItem( picker );
|
|
}
|
|
}
|
|
|
|
if( newList.GetCount() )
|
|
SaveCopyInUndoList( newList, UR_NEW );
|
|
|
|
Compile_Ratsnest( NULL, true );
|
|
m_canvas->Refresh( true );
|
|
}
|