libeditframe.cpp 37.9 KB
Newer Older
1 2 3
/*
 * This program source code file is part of KiCad, a free EDA CAD application.
 *
4 5 6
 * Copyright (C) 2013 Jean-Pierre Charras, jp.charras at wanadoo.fr
 * Copyright (C) 2008-2013 Wayne Stambaugh <stambaughw@verizon.net>
 * Copyright (C) 2004-2013 KiCad Developers, see change_log.txt for contributors.
7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, you may find one here:
 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
 * or you may search the http://www.gnu.org website for the version 2 license,
 * or you may write to the Free Software Foundation, Inc.,
 * 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA
 */

charras's avatar
charras committed
26
/**
27 28 29
 * @file libeditframe.cpp
 * @brief LIB_EDIT_FRAME class is the component library editor frame.
 */
plyatov's avatar
plyatov committed
30

31
#include <fctsys.h>
32 33
#include <pgm_base.h>
#include <kiface_i.h>
34 35 36 37
#include <class_drawpanel.h>
#include <confirm.h>
#include <eda_doc.h>
#include <gr_basic.h>
38
#include <schframe.h>
39
#include <msgpanel.h>
40

41 42 43 44 45
#include <general.h>
#include <eeschema_id.h>
#include <libeditframe.h>
#include <class_library.h>
#include <lib_polyline.h>
46
#include <lib_pin.h>
47

48 49
#include <kicad_device_context.h>
#include <hotkeys.h>
charras's avatar
charras committed
50

51 52 53
#include <dialogs/dialog_lib_edit_text.h>
#include <dialogs/dialog_edit_component_in_lib.h>
#include <dialogs/dialog_libedit_dimensions.h>
54

55
#include <menus_helpers.h>
56

57
#include <boost/foreach.hpp>
plyatov's avatar
plyatov committed
58

59

60
/* This method guarantees unique IDs for the library this run of Eeschema
61 62 63 64 65 66 67
 * which prevents ID conflicts and eliminates the need to recompile every
 * source file in the project when adding IDs to include/id.h. */
int ExportPartId = ::wxNewId();
int ImportPartId = ::wxNewId();
int CreateNewLibAndSavePartId = ::wxNewId();


68 69 70
wxString LIB_EDIT_FRAME::      m_aliasName;
int LIB_EDIT_FRAME::           m_unit    = 1;
int LIB_EDIT_FRAME::           m_convert = 1;
71 72
LIB_ITEM* LIB_EDIT_FRAME::m_lastDrawItem = NULL;
LIB_ITEM* LIB_EDIT_FRAME::m_drawItem = NULL;
73 74
bool LIB_EDIT_FRAME::          m_showDeMorgan    = false;
wxSize LIB_EDIT_FRAME::        m_clientSize      = wxSize( -1, -1 );
75
int LIB_EDIT_FRAME::           m_textSize        = -1;
76 77
int LIB_EDIT_FRAME::           m_textOrientation = TEXT_ORIENT_HORIZ;
int LIB_EDIT_FRAME::           m_drawLineWidth   = 0;
78 79 80 81 82 83

// these values are overridden when reading the config
int LIB_EDIT_FRAME::           m_textPinNumDefaultSize = DEFAULTPINNUMSIZE;
int LIB_EDIT_FRAME::           m_textPinNameDefaultSize = DEFAULTPINNAMESIZE;
int LIB_EDIT_FRAME::           m_defaultPinLength = DEFAULTPINLENGTH;

84 85 86
FILL_T LIB_EDIT_FRAME::        m_drawFillStyle   = NO_FILL;


87
BEGIN_EVENT_TABLE( LIB_EDIT_FRAME, EDA_DRAW_FRAME )
88 89 90
    EVT_CLOSE( LIB_EDIT_FRAME::OnCloseWindow )
    EVT_SIZE( LIB_EDIT_FRAME::OnSize )
    EVT_ACTIVATE( LIB_EDIT_FRAME::OnActivate )
91

92
    // Main horizontal toolbar.
93
    EVT_TOOL( ID_LIBEDIT_SAVE_CURRENT_LIB, LIB_EDIT_FRAME::OnSaveActiveLibrary )
94 95
    EVT_TOOL( ID_LIBEDIT_SELECT_CURRENT_LIB, LIB_EDIT_FRAME::Process_Special_Functions )
    EVT_TOOL( ID_LIBEDIT_DELETE_PART, LIB_EDIT_FRAME::DeleteOnePart )
96
    EVT_TOOL( ID_TO_LIBVIEW, LIB_EDIT_FRAME::OnOpenLibraryViewer )
97
    EVT_TOOL( ID_LIBEDIT_NEW_PART, LIB_EDIT_FRAME::CreateNewLibraryPart )
98 99
    EVT_TOOL( ID_LIBEDIT_NEW_PART_FROM_EXISTING, LIB_EDIT_FRAME::OnCreateNewPartFromExisting )

100 101 102 103 104 105 106 107 108 109 110 111 112 113 114
    EVT_TOOL( ID_LIBEDIT_SELECT_PART, LIB_EDIT_FRAME::LoadOneLibraryPart )
    EVT_TOOL( ID_LIBEDIT_SAVE_CURRENT_PART, LIB_EDIT_FRAME::Process_Special_Functions )
    EVT_TOOL( wxID_UNDO, LIB_EDIT_FRAME::GetComponentFromUndoList )
    EVT_TOOL( wxID_REDO, LIB_EDIT_FRAME::GetComponentFromRedoList )
    EVT_TOOL( ID_LIBEDIT_GET_FRAME_EDIT_PART, LIB_EDIT_FRAME::OnEditComponentProperties )
    EVT_TOOL( ID_LIBEDIT_GET_FRAME_EDIT_FIELDS, LIB_EDIT_FRAME::InstallFieldsEditorDialog )
    EVT_TOOL( ID_LIBEDIT_CHECK_PART, LIB_EDIT_FRAME::OnCheckComponent )
    EVT_TOOL( ID_DE_MORGAN_NORMAL_BUTT, LIB_EDIT_FRAME::OnSelectBodyStyle )
    EVT_TOOL( ID_DE_MORGAN_CONVERT_BUTT, LIB_EDIT_FRAME::OnSelectBodyStyle )
    EVT_TOOL( ID_LIBEDIT_VIEW_DOC, LIB_EDIT_FRAME::OnViewEntryDoc )
    EVT_TOOL( ID_LIBEDIT_EDIT_PIN_BY_PIN, LIB_EDIT_FRAME::Process_Special_Functions )
    EVT_TOOL( ExportPartId, LIB_EDIT_FRAME::OnExportPart )
    EVT_TOOL( CreateNewLibAndSavePartId, LIB_EDIT_FRAME::OnExportPart )
    EVT_TOOL( ImportPartId, LIB_EDIT_FRAME::OnImportPart )

115 116
    EVT_COMBOBOX( ID_LIBEDIT_SELECT_PART_NUMBER, LIB_EDIT_FRAME::OnSelectPart )
    EVT_COMBOBOX( ID_LIBEDIT_SELECT_ALIAS, LIB_EDIT_FRAME::OnSelectAlias )
117

118
    // Right vertical toolbar.
