summaryrefslogtreecommitdiff
path: root/pcbnew/loadcmp.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'pcbnew/loadcmp.cpp')
-rw-r--r--pcbnew/loadcmp.cpp580
1 files changed, 580 insertions, 0 deletions
diff --git a/pcbnew/loadcmp.cpp b/pcbnew/loadcmp.cpp
new file mode 100644
index 0000000..3bf9710
--- /dev/null
+++ b/pcbnew/loadcmp.cpp
@@ -0,0 +1,580 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2012 Jean-Pierre Charras, jean-pierre.charras@ujf-grenoble.fr
+ * Copyright (C) 2012 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
+ * Copyright (C) 1992-2015 KiCad Developers, see AUTHORS.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 pcbnew/loadcmp.cpp
+ * @brief Footprints selection and loading functions.
+ */
+
+#include <boost/bind.hpp>
+
+#include <fctsys.h>
+#include <class_drawpanel.h>
+#include <pcb_draw_panel_gal.h>
+#include <confirm.h>
+#include <eda_doc.h>
+#include <kicad_string.h>
+#include <pgm_base.h>
+#include <kiway.h>
+#include <wxPcbStruct.h>
+#include <dialog_helpers.h>
+#include <filter_reader.h>
+#include <gr_basic.h>
+#include <macros.h>
+#include <fp_lib_table.h>
+#include <fpid.h>
+
+#include <class_board.h>
+#include <class_module.h>
+#include <io_mgr.h>
+
+#include <pcbnew.h>
+#include <module_editor_frame.h>
+#include <footprint_info.h>
+#include <dialog_get_component.h>
+#include <modview_frame.h>
+#include <wildcards_and_files_ext.h>
+#include <class_pcb_layer_widget.h>
+
+
+static void DisplayCmpDoc( wxString& aName, void* aData );
+
+static FOOTPRINT_LIST MList;
+
+static void clearModuleItemFlags( BOARD_ITEM* aItem )
+{
+ aItem->ClearFlags();
+}
+
+bool FOOTPRINT_EDIT_FRAME::Load_Module_From_BOARD( MODULE* aModule )
+{
+ MODULE* newModule;
+ PCB_EDIT_FRAME* frame = (PCB_EDIT_FRAME*) Kiway().Player( FRAME_PCB, false );
+
+ if( frame == NULL ) // happens if no board editor opened
+ return false;
+
+ if( aModule == NULL )
+ {
+ if( ! frame->GetBoard() || ! frame->GetBoard()->m_Modules )
+ return false;
+
+ aModule = SelectFootprint( frame->GetBoard() );
+ }
+
+ if( aModule == NULL )
+ return false;
+
+ SetCurItem( NULL );
+
+ Clear_Pcb( false );
+
+ GetBoard()->m_Status_Pcb = 0;
+ newModule = new MODULE( *aModule );
+ newModule->SetParent( GetBoard() );
+ newModule->SetLink( aModule->GetTimeStamp() );
+
+ aModule = newModule;
+
+ newModule->ClearFlags();
+ newModule->RunOnChildren( boost::bind( &clearModuleItemFlags, _1 ) );
+
+ GetBoard()->Add( newModule );
+
+ // Clear references to any net info, because the footprint editor
+ // does know any thing about nets handled by the current edited board.
+ // Morever we do not want to save any reference to an unknown net when
+ // saving the footprint in lib cache
+ // so we force the ORPHANED dummy net info for all pads
+ newModule->ClearAllNets();
+
+ SetCrossHairPosition( wxPoint( 0, 0 ) );
+ PlaceModule( newModule, NULL );
+ newModule->SetPosition( wxPoint( 0, 0 ) ); // cursor in GAL may not be initialized at the moment
+
+ // Put it on FRONT layer,
+ // because this is the default in ModEdit, and in libs
+ if( newModule->GetLayer() != F_Cu )
+ newModule->Flip( newModule->GetPosition() );
+
+ // Put it in orientation 0,
+ // because this is the default orientation in ModEdit, and in libs
+ Rotate_Module( NULL, newModule, 0, false );
+ GetScreen()->ClrModify();
+ Zoom_Automatique( false );
+
+ if( IsGalCanvasActive() )
+ updateView();
+
+ return true;
+}
+
+
+wxString PCB_BASE_FRAME::SelectFootprintFromLibBrowser()
+{
+ // Close the current non-modal Lib browser if opened, and open a new one, in "modal" mode:
+ FOOTPRINT_VIEWER_FRAME* viewer;
+
+ viewer = (FOOTPRINT_VIEWER_FRAME*) Kiway().Player( FRAME_PCB_MODULE_VIEWER, false );
+
+ if( viewer )
+ viewer->Destroy();
+
+ viewer = (FOOTPRINT_VIEWER_FRAME*) Kiway().Player( FRAME_PCB_MODULE_VIEWER_MODAL, true );
+
+ wxString fpid;
+
+ int ret = viewer->ShowModal( &fpid, this );
+ (void) ret; // make static analyser quiet
+
+ //DBG(printf("%s: fpid:'%s'\n", __func__, TO_UTF8( fpid ) );)
+
+ viewer->Destroy();
+
+ return fpid;
+}
+
+
+MODULE* PCB_BASE_FRAME::LoadModuleFromLibrary( const wxString& aLibrary,
+ FP_LIB_TABLE* aTable,
+ bool aUseFootprintViewer,
+ wxDC* aDC )
+{
+ MODULE* module = NULL;
+ wxPoint curspos = GetCrossHairPosition();
+ wxString moduleName, keys;
+ wxString libName = aLibrary;
+ bool allowWildSeach = true;
+
+ static wxArrayString HistoryList;
+ static wxString lastComponentName;
+
+ // Ask for a component name or key words
+ DIALOG_GET_COMPONENT dlg( this, HistoryList, _( "Load Footprint" ), aUseFootprintViewer );
+
+ dlg.SetComponentName( lastComponentName );
+
+ if( dlg.ShowModal() == wxID_CANCEL )
+ return NULL;
+
+ if( dlg.m_GetExtraFunction )
+ {
+ // SelectFootprintFromLibBrowser() returns the "full" footprint name, i.e.
+ // <lib_name>/<footprint name> or FPID format "lib_name:fp_name:rev#"
+ moduleName = SelectFootprintFromLibBrowser();
+ }
+ else
+ {
+ moduleName = dlg.GetComponentName();
+ }
+
+ if( moduleName.IsEmpty() ) // Cancel command
+ {
+ m_canvas->MoveCursorToCrossHair();
+ return NULL;
+ }
+
+ if( dlg.IsKeyword() ) // Selection by keywords
+ {
+ allowWildSeach = false;
+ keys = moduleName;
+ moduleName = SelectFootprint( this, libName, wxEmptyString, keys, aTable );
+
+ if( moduleName.IsEmpty() ) // Cancel command
+ {
+ m_canvas->MoveCursorToCrossHair();
+ return NULL;
+ }
+ }
+ else if( moduleName.Contains( wxT( "?" ) )
+ || moduleName.Contains( wxT( "*" ) ) ) // Selection wild card
+ {
+ allowWildSeach = false;
+ moduleName = SelectFootprint( this, libName, moduleName, wxEmptyString, aTable );
+
+ if( moduleName.IsEmpty() )
+ {
+ m_canvas->MoveCursorToCrossHair();
+ return NULL; // Cancel command.
+ }
+ }
+
+ FPID fpid;
+
+ wxCHECK_MSG( fpid.Parse( moduleName ) < 0, NULL,
+ wxString::Format( wxT( "Could not parse FPID string '%s'." ),
+ GetChars( moduleName ) ) );
+
+ try
+ {
+ module = loadFootprint( fpid );
+ }
+ catch( const IO_ERROR& ioe )
+ {
+ wxLogDebug( wxT( "An error occurred attemping to load footprint '%s'.\n\nError: %s" ),
+ fpid.Format().c_str(), GetChars( ioe.errorText ) );
+ }
+
+ if( !module && allowWildSeach ) // Search with wild card
+ {
+ allowWildSeach = false;
+
+ wxString wildname = wxChar( '*' ) + moduleName + wxChar( '*' );
+ moduleName = wildname;
+
+ moduleName = SelectFootprint( this, libName, moduleName, wxEmptyString, aTable );
+
+ if( moduleName.IsEmpty() )
+ {
+ m_canvas->MoveCursorToCrossHair();
+ return NULL; // Cancel command.
+ }
+ else
+ {
+ FPID fpid;
+
+ wxCHECK_MSG( fpid.Parse( moduleName ) < 0, NULL,
+ wxString::Format( wxT( "Could not parse FPID string '%s'." ),
+ GetChars( moduleName ) ) );
+
+ try
+ {
+ module = loadFootprint( fpid );
+ }
+ catch( const IO_ERROR& ioe )
+ {
+ wxLogDebug( wxT( "An error occurred attemping to load footprint '%s'.\n\nError: %s" ),
+ fpid.Format().c_str(), GetChars( ioe.errorText ) );
+ }
+ }
+ }
+
+ SetCrossHairPosition( curspos );
+ m_canvas->MoveCursorToCrossHair();
+
+ if( module )
+ {
+ GetBoard()->Add( module, ADD_APPEND );
+
+ lastComponentName = moduleName;
+ AddHistoryComponentName( HistoryList, moduleName );
+
+ module->SetFlags( IS_NEW );
+ module->SetLink( 0 );
+
+ if( IsGalCanvasActive() )
+ module->SetPosition( wxPoint( 0, 0 ) ); // cursor in GAL may not be initialized at the moment
+ else
+ module->SetPosition( curspos );
+
+ module->SetTimeStamp( GetNewTimeStamp() );
+ GetBoard()->m_Status_Pcb = 0;
+
+ // Put it on FRONT layer,
+ // (Can be stored flipped if the lib is an archive built from a board)
+ if( module->IsFlipped() )
+ module->Flip( module->GetPosition() );
+
+ // Place it in orientation 0,
+ // even if it is not saved with orientation 0 in lib
+ // (Can happen if the lib is an archive built from a board)
+ Rotate_Module( NULL, module, 0, false );
+
+ RecalculateAllTracksNetcode();
+
+ if( aDC )
+ module->Draw( m_canvas, aDC, GR_OR );
+ }
+
+ return module;
+}
+
+
+MODULE* PCB_BASE_FRAME::LoadFootprint( const FPID& aFootprintId )
+{
+ MODULE* module = NULL;
+
+ try
+ {
+ module = loadFootprint( aFootprintId );
+ }
+ catch( const IO_ERROR& ioe )
+ {
+ wxLogDebug( wxT( "An error occurred attemping to load footprint '%s'.\n\nError: %s" ),
+ aFootprintId.Format().c_str(), GetChars( ioe.errorText ) );
+ }
+
+ return module;
+}
+
+
+MODULE* PCB_BASE_FRAME::loadFootprint( const FPID& aFootprintId )
+ throw( IO_ERROR, PARSE_ERROR, boost::interprocess::lock_exception )
+{
+ FP_LIB_TABLE* fptbl = Prj().PcbFootprintLibs();
+
+ wxCHECK_MSG( fptbl, NULL, wxT( "Cannot look up FPID in NULL FP_LIB_TABLE." ) );
+
+ MODULE* module = fptbl->FootprintLoadWithOptionalNickname( aFootprintId );
+
+ // If the module is found, clear all net info,
+ // to be sure there is no broken links
+ // to any netinfo list (should be not needed, but it can be edited from
+ // the footprint editor )
+ if( module )
+ module->ClearAllNets();
+
+ return module;
+}
+
+
+wxString PCB_BASE_FRAME::SelectFootprint( EDA_DRAW_FRAME* aWindow,
+ const wxString& aLibraryName,
+ const wxString& aMask,
+ const wxString& aKeyWord,
+ FP_LIB_TABLE* aTable )
+{
+ static wxString oldName; // Save the name of the last module loaded.
+
+ wxString fpname;
+ wxString msg;
+ wxArrayString libraries;
+
+ std::vector< wxArrayString > rows;
+
+ wxASSERT( aTable != NULL );
+
+ MList.ReadFootprintFiles( aTable, !aLibraryName ? NULL : &aLibraryName );
+
+ if( MList.GetErrorCount() )
+ {
+ MList.DisplayErrors( this );
+ return wxEmptyString;
+ }
+
+ if( MList.GetCount() == 0 )
+ {
+ wxString tmp;
+
+ for( unsigned i = 0; i < libraries.GetCount(); i++ )
+ {
+ tmp += libraries[i] + wxT( "\n" );
+ }
+
+ msg.Printf( _( "No footprints could be read from library file(s):\n\n%s\nin any of "
+ "the library search paths. Verify your system is configured properly "
+ "so the footprint libraries can be found." ), GetChars( tmp ) );
+ DisplayError( aWindow, msg );
+ return wxEmptyString;
+ }
+
+ if( !aKeyWord.IsEmpty() ) // Create a list of modules found by keyword.
+ {
+ for( unsigned ii = 0; ii < MList.GetCount(); ii++ )
+ {
+ if( KeyWordOk( aKeyWord, MList.GetItem( ii ).GetKeywords() ) )
+ {
+ wxArrayString cols;
+ cols.Add( MList.GetItem( ii ).GetFootprintName() );
+ cols.Add( MList.GetItem( ii ).GetNickname() );
+ rows.push_back( cols );
+ }
+ }
+ }
+ else if( !aMask.IsEmpty() ) // Create a list of modules found by pattern
+ {
+ for( unsigned ii = 0; ii < MList.GetCount(); ii++ )
+ {
+ const wxString& candidate = MList.GetItem( ii ).GetFootprintName();
+
+ if( WildCompareString( aMask, candidate, false ) )
+ {
+ wxArrayString cols;
+ cols.Add( MList.GetItem( ii ).GetFootprintName() );
+ cols.Add( MList.GetItem( ii ).GetNickname() );
+ rows.push_back( cols );
+ }
+ }
+ }
+ else // Create the full list of modules
+ {
+ for( unsigned ii = 0; ii < MList.GetCount(); ii++ )
+ {
+ wxArrayString cols;
+ cols.Add( MList.GetItem( ii ).GetFootprintName() );
+ cols.Add( MList.GetItem( ii ).GetNickname() );
+ rows.push_back( cols );
+ }
+ }
+
+ if( !rows.empty() )
+ {
+ wxArrayString headers;
+
+ headers.Add( _( "Footprint" ) );
+ headers.Add( _( "Library" ) );
+
+ msg.Printf( _( "Footprints [%d items]" ), (int) rows.size() );
+
+ EDA_LIST_DIALOG dlg( aWindow, msg, headers, rows, oldName, DisplayCmpDoc );
+
+ if( dlg.ShowModal() == wxID_OK )
+ {
+ fpname = dlg.GetTextSelection();
+
+ fpname = dlg.GetTextSelection( 1 ) + wxT( ":" ) + fpname;
+
+ SkipNextLeftButtonReleaseEvent();
+ }
+ else
+ fpname.Empty();
+ }
+ else
+ {
+ DisplayError( aWindow, _( "No footprint found." ) );
+ fpname.Empty();
+ }
+
+ if( fpname != wxEmptyString )
+ oldName = fpname;
+
+ wxLogDebug( wxT( "Footprint '%s' was selected." ), GetChars( fpname ) );
+
+ return fpname;
+}
+
+
+static void DisplayCmpDoc( wxString& aName, void* aData )
+{
+ FOOTPRINT_INFO* module_info = MList.GetModuleInfo( aName );
+
+ if( !module_info )
+ {
+ aName.Empty();
+ return;
+ }
+
+ aName = _( "Description: " ) + module_info->GetDoc();
+ aName += _( "\nKey words: " ) + module_info->GetKeywords();
+}
+
+
+MODULE* FOOTPRINT_EDIT_FRAME::SelectFootprint( BOARD* aPcb )
+{
+ static wxString oldName; // Save name of last module selected.
+
+ wxString fpname;
+ wxString msg;
+ wxArrayString listnames;
+ MODULE* module = aPcb->m_Modules;
+
+ for( ; module; module = module->Next() )
+ listnames.Add( module->GetReference() );
+
+ msg.Printf( _( "Footprints [%u items]" ), (unsigned) listnames.GetCount() );
+
+ wxArrayString headers;
+
+ headers.Add( _( "Footprint" ) );
+
+ std::vector<wxArrayString> itemsToDisplay;
+
+ // Conversion from wxArrayString to vector of ArrayString
+ for( unsigned i = 0; i < listnames.GetCount(); i++ )
+ {
+ wxArrayString item;
+
+ item.Add( listnames[i] );
+ itemsToDisplay.push_back( item );
+ }
+
+ EDA_LIST_DIALOG dlg( this, msg, headers, itemsToDisplay, wxEmptyString, NULL, NULL, SORT_LIST );
+
+ if( dlg.ShowModal() == wxID_OK )
+ fpname = dlg.GetTextSelection();
+ else
+ return NULL;
+
+ oldName = fpname;
+
+ module = aPcb->m_Modules;
+
+ for( ; module; module = module->Next() )
+ {
+ if( fpname == module->GetReference() )
+ break;
+ }
+
+ return module;
+}
+
+
+void FOOTPRINT_EDIT_FRAME::OnSaveLibraryAs( wxCommandEvent& aEvent )
+{
+ wxString curLibPath = getLibPath();
+ wxString dstLibPath = CreateNewLibrary();
+
+ if( !dstLibPath )
+ return; // user aborted in CreateNewLibrary()
+
+ wxBusyCursor dummy;
+ wxString msg;
+
+ IO_MGR::PCB_FILE_T dstType = IO_MGR::GuessPluginTypeFromLibPath( dstLibPath );
+ IO_MGR::PCB_FILE_T curType = IO_MGR::GuessPluginTypeFromLibPath( curLibPath );
+
+ try
+ {
+ PLUGIN::RELEASER cur( IO_MGR::PluginFind( curType ) );
+ PLUGIN::RELEASER dst( IO_MGR::PluginFind( dstType ) );
+
+ wxArrayString mods = cur->FootprintEnumerate( curLibPath );
+
+ for( unsigned i = 0; i < mods.size(); ++i )
+ {
+ std::auto_ptr<MODULE> m( cur->FootprintLoad( curLibPath, mods[i] ) );
+ dst->FootprintSave( dstLibPath, m.get() );
+
+ msg = wxString::Format( _( "Footprint '%s' saved" ),
+ GetChars( mods[i] ) );
+ SetStatusText( msg );
+
+ // m is deleted here by auto_ptr.
+ }
+ }
+ catch( const IO_ERROR& ioe )
+ {
+ DisplayError( this, ioe.errorText );
+ return;
+ }
+
+ msg = wxString::Format(
+ _( "Footprint library '%s' saved as '%s'." ),
+ GetChars( curLibPath ), GetChars( dstLibPath ) );
+
+ DisplayInfoMessage( this, msg );
+
+ SetStatusText( wxEmptyString );
+}