From e88dda2c0101430f9b3a862bda6bcb4d0d4f1cae Mon Sep 17 00:00:00 2001 From: Jeff Young Date: Thu, 30 Jul 2020 18:06:50 +0100 Subject: [PATCH] Handle wildcards and "other" item for insideCourtyard DRC function. Fixes https://gitlab.com/kicad/code/kicad/issues/4519 --- pcbnew/pcb_expr_evaluator.cpp | 58 ++++++++++++++++++++++------------- 1 file changed, 37 insertions(+), 21 deletions(-) diff --git a/pcbnew/pcb_expr_evaluator.cpp b/pcbnew/pcb_expr_evaluator.cpp index 8224cf2f88..32796393e5 100644 --- a/pcbnew/pcb_expr_evaluator.cpp +++ b/pcbnew/pcb_expr_evaluator.cpp @@ -80,8 +80,9 @@ static void isPlated( LIBEVAL::CONTEXT* aCtx, void* self ) static void insideCourtyard( LIBEVAL::CONTEXT* aCtx, void* self ) { - LIBEVAL::VALUE* arg = aCtx->Pop(); - LIBEVAL::VALUE* result = aCtx->AllocValue(); + PCB_EXPR_CONTEXT* context = static_cast( aCtx ); + LIBEVAL::VALUE* arg = aCtx->Pop(); + LIBEVAL::VALUE* result = aCtx->AllocValue(); result->Set( 0.0 ); aCtx->Push( result ); @@ -93,35 +94,50 @@ static void insideCourtyard( LIBEVAL::CONTEXT* aCtx, void* self ) return; } - wxString footprintRef = arg->AsString(); PCB_EXPR_VAR_REF* vref = static_cast( self ); BOARD_ITEM* item = vref ? vref->GetObject( aCtx ) : nullptr; + MODULE* footprint = nullptr; - if( item ) + if( !item ) + return; + + if( arg->AsString() == "A" ) { - for( MODULE* footprint : item->GetBoard()->Modules() ) + footprint = dynamic_cast( context->GetItem( 0 ) ); + } + else if( arg->AsString() == "B" ) + { + footprint = dynamic_cast( context->GetItem( 1 ) ); + } + else + { + for( MODULE* candidate : item->GetBoard()->Modules() ) { - if( footprint->GetReference() == footprintRef ) + if( candidate->GetReference().Matches( arg->AsString() ) ) { - SHAPE_POLY_SET footprintCourtyard; - - if( footprint->IsFlipped() ) - footprintCourtyard = footprint->GetPolyCourtyardBack(); - else - footprintCourtyard = footprint->GetPolyCourtyardFront(); - - SHAPE_POLY_SET testPoly; - - item->TransformShapeWithClearanceToPolygon( testPoly, 0 ); - testPoly.BooleanIntersection( footprintCourtyard, SHAPE_POLY_SET::PM_FAST ); - - if( testPoly.OutlineCount() ) - result->Set( 1.0 ); - + footprint = candidate; break; } } } + + if( footprint ) + { + SHAPE_POLY_SET footprintCourtyard; + + if( footprint->IsFlipped() ) + footprintCourtyard = footprint->GetPolyCourtyardBack(); + else + footprintCourtyard = footprint->GetPolyCourtyardFront(); + + SHAPE_POLY_SET testPoly; + + item->TransformShapeWithClearanceToPolygon( testPoly, 0 ); + testPoly.BooleanIntersection( footprintCourtyard, SHAPE_POLY_SET::PM_FAST ); + + if( testPoly.OutlineCount() ) + result->Set( 1.0 ); + } }