/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2019 CERN * Copyright (C) 2019-2020 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 #include #include #include #include #include #include #include #include // for KiROUND EE_INSPECTION_TOOL::EE_INSPECTION_TOOL() : EE_TOOL_BASE( "eeschema.InspectionTool" ), m_ercDialog( nullptr ) { } bool EE_INSPECTION_TOOL::Init() { EE_TOOL_BASE::Init(); auto singleMarkerCondition = SELECTION_CONDITIONS::OnlyType( SCH_MARKER_T ) && SELECTION_CONDITIONS::Count( 1 ); // Add inspection actions to the selection tool menu // CONDITIONAL_MENU& selToolMenu = m_selectionTool->GetToolMenu().GetMenu(); selToolMenu.AddItem( EE_ACTIONS::showDatasheet, EE_CONDITIONS::SingleSymbol && EE_CONDITIONS::Idle, 220 ); return true; } void EE_INSPECTION_TOOL::Reset( RESET_REASON aReason ) { EE_TOOL_BASE::Reset( aReason ); if( aReason == MODEL_RELOAD ) { DestroyERCDialog(); } } int EE_INSPECTION_TOOL::RunERC( const TOOL_EVENT& aEvent ) { if( m_frame->IsType( FRAME_SCH_SYMBOL_EDITOR ) ) { checkPart( static_cast( m_frame )->GetCurPart() ); } else if( m_frame->IsType( FRAME_SCH ) ) { if( m_ercDialog ) { // Needed at least on Windows. Raise() is not enough m_ercDialog->Show( true ); // Bring it to the top if already open. Dual monitor users need this. m_ercDialog->Raise(); } else { // This is a modeless dialog, so new it rather than instantiating on stack. m_ercDialog = new DIALOG_ERC( static_cast( m_frame ) ); m_ercDialog->Show( true ); } } return 0; } void EE_INSPECTION_TOOL::DestroyERCDialog() { if( m_ercDialog ) m_ercDialog->Destroy(); m_ercDialog = nullptr; } // helper function to sort pins by pin num bool sort_by_pin_number( const LIB_PIN* ref, const LIB_PIN* tst ) { // Use number as primary key int test = ref->GetNumber().Cmp( tst->GetNumber() ); // Use DeMorgan variant as secondary key if( test == 0 ) test = ref->GetConvert() - tst->GetConvert(); // Use unit as tertiary key if( test == 0 ) test = ref->GetUnit() - tst->GetUnit(); return test < 0; } void EE_INSPECTION_TOOL::checkPart( LIB_PART* aPart ) { if( !aPart ) return; wxString msg; const int min_grid_size = 25; const int grid_size = KiROUND( getView()->GetGAL()->GetGridSize().x ); const int clamped_grid_size = ( grid_size < min_grid_size ) ? min_grid_size : grid_size; LIB_PINS pinList; aPart->GetPins( pinList ); if( pinList.empty() ) { DisplayInfoMessage( m_frame, _( "No pins!" ) ); return; } // Sort pins by pin num, so 2 duplicate pins // (pins with the same number) will be consecutive in list sort( pinList.begin(), pinList.end(), sort_by_pin_number ); // Test for duplicates: DIALOG_DISPLAY_HTML_TEXT_BASE error_display( m_frame, wxID_ANY, _( "Marker Information" ), wxDefaultPosition, wxSize( 750, 600 ) ); std::vector messages; int dup_error = 0; for( unsigned ii = 1; ii < pinList.size(); ii++ ) { LIB_PIN* pin = pinList[ii - 1]; LIB_PIN* next = pinList[ii]; if( pin->GetNumber() != next->GetNumber() || pin->GetConvert() != next->GetConvert() ) continue; dup_error++; if( aPart->HasConversion() && next->GetConvert() ) { if( aPart->GetUnitCount() <= 1 ) { msg = wxString::Format( _( "Duplicate pin %s \"%s\" at location (%.3f, %.3f)" " conflicts with pin %s \"%s\" at location (%.3f, %.3f) of converted" ), next->GetNumber(), next->GetName(), next->GetPosition().x / 1000.0, -next->GetPosition().y / 1000.0, pin->GetNumber(), pin->GetName(), pin->GetPosition().x / 1000.0, -pin->GetPosition().y / 1000.0 ); } else { msg = wxString::Format( _( "Duplicate pin %s \"%s\" at location (%.3f, %.3f)" " conflicts with pin %s \"%s\" at location (%.3f, %.3f)" " in units %c and %c of converted" ), next->GetNumber(), next->GetName(), next->GetPosition().x / 1000.0, -next->GetPosition().y / 1000.0, pin->GetNumber(), pin->GetName(), pin->GetPosition().x / 1000.0, -pin->GetPosition().y / 1000.0, 'A' + next->GetUnit() - 1, 'A' + pin->GetUnit() - 1 ); } } else { if( aPart->GetUnitCount() <= 1 ) { msg = wxString::Format( _( "Duplicate pin %s \"%s\" at location (%.3f, %.3f)" " conflicts with pin %s \"%s\" at location (%.3f, %.3f)" ), next->GetNumber(), next->GetName(), next->GetPosition().x / 1000.0, -next->GetPosition().y / 1000.0, pin->GetNumber(), pin->GetName(), pin->GetPosition().x / 1000.0, -pin->GetPosition().y / 1000.0 ); } else { msg = wxString::Format( _( "Duplicate pin %s \"%s\" at location (%.3f, %.3f)" " conflicts with pin %s \"%s\" at location (%.3f, %.3f)" " in units %c and %c" ), next->GetNumber(), next->GetName(), next->GetPosition().x / 1000.0, -next->GetPosition().y / 1000.0, pin->GetNumber(), pin->GetName(), pin->GetPosition().x / 1000.0, -pin->GetPosition().y / 1000.0, 'A' + next->GetUnit() - 1, 'A' + pin->GetUnit() - 1 ); } } msg += wxT( ".
" ); messages.push_back( msg ); } // Test for off grid pins: int offgrid_error = 0; for( LIB_PIN* pin : pinList ) { if( ( (pin->GetPosition().x % clamped_grid_size) == 0 ) && ( (pin->GetPosition().y % clamped_grid_size) == 0 ) ) continue; // "pin" is off grid here. offgrid_error++; if( aPart->HasConversion() && pin->GetConvert() ) { if( aPart->GetUnitCount() <= 1 ) { msg = wxString::Format( _( "Off grid pin %s \"%s\" at location " "(%.3f, %.3f) of converted.
" ), pin->GetNumber(), pin->GetName(), pin->GetPosition().x / 1000.0, -pin->GetPosition().y / 1000.0 ); } else { msg = wxString::Format( _( "Off grid pin %s \"%s\" at location " "(%.3f, %.3f) in symbol %c of converted.
" ), pin->GetNumber(), pin->GetName(), pin->GetPosition().x / 1000.0, -pin->GetPosition().y / 1000.0, 'A' + pin->GetUnit() - 1 ); } } else { if( aPart->GetUnitCount() <= 1 ) { msg = wxString::Format( _( "Off grid pin %s \"%s\" at location " "(%.3f, %.3f).
" ), pin->GetNumber(), pin->GetName(), pin->GetPosition().x / 1000.0, -pin->GetPosition().y / 1000.0 ); } else { msg = wxString::Format( _( "Off grid pin %s \"%s\" at location " "(%.3f, %.3f) in symbol %c.
" ), pin->GetNumber(), pin->GetName(), pin->GetPosition().x / 1000.0, -pin->GetPosition().y / 1000.0, 'A' + pin->GetUnit() - 1 ); } } messages.push_back( msg ); } if( !dup_error && !offgrid_error ) DisplayInfoMessage( m_frame, _( "No off grid or duplicate pins were found." ) ); else { wxColour bgcolor = wxSystemSettings::GetColour( wxSYS_COLOUR_WINDOW ); wxColour fgcolor = wxSystemSettings::GetColour( wxSYS_COLOUR_WINDOWTEXT ); wxString outmsg = wxString::Format( "", bgcolor.GetAsString( wxC2S_HTML_SYNTAX ), fgcolor.GetAsString( wxC2S_HTML_SYNTAX ) ); for( auto& msgPart : messages ) outmsg += msgPart; outmsg += ""; error_display.m_htmlWindow->SetPage( outmsg ); error_display.ShowModal(); } } int EE_INSPECTION_TOOL::RunSimulation( const TOOL_EVENT& aEvent ) { #ifdef KICAD_SPICE SIM_PLOT_FRAME* simFrame = (SIM_PLOT_FRAME*) m_frame->Kiway().Player( FRAME_SIMULATOR, true ); simFrame->Show( true ); // On Windows, Raise() does not bring the window on screen, when iconized if( simFrame->IsIconized() ) simFrame->Iconize( false ); simFrame->Raise(); #endif /* KICAD_SPICE */ return 0; } int EE_INSPECTION_TOOL::ShowDatasheet( const TOOL_EVENT& aEvent ) { wxString datasheet; if( m_frame->IsType( FRAME_SCH_SYMBOL_EDITOR ) ) { LIB_PART* part = static_cast( m_frame )->GetCurPart(); if( !part ) return 0; datasheet = part->GetDatasheetField().GetText(); } else if( m_frame->IsType( FRAME_SCH_VIEWER ) || m_frame->IsType( FRAME_SCH_VIEWER_MODAL ) ) { LIB_PART* entry = static_cast( m_frame )->GetSelectedSymbol(); if( !entry ) return 0; datasheet = entry->GetDatasheetField().GetText(); } else if( m_frame->IsType( FRAME_SCH ) ) { EE_SELECTION& selection = m_selectionTool->RequestSelection( EE_COLLECTOR::ComponentsOnly ); if( selection.Empty() ) return 0; SCH_COMPONENT* component = (SCH_COMPONENT*) selection.Front(); datasheet = component->GetField( DATASHEET_FIELD )->GetText(); } if( !datasheet.IsEmpty() && datasheet != wxT( "~" ) ) GetAssociatedDocument( m_frame, datasheet, &m_frame->Prj() ); return 0; } int EE_INSPECTION_TOOL::UpdateMessagePanel( const TOOL_EVENT& aEvent ) { EE_SELECTION_TOOL* selTool = m_toolMgr->GetTool(); EE_SELECTION& selection = selTool->GetSelection(); if( selection.GetSize() == 1 ) { EDA_ITEM* item = (EDA_ITEM*) selection.Front(); MSG_PANEL_ITEMS msgItems; item->GetMsgPanelInfo( m_frame, msgItems ); m_frame->SetMsgPanel( msgItems ); } else { m_frame->ClearMsgPanel(); } if( SCH_EDIT_FRAME* editFrame = dynamic_cast( m_frame ) ) editFrame->UpdateNetHighlightStatus(); return 0; } void EE_INSPECTION_TOOL::setTransitions() { Go( &EE_INSPECTION_TOOL::RunERC, EE_ACTIONS::runERC.MakeEvent() ); Go( &EE_INSPECTION_TOOL::RunSimulation, EE_ACTIONS::runSimulation.MakeEvent() ); Go( &EE_INSPECTION_TOOL::ShowDatasheet, EE_ACTIONS::showDatasheet.MakeEvent() ); Go( &EE_INSPECTION_TOOL::UpdateMessagePanel, EVENTS::SelectedEvent ); Go( &EE_INSPECTION_TOOL::UpdateMessagePanel, EVENTS::UnselectedEvent ); Go( &EE_INSPECTION_TOOL::UpdateMessagePanel, EVENTS::ClearedEvent ); Go( &EE_INSPECTION_TOOL::UpdateMessagePanel, EVENTS::SelectedItemsModified ); }