KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
zone.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright (C) 2017 Jean-Pierre Charras, jp.charras at wanadoo.fr
5 * Copyright (C) 2012 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
6 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version 2
11 * of the License, or (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, you may find one here:
20 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
21 * or you may search the http://www.gnu.org website for the version 2 license,
22 * or you may write to the Free Software Foundation, Inc.,
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
24 */
25
26#include <advanced_config.h>
27#include <bitmaps.h>
29#include <geometry/shape_null.h>
30#include <pcb_edit_frame.h>
31#include <pcb_screen.h>
32#include <board.h>
34#include <lset.h>
35#include <pad.h>
36#include <zone.h>
37#include <footprint.h>
38#include <string_utils.h>
42#include <trigo.h>
43#include <i18n_utility.h>
44#include <mutex>
45#include <magic_enum.hpp>
46
47#include <google/protobuf/any.pb.h>
48#include <api/api_enums.h>
49#include <api/api_utils.h>
50#include <api/api_pcb_utils.h>
51#include <api/board/board_types.pb.h>
52
53
56 m_Poly( nullptr ),
57 m_cornerRadius( 0 ),
58 m_priority( 0 ),
59 m_isRuleArea( false ),
60 m_ruleAreaPlacementEnabled( false ),
61 m_ruleAreaPlacementSourceType( RULE_AREA_PLACEMENT_SOURCE_TYPE::SHEETNAME ),
62 m_teardropType( TEARDROP_TYPE::TD_NONE ),
63 m_PadConnection( ZONE_CONNECTION::NONE ),
64 m_ZoneClearance( 0 ),
65 m_ZoneMinThickness( 0 ),
66 m_islandRemovalMode( ISLAND_REMOVAL_MODE::ALWAYS ),
67 m_isFilled( false ),
68 m_thermalReliefGap( 0 ),
69 m_thermalReliefSpokeWidth( 0 ),
70 m_fillMode( ZONE_FILL_MODE::POLYGONS ),
71 m_hatchThickness( 0 ),
72 m_hatchGap( 0 ),
73 m_hatchOrientation( ANGLE_0 ),
74 m_hatchSmoothingLevel( 0 ),
75 m_hatchHoleMinArea( 0 ),
76 m_CornerSelection( nullptr ),
77 m_area( 0.0 ),
78 m_outlinearea( 0.0 )
79{
80 m_Poly = new SHAPE_POLY_SET(); // Outlines
81 SetLocalFlags( 0 ); // flags temporary used in zone calculations
82 m_fillVersion = 5; // set the "old" way to build filled polygon areas (< 6.0.x)
83
84 if( GetParentFootprint() )
85 SetIsRuleArea( true ); // Zones living in footprints have the rule area option
86
87 if( aParent->GetBoard() )
89 false );
90 else
92
93 m_needRefill = false; // True only after edits.
94}
95
96
97ZONE::ZONE( const ZONE& aZone ) :
98 BOARD_CONNECTED_ITEM( aZone ),
99 m_Poly( nullptr ),
100 m_CornerSelection( nullptr )
101{
103}
104
105
106ZONE& ZONE::operator=( const ZONE& aOther )
107{
109
111
112 return *this;
113}
114
115
116void ZONE::CopyFrom( const BOARD_ITEM* aOther )
117{
118 wxCHECK( aOther && aOther->Type() == PCB_ZONE_T, /* void */ );
119 *this = *static_cast<const ZONE*>( aOther );
120}
121
122
124{
125 delete m_Poly;
126 delete m_CornerSelection;
127
128 if( BOARD* board = GetBoard() )
129 board->IncrementTimeStamp();
130}
131
132
134{
135 // members are expected non initialize in this.
136 // InitDataFromSrcInCopyCtor() is expected to be called only from a copy constructor.
137
138 // Copy only useful EDA_ITEM flags:
139 m_flags = aZone.m_flags;
141
142 // Replace the outlines for aZone outlines.
143 delete m_Poly;
144 m_Poly = new SHAPE_POLY_SET( *aZone.m_Poly );
145
148 m_zoneName = aZone.m_zoneName;
149 m_priority = aZone.m_priority;
154 SetLayerSet( aZone.GetLayerSet() );
155
161
163 m_ZoneClearance = aZone.m_ZoneClearance; // clearance value
168
169 m_isFilled = aZone.m_isFilled;
172
175
176 m_fillMode = aZone.m_fillMode; // solid vs. hatched
178 m_hatchGap = aZone.m_hatchGap;
184
185 // For corner moving, corner index to drag, or nullptr if no selection
186 delete m_CornerSelection;
187 m_CornerSelection = nullptr;
188
189 aZone.GetLayerSet().RunOnLayers(
190 [&]( PCB_LAYER_ID layer )
191 {
192 std::shared_ptr<SHAPE_POLY_SET> fill = aZone.m_FilledPolysList.at( layer );
193
194 if( fill )
195 m_FilledPolysList[layer] = std::make_shared<SHAPE_POLY_SET>( *fill );
196 else
197 m_FilledPolysList[layer] = std::make_shared<SHAPE_POLY_SET>();
198
199 m_filledPolysHash[layer] = aZone.m_filledPolysHash.at( layer );
200 m_insulatedIslands[layer] = aZone.m_insulatedIslands.at( layer );
201 } );
202
203 m_layerProperties.clear();
204
205 std::ranges::copy( aZone.LayerProperties(),
206 std::inserter( m_layerProperties, std::end( m_layerProperties ) ) );
207
211
212 SetLocalFlags( aZone.GetLocalFlags() );
213
214 m_netinfo = aZone.m_netinfo;
215 m_area = aZone.m_area;
217}
218
219
221{
222 return new ZONE( *this );
223}
224
225
226void ZONE::Serialize( google::protobuf::Any& aContainer ) const
227{
228 using namespace kiapi::board;
229 types::Zone zone;
231
232 zone.mutable_id()->set_value( m_Uuid.AsStdString() );
233 PackLayerSet( *zone.mutable_layers(), GetLayerSet() );
234
235 if( m_isRuleArea )
236 zone.set_type( types::ZT_RULE_AREA );
237 else if( m_teardropType != TEARDROP_TYPE::TD_NONE )
238 zone.set_type( types::ZT_TEARDROP );
239 else if( IsOnCopperLayer() )
240 zone.set_type( types::ZT_COPPER );
241 else
242 zone.set_type( types::ZT_GRAPHICAL );
243
244 kiapi::common::PackPolySet( *zone.mutable_outline(), *m_Poly );
245
246 zone.set_name( m_zoneName.ToUTF8() );
247 zone.set_priority( m_priority );
248 zone.set_filled( m_isFilled );
249
250 if( m_isRuleArea )
251 {
252 types::RuleAreaSettings* ra = zone.mutable_rule_area_settings();
253 ra->set_keepout_copper( m_doNotAllowZoneFills );
254 ra->set_keepout_footprints( m_doNotAllowFootprints );
255 ra->set_keepout_pads( m_doNotAllowPads );
256 ra->set_keepout_tracks( m_doNotAllowTracks );
257 ra->set_keepout_vias( m_doNotAllowVias );
258
259 ra->set_placement_enabled( m_ruleAreaPlacementEnabled );
260 ra->set_placement_source( m_ruleAreaPlacementSource.ToUTF8() );
261 ra->set_placement_source_type(
262 ToProtoEnum<RULE_AREA_PLACEMENT_SOURCE_TYPE, types::PlacementRuleSourceType>(
264 }
265 else
266 {
267 types::CopperZoneSettings* cu = zone.mutable_copper_settings();
268 cu->mutable_connection()->set_zone_connection(
269 ToProtoEnum<ZONE_CONNECTION, types::ZoneConnectionStyle>( m_PadConnection ) );
270
271 types::ThermalSpokeSettings* thermals = cu->mutable_connection()->mutable_thermal_spokes();
272 thermals->mutable_width()->set_value_nm( m_thermalReliefSpokeWidth );
273 thermals->mutable_gap()->set_value_nm( m_thermalReliefGap );
274 // n.b. zones don't currently have an overall thermal angle override
275
276 cu->mutable_clearance()->set_value_nm( m_ZoneClearance );
277 cu->mutable_min_thickness()->set_value_nm( m_ZoneMinThickness );
278 cu->set_island_mode(
279 ToProtoEnum<ISLAND_REMOVAL_MODE, types::IslandRemovalMode>( m_islandRemovalMode ) );
280 cu->set_min_island_area( m_minIslandArea );
281 cu->set_fill_mode( ToProtoEnum<ZONE_FILL_MODE, types::ZoneFillMode>( m_fillMode ) );
282
283 types::HatchFillSettings* hatch = cu->mutable_hatch_settings();
284 hatch->mutable_thickness()->set_value_nm( m_hatchThickness );
285 hatch->mutable_gap()->set_value_nm( m_hatchGap );
286 hatch->mutable_orientation()->set_value_degrees( m_hatchOrientation.AsDegrees() );
287 hatch->set_hatch_smoothing_ratio( m_hatchSmoothingValue );
288 hatch->set_hatch_hole_min_area_ratio( m_hatchHoleMinArea );
289
290 switch( m_hatchBorderAlgorithm )
291 {
292 default:
293 case 0: hatch->set_border_mode( types::ZHFBM_USE_MIN_ZONE_THICKNESS ); break;
294 case 1: hatch->set_border_mode( types::ZHFBM_USE_HATCH_THICKNESS ); break;
295 }
296
297 cu->mutable_net()->mutable_code()->set_value( GetNetCode() );
298 cu->mutable_net()->set_name( GetNetname() );
299 cu->mutable_teardrop()->set_type(
300 ToProtoEnum<TEARDROP_TYPE, types::TeardropType>( m_teardropType ) );
301 }
302
303 for( const auto& [layer, shape] : m_FilledPolysList )
304 {
305 types::ZoneFilledPolygons* filledLayer = zone.add_filled_polygons();
306 filledLayer->set_layer( ToProtoEnum<PCB_LAYER_ID, types::BoardLayer>( layer ) );
307 kiapi::common::PackPolySet( *filledLayer->mutable_shapes(), *shape );
308 }
309
310 for( const auto& [layer, properties] : m_layerProperties )
311 {
312 types::ZoneLayerProperties* layerProperties = zone.add_layer_properties();
313 layerProperties->set_layer( ToProtoEnum<PCB_LAYER_ID, types::BoardLayer>( layer ) );
314
315 if( properties.hatching_offset.has_value() )
316 {
317 PackVector2( *layerProperties->mutable_hatching_offset(),
318 properties.hatching_offset.value() );
319 }
320 }
321
322 zone.mutable_border()->set_style(
323 ToProtoEnum<ZONE_BORDER_DISPLAY_STYLE, types::ZoneBorderStyle>( m_borderStyle ) );
324 zone.mutable_border()->mutable_pitch()->set_value_nm( m_borderHatchPitch );
325
326 aContainer.PackFrom( zone );
327}
328
329
330bool ZONE::Deserialize( const google::protobuf::Any& aContainer )
331{
332 using namespace kiapi::board;
333 types::Zone zone;
335
336 if( !aContainer.UnpackTo( &zone ) )
337 return false;
338
339 const_cast<KIID&>( m_Uuid ) = KIID( zone.id().value() );
340 SetLayerSet( UnpackLayerSet( zone.layers() ) );
341 SetAssignedPriority( zone.priority() );
342 SetZoneName( wxString::FromUTF8( zone.name() ) );
343
344 if( zone.type() == types::ZoneType::ZT_RULE_AREA )
345 m_isRuleArea = true;
346
347 if( !m_Poly )
349
350 *m_Poly = kiapi::common::UnpackPolySet( zone.outline() );
351
352 if( m_Poly->OutlineCount() == 0 )
353 return false;
354
355 if( m_isRuleArea )
356 {
357 const types::RuleAreaSettings& ra = zone.rule_area_settings();
358 m_doNotAllowZoneFills = ra.keepout_copper();
359 m_doNotAllowFootprints = ra.keepout_footprints();
360 m_doNotAllowPads = ra.keepout_pads();
361 m_doNotAllowTracks = ra.keepout_tracks();
362 m_doNotAllowVias = ra.keepout_vias();
363
364 m_ruleAreaPlacementEnabled = ra.placement_enabled();
365 m_ruleAreaPlacementSource = wxString::FromUTF8( ra.placement_source() );
367 FromProtoEnum<RULE_AREA_PLACEMENT_SOURCE_TYPE>( ra.placement_source_type() );
368 }
369 else
370 {
371 const types::CopperZoneSettings& cu = zone.copper_settings();
372 m_PadConnection = FromProtoEnum<ZONE_CONNECTION>( cu.connection().zone_connection() );
373 m_thermalReliefSpokeWidth = cu.connection().thermal_spokes().width().value_nm();
374 m_thermalReliefGap = cu.connection().thermal_spokes().gap().value_nm();
375 m_ZoneClearance = cu.clearance().value_nm();
376 m_ZoneMinThickness = cu.min_thickness().value_nm();
377 m_islandRemovalMode = FromProtoEnum<ISLAND_REMOVAL_MODE>( cu.island_mode() );
378 m_minIslandArea = cu.min_island_area();
379 m_fillMode = FromProtoEnum<ZONE_FILL_MODE>( cu.fill_mode() );
380
381 m_hatchThickness = cu.hatch_settings().thickness().value_nm();
382 m_hatchGap = cu.hatch_settings().gap().value_nm();
384 EDA_ANGLE( cu.hatch_settings().orientation().value_degrees(), DEGREES_T );
385 m_hatchSmoothingValue = cu.hatch_settings().hatch_smoothing_ratio();
386 m_hatchHoleMinArea = cu.hatch_settings().hatch_hole_min_area_ratio();
387
388 switch( cu.hatch_settings().border_mode() )
389 {
390 default:
391 case types::ZHFBM_USE_MIN_ZONE_THICKNESS: m_hatchBorderAlgorithm = 0; break;
392 case types::ZHFBM_USE_HATCH_THICKNESS: m_hatchBorderAlgorithm = 1; break;
393 }
394
395 SetNetCode( cu.net().code().value() );
396 m_teardropType = FromProtoEnum<TEARDROP_TYPE>( cu.teardrop().type() );
397
398 for( const auto& properties : zone.layer_properties() )
399 {
400 PCB_LAYER_ID layer = FromProtoEnum<PCB_LAYER_ID>( properties.layer() );
401
402 ZONE_LAYER_PROPERTIES layerProperties;
403
404 if( properties.has_hatching_offset() )
405 {
406 layerProperties.hatching_offset = UnpackVector2( properties.hatching_offset() );
407 }
408
409 m_layerProperties[layer] = layerProperties;
410 }
411 }
412
413 m_borderStyle = FromProtoEnum<ZONE_BORDER_DISPLAY_STYLE>( zone.border().style() );
414 m_borderHatchPitch = zone.border().pitch().value_nm();
415
416 if( zone.filled() )
417 {
418 // TODO(JE) check what else has to happen here
419 SetIsFilled( true );
420 SetNeedRefill( false );
421
422 for( const types::ZoneFilledPolygons& fillLayer : zone.filled_polygons() )
423 {
424 PCB_LAYER_ID layer = FromProtoEnum<PCB_LAYER_ID>( fillLayer.layer() );
425 SHAPE_POLY_SET shape = kiapi::common::UnpackPolySet( fillLayer.shapes() );
426 m_FilledPolysList[layer] = std::make_shared<SHAPE_POLY_SET>( shape );
427 }
428 }
429
430 HatchBorder();
431
432 return true;
433}
434
435
436bool ZONE::HigherPriority( const ZONE* aOther ) const
437{
438 // Teardrops are always higher priority than regular zones, so if one zone is a teardrop
439 // and the other is not, then return higher priority as the teardrop
440 if( ( m_teardropType == TEARDROP_TYPE::TD_NONE )
441 ^ ( aOther->m_teardropType == TEARDROP_TYPE::TD_NONE ) )
442 {
443 return static_cast<int>( m_teardropType ) > static_cast<int>( aOther->m_teardropType );
444 }
445
446 if( m_priority != aOther->m_priority )
447 return m_priority > aOther->m_priority;
448
449 return m_Uuid > aOther->m_Uuid;
450}
451
452
453bool ZONE::SameNet( const ZONE* aOther ) const
454{
455 return GetNetCode() == aOther->GetNetCode();
456}
457
458
460{
461 bool change = false;
462
463 for( std::pair<const PCB_LAYER_ID, std::shared_ptr<SHAPE_POLY_SET>>& pair : m_FilledPolysList )
464 {
465 change |= !pair.second->IsEmpty();
466 m_insulatedIslands[pair.first].clear();
467 pair.second->RemoveAllContours();
468 }
469
470 m_isFilled = false;
472
473 return change;
474}
475
476
478{
479 return HasFlag( COURTYARD_CONFLICT );
480}
481
482
484{
485 return GetCornerPosition( 0 );
486}
487
488
490{
491 if( m_layerSet.count() == 1 )
492 {
493 return GetFirstLayer();
494 }
495 return UNDEFINED_LAYER;
496}
497
498
500{
501 if( m_layerSet.count() == 0 )
502 {
503 return UNDEFINED_LAYER;
504 }
505
506 const LSEQ uiLayers = m_layerSet.UIOrder();
507
508 // This can't use m_layerSet.count() because it's possible to have a zone on
509 // a rescue layer that is not in the UI order.
510 if( uiLayers.size() )
511 {
512 return uiLayers[0];
513 }
514
515 // If it's not in the UI set at all, just return the first layer in the set.
516 // (we know the count > 0)
517 return m_layerSet.Seq()[0];
518}
519
520
522{
523 return ( m_layerSet & LSET::AllCuMask() ).count() > 0;
524}
525
526
528{
529 SetLayerSet( LSET( { aLayer } ) );
530}
531
532
533void ZONE::SetLayerSet( const LSET& aLayerSet )
534{
535 if( aLayerSet.count() == 0 )
536 return;
537
538 if( m_layerSet != aLayerSet )
539 {
540 SetNeedRefill( true );
541
542 UnFill();
543
544 m_FilledPolysList.clear();
545 m_filledPolysHash.clear();
546 m_insulatedIslands.clear();
547
548 aLayerSet.RunOnLayers(
549 [&]( PCB_LAYER_ID layer )
550 {
551 m_FilledPolysList[layer] = std::make_shared<SHAPE_POLY_SET>();
552 m_filledPolysHash[layer] = {};
553 m_insulatedIslands[layer] = {};
554 } );
555
556 std::erase_if( m_layerProperties,
557 [&]( const auto& item )
558 {
559 return !aLayerSet.Contains( item.first );
560 } );
561 }
562
563 m_layerSet = aLayerSet;
564}
565
566
568{
569 wxCHECK_MSG( m_layerProperties.contains( aLayer ), m_layerProperties.at( GetFirstLayer() ),
570 "Attempt to retrieve properties for layer "
571 + std::string( magic_enum::enum_name( aLayer ) )
572 + " from a "
573 "zone that does not contain it" );
574 return m_layerProperties.at( aLayer );
575}
576
577
578void ZONE::SetLayerProperties( const std::map<PCB_LAYER_ID, ZONE_LAYER_PROPERTIES>& aOther )
579{
580 m_layerProperties.clear();
581
582 std::ranges::copy( aOther, std::inserter( m_layerProperties, std::end( m_layerProperties ) ) );
583}
584
585
586const std::optional<VECTOR2I>& ZONE::HatchingOffset( PCB_LAYER_ID aLayer ) const
587{
588 wxCHECK_MSG( m_layerProperties.contains( aLayer ),
589 m_layerProperties.at( GetFirstLayer() ).hatching_offset,
590 "Attempt to retrieve properties for layer "
591 + std::string( magic_enum::enum_name( aLayer ) )
592 + " from a "
593 "zone that does not contain it" );
594 return m_layerProperties.at( aLayer ).hatching_offset;
595}
596
597
598std::vector<int> ZONE::ViewGetLayers() const
599{
600 std::vector<int> layers;
601 layers.reserve( 2 * m_layerSet.count() + 1 );
602
604 [&]( PCB_LAYER_ID layer )
605 {
606 layers.push_back( layer );
607 layers.push_back( layer + static_cast<int>( LAYER_ZONE_START ) );
608 } );
609
610 if( IsConflicting() )
611 layers.push_back( LAYER_CONFLICTS_SHADOW );
612
613 return layers;
614}
615
616
617double ZONE::ViewGetLOD( int aLayer, const KIGFX::VIEW* aView ) const
618{
619 if( !aView )
620 return LOD_SHOW;
621
622 if( !aView->IsLayerVisible( LAYER_ZONES ) )
623 return LOD_HIDE;
624
625 if( FOOTPRINT* parentFP = GetParentFootprint() )
626 {
627 bool flipped = parentFP->GetLayer() == B_Cu;
628
629 // Handle Render tab switches
630 if( !flipped && !aView->IsLayerVisible( LAYER_FOOTPRINTS_FR ) )
631 return LOD_HIDE;
632
633 if( flipped && !aView->IsLayerVisible( LAYER_FOOTPRINTS_BK ) )
634 return LOD_HIDE;
635 }
636
637 // Other layers are shown without any conditions
638 return LOD_SHOW;
639}
640
641
642bool ZONE::IsOnLayer( PCB_LAYER_ID aLayer ) const
643{
644 return m_layerSet.test( aLayer );
645}
646
647
649{
650 if( const BOARD* board = GetBoard() )
651 {
652 std::unordered_map<const ZONE*, BOX2I>& cache = board->m_ZoneBBoxCache;
653
654 {
655 std::shared_lock<std::shared_mutex> readLock( board->m_CachesMutex );
656
657 auto cacheIter = cache.find( this );
658
659 if( cacheIter != cache.end() )
660 return cacheIter->second;
661 }
662
663 BOX2I bbox = m_Poly->BBox();
664
665 {
666 std::unique_lock<std::shared_mutex> writeLock( board->m_CachesMutex );
667 cache[ this ] = bbox;
668 }
669
670 return bbox;
671 }
672
673 return m_Poly->BBox();
674}
675
676
678{
679 // GetBoundingBox() will cache it for us, and there's no sense duplicating the somewhat tricky
680 // locking code.
682}
683
684
685int ZONE::GetThermalReliefGap( PAD* aPad, wxString* aSource ) const
686{
687 if( aPad->GetLocalThermalGapOverride() == 0 )
688 {
689 if( aSource )
690 *aSource = _( "zone" );
691
692 return m_thermalReliefGap;
693 }
694
695 return aPad->GetLocalThermalGapOverride( aSource );
696
697}
698
699
700void ZONE::SetCornerRadius( unsigned int aRadius )
701{
702 if( m_cornerRadius != aRadius )
703 SetNeedRefill( true );
704
705 m_cornerRadius = aRadius;
706}
707
708
710
711
713{
714 if( !m_filledPolysHash.count( aLayer ) )
715 return g_nullPoly.GetHash();
716 else
717 return m_filledPolysHash.at( aLayer );
718}
719
720
722{
723 if( !m_FilledPolysList.count( aLayer ) )
725 else
726 m_filledPolysHash[aLayer] = m_FilledPolysList.at( aLayer )->GetHash();
727}
728
729
730bool ZONE::HitTest( const VECTOR2I& aPosition, int aAccuracy ) const
731{
732 // When looking for an "exact" hit aAccuracy will be 0 which works poorly for very thin
733 // lines. Give it a floor.
734 int accuracy = std::max( aAccuracy, pcbIUScale.mmToIU( 0.1 ) );
735
736 return HitTestForCorner( aPosition, accuracy * 2 ) || HitTestForEdge( aPosition, accuracy );
737}
738
739
740bool ZONE::HitTestForCorner( const VECTOR2I& refPos, int aAccuracy,
741 SHAPE_POLY_SET::VERTEX_INDEX* aCornerHit ) const
742{
743 return m_Poly->CollideVertex( VECTOR2I( refPos ), aCornerHit, aAccuracy );
744}
745
746
747bool ZONE::HitTestForEdge( const VECTOR2I& refPos, int aAccuracy,
748 SHAPE_POLY_SET::VERTEX_INDEX* aCornerHit ) const
749{
750 return m_Poly->CollideEdge( VECTOR2I( refPos ), aCornerHit, aAccuracy );
751}
752
753
754bool ZONE::HitTest( const BOX2I& aRect, bool aContained, int aAccuracy ) const
755{
756 // Calculate bounding box for zone
757 BOX2I bbox = GetBoundingBox();
758 bbox.Normalize();
759
760 BOX2I arect = aRect;
761 arect.Normalize();
762 arect.Inflate( aAccuracy );
763
764 if( aContained )
765 {
766 return arect.Contains( bbox );
767 }
768 else
769 {
770 // Fast test: if aBox is outside the polygon bounding box, rectangles cannot intersect
771 if( !arect.Intersects( bbox ) )
772 return false;
773
774 int count = m_Poly->TotalVertices();
775
776 for( int ii = 0; ii < count; ii++ )
777 {
778 VECTOR2I vertex = m_Poly->CVertex( ii );
779 VECTOR2I vertexNext = m_Poly->CVertex( ( ii + 1 ) % count );
780
781 // Test if the point is within the rect
782 if( arect.Contains( vertex ) )
783 return true;
784
785 // Test if this edge intersects the rect
786 if( arect.Intersects( vertex, vertexNext ) )
787 return true;
788 }
789
790 return false;
791 }
792}
793
794
795std::optional<int> ZONE::GetLocalClearance() const
796{
797 return m_isRuleArea ? 0 : m_ZoneClearance;
798}
799
800
801bool ZONE::HitTestFilledArea( PCB_LAYER_ID aLayer, const VECTOR2I& aRefPos, int aAccuracy ) const
802{
803 // Rule areas have no filled area, but it's generally nice to treat their interior as if it were
804 // filled so that people don't have to select them by their outline (which is min-width)
805 if( GetIsRuleArea() )
806 return m_Poly->Contains( aRefPos, -1, aAccuracy );
807
808 if( !m_FilledPolysList.count( aLayer ) )
809 return false;
810
811 return m_FilledPolysList.at( aLayer )->Contains( aRefPos, -1, aAccuracy );
812}
813
814
815bool ZONE::HitTestCutout( const VECTOR2I& aRefPos, int* aOutlineIdx, int* aHoleIdx ) const
816{
817 // Iterate over each outline polygon in the zone and then iterate over
818 // each hole it has to see if the point is in it.
819 for( int i = 0; i < m_Poly->OutlineCount(); i++ )
820 {
821 for( int j = 0; j < m_Poly->HoleCount( i ); j++ )
822 {
823 if( m_Poly->Hole( i, j ).PointInside( aRefPos ) )
824 {
825 if( aOutlineIdx )
826 *aOutlineIdx = i;
827
828 if( aHoleIdx )
829 *aHoleIdx = j;
830
831 return true;
832 }
833 }
834 }
835
836 return false;
837}
838
839
840void ZONE::GetMsgPanelInfo( EDA_DRAW_FRAME* aFrame, std::vector<MSG_PANEL_ITEM>& aList )
841{
842 wxString msg = GetFriendlyName();
843
844 // Display Cutout instead of Outline for holes inside a zone (i.e. when num contour !=0).
845 // Check whether the selected corner is in a hole; i.e., in any contour but the first one.
846 if( m_CornerSelection != nullptr && m_CornerSelection->m_contour > 0 )
847 msg << wxT( " " ) << _( "Cutout" );
848
849 aList.emplace_back( _( "Type" ), msg );
850
851 if( GetIsRuleArea() )
852 {
853 msg.Empty();
854
855 if( GetDoNotAllowVias() )
856 AccumulateDescription( msg, _( "No vias" ) );
857
858 if( GetDoNotAllowTracks() )
859 AccumulateDescription( msg, _( "No tracks" ) );
860
861 if( GetDoNotAllowPads() )
862 AccumulateDescription( msg, _( "No pads" ) );
863
865 AccumulateDescription( msg, _( "No zone fills" ) );
866
868 AccumulateDescription( msg, _( "No footprints" ) );
869
870 if( !msg.IsEmpty() )
871 aList.emplace_back( _( "Restrictions" ), msg );
872
874 {
875 aList.emplace_back( _( "Placement source" ),
877 }
878 }
879 else if( IsOnCopperLayer() )
880 {
881 if( aFrame->GetName() == PCB_EDIT_FRAME_NAME )
882 {
883 aList.emplace_back( _( "Net" ), UnescapeString( GetNetname() ) );
884
885 aList.emplace_back( _( "Resolved Netclass" ),
886 UnescapeString( GetEffectiveNetClass()->GetHumanReadableName() ) );
887 }
888
889 // Display priority level
890 aList.emplace_back( _( "Priority" ),
891 wxString::Format( wxT( "%d" ), GetAssignedPriority() ) );
892 }
893
894 if( aFrame->GetName() == PCB_EDIT_FRAME_NAME )
895 {
896 if( IsLocked() )
897 aList.emplace_back( _( "Status" ), _( "Locked" ) );
898 }
899
900 LSEQ layers = m_layerSet.Seq();
901 wxString layerDesc;
902
903 if( layers.size() == 1 )
904 {
905 layerDesc.Printf( _( "%s" ), GetBoard()->GetLayerName( layers[0] ) );
906 }
907 else if (layers.size() == 2 )
908 {
909 layerDesc.Printf( _( "%s and %s" ),
910 GetBoard()->GetLayerName( layers[0] ),
911 GetBoard()->GetLayerName( layers[1] ) );
912 }
913 else if (layers.size() == 3 )
914 {
915 layerDesc.Printf( _( "%s, %s and %s" ),
916 GetBoard()->GetLayerName( layers[0] ),
917 GetBoard()->GetLayerName( layers[1] ),
918 GetBoard()->GetLayerName( layers[2] ) );
919 }
920 else if( layers.size() > 3 )
921 {
922 layerDesc.Printf( _( "%s, %s and %d more" ),
923 GetBoard()->GetLayerName( layers[0] ),
924 GetBoard()->GetLayerName( layers[1] ),
925 static_cast<int>( layers.size() - 2 ) );
926 }
927
928 aList.emplace_back( _( "Layer" ), layerDesc );
929
930 if( !m_zoneName.empty() )
931 aList.emplace_back( _( "Name" ), m_zoneName );
932
933 switch( m_fillMode )
934 {
935 case ZONE_FILL_MODE::POLYGONS: msg = _( "Solid" ); break;
936 case ZONE_FILL_MODE::HATCH_PATTERN: msg = _( "Hatched" ); break;
937 default: msg = _( "Unknown" ); break;
938 }
939
940 aList.emplace_back( _( "Fill Mode" ), msg );
941
942 aList.emplace_back( _( "Filled Area" ),
943 aFrame->MessageTextFromValue( m_area, true, EDA_DATA_TYPE::AREA ) );
944
945 wxString source;
946 int clearance = GetOwnClearance( UNDEFINED_LAYER, &source );
947
948 if( !source.IsEmpty() )
949 {
950 aList.emplace_back( wxString::Format( _( "Min Clearance: %s" ),
951 aFrame->MessageTextFromValue( clearance ) ),
952 wxString::Format( _( "(from %s)" ),
953 source ) );
954 }
955
956 if( !m_FilledPolysList.empty() )
957 {
958 int count = 0;
959
960 for( std::pair<const PCB_LAYER_ID, std::shared_ptr<SHAPE_POLY_SET>>& ii: m_FilledPolysList )
961 count += ii.second->TotalVertices();
962
963 aList.emplace_back( _( "Corner Count" ), wxString::Format( wxT( "%d" ), count ) );
964 }
965}
966
967
968void ZONE::Move( const VECTOR2I& offset )
969{
970 /* move outlines */
971 m_Poly->Move( offset );
972
973 HatchBorder();
974
975 /* move fills */
976 for( std::pair<const PCB_LAYER_ID, std::shared_ptr<SHAPE_POLY_SET>>& pair : m_FilledPolysList )
977 pair.second->Move( offset );
978
979 /*
980 * move boundingbox cache
981 *
982 * While the cache will get nuked at the conclusion of the operation, we use it for some
983 * things (such as drawing the parent group) during the move.
984 */
985 if( GetBoard() )
986 {
987 auto it = GetBoard()->m_ZoneBBoxCache.find( this );
988
989 if( it != GetBoard()->m_ZoneBBoxCache.end() )
990 it->second.Move( offset );
991 }
992}
993
994
995wxString ZONE::GetFriendlyName() const
996{
997 if( GetIsRuleArea() )
998 return _( "Rule Area" );
999 else if( IsTeardropArea() )
1000 return _( "Teardrop Area" );
1001 else if( IsOnCopperLayer() )
1002 return _( "Copper Zone" );
1003 else
1004 return _( "Non-copper Zone" );
1005}
1006
1007
1008void ZONE::MoveEdge( const VECTOR2I& offset, int aEdge )
1009{
1010 int next_corner;
1011
1012 if( m_Poly->GetNeighbourIndexes( aEdge, nullptr, &next_corner ) )
1013 {
1014 m_Poly->SetVertex( aEdge, m_Poly->CVertex( aEdge ) + VECTOR2I( offset ) );
1015 m_Poly->SetVertex( next_corner, m_Poly->CVertex( next_corner ) + VECTOR2I( offset ) );
1016 HatchBorder();
1017
1018 SetNeedRefill( true );
1019 }
1020}
1021
1022
1023void ZONE::Rotate( const VECTOR2I& aCentre, const EDA_ANGLE& aAngle )
1024{
1025 m_Poly->Rotate( aAngle, aCentre );
1026 HatchBorder();
1027
1028 /* rotate filled areas: */
1029 for( std::pair<const PCB_LAYER_ID, std::shared_ptr<SHAPE_POLY_SET>>& pair : m_FilledPolysList )
1030 pair.second->Rotate( aAngle, aCentre );
1031}
1032
1033
1034void ZONE::Flip( const VECTOR2I& aCentre, FLIP_DIRECTION aFlipDirection )
1035{
1036 Mirror( aCentre, aFlipDirection );
1037
1038 std::map<PCB_LAYER_ID, SHAPE_POLY_SET> fillsCopy;
1039
1040 for( auto& [oldLayer, shapePtr] : m_FilledPolysList )
1041 {
1042 fillsCopy[oldLayer] = *shapePtr;
1043 }
1044
1045 std::map<PCB_LAYER_ID, ZONE_LAYER_PROPERTIES> layerPropertiesCopy;
1046
1047 std::ranges::copy( m_layerProperties,
1048 std::inserter( layerPropertiesCopy, std::end( layerPropertiesCopy ) ) );
1049
1050 LSET flipped;
1051
1052 for( PCB_LAYER_ID layer : GetLayerSet() )
1053 flipped.set( GetBoard()->FlipLayer( layer ) );
1054
1055 SetLayerSet( flipped );
1056
1057 for( auto& [oldLayer, properties] : layerPropertiesCopy )
1058 {
1059 PCB_LAYER_ID newLayer = GetBoard()->FlipLayer( oldLayer );
1060 m_layerProperties[newLayer] = properties;
1061 }
1062
1063 for( auto& [oldLayer, shape] : fillsCopy )
1064 {
1065 PCB_LAYER_ID newLayer = GetBoard()->FlipLayer( oldLayer );
1066 SetFilledPolysList( newLayer, shape );
1067 }
1068}
1069
1070
1071void ZONE::Mirror( const VECTOR2I& aMirrorRef, FLIP_DIRECTION aFlipDirection )
1072{
1073 m_Poly->Mirror( aMirrorRef, aFlipDirection );
1074
1075 HatchBorder();
1076
1077 for( std::pair<const PCB_LAYER_ID, std::shared_ptr<SHAPE_POLY_SET>>& pair : m_FilledPolysList )
1078 pair.second->Mirror( aMirrorRef, aFlipDirection );
1079}
1080
1081
1082void ZONE::RemoveCutout( int aOutlineIdx, int aHoleIdx )
1083{
1084 // Ensure the requested cutout is valid
1085 if( m_Poly->OutlineCount() < aOutlineIdx || m_Poly->HoleCount( aOutlineIdx ) < aHoleIdx )
1086 return;
1087
1088 SHAPE_POLY_SET cutPoly( m_Poly->Hole( aOutlineIdx, aHoleIdx ) );
1089
1090 // Add the cutout back to the zone
1091 m_Poly->BooleanAdd( cutPoly );
1092
1093 SetNeedRefill( true );
1094}
1095
1096
1097void ZONE::AddPolygon( const SHAPE_LINE_CHAIN& aPolygon )
1098{
1099 wxASSERT( aPolygon.IsClosed() );
1100
1101 // Add the outline as a new polygon in the polygon set
1102 if( m_Poly->OutlineCount() == 0 )
1103 m_Poly->AddOutline( aPolygon );
1104 else
1105 m_Poly->AddHole( aPolygon );
1106
1107 SetNeedRefill( true );
1108}
1109
1110
1111void ZONE::AddPolygon( std::vector<VECTOR2I>& aPolygon )
1112{
1113 if( aPolygon.empty() )
1114 return;
1115
1116 SHAPE_LINE_CHAIN outline;
1117
1118 // Create an outline and populate it with the points of aPolygon
1119 for( const VECTOR2I& pt : aPolygon )
1120 outline.Append( pt );
1121
1122 outline.SetClosed( true );
1123
1124 AddPolygon( outline );
1125}
1126
1127
1128bool ZONE::AppendCorner( VECTOR2I aPosition, int aHoleIdx, bool aAllowDuplication )
1129{
1130 // Ensure the main outline exists:
1131 if( m_Poly->OutlineCount() == 0 )
1132 m_Poly->NewOutline();
1133
1134 // If aHoleIdx >= 0, the corner musty be added to the hole, index aHoleIdx.
1135 // (remember: the index of the first hole is 0)
1136 // Return error if it does not exist.
1137 if( aHoleIdx >= m_Poly->HoleCount( 0 ) )
1138 return false;
1139
1140 m_Poly->Append( aPosition.x, aPosition.y, -1, aHoleIdx, aAllowDuplication );
1141
1142 SetNeedRefill( true );
1143
1144 return true;
1145}
1146
1147
1148wxString ZONE::GetItemDescription( UNITS_PROVIDER* aUnitsProvider, bool aFull ) const
1149{
1150 LSEQ layers = m_layerSet.Seq();
1151 wxString layerDesc;
1152
1153 if( layers.size() == 1 )
1154 {
1155 layerDesc.Printf( _( "on %s" ), GetBoard()->GetLayerName( layers[0] ) );
1156 }
1157 else if (layers.size() == 2 )
1158 {
1159 layerDesc.Printf( _( "on %s and %s" ),
1160 GetBoard()->GetLayerName( layers[0] ),
1161 GetBoard()->GetLayerName( layers[1] ) );
1162 }
1163 else if (layers.size() == 3 )
1164 {
1165 layerDesc.Printf( _( "on %s, %s and %s" ),
1166 GetBoard()->GetLayerName( layers[0] ),
1167 GetBoard()->GetLayerName( layers[1] ),
1168 GetBoard()->GetLayerName( layers[2] ) );
1169 }
1170 else if( layers.size() > 3 )
1171 {
1172 layerDesc.Printf( _( "on %s, %s and %zu more" ),
1173 GetBoard()->GetLayerName( layers[0] ),
1174 GetBoard()->GetLayerName( layers[1] ),
1175 layers.size() - 2 );
1176 }
1177
1178 // Check whether the selected contour is a hole (contour index > 0)
1179 if( m_CornerSelection != nullptr && m_CornerSelection->m_contour > 0 )
1180 {
1181 if( GetIsRuleArea() )
1182 return wxString::Format( _( "Rule Area Cutout %s" ), layerDesc );
1183 else
1184 return wxString::Format( _( "Zone Cutout %s" ), layerDesc );
1185 }
1186 else
1187 {
1188 if( GetIsRuleArea() )
1189 return wxString::Format( _( "Rule Area %s" ), layerDesc );
1190 else if( IsTeardropArea() )
1191 return wxString::Format( _( "Teardrop %s %s" ), GetNetnameMsg(), layerDesc );
1192 else
1193 return wxString::Format( _( "Zone %s %s" ), GetNetnameMsg(), layerDesc );
1194 }
1195}
1196
1197
1199{
1200 return m_borderHatchPitch;
1201}
1202
1203
1205 int aBorderHatchPitch, bool aRebuildBorderHatch )
1206{
1207 aBorderHatchPitch = std::max( aBorderHatchPitch,
1209 aBorderHatchPitch = std::min( aBorderHatchPitch,
1211 SetBorderHatchPitch( aBorderHatchPitch );
1212 m_borderStyle = aBorderHatchStyle;
1213
1214 if( aRebuildBorderHatch )
1215 HatchBorder();
1216}
1217
1218
1220{
1221 m_borderHatchPitch = aPitch;
1222}
1223
1224
1226{
1227 m_borderHatchLines.clear();
1228}
1229
1230
1232{
1233 UnHatchBorder();
1234
1235 if( m_borderStyle == ZONE_BORDER_DISPLAY_STYLE::NO_HATCH
1236 || m_borderHatchPitch == 0
1237 || m_Poly->IsEmpty() )
1238 {
1239 return;
1240 }
1241
1242 // Calculate spacing between 2 hatch lines
1243 int spacing;
1244
1245 if( m_borderStyle == ZONE_BORDER_DISPLAY_STYLE::DIAGONAL_EDGE )
1246 spacing = m_borderHatchPitch;
1247 else
1248 spacing = m_borderHatchPitch * 2;
1249
1250 // set the "length" of hatch lines (the length on horizontal axis)
1251 int hatch_line_len = m_borderHatchPitch;
1252
1253 // To have a better look, give a slope depending on the layer
1254 int layer = GetFirstLayer();
1255 std::vector<double> slopes;
1256
1257 if( IsTeardropArea() )
1258 slopes = { 0.7, -0.7 };
1259 else if( layer & 1 )
1260 slopes = { 1 };
1261 else
1262 slopes = { -1 };
1263
1264 m_borderHatchLines = m_Poly->GenerateHatchLines( slopes, spacing, hatch_line_len );
1265}
1266
1267
1269{
1271}
1272
1273
1275{
1276 return BITMAPS::add_zone;
1277}
1278
1279
1281{
1282 assert( aImage->Type() == PCB_ZONE_T );
1283
1284 std::swap( *static_cast<ZONE*>( this ), *static_cast<ZONE*>( aImage) );
1285}
1286
1287
1289{
1290 if( aLayer == UNDEFINED_LAYER )
1291 {
1292 for( auto& [ layer, poly ] : m_FilledPolysList )
1293 poly->CacheTriangulation();
1294
1295 m_Poly->CacheTriangulation( false );
1296 }
1297 else
1298 {
1299 if( m_FilledPolysList.count( aLayer ) )
1300 m_FilledPolysList[ aLayer ]->CacheTriangulation();
1301 }
1302}
1303
1304
1305bool ZONE::IsIsland( PCB_LAYER_ID aLayer, int aPolyIdx ) const
1306{
1307 if( GetNetCode() < 1 )
1308 return true;
1309
1310 if( !m_insulatedIslands.count( aLayer ) )
1311 return false;
1312
1313 return m_insulatedIslands.at( aLayer ).count( aPolyIdx );
1314}
1315
1316
1317void ZONE::GetInteractingZones( PCB_LAYER_ID aLayer, std::vector<ZONE*>* aSameNetCollidingZones,
1318 std::vector<ZONE*>* aOtherNetIntersectingZones ) const
1319{
1320 int epsilon = pcbIUScale.mmToIU( 0.001 );
1321 BOX2I bbox = GetBoundingBox();
1322
1323 bbox.Inflate( epsilon );
1324
1325 for( ZONE* candidate : GetBoard()->Zones() )
1326 {
1327 if( candidate == this )
1328 continue;
1329
1330 if( !candidate->GetLayerSet().test( aLayer ) )
1331 continue;
1332
1333 if( candidate->GetIsRuleArea() || candidate->IsTeardropArea() )
1334 continue;
1335
1336 if( !candidate->GetBoundingBox().Intersects( bbox ) )
1337 continue;
1338
1339 if( candidate->GetNetCode() == GetNetCode() )
1340 {
1341 if( m_Poly->Collide( candidate->m_Poly ) )
1342 aSameNetCollidingZones->push_back( candidate );
1343 }
1344 else
1345 {
1346 aOtherNetIntersectingZones->push_back( candidate );
1347 }
1348 }
1349}
1350
1351
1353 SHAPE_POLY_SET* aBoardOutline,
1354 SHAPE_POLY_SET* aSmoothedPolyWithApron ) const
1355{
1356 if( GetNumCorners() <= 2 ) // malformed zone. polygon calculations will not like it ...
1357 return false;
1358
1359 // Processing of arc shapes in zones is not yet supported because Clipper can't do boolean
1360 // operations on them. The poly outline must be converted to segments first.
1362 flattened.ClearArcs();
1363
1364 if( GetIsRuleArea() )
1365 {
1366 // We like keepouts just the way they are....
1367 aSmoothedPoly = flattened;
1368 return true;
1369 }
1370
1371 const BOARD* board = GetBoard();
1372 int maxError = ARC_HIGH_DEF;
1373 bool keepExternalFillets = false;
1376
1377 if( IsTeardropArea() )
1378 {
1379 // We use teardrop shapes with no smoothing; these shapes are already optimized
1380 smooth_requested = false;
1381 }
1382
1383 if( board )
1384 {
1386
1387 maxError = bds.m_MaxError;
1388 keepExternalFillets = bds.m_ZoneKeepExternalFillets;
1389 }
1390
1391 auto smooth =
1392 [&]( SHAPE_POLY_SET& aPoly )
1393 {
1394 if( !smooth_requested )
1395 return;
1396
1397 switch( m_cornerSmoothingType )
1398 {
1400 aPoly = aPoly.Chamfer( (int) m_cornerRadius );
1401 break;
1402
1404 aPoly = aPoly.Fillet( (int) m_cornerRadius, maxError );
1405 break;
1406
1407 default:
1408 break;
1409 }
1410 };
1411
1412 SHAPE_POLY_SET* maxExtents = &flattened;
1413 SHAPE_POLY_SET withFillets;
1414
1415 aSmoothedPoly = flattened;
1416
1417 // Should external fillets (that is, those applied to concave corners) be kept? While it
1418 // seems safer to never have copper extend outside the zone outline, 5.1.x and prior did
1419 // indeed fill them so we leave the mode available.
1420 if( keepExternalFillets && smooth_requested )
1421 {
1422 withFillets = flattened;
1423 smooth( withFillets );
1424 withFillets.BooleanAdd( flattened );
1425 maxExtents = &withFillets;
1426 }
1427
1428 // We now add in the areas of any same-net, intersecting zones. This keeps us from smoothing
1429 // corners at an intersection (which often produces undesired divots between the intersecting
1430 // zones -- see #2752).
1431 //
1432 // After smoothing, we'll subtract back out everything outside of our zone.
1433 std::vector<ZONE*> sameNetCollidingZones;
1434 std::vector<ZONE*> diffNetIntersectingZones;
1435 GetInteractingZones( aLayer, &sameNetCollidingZones, &diffNetIntersectingZones );
1436
1437 for( ZONE* sameNetZone : sameNetCollidingZones )
1438 {
1439 BOX2I sameNetBoundingBox = sameNetZone->GetBoundingBox();
1440
1441 // Note: a two-pass algorithm could use sameNetZone's actual fill instead of its outline.
1442 // This would obviate the need for the below wrinkles, in addition to fixing both issues
1443 // in #16095.
1444 // (And we wouldn't need to collect all the diffNetIntersectingZones either.)
1445
1446 SHAPE_POLY_SET sameNetPoly = sameNetZone->Outline()->CloneDropTriangulation();
1447 SHAPE_POLY_SET diffNetPoly;
1448
1449 // Of course there's always a wrinkle. The same-net intersecting zone *might* get knocked
1450 // out along the border by a higher-priority, different-net zone. #12797
1451 for( ZONE* diffNetZone : diffNetIntersectingZones )
1452 {
1453 if( diffNetZone->HigherPriority( sameNetZone )
1454 && diffNetZone->GetBoundingBox().Intersects( sameNetBoundingBox ) )
1455 {
1456 diffNetPoly.BooleanAdd( *diffNetZone->Outline() );
1457 }
1458 }
1459
1460 // Second wrinkle. After unioning the higher priority, different net zones together, we
1461 // need to check to see if they completely enclose our zone. If they do, then we need to
1462 // treat the enclosed zone as isolated, not connected to the outer zone. #13915
1463 bool isolated = false;
1464
1465 if( diffNetPoly.OutlineCount() )
1466 {
1468
1469 thisPoly.BooleanSubtract( diffNetPoly );
1470 isolated = thisPoly.OutlineCount() == 0;
1471 }
1472
1473 if( !isolated )
1474 {
1475 sameNetPoly.ClearArcs();
1476 aSmoothedPoly.BooleanAdd( sameNetPoly );
1477 }
1478 }
1479
1480 if( aBoardOutline )
1481 aSmoothedPoly.BooleanIntersection( *aBoardOutline );
1482
1483 SHAPE_POLY_SET withSameNetIntersectingZones = aSmoothedPoly.CloneDropTriangulation();
1484
1485 smooth( aSmoothedPoly );
1486
1487 if( aSmoothedPolyWithApron )
1488 {
1489 // The same-net intersecting-zone code above makes sure the corner-smoothing algorithm
1490 // doesn't produce divots. But the min-thickness algorithm applied in fillCopperZone()
1491 // is *also* going to perform a deflate/inflate cycle, again leading to divots. So we
1492 // pre-inflate the contour by the min-thickness within the same-net-intersecting-zones
1493 // envelope.
1494 SHAPE_POLY_SET poly = maxExtents->CloneDropTriangulation();
1495 poly.Inflate( m_ZoneMinThickness, CORNER_STRATEGY::ROUND_ALL_CORNERS, maxError );
1496
1497 if( !keepExternalFillets )
1498 poly.BooleanIntersection( withSameNetIntersectingZones );
1499
1500 *aSmoothedPolyWithApron = aSmoothedPoly;
1501 aSmoothedPolyWithApron->BooleanIntersection( poly );
1502 }
1503
1504 aSmoothedPoly.BooleanIntersection( *maxExtents );
1505
1506 return true;
1507}
1508
1509
1511{
1512 m_area = 0.0;
1513
1514 for( const auto& [layer, poly] : m_FilledPolysList )
1515 m_area += poly->Area();
1516
1517 return m_area;
1518}
1519
1520
1522{
1524 return m_outlinearea;
1525}
1526
1527
1529 int aMaxError, ERROR_LOC aErrorLoc,
1530 SHAPE_POLY_SET* aBoardOutline ) const
1531{
1532 // Creates the zone outline polygon (with holes if any)
1533 SHAPE_POLY_SET polybuffer;
1534
1535 // TODO: using GetFirstLayer() means it only works for single-layer zones....
1536 BuildSmoothedPoly( polybuffer, GetFirstLayer(), aBoardOutline );
1537
1538 // Calculate the polygon with clearance
1539 // holes are linked to the main outline, so only one polygon is created.
1540 if( aClearance )
1541 {
1542 const BOARD* board = GetBoard();
1543 int maxError = ARC_HIGH_DEF;
1544
1545 if( board )
1546 maxError = board->GetDesignSettings().m_MaxError;
1547
1548 if( aErrorLoc == ERROR_OUTSIDE )
1549 aClearance += maxError;
1550
1551 polybuffer.Inflate( aClearance, CORNER_STRATEGY::ROUND_ALL_CORNERS, maxError );
1552 }
1553
1554 polybuffer.Fracture();
1555 aBuffer.Append( polybuffer );
1556}
1557
1558
1559std::shared_ptr<SHAPE> ZONE::GetEffectiveShape( PCB_LAYER_ID aLayer, FLASHING aFlash ) const
1560{
1561 if( m_FilledPolysList.find( aLayer ) == m_FilledPolysList.end() )
1562 return std::make_shared<SHAPE_NULL>();
1563 else
1564 return m_FilledPolysList.at( aLayer );
1565}
1566
1567
1568void ZONE::TransformShapeToPolygon( SHAPE_POLY_SET& aBuffer, PCB_LAYER_ID aLayer, int aClearance,
1569 int aError, ERROR_LOC aErrorLoc, bool aIgnoreLineWidth ) const
1570{
1571 wxASSERT_MSG( !aIgnoreLineWidth, wxT( "IgnoreLineWidth has no meaning for zones." ) );
1572
1573 if( !m_FilledPolysList.count( aLayer ) )
1574 return;
1575
1576 if( !aClearance )
1577 {
1578 aBuffer.Append( *m_FilledPolysList.at( aLayer ) );
1579 return;
1580 }
1581
1582 SHAPE_POLY_SET temp_buf = m_FilledPolysList.at( aLayer )->CloneDropTriangulation();
1583
1584 // Rebuild filled areas only if clearance is not 0
1585 if( aClearance > 0 || aErrorLoc == ERROR_OUTSIDE )
1586 {
1587 if( aErrorLoc == ERROR_OUTSIDE )
1588 aClearance += aError;
1589
1590 temp_buf.InflateWithLinkedHoles( aClearance, CORNER_STRATEGY::ROUND_ALL_CORNERS, aError );
1591 }
1592
1593 aBuffer.Append( temp_buf );
1594}
1595
1596
1598{
1599 if( m_FilledPolysList.count( aLayer ) && !m_FilledPolysList.at( aLayer )->IsEmpty() )
1600 aBuffer.Append( *m_FilledPolysList.at( aLayer ) );
1601}
1602
1603
1605{
1606 if( aLayerSet.count() == 0 )
1607 return;
1608
1609 if( m_layerSet != aLayerSet )
1610 {
1611 aLayerSet.RunOnLayers(
1612 [&]( PCB_LAYER_ID layer )
1613 {
1614 // Only keep layers that are present in the new set
1615 if( !aLayerSet.Contains( layer ) )
1616 {
1617 m_FilledPolysList[layer] = std::make_shared<SHAPE_POLY_SET>();
1618 m_filledPolysHash[layer] = {};
1619 m_insulatedIslands[layer] = {};
1620 }
1621 } );
1622 }
1623
1624 m_layerSet = aLayerSet;
1625}
1626
1627
1628bool ZONE::operator==( const BOARD_ITEM& aOther ) const
1629{
1630 if( aOther.Type() != Type() )
1631 return false;
1632
1633 const ZONE& other = static_cast<const ZONE&>( aOther );
1634 return *this == other;
1635}
1636
1637
1638bool ZONE::operator==( const ZONE& aOther ) const
1639
1640{
1641 if( aOther.Type() != Type() )
1642 return false;
1643
1644 const ZONE& other = static_cast<const ZONE&>( aOther );
1645
1646 if( GetIsRuleArea() != other.GetIsRuleArea() )
1647 return false;
1648
1649 if( GetIsRuleArea() )
1650 {
1652 return false;
1653
1654 if( GetDoNotAllowTracks() != other.GetDoNotAllowTracks() )
1655 return false;
1656
1657 if( GetDoNotAllowVias() != other.GetDoNotAllowVias() )
1658 return false;
1659
1661 return false;
1662
1663 if( GetDoNotAllowPads() != other.GetDoNotAllowPads() )
1664 return false;
1665
1667 return false;
1668
1670 return false;
1671
1673 return false;
1674 }
1675 else
1676 {
1677 if( GetAssignedPriority() != other.GetAssignedPriority() )
1678 return false;
1679
1680 if( GetMinThickness() != other.GetMinThickness() )
1681 return false;
1682
1684 return false;
1685
1686 if( GetCornerRadius() != other.GetCornerRadius() )
1687 return false;
1688
1689 if( GetTeardropParams() != other.GetTeardropParams() )
1690 return false;
1691 }
1692
1693 if( GetNumCorners() != other.GetNumCorners() )
1694 return false;
1695
1696 for( int ii = 0; ii < GetNumCorners(); ii++ )
1697 {
1698 if( GetCornerPosition( ii ) != other.GetCornerPosition( ii ) )
1699 return false;
1700 }
1701
1702 return true;
1703}
1704
1705
1706double ZONE::Similarity( const BOARD_ITEM& aOther ) const
1707{
1708 if( aOther.Type() != Type() )
1709 return 0.0;
1710
1711 const ZONE& other = static_cast<const ZONE&>( aOther );
1712
1713 if( GetIsRuleArea() != other.GetIsRuleArea() )
1714 return 0.0;
1715
1716 double similarity = 1.0;
1717
1718 if( GetLayerSet() != other.GetLayerSet() )
1719 similarity *= 0.9;
1720
1721 if( GetNetCode() != other.GetNetCode() )
1722 similarity *= 0.9;
1723
1724 if( !GetIsRuleArea() )
1725 {
1726 if( GetAssignedPriority() != other.GetAssignedPriority() )
1727 similarity *= 0.9;
1728
1729 if( GetMinThickness() != other.GetMinThickness() )
1730 similarity *= 0.9;
1731
1733 similarity *= 0.9;
1734
1735 if( GetCornerRadius() != other.GetCornerRadius() )
1736 similarity *= 0.9;
1737
1738 if( GetTeardropParams() != other.GetTeardropParams() )
1739 similarity *= 0.9;
1740 }
1741 else
1742 {
1744 similarity *= 0.9;
1745 if( GetDoNotAllowTracks() != other.GetDoNotAllowTracks() )
1746 similarity *= 0.9;
1747 if( GetDoNotAllowVias() != other.GetDoNotAllowVias() )
1748 similarity *= 0.9;
1750 similarity *= 0.9;
1751 if( GetDoNotAllowPads() != other.GetDoNotAllowPads() )
1752 similarity *= 0.9;
1753 }
1754
1755 std::vector<VECTOR2I> corners;
1756 std::vector<VECTOR2I> otherCorners;
1757 VECTOR2I lastCorner( 0, 0 );
1758
1759 for( int ii = 0; ii < GetNumCorners(); ii++ )
1760 {
1761 corners.push_back( lastCorner - GetCornerPosition( ii ) );
1762 lastCorner = GetCornerPosition( ii );
1763 }
1764
1765 lastCorner = VECTOR2I( 0, 0 );
1766 for( int ii = 0; ii < other.GetNumCorners(); ii++ )
1767 {
1768 otherCorners.push_back( lastCorner - other.GetCornerPosition( ii ) );
1769 lastCorner = other.GetCornerPosition( ii );
1770 }
1771
1772 size_t longest = alg::longest_common_subset( corners, otherCorners );
1773
1774 similarity *= std::pow( 0.9, GetNumCorners() + other.GetNumCorners() - 2 * longest );
1775
1776 return similarity;
1777}
1778
1779
1780static struct ZONE_DESC
1781{
1783 {
1785
1786 if( layerEnum.Choices().GetCount() == 0 )
1787 {
1788 layerEnum.Undefined( UNDEFINED_LAYER );
1789
1790 for( PCB_LAYER_ID layer : LSET::AllLayersMask().Seq() )
1791 layerEnum.Map( layer, LSET::Name( layer ) );
1792 }
1793
1795
1796 if( zcMap.Choices().GetCount() == 0 )
1797 {
1798 zcMap.Undefined( ZONE_CONNECTION::INHERITED );
1799 zcMap.Map( ZONE_CONNECTION::INHERITED, _HKI( "Inherited" ) )
1800 .Map( ZONE_CONNECTION::NONE, _HKI( "None" ) )
1801 .Map( ZONE_CONNECTION::THERMAL, _HKI( "Thermal reliefs" ) )
1802 .Map( ZONE_CONNECTION::FULL, _HKI( "Solid" ) )
1803 .Map( ZONE_CONNECTION::THT_THERMAL, _HKI( "Thermal reliefs for PTH" ) );
1804 }
1805
1807
1808 if( zfmMap.Choices().GetCount() == 0 )
1809 {
1810 zfmMap.Undefined( ZONE_FILL_MODE::POLYGONS );
1811 zfmMap.Map( ZONE_FILL_MODE::POLYGONS, _HKI( "Solid fill" ) )
1812 .Map( ZONE_FILL_MODE::HATCH_PATTERN, _HKI( "Hatch pattern" ) );
1813 }
1814
1816
1817 if( irmMap.Choices().GetCount() == 0 )
1818 {
1819 irmMap.Undefined( ISLAND_REMOVAL_MODE::ALWAYS );
1820 irmMap.Map( ISLAND_REMOVAL_MODE::ALWAYS, _HKI( "Always" ) )
1821 .Map( ISLAND_REMOVAL_MODE::NEVER, _HKI( "Never" ) )
1822 .Map( ISLAND_REMOVAL_MODE::AREA, _HKI( "Below area limit" ) );
1823 }
1824
1827
1828 if( rapstMap.Choices().GetCount() == 0 )
1829 {
1830 rapstMap.Undefined( RULE_AREA_PLACEMENT_SOURCE_TYPE::SHEETNAME );
1831 rapstMap.Map( RULE_AREA_PLACEMENT_SOURCE_TYPE::SHEETNAME, _HKI( "Sheet Name" ) )
1832 .Map( RULE_AREA_PLACEMENT_SOURCE_TYPE::COMPONENT_CLASS,
1833 _HKI( "Component Class" ) );
1834 }
1835
1839
1840 // Mask layer and position properties; they aren't useful in current form
1841 auto posX = new PROPERTY<ZONE, int>( _HKI( "Position X" ), NO_SETTER( ZONE, int ),
1842 static_cast<int ( ZONE::* )() const>( &ZONE::GetX ),
1843 PROPERTY_DISPLAY::PT_COORD,
1845 posX->SetIsHiddenFromPropertiesManager();
1846
1847 auto posY = new PROPERTY<ZONE, int>( _HKI( "Position Y" ), NO_SETTER( ZONE, int ),
1848 static_cast<int ( ZONE::* )() const>( &ZONE::GetY ),
1849 PROPERTY_DISPLAY::PT_COORD,
1851 posY->SetIsHiddenFromPropertiesManager();
1852
1853 propMgr.ReplaceProperty( TYPE_HASH( BOARD_ITEM ), _HKI( "Position X" ), posX );
1854 propMgr.ReplaceProperty( TYPE_HASH( BOARD_ITEM ), _HKI( "Position Y" ), posY );
1855
1856 auto isCopperZone =
1857 []( INSPECTABLE* aItem ) -> bool
1858 {
1859 if( ZONE* zone = dynamic_cast<ZONE*>( aItem ) )
1860 return !zone->GetIsRuleArea() && IsCopperLayer( zone->GetFirstLayer() );
1861
1862 return false;
1863 };
1864
1865 auto isRuleArea =
1866 []( INSPECTABLE* aItem ) -> bool
1867 {
1868 if( ZONE* zone = dynamic_cast<ZONE*>( aItem ) )
1869 return zone->GetIsRuleArea();
1870
1871 return false;
1872 };
1873
1874 auto isHatchedFill =
1875 []( INSPECTABLE* aItem ) -> bool
1876 {
1877 if( ZONE* zone = dynamic_cast<ZONE*>( aItem ) )
1878 return zone->GetFillMode() == ZONE_FILL_MODE::HATCH_PATTERN;
1879
1880 return false;
1881 };
1882
1883 auto isAreaBasedIslandRemoval =
1884 []( INSPECTABLE* aItem ) -> bool
1885 {
1886 if( ZONE* zone = dynamic_cast<ZONE*>( aItem ) )
1887 return zone->GetIslandRemovalMode() == ISLAND_REMOVAL_MODE::AREA;
1888
1889 return false;
1890 };
1891
1892 // Layer property is hidden because it only holds a single layer and zones actually use
1893 // a layer set
1894 propMgr.ReplaceProperty( TYPE_HASH( BOARD_CONNECTED_ITEM ), _HKI( "Layer" ),
1897 &ZONE::GetLayer ) )
1898 .SetIsHiddenFromPropertiesManager();
1899
1901 _HKI( "Net" ), isCopperZone );
1903 _HKI( "Net Class" ), isCopperZone );
1904
1905 propMgr.AddProperty( new PROPERTY<ZONE, unsigned>( _HKI( "Priority" ),
1907 .SetAvailableFunc( isCopperZone );
1908
1909 propMgr.AddProperty( new PROPERTY<ZONE, wxString>( _HKI( "Name" ),
1911
1912 const wxString groupKeepout = _HKI( "Keepout" );
1913
1914 propMgr.AddProperty( new PROPERTY<ZONE, bool>( _HKI( "Keep Out Tracks" ),
1917 groupKeepout )
1918 .SetAvailableFunc( isRuleArea );
1919
1920 propMgr.AddProperty( new PROPERTY<ZONE, bool>( _HKI( "Keep Out Vias" ),
1923 groupKeepout )
1924 .SetAvailableFunc( isRuleArea );
1925
1926 propMgr.AddProperty( new PROPERTY<ZONE, bool>( _HKI( "Keep Out Pads" ),
1929 groupKeepout )
1930 .SetAvailableFunc( isRuleArea );
1931
1932 propMgr.AddProperty( new PROPERTY<ZONE, bool>( _HKI( "Keep Out Zone Fills" ),
1935 groupKeepout )
1936 .SetAvailableFunc( isRuleArea );
1937
1938 propMgr.AddProperty( new PROPERTY<ZONE, bool>( _HKI( "Keep Out Footprints" ),
1941 groupKeepout )
1942 .SetAvailableFunc( isRuleArea );
1943
1944
1945 const wxString groupPlacement = _HKI( "Placement" );
1946
1947 propMgr.AddProperty( new PROPERTY<ZONE, bool>( _HKI( "Enable" ),
1950 groupPlacement )
1951 .SetAvailableFunc( isRuleArea );
1952
1954 _HKI( "Source Type" ), &ZONE::SetRuleAreaPlacementSourceType,
1956 groupPlacement )
1957 .SetAvailableFunc( isRuleArea );
1958
1959 propMgr.AddProperty( new PROPERTY<ZONE, wxString>( _HKI( "Source Name" ),
1962 groupPlacement )
1963 .SetAvailableFunc( isRuleArea );
1964
1965
1966 const wxString groupFill = _HKI( "Fill Style" );
1967
1968 propMgr.AddProperty( new PROPERTY_ENUM<ZONE, ZONE_FILL_MODE>( _HKI( "Fill Mode" ),
1970 groupFill )
1971 .SetAvailableFunc( isCopperZone );
1972
1973 propMgr.AddProperty( new PROPERTY<ZONE, EDA_ANGLE>( _HKI( "Hatch Orientation" ),
1975 PROPERTY_DISPLAY::PT_DEGREE ),
1976 groupFill )
1977 .SetAvailableFunc( isCopperZone )
1978 .SetWriteableFunc( isHatchedFill );
1979
1980 // TODO: Switch to translated
1981 auto atLeastMinWidthValidator =
1982 []( const wxAny&& aValue, EDA_ITEM* aZone ) -> VALIDATOR_RESULT
1983 {
1984 int val = aValue.As<int>();
1985 ZONE* zone = dynamic_cast<ZONE*>( aZone );
1986 wxCHECK( zone, std::nullopt );
1987
1988 if( val < zone->GetMinThickness() )
1989 {
1990 return std::make_unique<VALIDATION_ERROR_MSG>(
1991 _( "Cannot be less than zone minimum width" ) );
1992 }
1993
1994 return std::nullopt;
1995 };
1996
1997 propMgr.AddProperty( new PROPERTY<ZONE, int>( _HKI( "Hatch Width" ),
1998 &ZONE::SetHatchThickness, &ZONE::GetHatchThickness, PROPERTY_DISPLAY::PT_SIZE ),
1999 groupFill )
2000 .SetAvailableFunc( isCopperZone )
2001 .SetWriteableFunc( isHatchedFill )
2002 .SetValidator( atLeastMinWidthValidator );
2003
2004 propMgr.AddProperty( new PROPERTY<ZONE, int>( _HKI( "Hatch Gap" ),
2005 &ZONE::SetHatchGap, &ZONE::GetHatchGap, PROPERTY_DISPLAY::PT_SIZE ),
2006 groupFill )
2007 .SetAvailableFunc( isCopperZone )
2008 .SetWriteableFunc( isHatchedFill )
2009 .SetValidator( atLeastMinWidthValidator );
2010
2011 propMgr.AddProperty( new PROPERTY<ZONE, double>( _HKI( "Hatch Minimum Hole Ratio" ),
2013 groupFill )
2014 .SetAvailableFunc( isCopperZone )
2015 .SetWriteableFunc( isHatchedFill )
2017
2018 // TODO: Smoothing effort needs to change to enum (in dialog too)
2019 propMgr.AddProperty( new PROPERTY<ZONE, int>( _HKI( "Smoothing Effort" ),
2021 groupFill )
2022 .SetAvailableFunc( isCopperZone )
2023 .SetWriteableFunc( isHatchedFill );
2024
2025 propMgr.AddProperty( new PROPERTY<ZONE, double>( _HKI( "Smoothing Amount" ),
2027 groupFill )
2028 .SetAvailableFunc( isCopperZone )
2029 .SetWriteableFunc( isHatchedFill );
2030
2031 propMgr.AddProperty( new PROPERTY_ENUM<ZONE, ISLAND_REMOVAL_MODE>( _HKI( "Remove Islands" ),
2033 groupFill )
2034 .SetAvailableFunc( isCopperZone );
2035
2036 propMgr.AddProperty( new PROPERTY<ZONE, long long int>( _HKI( "Minimum Island Area" ),
2037 &ZONE::SetMinIslandArea, &ZONE::GetMinIslandArea, PROPERTY_DISPLAY::PT_AREA ),
2038 groupFill )
2039 .SetAvailableFunc( isCopperZone )
2040 .SetWriteableFunc( isAreaBasedIslandRemoval );
2041
2042 const wxString groupElectrical = _HKI( "Electrical" );
2043
2044 auto clearanceOverride = new PROPERTY<ZONE, std::optional<int>>( _HKI( "Clearance" ),
2046 PROPERTY_DISPLAY::PT_SIZE );
2047 clearanceOverride->SetAvailableFunc( isCopperZone );
2048 constexpr int maxClearance = pcbIUScale.mmToIU( ZONE_CLEARANCE_MAX_VALUE_MM );
2049 clearanceOverride->SetValidator( PROPERTY_VALIDATORS::RangeIntValidator<0, maxClearance> );
2050
2051 auto minWidth = new PROPERTY<ZONE, int>( _HKI( "Minimum Width" ),
2053 PROPERTY_DISPLAY::PT_SIZE );
2054 minWidth->SetAvailableFunc( isCopperZone );
2055 constexpr int minMinWidth = pcbIUScale.mmToIU( ZONE_THICKNESS_MIN_VALUE_MM );
2056 minWidth->SetValidator( PROPERTY_VALIDATORS::RangeIntValidator<minMinWidth,
2057 INT_MAX> );
2058
2059 auto padConnections = new PROPERTY_ENUM<ZONE, ZONE_CONNECTION>( _HKI( "Pad Connections" ),
2061 padConnections->SetAvailableFunc( isCopperZone );
2062
2063 auto thermalGap = new PROPERTY<ZONE, int>( _HKI( "Thermal Relief Gap" ),
2065 PROPERTY_DISPLAY::PT_SIZE );
2066 thermalGap->SetAvailableFunc( isCopperZone );
2067 thermalGap->SetValidator( PROPERTY_VALIDATORS::PositiveIntValidator );
2068
2069 auto thermalSpokeWidth = new PROPERTY<ZONE, int>( _HKI( "Thermal Relief Spoke Width" ),
2071 PROPERTY_DISPLAY::PT_SIZE );
2072 thermalSpokeWidth->SetAvailableFunc( isCopperZone );
2073 thermalSpokeWidth->SetValidator( atLeastMinWidthValidator );
2074
2075 propMgr.AddProperty( clearanceOverride, groupElectrical );
2076 propMgr.AddProperty( minWidth, groupElectrical );
2077 propMgr.AddProperty( padConnections, groupElectrical );
2078 propMgr.AddProperty( thermalGap, groupElectrical );
2079 propMgr.AddProperty( thermalSpokeWidth, groupElectrical );
2080 }
2082
ERROR_LOC
When approximating an arc or circle, should the error be placed on the outside or inside of the curve...
Definition: approximation.h:32
@ ERROR_OUTSIDE
Definition: approximation.h:33
constexpr int ARC_HIGH_DEF
Definition: base_units.h:120
constexpr EDA_IU_SCALE pcbIUScale
Definition: base_units.h:108
BITMAPS
A list of all bitmap identifiers.
Definition: bitmaps_list.h:33
BASE_SET & reset(size_t pos)
Definition: base_set.h:143
BASE_SET & set(size_t pos)
Definition: base_set.h:116
A base class derived from BOARD_ITEM for items that can be connected and have a net,...
wxString GetNetnameMsg() const
virtual NETCLASS * GetEffectiveNetClass() const
Return the NETCLASS for this item.
bool SetNetCode(int aNetCode, bool aNoAssert)
Set net using a net code.
NETINFO_ITEM * m_netinfo
Store all information about the net that item belongs to.
virtual int GetOwnClearance(PCB_LAYER_ID aLayer, wxString *aSource=nullptr) const
Return an item's "own" clearance in internal units.
TEARDROP_PARAMETERS & GetTeardropParams()
Container for design settings for a BOARD object.
ZONE_SETTINGS & GetDefaultZoneSettings()
Abstract interface for BOARD_ITEMs capable of storing other items inside.
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition: board_item.h:78
int GetY() const
Definition: board_item.h:98
int GetX() const
Definition: board_item.h:92
virtual const BOARD * GetBoard() const
Return the BOARD in which this BOARD_ITEM resides, or NULL if none.
Definition: board_item.cpp:48
FOOTPRINT * GetParentFootprint() const
Definition: board_item.cpp:299
virtual bool IsLocked() const
Definition: board_item.cpp:76
wxString GetLayerName() const
Return the name of the PCB layer on which the item resides.
Definition: board_item.cpp:140
Information pertinent to a Pcbnew printed circuit board.
Definition: board.h:297
PCB_LAYER_ID FlipLayer(PCB_LAYER_ID aLayer) const
Definition: board.cpp:774
std::unordered_map< const ZONE *, BOX2I > m_ZoneBBoxCache
Definition: board.h:1343
BOARD_DESIGN_SETTINGS & GetDesignSettings() const
Definition: board.cpp:946
constexpr BOX2< Vec > & Inflate(coord_type dx, coord_type dy)
Inflates the rectangle horizontally by dx and vertically by dy.
Definition: box2.h:558
constexpr BOX2< Vec > & Normalize()
Ensure that the height and width are positive.
Definition: box2.h:146
constexpr bool Contains(const Vec &aPoint) const
Definition: box2.h:168
constexpr bool Intersects(const BOX2< Vec > &aRect) const
Definition: box2.h:311
double AsDegrees() const
Definition: eda_angle.h:113
The base class for create windows for drawing purpose.
A base class for most all the KiCad significant classes used in schematics and boards.
Definition: eda_item.h:96
EDA_ITEM & operator=(const EDA_ITEM &aItem)
Assign the members of aItem to another object.
Definition: eda_item.cpp:303
const KIID m_Uuid
Definition: eda_item.h:498
bool m_forceVisible
Definition: eda_item.h:512
KICAD_T Type() const
Returns the type of object.
Definition: eda_item.h:108
EDA_ITEM_FLAGS m_flags
Definition: eda_item.h:509
bool HasFlag(EDA_ITEM_FLAGS aFlag) const
Definition: eda_item.h:139
ENUM_MAP & Map(T aValue, const wxString &aName)
Definition: property.h:686
static ENUM_MAP< T > & Instance()
Definition: property.h:680
ENUM_MAP & Undefined(T aValue)
Definition: property.h:693
wxPGChoices & Choices()
Definition: property.h:729
Class that other classes need to inherit from, in order to be inspectable.
Definition: inspectable.h:37
static constexpr double LOD_HIDE
Return this constant from ViewGetLOD() to hide the item unconditionally.
Definition: view_item.h:174
static constexpr double LOD_SHOW
Return this constant from ViewGetLOD() to show the item unconditionally.
Definition: view_item.h:179
Hold a (potentially large) number of VIEW_ITEMs and renders them on a graphics device provided by the...
Definition: view.h:67
bool IsLayerVisible(int aLayer) const
Return information about visibility of a particular layer.
Definition: view.h:418
Definition: kiid.h:49
std::string AsStdString() const
Definition: kiid.cpp:252
LSEQ is a sequence (and therefore also a set) of PCB_LAYER_IDs.
Definition: lseq.h:47
LSET is a set of PCB_LAYER_IDs.
Definition: lset.h:37
LSEQ UIOrder() const
Return the copper, technical and user layers in the order shown in layer widget.
Definition: lset.cpp:712
static LSET AllLayersMask()
Definition: lset.cpp:601
void RunOnLayers(const std::function< void(PCB_LAYER_ID)> &aFunction) const
Execute a function on each layer of the LSET.
Definition: lset.h:255
static LSET AllCuMask(int aCuLayerCount=MAX_CU_LAYERS)
Return a mask holding the requested number of Cu PCB_LAYER_IDs.
Definition: lset.cpp:572
LSEQ Seq(const LSEQ &aSequence) const
Return an LSEQ from the union of this LSET and a desired sequence.
Definition: lset.cpp:297
static wxString Name(PCB_LAYER_ID aLayerId)
Return the fixed name association with aLayerId.
Definition: lset.cpp:188
bool Contains(PCB_LAYER_ID aLayer) const
See if the layer set contains a PCB layer.
Definition: lset.h:63
Definition: pad.h:54
int GetLocalThermalGapOverride(wxString *aSource) const
Definition: pad.cpp:1290
PROPERTY_BASE & SetAvailableFunc(std::function< bool(INSPECTABLE *)> aFunc)
Set a callback function to determine whether an object provides this property.
Definition: property.h:257
PROPERTY_BASE & SetWriteableFunc(std::function< bool(INSPECTABLE *)> aFunc)
Definition: property.h:268
PROPERTY_BASE & SetValidator(PROPERTY_VALIDATOR_FN &&aValidator)
Definition: property.h:330
Provide class metadata.Helper macro to map type hashes to names.
Definition: property_mgr.h:85
void InheritsAfter(TYPE_ID aDerived, TYPE_ID aBase)
Declare an inheritance relationship between types.
static PROPERTY_MANAGER & Instance()
Definition: property_mgr.h:87
PROPERTY_BASE & AddProperty(PROPERTY_BASE *aProperty, const wxString &aGroup=wxEmptyString)
Register a property.
void OverrideAvailability(TYPE_ID aDerived, TYPE_ID aBase, const wxString &aName, std::function< bool(INSPECTABLE *)> aFunc)
Sets an override availability functor for a base class property of a given derived class.
PROPERTY_BASE & ReplaceProperty(size_t aBase, const wxString &aName, PROPERTY_BASE *aNew, const wxString &aGroup=wxEmptyString)
Replace an existing property for a specific type.
static VALIDATOR_RESULT PositiveRatioValidator(const wxAny &&aValue, EDA_ITEM *aItem)
static VALIDATOR_RESULT PositiveIntValidator(const wxAny &&aValue, EDA_ITEM *aItem)
static VALIDATOR_RESULT RangeIntValidator(const wxAny &&aValue, EDA_ITEM *aItem)
bool PointInside(const VECTOR2I &aPt, int aAccuracy=0, bool aUseBBoxCache=false) const override
Check if point aP lies inside a closed shape.
Represent a polyline containing arcs as well as line segments: A chain of connected line and/or arc s...
bool IsClosed() const override
void SetClosed(bool aClosed)
Mark the line chain as closed (i.e.
void Append(int aX, int aY, bool aAllowDuplication=false)
Append a new point at the end of the line chain.
Represent a set of closed polygons.
void Rotate(const EDA_ANGLE &aAngle, const VECTOR2I &aCenter={ 0, 0 }) override
Rotate all vertices by a given angle.
bool CollideEdge(const VECTOR2I &aPoint, VERTEX_INDEX *aClosestVertex=nullptr, int aClearance=0) const
Check whether aPoint collides with any edge of any of the contours of the polygon.
HASH_128 GetHash() const
void BooleanAdd(const SHAPE_POLY_SET &b)
Perform boolean polyset union.
void ClearArcs()
Removes all arc references from all the outlines and holes in the polyset.
int AddOutline(const SHAPE_LINE_CHAIN &aOutline)
Adds a new outline to the set and returns its index.
virtual void CacheTriangulation(bool aPartition=true, bool aSimplify=false)
Build a polygon triangulation, needed to draw a polygon on OpenGL and in some other calculations.
double Area()
Return the area of this poly set.
void SetVertex(const VERTEX_INDEX &aIndex, const VECTOR2I &aPos)
Accessor function to set the position of a specific point.
bool IsEmpty() const
Return true if the set is empty (no polygons at all)
void Fracture()
Convert a set of polygons with holes to a single outline with "slits"/"fractures" connecting the oute...
bool Collide(const SHAPE *aShape, int aClearance=0, int *aActual=nullptr, VECTOR2I *aLocation=nullptr) const override
Check if the boundary of shape (this) lies closer to the shape aShape than aClearance,...
int TotalVertices() const
Return total number of vertices stored in the set.
void Inflate(int aAmount, CORNER_STRATEGY aCornerStrategy, int aMaxError, bool aSimplify=false)
Perform outline inflation/deflation.
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)
const std::vector< SEG > GenerateHatchLines(const std::vector< double > &aSlopes, int aSpacing, int aLineLength) const
int AddHole(const SHAPE_LINE_CHAIN &aHole, int aOutline=-1)
Adds a new hole to the given outline (default: last) and returns its index.
bool GetNeighbourIndexes(int aGlobalIndex, int *aPrevious, int *aNext) const
Return the global indexes of the previous and the next corner of the aGlobalIndex-th corner of a cont...
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 NewOutline()
Creates a new empty polygon in the set and returns its index.
bool CollideVertex(const VECTOR2I &aPoint, VERTEX_INDEX *aClosestVertex=nullptr, int aClearance=0) const
Check whether aPoint collides with any vertex of any of the contours of the polygon.
void BooleanIntersection(const SHAPE_POLY_SET &b)
Perform boolean polyset intersection.
void Mirror(const VECTOR2I &aRef, FLIP_DIRECTION aFlipDirection)
Mirror the line points about y or x (or both)
const VECTOR2I & CVertex(int aIndex, int aOutline, int aHole) const
Return the index-th vertex in a given hole outline within a given outline.
int OutlineCount() const
Return the number of outlines in the set.
void InflateWithLinkedHoles(int aFactor, CORNER_STRATEGY aCornerStrategy, int aMaxError)
Perform outline inflation/deflation, using round corners.
void Move(const VECTOR2I &aVector) override
bool Contains(const VECTOR2I &aP, int aSubpolyIndex=-1, int aAccuracy=0, bool aUseBBoxCaches=false) const
Return true if a given subpolygon contains the point aP.
SHAPE_POLY_SET CloneDropTriangulation() const
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().
ZONE_SETTINGS handles zones parameters.
Definition: zone_settings.h:87
void ExportSetting(ZONE &aTarget, bool aFullExport=true) const
Function ExportSetting copy settings to a given zone.
Handle a list of polygons defining a copper zone.
Definition: zone.h:74
void SetHatchThickness(int aThickness)
Definition: zone.h:317
SHAPE_POLY_SET::VERTEX_INDEX * m_CornerSelection
The index of the corner being moved or nullptr if no corner is selected.
Definition: zone.h:954
void SetNeedRefill(bool aNeedRefill)
Definition: zone.h:296
bool HitTest(const VECTOR2I &aPosition, int aAccuracy=0) const override
Test if a point is near an outline edge or a corner of this zone.
Definition: zone.cpp:730
wxString GetItemDescription(UNITS_PROVIDER *aUnitsProvider, bool aFull) const override
Return a user-visible description string of this item.
Definition: zone.cpp:1148
int m_borderHatchPitch
Definition: zone.h:975
bool m_isRuleArea
Definition: zone.h:883
void SetDoNotAllowPads(bool aEnable)
Definition: zone.h:777
void SetLayerProperties(const std::map< PCB_LAYER_ID, ZONE_LAYER_PROPERTIES > &aOther)
Definition: zone.cpp:578
ZONE & operator=(const ZONE &aOther)
Definition: zone.cpp:106
int m_cornerSmoothingType
Definition: zone.h:864
bool GetIsRuleArea() const
Accessors to parameters used in Rule Area zones:
Definition: zone.h:752
std::optional< int > GetLocalClearance() const override
Definition: zone.cpp:795
void SetRuleAreaPlacementEnabled(bool aEnabled)
Definition: zone.h:755
void SetLocalClearance(std::optional< int > aClearance)
Definition: zone.h:186
bool m_doNotAllowVias
Definition: zone.h:902
bool UnFill()
Removes the zone filling.
Definition: zone.cpp:459
wxString GetRuleAreaPlacementSource() const
Definition: zone.h:757
bool GetDoNotAllowVias() const
Definition: zone.h:769
void TransformSolidAreasShapesToPolygon(PCB_LAYER_ID aLayer, SHAPE_POLY_SET &aBuffer) const
Convert solid areas full shapes to polygon set (the full shape is the polygon area with a thick outli...
Definition: zone.cpp:1597
void SetCornerRadius(unsigned int aRadius)
Definition: zone.cpp:700
ZONE_FILL_MODE m_fillMode
How to fill areas:
Definition: zone.h:940
bool m_doNotAllowFootprints
Definition: zone.h:905
int m_ZoneMinThickness
Definition: zone.h:909
void AddPolygon(std::vector< VECTOR2I > &aPolygon)
Add a polygon to the zone outline.
Definition: zone.cpp:1111
ZONE_LAYER_PROPERTIES & LayerProperties(PCB_LAYER_ID aLayer)
Definition: zone.h:145
double m_hatchSmoothingValue
Definition: zone.h:948
void SetLocalFlags(int aFlags)
Definition: zone.h:366
void TransformSmoothedOutlineToPolygon(SHAPE_POLY_SET &aBuffer, int aClearance, int aError, ERROR_LOC aErrorLoc, SHAPE_POLY_SET *aBoardOutline) const
Convert the outlines shape to a polygon with no holes inflated (optional) by max( aClearanceValue,...
Definition: zone.cpp:1528
const std::optional< VECTOR2I > & HatchingOffset(PCB_LAYER_ID aLayer) const
Definition: zone.cpp:586
int m_thermalReliefSpokeWidth
Definition: zone.h:931
bool HitTestCutout(const VECTOR2I &aRefPos, int *aOutlineIdx=nullptr, int *aHoleIdx=nullptr) const
Test if the given point is contained within a cutout of the zone.
Definition: zone.cpp:815
EDA_ANGLE m_hatchOrientation
Definition: zone.h:943
void CacheTriangulation(PCB_LAYER_ID aLayer=UNDEFINED_LAYER)
Create a list of triangles that "fill" the solid areas used for instance to draw these solid areas on...
Definition: zone.cpp:1288
void Mirror(const VECTOR2I &aMirrorRef, FLIP_DIRECTION aFlipDirection) override
Mirror the outlines relative to a given horizontal axis the layer is not changed.
Definition: zone.cpp:1071
std::map< PCB_LAYER_ID, std::set< int > > m_insulatedIslands
For each layer, a set of insulated islands that were not removed.
Definition: zone.h:979
wxString m_zoneName
An optional unique name for this zone, used for identifying it in DRC checking.
Definition: zone.h:868
void HatchBorder()
Compute the hatch lines depending on the hatch parameters and stores it in the zone's attribute m_bor...
Definition: zone.cpp:1231
std::map< PCB_LAYER_ID, std::shared_ptr< SHAPE_POLY_SET > > m_FilledPolysList
Definition: zone.h:966
void SetBorderDisplayStyle(ZONE_BORDER_DISPLAY_STYLE aBorderHatchStyle, int aBorderHatchPitch, bool aRebuilBorderdHatch)
Set all hatch parameters for the zone.
Definition: zone.cpp:1204
bool GetDoNotAllowPads() const
Definition: zone.h:771
const BOX2I GetBoundingBox() const override
Definition: zone.cpp:648
void SetMinThickness(int aMinThickness)
Definition: zone.h:302
double ViewGetLOD(int aLayer, const KIGFX::VIEW *aView) const override
Return the level of detail (LOD) of the item.
Definition: zone.cpp:617
double m_outlinearea
Definition: zone.h:982
wxString GetFriendlyName() const override
Definition: zone.cpp:995
bool GetDoNotAllowTracks() const
Definition: zone.h:770
int GetLocalFlags() const
Definition: zone.h:365
EDA_ITEM * Clone() const override
Create a duplicate of this item with linked list members set to NULL.
Definition: zone.cpp:220
void SetHatchOrientation(const EDA_ANGLE &aStep)
Definition: zone.h:323
bool GetRuleAreaPlacementEnabled() const
Definition: zone.h:754
void SetHatchSmoothingValue(double aValue)
Definition: zone.h:329
bool m_needRefill
False when a zone was refilled, true after changes in zone params.
Definition: zone.h:928
std::map< PCB_LAYER_ID, ZONE_LAYER_PROPERTIES > m_layerProperties
Definition: zone.h:872
bool HitTestForCorner(const VECTOR2I &refPos, int aAccuracy, SHAPE_POLY_SET::VERTEX_INDEX *aCornerHit=nullptr) const
Test if the given VECTOR2I is near a corner.
Definition: zone.cpp:740
void SetHatchSmoothingLevel(int aLevel)
Definition: zone.h:326
bool m_doNotAllowTracks
Definition: zone.h:903
RULE_AREA_PLACEMENT_SOURCE_TYPE m_ruleAreaPlacementSourceType
Definition: zone.h:889
void SetThermalReliefSpokeWidth(int aThermalReliefSpokeWidth)
Definition: zone.h:237
virtual PCB_LAYER_ID GetLayer() const override
Return the primary layer this item is on.
Definition: zone.cpp:489
virtual void SetLayer(PCB_LAYER_ID aLayer) override
Set the layer this item is on.
Definition: zone.cpp:527
ISLAND_REMOVAL_MODE GetIslandRemovalMode() const
Definition: zone.h:780
LSET m_fillFlags
Temp variables used while filling.
Definition: zone.h:969
SHAPE_POLY_SET * Outline()
Definition: zone.h:368
bool m_doNotAllowPads
Definition: zone.h:904
ZONE(BOARD_ITEM_CONTAINER *parent)
Definition: zone.cpp:54
void Serialize(google::protobuf::Any &aContainer) const override
Serializes this object to the given Any message.
Definition: zone.cpp:226
void Move(const VECTOR2I &offset) override
Move the outlines.
Definition: zone.cpp:968
bool IsIsland(PCB_LAYER_ID aLayer, int aPolyIdx) const
Check if a given filled polygon is an insulated island.
Definition: zone.cpp:1305
SHAPE_POLY_SET * m_Poly
Outline of the zone.
Definition: zone.h:863
TEARDROP_TYPE m_teardropType
Definition: zone.h:896
std::map< PCB_LAYER_ID, HASH_128 > m_filledPolysHash
A hash value used in zone filling calculations to see if the filled areas are up to date.
Definition: zone.h:972
~ZONE()
Definition: zone.cpp:123
long long int GetMinIslandArea() const
Definition: zone.h:783
int m_hatchSmoothingLevel
Definition: zone.h:944
double Similarity(const BOARD_ITEM &aOther) const override
Return a measure of how likely the other object is to represent the same object.
Definition: zone.cpp:1706
LSET m_layerSet
Definition: zone.h:870
void SetRuleAreaPlacementSourceType(RULE_AREA_PLACEMENT_SOURCE_TYPE aType)
Definition: zone.h:763
void SetIsRuleArea(bool aEnable)
Definition: zone.h:753
std::map< PCB_LAYER_ID, ZONE_LAYER_PROPERTIES > & LayerProperties()
Definition: zone.h:152
int m_ZoneClearance
Definition: zone.h:908
void CopyFrom(const BOARD_ITEM *aOther) override
Definition: zone.cpp:116
void SetDoNotAllowTracks(bool aEnable)
Definition: zone.h:776
void SetFilledPolysList(PCB_LAYER_ID aLayer, const SHAPE_POLY_SET &aPolysList)
Set the list of filled polygons.
Definition: zone.h:669
const wxString & GetZoneName() const
Definition: zone.h:163
void CacheBoundingBox()
Used to preload the zone bounding box cache so we don't have to worry about mutex-locking it each tim...
Definition: zone.cpp:677
int GetMinThickness() const
Definition: zone.h:301
virtual void swapData(BOARD_ITEM *aImage) override
Definition: zone.cpp:1280
bool HitTestForEdge(const VECTOR2I &refPos, int aAccuracy, SHAPE_POLY_SET::VERTEX_INDEX *aCornerHit=nullptr) const
Test if the given VECTOR2I is near a segment defined by 2 corners.
Definition: zone.cpp:747
void RemoveCutout(int aOutlineIdx, int aHoleIdx)
Remove a cutout from the zone.
Definition: zone.cpp:1082
void Rotate(const VECTOR2I &aCentre, const EDA_ANGLE &aAngle) override
Rotate the outlines.
Definition: zone.cpp:1023
bool HigherPriority(const ZONE *aOther) const
Definition: zone.cpp:436
bool HitTestFilledArea(PCB_LAYER_ID aLayer, const VECTOR2I &aRefPos, int aAccuracy=0) const
Test if the given VECTOR2I is within the bounds of a filled area of this zone.
Definition: zone.cpp:801
void SetIsFilled(bool isFilled)
Definition: zone.h:293
void SetFillMode(ZONE_FILL_MODE aFillMode)
Definition: zone.h:223
BITMAPS GetMenuImage() const override
Return a pointer to an image to be used in menus.
Definition: zone.cpp:1274
int m_hatchGap
Definition: zone.h:942
ZONE_CONNECTION GetPadConnection() const
Definition: zone.h:298
void InitDataFromSrcInCopyCtor(const ZONE &aZone)
Copy aZone data to me.
Definition: zone.cpp:133
int GetHatchThickness() const
Definition: zone.h:316
double GetHatchHoleMinArea() const
Definition: zone.h:331
void SetLayerSet(const LSET &aLayerSet) override
Definition: zone.cpp:533
void GetMsgPanelInfo(EDA_DRAW_FRAME *aFrame, std::vector< MSG_PANEL_ITEM > &aList) override
Populate aList of MSG_PANEL_ITEM objects with it's internal state for display purposes.
Definition: zone.cpp:840
virtual bool IsOnLayer(PCB_LAYER_ID) const override
Test to see if this object is on the given layer.
Definition: zone.cpp:642
int m_hatchBorderAlgorithm
Definition: zone.h:950
void SetDoNotAllowVias(bool aEnable)
Definition: zone.h:775
bool IsTeardropArea() const
Definition: zone.h:727
std::vector< SEG > m_borderHatchLines
Definition: zone.h:976
wxString m_ruleAreaPlacementSource
Definition: zone.h:890
VECTOR2I GetPosition() const override
Definition: zone.cpp:483
int GetThermalReliefSpokeWidth() const
Definition: zone.h:245
int GetBorderHatchPitch() const
HatchBorder related methods.
Definition: zone.cpp:1198
virtual void Flip(const VECTOR2I &aCentre, FLIP_DIRECTION aFlipDirection) override
Flip this object, i.e.
Definition: zone.cpp:1034
void BuildHashValue(PCB_LAYER_ID aLayer)
Build the hash value of m_FilledPolysList, and store it internally in m_filledPolysHash.
Definition: zone.cpp:721
void SetThermalReliefGap(int aThermalReliefGap)
Definition: zone.h:226
EDA_ANGLE GetHatchOrientation() const
Definition: zone.h:322
bool BuildSmoothedPoly(SHAPE_POLY_SET &aSmoothedPoly, PCB_LAYER_ID aLayer, SHAPE_POLY_SET *aBoardOutline, SHAPE_POLY_SET *aSmoothedPolyWithApron=nullptr) const
Definition: zone.cpp:1352
int m_fillVersion
Definition: zone.h:910
const VECTOR2I & GetCornerPosition(int aCornerIndex) const
Definition: zone.h:587
bool GetDoNotAllowFootprints() const
Definition: zone.h:772
ZONE_FILL_MODE GetFillMode() const
Definition: zone.h:224
double m_hatchHoleMinArea
Definition: zone.h:949
virtual LSET GetLayerSet() const override
Return a std::bitset of all layers on which the item physically resides.
Definition: zone.h:136
void SetDoNotAllowFootprints(bool aEnable)
Definition: zone.h:778
void SetBorderHatchPitch(int aPitch)
Set the hatch pitch parameter for the zone.
Definition: zone.cpp:1219
void GetInteractingZones(PCB_LAYER_ID aLayer, std::vector< ZONE * > *aSameNetCollidingZones, std::vector< ZONE * > *aOtherNetIntersectingZones) const
Some intersecting zones, despite being on the same layer with the same net, cannot be merged due to o...
Definition: zone.cpp:1317
bool m_ruleAreaPlacementEnabled
Placement rule area data.
Definition: zone.h:888
void SetLayerSetAndRemoveUnusedFills(const LSET &aLayerSet)
Set the zone to be on the aLayerSet layers and only remove the fill polygons from the unused layers,...
Definition: zone.cpp:1604
int GetHatchGap() const
Definition: zone.h:319
double CalculateOutlineArea()
Compute the area of the zone outline (not the filled area).
Definition: zone.cpp:1521
RULE_AREA_PLACEMENT_SOURCE_TYPE GetRuleAreaPlacementSourceType() const
Definition: zone.h:759
void SetHatchHoleMinArea(double aPct)
Definition: zone.h:332
unsigned m_priority
Definition: zone.h:878
bool Deserialize(const google::protobuf::Any &aContainer) override
Deserializes the given protobuf message into this object.
Definition: zone.cpp:330
bool IsConflicting() const
For rule areas which exclude footprints (and therefore participate in courtyard conflicts during move...
Definition: zone.cpp:477
bool m_doNotAllowZoneFills
Definition: zone.h:901
ISLAND_REMOVAL_MODE m_islandRemovalMode
Definition: zone.h:912
bool m_isFilled
True when a zone was filled, false after deleting the filled areas.
Definition: zone.h:921
double GetHatchSmoothingValue() const
Definition: zone.h:328
bool AppendCorner(VECTOR2I aPosition, int aHoleIdx, bool aAllowDuplication=false)
Add a new corner to the zone outline (to the main outline or a hole)
Definition: zone.cpp:1128
bool GetDoNotAllowZoneFills() const
Definition: zone.h:768
void MoveEdge(const VECTOR2I &offset, int aEdge)
Move the outline Edge.
Definition: zone.cpp:1008
int GetHatchSmoothingLevel() const
Definition: zone.h:325
unsigned int GetCornerRadius() const
Definition: zone.h:698
int GetCornerSmoothingType() const
Definition: zone.h:694
int m_thermalReliefGap
Definition: zone.h:930
bool IsOnCopperLayer() const override
Definition: zone.cpp:521
double CalculateFilledArea()
Compute the area currently occupied by the zone fill.
Definition: zone.cpp:1510
void SetDoNotAllowZoneFills(bool aEnable)
Definition: zone.h:774
int m_hatchThickness
Definition: zone.h:941
void TransformShapeToPolygon(SHAPE_POLY_SET &aBuffer, PCB_LAYER_ID aLayer, int aClearance, int aError, ERROR_LOC aErrorLoc, bool ignoreLineWidth=false) const override
Convert the zone shape to a closed polygon Used in filling zones calculations Circles and arcs are ap...
Definition: zone.cpp:1568
void SetAssignedPriority(unsigned aPriority)
Definition: zone.h:121
double m_area
Definition: zone.h:981
unsigned int m_cornerRadius
Definition: zone.h:865
void SetPadConnection(ZONE_CONNECTION aPadConnection)
Definition: zone.h:299
void SetZoneName(const wxString &aName)
Definition: zone.h:164
bool operator==(const ZONE &aOther) const
Definition: zone.cpp:1638
void UnHatchBorder()
Clear the zone's hatch.
Definition: zone.cpp:1225
void SetIslandRemovalMode(ISLAND_REMOVAL_MODE aRemove)
Definition: zone.h:781
void SetOutline(SHAPE_POLY_SET *aOutline)
Definition: zone.h:371
PCB_LAYER_ID GetFirstLayer() const
Definition: zone.cpp:499
void SetMinIslandArea(long long int aArea)
Definition: zone.h:784
virtual std::vector< int > ViewGetLayers() const override
Return the all the layers within the VIEW the object is painted on.
Definition: zone.cpp:598
HASH_128 GetHashValue(PCB_LAYER_ID aLayer)
Definition: zone.cpp:712
ZONE_CONNECTION m_PadConnection
Definition: zone.h:907
void SetRuleAreaPlacementSource(const wxString &aSource)
Definition: zone.h:758
int GetThermalReliefGap() const
Definition: zone.h:234
void SetHatchGap(int aStep)
Definition: zone.h:320
static int GetDefaultHatchPitch()
Definition: zone.cpp:1268
unsigned GetAssignedPriority() const
Definition: zone.h:126
int GetNumCorners(void) const
Access to m_Poly parameters.
Definition: zone.h:547
bool SameNet(const ZONE *aOther) const
Definition: zone.cpp:453
virtual std::shared_ptr< SHAPE > GetEffectiveShape(PCB_LAYER_ID aLayer=UNDEFINED_LAYER, FLASHING aFlash=FLASHING::DEFAULT) const override
Some pad shapes can be complex (rounded/chamfered rectangle), even without considering custom shapes.
Definition: zone.cpp:1559
ZONE_BORDER_DISPLAY_STYLE m_borderStyle
Definition: zone.h:974
long long int m_minIslandArea
When island removal mode is set to AREA, islands below this area will be removed.
Definition: zone.h:918
#define _HKI(x)
#define _(s)
static constexpr EDA_ANGLE ANGLE_0
Definition: eda_angle.h:401
@ DEGREES_T
Definition: eda_angle.h:31
#define PCB_EDIT_FRAME_NAME
#define COURTYARD_CONFLICT
temporary set when moving footprints having courtyard overlapping
@ NONE
Definition: eda_shape.h:69
a few functions useful in geometry calculations.
Some functions to handle hotkeys in KiCad.
PCB_LAYER_ID FlipLayer(PCB_LAYER_ID aLayerId, int aCopperLayersCount)
Definition: layer_id.cpp:169
FLASHING
Enum used during connectivity building to ensure we do not query connectivity while building the data...
Definition: layer_ids.h:184
bool IsCopperLayer(int aLayerId)
Test whether a layer is a copper layer.
Definition: layer_ids.h:663
@ LAYER_CONFLICTS_SHADOW
Shadow layer for items flagged conflicting.
Definition: layer_ids.h:309
@ LAYER_FOOTPRINTS_FR
Show footprints on front.
Definition: layer_ids.h:258
@ LAYER_ZONES
Control for copper zone opacity/visibility (color ignored).
Definition: layer_ids.h:294
@ LAYER_ZONE_START
Virtual layers for stacking zones and tracks on a given copper layer.
Definition: layer_ids.h:327
@ LAYER_FOOTPRINTS_BK
Show footprints on back.
Definition: layer_ids.h:259
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ B_Cu
Definition: layer_ids.h:65
@ UNDEFINED_LAYER
Definition: layer_ids.h:61
FLIP_DIRECTION
Definition: mirror.h:27
size_t longest_common_subset(const _Container &__c1, const _Container &__c2)
Returns the length of the longest common subset of values between two containers.
Definition: kicad_algo.h:208
KICOMMON_API VECTOR2I UnpackVector2(const types::Vector2 &aInput)
Definition: api_utils.cpp:84
KICOMMON_API SHAPE_POLY_SET UnpackPolySet(const types::PolySet &aInput)
Definition: api_utils.cpp:196
KICOMMON_API void PackVector2(types::Vector2 &aOutput, const VECTOR2I &aInput)
Definition: api_utils.cpp:77
KICOMMON_API void PackPolySet(types::PolySet &aOutput, const SHAPE_POLY_SET &aInput)
Definition: api_utils.cpp:172
EDA_ANGLE abs(const EDA_ANGLE &aAngle)
Definition: eda_angle.h:390
#define TYPE_HASH(x)
Definition: property.h:71
#define IMPLEMENT_ENUM_TO_WXANY(type)
Definition: property.h:780
#define NO_SETTER(owner, type)
Definition: property.h:791
#define REGISTER_TYPE(x)
Definition: property_mgr.h:371
std::optional< std::unique_ptr< VALIDATION_ERROR > > VALIDATOR_RESULT
Null optional means validation succeeded.
const double epsilon
wxString UnescapeString(const wxString &aSource)
void AccumulateDescription(wxString &aDesc, const wxString &aItem)
Utility to build comma separated lists in messages.
Definition: string_utils.h:348
constexpr int mmToIU(double mm) const
Definition: base_units.h:88
A storage class for 128-bit hash value.
Definition: hash_128.h:36
Structure to hold the necessary information in order to index a vertex on a SHAPE_POLY_SET object: th...
ZONE_DESC()
Definition: zone.cpp:1782
std::optional< VECTOR2I > hatching_offset
Definition: zone_settings.h:51
TEARDROP_TYPE
define the type of a teardrop: on a via or pad, or a track end
int clearance
const int accuracy
@ PCB_ZONE_T
class ZONE, a copper pour area
Definition: typeinfo.h:107
VECTOR2< int32_t > VECTOR2I
Definition: vector2d.h:695
static SHAPE_POLY_SET g_nullPoly
Definition: zone.cpp:709
static struct ZONE_DESC _ZONE_DESC
RULE_AREA_PLACEMENT_SOURCE_TYPE
Definition: zone_settings.h:75
ISLAND_REMOVAL_MODE
Whether or not to remove isolated islands from a zone.
Definition: zone_settings.h:68
ZONE_FILL_MODE
Definition: zone_settings.h:44
ZONE_BORDER_DISPLAY_STYLE
Zone border styles.
Definition: zone_settings.h:59
#define ZONE_CLEARANCE_MAX_VALUE_MM
Definition: zones.h:38
ZONE_CONNECTION
How pads are covered by copper in zone.
Definition: zones.h:47
#define ZONE_BORDER_HATCH_DIST_MM
Definition: zones.h:39
#define ZONE_BORDER_HATCH_MINDIST_MM
Definition: zones.h:40
#define ZONE_THICKNESS_MIN_VALUE_MM
Definition: zones.h:36
#define ZONE_BORDER_HATCH_MAXDIST_MM
Definition: zones.h:41