119 120
    EVT_TOOL( ID_NO_TOOL_SELECTED, LIB_EDIT_FRAME::OnSelectTool )
    EVT_TOOL_RANGE( ID_LIBEDIT_PIN_BUTT, ID_LIBEDIT_DELETE_ITEM_BUTT,
121
                    LIB_EDIT_FRAME::OnSelectTool )
122

123
    // menubar commands
124
    EVT_MENU( wxID_EXIT, LIB_EDIT_FRAME::CloseWindow )
125
    EVT_MENU( ID_LIBEDIT_SAVE_CURRENT_LIB_AS, LIB_EDIT_FRAME::OnSaveActiveLibrary )
126 127
    EVT_MENU( ID_LIBEDIT_GEN_PNG_FILE, LIB_EDIT_FRAME::OnPlotCurrentComponent )
    EVT_MENU( ID_LIBEDIT_GEN_SVG_FILE, LIB_EDIT_FRAME::OnPlotCurrentComponent )
128
    EVT_MENU( wxID_HELP, EDA_DRAW_FRAME::GetKicadHelp )
129
    EVT_MENU( wxID_INDEX, EDA_DRAW_FRAME::GetKicadHelp )
130
    EVT_MENU( wxID_ABOUT, EDA_BASE_FRAME::GetKicadAbout )
131

132
    EVT_MENU( ID_COLORS_SETUP, LIB_EDIT_FRAME::OnColorConfig )
133
    EVT_MENU( wxID_PREFERENCES, LIB_EDIT_FRAME::OnPreferencesOptions )
134 135 136
    EVT_MENU( ID_CONFIG_REQ, LIB_EDIT_FRAME::InstallConfigFrame )
    EVT_MENU( ID_COLORS_SETUP, LIB_EDIT_FRAME::Process_Config )
    EVT_MENU( ID_LIBEDIT_DIMENSIONS, LIB_EDIT_FRAME::InstallDimensionsDialog )
137

138
    // Multiple item selection context menu commands.
139 140
    EVT_MENU_RANGE( ID_SELECT_ITEM_START, ID_SELECT_ITEM_END, LIB_EDIT_FRAME::OnSelectItem )

141
    EVT_MENU_RANGE( ID_PREFERENCES_HOTKEY_START, ID_PREFERENCES_HOTKEY_END,
142
                    LIB_EDIT_FRAME::Process_Config )
143

144
    // Context menu events and commands.
145
    EVT_MENU( ID_LIBEDIT_EDIT_PIN, LIB_EDIT_FRAME::OnEditPin )
146
    EVT_MENU( ID_LIBEDIT_ROTATE_ITEM, LIB_EDIT_FRAME::OnRotateItem )
147 148

    EVT_MENU_RANGE( ID_POPUP_LIBEDIT_PIN_GLOBAL_CHANGE_ITEM,
149
                    ID_POPUP_LIBEDIT_DELETE_CURRENT_POLY_SEGMENT,
150
                    LIB_EDIT_FRAME::Process_Special_Functions )
151 152

    EVT_MENU_RANGE( ID_POPUP_GENERAL_START_RANGE, ID_POPUP_GENERAL_END_RANGE,
153
                    LIB_EDIT_FRAME::Process_Special_Functions )
154

155
    // Update user interface elements.
156 157 158 159 160 161
    EVT_UPDATE_UI( ExportPartId, LIB_EDIT_FRAME::OnUpdateEditingPart )
    EVT_UPDATE_UI( CreateNewLibAndSavePartId, LIB_EDIT_FRAME::OnUpdateEditingPart )
    EVT_UPDATE_UI( ID_LIBEDIT_SAVE_CURRENT_PART, LIB_EDIT_FRAME::OnUpdateEditingPart )
    EVT_UPDATE_UI( ID_LIBEDIT_GET_FRAME_EDIT_FIELDS, LIB_EDIT_FRAME::OnUpdateEditingPart )
    EVT_UPDATE_UI( ID_LIBEDIT_CHECK_PART, LIB_EDIT_FRAME::OnUpdateEditingPart )
    EVT_UPDATE_UI( ID_LIBEDIT_GET_FRAME_EDIT_PART, LIB_EDIT_FRAME::OnUpdateEditingPart )
162
    EVT_UPDATE_UI( ID_LIBEDIT_NEW_PART_FROM_EXISTING, LIB_EDIT_FRAME::OnUpdateEditingPart )
163 164 165 166 167 168 169 170 171
    EVT_UPDATE_UI( wxID_UNDO, LIB_EDIT_FRAME::OnUpdateUndo )
    EVT_UPDATE_UI( wxID_REDO, LIB_EDIT_FRAME::OnUpdateRedo )
    EVT_UPDATE_UI( ID_LIBEDIT_SAVE_CURRENT_LIB, LIB_EDIT_FRAME::OnUpdateSaveCurrentLib )
    EVT_UPDATE_UI( ID_LIBEDIT_VIEW_DOC, LIB_EDIT_FRAME::OnUpdateViewDoc )
    EVT_UPDATE_UI( ID_LIBEDIT_EDIT_PIN_BY_PIN, LIB_EDIT_FRAME::OnUpdatePinByPin )
    EVT_UPDATE_UI( ID_LIBEDIT_SELECT_PART_NUMBER, LIB_EDIT_FRAME::OnUpdatePartNumber )
    EVT_UPDATE_UI( ID_LIBEDIT_SELECT_ALIAS, LIB_EDIT_FRAME::OnUpdateSelectAlias )
    EVT_UPDATE_UI( ID_DE_MORGAN_NORMAL_BUTT, LIB_EDIT_FRAME::OnUpdateDeMorganNormal )
    EVT_UPDATE_UI( ID_DE_MORGAN_CONVERT_BUTT, LIB_EDIT_FRAME::OnUpdateDeMorganConvert )
172 173
    EVT_UPDATE_UI( ID_NO_TOOL_SELECTED, LIB_EDIT_FRAME::OnUpdateEditingPart )
    EVT_UPDATE_UI_RANGE( ID_LIBEDIT_PIN_BUTT, ID_LIBEDIT_DELETE_ITEM_BUTT,
174
                         LIB_EDIT_FRAME::OnUpdateEditingPart )
plyatov's avatar
plyatov committed
175 176
END_EVENT_TABLE()

177
#define LIB_EDIT_FRAME_NAME wxT( "LibeditFrame" )
178

179
LIB_EDIT_FRAME::LIB_EDIT_FRAME( KIWAY* aKiway, wxWindow* aParent ) :
180
    SCH_BASE_FRAME( aKiway, aParent, FRAME_SCH_LIB_EDITOR, _( "Library Editor" ),
181 182 183
        wxDefaultPosition, wxDefaultSize, KICAD_DEFAULT_DRAWFRAME_STYLE, GetLibEditFrameName() ),
    m_my_part( 0 ),
    m_tempCopyComponent( 0 )
