1 changed files with 549 additions and 0 deletions
@ -0,0 +1,549 @@ |
|||
/****************************************************/ |
|||
/* Edition des pistes */ |
|||
/* Routines de duplication et deplacement de pistes */ |
|||
/****************************************************/ |
|||
|
|||
#include "fctsys.h" |
|||
#include "gr_basic.h" |
|||
|
|||
#include "common.h" |
|||
#include "pcbnew.h" |
|||
#include "autorout.h" |
|||
|
|||
#include "protos.h" |
|||
|
|||
|
|||
#define COPY_ROUTE 1 |
|||
#define MOVE_ROUTE 2 |
|||
|
|||
extern char marq_bitmap[]; // dans TRACEPCB : bitmap du marqueur "DRC" |
|||
|
|||
/* Routines externes */ |
|||
|
|||
#if 0 |
|||
/* Routines Locales */ |
|||
static void Exit_DuplicTrack( COMMAND* Cmd ); |
|||
static void Start_CopieMove_Route( COMMAND* Cmd ); |
|||
static void Duplic_Track( COMMAND* Cmd ); |
|||
static void Place_Dupl_Route( COMMAND* Cmd ); |
|||
static void Show_Move_Piste( wxDC* DC, int flag ); |
|||
|
|||
/* variables locales */ |
|||
static int startX, startY; |
|||
static int PosInitX, PosInitY; |
|||
static TRACK* NewTrack; /* Nouvelle piste creee ou piste deplacee */ |
|||
static int NbPtNewTrack; |
|||
static int FlagState; /* memoire de la commande (COPY_ROUTE ou MOVE_ROUTE) */ |
|||
/* variables externes */ |
|||
|
|||
|
|||
/**************************************************************/ |
|||
static void Exit_DuplicTrack( WinEDA_DrawFrame* frame, wxDC* DC ) |
|||
/***************************************************************/ |
|||
|
|||
/* routine d'annulation de la Commande Begin_Route si une piste est en cours |
|||
* de tracage, ou de sortie de l'application EDITRACK. |
|||
* Appel par la touche ESC |
|||
*/ |
|||
{ |
|||
TRACK* NextS; |
|||
int ii; |
|||
wxDC* DC = Cmd->DC; |
|||
|
|||
frame->DrawPanel->ManageCurseur = NULL; |
|||
|
|||
if( NewTrack ) |
|||
{ |
|||
/* Effacement du trace en cours */ |
|||
DisplayOpt.DisplayPcbTrackFill = FALSE; |
|||
Trace_Une_Piste( DC, NewTrack, NbPtNewTrack, GR_XOR ); |
|||
DisplayOpt.DisplayPcbTrackFill = Track_fill_copy; |
|||
|
|||
if( FlagState == COPY_ROUTE ) |
|||
{ |
|||
for( ii = 0; ii < NbPtNewTrack; ii++, NewTrack = NextS ) |
|||
{ |
|||
if( NewTrack == NULL ) |
|||
break; |
|||
NextS = (TRACK*) NewTrack->Pnext; |
|||
delete NewTrack; |
|||
} |
|||
} |
|||
else /* Move : remise en ancienne position */ |
|||
{ |
|||
TRACK* Track = NewTrack; |
|||
PosInitX -= Track->m_Start.x; |
|||
PosInitY -= Track->m_Start.y; |
|||
for( ii = 0; ii < NbPtNewTrack; ii++, Track = (TRACK*) Track->Pnext ) |
|||
{ |
|||
if( Track == NULL ) |
|||
break; |
|||
Track->m_Start.x += PosInitX; |
|||
Track->m_Start.y += PosInitY; |
|||
Track->m_End.x += PosInitX; |
|||
Track->m_End.y += PosInitY; |
|||
} |
|||
|
|||
Trace_Une_Piste( DC, NewTrack, NbPtNewTrack, GR_OR ); |
|||
} |
|||
|
|||
if( Etat_Surbrillance ) |
|||
Hight_Light( DC ); |
|||
EraseMsgBox(); |
|||
NewTrack = NULL; |
|||
} |
|||
else |
|||
{ |
|||
EraseMsgBox(); |
|||
} |
|||
} |
|||
|
|||
|
|||
/******************************************/ |
|||
static void Place_Dupl_Route( COMMAND* Cmd ) |
|||
/******************************************/ |
|||
|
|||
/* |
|||
* Routine de placement d'une piste (succession de segments) |
|||
*/ |
|||
{ |
|||
D_PAD* pt_pad; |
|||
TRACK* pt_track, * Track, * pt_classe, * NextS; |
|||
int masquelayer; |
|||
EDA_BaseStruct* LockPoint; |
|||
int ii, old_net_code, new_net_code, DRC_error = 0; |
|||
wxDC* DC = Cmd->DC; |
|||
|
|||
ActiveDrawPanel->ManageCurseur = NULL; |
|||
|
|||
if( NewTrack == NULL ) |
|||
return; |
|||
|
|||
old_net_code = NewTrack->net_code; |
|||
|
|||
/* Placement du flag BUSY de la piste originelle, qui ne doit |
|||
* pas etre vue dans les recherches de raccordement suivantes */ |
|||
ii = NbPtNewTrack; pt_track = NewTrack; |
|||
for( ; ii > 0; ii--, pt_track = (TRACK*) pt_track->Pnext ) |
|||
{ |
|||
pt_track->SetState( BUSY, ON ); |
|||
} |
|||
|
|||
/* Detection du nouveau net_code */ |
|||
ii = NbPtNewTrack; pt_track = NewTrack; |
|||
for( ; ii > 0; ii--, pt_track = (TRACK*) pt_track->Pnext ) |
|||
{ |
|||
pt_track->net_code = 0; |
|||
} |
|||
|
|||
new_net_code = 0; |
|||
ii = 0; pt_track = NewTrack; |
|||
for( ; ii < NbPtNewTrack; ii++, pt_track = (TRACK*) pt_track->Pnext ) |
|||
{ |
|||
/* Localisation de la pastille ou segment en debut de segment: */ |
|||
masquelayer = tab_layer[pt_track->Layer]; |
|||
LockPoint = LocateLockPoint( pt_track->m_Start.x, pt_track->m_Start.y, masquelayer ); |
|||
if( LockPoint ) |
|||
{ |
|||
if( LockPoint->m_StructType == TYPEPAD ) |
|||
{ |
|||
pt_pad = (D_PAD*) LockPoint; |
|||
new_net_code = pt_pad->net_code; |
|||
if( new_net_code > 0 ) |
|||
break; |
|||
} |
|||
else /* debut de piste sur un segment de piste */ |
|||
{ |
|||
Track = (TRACK*) LockPoint; |
|||
new_net_code = Track->net_code; |
|||
if( new_net_code > 0 ) |
|||
break; |
|||
} |
|||
} |
|||
LockPoint = LocateLockPoint( pt_track->m_End.x, pt_track->m_End.y, masquelayer ); |
|||
if( LockPoint ) |
|||
{ |
|||
if( LockPoint->m_StructType == TYPEPAD ) |
|||
{ |
|||
pt_pad = (D_PAD*) LockPoint; |
|||
new_net_code = pt_pad->net_code; |
|||
if( new_net_code > 0 ) |
|||
break; |
|||
} |
|||
else /* debut de piste sur un segment de piste */ |
|||
{ |
|||
Track = (TRACK*) LockPoint; |
|||
new_net_code = Track->net_code; |
|||
if( new_net_code > 0 ) |
|||
break; |
|||
} |
|||
} |
|||
} |
|||
|
|||
/* Mise a jour du nouveau net code de la piste */ |
|||
ii = 0; pt_track = NewTrack; |
|||
for( ; ii < NbPtNewTrack; ii++, pt_track = (TRACK*) pt_track->Pnext ) |
|||
{ |
|||
pt_track->net_code = new_net_code; |
|||
} |
|||
|
|||
/* Controle DRC de la nouvelle piste */ |
|||
ii = 0; pt_track = NewTrack; |
|||
for( ; ii < NbPtNewTrack; ii++, pt_track = pt_track->Next() ) |
|||
{ |
|||
if( Drc_On == RUN ) |
|||
if( drc( DC, pt_track, pt_pcb->Track, 1 ) == BAD_DRC ) |
|||
{ |
|||
if( confirmation( " Erreur DRC, Place piste:" ) == YES ) |
|||
continue; |
|||
else |
|||
{ |
|||
DRC_error = 1; break; |
|||
} |
|||
} |
|||
} |
|||
|
|||
if( DRC_error == 0 ) |
|||
{ |
|||
if( FlagState == MOVE_ROUTE ) |
|||
{ |
|||
/* copie nouvelle piste */ |
|||
pt_track = NewTrack; |
|||
NewTrack = pt_track->Copy( NbPtNewTrack ); |
|||
/* effacement ancienne ( chainage et liens mauvais */ |
|||
ii = NbPtNewTrack; |
|||
for( ; ii > 0; ii--, pt_track = NextS ) |
|||
{ |
|||
NextS = (TRACK*) pt_track->Pnext; |
|||
DeleteStructure( pt_track ); |
|||
} |
|||
|
|||
test_1_net_connexion( DC, old_net_code ); |
|||
} |
|||
|
|||
pt_classe = NewTrack->GetBestInsertPoint(); |
|||
NewTrack->Insert( pt_classe ); |
|||
|
|||
Trace_Une_Piste( DC, NewTrack, NbPtNewTrack, GR_OR ); |
|||
|
|||
/* Mise a jour des connexions sur pads et sur pistes */ |
|||
ii = 0; pt_track = NewTrack; |
|||
for( ; ii < NbPtNewTrack; ii++, pt_track = NextS ) |
|||
{ |
|||
NextS = (TRACK*) pt_track->Pnext; |
|||
pt_track->SetState( BEGIN_ONPAD | END_ONPAD, OFF ); |
|||
masquelayer = tab_layer[pt_track->Layer]; |
|||
|
|||
/* Localisation de la pastille ou segment sur debut segment: */ |
|||
LockPoint = LocateLockPoint( pt_track->m_Start.x, pt_track->m_Start.y, masquelayer ); |
|||
if( LockPoint ) |
|||
{ |
|||
pt_track->start = LockPoint; |
|||
if( LockPoint->m_StructType == TYPEPAD ) |
|||
{ /* fin de piste sur un pad */ |
|||
pt_pad = (D_PAD*) LockPoint; |
|||
pt_track->SetState( BEGIN_ONPAD, ON ); |
|||
} |
|||
else /* debut de piste sur un segment de piste */ |
|||
{ |
|||
Track = (TRACK*) LockPoint; |
|||
CreateLockPoint( &pt_track->m_Start.x, &pt_track->m_Start.y, Track, pt_track ); |
|||
} |
|||
} |
|||
|
|||
/* Localisation de la pastille ou segment sur fin de segment: */ |
|||
LockPoint = LocateLockPoint( pt_track->m_End.x, pt_track->m_End.y, masquelayer ); |
|||
if( LockPoint ) |
|||
{ |
|||
pt_track->end = LockPoint; |
|||
if( LockPoint->m_StructType == TYPEPAD ) |
|||
{ /* fin de piste sur un pad */ |
|||
pt_pad = (D_PAD*) LockPoint; |
|||
pt_track->SetState( END_ONPAD, ON ); |
|||
} |
|||
else /* debut de piste sur un segment de piste */ |
|||
{ |
|||
Track = (TRACK*) LockPoint; |
|||
CreateLockPoint( &pt_track->m_Start.x, &pt_track->m_Start.y, Track, pt_track ); |
|||
} |
|||
} |
|||
} |
|||
|
|||
/* Suppression du flag BUSY */ |
|||
ii = NbPtNewTrack; pt_track = NewTrack; |
|||
for( ; ii > 0; ii--, pt_track = (TRACK*) pt_track->Pnext ) |
|||
{ |
|||
pt_track->SetState( BUSY, OFF ); |
|||
} |
|||
|
|||
test_1_net_connexion( DC, new_net_code ); |
|||
ActiveScreen->SetModify(); |
|||
} |
|||
else /* Erreur DRC: Annulation commande */ |
|||
{ |
|||
DisplayOpt.DisplayPcbTrackFill = FALSE; |
|||
Trace_Une_Piste( DC, NewTrack, NbPtNewTrack, GR_XOR ); |
|||
DisplayOpt.DisplayPcbTrackFill = Track_fill_copy; |
|||
|
|||
if( FlagState == MOVE_ROUTE ) |
|||
{ /* Remise en position de la piste deplacee */ |
|||
Track = NewTrack; |
|||
PosInitX -= Track->m_Start.x; PosInitY -= Track->m_Start.y; |
|||
for( ii = 0; ii < NbPtNewTrack; ii++, Track = (TRACK*) Track->Pnext ) |
|||
{ |
|||
if( Track == NULL ) |
|||
break; |
|||
Track->m_Start.x += PosInitX; Track->m_Start.y += PosInitY; |
|||
Track->m_End.x += PosInitX; Track->m_End.y += PosInitY; |
|||
Track->SetState( BUSY, OFF ); |
|||
} |
|||
|
|||
Trace_Une_Piste( DC, NewTrack, NbPtNewTrack, GR_OR ); |
|||
} |
|||
|
|||
if( FlagState == COPY_ROUTE ) |
|||
{ /* Suppression copie */ |
|||
for( ii = 0; ii < NbPtNewTrack; NewTrack = NextS ) |
|||
{ |
|||
if( NewTrack == NULL ) |
|||
break; |
|||
NextS = (TRACK*) NewTrack->Pnext; |
|||
delete NewTrack; |
|||
} |
|||
} |
|||
} |
|||
NewTrack = NULL; |
|||
Affiche_Infos_Status_Pcb( Cmd ); |
|||
if( Etat_Surbrillance ) |
|||
Hight_Light( DC ); |
|||
} |
|||
|
|||
|
|||
/***********************************************/ |
|||
static void Show_Move_Piste( wxDC* DC, int flag ) |
|||
/***********************************************/ |
|||
/* redessin du contour de la piste lors des deplacements de la souris */ |
|||
{ |
|||
int ii, dx, dy; |
|||
TRACK* ptsegm; |
|||
|
|||
if( NewTrack == NULL ) |
|||
return; /* Pas de piste en cours (Erreur ) */ |
|||
|
|||
/* efface ancienne position si elle a ete deja dessinee */ |
|||
if( (flag == CURSEUR_MOVED ) && (FlagState == COPY_ROUTE ) ) |
|||
{ |
|||
DisplayOpt.DisplayPcbTrackFill = FALSE; |
|||
Trace_Une_Piste( DC, NewTrack, NbPtNewTrack, GR_XOR ); |
|||
} |
|||
|
|||
if( FlagState == MOVE_ROUTE ) |
|||
{ |
|||
if( flag == CURSEUR_MOVED ) |
|||
DisplayOpt.DisplayPcbTrackFill = FALSE; |
|||
Trace_Une_Piste( DC, NewTrack, NbPtNewTrack, GR_XOR ); |
|||
} |
|||
|
|||
/* mise a jour des coordonnees des segments de la piste */ |
|||
dx = ActiveScreen->Curseur_X - startX; |
|||
dy = ActiveScreen->Curseur_Y - startY; |
|||
startX = ActiveScreen->Curseur_X; |
|||
startY = ActiveScreen->Curseur_Y; |
|||
ii = NbPtNewTrack, ptsegm = NewTrack; |
|||
for( ; ii > 0; ii--, ptsegm = (TRACK*) ptsegm->Pnext ) |
|||
{ |
|||
ptsegm->m_Start.x += dx; ptsegm->m_Start.y += dy; |
|||
ptsegm->m_End.x += dx; ptsegm->m_End.y += dy; |
|||
} |
|||
|
|||
/* dessin de la nouvelle piste */ |
|||
DisplayOpt.DisplayPcbTrackFill = FALSE; |
|||
Trace_Une_Piste( DC, NewTrack, NbPtNewTrack, GR_XOR ); |
|||
DisplayOpt.DisplayPcbTrackFill = Track_fill_copy; |
|||
} |
|||
|
|||
|
|||
/************************************************/ |
|||
/* void Start_CopieMove_Route(COMMAND * Cmd) */ |
|||
/************************************************/ |
|||
|
|||
/* Routine permettant la recopie d'une piste deja tracee |
|||
*/ |
|||
static void Start_CopieMove_Route( COMMAND* Cmd ) |
|||
{ |
|||
int ii; |
|||
TRACK* pt_segm, * pt_track; |
|||
int masquelayer = tab_layer[ActiveScreen->Active_Layer]; |
|||
wxDC* DC = Cmd->DC; |
|||
|
|||
if( NewTrack ) |
|||
return; |
|||
|
|||
FlagState = (int) Cmd->Menu->param_inf; |
|||
|
|||
/* Recherche de la piste sur la couche active (non zone) */ |
|||
for( pt_segm = pt_pcb->Track; pt_segm != NULL; pt_segm = (TRACK*) pt_segm->Pnext ) |
|||
{ |
|||
pt_segm = Locate_Pistes( pt_segm, masquelayer, CURSEUR_OFF_GRILLE ); |
|||
if( pt_segm == NULL ) |
|||
break; |
|||
break; |
|||
} |
|||
|
|||
if( pt_segm != NULL ) |
|||
{ |
|||
if( FlagState == COPY_ROUTE ) |
|||
pt_track = Marque_Une_Piste( DC, pt_segm, &NbPtNewTrack, 0 ); |
|||
else |
|||
pt_track = Marque_Une_Piste( DC, pt_segm, &NbPtNewTrack, GR_XOR ); |
|||
|
|||
if( NbPtNewTrack ) /* Il y a NbPtNewTrack segments de piste a traiter */ |
|||
{ |
|||
/* effacement du flag BUSY de la piste originelle */ |
|||
ii = NbPtNewTrack; pt_segm = pt_track; |
|||
for( ; ii > 0; ii--, pt_segm = (TRACK*) pt_segm->Pnext ) |
|||
{ |
|||
pt_segm->SetState( BUSY, OFF ); |
|||
} |
|||
|
|||
if( FlagState == COPY_ROUTE ) |
|||
NewTrack = pt_track->Copy( NbPtNewTrack ); |
|||
else |
|||
NewTrack = pt_track; |
|||
|
|||
Affiche_Infos_Piste( Cmd, pt_track ); |
|||
|
|||
startX = ActiveScreen->Curseur_X; |
|||
startY = ActiveScreen->Curseur_Y; |
|||
Place_Dupl_Route_Item.State = WAIT; |
|||
ActiveDrawPanel->ManageCurseur = Show_Move_Piste; |
|||
DisplayOpt.DisplayPcbTrackFill = FALSE; |
|||
Trace_Une_Piste( DC, NewTrack, NbPtNewTrack, GR_XOR ); |
|||
DisplayOpt.DisplayPcbTrackFill = Track_fill_copy; |
|||
PosInitX = NewTrack->m_Start.x; PosInitY = NewTrack->m_Start.y; |
|||
} |
|||
} |
|||
} |
|||
|
|||
|
|||
#endif |
|||
|
|||
/************************************************************************/ |
|||
EDA_BaseStruct* LocateLockPoint( BOARD* Pcb, wxPoint pos, int LayerMask ) |
|||
/************************************************************************/ |
|||
|
|||
/* Routine trouvant le point " d'accrochage " d'une extremite de piste. |
|||
* Ce point peut etre un PAD ou un autre segment de piste |
|||
* Retourne: |
|||
* - pointeur sur ce PAD ou: |
|||
* - pointeur sur le segment ou: |
|||
* - NULL |
|||
* Parametres d'appel: |
|||
* coord pX, pY du point tst |
|||
* masque des couches a tester |
|||
*/ |
|||
{ |
|||
D_PAD* pt_pad; |
|||
TRACK* ptsegm; |
|||
MODULE* Module; |
|||
|
|||
/* detection du point type PAD */ |
|||
pt_pad = NULL; |
|||
Module = Pcb->m_Modules; |
|||
for( ; Module != NULL; Module = (MODULE*) Module->Pnext ) |
|||
{ |
|||
pt_pad = Locate_Pads( Module, pos.x, pos.y, LayerMask ); |
|||
if( pt_pad ) |
|||
return pt_pad; |
|||
} |
|||
|
|||
/* ici aucun pad n'a ete localise: detection d'un segment de piste */ |
|||
|
|||
ptsegm = Fast_Locate_Piste( Pcb->m_Track, NULL, pos.x, pos.y, LayerMask ); |
|||
if( ptsegm == NULL ) |
|||
ptsegm = Locate_Pistes( Pcb->m_Track, pos.x, pos.y, LayerMask ); |
|||
return ptsegm; |
|||
} |
|||
|
|||
|
|||
/******************************************************************************/ |
|||
TRACK* CreateLockPoint( int* pX, int* pY, TRACK* ptsegm, TRACK* refsegm ) |
|||
/******************************************************************************/ |
|||
|
|||
/* Routine de creation d'un point intermediaire sur un segment |
|||
* le segment ptsegm est casse en 2 segments se raccordant au point pX, pY |
|||
* retourne: |
|||
* NULL si pas de nouveau point ( c.a.d si pX, pY correspondait deja |
|||
* a une extremite ou: |
|||
* pointeur sur le segment cree |
|||
* si refsegm != NULL refsegm est pointeur sur le segment incident, |
|||
* et le point cree est l'intersection des 2 axes des segments ptsegm et |
|||
* refsegm |
|||
* retourne la valeur exacte de pX et pY |
|||
* Si ptsegm pointe sur une via: |
|||
* retourne la valeur exacte de pX et pY et ptsegm, |
|||
* mais ne cree pas de point supplementaire |
|||
* |
|||
*/ |
|||
{ |
|||
int cX, cY; |
|||
int dx, dy; /* Coord de l'extremite du segm ptsegm / origine */ |
|||
int ox, oy, fx, fy; /* coord de refsegm / origine de prsegm */ |
|||
TRACK* NewTrack; |
|||
|
|||
if( (ptsegm->m_Start.x == *pX) && (ptsegm->m_Start.y == *pY) ) |
|||
return NULL; |
|||
if( (ptsegm->m_End.x == *pX) && (ptsegm->m_End.y == *pY) ) |
|||
return NULL; |
|||
|
|||
/* le point n'est pas sur une extremite de piste */ |
|||
if( ptsegm->m_StructType == TYPEVIA ) |
|||
{ |
|||
*pX = ptsegm->m_Start.x; *pY = ptsegm->m_Start.y; |
|||
return ptsegm; |
|||
} |
|||
|
|||
/* calcul des coord vraies du point intermediaire dans le repere d'origine |
|||
* = origine de ptsegm */ |
|||
cX = *pX - ptsegm->m_Start.x; |
|||
cY = *pY - ptsegm->m_Start.y; |
|||
dx = ptsegm->m_End.x - ptsegm->m_Start.x; |
|||
dy = ptsegm->m_End.y - ptsegm->m_Start.y; |
|||
|
|||
// ***** A COMPLETER : non utilise |
|||
if( refsegm ) |
|||
{ |
|||
ox = refsegm->m_Start.x - ptsegm->m_Start.x; |
|||
oy = refsegm->m_Start.y - ptsegm->m_Start.y; |
|||
fx = refsegm->m_End.x - ptsegm->m_Start.x; |
|||
fy = refsegm->m_End.y - ptsegm->m_Start.y; |
|||
} |
|||
|
|||
/* pour que le point soit sur le segment ptsegm: cY/cX = dy/dx */ |
|||
if( dx == 0 ) |
|||
cX = 0; /* segm horizontal */ |
|||
else |
|||
cY = (cX * dy) / dx; |
|||
|
|||
/* creation du point intermediaire ( c'est a dire creation d'un nouveau |
|||
* segment, debutant au point intermediaire */ |
|||
|
|||
cX += ptsegm->m_Start.x; cY += ptsegm->m_Start.y; |
|||
NewTrack = ptsegm->Copy(); |
|||
|
|||
NewTrack->Insert( NULL, ptsegm ); |
|||
/* correction du pointeur de fin du nouveau segment */ |
|||
NewTrack->end = ptsegm->end; |
|||
|
|||
/* le segment primitif finit au nouveau point : */ |
|||
ptsegm->m_End.x = cX; ptsegm->m_End.y = cY; |
|||
ptsegm->SetState( END_ONPAD, OFF ); |
|||
|
|||
/* le nouveau segment debute au nouveau point : */ |
|||
ptsegm = NewTrack;; |
|||
ptsegm->m_Start.x = cX; ptsegm->m_Start.y = cY; |
|||
ptsegm->SetState( BEGIN_ONPAD, OFF ); |
|||
*pX = cX; *pY = cY; |
|||
|
|||
return ptsegm; |
|||
} |
Write
Preview
Loading…
Cancel
Save
Reference in new issue