Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Adding XML formatting #858

Draft
wants to merge 55 commits into
base: 1.0.0
Choose a base branch
from
Draft

Adding XML formatting #858

wants to merge 55 commits into from

Conversation

belav
Copy link
Owner

@belav belav commented Mar 27, 2023

closes #819

@belav belav marked this pull request as draft March 27, 2023 16:58
@belav
Copy link
Owner Author

belav commented Mar 28, 2023

The first pass at this was to just use the XmlTextWriter, which fixes indentation to be consistent, but also destroys line breaks.

https://github.com/belav/csharpier-repos/pull/63/files?diff=split&w=1

A notable example
image

Maybe instead for a first pass this can

  • Get indentation consistent
  • Clean up extra new lines - maybe new lines separating elements should be allowed, but if there are multiples, then remove them. It seems pretty common for people to leave a blank line between <ItemGroup> elements.
  • Get the end element tags consistent - <Tag/> should be <Tag />

I think that should be pretty easy to do using the existing Doc system.

Then the two remaining things I think are also needed

  • A way to specify indentation options by file type. At work we use 2 spaces in csproj, but 4 spaces in cs.
  • Maybe a way to specify which file types csharpier should format. I am leaning towards auto including cs, csproj, props and targets. The csharpierignore file is one way someone could opt out of formatting the none c# files. Maybe there should be an option instead? Something like
# this would be the default value
filesToFormat: 
  - cs
  - csproj
  - props
  - targets

Do you have any thoughts @shocklateboy92 ?

@belav
Copy link
Owner Author

belav commented Mar 28, 2023

Actually, maybe more like what prettier does. Which can be combined with options for files, and at some point also allow custom formatters

tabWidth: 4
overrides:
  - files: "*.csproj"
    options:
      tabWidth: 2
  - files: ["*.props", "*.targets"]
    formatter: none
  - files: "*.ps1"
    formatter: PluginThatFormatsPowershell

@belav belav force-pushed the csproj-formatting branch from 75f4d75 to a8d1c64 Compare April 17, 2023 16:15
@gdlol
Copy link

gdlol commented Oct 24, 2023

😃 Hi there, I am using both csharpier and prettier in my projects and wondering if this would be some duplicate effort.

I have been using prettier for everything besides .cs files (like JSON files), and for .csproj (XML) files there is @prettier/plugin-xml.

The plugin uses linguist-languages to identify XML file types base on extensions, and it already includes file types like .csproj, .props and .targets. The list of extensions for XML can be found here. So far the experience is pretty good.

There may also be tooling concerns (VSCode settings) for including csproj formatting in csharpier. For instance, users may need to set csharpier as the default XML formatter, but is it a good idea?

Anyway, csharpier is a nice tool, and thanks for the effort to keep improving it.

@belav
Copy link
Owner Author

belav commented Nov 1, 2023

😃 Hi there, I am using both csharpier and prettier in my projects and wondering if this would be some duplicate effort.

I have been using prettier for everything besides .cs files (like JSON files), and for .csproj (XML) files there is @prettier/plugin-xml.

My initial pass for this was just intended to clean up whitespace and indentation, but I already ran into reasons why I think it needs to do a bit more than that and can probably use their code as a starting point. Thanks for sharing the link!

Even if prettier already supports this type of formatting I still see value in csharpier providing the functionality. There are situations where someone may not want to deal with node/npm and still be able to format xml. For a web app, they probably already have node but there are plenty of other types projects out there.