plyatov's avatar
plyatov committed
184
{
185
    wxASSERT( aParent );
charras's avatar
charras committed
186

187
    m_showAxis   = true;            // true to draw axis
188
    m_configPath = wxT( "LibraryEditor" );
charras's avatar
charras committed
189
    SetShowDeMorgan( false );
190
    m_drawSpecificConvert = true;
191
    m_drawSpecificUnit    = false;
192
    m_HotkeysZoomAndGridList = g_Libedit_Hokeys_Descr;
193
    m_editPinsPerPartOrConvert = false;
194

195 196 197 198
    // Delayed initialization
    if( m_textSize == -1 )
        m_textSize = GetDefaultTextSize();

199
    // Initialize grid id to the default value 50 mils:
200
    m_LastGridSizeId = ID_POPUP_GRID_LEVEL_50 - ID_POPUP_GRID_LEVEL_1000;
201

202
    wxIcon icon;
203
    icon.CopyFromBitmap( KiBitmap( libedit_icon_xpm ) );
204 205
    SetIcon( icon );

206
    SetScreen( new SCH_SCREEN( aKiway ) );
207
    GetScreen()->m_Center = true;
208

209
    SetCrossHairPosition( wxPoint( 0, 0 ) );
210

211
    LoadSettings( config() );
212

213 214 215 216 217 218 219
    // Ensure m_LastGridSizeId is an offset inside the allowed schematic range
    if( m_LastGridSizeId < ID_POPUP_GRID_LEVEL_50 - ID_POPUP_GRID_LEVEL_1000 )
        m_LastGridSizeId = ID_POPUP_GRID_LEVEL_50 - ID_POPUP_GRID_LEVEL_1000;

    if( m_LastGridSizeId > ID_POPUP_GRID_LEVEL_1 - ID_POPUP_GRID_LEVEL_1000 )
        m_LastGridSizeId = ID_POPUP_GRID_LEVEL_1 - ID_POPUP_GRID_LEVEL_1000;

220
    SetSize( m_FramePos.x, m_FramePos.y, m_FrameSize.x, m_FrameSize.y );
221

222 223
    GetScreen()->SetGrid( ID_POPUP_GRID_LEVEL_1000 + m_LastGridSizeId  );

224
    if( m_canvas )
225
        m_canvas->SetEnableBlockCommands( true );
226

charras's avatar
charras committed
227
    ReCreateMenuBar();
228 229
    ReCreateHToolbar();
    ReCreateVToolbar();
230
    CreateOptionToolbar();
231
    DisplayLibInfos();
232
    DisplayCmpDoc();
233 234
    UpdateAliasSelectList();
    UpdatePartSelectList();
235

236 237
    m_auimgr.SetManagedWindow( this );

238 239 240 241 242 243 244 245 246
    EDA_PANEINFO horiz;
    horiz.HorizontalToolbarPane();

    EDA_PANEINFO vert;
    vert.VerticalToolbarPane();

    EDA_PANEINFO mesg;
    mesg.MessageToolbarPane();

247 248
    m_auimgr.AddPane( m_mainToolBar,
                      wxAuiPaneInfo( horiz ).Name( wxT( "m_mainToolBar" ) ).Top().Row( 0 ) );
249

250
    m_auimgr.AddPane( m_drawToolBar,
251
                      wxAuiPaneInfo( vert ).Name( wxT( "m_VToolBar" ) ).Right() );
252

253 254
    m_auimgr.AddPane( m_optionsToolBar,
                      wxAuiPaneInfo( vert ).Name( wxT( "m_optionsToolBar" ) ).Left() );
255

256
    m_auimgr.AddPane( m_canvas,
257
                      wxAuiPaneInfo().Name( wxT( "DrawFrame" ) ).CentrePane() );
258

259
    m_auimgr.AddPane( m_messagePanel,
260
                      wxAuiPaneInfo( mesg ).Name( wxT( "MsgPanel" ) ).Bottom().Layer(10) );
261

262
    m_auimgr.Update();
263

264 265
    Show( true );

266 267
    wxCommandEvent evt( wxEVT_COMMAND_MENU_SELECTED, ID_ZOOM_PAGE );
    wxPostEvent( this, evt );
plyatov's avatar
plyatov committed
268 269
}

270

271
LIB_EDIT_FRAME::~LIB_EDIT_FRAME()
plyatov's avatar
plyatov committed
272
{
273
    m_drawItem = m_lastDrawItem = NULL;
274

275 276
    delete m_tempCopyComponent;
    delete m_my_part;
plyatov's avatar
plyatov committed
277 278
}

279 280 281 282 283
const wxChar* LIB_EDIT_FRAME::GetLibEditFrameName()
{
    return LIB_EDIT_FRAME_NAME;
}

284

285
void LIB_EDIT_FRAME::SetDrawItem( LIB_ITEM* drawItem )
286 287 288 289 290
{
    m_drawItem = drawItem;
}


291
void LIB_EDIT_FRAME::OnCloseWindow( wxCloseEvent& Event )
plyatov's avatar
plyatov committed
292
{
293
    if( GetScreen()->IsModify() )
294
    {
295 296 297
        int ii = DisplayExitDialog( this, _( "Save the changes in the library before closing?" ) );

        switch( ii )
298
        {
299 300 301 302 303 304 305 306 307 308
        case wxID_NO:
            break;

        case wxID_YES:
            if ( this->SaveActiveLibrary( false ) )
                break;

            // fall through: cancel the close because of an error

        case wxID_CANCEL:
309 310
            Event.Veto();
            return;
311
        }
312
        GetScreen()->ClrModify();
313 314
    }

315 316 317
    PART_LIBS* libs = Prj().SchLibs();

    BOOST_FOREACH( const PART_LIB& lib, *libs )
318
    {
319
        if( lib.IsModified() )
320
        {
321 322 323 324
            wxString msg = wxString::Format( _(
                "Library '%s' was modified!\nDiscard changes?" ),
                GetChars( lib.GetName() )
                );
325

326 327
            if( !IsOK( this, msg ) )
            {
328 329
                Event.Veto();
                return;
330 331 332 333 334
            }
        }
    }

    Destroy();
plyatov's avatar
plyatov committed
335 336 337
}


338
double LIB_EDIT_FRAME::BestZoom()
plyatov's avatar
plyatov committed
339
{
340 341 342 343 344 345
    /* Please, note: wxMSW before version 2.9 seems have
     * problems with zoom values < 1 ( i.e. userscale > 1) and needs to be patched:
     * edit file <wxWidgets>/src/msw/dc.cpp
     * search for line static const int VIEWPORT_EXTENT = 1000;
     * and replace by static const int VIEWPORT_EXTENT = 10000;
     */
346
    int      dx, dy;
347

348 349 350
    LIB_PART*      part = GetCurPart();

    if( part )
351
    {
352 353 354 355
        EDA_RECT boundingBox = part->GetBoundingBox( m_unit, m_convert );

        dx = boundingBox.GetWidth();
        dy = boundingBox.GetHeight();
356
        SetScrollCenterPosition( wxPoint( 0, 0 ) );
357 358 359
    }
    else
    {
360 361 362 363 364
        const PAGE_INFO& pageInfo = GetScreen()->GetPageSettings();

        dx = pageInfo.GetSizeIU().x;
        dy = pageInfo.GetSizeIU().y;

365
        SetScrollCenterPosition( wxPoint( 0, 0 ) );
366 367
    }

368
    wxSize size = m_canvas->GetClientSize();
369

370 371
    // Reserve a 10% margin around component bounding box.
    double margin_scale_factor = 0.8;
372
    double zx =(double) dx / ( margin_scale_factor * (double)size.x );
373
    double zy = (double) dy / ( margin_scale_factor * (double)size.y );
374

375
    double bestzoom = std::max( zx, zy );
376 377 378 379 380

    // keep it >= minimal existing zoom (can happen for very small components
    // for instance when starting a new component
    if( bestzoom  < GetScreen()->m_ZoomList[0] )
        bestzoom  = GetScreen()->m_ZoomList[0];
381

382
    return bestzoom;
plyatov's avatar
plyatov committed
383 384 385
}


