2007-08-20 01:20:48 +00:00
|
|
|
/******************************************************/
|
|
|
|
/* Edition des contours du pcb: Routines */
|
|
|
|
/* d'effacement et d'edition de segments et contours */
|
|
|
|
/* du type PCB, draw, edgePCB */
|
|
|
|
/******************************************************/
|
2007-05-06 16:03:28 +00:00
|
|
|
|
|
|
|
#include "fctsys.h"
|
|
|
|
#include "gr_basic.h"
|
|
|
|
|
|
|
|
#include "common.h"
|
|
|
|
#include "pcbnew.h"
|
|
|
|
|
|
|
|
#include "protos.h"
|
|
|
|
|
|
|
|
/* Routines Locales */
|
2007-08-20 01:20:48 +00:00
|
|
|
static void Exit_EditEdge( WinEDA_DrawPanel* Panel, wxDC* DC );
|
|
|
|
static void Montre_Position_NewSegment( WinEDA_DrawPanel* panel, wxDC* DC, bool erase );
|
|
|
|
static void Move_Segment( WinEDA_DrawPanel* panel, wxDC* DC, bool erase );
|
2007-05-06 16:03:28 +00:00
|
|
|
|
|
|
|
/* Variables locales : */
|
2007-08-20 01:20:48 +00:00
|
|
|
static wxPoint cursor_pos; // position originelle du curseur souris (fct deplacement)
|
|
|
|
static wxPoint cursor_pos0; // position courante du curseur souris
|
2007-05-06 16:03:28 +00:00
|
|
|
|
|
|
|
/****************************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
void WinEDA_PcbFrame::Start_Move_DrawItem( DRAWSEGMENT* drawitem, wxDC* DC )
|
2007-05-06 16:03:28 +00:00
|
|
|
/****************************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
|
|
|
|
/* Routine de preparation du deplacement d'un element graphique type DRAWSEGMENT
|
|
|
|
*/
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
if( drawitem == NULL )
|
|
|
|
return;
|
|
|
|
Trace_DrawSegmentPcb( DrawPanel, DC, drawitem, GR_XOR );
|
|
|
|
drawitem->m_Flags |= IS_MOVED;
|
|
|
|
cursor_pos = cursor_pos0 = GetScreen()->m_Curseur;
|
2007-08-20 19:33:15 +00:00
|
|
|
drawitem->Display_Infos( this );
|
2007-08-20 01:20:48 +00:00
|
|
|
DrawPanel->ManageCurseur = Move_Segment;
|
|
|
|
DrawPanel->ForceCloseManageCurseur = Exit_EditEdge;
|
|
|
|
GetScreen()->SetCurItem( drawitem );
|
|
|
|
DrawPanel->ManageCurseur( DrawPanel, DC, FALSE );
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2007-05-06 16:03:28 +00:00
|
|
|
/*********************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
void WinEDA_PcbFrame::Place_DrawItem( DRAWSEGMENT* drawitem, wxDC* DC )
|
2007-05-06 16:03:28 +00:00
|
|
|
/*********************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2007-05-06 16:03:28 +00:00
|
|
|
/*
|
2007-08-20 01:20:48 +00:00
|
|
|
* Routine de placement de l'element graphique type DRAWSEGMENT en cours de deplacement
|
|
|
|
*/
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
if( drawitem == NULL )
|
|
|
|
return;
|
|
|
|
|
|
|
|
Trace_DrawSegmentPcb( DrawPanel, DC, drawitem, GR_OR );
|
|
|
|
DrawPanel->ManageCurseur = NULL;
|
|
|
|
DrawPanel->ForceCloseManageCurseur = NULL;
|
|
|
|
GetScreen()->SetCurItem( NULL );
|
|
|
|
GetScreen()->SetModify();
|
|
|
|
drawitem->m_Flags = 0;
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/************************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
static void Move_Segment( WinEDA_DrawPanel* panel, wxDC* DC, bool erase )
|
2007-05-06 16:03:28 +00:00
|
|
|
/************************************************************************/
|
|
|
|
/* redessin du contour du Segment Edge lors des deplacements de la souris */
|
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
DRAWSEGMENT* Segment = (DRAWSEGMENT*)
|
|
|
|
panel->GetScreen()->GetCurItem();
|
|
|
|
int t_fill = DisplayOpt.DisplayDrawItems;
|
|
|
|
|
|
|
|
if( Segment == NULL )
|
|
|
|
return;
|
|
|
|
|
|
|
|
DisplayOpt.DisplayDrawItems = SKETCH;
|
|
|
|
|
|
|
|
/* efface ancienne position */
|
|
|
|
if( erase )
|
|
|
|
Trace_DrawSegmentPcb( panel, DC, Segment, GR_XOR );
|
|
|
|
|
|
|
|
wxPoint delta;
|
|
|
|
delta.x = panel->GetScreen()->m_Curseur.x - cursor_pos.x;
|
|
|
|
delta.y = panel->GetScreen()->m_Curseur.y - cursor_pos.y;
|
|
|
|
Segment->m_Start.x += delta.x;
|
|
|
|
Segment->m_Start.y += delta.y;
|
|
|
|
Segment->m_End.x += delta.x;
|
|
|
|
Segment->m_End.y += delta.y;
|
|
|
|
cursor_pos = panel->GetScreen()->m_Curseur;
|
|
|
|
|
|
|
|
Trace_DrawSegmentPcb( panel, DC, Segment, GR_XOR );
|
|
|
|
DisplayOpt.DisplayDrawItems = t_fill;
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**************************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
void WinEDA_PcbFrame::Delete_Segment_Edge( DRAWSEGMENT* Segment, wxDC* DC )
|
2007-05-06 16:03:28 +00:00
|
|
|
/**************************************************************************/
|
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
EDA_BaseStruct* PtStruct;
|
|
|
|
int track_fill_copy = DisplayOpt.DisplayDrawItems;
|
|
|
|
|
|
|
|
if( Segment == NULL )
|
|
|
|
return;
|
|
|
|
|
|
|
|
if( Segment->m_Flags & IS_NEW ) // Trace en cours, on peut effacer le dernier segment
|
|
|
|
{
|
|
|
|
/* effacement du segment en cours de trace */
|
|
|
|
DisplayOpt.DisplayDrawItems = SKETCH;
|
|
|
|
Trace_DrawSegmentPcb( DrawPanel, DC, Segment, GR_XOR );
|
|
|
|
PtStruct = Segment->Pback;
|
|
|
|
DeleteStructure( Segment );
|
2007-09-01 12:00:30 +00:00
|
|
|
if( PtStruct && (PtStruct->Type() == TYPEDRAWSEGMENT ) )
|
2007-08-20 01:20:48 +00:00
|
|
|
Segment = (DRAWSEGMENT*) PtStruct;
|
|
|
|
DisplayOpt.DisplayDrawItems = track_fill_copy;
|
|
|
|
GetScreen()->SetCurItem( NULL );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Trace_DrawSegmentPcb( DrawPanel, DC, (DRAWSEGMENT*) Segment, GR_XOR );
|
|
|
|
Segment->m_Flags = 0;
|
|
|
|
DeleteStructure( Segment );
|
|
|
|
GetScreen()->SetCurItem( NULL );
|
|
|
|
GetScreen()->SetModify();
|
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*************************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
void WinEDA_PcbFrame::Drawing_SetNewWidth( DRAWSEGMENT* DrawSegm, wxDC* DC )
|
2007-05-06 16:03:28 +00:00
|
|
|
/*************************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2007-05-06 16:03:28 +00:00
|
|
|
/* Met a la largeur courante le segment pointe part la souris
|
2007-08-20 01:20:48 +00:00
|
|
|
*/
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
if( DrawSegm == NULL )
|
|
|
|
return;
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2007-08-20 01:20:48 +00:00
|
|
|
Trace_DrawSegmentPcb( DrawPanel, DC, DrawSegm, GR_XOR );
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2007-08-23 04:28:46 +00:00
|
|
|
if( DrawSegm->GetLayer() == EDGE_N )
|
2007-08-20 01:20:48 +00:00
|
|
|
DrawSegm->m_Width = g_DesignSettings.m_EdgeSegmentWidth;
|
|
|
|
else
|
|
|
|
DrawSegm->m_Width = g_DesignSettings.m_DrawSegmentWidth;
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2007-08-20 01:20:48 +00:00
|
|
|
Trace_DrawSegmentPcb( DrawPanel, DC, DrawSegm, GR_OR );
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2007-08-20 19:33:15 +00:00
|
|
|
DrawSegm->Display_Infos( this );
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2007-08-20 01:20:48 +00:00
|
|
|
GetScreen()->SetModify();
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/******************************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
void WinEDA_PcbFrame::Delete_Drawings_All_Layer( DRAWSEGMENT* Segment, wxDC* DC )
|
2007-05-06 16:03:28 +00:00
|
|
|
/******************************************************************************/
|
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
DRAWSEGMENT* pt_segm;
|
|
|
|
TEXTE_PCB* pt_txt;
|
2007-09-01 12:00:30 +00:00
|
|
|
BOARD_ITEM* PtStruct;
|
|
|
|
BOARD_ITEM* PtNext;
|
2007-08-20 01:20:48 +00:00
|
|
|
COTATION* Cotation;
|
2007-08-23 04:28:46 +00:00
|
|
|
int layer = Segment->GetLayer();
|
2007-08-20 01:20:48 +00:00
|
|
|
|
|
|
|
if( layer <= CMP_N )
|
|
|
|
{
|
|
|
|
DisplayError( this, _( "Copper layer global delete not allowed!" ), 20 );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( Segment->m_Flags )
|
|
|
|
{
|
|
|
|
DisplayError( this, _( "Segment is being edited" ), 10 );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
wxString msg = _( "Delete Layer " ) + ReturnPcbLayerName( layer );
|
|
|
|
if( !IsOK( this, msg ) )
|
|
|
|
return;
|
|
|
|
|
|
|
|
PtStruct = m_Pcb->m_Drawings;
|
|
|
|
for( ; PtStruct != NULL; PtStruct = PtNext )
|
|
|
|
{
|
|
|
|
GetScreen()->SetModify();
|
2007-09-01 12:00:30 +00:00
|
|
|
PtNext = PtStruct->Next();
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2007-09-01 12:00:30 +00:00
|
|
|
switch( PtStruct->Type() )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
|
|
|
case TYPEDRAWSEGMENT:
|
|
|
|
pt_segm = (DRAWSEGMENT*) PtStruct;
|
2007-08-23 04:28:46 +00:00
|
|
|
if( pt_segm->GetLayer() == layer )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
|
|
|
Trace_DrawSegmentPcb( DrawPanel, DC, pt_segm, GR_XOR );
|
|
|
|
DeleteStructure( PtStruct );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TYPETEXTE:
|
|
|
|
pt_txt = (TEXTE_PCB*) PtStruct;
|
2007-08-23 04:28:46 +00:00
|
|
|
if( pt_txt->GetLayer() == layer )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
|
|
|
pt_txt->Draw( DrawPanel, DC, wxPoint( 0, 0 ), GR_XOR );
|
|
|
|
DeleteStructure( PtStruct );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case TYPECOTATION:
|
|
|
|
Cotation = (COTATION*) PtStruct;
|
2007-08-23 04:28:46 +00:00
|
|
|
if( Cotation->GetLayer() == layer )
|
2007-08-20 01:20:48 +00:00
|
|
|
{
|
|
|
|
Cotation->Draw( DrawPanel, DC, wxPoint( 0, 0 ), GR_XOR );
|
|
|
|
DeleteStructure( PtStruct );
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
DisplayError( this, wxT( "Type Drawing Inconnu" ) );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
static void Exit_EditEdge( WinEDA_DrawPanel* Panel, wxDC* DC )
|
2007-05-06 16:03:28 +00:00
|
|
|
/*************************************************************/
|
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
DRAWSEGMENT* Segment = (DRAWSEGMENT*) Panel->GetScreen()->GetCurItem();
|
|
|
|
|
|
|
|
if( Segment == NULL )
|
|
|
|
return;
|
|
|
|
|
|
|
|
if( Segment->m_Flags & IS_NEW )
|
|
|
|
{
|
|
|
|
Panel->ManageCurseur( Panel, DC, FALSE );
|
|
|
|
DeleteStructure( Segment );
|
|
|
|
Segment = NULL;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
wxPoint pos = Panel->GetScreen()->m_Curseur;
|
|
|
|
Panel->GetScreen()->m_Curseur = cursor_pos0;
|
|
|
|
Panel->ManageCurseur( Panel, DC, TRUE );
|
|
|
|
Panel->GetScreen()->m_Curseur = pos;
|
|
|
|
Segment->m_Flags = 0;
|
|
|
|
Trace_DrawSegmentPcb( Panel, DC, Segment, GR_OR );
|
|
|
|
}
|
|
|
|
Panel->ManageCurseur = NULL;
|
|
|
|
Panel->ForceCloseManageCurseur = NULL;
|
|
|
|
Panel->GetScreen()->SetCurItem( NULL );
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2007-05-06 16:03:28 +00:00
|
|
|
/**********************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
DRAWSEGMENT* WinEDA_PcbFrame::Begin_DrawSegment( DRAWSEGMENT* Segment,
|
|
|
|
int shape, wxDC* DC )
|
2007-05-06 16:03:28 +00:00
|
|
|
/**********************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
|
2007-05-06 16:03:28 +00:00
|
|
|
/* Routine d'initialisation du trace d'un segment de type autre que piste
|
2007-08-20 01:20:48 +00:00
|
|
|
*/
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
int s_large;
|
|
|
|
int angle = 0;
|
|
|
|
DRAWSEGMENT* DrawItem;
|
|
|
|
|
|
|
|
s_large = g_DesignSettings.m_DrawSegmentWidth;
|
|
|
|
if( GetScreen()->m_Active_Layer == EDGE_N )
|
|
|
|
{
|
|
|
|
s_large = g_DesignSettings.m_EdgeSegmentWidth;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( shape == S_ARC )
|
|
|
|
angle = 900;
|
|
|
|
|
|
|
|
if( Segment == NULL ) /* debut reel du trace */
|
|
|
|
{
|
|
|
|
GetScreen()->SetCurItem( Segment = new DRAWSEGMENT( m_Pcb ) );
|
|
|
|
Segment->m_Flags = IS_NEW;
|
2007-08-23 04:28:46 +00:00
|
|
|
Segment->SetLayer( GetScreen()->m_Active_Layer );
|
2007-08-20 01:20:48 +00:00
|
|
|
Segment->m_Width = s_large;
|
|
|
|
Segment->m_Shape = shape;
|
|
|
|
Segment->m_Angle = 900;
|
|
|
|
Segment->m_Start = Segment->m_End = GetScreen()->m_Curseur;
|
|
|
|
DrawPanel->ManageCurseur = Montre_Position_NewSegment;
|
|
|
|
DrawPanel->ForceCloseManageCurseur = Exit_EditEdge;
|
|
|
|
}
|
|
|
|
else /* trace en cours : les coord du point d'arrivee ont ete mises
|
|
|
|
* a jour par la routine Montre_Position_NewSegment*/
|
|
|
|
{
|
|
|
|
if( (Segment->m_Start.x != Segment->m_End.x )
|
|
|
|
|| (Segment->m_Start.y != Segment->m_End.y ) )
|
|
|
|
{
|
|
|
|
if( Segment->m_Shape == S_SEGMENT )
|
|
|
|
{
|
|
|
|
Segment->Pnext = m_Pcb->m_Drawings;
|
|
|
|
Segment->Pback = m_Pcb;
|
|
|
|
if( m_Pcb->m_Drawings )
|
|
|
|
m_Pcb->m_Drawings->Pback = Segment;
|
|
|
|
m_Pcb->m_Drawings = Segment;
|
|
|
|
GetScreen()->SetModify();
|
|
|
|
Segment->m_Flags = 0;
|
|
|
|
|
|
|
|
Trace_DrawSegmentPcb( DrawPanel, DC, Segment, GR_OR );
|
|
|
|
|
|
|
|
DrawItem = Segment;
|
|
|
|
|
|
|
|
GetScreen()->SetCurItem( Segment = new DRAWSEGMENT( m_Pcb ) );
|
|
|
|
|
|
|
|
Segment->m_Flags = IS_NEW;
|
2007-08-23 04:28:46 +00:00
|
|
|
Segment->SetLayer( DrawItem->GetLayer() );
|
2007-08-20 01:20:48 +00:00
|
|
|
Segment->m_Width = s_large;
|
|
|
|
Segment->m_Shape = DrawItem->m_Shape;
|
|
|
|
Segment->m_Type = DrawItem->m_Type;
|
|
|
|
Segment->m_Angle = DrawItem->m_Angle;
|
|
|
|
Segment->m_Start = Segment->m_End = DrawItem->m_End;
|
|
|
|
Montre_Position_NewSegment( DrawPanel, DC, FALSE );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
End_Edge( Segment, DC );
|
|
|
|
Segment = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Segment;
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/***************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
void WinEDA_PcbFrame::End_Edge( DRAWSEGMENT* Segment, wxDC* DC )
|
2007-05-06 16:03:28 +00:00
|
|
|
/***************************************************************/
|
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
if( Segment == NULL )
|
|
|
|
return;
|
|
|
|
Trace_DrawSegmentPcb( DrawPanel, DC, (DRAWSEGMENT*) Segment, GR_OR );
|
|
|
|
|
|
|
|
/* Effacement si Longueur nulle */
|
|
|
|
if( (Segment->m_Start.x == Segment->m_End.x)
|
|
|
|
&& (Segment->m_Start.y == Segment->m_End.y) )
|
|
|
|
DeleteStructure( Segment );
|
|
|
|
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Segment->m_Flags = 0;
|
|
|
|
Segment->Pnext = m_Pcb->m_Drawings;
|
|
|
|
Segment->Pback = m_Pcb;
|
|
|
|
if( m_Pcb->m_Drawings )
|
|
|
|
m_Pcb->m_Drawings->Pback = Segment;
|
|
|
|
m_Pcb->m_Drawings = Segment;
|
|
|
|
GetScreen()->SetModify();
|
|
|
|
}
|
|
|
|
|
|
|
|
DrawPanel->ManageCurseur = NULL;
|
|
|
|
DrawPanel->ForceCloseManageCurseur = NULL;
|
|
|
|
GetScreen()->SetCurItem( NULL );
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
static void Montre_Position_NewSegment( WinEDA_DrawPanel* panel,
|
|
|
|
wxDC* DC, bool erase )
|
2007-05-06 16:03:28 +00:00
|
|
|
/************************************************************/
|
|
|
|
/* redessin du contour du Segment Edge lors des deplacements de la souris */
|
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
DRAWSEGMENT* Segment = (DRAWSEGMENT*)
|
|
|
|
panel->GetScreen()->GetCurItem();
|
|
|
|
int t_fill = DisplayOpt.DisplayDrawItems;
|
|
|
|
|
|
|
|
if( Segment == NULL )
|
|
|
|
return;
|
|
|
|
|
|
|
|
DisplayOpt.DisplayDrawItems = SKETCH;
|
|
|
|
|
|
|
|
/* efface ancienne position */
|
|
|
|
if( erase )
|
|
|
|
Trace_DrawSegmentPcb( panel, DC, Segment, GR_XOR );
|
|
|
|
|
|
|
|
if( Segments_45_Only && (Segment->m_Shape == S_SEGMENT ) )
|
|
|
|
{
|
|
|
|
Calcule_Coord_Extremite_45( Segment->m_Start.x, Segment->m_Start.y,
|
|
|
|
&Segment->m_End.x, &Segment->m_End.y );
|
|
|
|
}
|
|
|
|
else /* ici l'angle d'inclinaison est quelconque */
|
|
|
|
{
|
|
|
|
Segment->m_End = panel->GetScreen()->m_Curseur;
|
|
|
|
}
|
|
|
|
|
|
|
|
Trace_DrawSegmentPcb( panel, DC, Segment, GR_XOR );
|
|
|
|
DisplayOpt.DisplayDrawItems = t_fill;
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|