/********************************************************/ /* Modification de la place, orient, nombre des MODULES */ /********************************************************/ /* Fichier MODULES.Cpp */ #include "fctsys.h" #include "gr_basic.h" #include "common.h" #include "class_drawpanel.h" #include "confirm.h" #include "pcbnew.h" #include "autorout.h" #include "trigo.h" #include "protos.h" #include "drag.h" /* fonctions externes */ /* Fonctions locales */ static int ChangeSideMaskLayer( int masque ); static void Abort_MoveOrCopyModule( WinEDA_DrawPanel* Panel, wxDC* DC ); /* Variables locales : */ static int ModuleInitOrient; // Lors des moves, val init de l'orient (pour annulation) static int ModuleInitLayer; // Lors des moves, val init de la couche (pour annulation) /*************************************************************************/ void Show_Pads_On_Off( WinEDA_DrawPanel* panel, wxDC* DC, MODULE* module ) /**************************************************************************/ /* Fonction appelee lors de l'activation/desactivation de la visualisation * des Pads du module en deplacement * Effacement ou visu des Pads selon conditions initiales */ { D_PAD* pt_pad; bool pad_fill_tmp; if( module == 0 ) return; pad_fill_tmp = DisplayOpt.DisplayPadFill; DisplayOpt.DisplayPadFill = FALSE; /* Trace en SKETCH */ pt_pad = module->m_Pads; for( ; pt_pad != NULL; pt_pad = pt_pad->Next() ) { pt_pad->Draw( panel, DC, GR_XOR, g_Offset_Module ); } DisplayOpt.DisplayPadFill = pad_fill_tmp; } /***************************************************************************/ /* Fonction appelee lors de l'activation/desactivation de la visualisation */ /* du rastnest du module en deplacement */ /* Effacement ou visu du rastnest selon conditions initiales */ /***************************************************************************/ void Rastnest_On_Off( WinEDA_DrawPanel* panel, wxDC* DC, MODULE* module ) { WinEDA_BasePcbFrame* frame = (WinEDA_BasePcbFrame*) panel->m_Parent; frame->build_ratsnest_module( DC, module ); frame->trace_ratsnest_module( DC ); } /***************************************************/ MODULE* WinEDA_BasePcbFrame::GetModuleByName() /***************************************************/ /* Get a module name from user and return a pointer to the corresponding module */ { wxString modulename; MODULE* module = NULL; Get_Message( _( "Name:" ), _("Search footprint"), modulename, this ); if( !modulename.IsEmpty() ) { module = GetBoard()->m_Modules; while( module ) { if( module->m_Reference->m_Text.CmpNoCase( modulename ) == 0 ) break; module = module->Next(); } } return module; } /**********************************************************************/ void WinEDA_PcbFrame::StartMove_Module( MODULE* module, wxDC* DC ) /**********************************************************************/ { if( module == NULL ) return; SetCurItem( module ); GetBoard()->m_Status_Pcb &= ~CHEVELU_LOCAL_OK; module->m_Flags |= IS_MOVED; ModuleInitOrient = module->m_Orient; ModuleInitLayer = module->GetLayer(); /* Effacement chevelu general si necessaire */ if( g_Show_Ratsnest ) DrawGeneralRatsnest( DC ); if( g_DragSegmentList ) /* Anormal ! */ { EraseDragListe(); } if( g_Drag_Pistes_On ) { Build_Drag_Liste( DrawPanel, DC, module ); } GetBoard()->m_Status_Pcb |= DO_NOT_SHOW_GENERAL_RASTNEST; DrawPanel->ManageCurseur = Montre_Position_Empreinte; DrawPanel->ForceCloseManageCurseur = Abort_MoveOrCopyModule; DrawPanel->m_AutoPAN_Request = TRUE; // effacement module a l'ecran: if( DC ) { int tmp = module->m_Flags; module->m_Flags |= DO_NOT_DRAW; DrawPanel->PostDirtyRect( module->GetBoundingBox() ); module->m_Flags = tmp; } // Reaffichage DrawPanel->ManageCurseur( DrawPanel, DC, FALSE ); } /**************************************************************/ void Abort_MoveOrCopyModule( WinEDA_DrawPanel* Panel, wxDC* DC ) /****************************************************************/ /* Called on a move or copy module command abort */ { DRAG_SEGM* pt_drag; TRACK* pt_segm; MODULE* module; WinEDA_BasePcbFrame* pcbframe = (WinEDA_BasePcbFrame*) Panel->m_Parent; module = (MODULE*) pcbframe->GetScreen()->GetCurItem(); pcbframe->GetBoard()->m_Status_Pcb &= ~CHEVELU_LOCAL_OK; if( module ) { // Erase the current footprint on screen 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 { if( g_Drag_Pistes_On ) { /* Erase on screen dragged tracks */ pt_drag = g_DragSegmentList; for( ; pt_drag != NULL; pt_drag = pt_drag->Pnext ) { pt_segm = pt_drag->m_Segm; pt_segm->Draw( Panel, DC, GR_XOR ); } } /* Go to old position for dragged tracks */ pt_drag = g_DragSegmentList; for( ; pt_drag != NULL; pt_drag = pt_drag->Pnext ) { pt_segm = pt_drag->m_Segm; pt_segm->SetState( EDIT, OFF ); pt_drag->SetInitialValues(); pt_segm->Draw( Panel, DC, GR_OR ); } EraseDragListe(); module->m_Flags = 0; } if( module->m_Flags & IS_NEW ) // Copy command: delete new footprint { module->DeleteStructure(); module = NULL; pcbframe->GetBoard()->m_Status_Pcb = 0; pcbframe->build_liste_pads(); } } /* Reaffichage du module a l'ecran */ if( module ) { if( ModuleInitOrient != module->m_Orient ) pcbframe->Rotate_Module( NULL, module, ModuleInitOrient, FALSE ); if( ModuleInitLayer != module->GetLayer() ) pcbframe->GetBoard()->Change_Side_Module( module, NULL ); module->Draw( Panel, DC, GR_OR ); } g_Drag_Pistes_On = FALSE; Panel->ManageCurseur = NULL; Panel->ForceCloseManageCurseur = NULL; pcbframe->SetCurItem( NULL ); pcbframe->GetBoard()->m_Status_Pcb &= ~DO_NOT_SHOW_GENERAL_RASTNEST; // Display ratsnest is allowed if( g_Show_Ratsnest ) pcbframe->DrawGeneralRatsnest( DC ); } /**********************************************************/ MODULE* WinEDA_BasePcbFrame::Copie_Module( MODULE* module ) /**********************************************************/ /** * Function Copie_Module * Copy an existing footprint. The ne 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* newmodule; if( module == NULL ) return NULL; GetScreen()->SetModify(); /* Duplication du module */ GetBoard()->m_Status_Pcb = 0; newmodule = new MODULE( GetBoard() ); newmodule->Copy( module ); /* no, Add() below does this newmodule->SetParent( GetBoard() ); */ GetBoard()->Add( newmodule, ADD_APPEND ); newmodule->m_Flags = IS_NEW; build_liste_pads(); newmodule->Display_Infos( this ); GetBoard()->m_Status_Pcb &= ~CHEVELU_LOCAL_OK; return newmodule; } /**********************************************************************************/ void Montre_Position_Empreinte( WinEDA_DrawPanel* panel, wxDC* DC, bool erase ) /**********************************************************************************/ /* redessin du contour de l'empreinte lors des deplacements de la souris */ { MODULE* module = (MODULE*) panel->GetScreen()->GetCurItem(); if( module == NULL ) return; /* efface ancienne position */ if( erase ) { DrawModuleOutlines( panel, DC, module ); } /* Redessine le module a la nouvelle place */ g_Offset_Module = module->m_Pos - panel->GetScreen()->m_Curseur; DrawModuleOutlines( panel, DC, module ); Dessine_Segments_Dragges( panel, DC ); } /*****************************************************************************************/ bool WinEDA_PcbFrame::Delete_Module( MODULE* module, wxDC* DC, bool aAskBeforeDeleting ) /******************************************************************************************/ /** * Function Delete Module * Remove a footprint from m_Modules linked list and put it in undelete buffer * The net rastenes and pad list are recalcualed * @param module = footprint to delete * @param DC = currentDevice Context. if NULL: do not redraw new ratsnets and dirty rectange * @param aPromptBeforeDeleting : if true: ask for confirmation before deleting */ { wxString msg; /* Si l'empreinte est selectee , on ne peut pas l'effacer ! */ if( module == NULL ) return FALSE; /* Confirmation de l'effacement */ module->Display_Infos( this ); if( aAskBeforeDeleting ) { msg << _( "Delete Module" ) << wxT( " " ) << module->m_Reference->m_Text << wxT( " (" ) << _( "Value " ) << module->m_Value->m_Text << wxT( ") ?" ); if( !IsOK( this, msg ) ) { return FALSE; } } GetScreen()->SetModify(); /* Erase rastnest if needed * Dirty rectangle is not used here because usually using a XOR draw mode gives good results (very few artefacts) for ratsnest */ if( g_Show_Ratsnest ) DrawGeneralRatsnest( DC ); /* Sauvegarde en buffer des undelete */ SaveItemEfface( module, 1 ); GetBoard()->m_Status_Pcb = 0; build_liste_pads(); ReCompile_Ratsnest_After_Changes( DC ); // redraw the area where the module was if( DC ) DrawPanel->PostDirtyRect( module->GetBoundingBox() ); return TRUE; } /****************************************************************************/ void BOARD::Change_Side_Module( MODULE* Module, wxDC* DC ) /****************************************************************************/ /** * Function Change_Side_Module * Filp 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 */ { D_PAD* pt_pad; TEXTE_MODULE* pt_texte; EDGE_MODULE* pt_edgmod; EDA_BaseStruct* PtStruct; if( Module == NULL ) return; if( (Module->GetLayer() != CMP_N) && (Module->GetLayer() != COPPER_LAYER_N) ) return; m_PcbFrame->GetScreen()->SetModify(); if( !(Module->m_Flags & IS_MOVED) ) { m_Status_Pcb &= ~( LISTE_CHEVELU_OK | CONNEXION_OK); if( DC && m_PcbFrame ) { int tmp = Module->m_Flags; Module->m_Flags |= DO_NOT_DRAW; m_PcbFrame->DrawPanel->PostDirtyRect( Module->GetBoundingBox() ); Module->m_Flags = tmp; } /* Effacement chevelu general si necessaire */ if( DC && g_Show_Ratsnest ) m_PcbFrame->DrawGeneralRatsnest( DC ); /* Init des variables utilisees dans la routine Dessine_Drag_segment() */ g_Offset_Module.x = 0; g_Offset_Module.y = 0; } else // Module en deplacement { /* efface empreinte ( vue en contours) si elle a ete deja dessinee */ if( DC && m_PcbFrame ) { DrawModuleOutlines( m_PcbFrame->DrawPanel, DC, Module ); Dessine_Segments_Dragges( m_PcbFrame->DrawPanel, DC ); } } /* mise a jour du Flag de l'empreinte et des couches des contours et textes */ Module->SetLayer( ChangeSideNumLayer( Module->GetLayer() ) ); /* Inversion miroir de l'orientation */ Module->m_Orient = -Module->m_Orient; NORMALIZE_ANGLE_POS( Module->m_Orient ); /* Inversion miroir + layers des pastilles */ pt_pad = Module->m_Pads; for( ; pt_pad != NULL; pt_pad = pt_pad->Next() ) { pt_pad->m_Pos.y -= Module->m_Pos.y; pt_pad->m_Pos.y = -pt_pad->m_Pos.y; pt_pad->m_Pos.y += Module->m_Pos.y; pt_pad->m_Pos0.y = -pt_pad->m_Pos0.y; pt_pad->m_Offset.y = -pt_pad->m_Offset.y; pt_pad->m_DeltaSize.y = -pt_pad->m_DeltaSize.y; NEGATE_AND_NORMALIZE_ANGLE_POS( pt_pad->m_Orient ); /* change cote pour pastilles surfaciques */ pt_pad->m_Masque_Layer = ChangeSideMaskLayer( pt_pad->m_Masque_Layer ); } /* Inversion miroir de la Reference et mise en miroir : */ pt_texte = Module->m_Reference; pt_texte->m_Pos.y -= Module->m_Pos.y; pt_texte->m_Pos.y = -pt_texte->m_Pos.y; pt_texte->m_Pos.y += Module->m_Pos.y; pt_texte->m_Pos0.y = -pt_texte->m_Pos0.y; pt_texte->m_Mirror = false; NEGATE_AND_NORMALIZE_ANGLE_POS( pt_texte->m_Orient ); pt_texte->SetLayer( Module->GetLayer() ); pt_texte->SetLayer( ChangeSideNumLayer( pt_texte->GetLayer() ) ); if( Module->GetLayer() == COPPER_LAYER_N ) pt_texte->SetLayer( SILKSCREEN_N_CU ); if( Module->GetLayer() == CMP_N ) pt_texte->SetLayer( SILKSCREEN_N_CMP ); if( (Module->GetLayer() == SILKSCREEN_N_CU) || (Module->GetLayer() == ADHESIVE_N_CU) || (Module->GetLayer() == COPPER_LAYER_N) ) pt_texte->m_Mirror = true; /* Inversion miroir de la Valeur et mise en miroir : */ pt_texte = Module->m_Value; pt_texte->m_Pos.y -= Module->m_Pos.y; pt_texte->m_Pos.y = -pt_texte->m_Pos.y; pt_texte->m_Pos.y += Module->m_Pos.y; pt_texte->m_Pos0.y = -pt_texte->m_Pos0.y; pt_texte->m_Mirror = false; NEGATE_AND_NORMALIZE_ANGLE_POS( pt_texte->m_Orient ); pt_texte->SetLayer( Module->GetLayer() ); pt_texte->SetLayer( ChangeSideNumLayer( pt_texte->GetLayer() ) ); if( Module->GetLayer() == COPPER_LAYER_N ) pt_texte->SetLayer( SILKSCREEN_N_CU ); if( Module->GetLayer() == CMP_N ) pt_texte->SetLayer( SILKSCREEN_N_CMP ); if( (Module->GetLayer() == SILKSCREEN_N_CU) || (Module->GetLayer() == ADHESIVE_N_CU) || (Module->GetLayer() == COPPER_LAYER_N) ) pt_texte->m_Mirror = true; /* Inversion miroir des dessins de l'empreinte : */ PtStruct = Module->m_Drawings; for( ; PtStruct != NULL; PtStruct = PtStruct->Next() ) { switch( PtStruct->Type() ) { case TYPE_EDGE_MODULE: pt_edgmod = (EDGE_MODULE*) PtStruct; pt_edgmod->m_Start.y -= Module->m_Pos.y; pt_edgmod->m_Start.y = -pt_edgmod->m_Start.y; pt_edgmod->m_Start.y += Module->m_Pos.y; pt_edgmod->m_End.y -= Module->m_Pos.y; pt_edgmod->m_End.y = -pt_edgmod->m_End.y; pt_edgmod->m_End.y += Module->m_Pos.y; /* inversion des coords locales */ pt_edgmod->m_Start0.y = -pt_edgmod->m_Start0.y; pt_edgmod->m_End0.y = -pt_edgmod->m_End0.y; if( pt_edgmod->m_Shape == S_ARC ) { pt_edgmod->m_Angle = -pt_edgmod->m_Angle; } pt_edgmod->SetLayer( ChangeSideNumLayer( pt_edgmod->GetLayer() ) ); break; case TYPE_TEXTE_MODULE: /* Inversion miroir de la position et mise en miroir : */ pt_texte = (TEXTE_MODULE*) PtStruct; pt_texte->m_Pos.y -= Module->m_Pos.y; pt_texte->m_Pos.y = -pt_texte->m_Pos.y; pt_texte->m_Pos.y += Module->m_Pos.y; pt_texte->m_Pos0.y = - pt_texte->m_Pos0.y; pt_texte->m_Mirror = false; NEGATE_AND_NORMALIZE_ANGLE_POS( pt_texte->m_Orient ); pt_texte->SetLayer( Module->GetLayer() ); pt_texte->SetLayer( ChangeSideNumLayer( pt_texte->GetLayer() ) ); if( Module->GetLayer() == COPPER_LAYER_N ) pt_texte->SetLayer( SILKSCREEN_N_CU ); if( Module->GetLayer() == CMP_N ) pt_texte->SetLayer( SILKSCREEN_N_CMP ); if( Module->GetLayer() == SILKSCREEN_N_CU || Module->GetLayer() == ADHESIVE_N_CU || Module->GetLayer() == COPPER_LAYER_N ) { pt_texte->m_Mirror = true; } break; default: DisplayError( m_PcbFrame, wxT( "Unknown Draw Type" ) ); break; } } /* calcul du rectangle d'encadrement */ Module->Set_Rectangle_Encadrement(); if( m_PcbFrame ) Module->Display_Infos( m_PcbFrame ); if( !(Module->m_Flags & IS_MOVED) ) /* Inversion simple */ { if( DC && m_PcbFrame ) { Module->Draw( m_PcbFrame->DrawPanel, DC, GR_OR ); /* affichage chevelu general si necessaire */ m_PcbFrame->ReCompile_Ratsnest_After_Changes( DC ); } } else { if( DC && m_PcbFrame ) { DrawModuleOutlines( m_PcbFrame->DrawPanel, DC, Module ); Dessine_Segments_Dragges( m_PcbFrame->DrawPanel, DC ); } m_Status_Pcb &= ~CHEVELU_LOCAL_OK; } } /*********************************************/ static int ChangeSideMaskLayer( int masque ) /*********************************************/ /* Routine de recalcul du masque-layer lors des * echanges cote cu/cmp pour les couches CU/CMP specialisees * (cuivre, serigr., pate , soudure) */ { int newmasque; newmasque = masque & ~(CUIVRE_LAYER | CMP_LAYER | SILKSCREEN_LAYER_CU | SILKSCREEN_LAYER_CMP | ADHESIVE_LAYER_CU | ADHESIVE_LAYER_CMP | SOLDERMASK_LAYER_CU | SOLDERMASK_LAYER_CMP | SOLDERPASTE_LAYER_CU | SOLDERPASTE_LAYER_CMP | ADHESIVE_LAYER_CU | ADHESIVE_LAYER_CMP); if( masque & CUIVRE_LAYER ) newmasque |= CMP_LAYER; if( masque & CMP_LAYER ) newmasque |= CUIVRE_LAYER; if( masque & SILKSCREEN_LAYER_CU ) newmasque |= SILKSCREEN_LAYER_CMP; if( masque & SILKSCREEN_LAYER_CMP ) newmasque |= SILKSCREEN_LAYER_CU; if( masque & ADHESIVE_LAYER_CU ) newmasque |= ADHESIVE_LAYER_CMP; if( masque & ADHESIVE_LAYER_CMP ) newmasque |= ADHESIVE_LAYER_CU; if( masque & SOLDERMASK_LAYER_CU ) newmasque |= SOLDERMASK_LAYER_CMP; if( masque & SOLDERMASK_LAYER_CMP ) newmasque |= SOLDERMASK_LAYER_CU; if( masque & SOLDERPASTE_LAYER_CU ) newmasque |= SOLDERPASTE_LAYER_CMP; if( masque & SOLDERPASTE_LAYER_CMP ) newmasque |= SOLDERPASTE_LAYER_CU; if( masque & ADHESIVE_LAYER_CU ) newmasque |= ADHESIVE_LAYER_CMP; if( masque & ADHESIVE_LAYER_CMP ) newmasque |= ADHESIVE_LAYER_CU; return newmasque; } /*************************************/ int ChangeSideNumLayer( int oldlayer ) /*************************************/ /* Routine de recalcul du numero de couche lors des * echanges cote cu/cmp pour les couches CU/CMP specialisees * (cuivre, serigr., pate , soudure) */ { int newlayer; switch( oldlayer ) { case COPPER_LAYER_N: newlayer = CMP_N; break; case CMP_N: newlayer = COPPER_LAYER_N; break; case SILKSCREEN_N_CU: newlayer = SILKSCREEN_N_CMP; break; case SILKSCREEN_N_CMP: newlayer = SILKSCREEN_N_CU; break; case ADHESIVE_N_CU: newlayer = ADHESIVE_N_CMP; break; case ADHESIVE_N_CMP: newlayer = ADHESIVE_N_CU; break; case SOLDERMASK_N_CU: newlayer = SOLDERMASK_N_CMP; break; case SOLDERMASK_N_CMP: newlayer = SOLDERMASK_N_CU; break; case SOLDERPASTE_N_CU: newlayer = SOLDERPASTE_N_CMP; break; case SOLDERPASTE_N_CMP: newlayer = SOLDERPASTE_N_CU; break; default: newlayer = oldlayer; } return newlayer; } /*****************************************************************/ void WinEDA_BasePcbFrame::Place_Module( MODULE* module, wxDC* DC ) /*****************************************************************/ /* Place a l'endroit pointe par la souris le module deja existant selectionne * auparavant. * Entree: module = num du module a replacer * DC ( si NULL: pas d'affichage a l'�ran * Sortie : * mise a jour des nouvelles coord des differents elements du module * affichage a l'ecran du module */ { TRACK* pt_segm; DRAG_SEGM* pt_drag; wxPoint newpos; if( module == 0 ) return; GetScreen()->SetModify(); GetBoard()->m_Status_Pcb &= ~( LISTE_CHEVELU_OK | CONNEXION_OK); if( g_Show_Module_Ratsnest && (GetBoard()->m_Status_Pcb & LISTE_PAD_OK) && DC ) trace_ratsnest_module( DC ); newpos = GetScreen()->m_Curseur; module->SetPosition( newpos ); if( DC ) module->Draw( DrawPanel, DC, GR_OR ); /* Tracage des segments dragges et liberation memoire */ if( g_DragSegmentList ) { pt_drag = g_DragSegmentList; for( ; pt_drag != NULL; pt_drag = pt_drag->Pnext ) { pt_segm = pt_drag->m_Segm; pt_segm->SetState( EDIT, OFF ); if( DC ) pt_segm->Draw( DrawPanel, DC, GR_OR ); } EraseDragListe(); } /* affichage chevelu general si necessaire */ ReCompile_Ratsnest_After_Changes( DC ); module->Display_Infos( this ); DrawPanel->ManageCurseur = NULL; DrawPanel->ForceCloseManageCurseur = NULL; module->m_Flags = 0; g_Drag_Pistes_On = FALSE; } /***********************************************************************/ void WinEDA_BasePcbFrame::Rotate_Module( wxDC* DC, MODULE* module, int angle, bool incremental ) /***********************************************************************/ /* * Fait tourner l'empreinte de angle degres, dans le sens < 0. * Si incremental == TRUE, la rotation est faite a partir de la derniere orientation, * sinon le module est mis dans l'orientation absolue angle. * Si DC == NULL, le composant n'est pas redessine. * Sinon, il est efface, tourne et redessine */ { if( module == NULL ) return; GetScreen()->SetModify(); /* efface ancienne position */ if( !(module->m_Flags & IS_MOVED) ) /* Rotation simple */ { if( DC ) { int tmp = module->m_Flags; module->m_Flags |= DO_NOT_DRAW; DrawPanel->PostDirtyRect( module->GetBoundingBox() ); module->m_Flags = tmp; /* Reaffichage chevelu general si necessaire */ if( g_Show_Ratsnest ) DrawGeneralRatsnest( DC ); } } else { /* reaffiche module en mouvement */ if( DC ) { DrawModuleOutlines( DrawPanel, DC, module ); Dessine_Segments_Dragges( DrawPanel, DC ); } } GetBoard()->m_Status_Pcb &= ~(LISTE_CHEVELU_OK | CONNEXION_OK); if( incremental ) module->SetOrientation( module->m_Orient + angle ); else module->SetOrientation( angle ); module->Display_Infos( this ); if( DC ) { if( !(module->m_Flags & IS_MOVED) ) /* Rotation simple */ { module->Draw( DrawPanel, DC, GR_OR ); /* Reaffichage chevelu general si necessaire */ ReCompile_Ratsnest_After_Changes( DC ); } else { /* reaffiche module en mouvement */ DrawModuleOutlines( DrawPanel, DC, module ); Dessine_Segments_Dragges( DrawPanel, DC ); } } } /*************************************************/ /* Redessine en mode XOR la silouhette du module */ /*************************************************/ void DrawModuleOutlines( WinEDA_DrawPanel* panel, wxDC* DC, MODULE* module ) { 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 = SKETCH; /* Trace en SKETCH en deplacement */ pt_pad = module->m_Pads; for( ; pt_pad != NULL; pt_pad = pt_pad->Next() ) { pt_pad->Draw( panel, DC, GR_XOR, g_Offset_Module ); } DisplayOpt.DisplayPadFill = pad_fill_tmp; } if( g_Show_Module_Ratsnest && panel ) { WinEDA_BasePcbFrame* frame = (WinEDA_BasePcbFrame*) panel->m_Parent; frame->build_ratsnest_module( DC, module ); frame->trace_ratsnest_module( DC ); } }