32#include <nlohmann/json.hpp>
39#include <wx/datetime.h>
117 _(
"Castellated:" ) ) );
119 _(
"Press-fit:" ) ) );
173 switch( footprint->GetSide() )
196 auto trackShapeA = track->GetEffectiveShape( layer );
200 if( layer != otherTrack->GetLayer() )
203 if( track->GetNetCode() == otherTrack->GetNetCode() )
210 auto trackShapeB = otherTrack->GetEffectiveShape( layer );
233 std::vector<DRILL_LINE_ITEM> drills;
264 for(
int j = 0; j < polySet.
HoleCount( i ); ++j )
269 for(
PAD*
pad : footprint->Pads() )
271 if( !
pad->HasHole() )
274 std::shared_ptr<SHAPE_SEGMENT> hole =
pad->GetEffectiveHoleShape();
279 const SEG& seg = hole->GetSeg();
280 double width = hole->GetWidth();
281 double area = seg.
Length() * width;
283 area +=
M_PI * 0.25 * width * width;
293 double drill =
via->GetDrillValue();
314 int minPadClearanceOuter = netSettings->GetDefaultNetclass()->GetClearance();
322 frontShapesForArea.
Append( frontA );
325 backShapesForArea.
Append( backA );
329 for(
PAD*
pad : fp->Pads() )
331 if( !
pad->HasHole() )
336 pad->TransformShapeToPolygon( frontShapesForArea,
F_Cu,
337 std::min( minPadClearanceOuter,
pad->GetOwnClearance(
F_Cu ) ),
339 pad->TransformShapeToPolygon( backShapesForArea,
B_Cu,
340 std::min( minPadClearanceOuter,
pad->GetOwnClearance(
B_Cu ) ),
426 return wxString::Format( wxT(
"%i" ), aCount );
430static void appendTable(
const std::vector<std::vector<wxString>>& aRows,
bool aUseFirstColAsLabel, wxString& aOut )
435 size_t columnCount = 0;
437 for(
const std::vector<wxString>& row : aRows )
439 if( row.size() > columnCount )
440 columnCount = row.size();
443 if( columnCount == 0 )
446 std::vector<int> widths( columnCount, 0 );
448 for(
const std::vector<wxString>& row : aRows )
450 for(
size_t col = 0; col < columnCount; ++col )
452 if( col >= row.size() )
455 int cellWidth =
static_cast<int>( row[col].length() );
457 if( cellWidth > widths[col] )
458 widths[col] = cellWidth;
463 [&](
const std::vector<wxString>& row,
bool treatFirstAsLabel )
465 if( treatFirstAsLabel && aUseFirstColAsLabel )
468 wxString firstColumn;
471 firstColumn = row[0];
473 formatted.Printf( wxS(
"|%-*s |" ), widths[0], firstColumn );
476 for(
size_t col = 1; col < columnCount; ++col )
479 if( col < row.size() )
482 formatted.Printf( wxS(
" %*s |" ), widths[col], value );
490 for(
size_t col = 0; col < columnCount; ++col )
493 if( col < row.size() )
497 formatted.Printf( wxS(
" %*s |" ), widths[col], value );
505 appendDataRow( aRows.front(),
false );
509 for(
size_t col = 0; col < columnCount; ++col )
511 int dashCount = widths[col] + 2;
518 for(
int i = 0; i < dashCount; ++i )
519 dashes << wxS(
"-" );
521 aOut << dashes << wxS(
"|" );
526 for(
size_t rowIdx = 1; rowIdx < aRows.size(); ++rowIdx )
527 appendDataRow( aRows[rowIdx],
true );
532 const UNITS_PROVIDER& aUnitsProvider,
const wxString& aProjectName,
533 const wxString& aBoardName )
537 report <<
_(
"PCB statistics report\n=====================" ) << wxS(
"\n" );
538 report << wxS(
"- " ) <<
_(
"Date" ) << wxS(
": " ) << wxDateTime::Now().Format() << wxS(
"\n" );
539 report << wxS(
"- " ) <<
_(
"Project" ) << wxS(
": " ) << aProjectName << wxS(
"\n" );
540 report << wxS(
"- " ) <<
_(
"Board name" ) << wxS(
": " ) << aBoardName << wxS(
"\n\n" );
542 report <<
_(
"Board" ) << wxS(
"\n-----\n" );
548 report << wxS(
"- " ) <<
_(
"Height" ) << wxS(
": " )
550 report << wxS(
"- " ) <<
_(
"Area" ) << wxS(
": " )
555 report << wxS(
"- " ) <<
_(
"Dimensions" ) << wxS(
": " ) <<
_(
"unknown" ) << wxS(
"\n" );
556 report << wxS(
"- " ) <<
_(
"Area" ) << wxS(
": " ) <<
_(
"unknown" ) << wxS(
"\n" );
559 report << wxS(
"- " ) <<
_(
"Front copper area" ) << wxS(
": " )
561 report << wxS(
"- " ) <<
_(
"Back copper area" ) << wxS(
": " )
564 report << wxS(
"- " ) <<
_(
"Min track clearance" ) << wxS(
": " )
568 report << wxS(
"- " ) <<
_(
"Min track width" ) << wxS(
": " )
573 report << wxS(
"- " ) <<
_(
"Min drill diameter" ) << wxS(
": " )
576 report << wxS(
"- " ) <<
_(
"Board stackup thickness" ) << wxS(
": " )
580 report << wxS(
"- " ) <<
_(
"Front footprint area" ) << wxS(
": " )
582 report << wxS(
"- " ) <<
_(
"Back footprint area" ) << wxS(
": " )
585 report << wxS(
"- " ) <<
_(
"Front component density" ) << wxS(
": " );
590 report <<
_(
"unknown" );
592 report << wxS(
"\n" );
594 report << wxS(
"- " ) <<
_(
"Back component density" ) << wxS(
": " );
599 report <<
_(
"unknown" );
601 report << wxS(
"\n" );
603 report <<
_(
"Pads" ) << wxS(
"\n----\n" );
606 report << wxS(
"- " ) << padEntry.
title << wxS(
" " ) << padEntry.
quantity << wxS(
"\n" );
609 report << wxS(
"- " ) << propEntry.
title << wxS(
" " ) << propEntry.
quantity << wxS(
"\n" );
611 report << wxS(
"\n" );
612 report <<
_(
"Vias" ) << wxS(
"\n----\n" );
615 report << wxS(
"- " ) << viaEntry.
title << wxS(
" " ) << viaEntry.
quantity << wxS(
"\n" );
617 report << wxS(
"\n" );
618 report <<
_(
"Components" ) << wxS(
"\n----------\n\n" );
620 std::vector<std::vector<wxString>> componentRows;
621 std::vector<wxString> header;
622 header.push_back( wxString() );
623 header.push_back(
_(
"Front Side" ) );
624 header.push_back(
_(
"Back Side" ) );
625 header.push_back(
_(
"Total" ) );
626 componentRows.push_back( std::move( header ) );
633 std::vector<wxString> row;
634 row.push_back( fpEntry.
title );
638 componentRows.push_back( std::move( row ) );
644 std::vector<wxString> totalRow;
645 totalRow.push_back(
_(
"Total:" ) );
648 totalRow.push_back(
formatCount( frontTotal + backTotal ) );
649 componentRows.push_back( std::move( totalRow ) );
653 report << wxS(
"\n" );
654 report <<
_(
"Drill holes" ) << wxS(
"\n-----------\n\n" );
656 std::vector<std::vector<wxString>> drillRows;
657 std::vector<wxString> drillHeader;
658 drillHeader.push_back(
_(
"Count" ) );
659 drillHeader.push_back(
_(
"Shape" ) );
660 drillHeader.push_back(
_(
"X Size" ) );
661 drillHeader.push_back(
_(
"Y Size" ) );
662 drillHeader.push_back(
_(
"Plated" ) );
663 drillHeader.push_back(
_(
"Via/Pad" ) );
664 drillHeader.push_back(
_(
"Start Layer" ) );
665 drillHeader.push_back(
_(
"Stop Layer" ) );
666 drillRows.push_back( std::move( drillHeader ) );
672 switch( drill.
shape )
676 default: shapeStr =
_(
"???" );
break;
679 wxString platedStr = drill.
isPlated ?
_(
"PTH" ) :
_(
"NPTH" );
680 wxString itemStr = drill.
isPad ?
_(
"Pad" ) :
_(
"Via" );
682 wxString startLayerStr;
683 wxString stopLayerStr;
686 startLayerStr =
_(
"N/A" );
690 startLayerStr =
_(
"N/A" );
693 stopLayerStr =
_(
"N/A" );
697 stopLayerStr =
_(
"N/A" );
699 std::vector<wxString> row;
701 row.push_back( shapeStr );
704 row.push_back( platedStr );
705 row.push_back( itemStr );
706 row.push_back( startLayerStr );
707 row.push_back( stopLayerStr );
708 drillRows.push_back( std::move( row ) );
718 const UNITS_PROVIDER& aUnitsProvider,
const wxString& aProjectName,
719 const wxString& aBoardName )
721 nlohmann::ordered_json root;
723 nlohmann::ordered_json metadata;
726 metadata[
"project"] = aProjectName;
727 metadata[
"board_name"] = aBoardName;
728 root[
"metadata"] = metadata;
730 nlohmann::ordered_json board;
743 board[
"width"] = nlohmann::json();
744 board[
"height"] = nlohmann::json();
745 board[
"area"] = nlohmann::json();
746 board[
"front_component_density"] = nlohmann::json();
747 board[
"back_component_density"] = nlohmann::json();
770 board[
"front_footprint_density"] = nlohmann::json();
771 board[
"back_footprint_density"] = nlohmann::json();
774 root[
"board"] = board;
779 [](
const wxString& title,
bool removeSuffix ) -> wxString
781 wxString
json = title;
786 if(
json.EndsWith( wxS(
":" ) ) )
789 json.Replace( wxS(
" " ), wxS(
"_" ) );
790 json.Replace( wxS(
"-" ), wxS(
"_" ) );
792 return json.MakeLower();
795 nlohmann::ordered_json pads = nlohmann::ordered_json::object();
801 pads[jsonize( propEntry.
title,
false )] = propEntry.
quantity;
805 nlohmann::ordered_json vias = nlohmann::ordered_json::object();
814 nlohmann::ordered_json components = nlohmann::ordered_json::object();
818 nlohmann::ordered_json component;
822 components[jsonize( fpEntry.
title,
false )] = component;
828 nlohmann::ordered_json totals;
829 totals[
"front"] = frontTotal;
830 totals[
"back"] = backTotal;
831 totals[
"total"] = frontTotal + backTotal;
832 components[
"total"] = totals;
834 root[
"components"] = components;
836 nlohmann::ordered_json drillHoles = nlohmann::ordered_json::array();
840 nlohmann::ordered_json drillJson;
844 switch( drill.
shape )
848 default: shapeStr =
_(
"???" );
break;
851 wxString sourceStr = drill.
isPad ?
_(
"Pad" ) :
_(
"Via" );
852 wxString startLayerStr =
_(
"N/A" );
853 wxString stopLayerStr =
_(
"N/A" );
861 drillJson[
"count"] = drill.
m_Qty;
862 drillJson[
"shape"] = shapeStr;
865 drillJson[
"plated"] = drill.
isPlated;
866 drillJson[
"source"] = sourceStr;
867 drillJson[
"start_layer"] = startLayerStr;
868 drillJson[
"stop_layer"] = stopLayerStr;
869 drillHoles.push_back( std::move( drillJson ) );
872 root[
"drill_holes"] = drillHoles;
874 std::string jsonText = root.dump( 2 );
875 return wxString::FromUTF8( jsonText.c_str() );
constexpr int ARC_LOW_DEF
void CollectDrillLineItems(BOARD *board, std::vector< DRILL_LINE_ITEM > &out)
static void updatePadCounts(FOOTPRINT *aFootprint, BOARD_STATISTICS_DATA &aData)
wxString FormatBoardStatisticsJson(const BOARD_STATISTICS_DATA &aData, BOARD *aBoard, const UNITS_PROVIDER &aUnitsProvider, const wxString &aProjectName, const wxString &aBoardName)
void ComputeBoardStatistics(BOARD *aBoard, const BOARD_STATISTICS_OPTIONS &aOptions, BOARD_STATISTICS_DATA &aData)
static wxString formatCount(int aCount)
wxString FormatBoardStatisticsReport(const BOARD_STATISTICS_DATA &aData, BOARD *aBoard, const UNITS_PROVIDER &aUnitsProvider, const wxString &aProjectName, const wxString &aBoardName)
void InitializeBoardStatisticsData(BOARD_STATISTICS_DATA &aData)
static void appendTable(const std::vector< std::vector< wxString > > &aRows, bool aUseFirstColAsLabel, wxString &aOut)
wxString GetBuildVersion()
Get the full KiCad version string.
std::shared_ptr< NET_SETTINGS > m_NetSettings
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
virtual bool IsOnLayer(PCB_LAYER_ID aLayer) const
Test to see if this object is on the given layer.
virtual void TransformShapeToPolySet(SHAPE_POLY_SET &aBuffer, PCB_LAYER_ID aLayer, int aClearance, int aError, ERROR_LOC aErrorLoc, KIGFX::RENDER_SETTINGS *aRenderSettings=nullptr) const
Convert the item shape to a polyset.
int BuildBoardThicknessFromStackup() const
Information pertinent to a Pcbnew printed circuit board.
BOARD_STACKUP GetStackupOrDefault() const
const FOOTPRINTS & Footprints() const
const TRACKS & Tracks() const
bool GetBoardPolygonOutlines(SHAPE_POLY_SET &aOutlines, bool aInferOutlineIfNecessary, OUTLINE_ERROR_HANDLER *aErrorHandler=nullptr, bool aAllowUseArcsInPolygons=false, bool aIncludeNPTHAsOutlines=false)
Extract the board outlines and build a closed polygon from lines, arcs and circle items on edge cut l...
const wxString GetLayerName(PCB_LAYER_ID aLayer) const
Return the name of a aLayer.
void RunOnChildren(const std::function< void(BOARD_ITEM *)> &aFunction, RECURSE_MODE aMode) const override
Invoke a function on all children.
BOARD_DESIGN_SETTINGS & GetDesignSettings() const
constexpr size_type GetWidth() const
constexpr size_type GetHeight() const
KICAD_T Type() const
Returns the type of object.
int Length() const
Return the length (this).
Represent a polyline containing arcs as well as line segments: A chain of connected line and/or arc s...
double Area(bool aAbsolute=true) const
Return the area of this chain.
Represent a set of closed polygons.
double Area()
Return the area of this poly set.
int HoleCount(int aOutline) const
Returns the number of holes in a given outline.
int Append(int x, int y, int aOutline=-1, int aHole=-1, bool aAllowDuplication=false)
Appends a vertex at the end of the given outline/hole (default: the last outline)
void Simplify()
Simplify the polyset (merges overlapping polys, eliminates degeneracy/self-intersections)
SHAPE_LINE_CHAIN & Outline(int aIndex)
Return the reference to aIndex-th outline in the set.
SHAPE_LINE_CHAIN & Hole(int aOutline, int aHole)
Return the reference to aHole-th hole in the aIndex-th outline.
int OutlineCount() const
Return the number of outlines in the set.
void BooleanSubtract(const SHAPE_POLY_SET &b)
Perform boolean polyset difference.
const BOX2I BBox(int aClearance=0) const override
Compute a bounding box of the shape, with a margin of aClearance a collision.
wxString MessageTextFromValue(double aValue, bool aAddUnitLabel=true, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE) const
A lower-precision version of StringFromValue().
void TransformCircleToPolygon(SHAPE_LINE_CHAIN &aBuffer, const VECTOR2I &aCenter, int aRadius, int aError, ERROR_LOC aErrorLoc, int aMinSegCount=0)
Convert a circle to a polygon, using multiple straight lines.
Some functions to handle hotkeys in KiCad.
PCB_LAYER_ID
A quick note on layer IDs:
@ NPTH
like PAD_PTH, but not plated mechanical use only, no connection allowed
@ SMD
Smd pad, appears on the solder paste layer (default)
@ PTH
Plated through hole pad.
@ CONN
Like smd, does not appear on the solder paste layer (default) Note: also has a special attribute in G...
@ PRESSFIT
a PTH with a hole diameter with tight tolerances for press fit pin
@ CASTELLATED
a pad with a castellated through hole
bool collide(T aObject, U aAnotherObject, int aLayer, int aMinDistance)
Used by SHAPE_INDEX to implement Query().
wxString GetISO8601CurrentDateTime()
int minClearanceTrackToTrack
std::vector< BOARD_STATISTICS_INFO_ENTRY< PAD_ATTRIB > > padEntries
double backFootprintCourtyardArea
double frontFootprintDensity
std::vector< BOARD_STATISTICS_FP_ENTRY > footprintEntries
std::vector< BOARD_STATISTICS_INFO_ENTRY< VIATYPE > > viaEntries
double backFootprintDensity
double frontFootprintCourtyardArea
std::vector< BOARD_STATISTICS_INFO_ENTRY< PAD_PROP > > padPropertyEntries
std::vector< DRILL_LINE_ITEM > drillEntries
bool subtractHolesFromCopperAreas
bool subtractHolesFromBoardArea
bool excludeFootprintsWithoutPads
@ PCB_GENERATOR_T
class PCB_GENERATOR, generator on a layer
@ PCB_VIA_T
class PCB_VIA, a via (like a track segment on a copper layer)
@ PCB_GROUP_T
class PCB_GROUP, a set of BOARD_ITEMs
@ PCB_FOOTPRINT_T
class FOOTPRINT, a footprint
@ PCB_PAD_T
class PAD, a pad in a footprint
@ PCB_ARC_T
class PCB_ARC, an arc track segment on a copper layer
@ PCB_TRACE_T
class PCB_TRACK, a track segment (segment on a copper layer)
VECTOR2< int32_t > VECTOR2I