386
void LIB_EDIT_FRAME::UpdateAliasSelectList()
387
{
388
    if( m_aliasSelectBox == NULL )
389 390
        return;

391
    m_aliasSelectBox->Clear();
392

393 394 395
    LIB_PART*      part = GetCurPart();

    if( !part )
396 397
        return;

398
    m_aliasSelectBox->Append( part->GetAliasNames() );
399
    m_aliasSelectBox->SetSelection( 0 );
400

401
    int index = m_aliasSelectBox->FindString( m_aliasName );
402

403
    if( index != wxNOT_FOUND )
404
        m_aliasSelectBox->SetSelection( index );
405 406 407
}


408
void LIB_EDIT_FRAME::UpdatePartSelectList()
409
{
410
    if( m_partSelectBox == NULL )
411 412
        return;

413 414
    if( m_partSelectBox->GetCount() != 0 )
        m_partSelectBox->Clear();
415

416 417 418
    LIB_PART*      part = GetCurPart();

    if( !part || part->GetUnitCount() <= 1 )
419
    {
420
        m_unit = 1;
421
        m_partSelectBox->Append( wxEmptyString );
422 423 424
    }
    else
    {
425
        for( int i = 0; i < part->GetUnitCount(); i++ )
426
        {
427 428 429
            wxString sub  = LIB_PART::SubReference( i+1, false );
            wxString unit = wxString::Format( _( "Unit %s" ), GetChars( sub ) );
            m_partSelectBox->Append( unit );
430 431 432
        }
    }

433 434 435 436 437
    // Ensure the current selected unit is compatible with
    // the number of units of the current part:
    if( part && part->GetUnitCount() < m_unit )
        m_unit = 1;

438
    m_partSelectBox->SetSelection( ( m_unit > 0 ) ? m_unit - 1 : 0 );
439 440 441
}


442
void LIB_EDIT_FRAME::OnUpdateEditingPart( wxUpdateUIEvent& aEvent )
443
{
444
    LIB_PART*      part = GetCurPart();
445

446 447 448
    aEvent.Enable( part != NULL );

    if( part && aEvent.GetEventObject() == m_drawToolBar )
449
        aEvent.Check( GetToolId() == aEvent.GetId() );
450 451
}

452

453
void LIB_EDIT_FRAME::OnUpdateNotEditingPart( wxUpdateUIEvent& event )
454
{
455
    event.Enable( !GetCurPart() );
456 457 458
}


459
void LIB_EDIT_FRAME::OnUpdateUndo( wxUpdateUIEvent& event )
460
{
461 462
    event.Enable( GetCurPart() && GetScreen() &&
        GetScreen()->GetUndoCommandCount() != 0 && !IsEditingDrawItem() );
463 464 465
}


466
void LIB_EDIT_FRAME::OnUpdateRedo( wxUpdateUIEvent& event )
467
{
468 469
    event.Enable( GetCurPart() && GetScreen() &&
        GetScreen()->GetRedoCommandCount() != 0 && !IsEditingDrawItem() );
470 471 472
}


473
void LIB_EDIT_FRAME::OnUpdateSaveCurrentLib( wxUpdateUIEvent& event )
474
{
475 476 477 478
    PART_LIB* lib = GetCurLib();

    event.Enable( lib && !lib->IsReadOnly()
                  && ( lib->IsModified() || GetScreen()->IsModify() ) );
479 480 481
}


482
void LIB_EDIT_FRAME::OnUpdateViewDoc( wxUpdateUIEvent& event )
483 484 485
{
    bool enable = false;

486 487 488 489
    PART_LIB*    lib  = GetCurLib();
    LIB_PART*       part = GetCurPart();

    if( part && lib )
490
    {
491
        LIB_ALIAS* alias = part->GetAlias( m_aliasName );
492

493
        wxCHECK_RET( alias != NULL, wxT( "Alias <" ) + m_aliasName + wxT( "> not found." ) );
494 495

        enable = !alias->GetDocFileName().IsEmpty();
496 497 498 499 500 501
    }

    event.Enable( enable );
}


502
void LIB_EDIT_FRAME::OnUpdatePinByPin( wxUpdateUIEvent& event )
503
{
504 505 506
    LIB_PART*      part = GetCurPart();

    event.Enable( part && ( part->GetUnitCount() > 1 || m_showDeMorgan ) );
507

508
    event.Check( m_editPinsPerPartOrConvert );
509 510 511
}


512
void LIB_EDIT_FRAME::OnUpdatePartNumber( wxUpdateUIEvent& event )
513
{
514
    if( m_partSelectBox == NULL )
515 516
        return;

517 518 519 520 521
    LIB_PART*      part = GetCurPart();

    // Using the typical event.Enable() call doesn't seem to work with wxGTK
    // so use the pointer to alias combobox to directly enable or disable.
    m_partSelectBox->Enable( part && part->GetUnitCount() > 1 );
522 523 524
}


525
void LIB_EDIT_FRAME::OnUpdateDeMorganNormal( wxUpdateUIEvent& event )
526
{
527
    if( m_mainToolBar == NULL )
528 529
        return;

530 531 532
    LIB_PART*      part = GetCurPart();

    event.Enable( GetShowDeMorgan() || ( part && part->HasConversion() ) );
533
    event.Check( m_convert <= 1 );
534 535 536
}


537
void LIB_EDIT_FRAME::OnUpdateDeMorganConvert( wxUpdateUIEvent& event )
538
{
539
    if( m_mainToolBar == NULL )
540 541
        return;

542 543 544
    LIB_PART*      part = GetCurPart();

    event.Enable( GetShowDeMorgan() || ( part && part->HasConversion() ) );
545
    event.Check( m_convert > 1 );
546 547 548
}


549
void LIB_EDIT_FRAME::OnUpdateSelectAlias( wxUpdateUIEvent& event )
550
{
551
    if( m_aliasSelectBox == NULL )
552 553
        return;

554 555 556 557 558
    LIB_PART*      part = GetCurPart();

    // Using the typical event.Enable() call doesn't seem to work with wxGTK
    // so use the pointer to alias combobox to directly enable or disable.
    m_aliasSelectBox->Enable( part && part->GetAliasCount() > 1 );
559 560 561
}


562
void LIB_EDIT_FRAME::OnSelectAlias( wxCommandEvent& event )
563
{
564 565
    if( m_aliasSelectBox == NULL
        || ( m_aliasSelectBox->GetStringSelection().CmpNoCase( m_aliasName ) == 0)  )
566 567
        return;

568
    m_lastDrawItem = NULL;
569
    m_aliasName = m_aliasSelectBox->GetStringSelection();
570

571
    DisplayCmpDoc();
572
    m_canvas->Refresh();
573 574 575
}


