summaryrefslogtreecommitdiff
path: root/eeschema/libeditframe.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'eeschema/libeditframe.cpp')
-rw-r--r--eeschema/libeditframe.cpp1375
1 files changed, 1375 insertions, 0 deletions
diff --git a/eeschema/libeditframe.cpp b/eeschema/libeditframe.cpp
new file mode 100644
index 0000000..bd489a0
--- /dev/null
+++ b/eeschema/libeditframe.cpp
@@ -0,0 +1,1375 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * 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.
+ *
+ * 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
+ */
+
+/**
+ * @file libeditframe.cpp
+ * @brief LIB_EDIT_FRAME class is the component library editor frame.
+ */
+
+#include <fctsys.h>
+#include <pgm_base.h>
+#include <kiface_i.h>
+#include <class_drawpanel.h>
+#include <confirm.h>
+#include <eda_doc.h>
+#include <gr_basic.h>
+#include <schframe.h>
+#include <msgpanel.h>
+
+#include <general.h>
+#include <eeschema_id.h>
+#include <libeditframe.h>
+#include <class_library.h>
+#include <lib_polyline.h>
+#include <lib_pin.h>
+
+#include <kicad_device_context.h>
+#include <hotkeys.h>
+
+#include <dialogs/dialog_lib_edit_text.h>
+#include <dialogs/dialog_edit_component_in_lib.h>
+#include <dialogs/dialog_lib_edit_pin_table.h>
+
+#include <menus_helpers.h>
+
+#include <boost/foreach.hpp>
+
+
+/* This method guarantees unique IDs for the library this run of Eeschema
+ * 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();
+
+
+wxString LIB_EDIT_FRAME:: m_aliasName;
+int LIB_EDIT_FRAME:: m_unit = 1;
+int LIB_EDIT_FRAME:: m_convert = 1;
+LIB_ITEM* LIB_EDIT_FRAME::m_lastDrawItem = NULL;
+LIB_ITEM* LIB_EDIT_FRAME::m_drawItem = NULL;
+bool LIB_EDIT_FRAME:: m_showDeMorgan = false;
+wxSize LIB_EDIT_FRAME:: m_clientSize = wxSize( -1, -1 );
+int LIB_EDIT_FRAME:: m_textSize = -1;
+int LIB_EDIT_FRAME:: m_textOrientation = TEXT_ORIENT_HORIZ;
+int LIB_EDIT_FRAME:: m_drawLineWidth = 0;
+
+// 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;
+
+FILL_T LIB_EDIT_FRAME:: m_drawFillStyle = NO_FILL;
+
+
+BEGIN_EVENT_TABLE( LIB_EDIT_FRAME, EDA_DRAW_FRAME )
+ EVT_CLOSE( LIB_EDIT_FRAME::OnCloseWindow )
+ EVT_SIZE( LIB_EDIT_FRAME::OnSize )
+ EVT_ACTIVATE( LIB_EDIT_FRAME::OnActivate )
+
+ // Main horizontal toolbar.
+ EVT_TOOL( ID_LIBEDIT_SAVE_CURRENT_LIB, LIB_EDIT_FRAME::OnSaveActiveLibrary )
+ EVT_TOOL( ID_LIBEDIT_SELECT_CURRENT_LIB, LIB_EDIT_FRAME::Process_Special_Functions )
+ EVT_TOOL( ID_LIBEDIT_DELETE_PART, LIB_EDIT_FRAME::DeleteOnePart )
+ EVT_TOOL( ID_TO_LIBVIEW, LIB_EDIT_FRAME::OnOpenLibraryViewer )
+ EVT_TOOL( ID_LIBEDIT_NEW_PART, LIB_EDIT_FRAME::CreateNewLibraryPart )
+ EVT_TOOL( ID_LIBEDIT_NEW_PART_FROM_EXISTING, LIB_EDIT_FRAME::OnCreateNewPartFromExisting )
+
+ 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( ID_LIBEDIT_EDIT_PIN_BY_TABLE, LIB_EDIT_FRAME::OnOpenPinTable )
+ EVT_TOOL( ExportPartId, LIB_EDIT_FRAME::OnExportPart )
+ EVT_TOOL( CreateNewLibAndSavePartId, LIB_EDIT_FRAME::OnExportPart )
+ EVT_TOOL( ImportPartId, LIB_EDIT_FRAME::OnImportPart )
+
+ EVT_COMBOBOX( ID_LIBEDIT_SELECT_PART_NUMBER, LIB_EDIT_FRAME::OnSelectPart )
+ EVT_COMBOBOX( ID_LIBEDIT_SELECT_ALIAS, LIB_EDIT_FRAME::OnSelectAlias )
+
+ // Right vertical toolbar.
+ EVT_TOOL( ID_NO_TOOL_SELECTED, LIB_EDIT_FRAME::OnSelectTool )
+ EVT_TOOL_RANGE( ID_LIBEDIT_PIN_BUTT, ID_LIBEDIT_DELETE_ITEM_BUTT,
+ LIB_EDIT_FRAME::OnSelectTool )
+
+ // menubar commands
+ EVT_MENU( wxID_EXIT, LIB_EDIT_FRAME::CloseWindow )
+ EVT_MENU( ID_LIBEDIT_SAVE_CURRENT_LIB_AS, LIB_EDIT_FRAME::OnSaveActiveLibrary )
+ EVT_MENU( ID_LIBEDIT_GEN_PNG_FILE, LIB_EDIT_FRAME::OnPlotCurrentComponent )
+ EVT_MENU( ID_LIBEDIT_GEN_SVG_FILE, LIB_EDIT_FRAME::OnPlotCurrentComponent )
+ EVT_MENU( wxID_HELP, EDA_DRAW_FRAME::GetKicadHelp )
+ EVT_MENU( wxID_INDEX, EDA_DRAW_FRAME::GetKicadHelp )
+ EVT_MENU( wxID_ABOUT, EDA_BASE_FRAME::GetKicadAbout )
+
+ EVT_MENU( ID_COLORS_SETUP, LIB_EDIT_FRAME::OnColorConfig )
+ EVT_MENU( wxID_PREFERENCES, LIB_EDIT_FRAME::OnPreferencesOptions )
+ EVT_MENU( ID_CONFIG_REQ, LIB_EDIT_FRAME::InstallConfigFrame )
+ EVT_MENU( ID_COLORS_SETUP, LIB_EDIT_FRAME::Process_Config )
+
+ // Multiple item selection context menu commands.
+ EVT_MENU_RANGE( ID_SELECT_ITEM_START, ID_SELECT_ITEM_END, LIB_EDIT_FRAME::OnSelectItem )
+
+ EVT_MENU_RANGE( ID_PREFERENCES_HOTKEY_START, ID_PREFERENCES_HOTKEY_END,
+ LIB_EDIT_FRAME::Process_Config )
+
+ // Context menu events and commands.
+ EVT_MENU( ID_LIBEDIT_EDIT_PIN, LIB_EDIT_FRAME::OnEditPin )
+ EVT_MENU( ID_LIBEDIT_ROTATE_ITEM, LIB_EDIT_FRAME::OnRotateItem )
+
+ EVT_MENU_RANGE( ID_POPUP_LIBEDIT_PIN_GLOBAL_CHANGE_ITEM,
+ ID_POPUP_LIBEDIT_DELETE_CURRENT_POLY_SEGMENT,
+ LIB_EDIT_FRAME::Process_Special_Functions )
+
+ EVT_MENU_RANGE( ID_POPUP_GENERAL_START_RANGE, ID_POPUP_GENERAL_END_RANGE,
+ LIB_EDIT_FRAME::Process_Special_Functions )
+
+ // Update user interface elements.
+ 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 )
+ EVT_UPDATE_UI( ID_LIBEDIT_NEW_PART_FROM_EXISTING, LIB_EDIT_FRAME::OnUpdateEditingPart )
+ 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_EDIT_PIN_BY_TABLE, LIB_EDIT_FRAME::OnUpdatePinTable )
+ 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 )
+ EVT_UPDATE_UI( ID_NO_TOOL_SELECTED, LIB_EDIT_FRAME::OnUpdateEditingPart )
+ EVT_UPDATE_UI_RANGE( ID_LIBEDIT_PIN_BUTT, ID_LIBEDIT_DELETE_ITEM_BUTT,
+ LIB_EDIT_FRAME::OnUpdateEditingPart )
+END_EVENT_TABLE()
+
+#define LIB_EDIT_FRAME_NAME wxT( "LibeditFrame" )
+
+LIB_EDIT_FRAME::LIB_EDIT_FRAME( KIWAY* aKiway, wxWindow* aParent ) :
+ SCH_BASE_FRAME( aKiway, aParent, FRAME_SCH_LIB_EDITOR, _( "Library Editor" ),
+ wxDefaultPosition, wxDefaultSize, KICAD_DEFAULT_DRAWFRAME_STYLE, LIB_EDIT_FRAME_NAME )
+{
+ wxASSERT( aParent );
+
+ m_showAxis = true; // true to draw axis
+ SetShowDeMorgan( false );
+ m_drawSpecificConvert = true;
+ m_drawSpecificUnit = false;
+ m_hotkeysDescrList = g_Libedit_Hokeys_Descr;
+ m_editPinsPerPartOrConvert = false;
+ m_repeatPinStep = DEFAULT_REPEAT_OFFSET_PIN;
+
+ m_my_part = NULL;
+ m_tempCopyComponent = NULL;
+
+ // Delayed initialization
+ if( m_textSize == -1 )
+ m_textSize = GetDefaultTextSize();
+
+ // Initialize grid id to the default value 50 mils:
+ m_LastGridSizeId = ID_POPUP_GRID_LEVEL_50 - ID_POPUP_GRID_LEVEL_1000;
+
+ wxIcon icon;
+ icon.CopyFromBitmap( KiBitmap( libedit_icon_xpm ) );
+ SetIcon( icon );
+
+ LoadSettings( config() );
+
+ SetScreen( new SCH_SCREEN( aKiway ) );
+ GetScreen()->m_Center = true;
+ GetScreen()->SetMaxUndoItems( m_UndoRedoCountMax );
+
+ SetCrossHairPosition( wxPoint( 0, 0 ) );
+
+ // 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;
+
+ SetSize( m_FramePos.x, m_FramePos.y, m_FrameSize.x, m_FrameSize.y );
+
+ GetScreen()->SetGrid( ID_POPUP_GRID_LEVEL_1000 + m_LastGridSizeId );
+
+ if( m_canvas )
+ m_canvas->SetEnableBlockCommands( true );
+
+ ReCreateMenuBar();
+ ReCreateHToolbar();
+ ReCreateVToolbar();
+
+ // Ensure the current alias name is valid if a part is loaded
+ // Sometimes it is not valid. This is the case
+ // when a part value (the part lib name), or the alias list was modified
+ // during a previous session and the modifications not saved in lib.
+ // Reopen libedit in a new session gives a non valid m_aliasName
+ // because the curr part is reloaded from the library (and this is the unmodified part)
+ // and the old alias name (from the previous edition) can be invalid
+ LIB_PART* part = GetCurPart();
+
+ if( part == NULL )
+ m_aliasName.Empty();
+ else if( m_aliasName != part->GetName() )
+ {
+ LIB_ALIAS* alias = part->GetAlias( m_aliasName );
+
+ if( !alias )
+ m_aliasName = part->GetName();
+ }
+
+
+ CreateOptionToolbar();
+ DisplayLibInfos();
+ DisplayCmpDoc();
+ UpdateAliasSelectList();
+ UpdatePartSelectList();
+
+ m_auimgr.SetManagedWindow( this );
+
+ EDA_PANEINFO horiz;
+ horiz.HorizontalToolbarPane();
+
+ EDA_PANEINFO vert;
+ vert.VerticalToolbarPane();
+
+ EDA_PANEINFO mesg;
+ mesg.MessageToolbarPane();
+
+ m_auimgr.AddPane( m_mainToolBar,
+ wxAuiPaneInfo( horiz ).Name( wxT( "m_mainToolBar" ) ).Top().Row( 0 ) );
+
+ m_auimgr.AddPane( m_drawToolBar,
+ wxAuiPaneInfo( vert ).Name( wxT( "m_VToolBar" ) ).Right() );
+
+ m_auimgr.AddPane( m_optionsToolBar,
+ wxAuiPaneInfo( vert ).Name( wxT( "m_optionsToolBar" ) ).Left() );
+
+ m_auimgr.AddPane( m_canvas,
+ wxAuiPaneInfo().Name( wxT( "DrawFrame" ) ).CentrePane() );
+
+ m_auimgr.AddPane( m_messagePanel,
+ wxAuiPaneInfo( mesg ).Name( wxT( "MsgPanel" ) ).Bottom().Layer(10) );
+
+ m_auimgr.Update();
+
+ Raise();
+ Show( true );
+
+ wxCommandEvent evt( wxEVT_COMMAND_MENU_SELECTED, ID_ZOOM_PAGE );
+ wxPostEvent( this, evt );
+}
+
+
+LIB_EDIT_FRAME::~LIB_EDIT_FRAME()
+{
+ m_drawItem = m_lastDrawItem = NULL;
+
+ delete m_tempCopyComponent;
+ delete m_my_part;
+ m_my_part = NULL;
+ m_tempCopyComponent = NULL;
+}
+
+
+void LIB_EDIT_FRAME::SetDrawItem( LIB_ITEM* drawItem )
+{
+ m_drawItem = drawItem;
+}
+
+
+void LIB_EDIT_FRAME::OnCloseWindow( wxCloseEvent& Event )
+{
+ if( GetScreen()->IsModify() )
+ {
+ int ii = DisplayExitDialog( this, _( "Save the changes in the library before closing?" ) );
+
+ switch( ii )
+ {
+ case wxID_NO:
+ break;
+
+ case wxID_YES:
+ if ( this->SaveActiveLibrary( false ) )
+ break;
+
+ // fall through: cancel the close because of an error
+
+ case wxID_CANCEL:
+ Event.Veto();
+ return;
+ }
+ GetScreen()->ClrModify();
+ }
+
+ PART_LIBS* libs = Prj().SchLibs();
+
+ BOOST_FOREACH( const PART_LIB& lib, *libs )
+ {
+ if( lib.IsModified() )
+ {
+ wxString msg = wxString::Format( _(
+ "Library '%s' was modified!\nDiscard changes?" ),
+ GetChars( lib.GetName() )
+ );
+
+ if( !IsOK( this, msg ) )
+ {
+ Event.Veto();
+ return;
+ }
+ }
+ }
+
+ Destroy();
+}
+
+
+double LIB_EDIT_FRAME::BestZoom()
+{
+ /* 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;
+ */
+ int dx, dy;
+
+ LIB_PART* part = GetCurPart();
+
+ if( part )
+ {
+ EDA_RECT boundingBox = part->GetBoundingBox( m_unit, m_convert );
+
+ dx = boundingBox.GetWidth();
+ dy = boundingBox.GetHeight();
+ SetScrollCenterPosition( wxPoint( 0, 0 ) );
+ }
+ else
+ {
+ const PAGE_INFO& pageInfo = GetScreen()->GetPageSettings();
+
+ dx = pageInfo.GetSizeIU().x;
+ dy = pageInfo.GetSizeIU().y;
+
+ SetScrollCenterPosition( wxPoint( 0, 0 ) );
+ }
+
+ wxSize size = m_canvas->GetClientSize();
+
+ // Reserve a 10% margin around component bounding box.
+ double margin_scale_factor = 0.8;
+ double zx =(double) dx / ( margin_scale_factor * (double)size.x );
+ double zy = (double) dy / ( margin_scale_factor * (double)size.y );
+
+ double bestzoom = std::max( zx, zy );
+
+ // 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];
+
+ return bestzoom;
+}
+
+
+void LIB_EDIT_FRAME::UpdateAliasSelectList()
+{
+ if( m_aliasSelectBox == NULL )
+ return;
+
+ m_aliasSelectBox->Clear();
+
+ LIB_PART* part = GetCurPart();
+
+ if( !part )
+ return;
+
+ m_aliasSelectBox->Append( part->GetAliasNames() );
+ m_aliasSelectBox->SetSelection( 0 );
+
+ int index = m_aliasSelectBox->FindString( m_aliasName );
+
+ if( index != wxNOT_FOUND )
+ m_aliasSelectBox->SetSelection( index );
+}
+
+
+void LIB_EDIT_FRAME::UpdatePartSelectList()
+{
+ if( m_partSelectBox == NULL )
+ return;
+
+ if( m_partSelectBox->GetCount() != 0 )
+ m_partSelectBox->Clear();
+
+ LIB_PART* part = GetCurPart();
+
+ if( !part || part->GetUnitCount() <= 1 )
+ {
+ m_unit = 1;
+ m_partSelectBox->Append( wxEmptyString );
+ }
+ else
+ {
+ for( int i = 0; i < part->GetUnitCount(); i++ )
+ {
+ wxString sub = LIB_PART::SubReference( i+1, false );
+ wxString unit = wxString::Format( _( "Unit %s" ), GetChars( sub ) );
+ m_partSelectBox->Append( unit );
+ }
+ }
+
+ // 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;
+
+ m_partSelectBox->SetSelection( ( m_unit > 0 ) ? m_unit - 1 : 0 );
+}
+
+
+void LIB_EDIT_FRAME::OnUpdateEditingPart( wxUpdateUIEvent& aEvent )
+{
+ LIB_PART* part = GetCurPart();
+
+ aEvent.Enable( part != NULL );
+
+ if( part && aEvent.GetEventObject() == m_drawToolBar )
+ aEvent.Check( GetToolId() == aEvent.GetId() );
+}
+
+
+void LIB_EDIT_FRAME::OnUpdateNotEditingPart( wxUpdateUIEvent& event )
+{
+ event.Enable( !GetCurPart() );
+}
+
+
+void LIB_EDIT_FRAME::OnUpdateUndo( wxUpdateUIEvent& event )
+{
+ event.Enable( GetCurPart() && GetScreen() &&
+ GetScreen()->GetUndoCommandCount() != 0 && !IsEditingDrawItem() );
+}
+
+
+void LIB_EDIT_FRAME::OnUpdateRedo( wxUpdateUIEvent& event )
+{
+ event.Enable( GetCurPart() && GetScreen() &&
+ GetScreen()->GetRedoCommandCount() != 0 && !IsEditingDrawItem() );
+}
+
+
+void LIB_EDIT_FRAME::OnUpdateSaveCurrentLib( wxUpdateUIEvent& event )
+{
+ PART_LIB* lib = GetCurLib();
+
+ event.Enable( lib && !lib->IsReadOnly()
+ && ( lib->IsModified() || GetScreen()->IsModify() ) );
+}
+
+
+void LIB_EDIT_FRAME::OnUpdateViewDoc( wxUpdateUIEvent& event )
+{
+ bool enable = false;
+
+ PART_LIB* lib = GetCurLib();
+ LIB_PART* part = GetCurPart();
+
+ if( part && lib )
+ {
+ LIB_ALIAS* alias = part->GetAlias( m_aliasName );
+
+ wxCHECK_RET( alias != NULL, wxT( "Alias <" ) + m_aliasName + wxT( "> not found." ) );
+
+ enable = !alias->GetDocFileName().IsEmpty();
+ }
+
+ event.Enable( enable );
+}
+
+
+void LIB_EDIT_FRAME::OnUpdatePinByPin( wxUpdateUIEvent& event )
+{
+ LIB_PART* part = GetCurPart();
+
+ event.Enable( part && ( part->GetUnitCount() > 1 || m_showDeMorgan ) );
+
+ event.Check( m_editPinsPerPartOrConvert );
+}
+
+void LIB_EDIT_FRAME::OnUpdatePinTable( wxUpdateUIEvent& event )
+{
+ LIB_PART* part = GetCurPart();
+ event.Enable( part != NULL );
+}
+
+void LIB_EDIT_FRAME::OnUpdatePartNumber( wxUpdateUIEvent& event )
+{
+ if( m_partSelectBox == NULL )
+ return;
+
+ 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 );
+}
+
+
+void LIB_EDIT_FRAME::OnUpdateDeMorganNormal( wxUpdateUIEvent& event )
+{
+ if( m_mainToolBar == NULL )
+ return;
+
+ LIB_PART* part = GetCurPart();
+
+ event.Enable( GetShowDeMorgan() || ( part && part->HasConversion() ) );
+ event.Check( m_convert <= 1 );
+}
+
+
+void LIB_EDIT_FRAME::OnUpdateDeMorganConvert( wxUpdateUIEvent& event )
+{
+ if( m_mainToolBar == NULL )
+ return;
+
+ LIB_PART* part = GetCurPart();
+
+ event.Enable( GetShowDeMorgan() || ( part && part->HasConversion() ) );
+ event.Check( m_convert > 1 );
+}
+
+
+void LIB_EDIT_FRAME::OnUpdateSelectAlias( wxUpdateUIEvent& event )
+{
+ if( m_aliasSelectBox == NULL )
+ return;
+
+ 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 );
+}
+
+
+void LIB_EDIT_FRAME::OnSelectAlias( wxCommandEvent& event )
+{
+ if( m_aliasSelectBox == NULL
+ || ( m_aliasSelectBox->GetStringSelection().CmpNoCase( m_aliasName ) == 0) )
+ return;
+
+ m_lastDrawItem = NULL;
+ m_aliasName = m_aliasSelectBox->GetStringSelection();
+
+ DisplayCmpDoc();
+ m_canvas->Refresh();
+}
+
+
+void LIB_EDIT_FRAME::OnSelectPart( wxCommandEvent& event )
+{
+ int i = event.GetSelection();
+
+ if( ( i == wxNOT_FOUND ) || ( ( i + 1 ) == m_unit ) )
+ return;
+
+ m_lastDrawItem = NULL;
+ m_unit = i + 1;
+ m_canvas->Refresh();
+ DisplayCmpDoc();
+}
+
+
+void LIB_EDIT_FRAME::OnViewEntryDoc( wxCommandEvent& event )
+{
+ LIB_PART* part = GetCurPart();
+
+ if( !part )
+ return;
+
+ wxString fileName;
+ LIB_ALIAS* alias = part->GetAlias( m_aliasName );
+
+ wxCHECK_RET( alias != NULL, wxT( "Alias not found." ) );
+
+ fileName = alias->GetDocFileName();
+
+ if( !fileName.IsEmpty() )
+ {
+ SEARCH_STACK* lib_search = Prj().SchSearchS();
+
+ GetAssociatedDocument( this, fileName, lib_search );
+ }
+}
+
+
+void LIB_EDIT_FRAME::OnSelectBodyStyle( wxCommandEvent& event )
+{
+ m_canvas->EndMouseCapture( ID_NO_TOOL_SELECTED, m_canvas->GetDefaultCursor() );
+
+ if( event.GetId() == ID_DE_MORGAN_NORMAL_BUTT )
+ m_convert = 1;
+ else
+ m_convert = 2;
+
+ m_lastDrawItem = NULL;
+ m_canvas->Refresh();
+}
+
+
+void LIB_EDIT_FRAME::Process_Special_Functions( wxCommandEvent& event )
+{
+ int id = event.GetId();
+ wxPoint pos;
+
+ m_canvas->SetIgnoreMouseEvents( true );
+
+ wxGetMousePosition( &pos.x, &pos.y );
+ pos.y += 20;
+
+ switch( id ) // Stop placement commands before handling new command.
+ {
+ case ID_POPUP_LIBEDIT_END_CREATE_ITEM:
+ case ID_LIBEDIT_EDIT_PIN:
+ 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:
+ case ID_POPUP_MIRROR_X_BLOCK:
+ case ID_POPUP_MIRROR_Y_BLOCK:
+ case ID_POPUP_ROTATE_BLOCK:
+ case ID_POPUP_PLACE_BLOCK:
+ case ID_POPUP_LIBEDIT_DELETE_CURRENT_POLY_SEGMENT:
+ break;
+
+ case ID_POPUP_LIBEDIT_CANCEL_EDITING:
+ if( m_canvas->IsMouseCaptured() )
+ m_canvas->EndMouseCapture();
+ else
+ m_canvas->EndMouseCapture( ID_NO_TOOL_SELECTED, m_canvas->GetDefaultCursor() );
+ break;
+
+ case ID_POPUP_LIBEDIT_DELETE_ITEM:
+ m_canvas->EndMouseCapture();
+ break;
+
+ default:
+ m_canvas->EndMouseCapture( ID_NO_TOOL_SELECTED, m_canvas->GetDefaultCursor(),
+ wxEmptyString );
+ break;
+ }
+
+ INSTALL_UNBUFFERED_DC( dc, m_canvas );
+
+ switch( id )
+ {
+ case ID_POPUP_LIBEDIT_CANCEL_EDITING:
+ break;
+
+ case ID_LIBEDIT_SELECT_CURRENT_LIB:
+ SelectActiveLibrary();
+ break;
+
+ case ID_LIBEDIT_SAVE_CURRENT_PART:
+ {
+ 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 );
+ }
+ break;
+
+ case ID_LIBEDIT_EDIT_PIN_BY_PIN:
+ m_editPinsPerPartOrConvert = m_mainToolBar->GetToolToggled( ID_LIBEDIT_EDIT_PIN_BY_PIN );
+ break;
+
+ case ID_POPUP_LIBEDIT_END_CREATE_ITEM:
+ m_canvas->MoveCursorToCrossHair();
+ if( m_drawItem )
+ {
+ EndDrawGraphicItem( &dc );
+ }
+ break;
+
+ case ID_POPUP_LIBEDIT_BODY_EDIT_ITEM:
+ if( m_drawItem )
+ {
+ m_canvas->CrossHairOff( &dc );
+
+ switch( m_drawItem->Type() )
+ {
+ case LIB_ARC_T:
+ case LIB_CIRCLE_T:
+ case LIB_RECTANGLE_T:
+ case LIB_POLYLINE_T:
+ EditGraphicSymbol( &dc, m_drawItem );
+ break;
+
+ case LIB_TEXT_T:
+ EditSymbolText( &dc, m_drawItem );
+ break;
+
+ default:
+ ;
+ }
+
+ m_canvas->CrossHairOn( &dc );
+ }
+ break;
+
+ case ID_POPUP_LIBEDIT_DELETE_CURRENT_POLY_SEGMENT:
+ {
+ // Delete the last created segment, while creating a polyline draw item
+ if( m_drawItem == NULL )
+ break;
+
+ 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;
+ }
+ break;
+
+ case ID_POPUP_LIBEDIT_DELETE_ITEM:
+ if( m_drawItem )
+ deleteItem( &dc );
+
+ break;
+
+ case ID_POPUP_LIBEDIT_MOVE_ITEM_REQUEST:
+ if( m_drawItem == NULL )
+ break;
+
+ if( m_drawItem->Type() == LIB_PIN_T )
+ StartMovePin( &dc );
+ else
+ StartMoveDrawSymbol( &dc );
+ break;
+
+ case ID_POPUP_LIBEDIT_MODIFY_ITEM:
+
+ if( m_drawItem == NULL )
+ break;
+
+ m_canvas->MoveCursorToCrossHair();
+ 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
+ )
+ {
+ StartModifyDrawSymbol( &dc );
+ }
+
+ break;
+
+ case ID_POPUP_LIBEDIT_FIELD_EDIT_ITEM:
+ if( m_drawItem == NULL )
+ break;
+
+ m_canvas->CrossHairOff( &dc );
+
+ if( m_drawItem->Type() == LIB_FIELD_T )
+ {
+ EditField( (LIB_FIELD*) m_drawItem );
+ }
+
+ m_canvas->MoveCursorToCrossHair();
+ m_canvas->CrossHairOn( &dc );
+ 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:
+ {
+ if( !m_drawItem || m_drawItem->Type() != LIB_PIN_T )
+ break;
+
+ LIB_PART* part = GetCurPart();
+
+ SaveCopyInUndoList( part );
+
+ GlobalSetPins( (LIB_PIN*) m_drawItem, id );
+ m_canvas->MoveCursorToCrossHair();
+ m_canvas->Refresh();
+ }
+ break;
+
+ case ID_POPUP_ZOOM_BLOCK:
+ m_canvas->SetAutoPanRequest( false );
+ GetScreen()->m_BlockLocate.SetCommand( BLOCK_ZOOM );
+ HandleBlockEnd( &dc );
+ break;
+
+ case ID_POPUP_DELETE_BLOCK:
+ m_canvas->SetAutoPanRequest( false );
+ GetScreen()->m_BlockLocate.SetCommand( BLOCK_DELETE );
+ m_canvas->MoveCursorToCrossHair();
+ HandleBlockEnd( &dc );
+ break;
+
+ case ID_POPUP_COPY_BLOCK:
+ m_canvas->SetAutoPanRequest( false );
+ GetScreen()->m_BlockLocate.SetCommand( BLOCK_COPY );
+ m_canvas->MoveCursorToCrossHair();
+ HandleBlockPlace( &dc );
+ break;
+
+ case ID_POPUP_SELECT_ITEMS_BLOCK:
+ m_canvas->SetAutoPanRequest( false );
+ GetScreen()->m_BlockLocate.SetCommand( BLOCK_SELECT_ITEMS_ONLY );
+ m_canvas->MoveCursorToCrossHair();
+ HandleBlockEnd( &dc );
+ break;
+
+ case ID_POPUP_MIRROR_Y_BLOCK:
+ m_canvas->SetAutoPanRequest( false );
+ GetScreen()->m_BlockLocate.SetCommand( BLOCK_MIRROR_Y );
+ m_canvas->MoveCursorToCrossHair();
+ HandleBlockPlace( &dc );
+ break;
+
+ case ID_POPUP_MIRROR_X_BLOCK:
+ m_canvas->SetAutoPanRequest( false );
+ GetScreen()->m_BlockLocate.SetCommand( BLOCK_MIRROR_X );
+ m_canvas->MoveCursorToCrossHair();
+ HandleBlockPlace( &dc );
+ break;
+
+ case ID_POPUP_ROTATE_BLOCK:
+ m_canvas->SetAutoPanRequest( false );
+ GetScreen()->m_BlockLocate.SetCommand( BLOCK_ROTATE );
+ m_canvas->MoveCursorToCrossHair();
+ HandleBlockPlace( &dc );
+ break;
+
+ case ID_POPUP_PLACE_BLOCK:
+ m_canvas->SetAutoPanRequest( false );
+ m_canvas->MoveCursorToCrossHair();
+ HandleBlockPlace( &dc );
+ break;
+
+ default:
+ DisplayError( this, wxT( "LIB_EDIT_FRAME::Process_Special_Functions error" ) );
+ break;
+ }
+
+ m_canvas->SetIgnoreMouseEvents( false );
+
+ if( GetToolId() == ID_NO_TOOL_SELECTED )
+ m_lastDrawItem = NULL;
+}
+
+
+void LIB_EDIT_FRAME::OnActivate( wxActivateEvent& event )
+{
+ EDA_DRAW_FRAME::OnActivate( event );
+}
+
+
+PART_LIB* LIB_EDIT_FRAME::GetCurLib()
+{
+ wxString name = Prj().GetRString( PROJECT::SCH_LIBEDIT_CUR_LIB );
+
+ if( !!name )
+ {
+ PART_LIB* lib = Prj().SchLibs()->FindLibrary( name );
+
+ 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 );
+ else
+ Prj().SetRString( PROJECT::SCH_LIBEDIT_CUR_LIB, aLib->GetName() );
+
+ return old;
+}
+
+
+LIB_PART* LIB_EDIT_FRAME::GetCurPart()
+{
+ if( !m_my_part )
+ {
+ wxString name = Prj().GetRString( PROJECT::SCH_LIBEDIT_CUR_PART );
+ LIB_PART* part;
+
+ 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 );
+ }
+
+ return m_my_part;
+}
+
+
+void LIB_EDIT_FRAME::SetCurPart( LIB_PART* aPart )
+{
+ delete m_my_part;
+ m_my_part = aPart; // take ownership here
+
+ // retain in case this wxFrame is re-opened later on the same PROJECT
+ Prj().SetRString( PROJECT::SCH_LIBEDIT_CUR_PART,
+ aPart ? aPart->GetName() : wxString() );
+}
+
+
+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;
+ }
+}
+
+
+void LIB_EDIT_FRAME::ClearTempCopyComponent()
+{
+ delete m_tempCopyComponent;
+ m_tempCopyComponent = NULL;
+}
+
+
+void LIB_EDIT_FRAME::EditSymbolText( wxDC* DC, LIB_ITEM* DrawItem )
+{
+ if ( ( DrawItem == NULL ) || ( DrawItem->Type() != LIB_TEXT_T ) )
+ return;
+
+ // Deleting old text
+ if( DC && !DrawItem->InEditMode() )
+ DrawItem->Draw( m_canvas, DC, wxPoint( 0, 0 ), UNSPECIFIED_COLOR, g_XorMode, NULL,
+ DefaultTransform );
+
+ DIALOG_LIB_EDIT_TEXT* frame = new DIALOG_LIB_EDIT_TEXT( this, (LIB_TEXT*) DrawItem );
+ frame->ShowModal();
+ frame->Destroy();
+ OnModify();
+
+ // Display new text
+ if( DC && !DrawItem->InEditMode() )
+ DrawItem->Draw( m_canvas, DC, wxPoint( 0, 0 ), UNSPECIFIED_COLOR, GR_DEFAULT_DRAWMODE, NULL,
+ DefaultTransform );
+}
+
+
+void LIB_EDIT_FRAME::OnEditComponentProperties( wxCommandEvent& event )
+{
+ bool partLocked = GetCurPart()->UnitsLocked();
+
+ DIALOG_EDIT_COMPONENT_IN_LIBRARY dlg( this );
+
+ if( dlg.ShowModal() == wxID_CANCEL )
+ return;
+
+ if( partLocked != GetCurPart()->UnitsLocked() )
+ {
+ // m_editPinsPerPartOrConvert is set to the better value, if m_UnitSelectionLocked
+ // has changed
+ m_editPinsPerPartOrConvert = GetCurPart()->UnitsLocked() ? true : false;
+ }
+
+ UpdateAliasSelectList();
+ UpdatePartSelectList();
+ DisplayLibInfos();
+ DisplayCmpDoc();
+ OnModify();
+ m_canvas->Refresh();
+}
+
+
+void LIB_EDIT_FRAME::OnCreateNewPartFromExisting( wxCommandEvent& event )
+{
+ LIB_PART* part = GetCurPart();
+
+ wxCHECK_RET( part, wxT( "Cannot create new part from non-existent current part." ) );
+
+ INSTALL_UNBUFFERED_DC( dc, m_canvas );
+ m_canvas->CrossHairOff( &dc );
+
+ EditField( &part->GetValueField() );
+
+ m_canvas->MoveCursorToCrossHair();
+ m_canvas->CrossHairOn( &dc );
+}
+
+
+void LIB_EDIT_FRAME::OnSelectTool( wxCommandEvent& aEvent )
+{
+ int id = aEvent.GetId();
+
+ if( GetToolId() == ID_NO_TOOL_SELECTED )
+ m_lastDrawItem = NULL;
+
+ m_canvas->EndMouseCapture( ID_NO_TOOL_SELECTED, m_canvas->GetDefaultCursor(),
+ wxEmptyString );
+
+ LIB_PART* part = GetCurPart();
+
+ switch( id )
+ {
+ case ID_NO_TOOL_SELECTED:
+ SetToolID( id, m_canvas->GetDefaultCursor(), wxEmptyString );
+ break;
+
+ case ID_LIBEDIT_PIN_BUTT:
+ if( part )
+ {
+ SetToolID( id, wxCURSOR_PENCIL, _( "Add pin" ) );
+ }
+ else
+ {
+ SetToolID( id, wxCURSOR_ARROW, _( "Set pin options" ) );
+
+ wxCommandEvent cmd( wxEVT_COMMAND_MENU_SELECTED );
+
+ cmd.SetId( ID_LIBEDIT_EDIT_PIN );
+ GetEventHandler()->ProcessEvent( cmd );
+ SetToolID( ID_NO_TOOL_SELECTED, m_canvas->GetDefaultCursor(), wxEmptyString );
+ }
+ 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:
+ SetToolID( id, m_canvas->GetDefaultCursor(), _( "Import" ) );
+ LoadOneSymbol();
+ SetToolID( ID_NO_TOOL_SELECTED, m_canvas->GetDefaultCursor(), wxEmptyString );
+ break;
+
+ case ID_LIBEDIT_EXPORT_BODY_BUTT:
+ SetToolID( id, m_canvas->GetDefaultCursor(), _( "Export" ) );
+ SaveOneSymbol();
+ SetToolID( ID_NO_TOOL_SELECTED, m_canvas->GetDefaultCursor(), wxEmptyString );
+ break;
+
+ case ID_LIBEDIT_DELETE_ITEM_BUTT:
+ if( !part )
+ {
+ wxBell();
+ break;
+ }
+
+ SetToolID( id, wxCURSOR_BULLSEYE, _( "Delete item" ) );
+ break;
+
+ default:
+ break;
+ }
+
+ m_canvas->SetIgnoreMouseEvents( false );
+}
+
+
+void LIB_EDIT_FRAME::OnRotateItem( wxCommandEvent& aEvent )
+{
+ if( m_drawItem == NULL )
+ return;
+
+ if( !m_drawItem->InEditMode() )
+ {
+ LIB_PART* part = GetCurPart();
+
+ SaveCopyInUndoList( part );
+ m_drawItem->SetUnit( m_unit );
+ }
+
+ m_drawItem->Rotate();
+ OnModify();
+
+ if( !m_drawItem->InEditMode() )
+ m_drawItem->ClearFlags();
+
+ m_canvas->Refresh();
+
+ if( GetToolId() == ID_NO_TOOL_SELECTED )
+ m_lastDrawItem = NULL;
+}
+
+
+LIB_ITEM* LIB_EDIT_FRAME::LocateItemUsingCursor( const wxPoint& aPosition,
+ const KICAD_T aFilterList[] )
+{
+ LIB_PART* part = GetCurPart();
+
+ if( !part )
+ return NULL;
+
+ LIB_ITEM* item = locateItem( aPosition, aFilterList );
+
+ wxPoint pos = GetNearestGridPosition( aPosition );
+
+ if( item == NULL && aPosition != pos )
+ item = locateItem( pos, aFilterList );
+
+ return item;
+}
+
+
+LIB_ITEM* LIB_EDIT_FRAME::locateItem( const wxPoint& aPosition, const KICAD_T aFilterList[] )
+{
+ LIB_PART* part = GetCurPart();
+
+ if( !part )
+ return NULL;
+
+ LIB_ITEM* item = NULL;
+
+ m_collectedItems.Collect( part->GetDrawItemList(), aFilterList, aPosition,
+ 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++ )
+ {
+ wxString text = m_collectedItems[i]->GetSelectMenuText();
+ BITMAP_DEF xpm = m_collectedItems[i]->GetMenuImage();
+
+ AddMenuItem( &selectMenu, ID_SELECT_ITEM_START + i, text, KiBitmap( xpm ) );
+ }
+
+ // Set to NULL in case user aborts the clarification context menu.
+ m_drawItem = NULL;
+ m_canvas->SetAbortRequest( true ); // Changed to false if an item is selected
+ PopupMenu( &selectMenu );
+ m_canvas->MoveCursorToCrossHair();
+ item = m_drawItem;
+ }
+ }
+
+ if( item )
+ {
+ MSG_PANEL_ITEMS items;
+ item->GetMsgPanelInfo( items );
+ SetMsgPanel( items );
+ }
+ else
+ {
+ ClearMsgPanel();
+ }
+
+ return item;
+}
+
+
+void LIB_EDIT_FRAME::deleteItem( wxDC* aDC )
+{
+ wxCHECK_RET( m_drawItem != NULL, wxT( "No drawing item selected to delete." ) );
+
+ m_canvas->CrossHairOff( aDC );
+
+ LIB_PART* part = GetCurPart();
+
+ SaveCopyInUndoList( part );
+
+ if( m_drawItem->Type() == LIB_PIN_T )
+ {
+ LIB_PIN* pin = (LIB_PIN*) m_drawItem;
+ wxPoint pos = pin->GetPosition();
+
+ part->RemoveDrawItem( (LIB_ITEM*) pin, m_canvas, aDC );
+
+ if( SynchronizePins() )
+ {
+ LIB_PIN* tmp = part->GetNextPin();
+
+ while( tmp != NULL )
+ {
+ pin = tmp;
+ tmp = part->GetNextPin( pin );
+
+ if( pin->GetPosition() != pos )
+ continue;
+
+ part->RemoveDrawItem( (LIB_ITEM*) pin );
+ }
+ }
+
+ m_canvas->Refresh();
+ }
+ else
+ {
+ if( m_canvas->IsMouseCaptured() )
+ {
+ m_canvas->CallEndMouseCapture( aDC );
+ }
+ else
+ {
+ part->RemoveDrawItem( m_drawItem, m_canvas, aDC );
+ m_canvas->Refresh();
+ }
+ }
+
+ m_drawItem = NULL;
+ m_lastDrawItem = NULL;
+ OnModify();
+ m_canvas->CrossHairOn( aDC );
+}
+
+
+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];
+ m_canvas->SetAbortRequest( false );
+ m_drawItem = item;
+ }
+}
+
+void LIB_EDIT_FRAME::OnOpenPinTable( wxCommandEvent& aEvent )
+{
+ LIB_PART* part = GetCurPart();
+
+ DIALOG_LIB_EDIT_PIN_TABLE dlg( this, *part );
+
+ if( dlg.ShowModal() == wxID_CANCEL )
+ return;
+
+ return;
+}
+
+bool LIB_EDIT_FRAME::SynchronizePins()
+{
+ LIB_PART* part = GetCurPart();
+
+ return !m_editPinsPerPartOrConvert && ( part &&
+ ( part->HasConversion() || part->IsMulti() ) );
+}