547 lines
17 KiB
C++
547 lines
17 KiB
C++
/*********************************************************/
|
||
/* Modules de creations de Traits, Wires, Bus, Junctions */
|
||
/*********************************************************/
|
||
|
||
#include "fctsys.h"
|
||
#include "gr_basic.h"
|
||
#include "common.h"
|
||
#include "confirm.h"
|
||
|
||
#include "program.h"
|
||
#include "libcmp.h"
|
||
#include "general.h"
|
||
#include "protos.h"
|
||
|
||
/* Routines Locales */
|
||
static void Polyline_in_Ghost( WinEDA_DrawPanel* panel, wxDC* DC, bool erase );
|
||
static void Segment_in_Ghost( WinEDA_DrawPanel* panel, wxDC* DC, bool erase );
|
||
static void ExitTrace( WinEDA_DrawFrame* frame, wxDC* DC );
|
||
static bool IsTerminalPoint( SCH_SCREEN* screen, const wxPoint& pos, int layer );
|
||
|
||
|
||
/*************************************************************/
|
||
void WinEDA_SchematicFrame::BeginSegment( wxDC* DC, int type )
|
||
/*************************************************************/
|
||
|
||
/* Routine de Trace de segments ( WIRES, BUS ) pour lesquels chaque segment
|
||
* est une structure.
|
||
*/
|
||
// NOT USED!!!@!@!
|
||
{
|
||
DrawSegmentStruct* oldsegment, * newsegment;
|
||
wxPoint pos = GetScreen()->m_Curseur;
|
||
|
||
if( GetScreen()->GetCurItem()
|
||
&& (GetScreen()->GetCurItem()->m_Flags == 0) )
|
||
GetScreen()->SetCurItem( NULL );
|
||
|
||
if( GetScreen()->GetCurItem() )
|
||
{
|
||
switch( GetScreen()->GetCurItem()->Type() )
|
||
{
|
||
case DRAW_SEGMENT_STRUCT_TYPE:
|
||
case DRAW_POLYLINE_STRUCT_TYPE:
|
||
break;
|
||
|
||
default:
|
||
return;
|
||
}
|
||
}
|
||
|
||
oldsegment = newsegment =
|
||
(DrawSegmentStruct*) GetScreen()->GetCurItem();
|
||
|
||
if( !newsegment ) /* 1er point : creation de la 1ere structure */
|
||
{
|
||
switch( type )
|
||
{
|
||
default:
|
||
newsegment = new DrawSegmentStruct( pos, LAYER_NOTES );
|
||
break;
|
||
|
||
case LAYER_WIRE:
|
||
newsegment = new DrawSegmentStruct( pos, LAYER_WIRE );
|
||
if( LocatePinEnd( GetScreen()->EEDrawList, pos ) )
|
||
newsegment->m_StartIsDangling = FALSE;
|
||
break;
|
||
|
||
case LAYER_BUS:
|
||
newsegment = new DrawSegmentStruct( pos, LAYER_BUS );
|
||
break;
|
||
}
|
||
|
||
newsegment->m_Flags = IS_NEW;
|
||
GetScreen()->SetCurItem( newsegment );
|
||
GetScreen()->ManageCurseur = Segment_in_Ghost;
|
||
GetScreen()->ForceCloseManageCurseur = ExitTrace;
|
||
g_ItemToRepeat = NULL;
|
||
}
|
||
else /* Trace en cours: Placement d'un point supplementaire */
|
||
{
|
||
if( (oldsegment->m_Start.x == oldsegment->m_End.x)
|
||
&& (oldsegment->m_Start.y == oldsegment->m_End.y) ) /* Structure inutile */
|
||
return;
|
||
GetScreen()->ManageCurseur( DrawPanel, DC, FALSE );
|
||
oldsegment->m_EndIsDangling = FALSE;
|
||
|
||
/* Creation du segment suivant ou fin de trac<61> si point sur pin, jonction ...*/
|
||
if( IsTerminalPoint( GetScreen(), oldsegment->m_End, oldsegment->m_Layer ) )
|
||
{
|
||
EndSegment( DC ); return;
|
||
}
|
||
|
||
/* Placement en liste generale */
|
||
oldsegment->Pnext = GetScreen()->EEDrawList;
|
||
g_ItemToRepeat = GetScreen()->EEDrawList = oldsegment;
|
||
GetScreen()->CursorOff( DrawPanel, DC ); // Erase schematic cursor
|
||
RedrawOneStruct( DrawPanel, DC, oldsegment, GR_DEFAULT_DRAWMODE );
|
||
GetScreen()->CursorOn( DrawPanel, DC ); // Display schematic cursor
|
||
|
||
/* Creation du segment suivant */
|
||
newsegment = oldsegment->GenCopy();
|
||
newsegment->m_Start = oldsegment->m_End;
|
||
newsegment->m_End = pos;
|
||
oldsegment->m_Flags = 0;
|
||
newsegment->m_Flags = IS_NEW;
|
||
GetScreen()->SetCurItem( newsegment );
|
||
GetScreen()->ManageCurseur( DrawPanel, DC, FALSE );
|
||
newsegment->m_StartIsDangling = FALSE;
|
||
newsegment->m_EndIsDangling = TRUE;
|
||
}
|
||
}
|
||
|
||
|
||
/*************************************************************/
|
||
/* Routine de fin de trace d'une struct segment (Wire, Bus */
|
||
/*************************************************************/
|
||
void WinEDA_SchematicFrame::EndSegment( wxDC* DC )
|
||
{
|
||
DrawSegmentStruct* segment = (DrawSegmentStruct*) GetScreen()->GetCurItem();
|
||
|
||
if( GetScreen()->ManageCurseur == NULL )
|
||
return;
|
||
if( segment == NULL )
|
||
return;
|
||
if( (segment->m_Flags & IS_NEW) == 0 )
|
||
return;
|
||
|
||
if( (segment->m_Start.x == segment->m_End.x)
|
||
&& (segment->m_Start.y == segment->m_End.y) )/* Structure inutile */
|
||
{
|
||
EraseStruct( segment, (SCH_SCREEN*) GetScreen() );
|
||
segment = NULL;
|
||
}
|
||
else
|
||
{
|
||
/* Placement en liste generale */
|
||
GetScreen()->ManageCurseur( DrawPanel, DC, FALSE );
|
||
segment->Pnext = GetScreen()->EEDrawList;
|
||
g_ItemToRepeat = GetScreen()->EEDrawList = segment;
|
||
segment->m_Flags = 0;
|
||
}
|
||
|
||
/* Fin de trace */
|
||
GetScreen()->ManageCurseur = NULL;
|
||
GetScreen()->ForceCloseManageCurseur = NULL;
|
||
GetScreen()->SetCurItem( NULL );
|
||
|
||
TestDanglingEnds( GetScreen()->EEDrawList, DC );
|
||
SetFlagModify( GetScreen() );
|
||
if( segment )
|
||
{
|
||
GetScreen()->CursorOff( DrawPanel, DC ); // Erase schematic cursor
|
||
RedrawOneStruct( DrawPanel, DC, segment, GR_DEFAULT_DRAWMODE );
|
||
GetScreen()->CursorOn( DrawPanel, DC ); // Display schematic cursor
|
||
}
|
||
}
|
||
|
||
|
||
/****************************************************************************/
|
||
static void Segment_in_Ghost( WinEDA_DrawPanel* panel, wxDC* DC, bool erase )
|
||
/****************************************************************************/
|
||
|
||
/* Dessin du Segment Fantome lors des deplacements du curseur
|
||
*/
|
||
{
|
||
DrawSegmentStruct* segment =
|
||
(DrawSegmentStruct*) panel->m_Parent->GetScreen()->GetCurItem();
|
||
wxPoint endpos;
|
||
int color;
|
||
|
||
if( segment == NULL )
|
||
return;
|
||
|
||
color = ReturnLayerColor( segment->m_Layer ) ^ HIGHT_LIGHT_FLAG;
|
||
|
||
endpos = panel->m_Parent->GetScreen()->m_Curseur;
|
||
|
||
if( g_HVLines ) /* Coerce the line to vertical or horizontal one: */
|
||
{
|
||
if( ABS( endpos.x - segment->m_Start.x ) < ABS( endpos.y - segment->m_Start.y ) )
|
||
endpos.x = segment->m_Start.x;
|
||
else
|
||
endpos.y = segment->m_Start.y;
|
||
}
|
||
|
||
if( erase ) // Redraw if segment lengtht != 0
|
||
{
|
||
if( (segment->m_Start.x != segment->m_End.x)
|
||
|| (segment->m_Start.y != segment->m_End.y) )
|
||
RedrawOneStruct( panel, DC, segment, XOR_MODE, color );
|
||
}
|
||
segment->m_End = endpos;
|
||
|
||
// Redraw if segment lengtht != 0
|
||
|
||
if( (segment->m_Start.x != segment->m_End.x)
|
||
|| (segment->m_Start.y != segment->m_End.y) )
|
||
RedrawOneStruct( panel, DC, segment, XOR_MODE, color );
|
||
}
|
||
|
||
|
||
/*****************************************************************************/
|
||
static void Polyline_in_Ghost( WinEDA_DrawPanel* panel, wxDC* DC, bool erase )
|
||
/*****************************************************************************/
|
||
|
||
/* Dessin du du Polyline Fantome lors des deplacements du curseur
|
||
*/
|
||
{
|
||
DrawPolylineStruct* NewPoly =
|
||
(DrawPolylineStruct*) panel->m_Parent->GetScreen()->GetCurItem();
|
||
int color;
|
||
wxPoint endpos;
|
||
|
||
endpos = panel->m_Parent->GetScreen()->m_Curseur;
|
||
color = ReturnLayerColor( NewPoly->m_Layer );
|
||
|
||
GRSetDrawMode( DC, XOR_MODE );
|
||
|
||
if( g_HVLines )
|
||
{
|
||
/* Coerce the line to vertical or horizontal one: */
|
||
if( ABS( endpos.x - NewPoly->m_Points[NewPoly->m_NumOfPoints * 2 - 2] ) <
|
||
ABS( endpos.y - NewPoly->m_Points[NewPoly->m_NumOfPoints * 2 - 1] ) )
|
||
endpos.x = NewPoly->m_Points[NewPoly->m_NumOfPoints * 2 - 2];
|
||
else
|
||
endpos.y = NewPoly->m_Points[NewPoly->m_NumOfPoints * 2 - 1];
|
||
}
|
||
|
||
NewPoly->m_NumOfPoints++;
|
||
if( erase )
|
||
RedrawOneStruct( panel, DC, NewPoly, XOR_MODE, color );
|
||
|
||
NewPoly->m_Points[NewPoly->m_NumOfPoints * 2 - 2] = endpos.x;
|
||
NewPoly->m_Points[NewPoly->m_NumOfPoints * 2 - 1] = endpos.y;
|
||
RedrawOneStruct( panel, DC, NewPoly, XOR_MODE, color );
|
||
NewPoly->m_NumOfPoints--;
|
||
}
|
||
|
||
|
||
/**********************************************************/
|
||
void WinEDA_SchematicFrame::DeleteCurrentSegment( wxDC* DC )
|
||
/**********************************************************/
|
||
|
||
/*
|
||
* Routine effacant le dernier trait trace, ou l'element pointe par la souris
|
||
*/
|
||
{
|
||
g_ItemToRepeat = NULL;
|
||
|
||
if( (GetScreen()->GetCurItem() == NULL)
|
||
|| ( (GetScreen()->GetCurItem()->m_Flags & IS_NEW) == 0 ) )
|
||
{
|
||
return;
|
||
}
|
||
|
||
/* Trace en cours: annulation */
|
||
if( GetScreen()->GetCurItem()->Type() == DRAW_POLYLINE_STRUCT_TYPE )
|
||
{
|
||
Polyline_in_Ghost( DrawPanel, DC, FALSE ); /* Effacement du trace en cours */
|
||
}
|
||
else
|
||
{
|
||
Segment_in_Ghost( DrawPanel, DC, FALSE ); /* Effacement du trace en cours */
|
||
}
|
||
|
||
EraseStruct( GetScreen()->GetCurItem(), GetScreen() );
|
||
GetScreen()->ManageCurseur = NULL;
|
||
GetScreen()->SetCurItem( NULL );
|
||
}
|
||
|
||
|
||
/***************************************************************************/
|
||
EDA_BaseStruct* WinEDA_SchematicFrame::CreateNewJunctionStruct( wxDC* DC )
|
||
/***************************************************************************/
|
||
|
||
/* Routine to create new connection struct.
|
||
*/
|
||
{
|
||
DrawJunctionStruct* NewConnect;
|
||
|
||
NewConnect = new DrawJunctionStruct( GetScreen()->m_Curseur );
|
||
|
||
g_ItemToRepeat = NewConnect;
|
||
|
||
GetScreen()->CursorOff( DrawPanel, DC ); // Erase schematic cursor
|
||
RedrawOneStruct( DrawPanel, DC, NewConnect, GR_DEFAULT_DRAWMODE );
|
||
GetScreen()->CursorOn( DrawPanel, DC ); // Display schematic cursor
|
||
|
||
NewConnect->Pnext = GetScreen()->EEDrawList;
|
||
GetScreen()->EEDrawList = NewConnect;
|
||
SetFlagModify( GetScreen() );
|
||
return NewConnect;
|
||
}
|
||
|
||
|
||
/*********************************************************************************/
|
||
DrawNoConnectStruct * WinEDA_SchematicFrame::CreateNewNoConnectStruct( wxDC* DC )
|
||
/*********************************************************************************/
|
||
|
||
/*Routine to create new NoConnect struct. ( Symbole de Non Connexion)
|
||
*/
|
||
{
|
||
DrawNoConnectStruct* NewNoConnect;
|
||
|
||
NewNoConnect = new DrawNoConnectStruct( GetScreen()->m_Curseur );
|
||
g_ItemToRepeat = NewNoConnect;
|
||
|
||
GetScreen()->CursorOff( DrawPanel, DC ); // Erase schematic cursor
|
||
RedrawOneStruct( DrawPanel, DC, NewNoConnect, GR_DEFAULT_DRAWMODE );
|
||
GetScreen()->CursorOn( DrawPanel, DC ); // Display schematic cursor
|
||
|
||
NewNoConnect->Pnext = GetScreen()->EEDrawList;
|
||
GetScreen()->EEDrawList = NewNoConnect;
|
||
SetFlagModify( GetScreen() );
|
||
return NewNoConnect;
|
||
}
|
||
|
||
|
||
/**********************************************************/
|
||
static void ExitTrace( WinEDA_DrawFrame* frame, wxDC* DC )
|
||
/**********************************************************/
|
||
/* Routine de sortie des menus de trace */
|
||
{
|
||
BASE_SCREEN* Screen = frame->GetScreen();
|
||
|
||
if( Screen->GetCurItem() ) /* trace en cours */
|
||
{
|
||
Screen->ManageCurseur( frame->DrawPanel, DC, FALSE );
|
||
Screen->ManageCurseur = NULL;
|
||
Screen->ForceCloseManageCurseur = NULL;
|
||
EraseStruct( Screen->GetCurItem(), (SCH_SCREEN*) Screen );
|
||
Screen->SetCurItem( NULL );
|
||
return;
|
||
}
|
||
else
|
||
g_ItemToRepeat = NULL; // Fin de commande generale
|
||
}
|
||
|
||
|
||
/***************************************************/
|
||
void WinEDA_SchematicFrame::RepeatDrawItem( wxDC* DC )
|
||
/***************************************************/
|
||
|
||
/* Routine de recopie du dernier element dessine
|
||
* Les elements duplicables sont
|
||
* fils, bus, traits, textes, labels
|
||
* Les labels termines par un nombre seront incrementes
|
||
*/
|
||
{
|
||
char Line[256];
|
||
int ox = 0, oy = 0;
|
||
|
||
if( g_ItemToRepeat == NULL )
|
||
return;
|
||
|
||
switch( g_ItemToRepeat->Type() )
|
||
{
|
||
case DRAW_JUNCTION_STRUCT_TYPE:
|
||
#undef STRUCT
|
||
#define STRUCT ( (DrawJunctionStruct*) g_ItemToRepeat )
|
||
g_ItemToRepeat = STRUCT->GenCopy();
|
||
STRUCT->m_Pos.x += g_RepeatStep.x; ox = STRUCT->m_Pos.x;
|
||
STRUCT->m_Pos.y += g_RepeatStep.y; oy = STRUCT->m_Pos.y;
|
||
break;
|
||
|
||
case DRAW_NOCONNECT_STRUCT_TYPE:
|
||
#undef STRUCT
|
||
#define STRUCT ( (DrawNoConnectStruct*) g_ItemToRepeat )
|
||
g_ItemToRepeat = STRUCT->GenCopy();
|
||
STRUCT->m_Pos.x += g_RepeatStep.x; ox = STRUCT->m_Pos.x;
|
||
STRUCT->m_Pos.y += g_RepeatStep.y; oy = STRUCT->m_Pos.y;
|
||
break;
|
||
|
||
case TYPE_SCH_TEXT:
|
||
#undef STRUCT
|
||
#define STRUCT ( (SCH_TEXT*) g_ItemToRepeat )
|
||
g_ItemToRepeat = STRUCT->GenCopy();
|
||
STRUCT->m_Pos.x += g_RepeatStep.x; ox = STRUCT->m_Pos.x;
|
||
STRUCT->m_Pos.y += g_RepeatStep.y; oy = STRUCT->m_Pos.y;
|
||
/*** Increment du numero de label ***/
|
||
strcpy( Line, STRUCT->GetText() );
|
||
IncrementLabelMember( Line );
|
||
STRUCT->m_Text = Line;
|
||
break;
|
||
|
||
|
||
case TYPE_SCH_LABEL:
|
||
case TYPE_SCH_GLOBALLABEL:
|
||
case TYPE_SCH_HIERLABEL:
|
||
#undef STRUCT
|
||
#define STRUCT ( (SCH_LABEL*) g_ItemToRepeat )
|
||
g_ItemToRepeat = STRUCT->GenCopy();
|
||
STRUCT->m_Pos.x += g_RepeatStep.x; ox = STRUCT->m_Pos.x;
|
||
STRUCT->m_Pos.y += g_RepeatStep.y; oy = STRUCT->m_Pos.y;
|
||
/*** Increment du numero de label ***/
|
||
strcpy( Line, STRUCT->GetText() );
|
||
IncrementLabelMember( Line );
|
||
STRUCT->m_Text = Line;
|
||
break;
|
||
|
||
case DRAW_SEGMENT_STRUCT_TYPE:
|
||
#undef STRUCT
|
||
#define STRUCT ( (DrawSegmentStruct*) g_ItemToRepeat )
|
||
g_ItemToRepeat = STRUCT->GenCopy();
|
||
STRUCT->m_Start.x += g_RepeatStep.x; ox = STRUCT->m_Start.x;
|
||
STRUCT->m_Start.y += g_RepeatStep.y; oy = STRUCT->m_Start.y;
|
||
STRUCT->m_End.x += g_RepeatStep.x;
|
||
STRUCT->m_End.y += g_RepeatStep.y;
|
||
break;
|
||
|
||
case DRAW_RACCORD_STRUCT_TYPE:
|
||
#undef STRUCT
|
||
#define STRUCT ( (DrawRaccordStruct*) g_ItemToRepeat )
|
||
g_ItemToRepeat = STRUCT->GenCopy();
|
||
STRUCT->m_Pos.x += g_RepeatStep.x; ox = STRUCT->m_Pos.x;
|
||
STRUCT->m_Pos.y += g_RepeatStep.y; oy = STRUCT->m_Pos.y;
|
||
break;
|
||
|
||
default:
|
||
g_ItemToRepeat = NULL;
|
||
DisplayError( this, "Repeat Type Error", 10 );
|
||
break;
|
||
}
|
||
|
||
if( g_ItemToRepeat )
|
||
{
|
||
g_ItemToRepeat->Pnext = GetScreen()->EEDrawList;
|
||
GetScreen()->EEDrawList = g_ItemToRepeat;
|
||
TestDanglingEnds( GetScreen()->EEDrawList, NULL );
|
||
RedrawOneStruct( DrawPanel, DC, g_ItemToRepeat, GR_DEFAULT_DRAWMODE );
|
||
|
||
// GetScreen()->Curseur.x = ox; GetScreen()->Curseur.x = oy;
|
||
// DrawPanel->MouseTo( DrawPanel->CursorScreenPosition() );
|
||
}
|
||
}
|
||
|
||
|
||
/******************************************/
|
||
void IncrementLabelMember( char* Line )
|
||
/******************************************/
|
||
|
||
/* Routine incrementant les labels, c'est a dire pour les textes finissant
|
||
* par un nombre, ajoutant <RepeatDeltaLabel> a ce nombre
|
||
*/
|
||
{
|
||
char* strnum;
|
||
int ii;
|
||
|
||
strnum = Line + strlen( Line ) - 1;
|
||
if( !isdigit( *strnum ) )
|
||
return;
|
||
|
||
while( (strnum >= Line) && isdigit( *strnum ) )
|
||
strnum--;
|
||
|
||
strnum++; /* pointe le debut de la chaine des digits */
|
||
ii = atoi( strnum ) + g_RepeatDeltaLabel;
|
||
sprintf( strnum, "%d", ii );
|
||
}
|
||
|
||
|
||
/***************************************************************************/
|
||
static bool IsTerminalPoint( SCH_SCREEN* screen, const wxPoint& pos, int layer )
|
||
/***************************************************************************/
|
||
|
||
/* Returne TRUE si pos est un point possible pour terminer automatiquement un
|
||
* segment, c'est a dire pour
|
||
* - type WIRE, si il y a
|
||
* - une jonction
|
||
* - ou une pin
|
||
* - ou une extr<74>mit<69> unique de fil
|
||
*
|
||
* - type BUS, si il y a
|
||
* - ou une extr<74>mit<69> unique de BUS
|
||
*/
|
||
{
|
||
EDA_BaseStruct* item;
|
||
LibDrawPin* pin;
|
||
DrawLibItemStruct* LibItem = NULL;
|
||
Hierarchical_PIN_Sheet_Struct* pinsheet;
|
||
wxPoint itempos;
|
||
|
||
switch( layer )
|
||
{
|
||
case LAYER_BUS:
|
||
item = PickStruct( screen, BUSITEM );
|
||
if( item )
|
||
return TRUE;
|
||
pinsheet = LocateAnyPinSheet( pos, screen->EEDrawList );
|
||
if( pinsheet && IsBusLabel( pinsheet->GetText() ) )
|
||
{
|
||
itempos = pinsheet->m_Pos;
|
||
if( (itempos.x == pos.x) && (itempos.y == pos.y) )
|
||
return TRUE;
|
||
}
|
||
break;
|
||
|
||
case LAYER_NOTES:
|
||
item = PickStruct( screen, DRAWITEM );
|
||
if( item )
|
||
return TRUE;
|
||
break;
|
||
|
||
case LAYER_WIRE:
|
||
item = PickStruct( screen, RACCORDITEM | JUNCTIONITEM );
|
||
if( item )
|
||
return TRUE;
|
||
|
||
pin = LocateAnyPin( screen->EEDrawList, pos, &LibItem );
|
||
if( pin && LibItem )
|
||
{
|
||
// calcul de la position exacte du point de connexion de la pin,
|
||
// selon orientation du composant:
|
||
itempos = LibItem->GetScreenCoord( pin->m_Pos );
|
||
itempos.x += LibItem->m_Pos.x;
|
||
itempos.y += LibItem->m_Pos.y;
|
||
if( (itempos.x == pos.x) && (itempos.y == pos.y) )
|
||
return TRUE;
|
||
}
|
||
|
||
item = PickStruct( screen, WIREITEM );
|
||
if( item )
|
||
return TRUE;
|
||
|
||
item = PickStruct( screen, LABELITEM );
|
||
if( item && (item->Type() != TYPE_SCH_TEXT)
|
||
&& ( ( (SCH_GLOBALLABEL*) item )->m_Pos.x == pos.x )
|
||
&& ( ( (SCH_GLOBALLABEL*) item )->m_Pos.y == pos.y ) )
|
||
return TRUE;
|
||
|
||
pinsheet = LocateAnyPinSheet( pos, screen->EEDrawList );
|
||
if( pinsheet && !IsBusLabel( pinsheet->GetText() ) )
|
||
{
|
||
itempos = pinsheet->m_Pos;
|
||
if( (itempos.x == pos.x) && (itempos.y == pos.y) )
|
||
return TRUE;
|
||
}
|
||
|
||
break;
|
||
|
||
default:
|
||
break;
|
||
}
|
||
|
||
return FALSE;
|
||
}
|