576
void LIB_EDIT_FRAME::OnSelectPart( wxCommandEvent& event )
577 578
{
    int i = event.GetSelection();
579

580
    if( ( i == wxNOT_FOUND ) || ( ( i + 1 ) == m_unit ) )
581
        return;
582

583 584
    m_lastDrawItem = NULL;
    m_unit = i + 1;
585
    m_canvas->Refresh();
586
    DisplayCmpDoc();
587 588 589
}


590
void LIB_EDIT_FRAME::OnViewEntryDoc( wxCommandEvent& event )
591
{
592 593 594
    LIB_PART* part = GetCurPart();

    if( !part )
595 596
        return;

597
    wxString    fileName;
598
    LIB_ALIAS*  alias = part->GetAlias( m_aliasName );
599

600 601 602
    wxCHECK_RET( alias != NULL, wxT( "Alias not found." ) );

    fileName = alias->GetDocFileName();
603 604

    if( !fileName.IsEmpty() )
605
    {
606
        SEARCH_STACK* lib_search = Prj().SchSearchS();
607 608 609

        GetAssociatedDocument( this, fileName, lib_search );
    }
610 611 612
}


613
void LIB_EDIT_FRAME::OnSelectBodyStyle( wxCommandEvent& event )
614
{
615
    m_canvas->EndMouseCapture( ID_NO_TOOL_SELECTED, m_canvas->GetDefaultCursor() );
616 617 618 619 620 621 622

    if( event.GetId() == ID_DE_MORGAN_NORMAL_BUTT )
        m_convert = 1;
    else
        m_convert = 2;

    m_lastDrawItem = NULL;
623
    m_canvas->Refresh();
624 625 626
}


627
void LIB_EDIT_FRAME::Process_Special_Functions( wxCommandEvent& event )
plyatov's avatar
plyatov committed
628
{
629 630
    int     id = event.GetId();
    wxPoint pos;
631

632
    m_canvas->SetIgnoreMouseEvents( true );
633 634 635 636

    wxGetMousePosition( &pos.x, &pos.y );
    pos.y += 20;

637
    switch( id )   // Stop placement commands before handling new command.
638 639
    {
    case ID_POPUP_LIBEDIT_END_CREATE_ITEM:
640
    case ID_LIBEDIT_EDIT_PIN:
641 642 643 644 645 646 647 648 649
    case ID_POPUP_LIBEDIT_BODY_EDIT_ITEM:
    case ID_POPUP_LIBEDIT_FIELD_EDIT_ITEM:
    case ID_POPUP_LIBEDIT_PIN_GLOBAL_CHANGE_PINSIZE_ITEM:
    case ID_POPUP_LIBEDIT_PIN_GLOBAL_CHANGE_PINNAMESIZE_ITEM:
    case ID_POPUP_LIBEDIT_PIN_GLOBAL_CHANGE_PINNUMSIZE_ITEM:
    case ID_POPUP_ZOOM_BLOCK:
    case ID_POPUP_DELETE_BLOCK:
    case ID_POPUP_COPY_BLOCK:
    case ID_POPUP_SELECT_ITEMS_BLOCK:
650
    case ID_POPUP_MIRROR_X_BLOCK:
651
    case ID_POPUP_MIRROR_Y_BLOCK:
652
    case ID_POPUP_ROTATE_BLOCK:
653 654 655 656
    case ID_POPUP_PLACE_BLOCK:
    case ID_POPUP_LIBEDIT_DELETE_CURRENT_POLY_SEGMENT:
        break;

charras's avatar
charras committed
657
    case ID_POPUP_LIBEDIT_CANCEL_EDITING:
658 659
        if( m_canvas->IsMouseCaptured() )
            m_canvas->EndMouseCapture();
charras's avatar
charras committed
660
        else
661
            m_canvas->EndMouseCapture( ID_NO_TOOL_SELECTED, m_canvas->GetDefaultCursor() );
charras's avatar
charras committed
662 663 664
        break;

    case ID_POPUP_LIBEDIT_DELETE_ITEM:
665
        m_canvas->EndMouseCapture();
666 667 668
        break;

    default:
669
        m_canvas->EndMouseCapture( ID_NO_TOOL_SELECTED, m_canvas->GetDefaultCursor(),
670
                                    wxEmptyString );
671 672 673
        break;
    }

674
    INSTALL_UNBUFFERED_DC( dc, m_canvas );
675

676 677
    switch( id )
    {
678 679 680
    case ID_POPUP_LIBEDIT_CANCEL_EDITING:
        break;

681 682 683 684 685
    case ID_LIBEDIT_SELECT_CURRENT_LIB:
        SelectActiveLibrary();
        break;

    case ID_LIBEDIT_SAVE_CURRENT_PART:
686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709
        {
            LIB_PART*   part = GetCurPart();

            if( !part )
            {
                DisplayError( this, _( "No part to save." ) );
                break;
            }

            PART_LIB*   lib  = GetCurLib();

            if( !lib )
                SelectActiveLibrary();

            lib = GetCurLib();

            if( !lib )
            {
                DisplayError( this, _( "No library specified." ) );
                break;
            }

            SaveOnePart( lib );
        }
710 711 712
        break;

    case ID_LIBEDIT_EDIT_PIN_BY_PIN:
713
        m_editPinsPerPartOrConvert = m_mainToolBar->GetToolToggled( ID_LIBEDIT_EDIT_PIN_BY_PIN );
714 715 716
        break;

    case ID_POPUP_LIBEDIT_END_CREATE_ITEM:
717
        m_canvas->MoveCursorToCrossHair();
718
        if( m_drawItem )
719 720 721 722 723 724
        {
            EndDrawGraphicItem( &dc );
        }
        break;

    case ID_POPUP_LIBEDIT_BODY_EDIT_ITEM:
725
        if( m_drawItem )
726
        {
727
            m_canvas->CrossHairOff( &dc );
728

729
            switch( m_drawItem->Type() )
730
            {
731 732 733 734
            case LIB_ARC_T:
            case LIB_CIRCLE_T:
            case LIB_RECTANGLE_T:
            case LIB_POLYLINE_T:
735
                EditGraphicSymbol( &dc, m_drawItem );
736 737
                break;

738
            case LIB_TEXT_T:
739
                EditSymbolText( &dc, m_drawItem );
740 741 742 743 744 745
                break;

            default:
                ;
            }

746
            m_canvas->CrossHairOn( &dc );
747 748 749 750
        }
        break;

    case ID_POPUP_LIBEDIT_DELETE_CURRENT_POLY_SEGMENT:
751 752 753 754
        {
            // Delete the last created segment, while creating a polyline draw item
            if( m_drawItem == NULL )
                break;
755

756 757 758 759 760 761 762 763 764 765 766
            m_canvas->MoveCursorToCrossHair();
            STATUS_FLAGS oldFlags = m_drawItem->GetFlags();
            m_drawItem->ClearFlags();
            m_drawItem->Draw( m_canvas, &dc, wxPoint( 0, 0 ), UNSPECIFIED_COLOR, g_XorMode, NULL,
                              DefaultTransform );
            ( (LIB_POLYLINE*) m_drawItem )->DeleteSegment( GetCrossHairPosition( true ) );
            m_drawItem->Draw( m_canvas, &dc, wxPoint( 0, 0 ), UNSPECIFIED_COLOR, g_XorMode, NULL,
                              DefaultTransform );
            m_drawItem->SetFlags( oldFlags );
            m_lastDrawItem = NULL;
        }
767 768 769
        break;

    case ID_POPUP_LIBEDIT_DELETE_ITEM:
770 771
        if( m_drawItem )
            deleteItem( &dc );
772 773 774 775

        break;

    case ID_POPUP_LIBEDIT_MOVE_ITEM_REQUEST:
776
        if( m_drawItem == NULL )
777
            break;
778

779
        if( m_drawItem->Type() == LIB_PIN_T )
780 781 782 783 784
            StartMovePin( &dc );
        else
            StartMoveDrawSymbol( &dc );
        break;

785 786 787 788 789
    case ID_POPUP_LIBEDIT_MODIFY_ITEM:

        if( m_drawItem == NULL )
            break;

790
        m_canvas->MoveCursorToCrossHair();
791 792 793 794
        if( m_drawItem->Type() == LIB_RECTANGLE_T
            || m_drawItem->Type() == LIB_CIRCLE_T
            || m_drawItem->Type() == LIB_POLYLINE_T
            || m_drawItem->Type() == LIB_ARC_T
795 796 797 798 799 800 801
            )
        {
            StartModifyDrawSymbol( &dc );
        }

        break;

802
    case ID_POPUP_LIBEDIT_FIELD_EDIT_ITEM:
803
        if( m_drawItem == NULL )
804
            break;
805

806
        m_canvas->CrossHairOff( &dc );
807

808
        if( m_drawItem->Type() == LIB_FIELD_T )
809
        {
810
            EditField( (LIB_FIELD*) m_drawItem );
811
        }
812

813 814
        m_canvas->MoveCursorToCrossHair();
        m_canvas->CrossHairOn( &dc );
815 816 817 818 819
        break;

    case ID_POPUP_LIBEDIT_PIN_GLOBAL_CHANGE_PINSIZE_ITEM:
    case ID_POPUP_LIBEDIT_PIN_GLOBAL_CHANGE_PINNAMESIZE_ITEM:
    case ID_POPUP_LIBEDIT_PIN_GLOBAL_CHANGE_PINNUMSIZE_ITEM:
820 821 822
        {
            if( !m_drawItem || m_drawItem->Type() != LIB_PIN_T )
                break;
823

824 825 826 827 828 829 830 831
            LIB_PART*      part = GetCurPart();

            SaveCopyInUndoList( part );

            GlobalSetPins( (LIB_PIN*) m_drawItem, id );
            m_canvas->MoveCursorToCrossHair();
            m_canvas->Refresh();
        }
832 833 834
        break;

    case ID_POPUP_ZOOM_BLOCK:
835
        m_canvas->SetAutoPanRequest( false );
836
        GetScreen()->m_BlockLocate.SetCommand( BLOCK_ZOOM );
837 838 839 840
        HandleBlockEnd( &dc );
        break;

    case ID_POPUP_DELETE_BLOCK:
841
        m_canvas->SetAutoPanRequest( false );
842
        GetScreen()->m_BlockLocate.SetCommand( BLOCK_DELETE );
843
        m_canvas->MoveCursorToCrossHair();
844 845 846 847
        HandleBlockEnd( &dc );
        break;

    case ID_POPUP_COPY_BLOCK:
848
        m_canvas->SetAutoPanRequest( false );
849
        GetScreen()->m_BlockLocate.SetCommand( BLOCK_COPY );
850
        m_canvas->MoveCursorToCrossHair();
851 852 853 854
        HandleBlockPlace( &dc );
        break;

    case ID_POPUP_SELECT_ITEMS_BLOCK:
855
        m_canvas->SetAutoPanRequest( false );
856
        GetScreen()->m_BlockLocate.SetCommand( BLOCK_SELECT_ITEMS_ONLY );
857
        m_canvas->MoveCursorToCrossHair();
858 859 860
        HandleBlockEnd( &dc );
        break;

861
    case ID_POPUP_MIRROR_Y_BLOCK:
862
        m_canvas->SetAutoPanRequest( false );
863
        GetScreen()->m_BlockLocate.SetCommand( BLOCK_MIRROR_Y );
864
        m_canvas->MoveCursorToCrossHair();
865 866 867
        HandleBlockPlace( &dc );
        break;

868
    case ID_POPUP_MIRROR_X_BLOCK:
869
        m_canvas->SetAutoPanRequest( false );
870
        GetScreen()->m_BlockLocate.SetCommand( BLOCK_MIRROR_X );
871
        m_canvas->MoveCursorToCrossHair();
872 873 874 875
        HandleBlockPlace( &dc );
        break;

    case ID_POPUP_ROTATE_BLOCK:
876
        m_canvas->SetAutoPanRequest( false );
877
        GetScreen()->m_BlockLocate.SetCommand( BLOCK_ROTATE );
878
        m_canvas->MoveCursorToCrossHair();
879 880 881
        HandleBlockPlace( &dc );
        break;

882
    case ID_POPUP_PLACE_BLOCK:
883
        m_canvas->SetAutoPanRequest( false );
884
        m_canvas->MoveCursorToCrossHair();
885 886 887 888
        HandleBlockPlace( &dc );
        break;

    default:
889
        DisplayError( this, wxT( "LIB_EDIT_FRAME::Process_Special_Functions error" ) );
890 891 892
        break;
    }

893
    m_canvas->SetIgnoreMouseEvents( false );
894

895
    if( GetToolId() == ID_NO_TOOL_SELECTED )
896
        m_lastDrawItem = NULL;
plyatov's avatar
plyatov committed
897
}
898

