/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2019 CERN * Copyright (C) 2019-2021 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 // for KiROUND class DIALOG_DISPLAY_HTML_TEXT : public DIALOG_DISPLAY_HTML_TEXT_BASE { public: DIALOG_DISPLAY_HTML_TEXT( wxWindow* aParent, wxWindowID aId, const wxString& aTitle, const wxPoint& aPos, const wxSize& aSize, long aStyle = 0 ) : DIALOG_DISPLAY_HTML_TEXT_BASE( aParent, aId, aTitle, aPos, aSize, aStyle ) { } ~DIALOG_DISPLAY_HTML_TEXT() { } void SetPage( const wxString& message ) { m_htmlWindow->SetPage( message ); } }; EE_INSPECTION_TOOL::EE_INSPECTION_TOOL() : EE_TOOL_BASE( "eeschema.InspectionTool" ), m_ercDialog( nullptr ) { } bool EE_INSPECTION_TOOL::Init() { EE_TOOL_BASE::Init(); // Add inspection actions to the selection tool menu // CONDITIONAL_MENU& selToolMenu = m_selectionTool->GetToolMenu().GetMenu(); selToolMenu.AddItem( EE_ACTIONS::excludeMarker, EE_CONDITIONS::SingleNonExcludedMarker, 100 ); 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 ) { ShowERCDialog(); return 0; } void EE_INSPECTION_TOOL::ShowERCDialog() { 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 ); } } } void EE_INSPECTION_TOOL::DestroyERCDialog() { if( m_ercDialog ) m_ercDialog->Destroy(); m_ercDialog = nullptr; } int EE_INSPECTION_TOOL::PrevMarker( const TOOL_EVENT& aEvent ) { if( m_ercDialog ) { m_ercDialog->Show( true ); m_ercDialog->Raise(); m_ercDialog->PrevMarker(); } else { ShowERCDialog(); } return 0; } int EE_INSPECTION_TOOL::NextMarker( const TOOL_EVENT& aEvent ) { if( m_ercDialog ) { m_ercDialog->Show( true ); m_ercDialog->Raise(); m_ercDialog->NextMarker(); } else { ShowERCDialog(); } return 0; } int EE_INSPECTION_TOOL::ExcludeMarker( const TOOL_EVENT& aEvent ) { EE_SELECTION_TOOL* selTool = m_toolMgr->GetTool(); EE_SELECTION& selection = selTool->GetSelection(); SCH_MARKER* marker = nullptr; if( selection.GetSize() == 1 && selection.Front()->Type() == SCH_MARKER_T ) marker = static_cast( selection.Front() ); if( m_ercDialog ) { // Let the ERC dialog handle it since it has more update hassles to worry about // Note that if marker is nullptr the dialog will exclude whichever marker is selected // in the dialog itself m_ercDialog->ExcludeMarker( marker ); } else if( marker != nullptr ) { marker->SetExcluded( true ); m_frame->GetCanvas()->GetView()->Update( marker ); m_frame->GetCanvas()->Refresh(); m_frame->OnModify(); } return 0; } // 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; } int EE_INSPECTION_TOOL::CheckSymbol( const TOOL_EVENT& aEvent ) { LIB_SYMBOL* symbol = static_cast( m_frame )->GetCurSymbol(); EDA_UNITS units = m_frame->GetUserUnits(); if( !symbol ) return 0; LIB_PINS pinList; symbol->GetPins( pinList ); // Test for duplicates: // 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 ); // The minimal grid size allowed to place a pin is 25 mils // the best grid size is 50 mils, but 25 mils is still usable // this is because all symbols are using a 50 mils grid to place pins, and therefore // the wires must be on the 50 mils grid // So raise an error if a pin is not on a 25 (or bigger :50 or 100) mils grid const int min_grid_size = Mils2iu( 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; std::vector messages; wxString msg; for( unsigned ii = 1; ii < pinList.size(); ii++ ) { LIB_PIN* pin = pinList[ii - 1]; LIB_PIN* next = pinList[ii]; if( pin->GetNumber() != next->GetNumber() ) continue; // Pins are not duplicated only if they are in different convert bodies // (but GetConvert() == 0 means commun to all convert bodies) if( pin->GetConvert() != 0 && next->GetConvert() != 0 ) { if( pin->GetConvert() != next->GetConvert() ) continue; } wxString pinName; wxString nextName; if( pin->GetName() != wxT( "~" ) && !pin->GetName().IsEmpty() ) pinName = wxT( " '" ) + pin->GetName() + wxT( "'" ); if( next->GetName() != wxT( "~" ) && !next->GetName().IsEmpty() ) nextName = wxT( " '" ) + next->GetName() + wxT( "'" ); if( symbol->HasConversion() && next->GetConvert() ) { if( symbol->GetUnitCount() <= 1 ) { msg.Printf( _( "Duplicate pin %s %s at location (%.3f, %.3f)" " conflicts with pin %s%s at location (%.3f, %.3f)" " of converted." ), next->GetNumber(), nextName, MessageTextFromValue( units, next->GetPosition().x ), MessageTextFromValue( units, -next->GetPosition().y ), pin->GetNumber(), pin->GetName(), MessageTextFromValue( units, pin->GetPosition().x ), MessageTextFromValue( units, -pin->GetPosition().y ) ); } else { msg.Printf( _( "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(), nextName, MessageTextFromValue( units, next->GetPosition().x ), MessageTextFromValue( units, -next->GetPosition().y ), pin->GetNumber(), pinName, MessageTextFromValue( units, pin->GetPosition().x ), MessageTextFromValue( units, -pin->GetPosition().y ), 'A' + next->GetUnit() - 1, 'A' + pin->GetUnit() - 1 ); } } else { if( symbol->GetUnitCount() <= 1 ) { msg.Printf( _( "Duplicate pin %s %s at location (%s, %s)" " conflicts with pin %s%s at location (%s, %s)." ), next->GetNumber(), nextName, MessageTextFromValue( units, next->GetPosition().x ), MessageTextFromValue( units, -next->GetPosition().y ), pin->GetNumber(), pinName, MessageTextFromValue( units, pin->GetPosition().x ), MessageTextFromValue( units, -pin->GetPosition().y ) ); } else { msg.Printf( _( "Duplicate pin %s %s at location (%s, %s)" " conflicts with pin %s%s at location (%s, %s)" " in units %c and %c." ), next->GetNumber(), nextName, MessageTextFromValue( units, next->GetPosition().x ), MessageTextFromValue( units, -next->GetPosition().y ), pin->GetNumber(), pinName, MessageTextFromValue( units, pin->GetPosition().x ), MessageTextFromValue( units, -pin->GetPosition().y ), 'A' + next->GetUnit() - 1, 'A' + pin->GetUnit() - 1 ); } } msg += wxT( "

