55 VECTOR2I& aMidpoint,
int& aDistance )
60 std::vector<PCB_SHAPE*> shapes;
62 for(
int ii = 0; ii < items.
GetCount(); ++ii )
67 shapes.push_back( shape );
70 int best = std::numeric_limits<int>::max();
74 for(
size_t ii = 0; ii < shapes.size(); ++ii )
76 std::shared_ptr<SHAPE> shapeA = shapes[ii]->GetEffectiveShape();
78 for(
size_t jj = ii + 1; jj < shapes.size(); ++jj )
80 std::shared_ptr<SHAPE> shapeB = shapes[jj]->GetEffectiveShape();
84 if( shapeA && shapeB && shapeA->NearestPoints( shapeB.get(), ptA, ptB ) )
86 int dist = ( ptA - ptB ).EuclideanNorm();
100 if( aItemA && aItemB )
103 aMidpoint = ( bestA + bestB ) / 2;
123 virtual bool Run()
override;
125 virtual const wxString
GetName()
const override {
return wxT(
"miscellaneous" ); };
141 bool errorHandled =
false;
152 std::swap( itemA, itemB );
158 bool usedGap =
false;
162 shapeA =
static_cast<PCB_SHAPE*
>( itemA );
163 shapeB =
static_cast<PCB_SHAPE*
>( itemB );
165 else if( itemA && !itemB && itemA->Type() ==
PCB_SHAPE_T )
169 shapeA =
static_cast<PCB_SHAPE*
>( itemA );
177 usedGap = shapeA && shapeB;
180 if( shapeA && shapeB )
185 if( effectiveShapeA && effectiveShapeB )
187 BOX2I bboxA = effectiveShapeA->BBox();
188 BOX2I bboxB = effectiveShapeB->BBox();
193 markerPos = overlap.
Centre();
200 if( effectiveShapeA->NearestPoints( effectiveShapeB.get(), ptA, ptB ) )
202 gap = ( ptA - ptB ).EuclideanNorm();
203 markerPos = ( ptA + ptB ) / 2;
212 if( itemA && itemB && usedGap )
218 drcItem->SetErrorDetail( msg );
221 drcItem->SetItems( itemA, itemB );
228 const int minSizeForValideGraphics =
pcbIUScale.mmToIU( 0.001 );
239 drcItem->SetErrorDetail(
_(
"(Suspicious items found on Edge.Cuts layer)" ) );
249 int chainingEpsilon =
m_board->GetOutlinesChainingEpsilon();
263 drcItem->SetErrorDetail(
_(
"(no edges found on Edge.Cuts layer)" ) );
274 const int progressDelta = 2000;
290 auto checkDisabledLayers =
308 if( disabledLayers.test(
pad->GetPrincipalLayer() ) )
309 badLayer = item->GetLayer();
322 via->LayerPair( &
top, &bottom );
324 if( disabledLayers.test(
top ) )
326 else if( disabledLayers.test( bottom ) )
336 LSET badLayers = disabledLayers & item->GetLayerSet();
338 if( badLayers.any() )
339 badLayer = badLayers.
Seq().front();
345 drcItem->SetErrorDetail( wxString::Format(
_(
"(layer %s)" ),
LayerName( badLayer ) ) );
346 drcItem->SetItems( item );
364 std::vector<BOARD_ITEM*> allItems;
369 allItems.push_back( item );
373 std::atomic<size_t> itemsDone( 0 );
374 size_t itemCount = allItems.size();
388 drcItem->SetErrorDetail( wxString::Format( wxS(
"(%s)" ), c->
GetName() ) );
389 drcItem->SetItems( item );
396 itemsDone.fetch_add( 1 );
400 auto itemFutures =
tp.submit_loop( 0, itemCount, checkItem, itemCount );
402 while( itemsDone < itemCount )
409 itemFutures.wait_for( std::chrono::milliseconds( 250 ) );
420 const int progressDelta = 2000;
424 static const std::vector<KICAD_T> itemTypes = {
433 static wxRegEx warningExpr( wxS(
"^\\$\\{DRC_WARNING\\s*([^}]*)\\}(.*)$" ) );
434 static wxRegEx errorExpr( wxS(
"^\\$\\{DRC_ERROR\\s*([^}]*)\\}(.*)$" ) );
436 if( warningExpr.Matches(
text ) )
441 wxString drcText = warningExpr.GetMatch(
text, 1 );
444 drcItem->SetItems( item );
446 drcText +=
_(
" (in drawing sheet)" );
448 drcItem->SetErrorMessage( drcText );
456 if( errorExpr.Matches(
text ) )
461 wxString drcText = errorExpr.GetMatch(
text, 1 );
464 drcItem->SetItems( item );
466 drcText +=
_(
" (in drawing sheet)" );
468 drcItem->SetErrorMessage( drcText );
500 if(
result.Matches( wxT(
"*${*}*" ) ) )
503 drcItem->SetItems( item );
526 drawItems.
BuildDrawItemsList( drawingSheet->GetPageInfo(), drawingSheet->GetTitleBlock() );
542 else if(
text->GetShownText(
true ).Matches( wxT(
"*${*}*" ) ) )
545 drcItem->SetItems( drawingSheet );
559 std::shared_ptr<NET_SETTINGS> netSettings =
m_board->GetProject()->GetProjectFile().NetSettings();
560 const std::shared_ptr<TUNING_PROFILES> tuningProfiles =
561 m_board->GetProject()->GetProjectFile().TuningProfileParameters();
563 std::set<wxString> profileNames;
564 std::ranges::for_each( tuningProfiles->GetTuningProfiles(),
567 if( const wxString name = tuningProfile.m_ProfileName; name != wxEmptyString )
568 profileNames.insert( name );
571 for(
const auto& [
name, netclass] : netSettings->GetNetclasses() )
576 const wxString profileName = netclass->GetTuningProfile();
578 if( netclass->HasTuningProfile() && !profileNames.contains( profileName ) )
581 drcItem->SetErrorDetail( wxString::Format(
"(Net Class: %s, Tuning Profile: %s)",
605 if( !
reportPhase(
_(
"Checking disabled layers..." ) ) )
631 if( !
reportPhase(
_(
"Checking for missing tuning profiles..." ) ) )
constexpr EDA_IU_SCALE pcbIUScale
BASE_SET & flip(size_t pos)
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
virtual PCB_LAYER_ID GetLayer() const
Return the primary layer this item is on.
Information pertinent to a Pcbnew printed circuit board.
constexpr BOX2< Vec > Intersect(const BOX2< Vec > &aRect)
constexpr size_type GetWidth() const
constexpr Vec Centre() const
constexpr size_type GetHeight() const
int GetCount() const
Return the number of objects in the list.
DRC_RULE * GetParentRule() const
static std::shared_ptr< DRC_ITEM > Create(int aErrorCode)
Constructs a DRC_ITEM for the given error code.
virtual const wxString GetName() const override
void testMissingTuningProfiles()
virtual ~DRC_TEST_PROVIDER_MISC()=default
void testDisabledLayers()
virtual bool Run() override
Run this provider against the given PCB with configured options (if any).
virtual bool reportPhase(const wxString &aStageName)
int forEachGeometryItem(const std::vector< KICAD_T > &aTypes, const LSET &aLayers, const std::function< bool(BOARD_ITEM *)> &aFunc)
void reportViolation(std::shared_ptr< DRC_ITEM > &item, const VECTOR2I &aMarkerPos, int aMarkerLayer, const std::function< void(PCB_MARKER *)> &aPathGenerator=[](PCB_MARKER *){})
static std::vector< KICAD_T > s_allBasicItems
virtual bool reportProgress(size_t aCount, size_t aSize, size_t aDelta=1)
Base class to handle basic graphic items.
Store the list of graphic items: rect, lines, polygons and texts to draw/plot the title block and fra...
DS_DRAW_ITEM_BASE * GetFirst()
void BuildDrawItemsList(const PAGE_INFO &aPageInfo, const TITLE_BLOCK &aTitleBlock)
Drawing or plot the drawing sheet.
void SetFileName(const wxString &aFileName)
Set the filename to draw/plot.
void SetSheetName(const wxString &aSheetName)
Set the sheet name to draw/plot.
void SetSheetLayer(const wxString &aSheetLayer)
Set the sheet layer to draw/plot.
void SetSheetCount(int aSheetCount)
Set the value of the count of sheets, for basic inscriptions.
void SetPageNumber(const wxString &aPageNumber)
Set the value of the sheet number.
DS_DRAW_ITEM_BASE * GetNext()
void SetProject(const PROJECT *aProject)
virtual VECTOR2I GetPosition() const
A mix-in class (via multiple inheritance) that handles texts such as labels, parts,...
LSET is a set of PCB_LAYER_IDs.
static const LSET & AllCuMask()
return AllCuMask( MAX_CU_LAYERS );
LSEQ Seq(const LSEQ &aSequence) const
Return an LSEQ from the union of this LSET and a desired sequence.
static const LSET & AllLayersMask()
std::shared_ptr< SHAPE > GetEffectiveShape(PCB_LAYER_ID aLayer=UNDEFINED_LAYER, FLASHING aFlash=FLASHING::DEFAULT) const override
Make a set of SHAPE objects representing the PCB_SHAPE.
PCB_LAYER_ID GetLayer() const override
Return the primary layer this item is on.
Collect all BOARD_ITEM objects of a given set of KICAD_T type(s).
void Collect(BOARD_ITEM *aBoard, const std::vector< KICAD_T > &aTypes)
Collect BOARD_ITEM objects using this class's Inspector method, which does the collection.
Represent a set of closed polygons.
wxString MessageTextFromValue(double aValue, bool aAddUnitLabel=true, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE) const
A lower-precision version of StringFromValue().
const wxString ExpandEnvVarSubstitutions(const wxString &aString, const PROJECT *aProject)
Replace any environment variable & text variable references with their values.
#define FOR_ERC_DRC
Expand '${var-name}' templates in text.
bool BuildBoardPolygonOutlines(BOARD *aBoard, SHAPE_POLY_SET &aOutlines, int aErrorMax, int aChainingEpsilon, bool aInferOutlineIfNecessary, OUTLINE_ERROR_HANDLER *aErrorHandler, bool aAllowUseArcsInPolygons)
Extract the board outlines and build a closed polygon from lines, arcs and circle items on edge cut l...
bool TestBoardOutlinesGraphicItems(BOARD *aBoard, int aMinDist, OUTLINE_ERROR_HANDLER *aErrorHandler)
Test a board graphic items on edge cut layer for validity.
const std::function< void(const wxString &msg, BOARD_ITEM *itemA, BOARD_ITEM *itemB, const VECTOR2I &pt)> OUTLINE_ERROR_HANDLER
@ DRCE_DISABLED_LAYER_ITEM
@ DRCE_MISSING_TUNING_PROFILE
@ DRCE_UNRESOLVED_VARIABLE
static void findClosestOutlineGap(BOARD *aBoard, PCB_SHAPE *&aItemA, PCB_SHAPE *&aItemB, VECTOR2I &aMidpoint, int &aDistance)
wxString LayerName(int aLayer)
Returns the default display name for a given layer.
@ LAYER_DRAWINGSHEET
Sheet frame and title block.
PCB_LAYER_ID
A quick note on layer IDs:
static DRC_REGISTER_TEST_PROVIDER< DRC_TEST_PROVIDER_ANNULAR_WIDTH > dummy
@ SMD
Smd pad, appears on the solder paste layer (default)
@ CONN
Like smd, does not appear on the solder paste layer (default) Note: also has a special attribute in G...
Represents a single line in the tuning profile configuration grid.
KIBIS top(path, &reporter)
wxString result
Test unit parsing edge cases and error handling.
thread_pool & GetKiCadThreadPool()
Get a reference to the current thread pool.
BS::priority_thread_pool thread_pool
@ PCB_SHAPE_T
class PCB_SHAPE, a segment not on copper layers
@ PCB_VIA_T
class PCB_VIA, a via (like a track segment on a copper layer)
@ PCB_TEXTBOX_T
class PCB_TEXTBOX, wrapped text on a layer
@ PCB_ZONE_T
class ZONE, a copper pour area
@ PCB_TEXT_T
class PCB_TEXT, text on a layer
@ PCB_FIELD_T
class PCB_FIELD, text associated with a footprint property
@ PCB_TABLECELL_T
class PCB_TABLECELL, PCB_TEXTBOX for use in tables
@ PCB_PAD_T
class PAD, a pad in a footprint
@ PCB_DIMENSION_T
class PCB_DIMENSION_BASE: abstract dimension meta-type
VECTOR2< int32_t > VECTOR2I