/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2015 Jean-Pierre Charras, jp.charras at wanadoo.fr * Copyright (C) 2013 SoftPLC Corporation, Dick Hollenbeck * Copyright (C) 2013 Wayne Stambaugh * Copyright (C) 1992-2023 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 #define ID_MATCH_FP_ALL 4200 #define ID_MATCH_FP_SELECTED 4201 #define ID_MATCH_FP_REF 4202 #define ID_MATCH_FP_VAL 4203 #define ID_MATCH_FP_ID 4204 int g_matchModeForUpdate = ID_MATCH_FP_ALL; int g_matchModeForUpdateSelected = ID_MATCH_FP_SELECTED; int g_matchModeForExchange = ID_MATCH_FP_REF; int g_matchModeForExchangeSelected = ID_MATCH_FP_SELECTED; // { update, change } bool g_removeExtraTextItems[2] = { false, false }; bool g_resetTextItemLayers[2] = { false, true }; bool g_resetTextItemEffects[2] = { false, true }; bool g_resetFabricationAttrs[2] = { false, true }; bool g_reset3DModels[2] = { true, true }; DIALOG_EXCHANGE_FOOTPRINTS::DIALOG_EXCHANGE_FOOTPRINTS( PCB_EDIT_FRAME* aParent, FOOTPRINT* aFootprint, bool updateMode, bool selectedMode ) : DIALOG_EXCHANGE_FOOTPRINTS_BASE( aParent ), m_commit( aParent ), m_parent( aParent ), m_currentFootprint( aFootprint ), m_updateMode( updateMode ) { if( !updateMode ) { SetTitle( _( "Change Footprints" ) ); m_matchAll->SetLabel( _( "Change all footprints on board" ) ); m_matchSelected->SetLabel( _( "Change selected footprint(s)" ) ); m_matchSpecifiedRef->SetLabel( _( "Change footprints matching reference designator:" ) ); m_matchSpecifiedValue->SetLabel( _( "Change footprints matching value:" ) ); m_matchSpecifiedID->SetLabel( _( "Change footprints with library id:" ) ); m_resetTextItemLayers->SetLabel( _( "Update text layers and visibilities" ) ); m_resetTextItemEffects->SetLabel( _( "Update text sizes, styles and positions" ) ); m_resetFabricationAttrs->SetLabel( _( "Update fabrication attributes" ) ); m_reset3DModels->SetLabel( _( "Update 3D models" ) ); } #if 0 // translator hint wxString x = _( "Update/reset strings: there are two cases these descriptions need to cover: " "the user made overrides to a footprint on the PCB and wants to remove them, " "or the user made changes to the library footprint and wants to propagate " "them back to the PCB." ); #endif if( m_updateMode ) { m_changeSizer->Show( false ); } else { m_upperSizer->FindItem( m_matchAll )->Show( false ); m_newIDBrowseButton->SetBitmap( KiBitmapBundle( BITMAPS::small_library ) ); } if( m_currentFootprint ) m_newID->AppendText( From_UTF8( m_currentFootprint->GetFPID().Format().c_str() ) ); else m_upperSizer->FindItem( m_matchSelected )->Show( false ); // Use ChangeValue() instead of SetValue() so we don't generate events. if( m_currentFootprint ) m_specifiedRef->ChangeValue( m_currentFootprint->GetReference() ); if( m_currentFootprint ) m_specifiedValue->ChangeValue( m_currentFootprint->GetValue() ); if( m_currentFootprint ) m_specifiedID->ChangeValue( From_UTF8( m_currentFootprint->GetFPID().Format().c_str() ) ); m_specifiedIDBrowseButton->SetBitmap( KiBitmapBundle( BITMAPS::small_library ) ); m_upperSizer->SetEmptyCellSize( wxSize( 0, 0 ) ); // The upper sizer has its content modified: re-layout it: m_upperSizer->Layout(); // initialize match-mode if( m_updateMode ) m_matchMode = selectedMode ? &g_matchModeForUpdateSelected : &g_matchModeForUpdate; else m_matchMode = selectedMode ? &g_matchModeForExchangeSelected : &g_matchModeForExchange; wxCommandEvent event; event.SetEventObject( this ); switch( *m_matchMode ) { case ID_MATCH_FP_ALL: OnMatchAllClicked( event ); break; case ID_MATCH_FP_SELECTED: OnMatchSelectedClicked( event ); break; case ID_MATCH_FP_REF: OnMatchRefClicked( event ); break; case ID_MATCH_FP_VAL: OnMatchValueClicked( event ); break; case ID_MATCH_FP_ID: OnMatchIDClicked( event ); break; default: break; } m_removeExtraBox->SetValue( g_removeExtraTextItems[ m_updateMode ? 0 : 1 ] ); m_resetTextItemLayers->SetValue( g_resetTextItemLayers[ m_updateMode ? 0 : 1 ] ); m_resetTextItemEffects->SetValue( g_resetTextItemEffects[ m_updateMode ? 0 : 1 ] ); m_resetFabricationAttrs->SetValue( g_resetFabricationAttrs[ m_updateMode ? 0 : 1 ] ); m_reset3DModels->SetValue( g_reset3DModels[ m_updateMode ? 0 : 1 ] ); m_MessageWindow->SetLazyUpdate( true ); m_MessageWindow->SetFileName( Prj().GetProjectPath() + wxT( "report.txt" ) ); // DIALOG_SHIM needs a unique hash_key because classname is not sufficient // because the update and change versions of this dialog have different controls. m_hash_key = TO_UTF8( GetTitle() ); wxString okLabel = m_updateMode ? _( "Update" ) : _( "Change" ); SetupStandardButtons( { { wxID_OK, okLabel }, { wxID_CANCEL, _( "Close" ) } } ); // Now all widgets have the size fixed, call FinishDialogSettings finishDialogSettings(); } DIALOG_EXCHANGE_FOOTPRINTS::~DIALOG_EXCHANGE_FOOTPRINTS() { g_removeExtraTextItems[ m_updateMode ? 0 : 1 ] = m_removeExtraBox->GetValue(); g_resetTextItemLayers[ m_updateMode ? 0 : 1 ] = m_resetTextItemLayers->GetValue(); g_resetTextItemEffects[ m_updateMode ? 0 : 1 ] = m_resetTextItemEffects->GetValue(); g_resetFabricationAttrs[ m_updateMode ? 0 : 1 ] = m_resetFabricationAttrs->GetValue(); g_reset3DModels[ m_updateMode ? 0 : 1 ] = m_reset3DModels->GetValue(); } bool DIALOG_EXCHANGE_FOOTPRINTS::isMatch( FOOTPRINT* aFootprint ) { LIB_ID specifiedID; switch( *m_matchMode ) { case ID_MATCH_FP_ALL: return true; case ID_MATCH_FP_SELECTED: return aFootprint == m_currentFootprint || aFootprint->IsSelected(); case ID_MATCH_FP_REF: return WildCompareString( m_specifiedRef->GetValue(), aFootprint->GetReference(), false ); case ID_MATCH_FP_VAL: return WildCompareString( m_specifiedValue->GetValue(), aFootprint->GetValue(), false ); case ID_MATCH_FP_ID: specifiedID.Parse( m_specifiedID->GetValue() ); return aFootprint->GetFPID() == specifiedID; default: return false; // just to quiet compiler warnings.... } } wxRadioButton* DIALOG_EXCHANGE_FOOTPRINTS::getRadioButtonForMode() { switch( *m_matchMode ) { case ID_MATCH_FP_ALL: return m_matchAll; case ID_MATCH_FP_SELECTED: return m_matchSelected; case ID_MATCH_FP_REF: return m_matchSpecifiedRef; case ID_MATCH_FP_VAL: return m_matchSpecifiedValue; case ID_MATCH_FP_ID: return m_matchSpecifiedID; default: return nullptr; } } void DIALOG_EXCHANGE_FOOTPRINTS::updateMatchModeRadioButtons( wxUpdateUIEvent& ) { wxRadioButton* rb_button = getRadioButtonForMode(); wxRadioButton* rb_butt_list[] = { m_matchAll, m_matchSelected, m_matchSpecifiedRef, m_matchSpecifiedValue, m_matchSpecifiedID, nullptr // end of list }; // Ensure the button state is ok. Only one button can be checked // Change button state only if its state is incorrect, otherwise // we have issues depending on the platform. for( int ii = 0; rb_butt_list[ii]; ++ii ) { bool state = rb_butt_list[ii] == rb_button; if( rb_butt_list[ii]->GetValue() != state ) rb_butt_list[ii]->SetValue( state ); } } void DIALOG_EXCHANGE_FOOTPRINTS::OnMatchAllClicked( wxCommandEvent& event ) { *m_matchMode = ID_MATCH_FP_ALL; if( event.GetEventObject() == this ) SetInitialFocus( m_matchAll ); else m_matchAll->SetFocus(); } void DIALOG_EXCHANGE_FOOTPRINTS::OnMatchSelectedClicked( wxCommandEvent& event ) { *m_matchMode = ID_MATCH_FP_SELECTED; if( event.GetEventObject() == this ) SetInitialFocus( m_matchSelected ); else m_matchSelected->SetFocus(); } void DIALOG_EXCHANGE_FOOTPRINTS::OnMatchRefClicked( wxCommandEvent& event ) { *m_matchMode = ID_MATCH_FP_REF; if( event.GetEventObject() == this ) SetInitialFocus( m_specifiedRef ); else if( event.GetEventObject() != m_specifiedRef ) m_specifiedRef->SetFocus(); } void DIALOG_EXCHANGE_FOOTPRINTS::OnMatchValueClicked( wxCommandEvent& event ) { *m_matchMode = ID_MATCH_FP_VAL; if( event.GetEventObject() == this ) SetInitialFocus( m_specifiedValue ); else if( event.GetEventObject() != m_specifiedValue ) m_specifiedValue->SetFocus(); } void DIALOG_EXCHANGE_FOOTPRINTS::OnMatchIDClicked( wxCommandEvent& event ) { *m_matchMode = ID_MATCH_FP_ID; if( event.GetEventObject() == this ) SetInitialFocus( m_specifiedID ); else if( event.GetEventObject() != m_specifiedID ) m_specifiedID->SetFocus(); } void DIALOG_EXCHANGE_FOOTPRINTS::OnOKClicked( wxCommandEvent& event ) { wxBusyCursor dummy; m_MessageWindow->Clear(); m_MessageWindow->Flush( false ); processMatchingFootprints(); m_parent->Compile_Ratsnest( true ); m_parent->GetCanvas()->Refresh(); m_MessageWindow->Flush( false ); m_commit.Push( wxT( "Changed footprint" ) ); } void DIALOG_EXCHANGE_FOOTPRINTS::processMatchingFootprints() { LIB_ID newFPID; if( m_parent->GetBoard()->Footprints().empty() ) return; if( !m_updateMode ) { newFPID.Parse( m_newID->GetValue() ); if( !newFPID.IsValid() ) return; } /* * NB: the change is done from the last footprint because processFootprint() modifies the * last item in the list. */ for( auto it = m_parent->GetBoard()->Footprints().rbegin(); it != m_parent->GetBoard()->Footprints().rend(); it++ ) { FOOTPRINT* footprint = *it; if( !isMatch( footprint ) ) continue; if( m_updateMode ) processFootprint( footprint, footprint->GetFPID() ); else processFootprint( footprint, newFPID ); } } void DIALOG_EXCHANGE_FOOTPRINTS::processFootprint( FOOTPRINT* aFootprint, const LIB_ID& aNewFPID ) { LIB_ID oldFPID = aFootprint->GetFPID(); wxString msg; // Load new footprint. if( m_updateMode ) { msg.Printf( _( "Updated footprint %s (%s)" ) + wxS( ": " ), aFootprint->GetReference(), oldFPID.Format().c_str() ); } else { msg.Printf( _( "Changed footprint %s from '%s' to '%s'" ) + wxS( ": " ), aFootprint->GetReference(), oldFPID.Format().c_str(), aNewFPID.Format().c_str() ); } FOOTPRINT* newFootprint = m_parent->LoadFootprint( aNewFPID ); if( !newFootprint ) { msg += _( "*** library footprint not found ***" ); m_MessageWindow->Report( msg, RPT_SEVERITY_ERROR ); return; } bool updated = !m_updateMode || aFootprint->FootprintNeedsUpdate( newFootprint ); m_parent->ExchangeFootprint( aFootprint, newFootprint, m_commit, m_removeExtraBox->GetValue(), m_resetTextItemLayers->GetValue(), m_resetTextItemEffects->GetValue(), m_resetFabricationAttrs->GetValue(), m_reset3DModels->GetValue(), &updated ); // Update footprint field with the new FPID newFootprint->Footprint().SetText( aNewFPID.Format() ); if( aFootprint == m_currentFootprint ) m_currentFootprint = newFootprint; if( m_updateMode && !updated ) { msg += _( ": (no changes)" ); m_MessageWindow->Report( msg, RPT_SEVERITY_INFO ); } else { msg += _( ": OK" ); m_MessageWindow->Report( msg, RPT_SEVERITY_ACTION ); } } void DIALOG_EXCHANGE_FOOTPRINTS::ViewAndSelectFootprint( wxCommandEvent& event ) { wxString newname = m_newID->GetValue(); if( KIWAY_PLAYER* frame = Kiway().Player( FRAME_FOOTPRINT_CHOOSER, true, this ) ) { if( m_currentFootprint ) { /* * Symbol netlist format: * pinNumber pinName pinNumber pinName... * fpFilter fpFilter... */ wxString netlist; wxArrayString pins; for( const wxString& pad : m_currentFootprint->GetUniquePadNumbers() ) pins.push_back( pad + ' ' + wxEmptyString /* leave pinName empty */ ); if( !pins.IsEmpty() ) netlist << EscapeString( wxJoin( pins, '\t' ), CTX_LINE ); netlist << wxS( "\r" ); netlist << EscapeString( m_currentFootprint->GetFilters(), CTX_LINE ) << wxS( "\r" ); std::string payload( netlist.ToStdString() ); KIWAY_EXPRESS mail( FRAME_FOOTPRINT_CHOOSER, MAIL_SYMBOL_NETLIST, payload ); frame->KiwayMailIn( mail ); } if( frame->ShowModal( &newname, this ) ) { if( event.GetEventObject() == m_newIDBrowseButton ) m_newID->SetValue( newname ); else m_specifiedID->SetValue( newname ); } frame->Destroy(); } }