" ); messages.push_back( msg ); } for( LIB_PIN* pin : pinList ) { wxString pinName = pin->GetName(); if( pinName.IsEmpty() || pinName == wxT( "~" ) ) pinName = wxEmptyString; else pinName = wxT( "'" ) + pinName + wxT( "'" ); if( !symbol->IsPower() && pin->GetType() == ELECTRICAL_PINTYPE::PT_POWER_IN && !pin->IsVisible() ) { // hidden power pin if( symbol->HasConversion() && pin->GetConvert() ) { if( symbol->GetUnitCount() <= 1 ) { msg.Printf( _( "Info: Hidden power pin %s %s at location (%s, %s)" " of converted." ), pin->GetNumber(), pinName, MessageTextFromValue( units, pin->GetPosition().x ), MessageTextFromValue( units, -pin->GetPosition().y ) ); } else { msg.Printf( _( "Info: Hidden power pin %s %s at location (%s, %s)" " in unit %c of converted." ), pin->GetNumber(), pinName, MessageTextFromValue( units, pin->GetPosition().x ), MessageTextFromValue( units, -pin->GetPosition().y ), 'A' + pin->GetUnit() - 1 ); } } else { if( symbol->GetUnitCount() <= 1 ) { msg.Printf( _( "Info: Hidden power pin %s %s at location (%s, %s)." ), pin->GetNumber(), pinName, MessageTextFromValue( units, pin->GetPosition().x ), MessageTextFromValue( units, -pin->GetPosition().y ) ); } else { msg.Printf( _( "Info: Hidden power pin %s %s at location (%s, %s)" " in unit %c." ), pin->GetNumber(), pinName, MessageTextFromValue( units, pin->GetPosition().x ), MessageTextFromValue( units, -pin->GetPosition().y ), 'A' + pin->GetUnit() - 1 ); } } msg += wxT( "
" ); msg += _( "(Hidden power pins will drive their pin names on to any connected nets.)" ); msg += wxT( "

