I have a number of classes that are decorated with DebuggerDisplayAttribute.
I want to be able to add trace statements to Unit Tests that will display instances of these classes.
Does there exist a method in the .NET Framework that will display an object formatted using DebuggerDisplayAttribute (or fall back to using .ToString() if no DebuggerDisplayAttribute is defined)?
EDIT
To clarify, I was hoping there might be something built into the Framework. I know I can get the Value property from DebuggerDisplayAttribute, but I then need to format my instance using the format string represented by DebuggerDisplayAttribute.Value.
If I roll my own, I'd envisage an extension method along the following lines:
public string FormatDebugDisplay(this object value)
{
DebugDisplayAttribute attribute = ... get the attribute for value ...
if (attribute = null) return value.ToString();
string formatString = attribute.Value;
??? How do I format value using formatString ???
return SomeFormatMethod(formatString, value);
}
That might be nice -- but DebuggerDisplayAttribute's format string is evaluated by the debugger, the same way it evaluates expressions you type into the Watch windows or the Immediate window. That's why you can put arbitrary expressions inside the braces, like {FirstName + " " + LastName}
.
Therefore, to evaluate these in your code, you would need to embed the Visual Studio debugger into your app. Probably not gonna happen. (grin)
Your best bet is probably to take all the formatting logic that's currently in your DebuggerDisplay format string, and make it a method instead. Then you're free to call that method from your code. Your DebuggerDisplay attribute ends up doing nothing but calling the method.
[DebuggerDisplay("{Inspect()}")]
public class MyClass {
public string Inspect() { ... }
}
This method will not implement exactly what DebuggerDisplayAttribute provides in the debugger, but this is what I have been using in my code. It covers about 90% (or more) of the cases we experience in our codebase. If you fix it up to cover even more cases, I'd love to see your enhancements!
public static string ToDebuggerString(this object @this)
{
var display = @this.GetType().GetCustomAttributes(typeof (DebuggerDisplayAttribute),false).FirstOrDefault() as DebuggerDisplayAttribute;
if (display == null)
return @this.ToString();
var format = display.Value;
var builder = new StringBuilder();
for (var index = 0; index < format.Length; index++)
{
if (format[index] == '{')
{
var close = format.IndexOf('}', index);
if (close > index)
{
index++;
var name = format.Substring(index, close - index);
var property = @this.GetType().GetProperty(name);
if (property != null)
{
var value = property.GetValue(@this, null).ToString();
builder.Append(value);
index += name.Length;
}
}
}
else
builder.Append(format[index]);
}
return builder.ToString();
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With