# Conflicts:
#	Directory.Build.props
#	Scripts/UpdateCSharpierRepos.ps1
#	Src/CSharpier.Cli/CommandLineFormatter.cs
#	Src/CSharpier.Generators/CSharpier.Generators.csproj
#	Src/CSharpier.Playground/ClientApp/src/AppContext.ts
#	Src/CSharpier.Playground/ClientApp/src/Controls.tsx
#	Src/CSharpier.Playground/ClientApp/src/FormatCode.ts
#	Src/CSharpier.Playground/Controllers/FormatController.cs
#	Src/CSharpier.Tests.Generators/CSharpier.Tests.Generators.csproj
#	Src/CSharpier.Tests.Generators/FormattingTestsGenerator.cs
#	Src/CSharpier.Tests/CodeFormatterTests.cs
#	Src/CSharpier.Tests/FormattingTests/BaseTest.cs
#	Src/CSharpier.Tests/Samples/AllInOne.test
#	Src/CSharpier/CSharpFormatter.cs
#	Src/CSharpier/CodeFormatter.cs
belav and others added 27 commits October 5, 2024 15:04
# Conflicts:
#	Src/CSharpier/SyntaxPrinter/FormattingContext.cs
#	Src/CSharpier/SyntaxPrinter/TrailingComma.cs
# Conflicts:
#	Src/CSharpier.Cli/CommandLineFormatter.cs
#	Src/CSharpier.Playground/Controllers/FormatController.cs
#	Src/CSharpier/CSharpFormatter.cs
#	Src/CSharpier/SyntaxPrinter/FormattingContext.cs
#	Src/CSharpier/SyntaxPrinter/Token.cs
# Conflicts:
#	.editorconfig
#	Src/CSharpier.Cli.Tests/CliTests.cs
#	Src/CSharpier.Cli/EditorConfig/Section.cs
#	Src/CSharpier.Playground/Controllers/FormatController.cs
#	Src/CSharpier.Tests/CodeFormatterTests.cs
#	Src/CSharpier/CSharpFormatter.cs
#	Src/CSharpier/CodeFormatterResult.cs
#	Src/SyntaxFinder/Walkers/ModifiersWalker.cs
#	Src/SyntaxFinder/Walkers/ObjectInitializerWalker.cs
#	Src/SyntaxFinder/Walkers/SpreadWalker.cs
@belav
Copy link
Owner Author

belav commented Dec 26, 2024

@shocklateboy92 this is a bit too big to probably give a proper review, but it is mostly ready. If you have any feedback on the notes below I think that is more important than code review.

Things to do before actually releasing

  • address a couple TODOs I still had in code
  • clean up a lot of the commented out stuff from porting prettiers code for xml to c#
  • write up all the doc for this
  • maybe add some kind of xml validation to make sure csharpier isn't losing anything when it formats xml - I feel like I ran into a ton of edge cases in csharpier-repos so I don't know if it is worth the effort to do.

The most important bits

  • adds formatting for xml files by default
  • tries to format ".csproj", ".props", ".targets", ".xml", ".config" as if they were xml
    • If one of them is not valid xml it is treated as a warning.
  • defaults to 2 spaces for indentation instead of 4
  • defaults to 100 width - I am tempted to make this higher. When reviewing all the files in csharpier-repos that ends up breaking a lot of elements with two attributes that look fine on a single line.
  • does not break the starting tag for element with a single attribute
  • does not break an element with no attributes and a single text child (with no new lines in it)
<!-- examples of the two above rules, they appear a lot in csprojs -->
<Project>
  <Reference Include="Newtonsoft.Json.Bson, Version=1.0.0.0, Culture=neutral, PublicKeyToken=30ad4fe6b2a6aeed, processorArchitecture=MSIL">
    <HintPath>..\..\packages\Newtonsoft.Json.Bson.1.0.2\lib\net45\Newtonsoft.Json.Bson.dll</HintPath>
  </Reference>
  <AspNetCoreShippingAssembly Include="Microsoft.AspNetCore.Server.Kestrel.Transport.NamedPipes.SomeLongerName" />
</Project>
  • does not add/remove new lines in attributes. new lines are used often by csproj conditions to make them more readable, but officially xml spec calls for turning them into a single space
  • treats whitespace as strict within a text element
<!-- given an element like this -->
<ElementWithAttribute Attribute="AttributeValue__________________">TextValue</ElementWithAttribute>
<!-- breaks this way to not add new lines -->
<ElementWithAttribute Attribute="AttributeValue__________________"
  >TextValue</ElementWithAttribute
>
<!-- vs treating it as non-strict and adding line breaks wihin the text -->
<ElementWithAttribute Attribute="AttributeValue__________________">
  TextValue
</ElementWithAttribute>

Some real world examples

@belav belav changed the base branch from main to 1.0.0 December 28, 2024 02:31
@belav belav changed the title Adding csproj formatting Adding XML formatting Dec 28, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Support formatting .csproj files and other types of xml
2 participants