" ); messages.push_back( msg ); } if( ( (pin->GetPosition().x % clamped_grid_size) != 0 ) || ( (pin->GetPosition().y % clamped_grid_size) != 0 ) ) { // pin is off grid if( symbol->HasConversion() && pin->GetConvert() ) { if( symbol->GetUnitCount() <= 1 ) { msg.Printf( _( "Off grid pin %s %s at location (%s, %s)" " of converted." ), pin->GetNumber(), pinName, MessageTextFromValue( units, pin->GetPosition().x ), MessageTextFromValue( units, -pin->GetPosition().y ) ); } else { msg.Printf( _( "Off grid pin %s %s at location (%.3s, %.3s)" " in unit %c of converted." ), pin->GetNumber(), pinName, MessageTextFromValue( units, pin->GetPosition().x ), MessageTextFromValue( units, -pin->GetPosition().y ), 'A' + pin->GetUnit() - 1 ); } } else { if( symbol->GetUnitCount() <= 1 ) { msg.Printf( _( "Off grid pin %s %s at location (%s, %s)." ), pin->GetNumber(), pinName, MessageTextFromValue( units, pin->GetPosition().x ), MessageTextFromValue( units, -pin->GetPosition().y ) ); } else { msg.Printf( _( "Off grid pin %s %s at location (%s, %s)" " in unit %c." ), pin->GetNumber(), pinName, MessageTextFromValue( units, pin->GetPosition().x ), MessageTextFromValue( units, -pin->GetPosition().y ), 'A' + pin->GetUnit() - 1 ); } } msg += wxT( "

" ); messages.push_back( msg ); } } if( messages.empty() ) { DisplayInfoMessage( m_frame, _( "No symbol issues found." ) ); } else { wxString outmsg; for( const wxString& single_msg : messages ) outmsg += single_msg; DIALOG_DISPLAY_HTML_TEXT error_display( m_frame, wxID_ANY, _( "Symbol Warnings" ), wxDefaultPosition, wxSize( 700, 350 ) ); error_display.SetPage( outmsg ); error_display.ShowModal(); } return 0; } 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 ); if( !simFrame ) return -1; if( wxWindow* blocking_win = simFrame->Kiway().GetBlockingDialog() ) blocking_win->Close( 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_SYMBOL* symbol = static_cast( m_frame )->GetCurSymbol(); if( !symbol ) return 0; datasheet = symbol->GetDatasheetField().GetText(); } else if( m_frame->IsType( FRAME_SCH_VIEWER ) || m_frame->IsType( FRAME_SCH_VIEWER_MODAL ) ) { LIB_SYMBOL* 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::SymbolsOnly ); if( selection.Empty() ) return 0; SCH_SYMBOL* symbol = (SCH_SYMBOL*) selection.Front(); // Use GetShownText() to resolve any text variables datasheet = symbol->GetField( DATASHEET_FIELD )->GetShownText(); } if( datasheet.IsEmpty() || datasheet == wxT( "~" ) ) { m_frame->ShowInfoBarError( _( "No datasheet defined." ) ); } else { GetAssociatedDocument( m_frame, datasheet, &m_frame->Prj(), m_frame->Prj().SchSearchS() ); } 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(); std::vector 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::PrevMarker, EE_ACTIONS::prevMarker.MakeEvent() ); Go( &EE_INSPECTION_TOOL::NextMarker, EE_ACTIONS::nextMarker.MakeEvent() ); Go( &EE_INSPECTION_TOOL::ExcludeMarker, EE_ACTIONS::excludeMarker.MakeEvent() ); Go( &EE_INSPECTION_TOOL::CheckSymbol, EE_ACTIONS::checkSymbol.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 ); }