KiCad PCB EDA Suite
Loading...
Searching...
No Matches
pcbexpr_evaluator.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) 2019-2022 KiCad Developers, see AUTHORS.txt for contributors.
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, you may find one here:
18 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
19 * or you may search the http://www.gnu.org website for the version 2 license,
20 * or you may write to the Free Software Foundation, Inc.,
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
22 */
23
24
25#include <cstdio>
26#include <memory>
27#include <mutex>
28#include <board.h>
29#include <footprint.h>
30#include <lset.h>
32#include <pcbexpr_evaluator.h>
33#include <drc/drc_engine.h>
34
35/* --------------------------------------------------------------------------------------------
36 * Specialized Expression References
37 */
38
40{
41 wxASSERT( dynamic_cast<const PCBEXPR_CONTEXT*>( aCtx ) );
42
43 const PCBEXPR_CONTEXT* ctx = static_cast<const PCBEXPR_CONTEXT*>( aCtx );
44 BOARD_ITEM* item = ctx->GetItem( m_itemIndex );
45 return item;
46}
47
48
50{
51public:
53 LIBEVAL::VALUE( LayerName( aLayer ) ),
54 m_layer( aLayer )
55 {};
56
57 virtual bool EqualTo( LIBEVAL::CONTEXT* aCtx, const VALUE* b ) const override
58 {
59 // For boards with user-defined layer names there will be 2 entries for each layer
60 // in the ENUM_MAP: one for the canonical layer name and one for the user layer name.
61 // We need to check against both.
62
63 wxPGChoices& layerMap = ENUM_MAP<PCB_LAYER_ID>::Instance().Choices();
64 const wxString& layerName = b->AsString();
65 BOARD* board = static_cast<PCBEXPR_CONTEXT*>( aCtx )->GetBoard();
66
67 {
68 std::shared_lock<std::shared_mutex> readLock( board->m_CachesMutex );
69
70 auto i = board->m_LayerExpressionCache.find( layerName );
71
72 if( i != board->m_LayerExpressionCache.end() )
73 return i->second.Contains( m_layer );
74 }
75
76 LSET mask;
77
78 for( unsigned ii = 0; ii < layerMap.GetCount(); ++ii )
79 {
80 wxPGChoiceEntry& entry = layerMap[ii];
81
82 if( entry.GetText().Matches( layerName ) )
83 mask.set( ToLAYER_ID( entry.GetValue() ) );
84 }
85
86 {
87 std::unique_lock<std::shared_mutex> writeLock( board->m_CachesMutex );
88 board->m_LayerExpressionCache[ layerName ] = mask;
89 }
90
91 return mask.Contains( m_layer );
92 }
93
94protected:
96};
97
98
100{
101public:
102 PCBEXPR_PINTYPE_VALUE( const wxString& aPinTypeName ) :
103 LIBEVAL::VALUE( aPinTypeName )
104 {};
105
106 bool EqualTo( LIBEVAL::CONTEXT* aCtx, const VALUE* b ) const override
107 {
108 const wxString& thisStr = AsString();
109 const wxString& otherStr = b->AsString();
110
111 if( thisStr.IsSameAs( otherStr, false ) )
112 return true;
113
114 // Handle cases where the netlist token is different from the EEschema token
115 if( thisStr == wxT( "tri_state" ) )
116 return otherStr.IsSameAs( wxT( "Tri-state" ), false );
117
118 if( thisStr == wxT( "power_in" ) )
119 return otherStr.IsSameAs( wxT( "Power input" ), false );
120
121 if( thisStr == wxT( "power_out" ) )
122 return otherStr.IsSameAs( wxT( "Power output" ), false );
123
124 if( thisStr == wxT( "no_connect" ) )
125 return otherStr.IsSameAs( wxT( "Unconnected" ), false );
126
127 return false;
128 }
129};
130
131
133{
134public:
136 LIBEVAL::VALUE( wxEmptyString ),
137 m_item( aItem )
138 {};
139
140 const wxString& AsString() const override
141 {
142 const_cast<PCBEXPR_NETCLASS_VALUE*>( this )->Set(
145 }
146
147 bool EqualTo( LIBEVAL::CONTEXT* aCtx, const VALUE* b ) const override
148 {
149 if( const PCBEXPR_NETCLASS_VALUE* bValue = dynamic_cast<const PCBEXPR_NETCLASS_VALUE*>( b ) )
150 return *( m_item->GetEffectiveNetClass() )
151 == *( bValue->m_item->GetEffectiveNetClass() );
152 else
153 return LIBEVAL::VALUE::EqualTo( aCtx, b );
154 }
155
156 bool NotEqualTo( LIBEVAL::CONTEXT* aCtx, const LIBEVAL::VALUE* b ) const override
157 {
158 if( const PCBEXPR_NETCLASS_VALUE* bValue = dynamic_cast<const PCBEXPR_NETCLASS_VALUE*>( b ) )
159 return m_item->GetEffectiveNetClass() != bValue->m_item->GetEffectiveNetClass();
160 else
161 return LIBEVAL::VALUE::NotEqualTo( aCtx, b );
162 }
163
164protected:
166};
167
168
170{
171public:
173 LIBEVAL::VALUE( wxEmptyString ), m_item( dynamic_cast<FOOTPRINT*>( aItem ) )
174 {};
175
176 const wxString& AsString() const override
177 {
178 if( !m_item )
180
181 if( const COMPONENT_CLASS* compClass = m_item->GetComponentClass() )
182 {
183 const_cast<PCBEXPR_COMPONENT_CLASS_VALUE*>( this )->Set( compClass->GetFullName() );
184 }
185
187 }
188
189 bool EqualTo( LIBEVAL::CONTEXT* aCtx, const VALUE* b ) const override
190 {
191 if( const PCBEXPR_COMPONENT_CLASS_VALUE* bValue =
192 dynamic_cast<const PCBEXPR_COMPONENT_CLASS_VALUE*>( b ) )
193 {
194 if( !m_item || !bValue->m_item )
195 return LIBEVAL::VALUE::EqualTo( aCtx, b );
196
197 const COMPONENT_CLASS* aClass = m_item->GetComponentClass();
198 const COMPONENT_CLASS* bClass = bValue->m_item->GetComponentClass();
199
200 // Note this depends on COMPONENT_CLASS_MANAGER maintaining ownership
201 // of all unique component class objects
202 return aClass == bClass;
203 }
204 else
205 {
206 return LIBEVAL::VALUE::EqualTo( aCtx, b );
207 }
208 }
209
210 bool NotEqualTo( LIBEVAL::CONTEXT* aCtx, const LIBEVAL::VALUE* b ) const override
211 {
212 if( const PCBEXPR_COMPONENT_CLASS_VALUE* bValue =
213 dynamic_cast<const PCBEXPR_COMPONENT_CLASS_VALUE*>( b ) )
214 {
215 if( !m_item || !bValue->m_item )
216 return LIBEVAL::VALUE::EqualTo( aCtx, b );
217
218 const COMPONENT_CLASS* aClass = m_item->GetComponentClass();
219 const COMPONENT_CLASS* bClass = bValue->m_item->GetComponentClass();
220
221 // Note this depends on COMPONENT_CLASS_MANAGER maintaining ownership
222 // of all unique component class objects
223 return aClass != bClass;
224 }
225 else
226 {
227 return LIBEVAL::VALUE::NotEqualTo( aCtx, b );
228 }
229 }
230
231protected:
233};
234
235
237{
238public:
240 LIBEVAL::VALUE( wxEmptyString ),
241 m_item( aItem )
242 {};
243
244 const wxString& AsString() const override
245 {
246 const_cast<PCBEXPR_NET_VALUE*>( this )->Set( m_item->GetNetname() );
248 }
249
250 bool EqualTo( LIBEVAL::CONTEXT* aCtx, const VALUE* b ) const override
251 {
252 if( const PCBEXPR_NET_VALUE* bValue = dynamic_cast<const PCBEXPR_NET_VALUE*>( b ) )
253 return m_item->GetNetCode() == bValue->m_item->GetNetCode();
254 else
255 return LIBEVAL::VALUE::EqualTo( aCtx, b );
256 }
257
258 bool NotEqualTo( LIBEVAL::CONTEXT* aCtx, const LIBEVAL::VALUE* b ) const override
259 {
260 if( const PCBEXPR_NET_VALUE* bValue = dynamic_cast<const PCBEXPR_NET_VALUE*>( b ) )
261 return m_item->GetNetCode() != bValue->m_item->GetNetCode();
262 else
263 return LIBEVAL::VALUE::NotEqualTo( aCtx, b );
264 }
265
266protected:
268};
269
270
272{
273 PCBEXPR_CONTEXT* context = static_cast<PCBEXPR_CONTEXT*>( aCtx );
274
275 if( m_type == LIBEVAL::VT_NULL )
277
278 if( m_itemIndex == 2 )
279 return new PCBEXPR_LAYER_VALUE( context->GetLayer() );
280
281 BOARD_ITEM* item = GetObject( aCtx );
282
283 if( !item )
284 return new LIBEVAL::VALUE();
285
286 auto it = m_matchingTypes.find( TYPE_HASH( *item ) );
287
288 if( it == m_matchingTypes.end() )
289 {
290 // Don't force user to type "A.Type == 'via' && A.Via_Type == 'buried'" when the
291 // simpler "A.Via_Type == 'buried'" is perfectly clear. Instead, return an undefined
292 // value when the property doesn't appear on a particular object.
293
294 return new LIBEVAL::VALUE();
295 }
296 else
297 {
299 {
300 if( m_isOptional )
301 {
302 auto val = item->Get<std::optional<int>>( it->second );
303
304 if( val.has_value() )
305 return new LIBEVAL::VALUE( static_cast<double>( val.value() ) );
306
308 }
309
310 return new LIBEVAL::VALUE( static_cast<double>( item->Get<int>( it->second ) ) );
311 }
312 else
313 {
314 wxString str;
315
316 if( !m_isEnum )
317 {
318 str = item->Get<wxString>( it->second );
319
320 if( it->second->Name() == wxT( "Pin Type" ) )
321 return new PCBEXPR_PINTYPE_VALUE( str );
322 else
323 return new LIBEVAL::VALUE( str );
324 }
325 else
326 {
327 const wxAny& any = item->Get( it->second );
328 PCB_LAYER_ID layer;
329
330 if( it->second->Name() == wxT( "Layer" )
331 || it->second->Name() == wxT( "Layer Top" )
332 || it->second->Name() == wxT( "Layer Bottom" ) )
333 {
334 if( any.GetAs<PCB_LAYER_ID>( &layer ) )
335 return new PCBEXPR_LAYER_VALUE( layer );
336 else if( any.GetAs<wxString>( &str ) )
337 return new PCBEXPR_LAYER_VALUE( context->GetBoard()->GetLayerID( str ) );
338 }
339 else
340 {
341 if( any.GetAs<wxString>( &str ) )
342 return new LIBEVAL::VALUE( str );
343 }
344 }
345
346 return new LIBEVAL::VALUE();
347 }
348 }
349}
350
351
353{
354 BOARD_CONNECTED_ITEM* item = dynamic_cast<BOARD_CONNECTED_ITEM*>( GetObject( aCtx ) );
355
356 if( !item )
357 return new LIBEVAL::VALUE();
358
359 return new PCBEXPR_NETCLASS_VALUE( item );
360}
361
362
364{
365 BOARD_ITEM* item = dynamic_cast<BOARD_ITEM*>( GetObject( aCtx ) );
366
367 if( !item || item->Type() != PCB_FOOTPRINT_T )
368 return new LIBEVAL::VALUE();
369
370 return new PCBEXPR_COMPONENT_CLASS_VALUE( item );
371}
372
373
375{
376 BOARD_CONNECTED_ITEM* item = dynamic_cast<BOARD_CONNECTED_ITEM*>( GetObject( aCtx ) );
377
378 if( !item )
379 return new LIBEVAL::VALUE();
380
381 return new PCBEXPR_NET_VALUE( item );
382}
383
384
386{
387 BOARD_ITEM* item = GetObject( aCtx );
388
389 if( !item )
390 return new LIBEVAL::VALUE();
391
392 return new LIBEVAL::VALUE( ENUM_MAP<KICAD_T>::Instance().ToString( item->Type() ) );
393}
394
395
397{
399
400 return registry.Get( aName.Lower() );
401}
402
403
404std::unique_ptr<LIBEVAL::VAR_REF> PCBEXPR_UCODE::CreateVarRef( const wxString& aVar,
405 const wxString& aField )
406{
408 std::unique_ptr<PCBEXPR_VAR_REF> vref;
409
410 if( aVar.IsSameAs( wxT( "null" ), false ) )
411 {
412 vref = std::make_unique<PCBEXPR_VAR_REF>( 0 );
413 vref->SetType( LIBEVAL::VT_NULL );
414 return vref;
415 }
416
417 // Check for a couple of very common cases and compile them straight to "object code".
418
419 if( aField.CmpNoCase( wxT( "NetClass" ) ) == 0 )
420 {
421 if( aVar == wxT( "A" ) )
422 return std::make_unique<PCBEXPR_NETCLASS_REF>( 0 );
423 else if( aVar == wxT( "B" ) )
424 return std::make_unique<PCBEXPR_NETCLASS_REF>( 1 );
425 else
426 return nullptr;
427 }
428 else if( aField.CmpNoCase( wxT( "ComponentClass" ) ) == 0 )
429 {
430 if( aVar == wxT( "A" ) )
431 return std::make_unique<PCBEXPR_COMPONENT_CLASS_REF>( 0 );
432 else if( aVar == wxT( "B" ) )
433 return std::make_unique<PCBEXPR_COMPONENT_CLASS_REF>( 1 );
434 else
435 return nullptr;
436 }
437 else if( aField.CmpNoCase( wxT( "NetName" ) ) == 0 )
438 {
439 if( aVar == wxT( "A" ) )
440 return std::make_unique<PCBEXPR_NETNAME_REF>( 0 );
441 else if( aVar == wxT( "B" ) )
442 return std::make_unique<PCBEXPR_NETNAME_REF>( 1 );
443 else
444 return nullptr;
445 }
446 else if( aField.CmpNoCase( wxT( "Type" ) ) == 0 )
447 {
448 if( aVar == wxT( "A" ) )
449 return std::make_unique<PCBEXPR_TYPE_REF>( 0 );
450 else if( aVar == wxT( "B" ) )
451 return std::make_unique<PCBEXPR_TYPE_REF>( 1 );
452 else
453 return nullptr;
454 }
455
456 if( aVar == wxT( "A" ) || aVar == wxT( "AB" ) )
457 vref = std::make_unique<PCBEXPR_VAR_REF>( 0 );
458 else if( aVar == wxT( "B" ) )
459 vref = std::make_unique<PCBEXPR_VAR_REF>( 1 );
460 else if( aVar == wxT( "L" ) )
461 vref = std::make_unique<PCBEXPR_VAR_REF>( 2 );
462 else
463 return nullptr;
464
465 if( aField.length() == 0 ) // return reference to base object
466 {
467 return vref;
468 }
469
470 wxString field( aField );
471 field.Replace( wxT( "_" ), wxT( " " ) );
472
473 for( const PROPERTY_MANAGER::CLASS_INFO& cls : propMgr.GetAllClasses() )
474 {
475 if( propMgr.IsOfType( cls.type, TYPE_HASH( BOARD_ITEM ) ) )
476 {
477 PROPERTY_BASE* prop = propMgr.GetProperty( cls.type, field );
478
479 if( prop )
480 {
481 vref->AddAllowedClass( cls.type, prop );
482
483 if( prop->TypeHash() == TYPE_HASH( int ) )
484 {
485 vref->SetType( LIBEVAL::VT_NUMERIC );
486 }
487 if( prop->TypeHash() == TYPE_HASH( std::optional<int> ) )
488 {
489 vref->SetType( LIBEVAL::VT_NUMERIC );
490 vref->SetIsOptional();
491 }
492 else if( prop->TypeHash() == TYPE_HASH( bool ) )
493 {
494 vref->SetType( LIBEVAL::VT_NUMERIC );
495 }
496 else if( prop->TypeHash() == TYPE_HASH( wxString ) )
497 {
498 vref->SetType( LIBEVAL::VT_STRING );
499 }
500 else if ( prop->HasChoices() )
501 { // it's an enum, we treat it as string
502 vref->SetType( LIBEVAL::VT_STRING );
503 vref->SetIsEnum( true );
504 }
505 else
506 {
507 wxFAIL_MSG( wxT( "PCBEXPR_UCODE::createVarRef: Unknown property type." ) );
508 }
509 }
510 }
511 }
512
513 if( vref->GetType() == LIBEVAL::VT_UNDEFINED )
514 vref->SetType( LIBEVAL::VT_PARSE_ERROR );
515
516 return vref;
517}
518
519
521{
522 if( m_items[0] )
523 return m_items[0]->GetBoard();
524
525 return nullptr;
526}
527
528
529/* --------------------------------------------------------------------------------------------
530 * Unit Resolvers
531 */
532
533const std::vector<wxString>& PCBEXPR_UNIT_RESOLVER::GetSupportedUnits() const
534{
535 static const std::vector<wxString> pcbUnits = { wxT( "mil" ), wxT( "mm" ), wxT( "in" ) };
536
537 return pcbUnits;
538}
539
540
542{
543 return _( "must be mm, in, or mil" );
544}
545
546
547double PCBEXPR_UNIT_RESOLVER::Convert( const wxString& aString, int unitId ) const
548{
549 double v = wxAtof( aString );
550
551 switch( unitId )
552 {
553 case 0: return EDA_UNIT_UTILS::UI::DoubleValueFromString( pcbIUScale, EDA_UNITS::MILS, aString );
554 case 1: return EDA_UNIT_UTILS::UI::DoubleValueFromString( pcbIUScale, EDA_UNITS::MILLIMETRES, aString );
555 case 2: return EDA_UNIT_UTILS::UI::DoubleValueFromString( pcbIUScale, EDA_UNITS::INCHES, aString );
556 default: return v;
557 }
558};
559
560
561const std::vector<wxString>& PCBEXPR_UNITLESS_RESOLVER::GetSupportedUnits() const
562{
563 static const std::vector<wxString> emptyUnits;
564
565 return emptyUnits;
566}
567
568
569double PCBEXPR_UNITLESS_RESOLVER::Convert( const wxString& aString, int unitId ) const
570{
571 return wxAtof( aString );
572};
573
574
576{
577 m_unitResolver.reset( aUnitResolver );
578}
579
580
581/* --------------------------------------------------------------------------------------------
582 * PCB Expression Evaluator
583 */
584
586 m_result( 0 ),
587 m_compiler( aUnitResolver ),
588 m_ucode(),
589 m_errorStatus()
590{
591}
592
593
595{
596}
597
598
599bool PCBEXPR_EVALUATOR::Evaluate( const wxString& aExpr )
600{
601 PCBEXPR_UCODE ucode;
602 PCBEXPR_CONTEXT preflightContext( NULL_CONSTRAINT, F_Cu );
603
604 if( !m_compiler.Compile( aExpr.ToUTF8().data(), &ucode, &preflightContext ) )
605 return false;
606
607 PCBEXPR_CONTEXT evaluationContext( NULL_CONSTRAINT, F_Cu );
608 LIBEVAL::VALUE* result = ucode.Run( &evaluationContext );
609
610 if( result->GetType() == LIBEVAL::VT_NUMERIC )
611 m_result = KiROUND( result->AsDouble() );
612
613 return true;
614}
615
constexpr EDA_IU_SCALE pcbIUScale
Definition: base_units.h:108
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition: box2.h:990
BASE_SET & set(size_t pos)
Definition: base_set.h:115
A base class derived from BOARD_ITEM for items that can be connected and have a net,...
virtual NETCLASS * GetEffectiveNetClass() const
Return the NETCLASS for this item.
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition: board_item.h:79
virtual const BOARD * GetBoard() const
Return the BOARD in which this BOARD_ITEM resides, or NULL if none.
Definition: board_item.cpp:47
Information pertinent to a Pcbnew printed circuit board.
Definition: board.h:290
PCB_LAYER_ID GetLayerID(const wxString &aLayerName) const
Return the ID of a layer.
Definition: board.cpp:554
std::unordered_map< wxString, LSET > m_LayerExpressionCache
Definition: board.h:1297
std::shared_mutex m_CachesMutex
Definition: board.h:1291
KICAD_T Type() const
Returns the type of object.
Definition: eda_item.h:101
static ENUM_MAP< T > & Instance()
Definition: property.h:663
const COMPONENT_CLASS * GetComponentClass() const
Definition: footprint.h:1017
wxAny Get(PROPERTY_BASE *aProperty) const
Definition: inspectable.h:99
std::unique_ptr< UNIT_RESOLVER > m_unitResolver
bool Compile(const wxString &aString, UCODE *aCode, CONTEXT *aPreflightContext)
VALUE * Run(CONTEXT *ctx)
void Set(double aValue)
virtual const wxString & AsString() const
static VALUE * MakeNullValue()
virtual bool NotEqualTo(CONTEXT *aCtx, const VALUE *b) const
virtual double AsDouble() const
VAR_TYPE_T GetType() const
virtual bool EqualTo(CONTEXT *aCtx, const VALUE *b) const
LSET is a set of PCB_LAYER_IDs.
Definition: lset.h:36
bool Contains(PCB_LAYER_ID aLayer) const
See if the layer set contains a PCB layer.
Definition: lset.h:62
const wxString GetVariableSubstitutionName() const
Gets the name of this (maybe aggregate) netclass in a format for label variable substitutions.
Definition: netclass.cpp:179
LIBEVAL::FUNC_CALL_REF Get(const wxString &name)
static PCBEXPR_BUILTIN_FUNCTIONS & Instance()
PCBEXPR_COMPILER(LIBEVAL::UNIT_RESOLVER *aUnitResolver)
LIBEVAL::VALUE * GetValue(LIBEVAL::CONTEXT *aCtx) override
bool NotEqualTo(LIBEVAL::CONTEXT *aCtx, const LIBEVAL::VALUE *b) const override
const wxString & AsString() const override
PCBEXPR_COMPONENT_CLASS_VALUE(BOARD_ITEM *aItem)
bool EqualTo(LIBEVAL::CONTEXT *aCtx, const VALUE *b) const override
BOARD_ITEM * m_items[2]
BOARD * GetBoard() const
PCB_LAYER_ID GetLayer() const
BOARD_ITEM * GetItem(int index) const
PCBEXPR_EVALUATOR(LIBEVAL::UNIT_RESOLVER *aUnitResolver)
PCBEXPR_COMPILER m_compiler
bool Evaluate(const wxString &aExpr)
PCBEXPR_LAYER_VALUE(PCB_LAYER_ID aLayer)
virtual bool EqualTo(LIBEVAL::CONTEXT *aCtx, const VALUE *b) const override
LIBEVAL::VALUE * GetValue(LIBEVAL::CONTEXT *aCtx) override
bool EqualTo(LIBEVAL::CONTEXT *aCtx, const VALUE *b) const override
BOARD_CONNECTED_ITEM * m_item
PCBEXPR_NETCLASS_VALUE(BOARD_CONNECTED_ITEM *aItem)
const wxString & AsString() const override
bool NotEqualTo(LIBEVAL::CONTEXT *aCtx, const LIBEVAL::VALUE *b) const override
LIBEVAL::VALUE * GetValue(LIBEVAL::CONTEXT *aCtx) override
bool EqualTo(LIBEVAL::CONTEXT *aCtx, const VALUE *b) const override
BOARD_CONNECTED_ITEM * m_item
PCBEXPR_NET_VALUE(BOARD_CONNECTED_ITEM *aItem)
bool NotEqualTo(LIBEVAL::CONTEXT *aCtx, const LIBEVAL::VALUE *b) const override
const wxString & AsString() const override
bool EqualTo(LIBEVAL::CONTEXT *aCtx, const VALUE *b) const override
PCBEXPR_PINTYPE_VALUE(const wxString &aPinTypeName)
LIBEVAL::VALUE * GetValue(LIBEVAL::CONTEXT *aCtx) override
virtual std::unique_ptr< LIBEVAL::VAR_REF > CreateVarRef(const wxString &aVar, const wxString &aField) override
virtual LIBEVAL::FUNC_CALL_REF CreateFuncCall(const wxString &aName) override
const std::vector< wxString > & GetSupportedUnits() const override
double Convert(const wxString &aString, int unitId) const override
double Convert(const wxString &aString, int unitId) const override
wxString GetSupportedUnitsMessage() const override
const std::vector< wxString > & GetSupportedUnits() const override
std::unordered_map< TYPE_ID, PROPERTY_BASE * > m_matchingTypes
LIBEVAL::VALUE * GetValue(LIBEVAL::CONTEXT *aCtx) override
LIBEVAL::VAR_TYPE_T m_type
BOARD_ITEM * GetObject(const LIBEVAL::CONTEXT *aCtx) const
virtual size_t TypeHash() const =0
Return type-id of the property type.
virtual bool HasChoices() const
Return true if this PROPERTY has a limited set of possible values.
Definition: property.h:241
Provide class metadata.Helper macro to map type hashes to names.
Definition: property_mgr.h:85
CLASSES_INFO GetAllClasses()
static PROPERTY_MANAGER & Instance()
Definition: property_mgr.h:87
PROPERTY_BASE * GetProperty(TYPE_ID aType, const wxString &aProperty) const
Return a property for a specific type.
bool IsOfType(TYPE_ID aDerived, TYPE_ID aBase) const
Return true if aDerived is inherited from aBase.
@ NULL_CONSTRAINT
Definition: drc_rule.h:48
#define _(s)
wxString LayerName(int aLayer)
Returns the default display name for a given layer.
Definition: layer_id.cpp:31
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ F_Cu
Definition: layer_ids.h:64
PCB_LAYER_ID ToLAYER_ID(int aLayer)
Definition: lset.cpp:820
KICOMMON_API double DoubleValueFromString(const EDA_IU_SCALE &aIuScale, EDA_UNITS aUnits, const wxString &aTextValue, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE)
Function DoubleValueFromString converts aTextValue to a double.
Definition: eda_units.cpp:576
std::function< void(CONTEXT *, void *)> FUNC_CALL_REF
BOARD * GetBoard()
#define TYPE_HASH(x)
Definition: property.h:71
@ PCB_FOOTPRINT_T
class FOOTPRINT, a footprint
Definition: typeinfo.h:86