40#include <wx/datetime.h>
118 _(
"Castellated:" ) ) );
120 _(
"Press-fit:" ) ) );
174 switch( footprint->GetSide() )
197 auto trackShapeA = track->GetEffectiveShape( layer );
201 if( layer != otherTrack->GetLayer() )
204 if( track->GetNetCode() == otherTrack->GetNetCode() )
211 auto trackShapeB = otherTrack->GetEffectiveShape( layer );
234 std::vector<DRILL_LINE_ITEM> drills;
265 for(
int j = 0; j < polySet.
HoleCount( i ); ++j )
270 for(
PAD*
pad : footprint->Pads() )
272 if( !
pad->HasHole() )
275 std::shared_ptr<SHAPE_SEGMENT> hole =
pad->GetEffectiveHoleShape();
280 const SEG& seg = hole->GetSeg();
281 double width = hole->GetWidth();
282 double area = seg.
Length() * width;
284 area +=
M_PI * 0.25 * width * width;
294 double drill =
via->GetDrillValue();
315 int minPadClearanceOuter = netSettings->GetDefaultNetclass()->GetClearance();
323 frontShapesForArea.
Append( frontA );
326 backShapesForArea.
Append( backA );
330 for(
PAD*
pad : fp->Pads() )
332 if( !
pad->HasHole() )
337 pad->TransformShapeToPolygon( frontShapesForArea,
F_Cu,
338 std::min( minPadClearanceOuter,
pad->GetOwnClearance(
F_Cu ) ),
340 pad->TransformShapeToPolygon( backShapesForArea,
B_Cu,
341 std::min( minPadClearanceOuter,
pad->GetOwnClearance(
B_Cu ) ),
427 return wxString::Format( wxT(
"%i" ), aCount );
431static void appendTable(
const std::vector<std::vector<wxString>>& aRows,
bool aUseFirstColAsLabel, wxString& aOut )
436 size_t columnCount = 0;
438 for(
const std::vector<wxString>& row : aRows )
440 if( row.size() > columnCount )
441 columnCount = row.size();
444 if( columnCount == 0 )
447 std::vector<int> widths( columnCount, 0 );
449 for(
const std::vector<wxString>& row : aRows )
451 for(
size_t col = 0; col < columnCount; ++col )
453 if( col >= row.size() )
456 int cellWidth =
static_cast<int>( row[col].length() );
458 if( cellWidth > widths[col] )
459 widths[col] = cellWidth;
464 [&](
const std::vector<wxString>& row,
bool treatFirstAsLabel )
466 if( treatFirstAsLabel && aUseFirstColAsLabel )
469 wxString firstColumn;
472 firstColumn = row[0];
474 formatted.Printf( wxS(
"|%-*s |" ), widths[0], firstColumn );
477 for(
size_t col = 1; col < columnCount; ++col )
480 if( col < row.size() )
483 formatted.Printf( wxS(
" %*s |" ), widths[col], value );
491 for(
size_t col = 0; col < columnCount; ++col )
494 if( col < row.size() )
498 formatted.Printf( wxS(
" %*s |" ), widths[col], value );
506 appendDataRow( aRows.front(),
false );
510 for(
size_t col = 0; col < columnCount; ++col )
512 int dashCount = widths[col] + 2;
519 for(
int i = 0; i < dashCount; ++i )
520 dashes << wxS(
"-" );
522 aOut << dashes << wxS(
"|" );
527 for(
size_t rowIdx = 1; rowIdx < aRows.size(); ++rowIdx )
528 appendDataRow( aRows[rowIdx],
true );
533 const UNITS_PROVIDER& aUnitsProvider,
const wxString& aProjectName,
534 const wxString& aBoardName )
538 report <<
_(
"PCB statistics report\n=====================" ) << wxS(
"\n" );
539 report << wxS(
"- " ) <<
_(
"Date" ) << wxS(
": " ) << wxDateTime::Now().Format() << wxS(
"\n" );
540 report << wxS(
"- " ) <<
_(
"Project" ) << wxS(
": " ) << aProjectName << wxS(
"\n" );
541 report << wxS(
"- " ) <<
_(
"Board name" ) << wxS(
": " ) << aBoardName << wxS(
"\n\n" );
543 report <<
_(
"Board" ) << wxS(
"\n-----\n" );
549 report << wxS(
"- " ) <<
_(
"Height" ) << wxS(
": " )
551 report << wxS(
"- " ) <<
_(
"Area" ) << wxS(
": " )
556 report << wxS(
"- " ) <<
_(
"Dimensions" ) << wxS(
": " ) <<
_(
"unknown" ) << wxS(
"\n" );
557 report << wxS(
"- " ) <<
_(
"Area" ) << wxS(
": " ) <<
_(
"unknown" ) << wxS(
"\n" );
560 report << wxS(
"- " ) <<
_(
"Front copper area" ) << wxS(
": " )
562 report << wxS(
"- " ) <<
_(
"Back copper area" ) << wxS(
": " )
565 report << wxS(
"- " ) <<
_(
"Min track clearance" ) << wxS(
": " )
569 report << wxS(
"- " ) <<
_(
"Min track width" ) << wxS(
": " )
574 report << wxS(
"- " ) <<
_(
"Min drill diameter" ) << wxS(
": " )
577 report << wxS(
"- " ) <<
_(
"Board stackup thickness" ) << wxS(
": " )
581 report << wxS(
"- " ) <<
_(
"Front footprint area" ) << wxS(
": " )
583 report << wxS(
"- " ) <<
_(
"Back footprint area" ) << wxS(
": " )
586 report << wxS(
"- " ) <<
_(
"Front component density" ) << wxS(
": " );
591 report <<
_(
"unknown" );
593 report << wxS(
"\n" );
595 report << wxS(
"- " ) <<
_(
"Back component density" ) << wxS(
": " );
600 report <<
_(
"unknown" );
602 report << wxS(
"\n" );
604 report <<
_(
"Pads" ) << wxS(
"\n----\n" );
607 report << wxS(
"- " ) << padEntry.
title << wxS(
" " ) << padEntry.
quantity << wxS(
"\n" );
610 report << wxS(
"- " ) << propEntry.
title << wxS(
" " ) << propEntry.
quantity << wxS(
"\n" );
612 report << wxS(
"\n" );
613 report <<
_(
"Vias" ) << wxS(
"\n----\n" );
616 report << wxS(
"- " ) << viaEntry.
title << wxS(
" " ) << viaEntry.
quantity << wxS(
"\n" );
618 report << wxS(
"\n" );
619 report <<
_(
"Components" ) << wxS(
"\n----------\n\n" );
621 std::vector<std::vector<wxString>> componentRows;
622 std::vector<wxString> header;
623 header.push_back( wxString() );
624 header.push_back(
_(
"Front Side" ) );
625 header.push_back(
_(
"Back Side" ) );
626 header.push_back(
_(
"Total" ) );
627 componentRows.push_back( std::move( header ) );
634 std::vector<wxString> row;
635 row.push_back( fpEntry.
title );
639 componentRows.push_back( std::move( row ) );
645 std::vector<wxString> totalRow;
646 totalRow.push_back(
_(
"Total:" ) );
649 totalRow.push_back(
formatCount( frontTotal + backTotal ) );
650 componentRows.push_back( std::move( totalRow ) );
654 report << wxS(
"\n" );
655 report <<
_(
"Drill holes" ) << wxS(
"\n-----------\n\n" );
657 std::vector<std::vector<wxString>> drillRows;
658 std::vector<wxString> drillHeader;
659 drillHeader.push_back(
_(
"Count" ) );
660 drillHeader.push_back(
_(
"Shape" ) );
661 drillHeader.push_back(
_(
"X Size" ) );
662 drillHeader.push_back(
_(
"Y Size" ) );
663 drillHeader.push_back(
_(
"Plated" ) );
664 drillHeader.push_back(
_(
"Via/Pad" ) );
665 drillHeader.push_back(
_(
"Start Layer" ) );
666 drillHeader.push_back(
_(
"Stop Layer" ) );
667 drillRows.push_back( std::move( drillHeader ) );
673 switch( drill.
shape )
677 default: shapeStr =
_(
"???" );
break;
680 wxString platedStr = drill.
isPlated ?
_(
"PTH" ) :
_(
"NPTH" );
681 wxString itemStr = drill.
isPad ?
_(
"Pad" ) :
_(
"Via" );
683 wxString startLayerStr;
684 wxString stopLayerStr;
687 startLayerStr =
_(
"N/A" );
691 startLayerStr =
_(
"N/A" );
694 stopLayerStr =
_(
"N/A" );
698 stopLayerStr =
_(
"N/A" );
700 std::vector<wxString> row;
702 row.push_back( shapeStr );
705 row.push_back( platedStr );
706 row.push_back( itemStr );
707 row.push_back( startLayerStr );
708 row.push_back( stopLayerStr );
709 drillRows.push_back( std::move( row ) );
719 const UNITS_PROVIDER& aUnitsProvider,
const wxString& aProjectName,
720 const wxString& aBoardName )
722 nlohmann::ordered_json root;
724 nlohmann::ordered_json metadata;
727 metadata[
"project"] = aProjectName;
728 metadata[
"board_name"] = aBoardName;
729 root[
"metadata"] = metadata;
731 nlohmann::ordered_json board;
744 board[
"width"] = nlohmann::json();
745 board[
"height"] = nlohmann::json();
746 board[
"area"] = nlohmann::json();
747 board[
"front_component_density"] = nlohmann::json();
748 board[
"back_component_density"] = nlohmann::json();
771 board[
"front_footprint_density"] = nlohmann::json();
772 board[
"back_footprint_density"] = nlohmann::json();
775 root[
"board"] = board;
780 [](
const wxString& title,
bool removeSuffix ) -> wxString
782 wxString
json = title;
787 if(
json.EndsWith( wxS(
":" ) ) )
790 json.Replace( wxS(
" " ), wxS(
"_" ) );
791 json.Replace( wxS(
"-" ), wxS(
"_" ) );
793 return json.MakeLower();
796 nlohmann::ordered_json pads = nlohmann::ordered_json::object();
802 pads[jsonize( propEntry.
title,
false )] = propEntry.
quantity;
806 nlohmann::ordered_json vias = nlohmann::ordered_json::object();
815 nlohmann::ordered_json components = nlohmann::ordered_json::object();
819 nlohmann::ordered_json component;
823 components[jsonize( fpEntry.
title,
false )] = component;
829 nlohmann::ordered_json totals;
830 totals[
"front"] = frontTotal;
831 totals[
"back"] = backTotal;
832 totals[
"total"] = frontTotal + backTotal;
833 components[
"total"] = totals;
835 root[
"components"] = components;
837 nlohmann::ordered_json drillHoles = nlohmann::ordered_json::array();
841 nlohmann::ordered_json drillJson;
845 switch( drill.
shape )
849 default: shapeStr =
_(
"???" );
break;
852 wxString sourceStr = drill.
isPad ?
_(
"Pad" ) :
_(
"Via" );
853 wxString startLayerStr =
_(
"N/A" );
854 wxString stopLayerStr =
_(
"N/A" );
862 drillJson[
"count"] = drill.
m_Qty;
863 drillJson[
"shape"] = shapeStr;
866 drillJson[
"plated"] = drill.
isPlated;
867 drillJson[
"source"] = sourceStr;
868 drillJson[
"start_layer"] = startLayerStr;
869 drillJson[
"stop_layer"] = stopLayerStr;
870 drillHoles.push_back( std::move( drillJson ) );
873 root[
"drill_holes"] = drillHoles;
875 std::string jsonText = root.dump( 2 );
876 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