899

900
void LIB_EDIT_FRAME::OnActivate( wxActivateEvent& event )
901
{
902
    EDA_DRAW_FRAME::OnActivate( event );
903 904
}

905

906
PART_LIB* LIB_EDIT_FRAME::GetCurLib()
907
{
908
    wxString  name = Prj().GetRString( PROJECT::SCH_LIBEDIT_CUR_LIB );
909

910 911 912
    if( !!name )
    {
        PART_LIB* lib = Prj().SchLibs()->FindLibrary( name );
913

914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929
        if( !lib )
            Prj().SetRString( PROJECT::SCH_LIBEDIT_CUR_LIB, wxEmptyString );

        return lib;
    }

    return NULL;
}


PART_LIB* LIB_EDIT_FRAME::SetCurLib( PART_LIB* aLib )
{
    PART_LIB* old = GetCurLib();

    if( !aLib || !aLib->GetName() )
        Prj().SetRString( PROJECT::SCH_LIBEDIT_CUR_LIB, wxEmptyString );
930
    else
931 932 933
        Prj().SetRString( PROJECT::SCH_LIBEDIT_CUR_LIB, aLib->GetName() );

    return old;
934
}
charras's avatar
charras committed
935

936

937
LIB_PART* LIB_EDIT_FRAME::GetCurPart()
938
{
939 940 941 942
    if( !m_my_part )
    {
        wxString    name = Prj().GetRString( PROJECT::SCH_LIBEDIT_CUR_PART );
        LIB_PART*   part;
943

944 945 946 947 948 949 950 951
        if( !!name && ( part = Prj().SchLibs()->FindLibPart( name ) ) )
        {
            // clone it from the PART_LIB and own it.
            m_my_part = new LIB_PART( *part );
        }
        else
            Prj().SetRString( PROJECT::SCH_LIBEDIT_CUR_PART, wxEmptyString );
    }
952

953
    return m_my_part;
954 955
}

