diff --git a/common/properties/pg_properties.cpp b/common/properties/pg_properties.cpp
index 351b22ea21..b88ff39513 100644
--- a/common/properties/pg_properties.cpp
+++ b/common/properties/pg_properties.cpp
@@ -555,9 +555,17 @@ wxString PGPROPERTY_ANGLE::ValueToString( wxVariant& aVariant,
 wxString PGPROPERTY_ANGLE::ValueToString( wxVariant& aVariant, int aArgFlags ) const
 #endif
 {
-    if( aVariant.GetType() == wxPG_VARIANT_TYPE_DOUBLE )
+    if( aVariant.GetType() == wxT( "std::optional<double>" ) )
+    {
+        auto* variantData = static_cast<STD_OPTIONAL_DOUBLE_VARIANT_DATA*>( aVariant.GetData() );
+
+        if( variantData->Value().has_value() )
+            return wxString::Format( wxS( "%g\u00B0" ), variantData->Value().value() / m_scale );
+        else
+            return wxEmptyString;
+    }
+    else if( aVariant.GetType() == wxPG_VARIANT_TYPE_DOUBLE )
     {
-        // TODO(JE) Is this still needed?
         return wxString::Format( wxS( "%g\u00B0" ), aVariant.GetDouble() / m_scale );
     }
     else if( aVariant.GetType() == wxS( "EDA_ANGLE" ) )
diff --git a/common/widgets/unit_binder.cpp b/common/widgets/unit_binder.cpp
index 6a7b217786..54716aab78 100644
--- a/common/widgets/unit_binder.cpp
+++ b/common/widgets/unit_binder.cpp
@@ -655,15 +655,20 @@ bool UNIT_BINDER::IsIndeterminate() const
 
 bool UNIT_BINDER::IsNull() const
 {
-    wxTextEntry* te = dynamic_cast<wxTextEntry*>( m_valueCtrl );
-
-    if( te )
+    if( wxTextEntry* te = dynamic_cast<wxTextEntry*>( m_valueCtrl ) )
         return te->GetValue().IsEmpty();
 
     return false;
 }
 
 
+void UNIT_BINDER::SetNull()
+{
+    if( wxTextEntry* te = dynamic_cast<wxTextEntry*>( m_valueCtrl ) )
+        return te->SetValue( wxEmptyString );
+}
+
+
 void UNIT_BINDER::SetLabel( const wxString& aLabel )
 {
     m_label->SetLabel( aLabel );
diff --git a/include/properties/property_validators.h b/include/properties/property_validators.h
index 8f5fd04786..d5f782ee49 100644
--- a/include/properties/property_validators.h
+++ b/include/properties/property_validators.h
@@ -112,9 +112,16 @@ public:
         int val = 0;
 
         if( aValue.CheckType<int>() )
+        {
             val = aValue.As<int>();
+        }
         else if( aValue.CheckType<std::optional<int>>() )
-            val = aValue.As<std::optional<int>>().value_or( 0 );
+        {
+            if( aValue.As<std::optional<int>>().has_value() )
+                val = aValue.As<std::optional<int>>().value();
+            else
+                return std::nullopt;     // no value for a std::optional is always valid
+        }
 
         if( val > Max )
             return std::make_unique<VALIDATION_ERROR_TOO_LARGE<int>>( val, Max );
@@ -132,9 +139,16 @@ public:
         int val = 0;
 
         if( aValue.CheckType<int>() )
+        {
             val = aValue.As<int>();
+        }
         else if( aValue.CheckType<std::optional<int>>() )
-            val = aValue.As<std::optional<int>>().value_or( 0 );
+        {
+            if( aValue.As<std::optional<int>>().has_value() )
+                val = aValue.As<std::optional<int>>().value();
+            else
+                return std::nullopt;     // no value for a std::optional is always valid
+        }
 
         if( val < 0 )
             return std::make_unique<VALIDATION_ERROR_TOO_SMALL<int>>( val, 0 );
diff --git a/include/widgets/unit_binder.h b/include/widgets/unit_binder.h
index 825ede62cc..2454b822c0 100644
--- a/include/widgets/unit_binder.h
+++ b/include/widgets/unit_binder.h
@@ -157,6 +157,7 @@ public:
      * Return true if the control holds no value (ie: empty string, **not** 0).
      */
     bool IsNull() const;
+    void SetNull();
 
     /**
      * Validate the control against the given range, informing the user of any errors found.
diff --git a/pcbnew/dialogs/dialog_pad_properties.cpp b/pcbnew/dialogs/dialog_pad_properties.cpp
index b5345dcdb3..840aafeb74 100644
--- a/pcbnew/dialogs/dialog_pad_properties.cpp
+++ b/pcbnew/dialogs/dialog_pad_properties.cpp
@@ -672,9 +672,17 @@ void DIALOG_PAD_PROPERTIES::initValues()
     else
         m_pasteMarginRatio.ChangeValue( wxEmptyString );
 
-    m_spokeWidth.ChangeValue( m_previewPad->GetThermalSpokeWidth() );
+    if( m_previewPad->GetLocalThermalSpokeWidthOverride().has_value() )
+        m_spokeWidth.ChangeValue( m_previewPad->GetLocalThermalSpokeWidthOverride().value() );
+    else
+        m_spokeWidth.SetNull();
+
+    if( m_previewPad->GetLocalThermalGapOverride().has_value() )
+        m_thermalGap.ChangeValue( m_previewPad->GetLocalThermalGapOverride().value() );
+    else
+        m_thermalGap.SetNull();
+
     m_spokeAngle.ChangeAngleValue( m_previewPad->GetThermalSpokeAngle() );
-    m_thermalGap.ChangeValue( m_previewPad->GetThermalGap() );
     m_pad_orientation.ChangeAngleValue( m_previewPad->GetOrientation() );
 
     m_cbTeardrops->SetValue( m_previewPad->GetTeardropParams().m_Enabled );
@@ -1734,9 +1742,13 @@ bool DIALOG_PAD_PROPERTIES::transferDataToPad( PAD* aPad )
     else
         aPad->SetLocalSolderPasteMarginRatio( m_pasteMarginRatio.GetDoubleValue() / 100.0 );
 
-    aPad->SetThermalSpokeWidth( m_spokeWidth.GetIntValue() );
+    if( !m_spokeWidth.IsNull() )
+        aPad->SetLocalThermalSpokeWidthOverride( m_spokeWidth.GetIntValue() );
+
+    if( !m_thermalGap.IsNull() )
+        aPad->SetLocalThermalGapOverride( m_thermalGap.GetIntValue() );
+
     aPad->SetThermalSpokeAngle( m_spokeAngle.GetAngleValue() );
-    aPad->SetThermalGap( m_thermalGap.GetIntValue() );
 
     // And rotation
     aPad->SetOrientation( m_pad_orientation.GetAngleValue() );
diff --git a/pcbnew/drc/drc_test_provider_library_parity.cpp b/pcbnew/drc/drc_test_provider_library_parity.cpp
index 41f348ce8f..043627c6cf 100644
--- a/pcbnew/drc/drc_test_provider_library_parity.cpp
+++ b/pcbnew/drc/drc_test_provider_library_parity.cpp
@@ -233,13 +233,15 @@ bool padHasOverrides( const PAD* a, const PAD* b, REPORTER& aReporter )
         REPORT_MSG( _( "%s has zone connection override." ), PAD_DESC( a ) );
     }
 
-    if( a->GetThermalGap() != b->GetThermalGap() )
+    if( a->GetLocalThermalGapOverride().has_value()
+            && a->GetThermalGap() != b->GetThermalGap() )
     {
         diff = true;
         REPORT_MSG( _( "%s has thermal relief gap override." ), PAD_DESC( a ) );
     }
 
-    if( a->GetThermalSpokeWidth() != b->GetThermalSpokeWidth() )
+    if( a->GetLocalThermalSpokeWidthOverride().has_value()
+            && a->GetLocalThermalSpokeWidthOverride() != b->GetLocalThermalSpokeWidthOverride() )
     {
         diff = true;
         REPORT_MSG( _( "%s has thermal relief spoke width override." ), PAD_DESC( a ) );
diff --git a/pcbnew/pad.cpp b/pcbnew/pad.cpp
index 0e489b450a..e0255cb13d 100644
--- a/pcbnew/pad.cpp
+++ b/pcbnew/pad.cpp
@@ -1265,7 +1265,7 @@ int PAD::GetLocalThermalGapOverride( wxString* aSource ) const
     if( m_padStack.ThermalGap().has_value() && aSource )
         *aSource = _( "pad" );
 
-    return m_padStack.ThermalGap().value_or( 0 );
+    return GetLocalThermalGapOverride().value_or( 0 );
 }
 
 
@@ -1877,9 +1877,9 @@ void PAD::ImportSettingsFrom( const PAD& aMasterPad )
     SetLocalSolderPasteMarginRatio( aMasterPad.GetLocalSolderPasteMarginRatio() );
 
     SetLocalZoneConnection( aMasterPad.GetLocalZoneConnection() );
-    SetThermalSpokeWidth( aMasterPad.GetThermalSpokeWidth() );
+    SetLocalThermalSpokeWidthOverride( aMasterPad.GetLocalThermalSpokeWidthOverride() );
     SetThermalSpokeAngle( aMasterPad.GetThermalSpokeAngle() );
-    SetThermalGap( aMasterPad.GetThermalGap() );
+    SetLocalThermalGapOverride( aMasterPad.GetLocalThermalGapOverride() );
 
     SetCustomShapeInZoneOpt( aMasterPad.GetCustomShapeInZoneOpt() );
 
@@ -2720,17 +2720,20 @@ static struct PAD_DESC
 
         constexpr int minZoneWidth = pcbIUScale.mmToIU( ZONE_THICKNESS_MIN_VALUE_MM );
 
-        propMgr.AddProperty( new PROPERTY<PAD, int>( _HKI( "Thermal Relief Spoke Width" ),
-                    &PAD::SetThermalSpokeWidth, &PAD::GetThermalSpokeWidth,
+        propMgr.AddProperty( new PROPERTY<PAD, std::optional<int>>(
+                    _HKI( "Thermal Relief Spoke Width" ),
+                    &PAD::SetLocalThermalSpokeWidthOverride, &PAD::GetLocalThermalSpokeWidthOverride,
                     PROPERTY_DISPLAY::PT_SIZE ), groupOverrides )
                 .SetValidator( PROPERTY_VALIDATORS::RangeIntValidator<minZoneWidth, INT_MAX> );
 
-        propMgr.AddProperty( new PROPERTY<PAD, double>( _HKI( "Thermal Relief Spoke Angle" ),
+        propMgr.AddProperty( new PROPERTY<PAD, double>(
+                    _HKI( "Thermal Relief Spoke Angle" ),
                     &PAD::SetThermalSpokeAngleDegrees, &PAD::GetThermalSpokeAngleDegrees,
                     PROPERTY_DISPLAY::PT_DEGREE ), groupOverrides );
 
-        propMgr.AddProperty( new PROPERTY<PAD, int>( _HKI( "Thermal Relief Gap" ),
-                    &PAD::SetThermalGap, &PAD::GetThermalGap,
+        propMgr.AddProperty( new PROPERTY<PAD, std::optional<int>>(
+                    _HKI( "Thermal Relief Gap" ),
+                    &PAD::SetLocalThermalGapOverride, &PAD::GetLocalThermalGapOverride,
                     PROPERTY_DISPLAY::PT_SIZE ), groupOverrides )
                 .SetValidator( PROPERTY_VALIDATORS::PositiveIntValidator );
 
diff --git a/pcbnew/pad.h b/pcbnew/pad.h
index af917af59b..93f2eb977a 100644
--- a/pcbnew/pad.h
+++ b/pcbnew/pad.h
@@ -594,8 +594,14 @@ public:
      * Set the width of the thermal spokes connecting the pad to a zone.  If != 0 this will
      * override similar settings in the parent footprint and zone.
      */
-    void SetThermalSpokeWidth( int aWidth ) { m_padStack.ThermalSpokeWidth() = aWidth; }
-    int GetThermalSpokeWidth() const { return m_padStack.ThermalSpokeWidth().value_or( 0 ); }
+    void SetLocalThermalSpokeWidthOverride( std::optional<int> aWidth )
+    {
+        m_padStack.ThermalSpokeWidth() = aWidth;
+    }
+    std::optional<int> GetLocalThermalSpokeWidthOverride() const
+    {
+        return m_padStack.ThermalSpokeWidth();
+    }
 
     int GetLocalSpokeWidthOverride( wxString* aSource = nullptr ) const;
 
@@ -626,7 +632,16 @@ public:
     void SetThermalGap( int aGap ) { m_padStack.ThermalGap() = aGap; }
     int GetThermalGap() const { return m_padStack.ThermalGap().value_or( 0 ); }
 
-    int GetLocalThermalGapOverride( wxString* aSource = nullptr ) const;
+    int GetLocalThermalGapOverride( wxString* aSource ) const;
+
+    std::optional<int> GetLocalThermalGapOverride() const
+    {
+        return m_padStack.ThermalGap();
+    }
+    void SetLocalThermalGapOverride( const std::optional<int>& aOverride )
+    {
+        m_padStack.ThermalGap() = aOverride;
+    }
 
     /**
      * Has meaning only for rounded rectangle pads.
diff --git a/pcbnew/padstack.cpp b/pcbnew/padstack.cpp
index ad51ae1cf1..02c6125677 100644
--- a/pcbnew/padstack.cpp
+++ b/pcbnew/padstack.cpp
@@ -40,9 +40,9 @@ PADSTACK::PADSTACK( BOARD_ITEM* aParent ) :
 {
     m_copperProps[PADSTACK::ALL_LAYERS].shape = SHAPE_PROPS();
     m_copperProps[PADSTACK::ALL_LAYERS].zone_connection = ZONE_CONNECTION::INHERITED;
-    m_copperProps[PADSTACK::ALL_LAYERS].thermal_spoke_width = 0;
+    m_copperProps[PADSTACK::ALL_LAYERS].thermal_spoke_width = std::nullopt;
     m_copperProps[PADSTACK::ALL_LAYERS].thermal_spoke_angle = ANGLE_45;
-    m_copperProps[PADSTACK::ALL_LAYERS].thermal_gap = 0;
+    m_copperProps[PADSTACK::ALL_LAYERS].thermal_gap = std::nullopt;
 
     m_drill.shape = PAD_DRILL_SHAPE::CIRCLE;
     m_drill.start = F_Cu;
@@ -247,8 +247,8 @@ bool PADSTACK::Deserialize( const google::protobuf::Any& aContainer )
     else
     {
         CopperLayer( ALL_LAYERS ).zone_connection = ZONE_CONNECTION::INHERITED;
-        CopperLayer( ALL_LAYERS ).thermal_gap = 0;
-        CopperLayer( ALL_LAYERS ).thermal_spoke_width = 0;
+        CopperLayer( ALL_LAYERS ).thermal_gap = std::nullopt;
+        CopperLayer( ALL_LAYERS ).thermal_spoke_width = std::nullopt;
         CopperLayer( ALL_LAYERS ).thermal_spoke_angle = DefaultThermalSpokeAngleForShape( F_Cu );
     }
 
diff --git a/pcbnew/pcb_io/cadstar/cadstar_pcb_archive_loader.cpp b/pcbnew/pcb_io/cadstar/cadstar_pcb_archive_loader.cpp
index 2f50653f54..b3e0ea4ae2 100644
--- a/pcbnew/pcb_io/cadstar/cadstar_pcb_archive_loader.cpp
+++ b/pcbnew/pcb_io/cadstar/cadstar_pcb_archive_loader.cpp
@@ -1236,7 +1236,7 @@ PAD* CADSTAR_PCB_ARCHIVE_LOADER::getKiCadPad( const COMPONENT_PAD& aCadstarPad,
         pad->SetThermalGap( getKiCadLength( csPadcode.ReliefClearance ) );
 
     if( csPadcode.ReliefWidth != UNDEFINED_VALUE )
-        pad->SetThermalSpokeWidth( getKiCadLength( csPadcode.ReliefWidth ) );
+        pad->SetLocalThermalSpokeWidthOverride( getKiCadLength( csPadcode.ReliefWidth ) );
 
     if( csPadcode.DrillDiameter != UNDEFINED_VALUE )
     {
diff --git a/pcbnew/pcb_io/kicad_legacy/pcb_io_kicad_legacy.cpp b/pcbnew/pcb_io/kicad_legacy/pcb_io_kicad_legacy.cpp
index a6d3d68538..542b827e27 100644
--- a/pcbnew/pcb_io/kicad_legacy/pcb_io_kicad_legacy.cpp
+++ b/pcbnew/pcb_io/kicad_legacy/pcb_io_kicad_legacy.cpp
@@ -1567,12 +1567,12 @@ void PCB_IO_KICAD_LEGACY::loadPAD( FOOTPRINT* aFootprint )
         else if( TESTLINE( ".ThermalWidth" ) )
         {
             BIU tmp = biuParse( line + SZ( ".ThermalWidth" ) );
-            pad->SetThermalSpokeWidth( tmp );
+            pad->SetLocalThermalSpokeWidthOverride( tmp );
         }
         else if( TESTLINE( ".ThermalGap" ) )
         {
             BIU tmp = biuParse( line + SZ( ".ThermalGap" ) );
-            pad->SetThermalGap( tmp );
+            pad->SetLocalThermalGapOverride( tmp );
         }
         else if( TESTLINE( "$EndPAD" ) )
         {
diff --git a/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr.cpp b/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr.cpp
index a1e755d267..2645a050b0 100644
--- a/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr.cpp
+++ b/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr.cpp
@@ -1668,10 +1668,10 @@ void PCB_IO_KICAD_SEXPR::format( const PAD* aPad ) const
                       static_cast<int>( aPad->GetLocalZoneConnection() ) );
     }
 
-    if( aPad->GetThermalSpokeWidth() != 0 )
+    if( aPad->GetLocalThermalSpokeWidthOverride().has_value() )
     {
         m_out->Print( "(thermal_bridge_width %s)",
-                      formatInternalUnits( aPad->GetThermalSpokeWidth() ).c_str() );
+                      formatInternalUnits( aPad->GetLocalThermalSpokeWidthOverride().value() ).c_str() );
     }
 
     EDA_ANGLE defaultThermalSpokeAngle = ANGLE_90;
@@ -1689,10 +1689,10 @@ void PCB_IO_KICAD_SEXPR::format( const PAD* aPad ) const
                       EDA_UNIT_UTILS::FormatAngle( aPad->GetThermalSpokeAngle() ).c_str() );
     }
 
-    if( aPad->GetThermalGap() != 0 )
+    if( aPad->GetLocalThermalGapOverride().has_value() )
     {
         m_out->Print( "(thermal_gap %s)",
-                      formatInternalUnits( aPad->GetThermalGap() ).c_str() );
+                      formatInternalUnits( aPad->GetLocalThermalGapOverride().value() ).c_str() );
     }
 
     auto anchorShape =
diff --git a/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr_parser.cpp b/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr_parser.cpp
index 78158184e1..226fca464d 100644
--- a/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr_parser.cpp
+++ b/pcbnew/pcb_io/kicad_sexpr/pcb_io_kicad_sexpr_parser.cpp
@@ -5235,7 +5235,7 @@ PAD* PCB_IO_KICAD_SEXPR_PARSER::parsePAD( FOOTPRINT* aParent )
 
         case T_thermal_width:       // legacy token
         case T_thermal_bridge_width:
-            pad->SetThermalSpokeWidth( parseBoardUnits( token ) );
+            pad->SetLocalThermalSpokeWidthOverride( parseBoardUnits( token ) );
             NeedRIGHT();
             break;
 
diff --git a/pcbnew/pcb_shape.cpp b/pcbnew/pcb_shape.cpp
index e78aa8b798..48b87710f2 100644
--- a/pcbnew/pcb_shape.cpp
+++ b/pcbnew/pcb_shape.cpp
@@ -550,8 +550,8 @@ void PCB_SHAPE::SetIsProxyItem( bool aIsProxy )
     {
         if( GetShape() == SHAPE_T::SEGMENT )
         {
-            if( parentPad && parentPad->GetThermalSpokeWidth() )
-                SetWidth( parentPad->GetThermalSpokeWidth() );
+            if( parentPad && parentPad->GetLocalThermalSpokeWidthOverride().has_value() )
+                SetWidth( parentPad->GetLocalThermalSpokeWidthOverride().value() );
             else
                 SetWidth( pcbIUScale.mmToIU( ZONE_THERMAL_RELIEF_COPPER_WIDTH_MM ) );
         }
diff --git a/pcbnew/tools/pad_tool.cpp b/pcbnew/tools/pad_tool.cpp
index e09439f56a..ca13b52f2a 100644
--- a/pcbnew/tools/pad_tool.cpp
+++ b/pcbnew/tools/pad_tool.cpp
@@ -896,8 +896,8 @@ void PAD_TOOL::explodePad( PAD* aPad, PCB_LAYER_ID* aLayer, BOARD_COMMIT& aCommi
 
             if( shape->IsProxyItem() && shape->GetShape() == SHAPE_T::SEGMENT )
             {
-                if( aPad->GetThermalSpokeWidth() )
-                    shape->SetWidth( aPad->GetThermalSpokeWidth() );
+                if( aPad->GetLocalThermalSpokeWidthOverride().has_value() )
+                    shape->SetWidth( aPad->GetLocalThermalSpokeWidthOverride().value() );
                 else
                     shape->SetWidth( pcbIUScale.mmToIU( ZONE_THERMAL_RELIEF_COPPER_WIDTH_MM ) );
             }
diff --git a/qa/pcbnew_utils/board_test_utils.cpp b/qa/pcbnew_utils/board_test_utils.cpp
index 9d4cc09a43..cca50238ca 100644
--- a/qa/pcbnew_utils/board_test_utils.cpp
+++ b/qa/pcbnew_utils/board_test_utils.cpp
@@ -367,7 +367,8 @@ void CheckFpPad( const PAD* expected, const PAD* pad )
         BOOST_CHECK_EQUAL( expected->GetLocalClearance().value_or( 0 ),
                            pad->GetLocalClearance().value_or( 0 ) );
         CHECK_ENUM_CLASS_EQUAL( expected->GetLocalZoneConnection(), pad->GetLocalZoneConnection() );
-        BOOST_CHECK_EQUAL( expected->GetThermalSpokeWidth(), pad->GetThermalSpokeWidth() );
+        BOOST_CHECK_EQUAL( expected->GetLocalThermalSpokeWidthOverride().value_or( 0 ),
+                           pad->GetLocalThermalSpokeWidthOverride().value_or( 0 ) );
         BOOST_CHECK_EQUAL( expected->GetThermalSpokeAngle(), pad->GetThermalSpokeAngle() );
         BOOST_CHECK_EQUAL( expected->GetThermalGap(), pad->GetThermalGap() );
         BOOST_CHECK_EQUAL( expected->GetRoundRectRadiusRatio( PADSTACK::ALL_LAYERS ),