I have a Silverlight 3 project with something like this:
You have asked specific sub-questions with a view to achieving your overall goal, I presume you want to learn about MSBuild, rather than get a rote answer to your overall task (which is what you are gonna get from many other people due to your bounty), so I will answer your individual questions and then attempt to roll them all up into a solution.
So let's say you want to convert all .jpg files to .png.
Create a sub-list from the content item list based on extension:
<ItemGroup>
<Sublist Include="@(Content)" Condition=" '%(Extension)' == '.jpg' " />
</ItemGroup>
Receive the path of the item in a task.
Two ways - depends on the input that your task can accept. This way is like a "foreach" over each item in Sublist, and I would tend to use it with an Exec task:
<Exec Command="convert.exe /Input:%(Sublist.FullPath)" />
Specifying an output path also depends on the .exe or the task that you are and what an output path means to a particular task:
is it a directory, or just a filename with a different extension. But I'll assume you want to output files with the same name, but a different extension:
<Exec Command="convert.exe "%(Sublist.FullPath)" "%(Sublist.RootDir)%(Sublist.Directory)%(Sublist.Filename).png"" />
How to rebuild the png if the jpg changes (or is cleaned).
Well, this is using the Inputs and Outputs attribute of the containing target element where our convert command is executed. Inputs specifies what the source files are, and outputs specifies what the target will produce. MSBuild then compares the datetime of the Inputs with the datetime of the output and if they are out of date, then the outputs get rebuilt
<Target Name="ConvertJpg"
Inputs="@(Content)"
Outputs="@(Content -> '%(RootDir)%(Directory)%(Filename).png' )"
Condition=" '%(Extension)' == '.jpg' "
Lastly, you correctly have spotted that you need to re-inject the generated .png files back into the @Content item group - well, that's easy, you just Include them into the Content item. Recall that Sublist contains .jpg files - we want those files but with a .png ending. We also do NOT want the .jpg files in the Content item group once a png has been generated
<Content Remove="@(Sublist)" />
<Content Include="@(Sublist -> '%(RootDir)%(Directory)%(Filename).png' )" />
So summing up, your target would looks something like this I believe:
<Target Name="ConvertJpg"
Inputs="@(Content)"
Outputs="@(Content -> '%(RootDir)%(Directory)%(Filename).png' )"
Condition=" '%(Extension)' == '.jpg' "
<ItemGroup>
<Sublist Include="@(Content)" Condition=" '%(Extension)' == '.jpg' " />
</ItemGroup>
<Exec Command="convert.exe /Input:%(Sublist.FullPath) Output=%(Sublist.RootDir)%(Sublist.Directory)%(Sublist.Filename).png" />
<Content Remove="@(Sublist)" />
<Content Include="@(Sublist -> '%(RootDir)%(Directory)%(Filename).png' )" />
</Target>
By the way, ImageMagik has a command line tool that will convert jpg to png...
I believe you want something like this:
<ItemGroup>
<JPGContent Include="foo.jpg" />
</ItemGroup>
<Target Name="BeforeBuild"
Inputs="@(JPGContent)" Outputs="%(JPGContent.Filename).png">
<!-- replace this with call to jpg->png converter -->
<Exec Command="copy %22%(JPGContent.FullPath)%22 %(JPGContent.Filename).png" />
<ItemGroup>
<Content Include="%(JPGContent.Filename).png" />
</ItemGroup>
</Target>
<Target Name="AfterBuild">
<!-- just demoing that 'Content' now has right value -->
<Warning Text ="Content=@(Content)" />
</Target>
where you specify a new JPGContent
BuildAction that it converts.
(Possibly see also http://msdn.microsoft.com/en-us/library/ms164313.aspx and note that %22
is just a way to embed a quotation mark in an attribute.)
Instead of creating a MSBuild task you can also create a Custom tool and specify this in Custom Tool in properties of your image file.
"a custom tool will transform a file at design time and will place the output of the transformation into another file"
e.g. DataSet. it has its own transformation tool through which we get a class to use in our application.
the advantage of this approach is you can use the generated file at design time as well.
as sample implementation can be found at http://www.drewnoakes.com/snippets/WritingACustomCodeGeneratorToolForVisualStudio/