kicad/pcbnew/modules.cpp

600 lines
17 KiB
C++
Raw Normal View History

/***************/
/* modules.cpp */
/***************/
#include "fctsys.h"
#include "gr_basic.h"
#include "common.h"
#include "class_drawpanel.h"
#include "confirm.h"
#include "pcbnew.h"
2009-07-30 11:04:07 +00:00
#include "wxPcbStruct.h"
#include "trigo.h"
#include "protos.h"
#include "drag.h"
static void Abort_MoveOrCopyModule( WinEDA_DrawPanel* Panel, wxDC* DC );
2010-07-20 18:11:34 +00:00
static MODULE* s_ModuleInitialCopy = NULL; /* Copy of module for
* abort/undo command
*/
static PICKED_ITEMS_LIST s_PickedList; /* a picked list to
* save initial module
* and dragged tracks
*/
2007-08-10 19:14:51 +00:00
/* Show or hide module pads.
2007-08-10 19:14:51 +00:00
*/
void Show_Pads_On_Off( WinEDA_DrawPanel* panel, wxDC* DC, MODULE* module )
{
2007-08-10 19:14:51 +00:00
D_PAD* pt_pad;
bool pad_fill_tmp;
if( module == 0 )
return;
pad_fill_tmp = DisplayOpt.DisplayPadFill;
DisplayOpt.DisplayPadFill = true; /* Trace en SKETCH */
2007-08-10 19:14:51 +00:00
pt_pad = module->m_Pads;
for( ; pt_pad != NULL; pt_pad = pt_pad->Next() )
2007-08-10 19:14:51 +00:00
{
2008-04-01 05:21:50 +00:00
pt_pad->Draw( panel, DC, GR_XOR, g_Offset_Module );
2007-08-10 19:14:51 +00:00
}
DisplayOpt.DisplayPadFill = pad_fill_tmp;
}
2007-08-10 19:14:51 +00:00
/* Show or hide ratsnest
*/
2007-08-10 19:14:51 +00:00
void Rastnest_On_Off( WinEDA_DrawPanel* panel, wxDC* DC, MODULE* module )
{
WinEDA_BasePcbFrame* frame = (WinEDA_BasePcbFrame*) panel->GetParent();
2007-08-10 19:14:51 +00:00
frame->build_ratsnest_module( DC, module );
frame->trace_ratsnest_module( DC );
}
2007-08-10 19:14:51 +00:00
/* Get a module name from user and return a pointer to the corresponding module
2007-08-10 19:14:51 +00:00
*/
MODULE* WinEDA_BasePcbFrame::GetModuleByName()
{
2010-07-20 18:11:34 +00:00
wxString moduleName;
MODULE* module = NULL;
2007-08-10 19:14:51 +00:00
2010-07-20 18:11:34 +00:00
wxTextEntryDialog dlg( this, _( "Name:" ), _( "Search footprint" ), moduleName );
if( dlg.ShowModal() != wxID_OK )
return NULL; //Aborted by user
moduleName = dlg.GetValue();
2010-07-20 18:11:34 +00:00
moduleName.Trim( true );
moduleName.Trim( false );
if( !moduleName.IsEmpty() )
2007-08-10 19:14:51 +00:00
{
module = GetBoard()->m_Modules;
2007-08-10 19:14:51 +00:00
while( module )
{
2010-07-20 18:11:34 +00:00
if( module->m_Reference->m_Text.CmpNoCase( moduleName ) == 0 )
2007-08-10 19:14:51 +00:00
break;
module = module->Next();
}
}
return module;
}
2007-08-10 19:14:51 +00:00
void WinEDA_PcbFrame::StartMove_Module( MODULE* module, wxDC* DC )
{
2007-08-10 19:14:51 +00:00
if( module == NULL )
return;
2009-08-06 07:11:04 +00:00
if( s_ModuleInitialCopy )
delete s_ModuleInitialCopy;
s_PickedList.ClearItemsList(); // Should be empty, but...
// Creates a copy of the current module, for abort and undo commands
s_ModuleInitialCopy = new MODULE( GetBoard() );
2009-08-06 07:11:04 +00:00
s_ModuleInitialCopy->Copy( module );
s_ModuleInitialCopy->m_Flags = 0;
SetCurItem( module );
GetBoard()->m_Status_Pcb &= ~RATSNEST_ITEM_LOCAL_OK;
2007-08-10 19:14:51 +00:00
module->m_Flags |= IS_MOVED;
/* Show ratsnest. */
2010-07-20 18:11:34 +00:00
if( GetBoard()->IsElementVisible( RATSNEST_VISIBLE ) )
2007-08-10 19:14:51 +00:00
DrawGeneralRatsnest( DC );
EraseDragList();
2007-08-10 19:14:51 +00:00
if( g_Drag_Pistes_On )
{
Build_Drag_Liste( DrawPanel, DC, module );
ITEM_PICKER itemWrapper( NULL, UR_CHANGED );
for( unsigned ii = 0; ii < g_DragSegmentList.size(); ii++ )
2009-08-06 07:11:04 +00:00
{
TRACK* segm = g_DragSegmentList[ii].m_Segm;
2009-08-06 07:11:04 +00:00
itemWrapper.m_PickedItem = segm;
itemWrapper.m_Link = segm->Copy();
itemWrapper.m_Link->SetState( EDIT, OFF );
s_PickedList.PushItem( itemWrapper );
2009-08-06 07:11:04 +00:00
}
2007-08-10 19:14:51 +00:00
}
GetBoard()->m_Status_Pcb |= DO_NOT_SHOW_GENERAL_RASTNEST;
DrawPanel->ManageCurseur = Montre_Position_Empreinte;
DrawPanel->ForceCloseManageCurseur = Abort_MoveOrCopyModule;
DrawPanel->m_AutoPAN_Request = true;
2007-08-10 19:14:51 +00:00
// Erase the module.
if( DC )
2008-04-01 05:21:50 +00:00
{
int tmp = module->m_Flags;
module->m_Flags |= DO_NOT_DRAW;
DrawPanel->PostDirtyRect( module->GetBoundingBox() );
module->m_Flags = tmp;
}
2007-08-10 19:14:51 +00:00
DrawPanel->ManageCurseur( DrawPanel, DC, FALSE );
}
2007-08-10 19:14:51 +00:00
/* Called on a move or copy module command abort
*/
void Abort_MoveOrCopyModule( WinEDA_DrawPanel* Panel, wxDC* DC )
{
TRACK* pt_segm;
MODULE* module;
WinEDA_PcbFrame* pcbframe = (WinEDA_PcbFrame*) Panel->GetParent();
2007-08-10 19:14:51 +00:00
module = (MODULE*) pcbframe->GetScreen()->GetCurItem();
pcbframe->GetBoard()->m_Status_Pcb &= ~RATSNEST_ITEM_LOCAL_OK;
2007-08-10 19:14:51 +00:00
if( module )
{
// Erase the current footprint on screen
2007-08-10 19:14:51 +00:00
DrawModuleOutlines( Panel, DC, module );
/* If a move command: return to old position
* If a copy command, delete the new footprint
*/
if( module->m_Flags & IS_MOVED ) // Move command
2007-08-10 19:14:51 +00:00
{
if( g_Drag_Pistes_On )
{
/* Erase on screen dragged tracks */
for( unsigned ii = 0; ii < g_DragSegmentList.size(); ii++ )
2007-08-10 19:14:51 +00:00
{
pt_segm = g_DragSegmentList[ii].m_Segm;
2007-08-10 19:14:51 +00:00
pt_segm->Draw( Panel, DC, GR_XOR );
}
}
/* Go to old position for dragged tracks */
for( unsigned ii = 0; ii < g_DragSegmentList.size(); ii++ )
2007-08-10 19:14:51 +00:00
{
pt_segm = g_DragSegmentList[ii].m_Segm;
pt_segm->SetState( EDIT, OFF );
g_DragSegmentList[ii].SetInitialValues();
2007-08-10 19:14:51 +00:00
pt_segm->Draw( Panel, DC, GR_OR );
}
EraseDragList();
2007-08-10 19:14:51 +00:00
module->m_Flags = 0;
}
if( (module->m_Flags & IS_NEW) ) // Copy command: delete new footprint
2007-08-10 19:14:51 +00:00
{
module->DeleteStructure();
2007-08-10 19:14:51 +00:00
module = NULL;
pcbframe->GetBoard()->m_Status_Pcb = 0;
pcbframe->GetBoard()->m_NetInfo->BuildListOfNets();
2007-08-10 19:14:51 +00:00
}
}
/* Redraw the module. */
2009-08-06 07:11:04 +00:00
if( module && s_ModuleInitialCopy )
2007-08-10 19:14:51 +00:00
{
2009-08-06 07:11:04 +00:00
if( s_ModuleInitialCopy->m_Orient != module->m_Orient )
pcbframe->Rotate_Module( NULL,
module,
s_ModuleInitialCopy->m_Orient,
FALSE );
2009-08-06 07:11:04 +00:00
if( s_ModuleInitialCopy->GetLayer() != module->GetLayer() )
pcbframe->Change_Side_Module( module, NULL );
2008-04-01 05:21:50 +00:00
module->Draw( Panel, DC, GR_OR );
2007-08-10 19:14:51 +00:00
}
g_Drag_Pistes_On = FALSE;
Panel->ManageCurseur = NULL;
Panel->ForceCloseManageCurseur = NULL;
pcbframe->SetCurItem( NULL );
2009-08-06 07:11:04 +00:00
delete s_ModuleInitialCopy;
s_ModuleInitialCopy = NULL;
s_PickedList.ClearListAndDeleteItems();
// Display ratsnest is allowed
pcbframe->GetBoard()->m_Status_Pcb &= ~DO_NOT_SHOW_GENERAL_RASTNEST;
2010-07-20 18:11:34 +00:00
if( pcbframe->GetBoard()->IsElementVisible( RATSNEST_VISIBLE ) )
pcbframe->DrawGeneralRatsnest( DC );
#ifdef __WXMAC__
pcbframe->GetScreen()->SetRefreshReq();
#endif
}
/**
* Function Copie_Module
* Copy an existing footprint. The new footprint is added in module list
* @param module = footprint to copy
* @return a pointer on the new footprint (the copy of the existing footprint)
*/
MODULE* WinEDA_BasePcbFrame::Copie_Module( MODULE* module )
{
2007-08-10 19:14:51 +00:00
MODULE* newmodule;
2007-08-10 19:14:51 +00:00
if( module == NULL )
return NULL;
OnModify();
/* Duplicate module */
GetBoard()->m_Status_Pcb = 0;
newmodule = new MODULE( GetBoard() );
2007-08-10 19:14:51 +00:00
newmodule->Copy( module );
2008-12-06 08:21:54 +00:00
GetBoard()->Add( newmodule, ADD_APPEND );
2008-12-06 08:21:54 +00:00
2007-08-10 19:14:51 +00:00
newmodule->m_Flags = IS_NEW;
GetBoard()->m_NetInfo->BuildListOfNets();
newmodule->DisplayInfo( this );
GetBoard()->m_Status_Pcb &= ~RATSNEST_ITEM_LOCAL_OK;
2007-08-10 19:14:51 +00:00
return newmodule;
}
2007-08-10 19:14:51 +00:00
/* Redraw the footprint when moving the mouse.
2007-08-10 19:14:51 +00:00
*/
void Montre_Position_Empreinte( WinEDA_DrawPanel* panel, wxDC* DC, bool erase )
{
MODULE* module = (MODULE*) panel->GetScreen()->GetCurItem();
2007-08-10 19:14:51 +00:00
if( module == NULL )
return;
/* Erase current footprint. */
2007-08-10 19:14:51 +00:00
if( erase )
{
DrawModuleOutlines( panel, DC, module );
}
/* Redraw the module at the new position. */
g_Offset_Module = module->m_Pos - panel->GetScreen()->m_Curseur;
2007-08-10 19:14:51 +00:00
DrawModuleOutlines( panel, DC, module );
2007-08-10 19:14:51 +00:00
Dessine_Segments_Dragges( panel, DC );
}
/**
* Function Delete Module
* Remove a footprint from m_Modules linked list and put it in undelete buffer
* The ratsnest and pad list are recalculated
* @param module = footprint to delete
* @param DC = currentDevice Context. if NULL: do not redraw new ratsnest and
* screen
* @param aPromptBeforeDeleting : if true: ask for confirmation before deleting
2007-08-10 19:14:51 +00:00
*/
bool WinEDA_PcbFrame::Delete_Module( MODULE* module,
wxDC* DC,
bool aAskBeforeDeleting )
{
wxString msg;
2007-08-10 19:14:51 +00:00
if( module == NULL )
return FALSE;
module->DisplayInfo( this );
2007-08-10 19:14:51 +00:00
/* Confirm module delete. */
if( aAskBeforeDeleting )
2007-08-10 19:14:51 +00:00
{
2010-07-20 18:11:34 +00:00
msg.Printf( _( "Delete Module %s (value %s) ?" ),
GetChars( module->m_Reference->m_Text ),
GetChars( module->m_Value->m_Text ) );
if( !IsOK( this, msg ) )
{
return FALSE;
}
2007-08-10 19:14:51 +00:00
}
OnModify();
2007-08-10 19:14:51 +00:00
/* Remove module from list, and put it in undo command list */
m_Pcb->m_Modules.Remove( module );
module->SetState( DELETED, ON );
SaveCopyInUndoList( module, UR_DELETED );
2007-08-10 19:14:51 +00:00
if( DC && GetBoard()->IsElementVisible( RATSNEST_VISIBLE ) )
Compile_Ratsnest( DC, true );
// Redraw the full screen to ensure perfect display of board and ratsnest.
if( DC )
2010-07-20 18:11:34 +00:00
DrawPanel->Refresh();
return true;
}
2007-08-10 19:14:51 +00:00
/**
* Function Change_Side_Module
* Flip a footprint (switch layer from component or component to copper)
* The mirroring is made from X axis
* if a footprint is not on copper or component layer it is not flipped
* (it could be on an adhesive layer, not supported at this time)
* @param Module the footprint to flip
* @param DC Current Device Context. if NULL, no redraw
2007-08-10 19:14:51 +00:00
*/
void WinEDA_PcbFrame::Change_Side_Module( MODULE* Module, wxDC* DC )
{
2007-08-10 19:14:51 +00:00
if( Module == NULL )
return;
if( ( Module->GetLayer() != LAYER_N_FRONT )
2010-07-20 18:11:34 +00:00
&& ( Module->GetLayer() != LAYER_N_BACK ) )
2007-08-10 19:14:51 +00:00
return;
OnModify();
2007-08-10 19:14:51 +00:00
if( !( Module->m_Flags & IS_MOVED ) ) /* This is a simple flip, no other
2010-07-20 18:11:34 +00:00
*edition in progress */
2007-08-10 19:14:51 +00:00
{
GetBoard()->m_Status_Pcb &= ~( LISTE_RATSNEST_ITEM_OK | CONNEXION_OK );
2009-08-06 07:11:04 +00:00
if( DC )
2008-04-01 05:21:50 +00:00
{
int tmp = Module->m_Flags;
Module->m_Flags |= DO_NOT_DRAW;
2009-08-06 07:11:04 +00:00
DrawPanel->PostDirtyRect( Module->GetBoundingBox() );
2008-04-01 05:21:50 +00:00
Module->m_Flags = tmp;
}
2007-08-10 19:14:51 +00:00
/* Show ratsnest if necessary. */
2010-07-20 18:11:34 +00:00
if( DC && GetBoard()->IsElementVisible( RATSNEST_VISIBLE ) )
2009-08-06 07:11:04 +00:00
DrawGeneralRatsnest( DC );
2007-08-10 19:14:51 +00:00
g_Offset_Module.x = 0;
g_Offset_Module.y = 0;
}
else // Module is being moved.
2007-08-10 19:14:51 +00:00
{
/* Erase footprint and draw outline if it has been already drawn. */
2009-08-06 07:11:04 +00:00
if( DC )
2007-08-10 19:14:51 +00:00
{
2009-08-06 07:11:04 +00:00
DrawModuleOutlines( DrawPanel, DC, Module );
Dessine_Segments_Dragges( DrawPanel, DC );
2007-08-10 19:14:51 +00:00
}
}
2009-08-01 19:26:05 +00:00
/* Flip the module */
Module->Flip( Module->m_Pos );
2007-08-10 19:14:51 +00:00
2009-08-06 07:11:04 +00:00
Module->DisplayInfo( this );
2007-08-10 19:14:51 +00:00
if( !( Module->m_Flags & IS_MOVED ) ) /* Inversion simple */
2007-08-10 19:14:51 +00:00
{
2009-08-06 07:11:04 +00:00
if( DC )
2007-08-10 19:14:51 +00:00
{
2009-08-06 07:11:04 +00:00
Module->Draw( DrawPanel, DC, GR_OR );
if( GetBoard()->IsElementVisible( RATSNEST_VISIBLE ) )
Compile_Ratsnest( DC, true );
2007-08-10 19:14:51 +00:00
}
}
else
{
2009-08-06 07:11:04 +00:00
if( DC )
2007-08-10 19:14:51 +00:00
{
2009-08-06 07:11:04 +00:00
DrawModuleOutlines( DrawPanel, DC, Module );
Dessine_Segments_Dragges( DrawPanel, DC );
2007-08-10 19:14:51 +00:00
}
2009-08-06 07:11:04 +00:00
GetBoard()->m_Status_Pcb &= ~RATSNEST_ITEM_LOCAL_OK;
2007-08-10 19:14:51 +00:00
}
}
/* Place module at cursor position.
*
* DC (if NULL: no display screen has the output.
* Update module coordinates with the new position.
2007-08-10 19:14:51 +00:00
*/
void WinEDA_BasePcbFrame::Place_Module( MODULE* module,
wxDC* DC,
bool aDoNotRecreateRatsnest )
{
TRACK* pt_segm;
wxPoint newpos;
2007-08-10 19:14:51 +00:00
if( module == 0 )
return;
OnModify();
GetBoard()->m_Status_Pcb &= ~( LISTE_RATSNEST_ITEM_OK | CONNEXION_OK);
2007-08-10 19:14:51 +00:00
if( module->m_Flags & IS_NEW )
2009-08-06 07:11:04 +00:00
{
SaveCopyInUndoList( module, UR_NEW );
2009-08-06 07:11:04 +00:00
}
else if( (module->m_Flags & IS_MOVED ) )
{
ITEM_PICKER picker( module, UR_CHANGED );
2009-08-06 07:11:04 +00:00
picker.m_Link = s_ModuleInitialCopy;
s_PickedList.PushItem( picker );
s_ModuleInitialCopy = NULL; // the picker is now owner of
// s_ModuleInitialCopy.
2009-08-06 07:11:04 +00:00
}
if( s_PickedList.GetCount() )
2009-08-06 07:11:04 +00:00
{
SaveCopyInUndoList( s_PickedList, UR_UNSPECIFIED );
2009-08-06 07:11:04 +00:00
// Clear list, but DO NOT delete items,
// because they are owned by the saved undo list and they therefore in
// use
2009-08-06 07:11:04 +00:00
s_PickedList.ClearItemsList();
}
if( g_Show_Module_Ratsnest && ( GetBoard()->m_Status_Pcb & LISTE_PAD_OK )
&& DC )
2007-08-10 19:14:51 +00:00
trace_ratsnest_module( DC );
newpos = GetScreen()->m_Curseur;
2007-08-10 19:14:51 +00:00
module->SetPosition( newpos );
2010-07-20 18:11:34 +00:00
module->m_Flags = 0;
delete s_ModuleInitialCopy;
s_ModuleInitialCopy = NULL;
2007-08-10 19:14:51 +00:00
if( DC )
2008-04-01 05:21:50 +00:00
module->Draw( DrawPanel, DC, GR_OR );
2007-08-10 19:14:51 +00:00
if( g_DragSegmentList.size() )
2007-08-10 19:14:51 +00:00
{
2009-08-06 07:11:04 +00:00
/* Redraw dragged track segments */
for( unsigned ii = 0; ii < g_DragSegmentList.size(); ii++ )
2007-08-10 19:14:51 +00:00
{
pt_segm = g_DragSegmentList[ii].m_Segm;
2007-08-10 19:14:51 +00:00
pt_segm->SetState( EDIT, OFF );
if( DC )
pt_segm->Draw( DrawPanel, DC, GR_OR );
}
2009-08-06 07:11:04 +00:00
// Delete drag list
EraseDragList();
2007-08-10 19:14:51 +00:00
}
g_Drag_Pistes_On = FALSE;
DrawPanel->ManageCurseur = NULL;
DrawPanel->ForceCloseManageCurseur = NULL;
if( GetBoard()->IsElementVisible( RATSNEST_VISIBLE ) )
if( !aDoNotRecreateRatsnest )
Compile_Ratsnest( DC, true );
2007-08-10 19:14:51 +00:00
if( DC )
2010-07-20 18:11:34 +00:00
DrawPanel->Refresh();
module->DisplayInfo( this );
}
/*
* Rotate the footprint angle degrees in the direction < 0.
* If incremental == true, the rotation is made from the last orientation,
* If the module is placed in the absolute orientation angle.
* If DC == NULL, the component does not redraw.
* Otherwise, it erases and redraws turns
2007-08-10 19:14:51 +00:00
*/
void WinEDA_BasePcbFrame::Rotate_Module( wxDC* DC, MODULE* module,
int angle, bool incremental )
{
2007-08-10 19:14:51 +00:00
if( module == NULL )
return;
OnModify();
2007-08-10 19:14:51 +00:00
if( !( module->m_Flags & IS_MOVED ) ) /* This is a simple rotation, no other
* edition in progress */
2007-08-10 19:14:51 +00:00
{
if( DC ) // Erase footprint to screen
2007-08-10 19:14:51 +00:00
{
2008-04-01 05:21:50 +00:00
int tmp = module->m_Flags;
module->m_Flags |= DO_NOT_DRAW;
DrawPanel->PostDirtyRect( module->GetBoundingBox() );
module->m_Flags = tmp;
2010-07-20 18:11:34 +00:00
if( GetBoard()->IsElementVisible( RATSNEST_VISIBLE ) )
2007-08-10 19:14:51 +00:00
DrawGeneralRatsnest( DC );
}
}
else
{
if( DC )
{
DrawModuleOutlines( DrawPanel, DC, module );
Dessine_Segments_Dragges( DrawPanel, DC );
}
}
GetBoard()->m_Status_Pcb &= ~( LISTE_RATSNEST_ITEM_OK | CONNEXION_OK );
2007-08-10 19:14:51 +00:00
if( incremental )
module->SetOrientation( module->m_Orient + angle );
else
module->SetOrientation( angle );
module->DisplayInfo( this );
2007-08-10 19:14:51 +00:00
if( DC )
{
if( !( module->m_Flags & IS_MOVED ) )
2010-07-20 18:11:34 +00:00
{
// not beiing moved: redraw the module and update ratsnest
2008-04-01 05:21:50 +00:00
module->Draw( DrawPanel, DC, GR_OR );
if( GetBoard()->IsElementVisible( RATSNEST_VISIBLE ) )
Compile_Ratsnest( DC, true );
2007-08-10 19:14:51 +00:00
}
else
2010-07-20 18:11:34 +00:00
{
// Beiing moved: just redraw it
2007-08-10 19:14:51 +00:00
DrawModuleOutlines( DrawPanel, DC, module );
Dessine_Segments_Dragges( DrawPanel, DC );
}
if( module->m_Flags == 0 ) // module not in edit: redraw full screen
2010-07-20 18:11:34 +00:00
DrawPanel->Refresh();
2007-08-10 19:14:51 +00:00
}
}
/*************************************************/
/* Redraw mode XOR the silhouette of the module. */
/*************************************************/
2007-08-10 19:14:51 +00:00
void DrawModuleOutlines( WinEDA_DrawPanel* panel, wxDC* DC, MODULE* module )
{
2007-08-10 19:14:51 +00:00
int pad_fill_tmp;
D_PAD* pt_pad;
if( module == NULL )
return;
module->DrawEdgesOnly( panel, DC, g_Offset_Module, GR_XOR );
if( g_Show_Pads_Module_in_Move )
{
pad_fill_tmp = DisplayOpt.DisplayPadFill;
DisplayOpt.DisplayPadFill = true;
2007-08-10 19:14:51 +00:00
pt_pad = module->m_Pads;
for( ; pt_pad != NULL; pt_pad = pt_pad->Next() )
2007-08-10 19:14:51 +00:00
{
2008-04-01 05:21:50 +00:00
pt_pad->Draw( panel, DC, GR_XOR, g_Offset_Module );
2007-08-10 19:14:51 +00:00
}
DisplayOpt.DisplayPadFill = pad_fill_tmp;
}
if( g_Show_Module_Ratsnest && panel )
{
WinEDA_BasePcbFrame* frame = (WinEDA_BasePcbFrame*) panel->GetParent();
2007-08-10 19:14:51 +00:00
frame->build_ratsnest_module( DC, module );
frame->trace_ratsnest_module( DC );
}
}