2007-08-20 01:20:48 +00:00
|
|
|
|
/****************/
|
|
|
|
|
/* modedit.cpp */
|
|
|
|
|
/****************/
|
2007-05-06 16:03:28 +00:00
|
|
|
|
|
|
|
|
|
#ifdef __GNUG__
|
|
|
|
|
#pragma implementation
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#include "fctsys.h"
|
|
|
|
|
#include "common.h"
|
|
|
|
|
|
|
|
|
|
#include "pcbnew.h"
|
|
|
|
|
#include "trigo.h"
|
|
|
|
|
#include "3d_viewer.h"
|
|
|
|
|
|
|
|
|
|
#include "bitmaps.h"
|
|
|
|
|
#include "protos.h"
|
|
|
|
|
#include "id.h"
|
|
|
|
|
|
2007-10-27 18:05:50 +00:00
|
|
|
|
#include "collectors.h"
|
2007-05-06 16:03:28 +00:00
|
|
|
|
|
2007-10-27 18:05:50 +00:00
|
|
|
|
/****************************************************************************/
|
|
|
|
|
BOARD_ITEM* WinEDA_ModuleEditFrame::ModeditLocateAndDisplay( int aHotKeyCode )
|
|
|
|
|
/****************************************************************************/
|
2007-05-06 16:03:28 +00:00
|
|
|
|
{
|
2007-10-27 18:05:50 +00:00
|
|
|
|
BOARD_ITEM* item = GetCurItem();
|
2007-08-20 01:20:48 +00:00
|
|
|
|
MODULE* Module = m_Pcb->m_Modules;
|
2007-05-06 16:03:28 +00:00
|
|
|
|
|
2007-08-20 01:20:48 +00:00
|
|
|
|
if( Module == NULL )
|
|
|
|
|
return NULL;
|
2007-05-06 16:03:28 +00:00
|
|
|
|
|
2007-10-27 18:05:50 +00:00
|
|
|
|
GENERAL_COLLECTORS_GUIDE guide = GetCollectorsGuide();
|
|
|
|
|
|
|
|
|
|
// Assign to scanList the proper item types desired based on tool type
|
|
|
|
|
// or hotkey that is in play.
|
|
|
|
|
|
|
|
|
|
const KICAD_T* scanList = NULL;
|
|
|
|
|
|
|
|
|
|
if( aHotKeyCode )
|
2007-08-20 01:20:48 +00:00
|
|
|
|
{
|
2007-10-27 18:05:50 +00:00
|
|
|
|
// @todo: add switch here and add calls to PcbGeneralLocateAndDisplay( int aHotKeyCode )
|
|
|
|
|
// when searching is needed from a hotkey handler
|
2007-08-20 01:20:48 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
2007-10-27 18:05:50 +00:00
|
|
|
|
{
|
|
|
|
|
scanList = GENERAL_COLLECTOR::ModulesAndTheirItems;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
m_Collector->Collect( m_Pcb, scanList, GetScreen()->RefPos( true ), guide );
|
|
|
|
|
|
|
|
|
|
/* Remove redundancies: when an item is found, we can remove the
|
2007-11-12 05:53:48 +00:00
|
|
|
|
* module from list
|
2007-10-27 18:05:50 +00:00
|
|
|
|
*/
|
2007-11-12 05:53:48 +00:00
|
|
|
|
if( m_Collector->GetCount() > 1 )
|
|
|
|
|
{
|
|
|
|
|
for( int ii = 0; ii < m_Collector->GetCount(); ii++ )
|
|
|
|
|
{
|
|
|
|
|
item = (*m_Collector)[ii];
|
|
|
|
|
if( item->Type() != TYPEMODULE )
|
|
|
|
|
continue;
|
|
|
|
|
m_Collector->Remove( ii );
|
|
|
|
|
ii--;
|
|
|
|
|
}
|
|
|
|
|
}
|
2007-10-27 18:05:50 +00:00
|
|
|
|
|
|
|
|
|
if( m_Collector->GetCount() <= 1 )
|
|
|
|
|
{
|
|
|
|
|
item = (*m_Collector)[0];
|
|
|
|
|
SetCurItem( item );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
else // we can't figure out which item user wants, do popup menu so user can choose
|
|
|
|
|
{
|
|
|
|
|
wxMenu itemMenu;
|
|
|
|
|
|
|
|
|
|
/* Give a title to the selection menu. This is also a cancel menu item */
|
|
|
|
|
wxMenuItem * item_title = new wxMenuItem(&itemMenu, -1, _( "Selection Clarification" ) );
|
|
|
|
|
#ifdef __WINDOWS__
|
|
|
|
|
wxFont bold_font(*wxNORMAL_FONT);
|
|
|
|
|
bold_font.SetWeight(wxFONTWEIGHT_BOLD);
|
|
|
|
|
bold_font.SetStyle( wxFONTSTYLE_ITALIC);
|
|
|
|
|
item_title->SetFont(bold_font);
|
|
|
|
|
#endif
|
|
|
|
|
itemMenu.Append(item_title);
|
|
|
|
|
itemMenu.AppendSeparator();
|
|
|
|
|
|
|
|
|
|
int limit = MIN( MAX_ITEMS_IN_PICKER, m_Collector->GetCount() );
|
|
|
|
|
|
|
|
|
|
for( int ii = 0; ii<limit; ++ii )
|
|
|
|
|
{
|
|
|
|
|
wxString text;
|
|
|
|
|
const char** xpm;
|
|
|
|
|
|
|
|
|
|
item = (*m_Collector)[ii];
|
|
|
|
|
|
|
|
|
|
text = item->MenuText( m_Pcb );
|
|
|
|
|
xpm = item->MenuIcon();
|
|
|
|
|
|
|
|
|
|
ADD_MENUITEM( &itemMenu, ID_POPUP_PCB_ITEM_SELECTION_START + ii, text, xpm );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// this menu's handler is void WinEDA_BasePcbFrame::ProcessItemSelection()
|
|
|
|
|
// and it calls SetCurItem() which in turn calls Display_Infos() on the item.
|
|
|
|
|
DrawPanel->m_AbortRequest = true; // changed in false if an item
|
|
|
|
|
PopupMenu( &itemMenu ); // m_AbortRequest = false if an item is selected
|
|
|
|
|
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
2007-05-06 16:03:28 +00:00
|
|
|
|
|
2007-10-27 18:05:50 +00:00
|
|
|
|
DrawPanel->m_IgnoreMouseEvents = FALSE;
|
|
|
|
|
|
|
|
|
|
// The function ProcessItemSelection() has set the current item, return it.
|
|
|
|
|
item = GetCurItem();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if( item )
|
|
|
|
|
{
|
|
|
|
|
item->Display_Infos( this );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return item;
|
2007-05-06 16:03:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2007-10-27 18:05:50 +00:00
|
|
|
|
|
2007-05-06 16:03:28 +00:00
|
|
|
|
/****************************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
|
void WinEDA_ModuleEditFrame::Process_Special_Functions( wxCommandEvent& event )
|
2007-05-06 16:03:28 +00:00
|
|
|
|
/****************************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
|
|
2007-05-06 16:03:28 +00:00
|
|
|
|
/* Traite les selections d'outils et les commandes appelees du menu POPUP
|
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 id = event.GetId();
|
|
|
|
|
wxPoint pos;
|
|
|
|
|
wxClientDC dc( DrawPanel );
|
|
|
|
|
|
|
|
|
|
DrawPanel->CursorOff( &dc );
|
|
|
|
|
DrawPanel->PrepareGraphicContext( &dc );
|
|
|
|
|
|
|
|
|
|
wxGetMousePosition( &pos.x, &pos.y );
|
|
|
|
|
|
|
|
|
|
pos.y += 20;
|
|
|
|
|
|
|
|
|
|
switch( id ) // Arret eventuel de la commande de d<>placement en cours
|
|
|
|
|
{
|
|
|
|
|
case wxID_CUT:
|
|
|
|
|
case wxID_COPY:
|
|
|
|
|
case ID_TOOLBARH_PCB_SELECT_LAYER:
|
|
|
|
|
case ID_MODEDIT_PAD_SETTINGS:
|
|
|
|
|
case ID_PCB_USER_GRID_SETUP:
|
|
|
|
|
case ID_POPUP_PCB_ROTATE_TEXTEPCB:
|
|
|
|
|
case ID_POPUP_PCB_EDIT_TEXTEPCB:
|
|
|
|
|
case ID_POPUP_PCB_ROTATE_TEXTMODULE:
|
|
|
|
|
case ID_POPUP_PCB_ROTATE_MODULE_CLOCKWISE:
|
|
|
|
|
case ID_POPUP_PCB_ROTATE_MODULE_COUNTERCLOCKWISE:
|
|
|
|
|
case ID_POPUP_PCB_EDIT_TEXTMODULE:
|
|
|
|
|
case ID_POPUP_PCB_IMPORT_PAD_SETTINGS:
|
|
|
|
|
case ID_POPUP_PCB_EXPORT_PAD_SETTINGS:
|
|
|
|
|
case ID_POPUP_PCB_GLOBAL_IMPORT_PAD_SETTINGS:
|
|
|
|
|
case ID_POPUP_PCB_STOP_CURRENT_DRAWING:
|
|
|
|
|
case ID_POPUP_PCB_EDIT_EDGE:
|
|
|
|
|
case ID_POPUP_PCB_EDIT_WIDTH_CURRENT_EDGE:
|
|
|
|
|
case ID_POPUP_PCB_EDIT_WIDTH_ALL_EDGE:
|
|
|
|
|
case ID_POPUP_PCB_EDIT_LAYER_CURRENT_EDGE:
|
|
|
|
|
case ID_POPUP_PCB_EDIT_LAYER_ALL_EDGE:
|
|
|
|
|
case ID_POPUP_PCB_ENTER_EDGE_WIDTH:
|
|
|
|
|
case ID_POPUP_DELETE_BLOCK:
|
|
|
|
|
case ID_POPUP_PLACE_BLOCK:
|
|
|
|
|
case ID_POPUP_ZOOM_BLOCK:
|
|
|
|
|
case ID_POPUP_MIRROR_Y_BLOCK:
|
|
|
|
|
case ID_POPUP_MIRROR_X_BLOCK:
|
|
|
|
|
case ID_POPUP_ROTATE_BLOCK:
|
|
|
|
|
case ID_POPUP_COPY_BLOCK:
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_CANCEL_CURRENT_COMMAND:
|
|
|
|
|
if( DrawPanel->ManageCurseur
|
|
|
|
|
&& DrawPanel->ForceCloseManageCurseur )
|
|
|
|
|
{
|
|
|
|
|
DrawPanel->ForceCloseManageCurseur( DrawPanel, &dc );
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default: // Arret dea commande de d<>placement en cours
|
|
|
|
|
if( DrawPanel->ManageCurseur
|
|
|
|
|
&& DrawPanel->ForceCloseManageCurseur )
|
|
|
|
|
{
|
|
|
|
|
DrawPanel->ForceCloseManageCurseur( DrawPanel, &dc );
|
|
|
|
|
}
|
|
|
|
|
SetToolID( 0, wxCURSOR_ARROW, wxEmptyString );
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch( id ) // Traitement des commandes
|
|
|
|
|
{
|
|
|
|
|
case ID_EXIT:
|
|
|
|
|
Close( TRUE );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_LIBEDIT_SELECT_CURRENT_LIB:
|
|
|
|
|
Select_Active_Library();
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_LIBEDIT_DELETE_PART:
|
|
|
|
|
{
|
|
|
|
|
wxString Line;
|
|
|
|
|
Line = MakeFileName( g_RealLibDirBuffer, m_CurrentLib, LibExtBuffer );
|
|
|
|
|
Delete_Module_In_Library( Line );
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_NEW_MODULE:
|
|
|
|
|
Clear_Pcb( &dc, TRUE );
|
|
|
|
|
GetScreen()->ClearUndoRedoList();
|
2007-09-12 02:14:07 +00:00
|
|
|
|
SetCurItem( NULL );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
GetScreen()->m_Curseur = wxPoint( 0, 0 );
|
|
|
|
|
Create_1_Module( &dc, wxEmptyString );
|
|
|
|
|
if( m_Pcb->m_Modules )
|
|
|
|
|
m_Pcb->m_Modules->m_Flags = 0;
|
|
|
|
|
Zoom_Automatique( TRUE );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_SAVE_LIBMODULE:
|
|
|
|
|
{
|
|
|
|
|
wxString Line;
|
|
|
|
|
Line = MakeFileName( g_RealLibDirBuffer, m_CurrentLib.GetData(), LibExtBuffer );
|
|
|
|
|
Save_1_Module( Line, m_Pcb->m_Modules, TRUE, TRUE );
|
|
|
|
|
GetScreen()->ClrModify();
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_LOAD_MODULE_FROM_BOARD:
|
|
|
|
|
GetScreen()->ClearUndoRedoList();
|
|
|
|
|
Load_Module_Module_From_BOARD( NULL );
|
|
|
|
|
GetScreen()->ClrModify();
|
|
|
|
|
if( m_Draw3DFrame )
|
|
|
|
|
m_Draw3DFrame->NewDisplay();
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_INSERT_MODULE_IN_BOARD:
|
|
|
|
|
case ID_MODEDIT_UPDATE_MODULE_IN_BOARD:
|
|
|
|
|
{
|
|
|
|
|
// update module in the current board,
|
|
|
|
|
// not just add it to the board with total disregard for the netlist...
|
|
|
|
|
WinEDA_PcbFrame* pcbframe = m_Parent->m_PcbFrame;
|
|
|
|
|
BOARD* mainpcb = pcbframe->m_Pcb;
|
|
|
|
|
MODULE* source_module = NULL;
|
|
|
|
|
MODULE* module_in_edit = m_Pcb->m_Modules;
|
|
|
|
|
|
|
|
|
|
// Search the old module (source) if exists
|
|
|
|
|
// Because this source could be deleted when editing the main board...
|
|
|
|
|
if( module_in_edit->m_Link ) // this is not a new module ...
|
|
|
|
|
{
|
|
|
|
|
source_module = mainpcb->m_Modules;
|
|
|
|
|
for( ; source_module != NULL; source_module = (MODULE*) source_module->Pnext )
|
|
|
|
|
{
|
|
|
|
|
if( module_in_edit->m_Link == source_module->m_TimeStamp )
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if( (source_module == NULL) && id == (ID_MODEDIT_UPDATE_MODULE_IN_BOARD) ) // source not found
|
|
|
|
|
{
|
|
|
|
|
wxString msg;
|
|
|
|
|
msg.Printf( _( "Unable to find the footprint source on the main board" ) );
|
|
|
|
|
msg << _( "\nCannot update the footprint" );
|
|
|
|
|
DisplayError( this, msg );
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if( (source_module != NULL) && id == (ID_MODEDIT_INSERT_MODULE_IN_BOARD) ) // source not found
|
|
|
|
|
{
|
|
|
|
|
wxString msg;
|
|
|
|
|
msg.Printf( _( "A footprint source was found on the main board" ) );
|
|
|
|
|
msg << _( "\nCannot insert this footprint" );
|
|
|
|
|
DisplayError( this, msg );
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Create the "new" module
|
|
|
|
|
MODULE* newmodule = new MODULE( mainpcb );
|
|
|
|
|
newmodule->Copy( module_in_edit );
|
|
|
|
|
newmodule->m_Parent = mainpcb; // modifie par la copie
|
|
|
|
|
newmodule->m_Link = 0;
|
|
|
|
|
|
|
|
|
|
// Put the footprint in the main pcb linked list.
|
|
|
|
|
newmodule->Pnext = mainpcb->m_Modules;
|
|
|
|
|
mainpcb->m_Modules = newmodule;
|
|
|
|
|
newmodule->Pback = mainpcb;
|
|
|
|
|
if( newmodule->Pnext )
|
|
|
|
|
newmodule->Pnext->Pback = newmodule;
|
|
|
|
|
|
|
|
|
|
if( source_module ) // this is an update command
|
|
|
|
|
{
|
|
|
|
|
// The new module replace the old module (pos, orient, ref, value and connexions are kept)
|
|
|
|
|
// and the source_module (old module) is deleted
|
|
|
|
|
newmodule = pcbframe->Exchange_Module( this, source_module, newmodule );
|
|
|
|
|
newmodule->m_TimeStamp = module_in_edit->m_Link;
|
|
|
|
|
}
|
|
|
|
|
else // This is an insert command
|
|
|
|
|
{
|
|
|
|
|
wxPoint cursor_pos = pcbframe->GetScreen()->m_Curseur;
|
|
|
|
|
pcbframe->GetScreen()->m_Curseur = wxPoint( 0, 0 );
|
|
|
|
|
pcbframe->Place_Module( newmodule, NULL );
|
|
|
|
|
pcbframe->GetScreen()->m_Curseur = cursor_pos;
|
|
|
|
|
newmodule->m_TimeStamp = GetTimeStamp();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
newmodule->m_Flags = 0;
|
|
|
|
|
GetScreen()->ClrModify();
|
2007-09-12 02:14:07 +00:00
|
|
|
|
pcbframe->SetCurItem( NULL );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
mainpcb->m_Status_Pcb = 0;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_LIBEDIT_IMPORT_PART:
|
|
|
|
|
GetScreen()->ClearUndoRedoList();
|
2007-09-12 02:14:07 +00:00
|
|
|
|
SetCurItem( NULL );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
Clear_Pcb( &dc, TRUE );
|
|
|
|
|
GetScreen()->m_Curseur = wxPoint( 0, 0 );
|
|
|
|
|
Import_Module( &dc );
|
|
|
|
|
if( m_Pcb->m_Modules )
|
|
|
|
|
m_Pcb->m_Modules->m_Flags = 0;
|
|
|
|
|
GetScreen()->ClrModify();
|
|
|
|
|
Zoom_Automatique( TRUE );
|
|
|
|
|
if( m_Draw3DFrame )
|
|
|
|
|
m_Draw3DFrame->NewDisplay();
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_LIBEDIT_EXPORT_PART:
|
|
|
|
|
if( m_Pcb->m_Modules )
|
|
|
|
|
Export_Module( m_Pcb->m_Modules, FALSE );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_LIBEDIT_CREATE_NEW_LIB_AND_SAVE_CURRENT_PART:
|
|
|
|
|
if( m_Pcb->m_Modules )
|
|
|
|
|
Export_Module( m_Pcb->m_Modules, TRUE );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_SHEET_SET:
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_LOAD_MODULE:
|
|
|
|
|
{
|
|
|
|
|
GetScreen()->ClearUndoRedoList();
|
2007-09-12 02:14:07 +00:00
|
|
|
|
SetCurItem( NULL );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
Clear_Pcb( &dc, TRUE );
|
|
|
|
|
GetScreen()->m_Curseur = wxPoint( 0, 0 );
|
|
|
|
|
Load_Module_From_Library( m_CurrentLib, &dc );
|
|
|
|
|
if( m_Pcb->m_Modules )
|
|
|
|
|
m_Pcb->m_Modules->m_Flags = 0;
|
|
|
|
|
|
|
|
|
|
//if either m_Reference or m_Value are gone, reinstate them -
|
|
|
|
|
//otherwise it becomes hard to see what you are working with in the layout!
|
|
|
|
|
if( m_Pcb && m_Pcb->m_Modules )
|
|
|
|
|
{
|
|
|
|
|
TEXTE_MODULE* ref = m_Pcb->m_Modules->m_Reference;
|
|
|
|
|
TEXTE_MODULE* val = m_Pcb->m_Modules->m_Value;
|
|
|
|
|
if( val && ref )
|
|
|
|
|
{
|
|
|
|
|
ref->m_NoShow = 0;
|
|
|
|
|
val->m_NoShow = 0;
|
|
|
|
|
ref->m_Type = 0;
|
|
|
|
|
val->m_Type = 1;
|
|
|
|
|
if( ref->m_Text.Length() == 0 )
|
|
|
|
|
ref->m_Text = L"Ref**";
|
|
|
|
|
if( val->m_Text.Length() == 0 )
|
|
|
|
|
val->m_Text = L"Val**";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
GetScreen()->ClrModify();
|
|
|
|
|
Zoom_Automatique( TRUE );
|
|
|
|
|
if( m_Draw3DFrame )
|
|
|
|
|
m_Draw3DFrame->NewDisplay();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_PAD_SETTINGS:
|
|
|
|
|
InstallPadOptionsFrame( NULL, NULL, wxPoint( -1, -1 ) );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_CHECK:
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_EDIT_MODULE_PROPERTIES:
|
|
|
|
|
if( m_Pcb->m_Modules )
|
|
|
|
|
{
|
2007-09-12 02:14:07 +00:00
|
|
|
|
SetCurItem( m_Pcb->m_Modules );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
InstallModuleOptionsFrame( (MODULE*) GetScreen()->GetCurItem(),
|
|
|
|
|
&dc, pos );
|
|
|
|
|
GetScreen()->GetCurItem()->m_Flags = 0;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_ADD_PAD:
|
|
|
|
|
if( m_Pcb->m_Modules )
|
|
|
|
|
SetToolID( id, wxCURSOR_PENCIL, _( "Add Pad" ) );
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
SetToolID( id, wxCURSOR_ARROW, _( "Pad Settings" ) );
|
|
|
|
|
InstallPadOptionsFrame( NULL, NULL, wxPoint( -1, -1 ) );
|
|
|
|
|
SetToolID( 0, wxCURSOR_ARROW, wxEmptyString );
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_LINE_COMMENT_BUTT:
|
|
|
|
|
case ID_PCB_ARC_BUTT:
|
|
|
|
|
case ID_PCB_CIRCLE_BUTT:
|
|
|
|
|
case ID_TEXT_COMMENT_BUTT:
|
|
|
|
|
SetToolID( id, wxCURSOR_PENCIL, _( "Add Drawing" ) );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_PLACE_ANCHOR:
|
|
|
|
|
SetToolID( id, wxCURSOR_PENCIL, _( "Place anchor" ) );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_NO_SELECT_BUTT:
|
|
|
|
|
SetToolID( 0, wxCURSOR_ARROW, wxEmptyString );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_CLOSE_CURRENT_TOOL:
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_CANCEL_CURRENT_COMMAND:
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_DELETE_ITEM_BUTT:
|
|
|
|
|
SetToolID( id, wxCURSOR_BULLSEYE, _( "Delete item" ) );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_ROTATE_MODULE_COUNTERCLOCKWISE:
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
Rotate_Module( &dc, (MODULE*) GetScreen()->GetCurItem(), 900, TRUE );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_ROTATE_MODULE_CLOCKWISE:
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
Rotate_Module( &dc, (MODULE*) GetScreen()->GetCurItem(), -900, TRUE );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_EDIT_MODULE:
|
|
|
|
|
InstallModuleOptionsFrame( (MODULE*) GetScreen()->GetCurItem(),
|
|
|
|
|
&dc, pos );
|
|
|
|
|
GetScreen()->GetCurItem()->m_Flags = 0;
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_MOVE_PAD_REQUEST:
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
StartMovePad( (D_PAD*) GetScreen()->GetCurItem(), &dc );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_EDIT_PAD:
|
|
|
|
|
InstallPadOptionsFrame( (D_PAD*) GetScreen()->GetCurItem(),
|
|
|
|
|
&dc, pos );
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_DELETE_PAD:
|
|
|
|
|
SaveCopyInUndoList( m_Pcb->m_Modules );
|
|
|
|
|
DeletePad( (D_PAD*) GetScreen()->GetCurItem(), &dc );
|
2007-09-12 02:14:07 +00:00
|
|
|
|
SetCurItem( NULL );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_IMPORT_PAD_SETTINGS:
|
|
|
|
|
SaveCopyInUndoList( m_Pcb->m_Modules );
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
Import_Pad_Settings( (D_PAD*) GetScreen()->GetCurItem(), &dc );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_GLOBAL_IMPORT_PAD_SETTINGS:
|
|
|
|
|
SaveCopyInUndoList( m_Pcb->m_Modules );
|
|
|
|
|
Global_Import_Pad_Settings( (D_PAD*) GetScreen()->GetCurItem(), &dc );
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_EXPORT_PAD_SETTINGS:
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
Export_Pad_Settings( (D_PAD*) GetScreen()->GetCurItem() );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_EDIT_TEXTMODULE:
|
|
|
|
|
InstallTextModOptionsFrame( (TEXTE_MODULE*) GetScreen()->GetCurItem(),
|
|
|
|
|
&dc, pos );
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_MOVE_TEXTMODULE_REQUEST:
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
StartMoveTexteModule( (TEXTE_MODULE*) GetScreen()->GetCurItem(),
|
|
|
|
|
&dc );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_ROTATE_TEXTMODULE:
|
|
|
|
|
RotateTextModule( (TEXTE_MODULE*) GetScreen()->GetCurItem(),
|
|
|
|
|
&dc );
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_DELETE_TEXTMODULE:
|
|
|
|
|
SaveCopyInUndoList( m_Pcb->m_Modules );
|
|
|
|
|
DeleteTextModule( (TEXTE_MODULE*) GetScreen()->GetCurItem(),
|
|
|
|
|
&dc );
|
2007-09-12 02:14:07 +00:00
|
|
|
|
SetCurItem( NULL );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_MOVE_EDGE:
|
|
|
|
|
Start_Move_EdgeMod( (EDGE_MODULE*) GetScreen()->GetCurItem(), &dc );
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_STOP_CURRENT_DRAWING:
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
if( (GetScreen()->GetCurItem()->m_Flags & IS_NEW) )
|
|
|
|
|
{
|
|
|
|
|
End_Edge_Module( (EDGE_MODULE*) GetScreen()->GetCurItem(), &dc );
|
2007-09-12 02:14:07 +00:00
|
|
|
|
SetCurItem( NULL );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_ENTER_EDGE_WIDTH:
|
|
|
|
|
{
|
|
|
|
|
EDGE_MODULE* edge = NULL;
|
|
|
|
|
if( GetScreen()->GetCurItem()
|
|
|
|
|
&& ( GetScreen()->GetCurItem()->m_Flags & IS_NEW)
|
2007-09-01 12:00:30 +00:00
|
|
|
|
&& (GetScreen()->GetCurItem()->Type() == TYPEEDGEMODULE) )
|
2007-08-20 01:20:48 +00:00
|
|
|
|
{
|
|
|
|
|
edge = (EDGE_MODULE*) GetScreen()->GetCurItem();
|
|
|
|
|
}
|
|
|
|
|
Enter_Edge_Width( edge, &dc );
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_EDIT_WIDTH_CURRENT_EDGE:
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
Edit_Edge_Width( (EDGE_MODULE*) GetScreen()->GetCurItem(), &dc );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_EDIT_WIDTH_ALL_EDGE:
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
Edit_Edge_Width( NULL, &dc );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_EDIT_LAYER_CURRENT_EDGE:
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
Edit_Edge_Layer( (EDGE_MODULE*) GetScreen()->GetCurItem(), &dc );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_EDIT_LAYER_ALL_EDGE:
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
Edit_Edge_Layer( NULL, &dc );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PCB_DELETE_EDGE:
|
|
|
|
|
SaveCopyInUndoList( m_Pcb->m_Modules );
|
|
|
|
|
DrawPanel->MouseToCursorSchema();
|
|
|
|
|
RemoveStruct( GetScreen()->GetCurItem(), &dc );
|
2007-09-12 02:14:07 +00:00
|
|
|
|
SetCurItem( NULL );
|
2007-08-20 01:20:48 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_MODULE_ROTATE:
|
|
|
|
|
case ID_MODEDIT_MODULE_MIRROR:
|
|
|
|
|
case ID_MODEDIT_MODULE_SCALE:
|
|
|
|
|
case ID_MODEDIT_MODULE_SCALEX:
|
|
|
|
|
case ID_MODEDIT_MODULE_SCALEY:
|
|
|
|
|
SaveCopyInUndoList( m_Pcb->m_Modules );
|
|
|
|
|
Transform( (MODULE*) GetScreen()->GetCurItem(), &dc, id );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_PCB_DRAWINGS_WIDTHS_SETUP:
|
|
|
|
|
InstallOptionsFrame( pos );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_PCB_PAD_SETUP:
|
2007-11-12 05:53:48 +00:00
|
|
|
|
{
|
|
|
|
|
BOARD_ITEM* item = GetCurItem();
|
2007-11-12 06:00:42 +00:00
|
|
|
|
if( item )
|
|
|
|
|
{
|
|
|
|
|
if( item->Type() != TYPEPAD )
|
|
|
|
|
item = NULL;
|
|
|
|
|
}
|
|
|
|
|
InstallPadOptionsFrame( (D_PAD*) item, &dc, pos );
|
2007-11-12 05:53:48 +00:00
|
|
|
|
}
|
2007-08-20 01:20:48 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_PCB_USER_GRID_SETUP:
|
|
|
|
|
InstallGridFrame( pos );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_UNDO:
|
|
|
|
|
GetComponentFromUndoList();
|
|
|
|
|
DrawPanel->Refresh( TRUE );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_REDO:
|
|
|
|
|
GetComponentFromRedoList();
|
|
|
|
|
DrawPanel->Refresh( TRUE );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_PLACE_BLOCK:
|
|
|
|
|
GetScreen()->BlockLocate.m_Command = BLOCK_MOVE;
|
|
|
|
|
DrawPanel->m_AutoPAN_Request = FALSE;
|
|
|
|
|
HandleBlockPlace( &dc );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_COPY_BLOCK:
|
|
|
|
|
GetScreen()->BlockLocate.m_Command = BLOCK_COPY;
|
|
|
|
|
m_CurrentScreen->BlockLocate.SetMessageBlock( this );
|
|
|
|
|
DrawPanel->m_AutoPAN_Request = FALSE;
|
|
|
|
|
HandleBlockPlace( &dc );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_ZOOM_BLOCK:
|
|
|
|
|
GetScreen()->BlockLocate.m_Command = BLOCK_ZOOM;
|
|
|
|
|
m_CurrentScreen->BlockLocate.SetMessageBlock( this );
|
|
|
|
|
m_CurrentScreen->BlockLocate.SetMessageBlock( this );
|
|
|
|
|
HandleBlockEnd( &dc );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_DELETE_BLOCK:
|
|
|
|
|
GetScreen()->BlockLocate.m_Command = BLOCK_DELETE;
|
|
|
|
|
m_CurrentScreen->BlockLocate.SetMessageBlock( this );
|
|
|
|
|
HandleBlockEnd( &dc );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_ROTATE_BLOCK:
|
|
|
|
|
GetScreen()->BlockLocate.m_Command = BLOCK_ROTATE;
|
|
|
|
|
m_CurrentScreen->BlockLocate.SetMessageBlock( this );
|
|
|
|
|
HandleBlockEnd( &dc );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_POPUP_MIRROR_X_BLOCK:
|
|
|
|
|
case ID_POPUP_MIRROR_Y_BLOCK:
|
|
|
|
|
case ID_POPUP_INVERT_BLOCK:
|
|
|
|
|
GetScreen()->BlockLocate.m_Command = BLOCK_INVERT;
|
|
|
|
|
m_CurrentScreen->BlockLocate.SetMessageBlock( this );
|
|
|
|
|
HandleBlockEnd( &dc );
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
DisplayError( this,
|
|
|
|
|
wxT( "WinEDA_ModuleEditFrame::Process_Special_Functions error" ) );
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
SetToolbars();
|
|
|
|
|
DrawPanel->CursorOn( &dc );
|
2007-05-06 16:03:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/******************************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
|
void WinEDA_ModuleEditFrame::Transform( MODULE* module, wxDC* DC, int transform )
|
2007-05-06 16:03:28 +00:00
|
|
|
|
/******************************************************************************/
|
2007-08-20 01:20:48 +00:00
|
|
|
|
|
2007-06-10 14:27:07 +00:00
|
|
|
|
/* Execute les transformations de la repr<70>sentation des modules.
|
2007-08-20 01:20:48 +00:00
|
|
|
|
* le module, apres transformation est toujours en position de reference:
|
|
|
|
|
* position 0,0
|
|
|
|
|
* orientation 0, cot<EFBFBD> composant.
|
|
|
|
|
*/
|
2007-05-06 16:03:28 +00:00
|
|
|
|
{
|
2007-08-20 01:20:48 +00:00
|
|
|
|
D_PAD* pad = module->m_Pads;
|
|
|
|
|
EDA_BaseStruct* PtStruct = module->m_Drawings;
|
|
|
|
|
TEXTE_MODULE* textmod;
|
|
|
|
|
EDGE_MODULE* edgemod;
|
|
|
|
|
int angle = 900; // NECESSAIREMENT +- 900 (+- 90 degres) )
|
|
|
|
|
|
|
|
|
|
switch( transform )
|
|
|
|
|
{
|
|
|
|
|
case ID_MODEDIT_MODULE_ROTATE:
|
|
|
|
|
module->SetOrientation( angle );
|
|
|
|
|
|
|
|
|
|
for( ; pad != NULL; pad = (D_PAD*) pad->Pnext )
|
|
|
|
|
{
|
|
|
|
|
pad->m_Pos0 = pad->m_Pos;
|
|
|
|
|
pad->m_Orient -= angle;
|
|
|
|
|
RotatePoint( &pad->m_Offset.x, &pad->m_Offset.y, angle );
|
|
|
|
|
EXCHG( pad->m_Size.x, pad->m_Size.y );
|
|
|
|
|
RotatePoint( &pad->m_DeltaSize.x, &pad->m_DeltaSize.y, -angle );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
module->m_Reference->m_Pos0 = module->m_Reference->m_Pos;
|
|
|
|
|
module->m_Reference->m_Orient += angle;
|
|
|
|
|
if( module->m_Reference->m_Orient >= 1800 )
|
|
|
|
|
module->m_Reference->m_Orient -= 1800;
|
|
|
|
|
module->m_Value->m_Pos0 = module->m_Value->m_Pos;
|
|
|
|
|
module->m_Value->m_Orient += angle;
|
|
|
|
|
if( module->m_Value->m_Orient >= 1800 )
|
|
|
|
|
module->m_Value->m_Orient -= 1800;
|
|
|
|
|
|
|
|
|
|
/* Rectification des contours et textes de l'empreinte : */
|
|
|
|
|
for( ; PtStruct != NULL; PtStruct = PtStruct->Pnext )
|
|
|
|
|
{
|
2007-09-01 12:00:30 +00:00
|
|
|
|
if( PtStruct->Type() == TYPEEDGEMODULE )
|
2007-08-20 01:20:48 +00:00
|
|
|
|
{
|
|
|
|
|
edgemod = (EDGE_MODULE*) PtStruct;
|
|
|
|
|
edgemod->m_Start0 = edgemod->m_Start;
|
|
|
|
|
edgemod->m_End0 = edgemod->m_End;
|
|
|
|
|
}
|
2007-09-01 12:00:30 +00:00
|
|
|
|
if( PtStruct->Type() == TYPETEXTEMODULE )
|
2007-08-20 01:20:48 +00:00
|
|
|
|
{
|
|
|
|
|
/* deplacement des inscriptions : */
|
|
|
|
|
textmod = (TEXTE_MODULE*) PtStruct;
|
|
|
|
|
textmod->m_Pos0 = textmod->m_Pos;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
module->m_Orient = 0;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_MODULE_MIRROR:
|
|
|
|
|
for( ; pad != NULL; pad = (D_PAD*) pad->Pnext )
|
|
|
|
|
{
|
|
|
|
|
pad->m_Pos.y = -pad->m_Pos.y;
|
|
|
|
|
pad->m_Pos0.y = -pad->m_Pos0.y;
|
|
|
|
|
pad->m_Offset.y = -pad->m_Offset.y;
|
|
|
|
|
pad->m_DeltaSize.y = -pad->m_DeltaSize.y;
|
|
|
|
|
if( pad->m_Orient )
|
|
|
|
|
pad->m_Orient = 3600 - pad->m_Orient;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Inversion miroir de la Reference */
|
|
|
|
|
textmod = module->m_Reference;
|
|
|
|
|
textmod->m_Pos.y = -textmod->m_Pos.y;
|
|
|
|
|
textmod->m_Pos0.y = textmod->m_Pos0.y;
|
|
|
|
|
if( textmod->m_Orient )
|
|
|
|
|
textmod->m_Orient = 3600 - textmod->m_Orient;
|
|
|
|
|
|
|
|
|
|
/* Inversion miroir de la Valeur */
|
|
|
|
|
textmod = module->m_Value;
|
|
|
|
|
textmod->m_Pos.y = -textmod->m_Pos.y;
|
|
|
|
|
textmod->m_Pos0.y = textmod->m_Pos0.y;
|
|
|
|
|
if( textmod->m_Orient )
|
|
|
|
|
textmod->m_Orient = 3600 - textmod->m_Orient;
|
|
|
|
|
|
|
|
|
|
/* Inversion miroir des dessins de l'empreinte : */
|
|
|
|
|
PtStruct = module->m_Drawings;
|
|
|
|
|
for( ; PtStruct != NULL; PtStruct = PtStruct->Pnext )
|
|
|
|
|
{
|
2007-09-01 12:00:30 +00:00
|
|
|
|
switch( PtStruct->Type() )
|
2007-08-20 01:20:48 +00:00
|
|
|
|
{
|
|
|
|
|
case TYPEEDGEMODULE:
|
|
|
|
|
edgemod = (EDGE_MODULE*) PtStruct;
|
|
|
|
|
edgemod->m_Start.y = -edgemod->m_Start.y;
|
|
|
|
|
edgemod->m_End.y = -edgemod->m_End.y;
|
|
|
|
|
/* inversion des coords locales */
|
|
|
|
|
edgemod->m_Start0.y = -edgemod->m_Start0.y;
|
|
|
|
|
edgemod->m_End0.y = -edgemod->m_End0.y;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case TYPETEXTEMODULE:
|
|
|
|
|
/* Inversion miroir de la position et mise en miroir : */
|
|
|
|
|
textmod = (TEXTE_MODULE*) PtStruct;
|
|
|
|
|
textmod->m_Pos.y = -textmod->m_Pos.y;
|
|
|
|
|
textmod->m_Pos0.y = textmod->m_Pos0.y;
|
|
|
|
|
if( textmod->m_Orient )
|
|
|
|
|
textmod->m_Orient = 3600 - textmod->m_Orient;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
DisplayError( this, wxT( "Type Draw Indefini" ) );
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ID_MODEDIT_MODULE_SCALE:
|
|
|
|
|
case ID_MODEDIT_MODULE_SCALEX:
|
|
|
|
|
case ID_MODEDIT_MODULE_SCALEY:
|
|
|
|
|
DisplayInfo( this, wxT( "Not availlable" ) );
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
module->Set_Rectangle_Encadrement();
|
|
|
|
|
DrawPanel->ReDraw( DC );
|
2007-05-06 16:03:28 +00:00
|
|
|
|
}
|