/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2017 CERN * @author Maciej Suminski * * 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 3 * 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: * https://www.gnu.org/licenses/gpl-3.0.html * or you may search the http://www.gnu.org website for the version 3 license, * or you may write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA */ #include "dialog_update_fields.h" #include #include #include #include int InvokeDialogUpdateFields( SCH_EDIT_FRAME* aCaller, const list aComponents, bool aCreateUndoEntry ) { DIALOG_UPDATE_FIELDS dlg( aCaller, aComponents, aCreateUndoEntry ); return dlg.ShowQuasiModal(); } DIALOG_UPDATE_FIELDS::DIALOG_UPDATE_FIELDS( SCH_EDIT_FRAME* aParent, const list& aComponents, bool aCreateUndoEntry ) : DIALOG_UPDATE_FIELDS_BASE( aParent ), m_frame( aParent ), m_components( aComponents ), m_createUndo( aCreateUndoEntry ) { m_sdbSizerOK->SetDefault(); } bool DIALOG_UPDATE_FIELDS::TransferDataFromWindow() { if( !wxDialog::TransferDataFromWindow() ) return false; if( m_components.empty() ) return true; // nothing to process // Create the set of fields to be updated m_fields.clear(); for( unsigned i = 0; i < m_fieldsBox->GetCount(); ++i ) { if( m_fieldsBox->IsChecked( i ) ) m_fields.insert( m_fieldsBox->GetString( i ) ); } // Undo buffer entry if( m_createUndo ) { PICKED_ITEMS_LIST itemsList; for( auto component : m_components ) itemsList.PushItem( ITEM_PICKER( component, UR_CHANGED ) ); m_frame->SaveCopyInUndoList( itemsList, UR_CHANGED ); } // Do it! for( auto component : m_components ) updateFields( component ); m_frame->SyncView(); m_frame->GetCanvas()->Refresh(); m_frame->OnModify(); return true; } bool DIALOG_UPDATE_FIELDS::TransferDataToWindow() { if( !wxDialog::TransferDataToWindow() || !m_components.size() ) return false; // Collect all user field names from library parts of components that are going to be updated { for( auto component : m_components ) { const std::unique_ptr< LIB_PART >& part = component->GetPartRef(); if( !part ) continue; const auto& drawItems = part->GetDrawItems(); for( auto it = drawItems.begin( LIB_FIELD_T ); it != drawItems.end( LIB_FIELD_T ); ++it ) { const LIB_FIELD* field = static_cast( &( *it ) ); if( field->GetId() >= MANDATORY_FIELDS ) m_fields.insert( field->GetName( false ) ); } } } // Update the listbox widget m_fieldsBox->Clear(); for( int i = 0; i < MANDATORY_FIELDS; ++i ) { m_fieldsBox->Append( m_components.front()->GetField( i )->GetName( false ) ); if( i != REFERENCE && i != VALUE ) m_fieldsBox->Check( i, true ); } for( const auto& field : m_fields ) { int idx = m_fieldsBox->Append( field ); m_fieldsBox->Check( idx, true ); } // Now all widgets have the size fixed, call FinishDialogSettings FinishDialogSettings(); return true; } void DIALOG_UPDATE_FIELDS::updateFields( SCH_COMPONENT* aComponent ) { std::vector oldFields; SCH_FIELDS newFields; std::unique_ptr< LIB_PART >& libPart = aComponent->GetPartRef(); if( !libPart ) // the symbol is not found in lib: cannot update fields return; LIB_PART* alias = m_frame->GetLibPart( aComponent->GetLibId() ); aComponent->GetFields( oldFields, false ); for( auto compField : oldFields ) { // If requested, transfer only fields that occur also in the original library part if( !m_removeExtraBox->IsChecked() || libPart->FindField( compField->GetName() ) ) newFields.push_back( *compField ); } // Update the requested field values for( const auto& partField : m_fields ) { LIB_FIELD* libField = libPart->FindField( partField ); if( !libField ) continue; SCH_FIELD* field = nullptr; auto it = std::find_if( newFields.begin(), newFields.end(), [&] ( const SCH_FIELD& f ) { return f.GetName() == partField; } ); if( it != newFields.end() ) { field = &*it; } else { // Missing field, it has to be added to the component SCH_FIELD f( wxPoint( 0, 0 ), newFields.size(), aComponent, partField ); newFields.push_back( f ); field = &newFields.back(); } wxString fieldValue = libField->GetText(); if( alias ) { if( partField == TEMPLATE_FIELDNAME::GetDefaultFieldName( VALUE ) ) fieldValue = alias->GetName(); else if( partField == TEMPLATE_FIELDNAME::GetDefaultFieldName( DATASHEET ) ) fieldValue = alias->GetDocFileName(); } // If the library field is empty an update would clear an existing entry. // Check if this is the desired behavior. if( !fieldValue.empty() || m_resetEmpty->IsChecked() ) field->SetText( fieldValue ); if( m_resetVisibility->IsChecked() ) field->SetVisible( libField->IsVisible() ); if( m_resetPosition->IsChecked() ) { field->SetTextAngle( libField->GetTextAngle() ); // Board fields are board-relative; symbol editor fields are component-relative if( m_createUndo ) field->SetTextPos( libField->GetTextPos() + aComponent->GetPosition() ); else field->SetTextPos( libField->GetTextPos() ); } if( m_resetSizeAndStyle->IsChecked() ) { field->SetHorizJustify( libField->GetHorizJustify() ); field->SetVertJustify( libField->GetVertJustify() ); field->SetTextSize( libField->GetTextSize() ); field->SetItalic( libField->IsItalic() ); field->SetBold( libField->IsBold() ); } } // Apply changes & clean-up aComponent->SetFields( newFields ); } void DIALOG_UPDATE_FIELDS::checkAll( bool aCheck ) { for( unsigned i = 0; i < m_fieldsBox->GetCount(); ++i ) m_fieldsBox->Check( i, aCheck ); }