956

957
void LIB_EDIT_FRAME::SetCurPart( LIB_PART* aPart )
958
{
959 960
    delete m_my_part;
    m_my_part = aPart;      // take ownership here
961

962 963 964 965
    // retain in case this wxFrame is re-opened later on the same PROJECT
    Prj().SetRString( PROJECT::SCH_LIBEDIT_CUR_PART,
            aPart ? aPart->GetName() : wxString() );
}
966

967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988

void LIB_EDIT_FRAME::TempCopyComponent()
{
    delete m_tempCopyComponent;

    if( LIB_PART* part = GetCurPart() )
        // clone it and own the clone.
        m_tempCopyComponent = new LIB_PART( *part );
    else
        // clear it, there was no CurPart
        m_tempCopyComponent = NULL;
}


void LIB_EDIT_FRAME::RestoreComponent()
{
    if( m_tempCopyComponent )
    {
        // transfer ownership to CurPart
        SetCurPart( m_tempCopyComponent );
        m_tempCopyComponent = NULL;
    }
989
}
990

991

992
void LIB_EDIT_FRAME::ClearTempCopyComponent()
993 994 995 996
{
    delete m_tempCopyComponent;
    m_tempCopyComponent = NULL;
}
997 998


999
void LIB_EDIT_FRAME::EditSymbolText( wxDC* DC, LIB_ITEM* DrawItem )
1000
{
1001
    if ( ( DrawItem == NULL ) || ( DrawItem->Type() != LIB_TEXT_T ) )
1002 1003
        return;

1004
    // Deleting old text
1005
    if( DC && !DrawItem->InEditMode() )
1006 1007
        DrawItem->Draw( m_canvas, DC, wxPoint( 0, 0 ), UNSPECIFIED_COLOR, g_XorMode, NULL,
                        DefaultTransform );
1008 1009 1010 1011 1012 1013

    DIALOG_LIB_EDIT_TEXT* frame = new DIALOG_LIB_EDIT_TEXT( this, (LIB_TEXT*) DrawItem );
    frame->ShowModal();
    frame->Destroy();
    OnModify();

1014
    // Display new text
1015
    if( DC && !DrawItem->InEditMode() )
1016
        DrawItem->Draw( m_canvas, DC, wxPoint( 0, 0 ), UNSPECIFIED_COLOR, GR_DEFAULT_DRAWMODE, NULL,
1017 1018
                        DefaultTransform );
}
1019 1020


1021
void LIB_EDIT_FRAME::OnEditComponentProperties( wxCommandEvent& event )
1022
{
1023
    bool partLocked = GetCurPart()->UnitsLocked();
1024 1025 1026 1027 1028 1029

    DIALOG_EDIT_COMPONENT_IN_LIBRARY dlg( this );

    if( dlg.ShowModal() == wxID_CANCEL )
        return;

1030
    if( partLocked != GetCurPart()->UnitsLocked() )
1031
    {
1032 1033
        // m_editPinsPerPartOrConvert is set to the better value, if m_UnitSelectionLocked
        // has changed
1034
        m_editPinsPerPartOrConvert = GetCurPart()->UnitsLocked() ? true : false;
1035 1036 1037 1038 1039 1040 1041
    }

    UpdateAliasSelectList();
    UpdatePartSelectList();
    DisplayLibInfos();
    DisplayCmpDoc();
    OnModify();
1042
    m_canvas->Refresh();
1043
}
1044 1045 1046 1047 1048 1049 1050


void LIB_EDIT_FRAME::InstallDimensionsDialog( wxCommandEvent& event )
{
    DIALOG_LIBEDIT_DIMENSIONS dlg( this );
    dlg.ShowModal();
}
1051 1052 1053 1054


void LIB_EDIT_FRAME::OnCreateNewPartFromExisting( wxCommandEvent& event )
{
1055 1056 1057
    LIB_PART*      part = GetCurPart();

    wxCHECK_RET( part, wxT( "Cannot create new part from non-existent current part." ) );
1058

1059 1060
    INSTALL_UNBUFFERED_DC( dc, m_canvas );
    m_canvas->CrossHairOff( &dc );
1061 1062 1063

    EditField( &part->GetValueField() );

1064 1065
    m_canvas->MoveCursorToCrossHair();
    m_canvas->CrossHairOn( &dc );
1066
}
1067

1068

1069 1070 1071 1072
void LIB_EDIT_FRAME::OnSelectTool( wxCommandEvent& aEvent )
{
    int id = aEvent.GetId();

1073
    if( GetToolId() == ID_NO_TOOL_SELECTED )
1074 1075
        m_lastDrawItem = NULL;

1076
    m_canvas->EndMouseCapture( ID_NO_TOOL_SELECTED, m_canvas->GetDefaultCursor(),
1077
                               wxEmptyString );
1078

1079 1080
    LIB_PART*      part = GetCurPart();

1081 1082
    switch( id )
    {
1083
    case ID_NO_TOOL_SELECTED:
1084
        SetToolID( id, m_canvas->GetDefaultCursor(), wxEmptyString );
1085 1086 1087
        break;

    case ID_LIBEDIT_PIN_BUTT:
1088
        if( part )
1089 1090 1091 1092 1093 1094
        {
            SetToolID( id, wxCURSOR_PENCIL, _( "Add pin" ) );
        }
        else
        {
            SetToolID( id, wxCURSOR_ARROW, _( "Set pin options" ) );
1095

1096
            wxCommandEvent cmd( wxEVT_COMMAND_MENU_SELECTED );
1097

1098 1099
            cmd.SetId( ID_LIBEDIT_EDIT_PIN );
            GetEventHandler()->ProcessEvent( cmd );
1100
            SetToolID( ID_NO_TOOL_SELECTED, m_canvas->GetDefaultCursor(), wxEmptyString );
1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128
        }
        break;

    case ID_LIBEDIT_BODY_TEXT_BUTT:
        SetToolID( id, wxCURSOR_PENCIL, _( "Add text" ) );
        break;

    case ID_LIBEDIT_BODY_RECT_BUTT:
        SetToolID( id, wxCURSOR_PENCIL, _( "Add rectangle" ) );
        break;

    case ID_LIBEDIT_BODY_CIRCLE_BUTT:
        SetToolID( id, wxCURSOR_PENCIL, _( "Add circle" ) );
        break;

    case ID_LIBEDIT_BODY_ARC_BUTT:
        SetToolID( id, wxCURSOR_PENCIL, _( "Add arc" ) );
        break;

    case ID_LIBEDIT_BODY_LINE_BUTT:
        SetToolID( id, wxCURSOR_PENCIL, _( "Add line" ) );
        break;

    case ID_LIBEDIT_ANCHOR_ITEM_BUTT:
        SetToolID( id, wxCURSOR_HAND, _( "Set anchor position" ) );
        break;

    case ID_LIBEDIT_IMPORT_BODY_BUTT:
1129
        SetToolID( id, m_canvas->GetDefaultCursor(), _( "Import" ) );
1130
        LoadOneSymbol();
1131
        SetToolID( ID_NO_TOOL_SELECTED, m_canvas->GetDefaultCursor(), wxEmptyString );
1132 1133 1134
        break;

    case ID_LIBEDIT_EXPORT_BODY_BUTT:
1135
        SetToolID( id, m_canvas->GetDefaultCursor(), _( "Export" ) );
1136
        SaveOneSymbol();
1137
        SetToolID( ID_NO_TOOL_SELECTED, m_canvas->GetDefaultCursor(), wxEmptyString );
1138 1139 1140
        break;

    case ID_LIBEDIT_DELETE_ITEM_BUTT:
1141
        if( !part )
1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153
        {
            wxBell();
            break;
        }

        SetToolID( id, wxCURSOR_BULLSEYE, _( "Delete item" ) );
        break;

    default:
        break;
    }

1154
    m_canvas->SetIgnoreMouseEvents( false );
1155
}
1156 1157


