On my Silverlight client, I’m generating a class at run-time to bind it to a datagrid. I’m using a method based on this blog post.
Now I want to use datagrid cellvalidation by calling ValidateProperty in the property setter. But as the properties are generated at runtime, I need to do this in Reflection.Emit.
This is the C# I want to generate in IL:
public int TestProperty
{
get { return testProperty; }
set
{
Validator.ValidateProperty(value, new ValidationContext(this, null, null) { MemberName = "TestProperty" });
testProperty = value;
}
}
This is how ILspy decompiles this method in IL:
.property int32 TestProperty
{
.get public hidebysig specialname
instance int32 get_TestProperty () cil managed
{
// Method begins at RVA 0x224c
// Code size 12 (0xc)
.maxstack 1
.locals init (
[0] int32
)
IL_0000: nop
IL_0001: ldarg.0
IL_0002: ldfld int32 class SilverlightApplication2.testclass::testProperty
IL_0007: stloc.0
IL_0008: br.s IL_000a
IL_000a: ldloc.0
IL_000b: ret
} // End of method testclass.get_TestProperty
.set public hidebysig specialname
instance void set_TestProperty (
int32 value
) cil managed
{
// Method begins at RVA 0x2264
// Code size 43 (0x2b)
.maxstack 5
.locals init (
[0] class [System.ComponentModel.DataAnnotations]System.ComponentModel.DataAnnotations.ValidationContext
)
IL_0000: nop
IL_0001: ldarg.1
IL_0002: box int32
IL_0007: ldarg.0
IL_0008: ldnull
IL_0009: ldnull
IL_000a: newobj instance void [System.ComponentModel.DataAnnotations]System.ComponentModel.DataAnnotations.ValidationContext::.ctor(object, class [mscorlib]System.IServiceProvider, class [mscorlib]System.Collections.Generic.IDictionary`2<object, object>)
IL_000f: stloc.0
IL_0010: ldloc.0
IL_0011: ldstr "TestProperty"
IL_0016: callvirt instance void [System.ComponentModel.DataAnnotations]System.ComponentModel.DataAnnotations.ValidationContext::set_MemberName(string)
IL_001b: nop
IL_001c: ldloc.0
IL_001d: call void [System.ComponentModel.DataAnnotations]System.ComponentModel.DataAnnotations.Validator::ValidateProperty(object, class [System.ComponentModel.DataAnnotations]System.ComponentModel.DataAnnotations.ValidationContext)
IL_0022: nop
IL_0023: ldarg.0
IL_0024: ldarg.1
IL_0025: stfld int32 class SilverlightApplication2.testclass::testProperty
IL_002a: ret
} // End of method testclass.set_TestProperty
}
And this is me trying to write it in Reflection.Emit:
setIL.Emit(OpCodes.Ldarg_1);
setIL.Emit(OpCodes.Box, typeof(Int32));
setIL.Emit(OpCodes.Ldarg_0);
setIL.Emit(OpCodes.Ldnull);
setIL.Emit(OpCodes.Ldnull);
Type[] types = new Type[3];
types[0] = typeof(object);
types[1] = typeof(IServiceProvider);
types[2] = typeof(IDictionary<object, object>);
setIL.Emit(OpCodes.Newobj, typeof(System.ComponentModel.DataAnnotations.ValidationContext).GetConstructor(types));
setIL.Emit(OpCodes.Stloc_0);
setIL.Emit(OpCodes.Ldloc_0);
setIL.Emit(OpCodes.Ldstr, "TestProperty");
setIL.Emit(OpCodes.Callvirt, typeof(System.ComponentModel.DataAnnotations.ValidationContext).GetMethod("set_MemberName"));
setIL.Emit(OpCodes.Ldloc_0);
setIL.Emit(OpCodes.Call, typeof(System.ComponentModel.DataAnnotations.Validator).GetMethod("ValidateProperty"));
setIL.Emit(OpCodes.Ldarg_0);
setIL.Emit(OpCodes.Ldarg_1);
setIL.Emit(OpCodes.Stfld, fieldBuilder);
setIL.Emit(OpCodes.Ret);
This is only a part of my code, I managed to get it working without the Validator.ValidateProperty with the last 4 lines of code. With the 16 other lines I want to add the validate functionality, but right now this results in a ‘Operation could destabilize the runtime’ exception.
Figured it out myself 🙂
This is the code you need to emit Validator.Validate in your property setter: