views:

1508

answers:

6

We have a code generator that munges the schema of a given database to automate our inhouse n-tier architecture. The output is various C# partial classes, one per file.

In the code to munge all the strings, we try and keep on top of the indenting and formatting as much as possible, but invariably when you come to open the file in Visual Studio the formatting is awry. A quick ctrl-k, ctrl-d fixes it, but obviously this reformatting is lost the next time the class is generated.

What I'd like to know, is if there's a way I can somehow automatically format the contents of the textfile in the same way Visual Studio does?

Pseudocode

Create "code" object, passing text file to constructor
Invoke "format" method
Re-save text file

Any help greatly appreciated.

EDIT: I should clarify - I want to be able to invoke the formatting from my C# code that creates the textfile containing my generated C#. The format of the code can be standardised (doesn't have to be per-developer), and I don't want to have to install any 3rd-party apps.

I seem to remember there's a namespace containing loads of classes for creating C# in C#: http://msdn.microsoft.com/en-us/library/system.codedom(VS.80).aspx, but I'm not sure if it contains any classes that could help.

FURTHER EDIT: My code generator is a winforms app deployed via a click-once install. It's used by many developers in-house. I need a solution that doesn't require each developer to have a tool installed on their machine.

A: 

I would take a look at Artistic Style - it is very good at pretty-printing C-ish languages.

Andrew Hare
Thanks for the answer, but this won't help me as I'd like to format the code from my code (IYSWIM).
Paul Suart
A: 

Only if you're running the code generator as a VS add-on - each developer is going to have different settings.

Greg Hurlman
No, it's a winforms app. Thanks anyway.
Paul Suart
+4  A: 

Take a look at Narrange.You'll probably need to automate these things as part of the build.
Not sure if it meets all your requirements though.
To quote:

NArrange is a .NET code beautifier that automatically organizes code members and elements within .NET classes.

Cherian
Thanks for the answer, but this won't help me as I'd like to format the code from my code.
Paul Suart
I love NArrange. NArrange is called from the command line, so you can call it from the code
phsr
Wouldn't it require me to have it installed on the machine I'm running the code on, or is there a redistributable dll I can include in my app? Thanks.
Paul Suart
Source is available, so you should be able to integrate and redistribute. If you do, be sure to comply with the license
phsr
A: 

Here's how to do it from the context of a macro or add-in:

var dte = (EnvDTE80.DTE2)System.Runtime.InteropServices.Marshal.GetActiveObject("VisualStudio.DTE.8.0");
dte.ExecuteCommand("File.OpenFile", filename);
dte.ExecuteCommand("Edit.FormatDocument", filename);
dte.ActiveDocument.Close(vsSaveChanges.vsSaveChangesYes);

Warning: As @Greg Hurlman says, the output will vary depending on the user's current options.

Edit:

unfortunately your method requires me to have an instance of VS running alongside my winforms app. Can you think of a way to create an instance of VS from within my app (if that's even possible)?

I think it might be possible to do from within your Win.Form app. However, you'll have to have Visual Studio installed on the machine running the code.

Try this:

var dte = (EnvDTE80.DTE2)Microsoft.VisualBasic.Interaction.CreateObject("VisualStudio.DTE.8.0", "");
dte.ExecuteCommand("File.OpenFile", filename);
dte.ExecuteCommand("Edit.FormatDocument", filename);
dte.ActiveDocument.Close(vsSaveChanges.vsSaveChangesYes);

Keep in mind that you'll need references to the EnvDTE80.dll assembly.

Randolpho
Hi Randolpho - You're the closest one yet, but unfortunately your method requires me to have an instance of VS running alongside my winforms app. Can you think of a way to create an instance of VS from within my app (if that's even possible)? Thanks.
Paul Suart
A: 

You can use CodeDOM and the CSharpCodeProvider. It is all in the namespaces Microsoft.CSharp and System.CodeDom.

Her is an example of a property:

StringWriter writer = new StringWriter();
CSharpCodeProvider provider = new CSharpCodeProvider();
CodeMemberProperty property = new CodeMemberProperty();
property.Type = new CodeTypeReference(typeof(int));
property.Name = "MeaningOfLifeUniverseAndEverything";
property.GetStatements.Add(new CodeMethodReturnStatement(new CodePrimitiveExpression(42)));
provider.GenerateCodeFromMember(property, writer, null);
Console.WriteLine(writer.GetStringBuilder().ToString());

This code will generate:

private int MeaningOfLifeUniverseAndEverything {
    get {
        return 42;
    }
}

The CodeDOM is a quite chatty way to generate code. The good thing is that you can generate multiple languages. Perhaps you can find a Erlang.NET CodeProvider?

You might be able to do a few shortcuts by using CodeSnippetExpression.

Hallgrim
Hi Hallgrim. Yes, I discounted this way of generating code because of its verbosity. Without re-writing our code generator, there's not much I can do. Do you know if there's a method in System.CodeDom to format a class file?
Paul Suart
CSharpCodeProvider had a CreateParser() method, but it never returned anything but null and is now obsolete.
Hallgrim
+2  A: 

To Expand on Cherian's answer:

NArrange will allow you to do a lot of code formatting. It is open source, and the source is available on their site, so you could potentially integrate and redistbute it with your tool.

You should only need the dll's and just look at the exe on how to call the formatting. It will also create a backup of the code being formatted. It is really a nice tool if it fits your needs.

phsr