I'm frequently adding a lot of content files (mostly images and js) to my ASP.NET project. I'm using VS publish system, and on publish, new files are not published until I include them in the project. I would like to auto include all files in specified directory. Is there a way to specify which directories should be auto-included in csproj file or anywhere else?
-
see: this may help you stackoverflow.com/questions/1743432/…Saar– Saar2010-03-31 08:28:07 +00:00Commented Mar 31, 2010 at 8:28
-
1not exactly what I'm looking forMarko– Marko2010-03-31 10:53:05 +00:00Commented Mar 31, 2010 at 10:53
-
I updated my answer concering your issue when modifying the folder inside your vs solution browser.Filburt– Filburt2010-04-07 20:46:24 +00:00Commented Apr 7, 2010 at 20:46
9 Answers
Old thread, I know, but I found a way to do this that I keep forgetting, and on my search to find it one last time, I stumbled upon this question. The best way I've found to this is is to use the BeforeBuild target in the .csproj file.
<Target Name="BeforeBuild">
<ItemGroup>
<Content Include="**\*.less" />
</ItemGroup>
</Target>
VS 2010 won't touch this section, and it ensures that your files are included as content every time the project is built.
12 Comments
**\*.less ?**\*.less means include all *.less files in all directories. In MSBUILD speak, ** means 'all directories recursively'You simply can extend your website .csproj file. Just add your content root folder with a recursive wildcard:
...
<ItemGroup>
<!-- your normal project content -->
<Content Include="Default.aspx" />
<!-- your static content you like to publish -->
<Content Include="Images\**\*.*" />
</ItemGroup>
...
Doing so makes this folder and all content below visible inside your solution browser.
If you try to hide the folder inside the solution browser by specifying
<Content Include="Images\**.*.*">
<Visible>false</Visible>
</Content>
it will not be published.
Update
As you already discovered the wildcard will be replaced as soon as you touch the folder inside your solution because VS projects are not designed to contain arbitrary content.
So you will have to make sure the folder and its contents are never modified from within VS - adding or removing files can only be done on the file system ... which is what you wanted as i understood your question.
It would be easier if the folder could be hidden in VS but i couldn't find a way to hide it AND publish.
Another unsuccessful approach was to include the folder by a CreateItem Task.
This resulted in the contents of folder being published to \bin\app.publish\... and could not be convinced to publish it together with the content items inside the .csproj so i did not present it in my answer.
3 Comments
<Content Include="Images\**\*.*" /> it worked. Once you add more images the .csproj is changed and is back to listing all files in the images/ ... and the <Content Include="Images**." /> is gone.For those having issues using Chris' answer, this is the solution for Visual Studio 2012 and newer:
<Target Name="ContentsBeforeBuild" AfterTargets="BeforeBuild">
<ItemGroup>
<Content Include="images\**" />
</ItemGroup>
</Target>
As Chris mentioned in his answer - Visual Studio will not touch this <Target> section, even if you manually fiddle around (adding/removing files) with the target directory.
Please note that you should include a subdirectory where the files are located (in the case above, it's images). Visual Studio/MSBuild will place those files in the same directory within the project structure. If you don't use a subdirectory, the files will be placed at the root of the project structure.
For a quick explanation of the wildcards:
**means everything recursively (files, subdirectories, and files within those)*.extwill include all files with extensionextwithin the top-level directory, but not subdirectories- For example,
*.extcould be*.png,*.js, etc. Any file extension will work
- For example,
**\*.extwill include all files with extensionextfrom the top-level directory and all subdirectories.- See the answer from How do I use Nant/Ant naming patterns? for a more complete explanation with examples.
For completion, please note that there is a difference between using <Target> and not using it.
With the <Target> approach, Visual Studio will not show the files within the Solution Explorer.
<Target Name="ContentsBeforeBuild" AfterTargets="BeforeBuild">
<ItemGroup>
<Content Include="images\**" />
</ItemGroup>
</Target>
The non-<Target> approach will instruct Visual Studio to show the files within the Solution Explorer. The drawback with this one is that any manipulation of the automatic directories will cause Visual Studio to override the wildcard entry. It should also be noted that the approach below will only update the Solution Explorer upon opening the Solution/Project in VS. Even the Solution Explorer's "refresh" toolbar button won't do it.
<ItemGroup>
<Content Include="images\**" />
</ItemGroup>
9 Comments
.. AfterTargets="BeforeBuild". Meaning, yuour custom target must execute after BeforeBuild, but it does not specify how much after. Though, my mistake, by the current target ordering algorithm it should be ok: msdn.microsoft.com/en-us/library/ee216359.aspxYou can use the framework's System.IO.Directory.GetFile(string) method and its overloads to recursively include all files.
<ItemGroup>
<Content Include="$([System.IO.Directory]::GetFiles('$(ProjectDir)Scripts\', '*.js', SearchOption.AllDirectories))" />
<Content Include="$([System.IO.Directory]::GetFiles('$(ProjectDir)Images\', '*.png', SearchOption.AllDirectories))" />
</ItemGroup>
4 Comments
Include="**\*.ext" with wildcards.You can add folder with files recursively like this:
<ItemGroup>
<Content Include="somefolder\*.txt">
<Link>%(RecursiveDir)myown_somefolder\%(Filename)%(Extension)</Link>
</Content>
</ItemGroup>
1 Comment
You can add files with links like this, they are searchable, view-able, but they do not checkout if you try to change them, also visual studio leaves the wildcards in place:
<ItemGroup>
<Content Include="..\Database Schema\Views\*.sql">
<Link>Views\*.sql</Link>
</Content>
</ItemGroup>
This goes inside the .proj file.
3 Comments
I've written up how I was able to get the content includes created with a small powershell script:
$folders = Get-ChildItem .\ -r -Directory
$filtered = $folders |Select-Object @{Name='FullName';Expression={$_.fullname.replace($pwd,'')}}, @{Name='FolderDepth';Expression={($_.fullname.Split('\').Count) - ($Pwd.Path.split('\').count)}} | Sort-Object -Descending FullName,folderDepth
$basefolders = $filtered | Where-Object{$_.folderdepth -eq 1}
$basefoldersobj = @()
foreach($basefolder in $basefolders)
{
$basefoldername =$baseFolder.fullname
$filteredbase = $filtered -match "\$basefoldername\\" | Sort-Object -Descending FolderDepth | Select-Object -first 1
if($filteredbase -eq $null)
{
$filteredbase = $filtered -match "\$basefoldername" | Sort-Object -Descending FolderDepth | Select-Object -first 1
}
$obj = New-Object psobject
Add-Member -InputObject $obj -MemberType NoteProperty -Name 'Folder' -Value $basefolder.fullname.trim('\')
Add-member -InputObject $obj -MemberType NoteProperty -Name 'DeepestPath' -Value $filteredbase.folderDepth
$basefoldersobj += $obj
}
$include = '*.*'
foreach($bfolderObj in $basefoldersobj)
{
$includecount = ''
$includecount = "\$include" * ($bfolderObj.Deepestpath)
Write-Output "<content Include=`"$($bfolderObj.folder)$includecount`" /> "
}
This should produce the necessary include statement at the powershell prompt
Comments
I realized that best solution for this is manually add files, one by one. If you have hundreds of them as I did it was just a matter of few hours. Funny that even in 2016 with VS 2015 this serious problem is still not solved. Ahh, how I love Xcode.