/** * @file cvpcb/readwrite_dlgs.cpp */ /* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2018 Jean-Pierre Charras, jean-pierre.charras * Copyright (C) 2011-2016 Wayne Stambaugh * Copyright (C) 1992-2018 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 */ #include #include #include #include #include #include #include #include #include #include #include #include #include void CVPCB_MAINFRAME::SetNewPkg( const wxString& aFootprintName ) { COMPONENT* component; int componentIndex; if( m_netlist.IsEmpty() ) return; // If no component is selected, select the first one if( m_compListBox->GetFirstSelected() < 0 ) { componentIndex = 0; m_compListBox->SetSelection( componentIndex, true ); } // iterate over the selection while( m_compListBox->GetFirstSelected() != -1 ) { // Get the component for the current iteration componentIndex = m_compListBox->GetFirstSelected(); component = m_netlist.GetComponent( componentIndex ); if( component == NULL ) return; SetNewPkg( aFootprintName, componentIndex ); m_compListBox->SetSelection( componentIndex, false ); } // select the next component, if there is one if( componentIndex < (m_compListBox->GetCount() - 1) ) componentIndex++; m_compListBox->SetSelection( componentIndex, true ); // update the statusbar DisplayStatus(); } void CVPCB_MAINFRAME::SetNewPkg( const wxString& aFootprintName, int aIndex ) { COMPONENT* component; if( m_netlist.IsEmpty() ) return; component = m_netlist.GetComponent( aIndex ); if( component == NULL ) return; LIB_ID fpid; if( !aFootprintName.IsEmpty() ) { wxCHECK_RET( fpid.Parse( aFootprintName, LIB_ID::ID_PCB ) < 0, wxString::Format( _( "\"%s\" is not a valid LIB_ID." ), aFootprintName ) ); } component->SetFPID( fpid ); // create the new component description wxString description = wxString::Format( CMP_FORMAT, aIndex + 1, GetChars( component->GetReference() ), GetChars( component->GetValue() ), GetChars( FROM_UTF8( component->GetFPID().Format().c_str() ) ) ); // Set the new description and deselect the processed component m_compListBox->SetString( aIndex, description ); // Mark this "session" as modified m_modified = true; // update the statusbar DisplayStatus(); } /// Return true if the resultant LIB_ID has a certain nickname. The guess /// is only made if this footprint resides in only one library. /// @return int - 0 on success, 1 on not found, 2 on ambiguous i.e. multiple matches static int guessNickname( FP_LIB_TABLE* aTbl, LIB_ID* aFootprintId ) { if( aFootprintId->GetLibNickname().size() ) return 0; wxString nick; wxString fpname = aFootprintId->GetLibItemName(); std::vector nicks = aTbl->GetLogicalLibs(); // Search each library going through libraries alphabetically. for( unsigned libNdx = 0; libNdxFootprintEnumerate( fpnames, nicks[libNdx] ); for( unsigned nameNdx = 0; nameNdxSetLibNickname( nick ); return 0; } return 1; } bool CVPCB_MAINFRAME::ReadNetListAndFpFiles( const std::string& aNetlist ) { wxString msg; bool hasMissingNicks = false; ReadSchematicNetlist( aNetlist ); if( m_compListBox == NULL ) return false; LoadProjectFile(); wxSafeYield(); LoadFootprintFiles(); BuildFOOTPRINTS_LISTBOX(); BuildLIBRARY_LISTBOX(); m_compListBox->Clear(); if( m_netlist.AnyFootprintsLinked() ) { for( unsigned i = 0; i < m_netlist.GetCount(); i++ ) { COMPONENT* component = m_netlist.GetComponent( i ); if( component->GetFPID().empty() ) continue; if( component->GetFPID().IsLegacy() ) hasMissingNicks = true; } } // Check if footprint links were generated before the footprint library table was implemented. if( hasMissingNicks ) { msg = _( "Some of the assigned footprints are legacy entries (are missing lib nicknames). " "Would you like CvPcb to attempt to convert them to the new required LIB_ID format? " "(If you answer no, then these assignments will be cleared out and you will " "have to re-assign these footprints yourself.)" ); if( IsOK( this, msg ) ) { msg.Clear(); try { for( unsigned i = 0; i < m_netlist.GetCount(); i++ ) { COMPONENT* component = m_netlist.GetComponent( i ); if( component->GetFPID().IsLegacy() ) { // get this first here, it's possibly obsoleted if we get it too soon. FP_LIB_TABLE* tbl = Prj().PcbFootprintLibs( Kiway() ); int guess = guessNickname( tbl, (LIB_ID*) &component->GetFPID() ); switch( guess ) { case 0: DBG(printf("%s: guessed OK ref:%s fpid:%s\n", __func__, TO_UTF8( component->GetReference() ), component->GetFPID().Format().c_str() );) m_modified = true; break; case 1: msg += wxString::Format( _( "Component \"%s\" footprint \"%s\" was not found in any library.\n" ), GetChars( component->GetReference() ), GetChars( component->GetFPID().GetLibItemName() ) ); break; case 2: msg += wxString::Format( _( "Component \"%s\" footprint \"%s\" was found in multiple libraries.\n" ), GetChars( component->GetReference() ), GetChars( component->GetFPID().GetLibItemName() ) ); break; } } } } catch( const IO_ERROR& ioe ) { msg = ioe.What(); msg += wxT( "\n\n" ); msg += _( "First check your footprint library table entries." ); wxMessageBox( msg, _( "Problematic Footprint Library Tables" ) ); return false; } if( msg.size() ) { HTML_MESSAGE_BOX dlg( this, wxEmptyString ); dlg.MessageSet( _( "The following errors occurred attempting to convert the " "footprint assignments:\n\n" ) ); dlg.ListSet( msg ); dlg.MessageSet( _( "\nYou will need to reassign them manually if you want them " "to be updated correctly the next time you import the " "netlist in Pcbnew." ) ); #if 1 dlg.ShowModal(); #else dlg.Fit(); dlg.Show( true ); // modeless lets user watch while fixing the problems, but its not working. #endif } } else { // Clear the legacy footprint assignments. for( unsigned i = 0; i < m_netlist.GetCount(); i++ ) { COMPONENT* component = m_netlist.GetComponent( i ); if( component->GetFPID().IsLegacy() ) { component->SetFPID( LIB_ID() /* empty */ ); m_modified = true; } } } } // Display a dialog to select footprint selection, if the netlist // and the .cmp file give 2 different valid footprints std::vector m_indexes; // indexes of footprints in netlist for( unsigned ii = 0; ii < m_netlist.GetCount(); ii++ ) { COMPONENT* component = m_netlist.GetComponent( ii ); if( component->GetAltFPID().empty() ) continue; if( component->GetFPID().IsLegacy() || component->GetAltFPID().IsLegacy()) continue; m_indexes.push_back( ii ); } // If a n assignment conflict is found, // open a dialog to chose between schematic assignment // and .cmp file assignment: if( m_indexes.size() > 0 ) { DIALOG_FP_CONFLICT_ASSIGNMENT_SELECTOR dlg( this ); for( unsigned ii = 0; ii < m_indexes.size(); ii++ ) { COMPONENT* component = m_netlist.GetComponent( m_indexes[ii] ); wxString cmpfpid = component->GetFPID().Format(); wxString schfpid = component->GetAltFPID().Format(); dlg.Add( component->GetReference(), schfpid, cmpfpid ); } if( dlg.ShowModal() == wxID_OK ) { // Update the fp selection: for( unsigned ii = 0; ii < m_indexes.size(); ii++ ) { COMPONENT* component = m_netlist.GetComponent( m_indexes[ii] ); int choice = dlg.GetSelection( component->GetReference() ); if( choice == 0 ) // the schematic (alt fpid) is chosen: component->SetFPID( component->GetAltFPID() ); } } } // Populates the component list box: for( unsigned i = 0; i < m_netlist.GetCount(); i++ ) { COMPONENT* component = m_netlist.GetComponent( i ); msg.Printf( CMP_FORMAT, m_compListBox->GetCount() + 1, GetChars( component->GetReference() ), GetChars( component->GetValue() ), GetChars( FROM_UTF8( component->GetFPID().Format().c_str() ) ) ); m_compListBox->AppendLine( msg ); } if( !m_netlist.IsEmpty() ) m_compListBox->SetSelection( 0, true ); DisplayStatus(); return true; } void CVPCB_MAINFRAME::SaveFootprintAssociation( bool doSaveSchematic ) { STRING_FORMATTER sf; m_netlist.FormatBackAnnotation( &sf ); Kiway().ExpressMail( FRAME_SCH, MAIL_BACKANNOTATE_FOOTPRINTS, sf.GetString() ); if( doSaveSchematic ) Kiway().ExpressMail( FRAME_SCH, MAIL_SCH_SAVE, std::string( "" ) ); }