2007-05-06 16:03:28 +00:00
|
|
|
/*****************************************************************************
|
|
|
|
* Program to draw EE diagrams. *
|
|
|
|
* This module redraw/draw all structs. *
|
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
#include "fctsys.h"
|
|
|
|
#include "gr_basic.h"
|
|
|
|
#include "common.h"
|
2009-02-04 15:25:03 +00:00
|
|
|
#include "class_drawpanel.h"
|
2009-04-05 20:49:15 +00:00
|
|
|
#include "class_drawpickedstruct.h"
|
|
|
|
#include "appl_wxstruct.h"
|
2009-02-04 15:25:03 +00:00
|
|
|
|
2007-05-06 16:03:28 +00:00
|
|
|
#include "program.h"
|
|
|
|
#include "libcmp.h"
|
|
|
|
#include "general.h"
|
|
|
|
#include "protos.h"
|
|
|
|
|
|
|
|
|
2009-04-05 20:49:15 +00:00
|
|
|
extern char marq_bitmap[];
|
2008-03-13 06:36:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
static EDA_BaseStruct* HighLightStruct = NULL;
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2009-04-05 20:49:15 +00:00
|
|
|
|
2008-03-27 21:02:42 +00:00
|
|
|
void DrawDanglingSymbol( WinEDA_DrawPanel* panel, wxDC* DC,
|
2009-01-02 17:07:50 +00:00
|
|
|
const wxPoint& pos, int Color )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2009-04-05 20:49:15 +00:00
|
|
|
BASE_SCREEN* screen = panel->GetScreen();
|
|
|
|
|
|
|
|
if( !screen->m_IsPrinting ) /* Draw but do not print the Dangling Symbol */
|
2008-03-13 06:36:53 +00:00
|
|
|
{
|
|
|
|
GRRect( &panel->m_ClipBox, DC,
|
2009-01-02 17:07:50 +00:00
|
|
|
pos.x - DANGLING_SYMBOL_SIZE, pos.y - DANGLING_SYMBOL_SIZE,
|
|
|
|
pos.x + DANGLING_SYMBOL_SIZE, pos.y + DANGLING_SYMBOL_SIZE,
|
|
|
|
0, Color );
|
2008-03-13 06:36:53 +00:00
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-03-13 06:36:53 +00:00
|
|
|
void SetHighLightStruct( EDA_BaseStruct* HighLight )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2008-03-13 06:36:53 +00:00
|
|
|
HighLightStruct = HighLight;
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
2008-03-13 06:36:53 +00:00
|
|
|
|
2007-05-06 16:03:28 +00:00
|
|
|
/*
|
2009-01-02 17:07:50 +00:00
|
|
|
* Redraws only the active window which is assumed to be whole visible.
|
2008-03-13 06:36:53 +00:00
|
|
|
*/
|
2009-04-05 20:49:15 +00:00
|
|
|
void WinEDA_SchematicFrame::RedrawActiveWindow( wxDC* DC, bool EraseBg )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2008-03-13 06:36:53 +00:00
|
|
|
wxString title;
|
|
|
|
|
|
|
|
if( GetScreen() == NULL )
|
|
|
|
return;
|
|
|
|
|
|
|
|
ActiveScreen = GetScreen();
|
|
|
|
|
2008-04-16 08:40:31 +00:00
|
|
|
/* Reinit draw and pen parameters */
|
2008-03-13 06:36:53 +00:00
|
|
|
GRResetPenAndBrush( DC );
|
|
|
|
DC->SetBackground( *wxBLACK_BRUSH );
|
|
|
|
DC->SetBackgroundMode( wxTRANSPARENT );
|
|
|
|
|
|
|
|
DrawPanel->CursorOff( DC ); // effacement curseur
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2008-03-13 06:36:53 +00:00
|
|
|
if( DrawPanel->ManageCurseur )
|
|
|
|
{
|
|
|
|
DrawPanel->ManageCurseur( DrawPanel, DC, FALSE );
|
|
|
|
}
|
|
|
|
|
|
|
|
if( EraseBg )
|
|
|
|
DrawPanel->EraseScreen( DC );
|
|
|
|
|
|
|
|
DrawPanel->DrawBackGround( DC );
|
|
|
|
|
2009-04-05 20:49:15 +00:00
|
|
|
RedrawStructList( DrawPanel, DC, GetScreen()->EEDrawList,
|
|
|
|
GR_DEFAULT_DRAWMODE );
|
2008-03-13 06:36:53 +00:00
|
|
|
|
|
|
|
TraceWorkSheet( DC, GetScreen(), g_DrawMinimunLineWidth );
|
|
|
|
|
|
|
|
DrawPanel->CursorOn( DC ); // reaffichage curseur
|
|
|
|
if( DrawPanel->ManageCurseur )
|
|
|
|
{
|
|
|
|
DrawPanel->ManageCurseur( DrawPanel, DC, FALSE );
|
|
|
|
}
|
|
|
|
|
2009-04-05 20:49:15 +00:00
|
|
|
UpdateStatusBar();
|
2008-03-13 06:36:53 +00:00
|
|
|
GetScreen()->ClrRefreshReq();
|
2008-04-16 08:40:31 +00:00
|
|
|
|
|
|
|
// Display the sheet filename, and the sheet path, for non root sheets
|
2009-04-05 20:49:15 +00:00
|
|
|
if( GetScreen()->m_FileName == g_DefaultSchematicFileName )
|
2008-03-13 06:36:53 +00:00
|
|
|
{
|
2009-04-05 20:49:15 +00:00
|
|
|
wxString msg = wxGetApp().GetAppName() + wxT( " " ) + GetBuildVersion();
|
|
|
|
title.Printf( wxT( "%s [%s]" ), msg.GetData(),
|
|
|
|
GetScreen()->m_FileName.GetData() );
|
2008-03-13 06:36:53 +00:00
|
|
|
SetTitle( title );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-04-16 08:40:31 +00:00
|
|
|
title = wxT( "[" );
|
2009-01-02 17:07:50 +00:00
|
|
|
title << GetScreen()->m_FileName << wxT( "] " ) << _( "Sheet" );
|
2008-04-16 08:40:31 +00:00
|
|
|
title << wxT( " " ) << m_CurrentSheet->PathHumanReadable();
|
2008-03-13 06:36:53 +00:00
|
|
|
SetTitle( title );
|
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
2008-03-13 06:36:53 +00:00
|
|
|
|
2009-04-05 20:49:15 +00:00
|
|
|
/**
|
|
|
|
* PrintPage
|
2008-07-31 15:30:57 +00:00
|
|
|
* used to print a page.
|
|
|
|
* Print the page pointed by ActiveScreen, set by the calling print function
|
|
|
|
* @param DC = wxDC given by the calling print function
|
|
|
|
* @param Print_Sheet_Ref = true to print page references
|
|
|
|
* @param PrintMask = not used here
|
2008-11-02 19:52:57 +00:00
|
|
|
* @param aPrintMirrorMode = not used here (Set when printing in mirror mode)
|
2008-07-31 15:30:57 +00:00
|
|
|
*/
|
2009-04-05 20:49:15 +00:00
|
|
|
void WinEDA_DrawPanel::PrintPage( wxDC* DC, bool Print_Sheet_Ref,
|
|
|
|
int PrintMask, bool aPrintMirrorMode )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2008-03-13 06:36:53 +00:00
|
|
|
wxBeginBusyCursor();
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2008-07-31 15:30:57 +00:00
|
|
|
RedrawStructList( this, DC, ActiveScreen->EEDrawList, GR_COPY );
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2008-03-13 06:36:53 +00:00
|
|
|
if( Print_Sheet_Ref )
|
2008-07-31 15:30:57 +00:00
|
|
|
m_Parent->TraceWorkSheet( DC, ActiveScreen, g_DrawMinimunLineWidth );
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2008-03-13 06:36:53 +00:00
|
|
|
wxEndBusyCursor();
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*****************************************************************************
|
|
|
|
* Routine to redraw list of structs. *
|
|
|
|
* If the list is of DrawPickStruct types then the picked item are drawn. *
|
|
|
|
*****************************************************************************/
|
2008-03-13 06:36:53 +00:00
|
|
|
void RedrawStructList( WinEDA_DrawPanel* panel, wxDC* DC,
|
2008-04-14 19:22:48 +00:00
|
|
|
SCH_ITEM* Structs, int DrawMode, int Color )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2008-03-13 06:36:53 +00:00
|
|
|
while( Structs )
|
|
|
|
{
|
|
|
|
if( Structs->Type() == DRAW_PICK_ITEM_STRUCT_TYPE )
|
|
|
|
{
|
2009-04-05 20:49:15 +00:00
|
|
|
SCH_ITEM* item =
|
|
|
|
(SCH_ITEM*) ( (DrawPickedStruct*) Structs )->m_PickedStruct;
|
2008-03-13 16:45:07 +00:00
|
|
|
|
|
|
|
// uncomment line below when there is a virtual EDA_BaseStruct::GetBoundingBox()
|
2009-01-02 17:07:50 +00:00
|
|
|
// if( panel->m_ClipBox.Intersects( item->GetBoundingBox() ) )
|
2008-03-13 06:36:53 +00:00
|
|
|
{
|
|
|
|
RedrawOneStruct( panel, DC, item, DrawMode, Color );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if( !(Structs->m_Flags & IS_MOVED) )
|
|
|
|
{
|
2008-03-13 16:45:07 +00:00
|
|
|
// uncomment line below when there is a virtual EDA_BaseStruct::GetBoundingBox()
|
2009-01-02 17:07:50 +00:00
|
|
|
// if( panel->m_ClipBox.Intersects( Structs->GetBoundingBox() ) )
|
|
|
|
RedrawOneStruct( panel, DC, Structs, DrawMode, Color );
|
2008-03-13 06:36:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-04-14 19:22:48 +00:00
|
|
|
Structs = Structs->Next();
|
2008-03-13 06:36:53 +00:00
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
2008-03-13 06:36:53 +00:00
|
|
|
|
2007-05-06 16:03:28 +00:00
|
|
|
/*****************************************************************************
|
|
|
|
* Routine to redraw list of structs. *
|
|
|
|
*****************************************************************************/
|
2008-03-13 06:36:53 +00:00
|
|
|
void RedrawOneStruct( WinEDA_DrawPanel* panel, wxDC* DC,
|
2008-04-14 19:22:48 +00:00
|
|
|
SCH_ITEM* Struct, int DrawMode, int Color )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2008-03-13 06:36:53 +00:00
|
|
|
if( Struct == NULL )
|
|
|
|
return;
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2008-03-13 06:36:53 +00:00
|
|
|
if( HighLightStruct == Struct )
|
|
|
|
Color = HIGHLIGHT_COLOR;
|
2007-05-06 16:03:28 +00:00
|
|
|
|
2008-03-13 06:36:53 +00:00
|
|
|
Struct->Draw( panel, DC, wxPoint( 0, 0 ), DrawMode, Color );
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-03-13 06:36:53 +00:00
|
|
|
|
2007-05-06 16:03:28 +00:00
|
|
|
/* Routine de redessin en mode fantome (Dessin simplifie en g_XorMode et
|
2009-01-02 17:07:50 +00:00
|
|
|
* g_GhostColor
|
|
|
|
* de structures.
|
|
|
|
* Utilisee dans les deplacements de blocs
|
2008-03-13 06:36:53 +00:00
|
|
|
*/
|
2009-04-05 20:49:15 +00:00
|
|
|
void DrawStructsInGhost( WinEDA_DrawPanel* panel, wxDC* DC,
|
|
|
|
SCH_ITEM* DrawStruct, int dx, int dy )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2008-03-13 06:36:53 +00:00
|
|
|
int DrawMode = g_XorMode;
|
|
|
|
int width = g_DrawMinimunLineWidth;
|
|
|
|
|
|
|
|
|
|
|
|
GRSetDrawMode( DC, DrawMode );
|
|
|
|
|
|
|
|
switch( DrawStruct->Type() )
|
|
|
|
{
|
|
|
|
case DRAW_POLYLINE_STRUCT_TYPE:
|
|
|
|
{
|
2009-01-02 17:07:50 +00:00
|
|
|
DrawPolylineStruct* Struct = (DrawPolylineStruct*) DrawStruct;
|
2009-04-05 20:49:15 +00:00
|
|
|
GRMoveTo( Struct->m_PolyPoints[0].x + dx,
|
|
|
|
Struct->m_PolyPoints[0].y + dy );
|
2009-01-02 17:07:50 +00:00
|
|
|
for( unsigned ii = 1; ii < Struct->GetCornerCount(); ii++ )
|
|
|
|
GRLineTo( &panel->m_ClipBox, DC, Struct->m_PolyPoints[ii].x + dx,
|
|
|
|
Struct->m_PolyPoints[ii].y + dy, width, g_GhostColor );
|
2008-03-13 06:36:53 +00:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case DRAW_SEGMENT_STRUCT_TYPE:
|
|
|
|
{
|
|
|
|
EDA_DrawLineStruct* Struct;
|
|
|
|
Struct = (EDA_DrawLineStruct*) DrawStruct;
|
|
|
|
if( (Struct->m_Flags & STARTPOINT) == 0 )
|
|
|
|
{
|
|
|
|
GRMoveTo( Struct->m_Start.x + dx, Struct->m_Start.y + dy );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
GRMoveTo( Struct->m_Start.x, Struct->m_Start.y );
|
|
|
|
}
|
|
|
|
if( (Struct->m_Flags & ENDPOINT) == 0 )
|
|
|
|
{
|
2009-04-05 20:49:15 +00:00
|
|
|
GRLineTo( &panel->m_ClipBox, DC, Struct->m_End.x + dx,
|
|
|
|
Struct->m_End.y + dy, width, g_GhostColor );
|
2008-03-13 06:36:53 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-04-05 20:49:15 +00:00
|
|
|
GRLineTo( &panel->m_ClipBox, DC, Struct->m_End.x,
|
|
|
|
Struct->m_End.y, width, g_GhostColor );
|
2008-03-13 06:36:53 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case DRAW_BUSENTRY_STRUCT_TYPE:
|
|
|
|
{
|
|
|
|
DrawBusEntryStruct* Struct = (DrawBusEntryStruct*) DrawStruct;
|
|
|
|
int xx = Struct->m_Pos.x + dx, yy = Struct->m_Pos.y + dy;
|
|
|
|
GRMoveTo( xx, yy );
|
2009-04-05 20:49:15 +00:00
|
|
|
GRLineTo( &panel->m_ClipBox, DC, Struct->m_Size.x + xx,
|
|
|
|
Struct->m_Size.y + yy, width, g_GhostColor );
|
2008-03-13 06:36:53 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case DRAW_JUNCTION_STRUCT_TYPE:
|
|
|
|
{
|
|
|
|
DrawJunctionStruct* Struct;
|
|
|
|
Struct = (DrawJunctionStruct*) DrawStruct;
|
2009-01-31 18:08:47 +00:00
|
|
|
Struct->Draw( panel, DC, wxPoint(0,0), DrawMode, g_GhostColor );
|
2008-03-13 06:36:53 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2008-03-20 01:50:21 +00:00
|
|
|
case TYPE_SCH_TEXT:
|
2008-03-13 06:36:53 +00:00
|
|
|
{
|
2008-03-20 01:50:21 +00:00
|
|
|
SCH_TEXT* Struct;
|
|
|
|
Struct = (SCH_TEXT*) DrawStruct;
|
2008-04-14 19:22:48 +00:00
|
|
|
Struct->Draw( panel, DC, wxPoint( dx, dy ), DrawMode, g_GhostColor );
|
2008-03-13 06:36:53 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2008-03-20 01:50:21 +00:00
|
|
|
case TYPE_SCH_LABEL:
|
|
|
|
case TYPE_SCH_GLOBALLABEL:
|
|
|
|
case TYPE_SCH_HIERLABEL:
|
2008-03-13 06:36:53 +00:00
|
|
|
{
|
2008-03-20 01:50:21 +00:00
|
|
|
SCH_LABEL* Struct;
|
|
|
|
Struct = (SCH_LABEL*) DrawStruct;
|
2008-04-14 19:22:48 +00:00
|
|
|
Struct->Draw( panel, DC, wxPoint( dx, dy ), DrawMode, g_GhostColor );
|
2008-03-13 06:36:53 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case DRAW_NOCONNECT_STRUCT_TYPE:
|
|
|
|
{
|
|
|
|
DrawNoConnectStruct* Struct;
|
|
|
|
Struct = (DrawNoConnectStruct*) DrawStruct;
|
2008-04-14 19:22:48 +00:00
|
|
|
Struct->Draw( panel, DC, wxPoint( dx, dy ), DrawMode, g_GhostColor );
|
2008-03-13 06:36:53 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2008-03-20 01:50:21 +00:00
|
|
|
case TYPE_SCH_COMPONENT:
|
2008-03-13 06:36:53 +00:00
|
|
|
{
|
|
|
|
EDA_LibComponentStruct* LibEntry;
|
2009-01-02 17:07:50 +00:00
|
|
|
SCH_COMPONENT* Struct;
|
2008-03-20 01:50:21 +00:00
|
|
|
Struct = (SCH_COMPONENT*) DrawStruct;
|
2009-04-05 20:49:15 +00:00
|
|
|
LibEntry = FindLibPart( Struct->m_ChipName.GetData(), wxEmptyString,
|
|
|
|
FIND_ROOT );
|
2008-03-13 06:36:53 +00:00
|
|
|
if( LibEntry == NULL )
|
|
|
|
break;
|
|
|
|
DrawingLibInGhost( panel, DC, LibEntry, Struct, Struct->m_Pos.x + dx,
|
2009-04-05 20:49:15 +00:00
|
|
|
Struct->m_Pos.y + dy, Struct->m_Multi,
|
|
|
|
Struct->m_Convert, g_GhostColor, FALSE );
|
2008-03-13 06:36:53 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case DRAW_SHEET_STRUCT_TYPE:
|
|
|
|
{
|
|
|
|
DrawSheetStruct* Struct = (DrawSheetStruct*) DrawStruct;
|
2009-04-05 20:49:15 +00:00
|
|
|
GRRect( &panel->m_ClipBox, DC, Struct->m_Pos.x + dx,
|
|
|
|
Struct->m_Pos.y + dy, Struct->m_Pos.x + Struct->m_Size.x + dx,
|
2009-01-02 17:07:50 +00:00
|
|
|
Struct->m_Pos.y + Struct->m_Size.y + dy, width, g_GhostColor );
|
2008-03-13 06:36:53 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2008-04-15 19:38:19 +00:00
|
|
|
case DRAW_HIERARCHICAL_PIN_SHEET_STRUCT_TYPE:
|
2008-03-13 06:36:53 +00:00
|
|
|
case DRAW_MARKER_STRUCT_TYPE:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-03-13 06:36:53 +00:00
|
|
|
|
2007-05-06 16:03:28 +00:00
|
|
|
/*
|
2009-01-02 17:07:50 +00:00
|
|
|
* Place un repere sur l'ecran au point de coordonnees PCB pos_X, pos_Y
|
|
|
|
* Le marqueur est defini par un tableau de 2 + (lig*col) elements:
|
|
|
|
* 1er element: dim nbre ligne
|
|
|
|
* 2er element: dim nbre col
|
|
|
|
* suite: lig * col elements a 0 ou 1 : si 1 mise a color du pixel
|
2008-03-13 06:36:53 +00:00
|
|
|
*
|
2009-01-02 17:07:50 +00:00
|
|
|
* copie la description du marqueur en current_marqueur (global)
|
2008-03-13 06:36:53 +00:00
|
|
|
*/
|
2009-04-05 20:49:15 +00:00
|
|
|
void Draw_Marqueur( WinEDA_DrawPanel* panel, wxDC* DC,
|
|
|
|
wxPoint pos, char* pt_bitmap, int DrawMode, int Color )
|
2007-05-06 16:03:28 +00:00
|
|
|
{
|
2008-03-13 06:36:53 +00:00
|
|
|
int px, py, color;
|
|
|
|
char ii, ii_max, jj, jj_max;
|
|
|
|
|
|
|
|
if( pt_bitmap == NULL )
|
|
|
|
pt_bitmap = marq_bitmap;
|
|
|
|
|
2009-04-05 20:49:15 +00:00
|
|
|
px = GRMapX( pos.x );
|
|
|
|
py = GRMapY( pos.y );
|
2008-03-13 06:36:53 +00:00
|
|
|
|
|
|
|
/* Lecture des dimensions */
|
2009-04-05 20:49:15 +00:00
|
|
|
ii_max = *(pt_bitmap++);
|
|
|
|
jj_max = *(pt_bitmap++);
|
2008-03-13 06:36:53 +00:00
|
|
|
|
|
|
|
/* lecture des offsets */
|
2009-04-05 20:49:15 +00:00
|
|
|
px += *(pt_bitmap++);
|
|
|
|
py += *(pt_bitmap++);
|
2008-03-13 06:36:53 +00:00
|
|
|
|
|
|
|
color = *(pt_bitmap++);
|
|
|
|
if( (Color > 0) )
|
|
|
|
color = Color;
|
|
|
|
if( color < 0 )
|
|
|
|
color = 0;
|
|
|
|
GRSetDrawMode( DC, DrawMode );
|
|
|
|
|
|
|
|
/* Trace du bitmap */
|
|
|
|
for( ii = 0; ii < ii_max; ii++ )
|
|
|
|
{
|
|
|
|
for( jj = 0; jj < jj_max; jj++, pt_bitmap++ )
|
|
|
|
{
|
|
|
|
if( *pt_bitmap )
|
|
|
|
GRSPutPixel( &panel->m_ClipBox, DC, px + ii, py + jj, color );
|
|
|
|
}
|
|
|
|
}
|
2007-05-06 16:03:28 +00:00
|
|
|
}
|