You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1472 lines
49 KiB

5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
  1. /*
  2. * This program source code file is part of KiCad, a free EDA CAD application.
  3. *
  4. * Copyright (C) 2019-2021 KiCad Developers, see AUTHORS.txt for contributors.
  5. *
  6. * This program is free software; you can redistribute it and/or
  7. * modify it under the terms of the GNU General Public License
  8. * as published by the Free Software Foundation; either version 2
  9. * of the License, or (at your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. * GNU General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License
  17. * along with this program; if not, you may find one here:
  18. * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
  19. * or you may search the http://www.gnu.org website for the version 2 license,
  20. * or you may write to the Free Software Foundation, Inc.,
  21. * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
  22. */
  23. #include <bitmaps.h>
  24. #include <pcb_group.h>
  25. #include <tool/tool_manager.h>
  26. #include <tools/pcb_selection_tool.h>
  27. #include <tools/pcb_picker_tool.h>
  28. #include <tools/edit_tool.h>
  29. #include <pcb_painter.h>
  30. #include <connectivity/connectivity_data.h>
  31. #include <dialogs/wx_html_report_box.h>
  32. #include <drc/drc_engine.h>
  33. #include <dialogs/panel_setup_rules_base.h>
  34. #include <dialogs/dialog_constraints_reporter.h>
  35. #include <string_utils.h>
  36. #include "board_inspection_tool.h"
  37. #include <pcbnew_settings.h>
  38. #include <widgets/appearance_controls.h>
  39. #include <drc/drc_item.h>
  40. #include <pad.h>
  41. BOARD_INSPECTION_TOOL::BOARD_INSPECTION_TOOL() :
  42. PCB_TOOL_BASE( "pcbnew.InspectionTool" ),
  43. m_frame( nullptr )
  44. {
  45. m_probingSchToPcb = false;
  46. m_dynamicData = nullptr;
  47. }
  48. class NET_CONTEXT_MENU : public ACTION_MENU
  49. {
  50. public:
  51. NET_CONTEXT_MENU() : ACTION_MENU( true )
  52. {
  53. SetIcon( BITMAPS::show_ratsnest );
  54. SetTitle( _( "Net Tools" ) );
  55. Add( PCB_ACTIONS::showNet );
  56. Add( PCB_ACTIONS::hideNet );
  57. Add( PCB_ACTIONS::highlightNetSelection );
  58. Add( PCB_ACTIONS::clearHighlight );
  59. }
  60. private:
  61. ACTION_MENU* create() const override
  62. {
  63. return new NET_CONTEXT_MENU();
  64. }
  65. };
  66. bool BOARD_INSPECTION_TOOL::Init()
  67. {
  68. PCB_SELECTION_TOOL* selectionTool = m_toolMgr->GetTool<PCB_SELECTION_TOOL>();
  69. auto netSubMenu = std::make_shared<NET_CONTEXT_MENU>();
  70. netSubMenu->SetTool( this );
  71. static KICAD_T connectedTypes[] = { PCB_TRACE_T, PCB_VIA_T, PCB_ARC_T, PCB_PAD_T, PCB_ZONE_T,
  72. EOT };
  73. CONDITIONAL_MENU& menu = selectionTool->GetToolMenu().GetMenu();
  74. selectionTool->GetToolMenu().AddSubMenu( netSubMenu );
  75. menu.AddMenu( netSubMenu.get(), SELECTION_CONDITIONS::OnlyTypes( connectedTypes ), 200 );
  76. menu.AddItem( PCB_ACTIONS::inspectClearance, SELECTION_CONDITIONS::Count( 2 ), 200 );
  77. return true;
  78. }
  79. void BOARD_INSPECTION_TOOL::Reset( RESET_REASON aReason )
  80. {
  81. m_frame = getEditFrame<PCB_EDIT_FRAME>();
  82. }
  83. int BOARD_INSPECTION_TOOL::ShowStatisticsDialog( const TOOL_EVENT& aEvent )
  84. {
  85. DIALOG_BOARD_STATISTICS dialog( m_frame );
  86. dialog.ShowModal();
  87. return 0;
  88. }
  89. DRC_ENGINE BOARD_INSPECTION_TOOL::makeDRCEngine( bool* aCompileError, bool* aCourtyardError )
  90. {
  91. DRC_ENGINE engine( m_frame->GetBoard(), &m_frame->GetBoard()->GetDesignSettings() );
  92. try
  93. {
  94. engine.InitEngine( m_frame->GetDesignRulesPath() );
  95. }
  96. catch( PARSE_ERROR& )
  97. {
  98. if( aCompileError )
  99. *aCompileError = true;
  100. }
  101. for( ZONE* zone : m_frame->GetBoard()->Zones() )
  102. zone->CacheBoundingBox();
  103. for( FOOTPRINT* footprint : m_frame->GetBoard()->Footprints() )
  104. {
  105. for( ZONE* zone : footprint->Zones() )
  106. zone->CacheBoundingBox();
  107. footprint->BuildPolyCourtyards();
  108. if( aCourtyardError && ( footprint->GetFlags() & MALFORMED_COURTYARDS ) != 0 )
  109. *aCourtyardError = true;
  110. }
  111. return engine;
  112. }
  113. wxString BOARD_INSPECTION_TOOL::getItemDescription( BOARD_ITEM* aItem )
  114. {
  115. // Null items have no description
  116. if( !aItem )
  117. return wxString();
  118. wxString s = aItem->GetSelectMenuText( m_frame->GetUserUnits() );
  119. if( aItem->IsConnected() )
  120. {
  121. BOARD_CONNECTED_ITEM* cItem = static_cast<BOARD_CONNECTED_ITEM*>( aItem );
  122. s += wxS( " " ) + wxString::Format( _( "[netclass %s]" ),
  123. cItem->GetNetClass()->GetName() );
  124. }
  125. return s;
  126. };
  127. void BOARD_INSPECTION_TOOL::reportCompileError( REPORTER* r )
  128. {
  129. r->Report( "" );
  130. r->Report( _( "Report incomplete: could not compile custom design rules. " )
  131. + "<a href='boardsetup'>" + _( "Show design rules." ) + "</a>" );
  132. }
  133. void BOARD_INSPECTION_TOOL::reportHeader( const wxString& aTitle, BOARD_ITEM* a, REPORTER* r )
  134. {
  135. r->Report( "<h7>" + EscapeHTML( aTitle ) + "</h7>" );
  136. r->Report( "<ul><li>" + EscapeHTML( getItemDescription( a ) ) + "</li></ul>" );
  137. }
  138. void BOARD_INSPECTION_TOOL::reportHeader( const wxString& aTitle, BOARD_ITEM* a, BOARD_ITEM* b,
  139. REPORTER* r )
  140. {
  141. r->Report( "<h7>" + EscapeHTML( aTitle ) + "</h7>" );
  142. r->Report( "<ul><li>" + EscapeHTML( getItemDescription( a ) ) + "</li>"
  143. + "<li>" + EscapeHTML( getItemDescription( b ) ) + "</li></ul>" );
  144. }
  145. void BOARD_INSPECTION_TOOL::reportHeader( const wxString& aTitle, BOARD_ITEM* a, BOARD_ITEM* b,
  146. PCB_LAYER_ID aLayer, REPORTER* r )
  147. {
  148. wxString layerStr = _( "Layer" ) + wxS( " " ) + m_frame->GetBoard()->GetLayerName( aLayer );
  149. r->Report( "<h7>" + EscapeHTML( aTitle ) + "</h7>" );
  150. r->Report( "<ul><li>" + EscapeHTML( layerStr ) + "</li>"
  151. + "<li>" + EscapeHTML( getItemDescription( a ) ) + "</li>"
  152. + "<li>" + EscapeHTML( getItemDescription( b ) ) + "</li></ul>" );
  153. }
  154. void BOARD_INSPECTION_TOOL::InspectDRCError( const std::shared_ptr<RC_ITEM>& aDRCItem )
  155. {
  156. BOARD_ITEM* a = m_frame->GetBoard()->GetItem( aDRCItem->GetMainItemID() );
  157. BOARD_ITEM* b = m_frame->GetBoard()->GetItem( aDRCItem->GetAuxItemID() );
  158. PCB_LAYER_ID layer = m_frame->GetActiveLayer();
  159. if( !a || !b )
  160. return;
  161. if( m_inspectClearanceDialog == nullptr )
  162. {
  163. m_inspectClearanceDialog = std::make_unique<DIALOG_CONSTRAINTS_REPORTER>( m_frame );
  164. m_inspectClearanceDialog->SetTitle( _( "Clearance Report" ) );
  165. m_inspectClearanceDialog->Connect( wxEVT_CLOSE_WINDOW,
  166. wxCommandEventHandler( BOARD_INSPECTION_TOOL::onInspectClearanceDialogClosed ),
  167. nullptr, this );
  168. }
  169. WX_HTML_REPORT_BOX* r = nullptr;
  170. bool compileError = false;
  171. DRC_ENGINE drcEngine = makeDRCEngine( &compileError );
  172. DRC_CONSTRAINT constraint;
  173. int clearance = 0;
  174. wxString clearanceStr;
  175. switch( aDRCItem->GetErrorCode() )
  176. {
  177. case DRCE_EDGE_CLEARANCE:
  178. r = m_inspectClearanceDialog->AddPage( _( "Clearance" ) );
  179. reportHeader( _( "Edge clearance resolution for:" ), a, b, r );
  180. if( compileError )
  181. reportCompileError( r );
  182. constraint = drcEngine.EvalRules( EDGE_CLEARANCE_CONSTRAINT, a, b, layer, r );
  183. clearance = constraint.m_Value.Min();
  184. clearanceStr = StringFromValue( r->GetUnits(), clearance, true );
  185. r->Report( "" );
  186. r->Report( wxString::Format( _( "Resolved clearance: %s." ), clearanceStr ) );
  187. break;
  188. case DRCE_CLEARANCE:
  189. if( a->Type() == PCB_TRACE_T || a->Type() == PCB_ARC_T )
  190. {
  191. layer = a->GetLayer();
  192. }
  193. else if( b->Type() == PCB_TRACE_T || b->Type() == PCB_ARC_T )
  194. {
  195. layer = b->GetLayer();
  196. }
  197. else if( a->Type() == PCB_PAD_T && static_cast<PAD*>( a )->GetAttribute() == PAD_ATTRIB::SMD )
  198. {
  199. PAD* pad = static_cast<PAD*>( a );
  200. if( pad->IsOnLayer( F_Cu ) )
  201. layer = F_Cu;
  202. else
  203. layer = B_Cu;
  204. }
  205. else if( b->Type() == PCB_PAD_T && static_cast<PAD*>( a )->GetAttribute() == PAD_ATTRIB::SMD )
  206. {
  207. PAD* pad = static_cast<PAD*>( b );
  208. if( pad->IsOnLayer( F_Cu ) )
  209. layer = F_Cu;
  210. else
  211. layer = B_Cu;
  212. }
  213. r = m_inspectClearanceDialog->AddPage( _( "Clearance" ) );
  214. reportHeader( _( "Clearance resolution for:" ), a, b, layer, r );
  215. if( compileError )
  216. reportCompileError( r );
  217. constraint = drcEngine.EvalRules( CLEARANCE_CONSTRAINT, a, a, layer, r );
  218. clearance = constraint.m_Value.Min();
  219. clearanceStr = StringFromValue( r->GetUnits(), clearance, true );
  220. r->Report( "" );
  221. r->Report( wxString::Format( _( "Resolved clearance: %s." ), clearanceStr ) );
  222. break;
  223. default:
  224. return;
  225. }
  226. r->Flush();
  227. m_inspectClearanceDialog->Raise();
  228. m_inspectClearanceDialog->Show( true );
  229. }
  230. bool hasHole( BOARD_ITEM* aItem )
  231. {
  232. PAD* pad = dynamic_cast<PAD*>( aItem );
  233. if( pad && pad->GetDrillSizeX() > 0 && pad->GetDrillSizeY() > 0 )
  234. return true;
  235. PCB_VIA* via = dynamic_cast<PCB_VIA*>( aItem );
  236. if( via )
  237. return true;
  238. return false;
  239. };
  240. int BOARD_INSPECTION_TOOL::InspectClearance( const TOOL_EVENT& aEvent )
  241. {
  242. PCB_SELECTION_TOOL* selTool = m_toolMgr->GetTool<PCB_SELECTION_TOOL>();
  243. const PCB_SELECTION& selection = selTool->GetSelection();
  244. if( selection.Size() != 2 )
  245. {
  246. m_frame->ShowInfoBarError( _( "Select two items for a clearance resolution report." ) );
  247. return 0;
  248. }
  249. BOARD_ITEM* a = static_cast<BOARD_ITEM*>( selection.GetItem( 0 ) );
  250. BOARD_ITEM* b = static_cast<BOARD_ITEM*>( selection.GetItem( 1 ) );
  251. wxCHECK( a && b, 0 );
  252. if( a->Type() == PCB_GROUP_T )
  253. {
  254. PCB_GROUP* ag = static_cast<PCB_GROUP*>( a );
  255. if( ag->GetItems().empty() )
  256. {
  257. m_frame->ShowInfoBarError( _( "Cannot generate clearance report on empty group." ) );
  258. return 0;
  259. }
  260. a = *ag->GetItems().begin();
  261. }
  262. if( b->Type() == PCB_GROUP_T )
  263. {
  264. PCB_GROUP* bg = static_cast<PCB_GROUP*>( b );
  265. if( bg->GetItems().empty() )
  266. {
  267. m_frame->ShowInfoBarError( _( "Cannot generate clearance report on empty group." ) );
  268. return 0;
  269. }
  270. b = *bg->GetItems().begin();
  271. }
  272. // a and b could be null after group tests above.
  273. wxCHECK( a && b, 0 );
  274. if( m_inspectClearanceDialog == nullptr )
  275. {
  276. m_inspectClearanceDialog = std::make_unique<DIALOG_CONSTRAINTS_REPORTER>( m_frame );
  277. m_inspectClearanceDialog->SetTitle( _( "Clearance Report" ) );
  278. m_inspectClearanceDialog->Connect( wxEVT_CLOSE_WINDOW,
  279. wxCommandEventHandler( BOARD_INSPECTION_TOOL::onInspectClearanceDialogClosed ),
  280. nullptr, this );
  281. }
  282. m_inspectClearanceDialog->DeleteAllPages();
  283. if( a->Type() != PCB_ZONE_T && b->Type() == PCB_ZONE_T )
  284. std::swap( a, b );
  285. else if( !a->IsConnected() && b->IsConnected() )
  286. std::swap( a, b );
  287. WX_HTML_REPORT_BOX* r = nullptr;
  288. EDA_UNITS units = m_frame->GetUserUnits();
  289. PCB_LAYER_ID active = m_frame->GetActiveLayer();
  290. LSET layerIntersection = a->GetLayerSet() & b->GetLayerSet();
  291. LSET copperIntersection = layerIntersection & LSET::AllCuMask();
  292. BOARD_CONNECTED_ITEM* ac = dynamic_cast<BOARD_CONNECTED_ITEM*>( a );
  293. BOARD_CONNECTED_ITEM* bc = dynamic_cast<BOARD_CONNECTED_ITEM*>( b );
  294. ZONE* zone = dynamic_cast<ZONE*>( a );
  295. PAD* pad = dynamic_cast<PAD*>( b );
  296. FOOTPRINT* aFP = dynamic_cast<FOOTPRINT*>( a );
  297. FOOTPRINT* bFP = dynamic_cast<FOOTPRINT*>( b );
  298. bool compileError = false;
  299. DRC_ENGINE drcEngine = makeDRCEngine( &compileError );
  300. DRC_CONSTRAINT constraint;
  301. int clearance = 0;
  302. if( copperIntersection.any() && zone && pad && zone->GetNetCode() == pad->GetNetCode() )
  303. {
  304. PCB_LAYER_ID layer = active;
  305. if( !zone->IsOnLayer( active ) )
  306. layer = zone->GetLayerSet().Seq().front();
  307. r = m_inspectClearanceDialog->AddPage( _( "Zone" ) );
  308. reportHeader( _( "Zone connection resolution for:" ), a, b, layer, r );
  309. constraint = drcEngine.EvalZoneConnection( pad, zone, layer, r );
  310. if( constraint.m_ZoneConnection == ZONE_CONNECTION::THERMAL )
  311. {
  312. constraint = drcEngine.EvalRules( THERMAL_RELIEF_GAP_CONSTRAINT, pad, zone, layer, r );
  313. int gap = constraint.m_Value.Min();
  314. r->Report( wxString::Format( _( "Resolved thermal relief gap: %s." ),
  315. StringFromValue( units, gap, true ) ) );
  316. constraint = drcEngine.EvalRules( THERMAL_SPOKE_WIDTH_CONSTRAINT, pad, zone, layer, r );
  317. int width = constraint.m_Value.Opt();
  318. if( compileError )
  319. reportCompileError( r );
  320. r->Report( "" );
  321. r->Report( wxString::Format( _( "Resolved thermal spoke width: %s." ),
  322. StringFromValue( units, width, true ) ) );
  323. }
  324. else if( constraint.m_ZoneConnection == ZONE_CONNECTION::NONE )
  325. {
  326. clearance = zone->GetLocalClearance();
  327. r->Report( wxString::Format( _( "Zone clearance: %s." ),
  328. StringFromValue( units, clearance, true ) ) );
  329. constraint = drcEngine.EvalRules( THERMAL_RELIEF_GAP_CONSTRAINT, pad, zone, layer, r );
  330. if( constraint.m_Value.Min() > clearance )
  331. {
  332. clearance = constraint.m_Value.Min();
  333. r->Report( wxString::Format( _( "Overridden by larger thermal relief from %s;"
  334. "clearance: %s." ),
  335. EscapeHTML( constraint.GetName() ),
  336. StringFromValue( units, clearance, true ) ) );
  337. }
  338. if( compileError )
  339. reportCompileError( r );
  340. r->Report( "" );
  341. r->Report( wxString::Format( _( "Clearance: %s." ),
  342. StringFromValue( units, 0, true ) ) );
  343. }
  344. else
  345. {
  346. if( compileError )
  347. reportCompileError( r );
  348. // Report a 0 clearance for solid connections
  349. r->Report( "" );
  350. r->Report( wxString::Format( _( "Clearance: %s." ),
  351. StringFromValue( units, 0, true ) ) );
  352. }
  353. r->Flush();
  354. }
  355. else if( copperIntersection.any() && !aFP && !bFP )
  356. {
  357. PCB_LAYER_ID layer = active;
  358. if( !copperIntersection.test( layer ) )
  359. layer = copperIntersection.Seq().front();
  360. r = m_inspectClearanceDialog->AddPage( m_frame->GetBoard()->GetLayerName( layer ) );
  361. reportHeader( _( "Clearance resolution for:" ), a, b, layer, r );
  362. if( ac && bc && ac->GetNetCode() > 0 && ac->GetNetCode() == bc->GetNetCode() )
  363. {
  364. // Same nets....
  365. r->Report( _( "Items belong to the same net. Clearance is 0." ) );
  366. }
  367. else
  368. {
  369. // Different nets (or one or both unconnected)....
  370. constraint = drcEngine.EvalRules( CLEARANCE_CONSTRAINT, a, b, layer, r );
  371. clearance = constraint.m_Value.Min();
  372. if( compileError )
  373. reportCompileError( r );
  374. r->Report( "" );
  375. r->Report( wxString::Format( _( "Resolved clearance: %s." ),
  376. StringFromValue( units, clearance, true ) ) );
  377. }
  378. r->Flush();
  379. }
  380. for( PCB_LAYER_ID layer : { F_SilkS, B_SilkS } )
  381. {
  382. PCB_LAYER_ID correspondingMask = IsFrontLayer( layer ) ? F_Mask : B_Mask;
  383. if( ( a->IsOnLayer( layer ) && b->IsOnLayer( layer ) )
  384. || ( a->IsOnLayer( layer ) && b->IsOnLayer( correspondingMask ) )
  385. || ( b->IsOnLayer( layer ) && a->IsOnLayer( correspondingMask ) ) )
  386. {
  387. r = m_inspectClearanceDialog->AddPage( m_frame->GetBoard()->GetLayerName( layer ) );
  388. reportHeader( _( "Silkscreen clearance resolution for:" ), a, b, layer, r );
  389. constraint = drcEngine.EvalRules( SILK_CLEARANCE_CONSTRAINT, a, b, layer, r );
  390. clearance = constraint.m_Value.Min();
  391. if( compileError )
  392. reportCompileError( r );
  393. r->Report( "" );
  394. r->Report( wxString::Format( _( "Resolved clearance: %s." ),
  395. StringFromValue( units, clearance, true ) ) );
  396. r->Flush();
  397. }
  398. }
  399. for( PCB_LAYER_ID layer : { F_CrtYd, B_CrtYd } )
  400. {
  401. bool aCourtyard = aFP && !aFP->GetPolyCourtyard( layer ).IsEmpty();
  402. bool bCourtyard = bFP && !bFP->GetPolyCourtyard( layer ).IsEmpty();
  403. if( aCourtyard && bCourtyard )
  404. {
  405. r = m_inspectClearanceDialog->AddPage( m_frame->GetBoard()->GetLayerName( layer ) );
  406. reportHeader( _( "Courtyard clearance resolution for:" ), a, b, layer, r );
  407. constraint = drcEngine.EvalRules( COURTYARD_CLEARANCE_CONSTRAINT, a, b, layer, r );
  408. clearance = constraint.m_Value.Min();
  409. if( compileError )
  410. reportCompileError( r );
  411. r->Report( "" );
  412. r->Report( wxString::Format( _( "Resolved clearance: %s." ),
  413. StringFromValue( units, clearance, true ) ) );
  414. r->Flush();
  415. }
  416. }
  417. if( hasHole( a ) || hasHole( b ) )
  418. {
  419. PCB_LAYER_ID layer = UNDEFINED_LAYER;
  420. if( hasHole( a ) && b->IsOnLayer( active ) && IsCopperLayer( active ) )
  421. layer = active;
  422. else if( hasHole( b ) && a->IsOnLayer( active ) && IsCopperLayer( active ) )
  423. layer = active;
  424. else if( hasHole( a ) && IsCopperLayer( b->GetLayer() ) )
  425. layer = b->GetLayer();
  426. else if( hasHole( b ) && IsCopperLayer( a->GetLayer() ) )
  427. layer = a->GetLayer();
  428. if( layer >= 0 )
  429. {
  430. r = m_inspectClearanceDialog->AddPage( _( "Hole" ) );
  431. reportHeader( _( "Hole clearance resolution for:" ), a, b, layer, r );
  432. constraint = drcEngine.EvalRules( HOLE_CLEARANCE_CONSTRAINT, a, b, layer, r );
  433. clearance = constraint.m_Value.Min();
  434. if( compileError )
  435. reportCompileError( r );
  436. r->Report( "" );
  437. r->Report( wxString::Format( _( "Resolved clearance: %s." ),
  438. StringFromValue( units, clearance, true ) ) );
  439. r->Flush();
  440. }
  441. }
  442. for( PCB_LAYER_ID edgeLayer : { Edge_Cuts, Margin } )
  443. {
  444. PCB_LAYER_ID layer = UNDEFINED_LAYER;
  445. if( a->IsOnLayer( edgeLayer ) && b->IsOnLayer( active ) && IsCopperLayer( active ) )
  446. layer = active;
  447. else if( b->IsOnLayer( edgeLayer ) && a->IsOnLayer( active ) && IsCopperLayer( active ) )
  448. layer = active;
  449. else if( a->IsOnLayer( edgeLayer ) && IsCopperLayer( b->GetLayer() ) )
  450. layer = b->GetLayer();
  451. else if( b->IsOnLayer( edgeLayer ) && IsCopperLayer( a->GetLayer() ) )
  452. layer = a->GetLayer();
  453. if( layer >= 0 )
  454. {
  455. wxString layerName = m_frame->GetBoard()->GetLayerName( edgeLayer );
  456. r = m_inspectClearanceDialog->AddPage( layerName + wxS( " " ) + _( "Clearance" ) );
  457. reportHeader( _( "Edge clearance resolution for:" ), a, b, layer, r );
  458. constraint = drcEngine.EvalRules( EDGE_CLEARANCE_CONSTRAINT, a, b, layer, r );
  459. clearance = constraint.m_Value.Min();
  460. if( compileError )
  461. reportCompileError( r );
  462. r->Report( "" );
  463. r->Report( wxString::Format( _( "Resolved clearance: %s." ),
  464. StringFromValue( units, clearance, true ) ) );
  465. r->Flush();
  466. }
  467. }
  468. r = m_inspectClearanceDialog->AddPage( _( "Mechanical" ) );
  469. if( layerIntersection.any() )
  470. {
  471. PCB_LAYER_ID layer = active;
  472. if( !layerIntersection.test( layer ) )
  473. layer = layerIntersection.Seq().front();
  474. reportHeader( _( "Mechanical clearance resolution for:" ), a, b, layer, r );
  475. constraint = drcEngine.EvalRules( MECHANICAL_CLEARANCE_CONSTRAINT, a, b, layer, r );
  476. clearance = constraint.m_Value.Min();
  477. if( compileError )
  478. {
  479. reportCompileError( r );
  480. }
  481. else if( !drcEngine.HasRulesForConstraintType( MECHANICAL_CLEARANCE_CONSTRAINT ) )
  482. {
  483. r->Report( "" );
  484. r->Report( _( "No 'mechanical_clearance' constraints defined." ) );
  485. }
  486. else
  487. {
  488. r->Report( "" );
  489. r->Report( wxString::Format( _( "Resolved clearance: %s." ),
  490. StringFromValue( units, clearance, true ) ) );
  491. }
  492. r->Report( "" );
  493. r->Report( "" );
  494. r->Report( "" );
  495. }
  496. if( hasHole( a ) || hasHole( b ) )
  497. {
  498. PCB_LAYER_ID layer;
  499. if( hasHole( a ) && b->IsOnLayer( active ) )
  500. layer = active;
  501. else if( hasHole( b ) && a->IsOnLayer( active ) )
  502. layer = active;
  503. else if( hasHole( a ) )
  504. layer = b->GetLayer();
  505. else
  506. layer = a->GetLayer();
  507. reportHeader( _( "Mechanical hole clearance resolution for:" ), a, b, layer, r );
  508. constraint = drcEngine.EvalRules( MECHANICAL_HOLE_CLEARANCE_CONSTRAINT, a, b, layer, r );
  509. clearance = constraint.m_Value.Min();
  510. if( compileError )
  511. {
  512. reportCompileError( r );
  513. }
  514. else if( !drcEngine.HasRulesForConstraintType( MECHANICAL_HOLE_CLEARANCE_CONSTRAINT ) )
  515. {
  516. r->Report( "" );
  517. r->Report( _( "No 'mechanical_hole_clearance' constraints defined." ) );
  518. }
  519. else
  520. {
  521. r->Report( "" );
  522. r->Report( wxString::Format( _( "Resolved clearance: %s." ),
  523. StringFromValue( units, clearance, true ) ) );
  524. }
  525. }
  526. r->Flush();
  527. m_inspectClearanceDialog->Raise();
  528. m_inspectClearanceDialog->Show( true );
  529. return 0;
  530. }
  531. wxString reportMin( EDA_UNITS aUnits, DRC_CONSTRAINT& aConstraint )
  532. {
  533. if( aConstraint.m_Value.HasMin() )
  534. return StringFromValue( aUnits, aConstraint.m_Value.Min(), true );
  535. else
  536. return wxT( "<i>" ) + _( "undefined" ) + wxT( "</i>" );
  537. }
  538. wxString reportOpt( EDA_UNITS aUnits, DRC_CONSTRAINT& aConstraint )
  539. {
  540. if( aConstraint.m_Value.HasOpt() )
  541. return StringFromValue( aUnits, aConstraint.m_Value.Opt(), true );
  542. else
  543. return wxT( "<i>" ) + _( "undefined" ) + wxT( "</i>" );
  544. }
  545. wxString reportMax( EDA_UNITS aUnits, DRC_CONSTRAINT& aConstraint )
  546. {
  547. if( aConstraint.m_Value.HasMax() )
  548. return StringFromValue( aUnits, aConstraint.m_Value.Max(), true );
  549. else
  550. return wxT( "<i>" ) + _( "undefined" ) + wxT( "</i>" );
  551. }
  552. int BOARD_INSPECTION_TOOL::InspectConstraints( const TOOL_EVENT& aEvent )
  553. {
  554. #define EVAL_RULES( constraint, a, b, layer, r ) drcEngine.EvalRules( constraint, a, b, layer, r )
  555. PCB_SELECTION_TOOL* selTool = m_toolMgr->GetTool<PCB_SELECTION_TOOL>();
  556. const PCB_SELECTION& selection = selTool->GetSelection();
  557. if( selection.Size() != 1 )
  558. {
  559. m_frame->ShowInfoBarError( _( "Select an item for a constraints resolution report." ) );
  560. return 0;
  561. }
  562. if( m_inspectConstraintsDialog == nullptr )
  563. {
  564. m_inspectConstraintsDialog = std::make_unique<DIALOG_CONSTRAINTS_REPORTER>( m_frame );
  565. m_inspectConstraintsDialog->SetTitle( _( "Constraints Report" ) );
  566. m_inspectConstraintsDialog->Connect( wxEVT_CLOSE_WINDOW,
  567. wxCommandEventHandler( BOARD_INSPECTION_TOOL::onInspectConstraintsDialogClosed ),
  568. nullptr, this );
  569. }
  570. m_inspectConstraintsDialog->DeleteAllPages();
  571. BOARD_ITEM* item = static_cast<BOARD_ITEM*>( selection.GetItem( 0 ) );
  572. bool compileError = false;
  573. bool courtyardError = false;
  574. DRC_ENGINE drcEngine = makeDRCEngine( &compileError, &courtyardError );
  575. DRC_CONSTRAINT constraint;
  576. WX_HTML_REPORT_BOX* r = nullptr;
  577. if( item->Type() == PCB_TRACE_T )
  578. {
  579. r = m_inspectConstraintsDialog->AddPage( _( "Track Width" ) );
  580. reportHeader( _( "Track width resolution for:" ), item, r );
  581. constraint = EVAL_RULES( TRACK_WIDTH_CONSTRAINT, item, nullptr, item->GetLayer(), r );
  582. if( compileError )
  583. reportCompileError( r );
  584. r->Report( "" );
  585. r->Report( wxString::Format( _( "Width constraints: min %s; opt %s; max %s." ),
  586. reportMin( r->GetUnits(), constraint ),
  587. reportOpt( r->GetUnits(), constraint ),
  588. reportMax( r->GetUnits(), constraint ) ) );
  589. r->Flush();
  590. }
  591. if( item->Type() == PCB_VIA_T )
  592. {
  593. r = m_inspectConstraintsDialog->AddPage( _( "Via Diameter" ) );
  594. reportHeader( _( "Via diameter resolution for:" ), item, r );
  595. // PADSTACKS TODO: once we have padstacks we'll need to run this per-layer....
  596. constraint = EVAL_RULES( VIA_DIAMETER_CONSTRAINT, item, nullptr, UNDEFINED_LAYER, r );
  597. if( compileError )
  598. reportCompileError( r );
  599. r->Report( "" );
  600. r->Report( wxString::Format( _( "Diameter constraints: min %s; opt %s; max %s." ),
  601. reportMin( r->GetUnits(), constraint ),
  602. reportOpt( r->GetUnits(), constraint ),
  603. reportMax( r->GetUnits(), constraint ) ) );
  604. r->Flush();
  605. r = m_inspectConstraintsDialog->AddPage( _( "Via Annular Width" ) );
  606. reportHeader( _( "Via annular width resolution for:" ), item, r );
  607. // PADSTACKS TODO: once we have padstacks we'll need to run this per-layer....
  608. constraint = EVAL_RULES( ANNULAR_WIDTH_CONSTRAINT, item, nullptr, UNDEFINED_LAYER, r );
  609. if( compileError )
  610. reportCompileError( r );
  611. r->Report( "" );
  612. r->Report( wxString::Format( _( "Annular width constraints: min %s; opt %s; max %s." ),
  613. reportMin( r->GetUnits(), constraint ),
  614. reportOpt( r->GetUnits(), constraint ),
  615. reportMax( r->GetUnits(), constraint ) ) );
  616. r->Flush();
  617. }
  618. if( ( item->Type() == PCB_PAD_T && static_cast<PAD*>( item )->GetDrillSize().x > 0 )
  619. || item->Type() == PCB_VIA_T )
  620. {
  621. r = m_inspectConstraintsDialog->AddPage( _( "Hole Size" ) );
  622. reportHeader( _( "Hole diameter resolution for:" ), item, r );
  623. // PADSTACKS TODO: once we have padstacks we'll need to run this per-layer....
  624. constraint = EVAL_RULES( HOLE_SIZE_CONSTRAINT, item, nullptr, UNDEFINED_LAYER, r );
  625. if( compileError )
  626. reportCompileError( r );
  627. wxString min = _( "undefined" );
  628. if( constraint.m_Value.HasMin() )
  629. min = StringFromValue( r->GetUnits(), constraint.m_Value.Min(), true );
  630. r->Report( "" );
  631. r->Report( wxString::Format( _( "Hole constraint: min %s." ), min ) );
  632. r->Flush();
  633. }
  634. r = m_inspectConstraintsDialog->AddPage( _( "Keepouts" ) );
  635. reportHeader( _( "Keepout resolution for:" ), item, r );
  636. constraint = EVAL_RULES( DISALLOW_CONSTRAINT, item, nullptr, item->GetLayer(), r );
  637. if( compileError )
  638. reportCompileError( r );
  639. if( courtyardError )
  640. {
  641. r->Report( "" );
  642. r->Report( _( "Report may be incomplete: some footprint courtyards are malformed." )
  643. + " <a href='drc'>" + _( "Run DRC for a full analysis." ) + "</a>" );
  644. }
  645. r->Report( "" );
  646. if( constraint.m_DisallowFlags )
  647. r->Report( _( "Item <b>disallowed</b> at current location." ) );
  648. else
  649. r->Report( _( "Item allowed at current location." ) );
  650. r->Flush();
  651. r = m_inspectConstraintsDialog->AddPage( _( "Assertions" ) );
  652. reportHeader( _( "Assertions for:" ), item, r );
  653. if( compileError )
  654. reportCompileError( r );
  655. if( courtyardError )
  656. {
  657. r->Report( "" );
  658. r->Report( _( "Report may be incomplete: some footprint courtyards are malformed." )
  659. + " <a href='drc'>" + _( "Run DRC for a full analysis." ) + "</a>" );
  660. }
  661. drcEngine.ProcessAssertions( item, []( const DRC_CONSTRAINT* c ){}, r );
  662. r->Flush();
  663. m_inspectConstraintsDialog->FinishInitialization();
  664. m_inspectConstraintsDialog->Raise();
  665. m_inspectConstraintsDialog->Show( true );
  666. return 0;
  667. }
  668. int BOARD_INSPECTION_TOOL::CrossProbePcbToSch( const TOOL_EVENT& aEvent )
  669. {
  670. // Don't get in an infinite loop PCB -> SCH -> PCB -> SCH -> ...
  671. if( m_probingSchToPcb )
  672. return 0;
  673. PCB_SELECTION_TOOL* selTool = m_toolMgr->GetTool<PCB_SELECTION_TOOL>();
  674. const PCB_SELECTION& selection = selTool->GetSelection();
  675. if( selection.Size() == 1 )
  676. m_frame->SendMessageToEESCHEMA( static_cast<BOARD_ITEM*>( selection.Front() ) );
  677. else
  678. m_frame->SendMessageToEESCHEMA( nullptr );
  679. // Update 3D viewer highlighting
  680. m_frame->Update3DView( false, frame()->Settings().m_Display.m_Live3DRefresh );
  681. return 0;
  682. }
  683. int BOARD_INSPECTION_TOOL::HighlightItem( const TOOL_EVENT& aEvent )
  684. {
  685. BOARD_ITEM* item = aEvent.Parameter<BOARD_ITEM*>();
  686. m_probingSchToPcb = true; // recursion guard
  687. {
  688. m_toolMgr->RunAction( PCB_ACTIONS::selectionClear, true );
  689. if( item )
  690. m_toolMgr->RunAction( PCB_ACTIONS::selectItem, true, (void*) item );
  691. }
  692. m_probingSchToPcb = false;
  693. bool request3DviewRedraw = frame()->Settings().m_Display.m_Live3DRefresh;
  694. if( item && item->Type() != PCB_FOOTPRINT_T )
  695. request3DviewRedraw = false;
  696. // Update 3D viewer highlighting
  697. if( request3DviewRedraw )
  698. m_frame->Update3DView( false, true );
  699. return 0;
  700. }
  701. bool BOARD_INSPECTION_TOOL::highlightNet( const VECTOR2D& aPosition, bool aUseSelection )
  702. {
  703. BOARD* board = static_cast<BOARD*>( m_toolMgr->GetModel() );
  704. KIGFX::RENDER_SETTINGS* settings = getView()->GetPainter()->GetSettings();
  705. PCB_SELECTION_TOOL* selectionTool = m_toolMgr->GetTool<PCB_SELECTION_TOOL>();
  706. int net = -1;
  707. bool enableHighlight = false;
  708. if( aUseSelection )
  709. {
  710. const PCB_SELECTION& selection = selectionTool->GetSelection();
  711. std::set<int> netcodes;
  712. for( EDA_ITEM* item : selection )
  713. {
  714. if( BOARD_CONNECTED_ITEM* ci = dynamic_cast<BOARD_CONNECTED_ITEM*>( item ) )
  715. netcodes.insert( ci->GetNetCode() );
  716. }
  717. enableHighlight = !netcodes.empty();
  718. if( enableHighlight && netcodes.size() > 1 )
  719. {
  720. // If we are doing a multi-highlight, cross-probing back and other stuff is not
  721. // yet supported
  722. settings->SetHighlight( netcodes );
  723. board->ResetNetHighLight();
  724. for( int multiNet : netcodes )
  725. board->SetHighLightNet( multiNet, true );
  726. board->HighLightON();
  727. m_toolMgr->GetView()->UpdateAllLayersColor();
  728. m_currentlyHighlighted = netcodes;
  729. return true;
  730. }
  731. else if( enableHighlight )
  732. {
  733. net = *netcodes.begin();
  734. }
  735. }
  736. // If we didn't get a net to highlight from the selection, use the cursor
  737. if( net < 0 )
  738. {
  739. GENERAL_COLLECTORS_GUIDE guide = m_frame->GetCollectorsGuide();
  740. guide.SetIgnoreZoneFills( false );
  741. PCB_LAYER_ID activeLayer = static_cast<PCB_LAYER_ID>( view()->GetTopLayer() );
  742. guide.SetPreferredLayer( activeLayer );
  743. GENERAL_COLLECTOR collector;
  744. collector.Collect( board, GENERAL_COLLECTOR::PadsOrTracks, (wxPoint) aPosition, guide );
  745. if( collector.GetCount() == 0 )
  746. collector.Collect( board, GENERAL_COLLECTOR::Zones, (wxPoint) aPosition, guide );
  747. // Apply the active selection filter, except we want to allow picking locked items for
  748. // highlighting even if the user has disabled them for selection
  749. SELECTION_FILTER_OPTIONS& filter = selectionTool->GetFilter();
  750. bool saved = filter.lockedItems;
  751. filter.lockedItems = true;
  752. selectionTool->FilterCollectedItems( collector, true );
  753. filter.lockedItems = saved;
  754. // Clear the previous highlight
  755. m_frame->SendMessageToEESCHEMA( nullptr );
  756. bool highContrast = settings->GetHighContrast();
  757. PCB_LAYER_ID contrastLayer = settings->GetPrimaryHighContrastLayer();
  758. for( int i = collector.GetCount() - 1; i >= 0; i-- )
  759. {
  760. LSET itemLayers = collector[i]->GetLayerSet();
  761. if( ( itemLayers & LSET::AllCuMask() ).none() ||
  762. ( highContrast && !itemLayers.Contains( contrastLayer ) ) )
  763. {
  764. collector.Remove( i );
  765. continue;
  766. }
  767. }
  768. enableHighlight = ( collector.GetCount() > 0 );
  769. // Obtain net code for the clicked item
  770. if( enableHighlight )
  771. {
  772. BOARD_CONNECTED_ITEM* targetItem = static_cast<BOARD_CONNECTED_ITEM*>( collector[0] );
  773. if( targetItem->Type() == PCB_PAD_T )
  774. m_frame->SendMessageToEESCHEMA( targetItem );
  775. net = targetItem->GetNetCode();
  776. }
  777. }
  778. auto& netcodes = settings->GetHighlightNetCodes();
  779. // Toggle highlight when the same net was picked
  780. if( net > 0 && netcodes.count( net ) )
  781. enableHighlight = !settings->IsHighlightEnabled();
  782. if( enableHighlight != settings->IsHighlightEnabled() || !netcodes.count( net ) )
  783. {
  784. if( !netcodes.empty() )
  785. m_lastHighlighted = netcodes;
  786. settings->SetHighlight( enableHighlight, net );
  787. m_toolMgr->GetView()->UpdateAllLayersColor();
  788. }
  789. // Store the highlighted netcode in the current board (for dialogs for instance)
  790. if( enableHighlight && net >= 0 )
  791. {
  792. m_currentlyHighlighted = netcodes;
  793. board->SetHighLightNet( net );
  794. board->HighLightON();
  795. NETINFO_ITEM* netinfo = board->FindNet( net );
  796. if( netinfo )
  797. {
  798. std::vector<MSG_PANEL_ITEM> items;
  799. netinfo->GetMsgPanelInfo( m_frame, items );
  800. m_frame->SetMsgPanel( items );
  801. m_frame->SendCrossProbeNetName( netinfo->GetNetname() );
  802. }
  803. }
  804. else
  805. {
  806. m_currentlyHighlighted.clear();
  807. board->ResetNetHighLight();
  808. m_frame->SetMsgPanel( board );
  809. m_frame->SendCrossProbeNetName( "" );
  810. }
  811. return true;
  812. }
  813. int BOARD_INSPECTION_TOOL::HighlightNet( const TOOL_EVENT& aEvent )
  814. {
  815. int netcode = aEvent.Parameter<intptr_t>();
  816. KIGFX::RENDER_SETTINGS* settings = m_toolMgr->GetView()->GetPainter()->GetSettings();
  817. const std::set<int>& highlighted = settings->GetHighlightNetCodes();
  818. if( netcode > 0 )
  819. {
  820. m_lastHighlighted = highlighted;
  821. settings->SetHighlight( true, netcode );
  822. m_toolMgr->GetView()->UpdateAllLayersColor();
  823. m_currentlyHighlighted.clear();
  824. m_currentlyHighlighted.insert( netcode );
  825. }
  826. else if( aEvent.IsAction( &PCB_ACTIONS::highlightNetSelection ) )
  827. {
  828. // Highlight selection (cursor position will be ignored)
  829. highlightNet( getViewControls()->GetMousePosition(), true );
  830. }
  831. else if( aEvent.IsAction( &PCB_ACTIONS::toggleLastNetHighlight ) )
  832. {
  833. std::set<int> temp = highlighted;
  834. settings->SetHighlight( m_lastHighlighted );
  835. m_toolMgr->GetView()->UpdateAllLayersColor();
  836. m_currentlyHighlighted = m_lastHighlighted;
  837. m_lastHighlighted = temp;
  838. }
  839. else if( aEvent.IsAction( &PCB_ACTIONS::toggleNetHighlight ) )
  840. {
  841. bool turnOn = highlighted.empty() && !m_currentlyHighlighted.empty();
  842. settings->SetHighlight( m_currentlyHighlighted, turnOn );
  843. m_toolMgr->GetView()->UpdateAllLayersColor();
  844. }
  845. else // Highlight the net belonging to the item under the cursor
  846. {
  847. highlightNet( getViewControls()->GetMousePosition(), false );
  848. }
  849. return 0;
  850. }
  851. int BOARD_INSPECTION_TOOL::ClearHighlight( const TOOL_EVENT& aEvent )
  852. {
  853. BOARD* board = static_cast<BOARD*>( m_toolMgr->GetModel() );
  854. KIGFX::RENDER_SETTINGS* settings = m_toolMgr->GetView()->GetPainter()->GetSettings();
  855. m_currentlyHighlighted.clear();
  856. m_lastHighlighted.clear();
  857. board->ResetNetHighLight();
  858. settings->SetHighlight( false );
  859. m_toolMgr->GetView()->UpdateAllLayersColor();
  860. m_frame->SetMsgPanel( board );
  861. m_frame->SendCrossProbeNetName( "" );
  862. return 0;
  863. }
  864. #if 0
  865. int BOARD_INSPECTION_TOOL::HighlightNetTool( const TOOL_EVENT& aEvent )
  866. {
  867. std::string tool = aEvent.GetCommandStr().get();
  868. PCB_PICKER_TOOL* picker = m_toolMgr->GetTool<PCB_PICKER_TOOL>();
  869. // Deactivate other tools; particularly important if another PICKER is currently running
  870. Activate();
  871. // If the keyboard hotkey was triggered and we are already in the highlight tool, behave
  872. // the same as a left-click. Otherwise highlight the net of the selected item(s), or if
  873. // there is no selection, then behave like a ctrl-left-click.
  874. if( aEvent.IsAction( &PCB_ACTIONS::highlightNetSelection ) )
  875. {
  876. bool use_selection = m_frame->IsCurrentTool( PCB_ACTIONS::highlightNetTool );
  877. highlightNet( getViewControls()->GetMousePosition(), use_selection );
  878. }
  879. picker->SetClickHandler(
  880. [this] ( const VECTOR2D& pt ) -> bool
  881. {
  882. highlightNet( pt, false );
  883. return true;
  884. } );
  885. picker->SetLayerSet( LSET::AllCuMask() );
  886. picker->SetSnapping( false );
  887. m_toolMgr->RunAction( ACTIONS::pickerTool, true, &tool );
  888. return 0;
  889. }
  890. #endif
  891. int BOARD_INSPECTION_TOOL::LocalRatsnestTool( const TOOL_EVENT& aEvent )
  892. {
  893. std::string tool = aEvent.GetCommandStr().get();
  894. PCB_PICKER_TOOL* picker = m_toolMgr->GetTool<PCB_PICKER_TOOL>();
  895. BOARD* board = getModel<BOARD>();
  896. // Deactivate other tools; particularly important if another PICKER is currently running
  897. Activate();
  898. picker->SetClickHandler(
  899. [this, board]( const VECTOR2D& pt ) -> bool
  900. {
  901. PCB_SELECTION_TOOL* selectionTool = m_toolMgr->GetTool<PCB_SELECTION_TOOL>();
  902. m_toolMgr->RunAction( PCB_ACTIONS::selectionClear, true );
  903. m_toolMgr->RunAction( PCB_ACTIONS::selectionCursor, true, EDIT_TOOL::PadFilter );
  904. PCB_SELECTION& selection = selectionTool->GetSelection();
  905. if( selection.Empty() )
  906. {
  907. m_toolMgr->RunAction( PCB_ACTIONS::selectionCursor, true,
  908. EDIT_TOOL::FootprintFilter );
  909. selection = selectionTool->GetSelection();
  910. }
  911. if( selection.Empty() )
  912. {
  913. // Clear the previous local ratsnest if we click off all items
  914. for( FOOTPRINT* fp : board->Footprints() )
  915. {
  916. for( PAD* pad : fp->Pads() )
  917. pad->SetLocalRatsnestVisible( displayOptions().m_ShowGlobalRatsnest );
  918. }
  919. }
  920. else
  921. {
  922. for( EDA_ITEM* item : selection )
  923. {
  924. if( PAD* pad = dyn_cast<PAD*>( item) )
  925. {
  926. pad->SetLocalRatsnestVisible( !pad->GetLocalRatsnestVisible() );
  927. }
  928. else if( FOOTPRINT* fp = dyn_cast<FOOTPRINT*>( item) )
  929. {
  930. if( !fp->Pads().empty() )
  931. {
  932. bool enable = !fp->Pads()[0]->GetLocalRatsnestVisible();
  933. for( PAD* childPad : fp->Pads() )
  934. childPad->SetLocalRatsnestVisible( enable );
  935. }
  936. }
  937. }
  938. }
  939. m_toolMgr->GetView()->MarkTargetDirty( KIGFX::TARGET_OVERLAY );
  940. return true;
  941. } );
  942. picker->SetFinalizeHandler(
  943. [this, board]( int aCondition )
  944. {
  945. if( aCondition != PCB_PICKER_TOOL::END_ACTIVATE )
  946. {
  947. for( FOOTPRINT* fp : board->Footprints() )
  948. {
  949. for( PAD* pad : fp->Pads() )
  950. pad->SetLocalRatsnestVisible( displayOptions().m_ShowGlobalRatsnest );
  951. }
  952. }
  953. } );
  954. m_toolMgr->RunAction( ACTIONS::pickerTool, true, &tool );
  955. return 0;
  956. }
  957. int BOARD_INSPECTION_TOOL::UpdateSelectionRatsnest( const TOOL_EVENT& aEvent )
  958. {
  959. VECTOR2I delta;
  960. // If we have passed the simple move vector, we can update without recalculation
  961. if( aEvent.Parameter<VECTOR2I*>() )
  962. {
  963. delta = *aEvent.Parameter<VECTOR2I*>();
  964. delete aEvent.Parameter<VECTOR2I*>();
  965. }
  966. else
  967. {
  968. // We can delete the existing map to force a recalculation
  969. delete m_dynamicData;
  970. m_dynamicData = nullptr;
  971. }
  972. auto selectionTool = m_toolMgr->GetTool<PCB_SELECTION_TOOL>();
  973. auto& selection = selectionTool->GetSelection();
  974. auto connectivity = getModel<BOARD>()->GetConnectivity();
  975. if( selection.Empty() )
  976. {
  977. connectivity->ClearDynamicRatsnest();
  978. delete m_dynamicData;
  979. m_dynamicData = nullptr;
  980. }
  981. else
  982. {
  983. calculateSelectionRatsnest( delta );
  984. }
  985. return 0;
  986. }
  987. int BOARD_INSPECTION_TOOL::HideDynamicRatsnest( const TOOL_EVENT& aEvent )
  988. {
  989. getModel<BOARD>()->GetConnectivity()->ClearDynamicRatsnest();
  990. delete m_dynamicData;
  991. m_dynamicData = nullptr;
  992. return 0;
  993. }
  994. void BOARD_INSPECTION_TOOL::calculateSelectionRatsnest( const VECTOR2I& aDelta )
  995. {
  996. PCB_SELECTION_TOOL* selectionTool = m_toolMgr->GetTool<PCB_SELECTION_TOOL>();
  997. SELECTION& selection = selectionTool->GetSelection();
  998. std::shared_ptr<CONNECTIVITY_DATA> connectivity = board()->GetConnectivity();
  999. std::vector<BOARD_ITEM*> items;
  1000. std::deque<EDA_ITEM*> queued_items( selection.begin(), selection.end() );
  1001. for( std::size_t i = 0; i < queued_items.size(); ++i )
  1002. {
  1003. BOARD_ITEM* item = static_cast<BOARD_ITEM*>( queued_items[i] );
  1004. if( item->Type() == PCB_FOOTPRINT_T )
  1005. {
  1006. for( PAD* pad : static_cast<FOOTPRINT*>( item )->Pads() )
  1007. {
  1008. if( pad->GetLocalRatsnestVisible() || displayOptions().m_ShowModuleRatsnest )
  1009. items.push_back( pad );
  1010. }
  1011. }
  1012. else if( item->Type() == PCB_GROUP_T )
  1013. {
  1014. PCB_GROUP *group = static_cast<PCB_GROUP*>( item );
  1015. group->RunOnDescendants( [ &queued_items ]( BOARD_ITEM *aItem )
  1016. {
  1017. queued_items.push_back( aItem );
  1018. } );
  1019. }
  1020. else if( BOARD_CONNECTED_ITEM* boardItem = dyn_cast<BOARD_CONNECTED_ITEM*>( item ) )
  1021. {
  1022. if( boardItem->GetLocalRatsnestVisible() || displayOptions().m_ShowModuleRatsnest )
  1023. items.push_back( boardItem );
  1024. }
  1025. }
  1026. if( items.empty() || std::none_of( items.begin(), items.end(),
  1027. []( const BOARD_ITEM* aItem )
  1028. {
  1029. return( aItem->Type() == PCB_TRACE_T
  1030. || aItem->Type() == PCB_PAD_T
  1031. || aItem->Type() == PCB_ARC_T
  1032. || aItem->Type() == PCB_ZONE_T
  1033. || aItem->Type() == PCB_FOOTPRINT_T
  1034. || aItem->Type() == PCB_VIA_T );
  1035. } ) )
  1036. {
  1037. return;
  1038. }
  1039. if( !m_dynamicData )
  1040. {
  1041. m_dynamicData = new CONNECTIVITY_DATA( items, true );
  1042. connectivity->BlockRatsnestItems( items );
  1043. }
  1044. else
  1045. {
  1046. m_dynamicData->Move( aDelta );
  1047. }
  1048. connectivity->ComputeDynamicRatsnest( items, m_dynamicData );
  1049. }
  1050. int BOARD_INSPECTION_TOOL::ListNets( const TOOL_EVENT& aEvent )
  1051. {
  1052. if( m_listNetsDialog == nullptr )
  1053. {
  1054. m_listNetsDialog =
  1055. std::make_unique<DIALOG_NET_INSPECTOR>( m_frame, m_listNetsDialogSettings );
  1056. m_listNetsDialog->Connect( wxEVT_CLOSE_WINDOW,
  1057. wxCommandEventHandler( BOARD_INSPECTION_TOOL::onListNetsDialogClosed ), nullptr,
  1058. this );
  1059. m_listNetsDialog->Connect( wxEVT_BUTTON,
  1060. wxCommandEventHandler( BOARD_INSPECTION_TOOL::onListNetsDialogClosed ), nullptr,
  1061. this );
  1062. }
  1063. m_listNetsDialog->Raise();
  1064. m_listNetsDialog->Show( true );
  1065. return 0;
  1066. }
  1067. void BOARD_INSPECTION_TOOL::onListNetsDialogClosed( wxCommandEvent& event )
  1068. {
  1069. m_listNetsDialogSettings = m_listNetsDialog->Settings();
  1070. m_listNetsDialog->Disconnect( wxEVT_CLOSE_WINDOW,
  1071. wxCommandEventHandler( BOARD_INSPECTION_TOOL::onListNetsDialogClosed ), nullptr, this );
  1072. m_listNetsDialog->Disconnect( wxEVT_BUTTON,
  1073. wxCommandEventHandler( BOARD_INSPECTION_TOOL::onListNetsDialogClosed ), nullptr, this );
  1074. m_listNetsDialog->Destroy();
  1075. m_listNetsDialog.release();
  1076. }
  1077. void BOARD_INSPECTION_TOOL::onInspectClearanceDialogClosed( wxCommandEvent& event )
  1078. {
  1079. m_inspectClearanceDialog->Disconnect( wxEVT_CLOSE_WINDOW,
  1080. wxCommandEventHandler( BOARD_INSPECTION_TOOL::onInspectClearanceDialogClosed ),
  1081. nullptr, this );
  1082. m_inspectClearanceDialog->Destroy();
  1083. m_inspectClearanceDialog.release();
  1084. }
  1085. void BOARD_INSPECTION_TOOL::onInspectConstraintsDialogClosed( wxCommandEvent& event )
  1086. {
  1087. m_inspectConstraintsDialog->Disconnect( wxEVT_CLOSE_WINDOW,
  1088. wxCommandEventHandler( BOARD_INSPECTION_TOOL::onInspectConstraintsDialogClosed ),
  1089. nullptr, this );
  1090. m_inspectConstraintsDialog->Destroy();
  1091. m_inspectConstraintsDialog.release();
  1092. }
  1093. int BOARD_INSPECTION_TOOL::HideNet( const TOOL_EVENT& aEvent )
  1094. {
  1095. doHideNet( aEvent.Parameter<intptr_t>(), true );
  1096. return 0;
  1097. }
  1098. int BOARD_INSPECTION_TOOL::ShowNet( const TOOL_EVENT& aEvent )
  1099. {
  1100. doHideNet( aEvent.Parameter<intptr_t>(), false );
  1101. return 0;
  1102. }
  1103. void BOARD_INSPECTION_TOOL::doHideNet( int aNetCode, bool aHide )
  1104. {
  1105. KIGFX::PCB_RENDER_SETTINGS* rs = static_cast<KIGFX::PCB_RENDER_SETTINGS*>(
  1106. m_toolMgr->GetView()->GetPainter()->GetSettings() );
  1107. PCB_SELECTION_TOOL* selectionTool = m_toolMgr->GetTool<PCB_SELECTION_TOOL>();
  1108. SELECTION& selection = selectionTool->GetSelection();
  1109. if( aNetCode <= 0 && !selection.Empty() )
  1110. {
  1111. for( EDA_ITEM* item : selection )
  1112. {
  1113. if( BOARD_CONNECTED_ITEM* bci = dynamic_cast<BOARD_CONNECTED_ITEM*>( item ) )
  1114. {
  1115. if( bci->GetNetCode() > 0 )
  1116. doHideNet( bci->GetNetCode(), aHide );
  1117. }
  1118. }
  1119. return;
  1120. }
  1121. if( aHide )
  1122. rs->GetHiddenNets().insert( aNetCode );
  1123. else
  1124. rs->GetHiddenNets().erase( aNetCode );
  1125. m_frame->GetCanvas()->RedrawRatsnest();
  1126. m_frame->GetCanvas()->Refresh();
  1127. m_frame->GetAppearancePanel()->OnNetVisibilityChanged( aNetCode, !aHide );
  1128. }
  1129. void BOARD_INSPECTION_TOOL::setTransitions()
  1130. {
  1131. Go( &BOARD_INSPECTION_TOOL::CrossProbePcbToSch, EVENTS::SelectedEvent );
  1132. Go( &BOARD_INSPECTION_TOOL::CrossProbePcbToSch, EVENTS::UnselectedEvent );
  1133. Go( &BOARD_INSPECTION_TOOL::CrossProbePcbToSch, EVENTS::ClearedEvent );
  1134. Go( &BOARD_INSPECTION_TOOL::LocalRatsnestTool,
  1135. PCB_ACTIONS::localRatsnestTool.MakeEvent() );
  1136. Go( &BOARD_INSPECTION_TOOL::HideDynamicRatsnest,
  1137. PCB_ACTIONS::hideDynamicRatsnest.MakeEvent() );
  1138. Go( &BOARD_INSPECTION_TOOL::UpdateSelectionRatsnest,
  1139. PCB_ACTIONS::updateLocalRatsnest.MakeEvent() );
  1140. Go( &BOARD_INSPECTION_TOOL::ListNets, PCB_ACTIONS::listNets.MakeEvent() );
  1141. Go( &BOARD_INSPECTION_TOOL::ShowStatisticsDialog, PCB_ACTIONS::boardStatistics.MakeEvent() );
  1142. Go( &BOARD_INSPECTION_TOOL::InspectClearance, PCB_ACTIONS::inspectClearance.MakeEvent() );
  1143. Go( &BOARD_INSPECTION_TOOL::InspectConstraints, PCB_ACTIONS::inspectConstraints.MakeEvent() );
  1144. Go( &BOARD_INSPECTION_TOOL::HighlightNet, PCB_ACTIONS::highlightNet.MakeEvent() );
  1145. Go( &BOARD_INSPECTION_TOOL::HighlightNet, PCB_ACTIONS::highlightNetSelection.MakeEvent() );
  1146. Go( &BOARD_INSPECTION_TOOL::HighlightNet, PCB_ACTIONS::toggleLastNetHighlight.MakeEvent() );
  1147. Go( &BOARD_INSPECTION_TOOL::ClearHighlight, PCB_ACTIONS::clearHighlight.MakeEvent() );
  1148. Go( &BOARD_INSPECTION_TOOL::HighlightNet, PCB_ACTIONS::toggleNetHighlight.MakeEvent() );
  1149. Go( &BOARD_INSPECTION_TOOL::HighlightItem, PCB_ACTIONS::highlightItem.MakeEvent() );
  1150. Go( &BOARD_INSPECTION_TOOL::HideNet, PCB_ACTIONS::hideNet.MakeEvent() );
  1151. Go( &BOARD_INSPECTION_TOOL::ShowNet, PCB_ACTIONS::showNet.MakeEvent() );
  1152. }