1158 1159 1160 1161 1162 1163 1164
void LIB_EDIT_FRAME::OnRotateItem( wxCommandEvent& aEvent )
{
    if( m_drawItem == NULL )
        return;

    if( !m_drawItem->InEditMode() )
    {
1165 1166 1167
        LIB_PART*      part = GetCurPart();

        SaveCopyInUndoList( part );
1168 1169 1170 1171
        m_drawItem->SetUnit( m_unit );
    }

    m_drawItem->Rotate();
1172
    OnModify();
1173 1174 1175 1176

    if( !m_drawItem->InEditMode() )
        m_drawItem->ClearFlags();

1177
    m_canvas->Refresh();
1178 1179 1180 1181 1182 1183

    if( GetToolId() == ID_NO_TOOL_SELECTED )
        m_lastDrawItem = NULL;
}


1184 1185
LIB_ITEM* LIB_EDIT_FRAME::LocateItemUsingCursor( const wxPoint& aPosition,
                                                 const KICAD_T aFilterList[] )
1186
{
1187 1188 1189
    LIB_PART*      part = GetCurPart();

    if( !part )
1190 1191
        return NULL;

1192 1193
    LIB_ITEM* item = locateItem( aPosition, aFilterList );

1194
    wxPoint pos = GetNearestGridPosition( aPosition );
1195 1196

    if( item == NULL && aPosition != pos )
1197 1198 1199 1200 1201 1202 1203 1204
        item = locateItem( pos, aFilterList );

    return item;
}


LIB_ITEM* LIB_EDIT_FRAME::locateItem( const wxPoint& aPosition, const KICAD_T aFilterList[] )
{
1205 1206 1207
    LIB_PART*      part = GetCurPart();

    if( !part )
1208 1209 1210 1211
        return NULL;

    LIB_ITEM* item = NULL;

1212
    m_collectedItems.Collect( part->GetDrawItemList(), aFilterList, aPosition,
1213 1214 1215 1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234 1235 1236 1237
                              m_unit, m_convert );

    if( m_collectedItems.GetCount() == 0 )
    {
        ClearMsgPanel();
    }
    else if( m_collectedItems.GetCount() == 1 )
    {
        item = m_collectedItems[0];
    }
    else
    {
        if( item == NULL )
        {
            wxASSERT_MSG( m_collectedItems.GetCount() <= MAX_SELECT_ITEM_IDS,
                          wxT( "Select item clarification context menu size limit exceeded." ) );

            wxMenu selectMenu;
            wxMenuItem* title = new wxMenuItem( &selectMenu, wxID_NONE, _( "Clarify Selection" ) );

            selectMenu.Append( title );
            selectMenu.AppendSeparator();

            for( int i = 0;  i < m_collectedItems.GetCount() && i < MAX_SELECT_ITEM_IDS;  i++ )
            {
1238 1239 1240
                wxString    text = m_collectedItems[i]->GetSelectMenuText();
                BITMAP_DEF  xpm = m_collectedItems[i]->GetMenuImage();

1241
                AddMenuItem( &selectMenu, ID_SELECT_ITEM_START + i, text, KiBitmap( xpm ) );
1242 1243 1244 1245
            }

            // Set to NULL in case user aborts the clarification context menu.
            m_drawItem = NULL;
1246
            m_canvas->SetAbortRequest( true );   // Changed to false if an item is selected
1247
            PopupMenu( &selectMenu );
1248
            m_canvas->MoveCursorToCrossHair();
1249 1250 1251 1252 1253
            item = m_drawItem;
        }
    }

    if( item )
1254 1255 1256 1257 1258
    {
        MSG_PANEL_ITEMS items;
        item->GetMsgPanelInfo( items );
        SetMsgPanel( items );
    }
1259
    else
1260
    {
1261
        ClearMsgPanel();
1262
    }
1263 1264 1265 1266 1267

    return item;
}


1268 1269 1270 1271
void LIB_EDIT_FRAME::deleteItem( wxDC* aDC )
{
    wxCHECK_RET( m_drawItem != NULL, wxT( "No drawing item selected to delete." ) );

1272
    m_canvas->CrossHairOff( aDC );
1273 1274 1275 1276

    LIB_PART*      part = GetCurPart();

    SaveCopyInUndoList( part );
1277 1278 1279

    if( m_drawItem->Type() == LIB_PIN_T )
    {
1280 1281
        LIB_PIN*    pin = (LIB_PIN*) m_drawItem;
        wxPoint     pos = pin->GetPosition();
1282

1283
        part->RemoveDrawItem( (LIB_ITEM*) pin, m_canvas, aDC );
1284

1285
        if( SynchronizePins() )
1286
        {
1287
            LIB_PIN* tmp = part->GetNextPin();
1288 1289 1290 1291

            while( tmp != NULL )
            {
                pin = tmp;
1292
                tmp = part->GetNextPin( pin );
1293 1294 1295 1296

                if( pin->GetPosition() != pos )
                    continue;

1297
                part->RemoveDrawItem( (LIB_ITEM*) pin );
1298 1299
            }
        }
1300

1301
        m_canvas->Refresh();
1302 1303 1304
    }
    else
    {
1305
        if( m_canvas->IsMouseCaptured() )
1306
        {
1307
            m_canvas->CallEndMouseCapture( aDC );
1308
        }
1309
        else
1310
        {
1311
            part->RemoveDrawItem( m_drawItem, m_canvas, aDC );
1312
            m_canvas->Refresh();
1313
        }
1314 1315 1316
    }

    m_drawItem = NULL;
1317
    m_lastDrawItem = NULL;
1318
    OnModify();
1319
    m_canvas->CrossHairOn( aDC );
1320
}
1321 1322 1323 1324 1325 1326 1327 1328 1329 1330 1331


void LIB_EDIT_FRAME::OnSelectItem( wxCommandEvent& aEvent )
{
    int id = aEvent.GetId();
    int index = id - ID_SELECT_ITEM_START;

    if( (id >= ID_SELECT_ITEM_START && id <= ID_SELECT_ITEM_END)
        && (index >= 0 && index < m_collectedItems.GetCount()) )
    {
        LIB_ITEM* item = m_collectedItems[index];
1332
        m_canvas->SetAbortRequest( false );
1333 1334 1335
        m_drawItem = item;
    }
}
1336 1337


1338
bool LIB_EDIT_FRAME::SynchronizePins()
1339
{
1340 1341 1342 1343
    LIB_PART*      part = GetCurPart();

    return !m_editPinsPerPartOrConvert && ( part &&
        ( part->HasConversion() || part->IsMulti() ) );
1344
}