[3.2] Sync csproj when files are changed from the Godot FileSystem dock

This commit is contained in:
Ignacio Etcheverry 2020-03-18 15:05:50 +01:00
parent 110523fecc
commit d1a5f8dbf2
7 changed files with 186 additions and 33 deletions

View file

@ -2881,22 +2881,6 @@ void CSharpScript::initialize_for_managed_type(Ref<CSharpScript> p_script, GDMon
bool CSharpScript::can_instance() const { bool CSharpScript::can_instance() const {
#ifdef TOOLS_ENABLED
if (Engine::get_singleton()->is_editor_hint()) {
// Hack to lower the risk of attached scripts not being added to the C# project
if (!get_path().empty() && get_path().find("::") == -1) { // Ignore if built-in script. Can happen if the file is deleted...
if (_create_project_solution_if_needed()) {
CSharpProject::add_item(GodotSharpDirs::get_project_csproj_path(),
"Compile",
ProjectSettings::get_singleton()->globalize_path(get_path()));
} else {
ERR_PRINTS("C# project could not be created; cannot add file: '" + get_path() + "'.");
}
}
}
#endif
#ifdef TOOLS_ENABLED #ifdef TOOLS_ENABLED
bool extra_cond = tool || ScriptServer::is_scripting_enabled(); bool extra_cond = tool || ScriptServer::is_scripting_enabled();
#else #else

View file

@ -30,7 +30,7 @@ namespace GodotTools.Core
path = string.Join(Path.DirectorySeparatorChar.ToString(), parts).Trim(); path = string.Join(Path.DirectorySeparatorChar.ToString(), parts).Trim();
return rooted ? Path.DirectorySeparatorChar.ToString() + path : path; return rooted ? Path.DirectorySeparatorChar + path : path;
} }
private static readonly string driveRoot = Path.GetPathRoot(Environment.CurrentDirectory); private static readonly string driveRoot = Path.GetPathRoot(Environment.CurrentDirectory);

View file

@ -1,5 +1,7 @@
using GodotTools.Core; using GodotTools.Core;
using System; using System;
using System.Collections.Generic;
using System.IO;
using DotNet.Globbing; using DotNet.Globbing;
using Microsoft.Build.Construction; using Microsoft.Build.Construction;
@ -7,16 +9,15 @@ namespace GodotTools.ProjectEditor
{ {
public static class ProjectExtensions public static class ProjectExtensions
{ {
public static bool HasItem(this ProjectRootElement root, string itemType, string include) public static ProjectItemElement FindItemOrNull(this ProjectRootElement root, string itemType, string include, bool noCondition = false)
{ {
GlobOptions globOptions = new GlobOptions(); GlobOptions globOptions = new GlobOptions {Evaluation = {CaseInsensitive = false}};
globOptions.Evaluation.CaseInsensitive = false;
string normalizedInclude = include.NormalizePath(); string normalizedInclude = include.NormalizePath();
foreach (var itemGroup in root.ItemGroups) foreach (var itemGroup in root.ItemGroups)
{ {
if (itemGroup.Condition.Length != 0) if (noCondition && itemGroup.Condition.Length != 0)
continue; continue;
foreach (var item in itemGroup.Items) foreach (var item in itemGroup.Items)
@ -27,20 +28,79 @@ namespace GodotTools.ProjectEditor
var glob = Glob.Parse(item.Include.NormalizePath(), globOptions); var glob = Glob.Parse(item.Include.NormalizePath(), globOptions);
if (glob.IsMatch(normalizedInclude)) if (glob.IsMatch(normalizedInclude))
return item;
}
}
return null;
}
public static ProjectItemElement FindItemOrNullAbs(this ProjectRootElement root, string itemType, string include, bool noCondition = false)
{ {
GlobOptions globOptions = new GlobOptions {Evaluation = {CaseInsensitive = false}};
string normalizedInclude = Path.GetFullPath(include).NormalizePath();
foreach (var itemGroup in root.ItemGroups)
{
if (noCondition && itemGroup.Condition.Length != 0)
continue;
foreach (var item in itemGroup.Items)
{
if (item.ItemType != itemType)
continue;
var glob = Glob.Parse(Path.GetFullPath(item.Include).NormalizePath(), globOptions);
if (glob.IsMatch(normalizedInclude))
return item;
}
}
return null;
}
public static IEnumerable<ProjectItemElement> FindAllItemsInFolder(this ProjectRootElement root, string itemType, string folder)
{
string absFolderNormalizedWithSep = Path.GetFullPath(folder).NormalizePath() + Path.DirectorySeparatorChar;
foreach (var itemGroup in root.ItemGroups)
{
foreach (var item in itemGroup.Items)
{
if (item.ItemType != itemType)
continue;
string absPathNormalized = Path.GetFullPath(item.Include).NormalizePath();
if (absPathNormalized.StartsWith(absFolderNormalizedWithSep))
yield return item;
}
}
}
public static bool HasItem(this ProjectRootElement root, string itemType, string include, bool noCondition = false)
{
return root.FindItemOrNull(itemType, include, noCondition) != null;
}
public static bool AddItemChecked(this ProjectRootElement root, string itemType, string include)
{
if (!root.HasItem(itemType, include, noCondition: true))
{
root.AddItem(itemType, include);
return true; return true;
} }
}
}
return false; return false;
} }
public static bool AddItemChecked(this ProjectRootElement root, string itemType, string include) public static bool RemoveItemChecked(this ProjectRootElement root, string itemType, string include)
{ {
if (!root.HasItem(itemType, include)) var item = root.FindItemOrNullAbs(itemType, include);
if (item != null)
{ {
root.AddItem(itemType, include); item.Parent.RemoveChild(item);
return true; return true;
} }

View file

@ -22,6 +22,79 @@ namespace GodotTools.ProjectEditor
root.Save(); root.Save();
} }
public static void RenameItemInProjectChecked(string projectPath, string itemType, string oldInclude, string newInclude)
{
var dir = Directory.GetParent(projectPath).FullName;
var root = ProjectRootElement.Open(projectPath);
Debug.Assert(root != null);
var normalizedOldInclude = oldInclude.NormalizePath();
var normalizedNewInclude = newInclude.NormalizePath();
var item = root.FindItemOrNullAbs(itemType, normalizedOldInclude);
if (item == null)
return;
item.Include = normalizedNewInclude.RelativeToPath(dir).Replace("/", "\\");
root.Save();
}
public static void RemoveItemFromProjectChecked(string projectPath, string itemType, string include)
{
var dir = Directory.GetParent(projectPath).FullName;
var root = ProjectRootElement.Open(projectPath);
Debug.Assert(root != null);
var normalizedInclude = include.NormalizePath();
if (root.RemoveItemChecked(itemType, normalizedInclude))
root.Save();
}
public static void RenameItemsToNewFolderInProjectChecked(string projectPath, string itemType, string oldFolder, string newFolder)
{
var dir = Directory.GetParent(projectPath).FullName;
var root = ProjectRootElement.Open(projectPath);
Debug.Assert(root != null);
bool dirty = false;
var oldFolderNormalized = oldFolder.NormalizePath();
var newFolderNormalized = newFolder.NormalizePath();
string absOldFolderNormalized = Path.GetFullPath(oldFolderNormalized).NormalizePath();
string absNewFolderNormalized = Path.GetFullPath(newFolderNormalized).NormalizePath();
foreach (var item in root.FindAllItemsInFolder(itemType, oldFolderNormalized))
{
string absPathNormalized = Path.GetFullPath(item.Include).NormalizePath();
string absNewIncludeNormalized = absNewFolderNormalized + absPathNormalized.Substring(absOldFolderNormalized.Length);
item.Include = absNewIncludeNormalized.RelativeToPath(dir).Replace("/", "\\");
dirty = true;
}
if (dirty)
root.Save();
}
public static void RemoveItemsInFolderFromProjectChecked(string projectPath, string itemType, string folder)
{
var root = ProjectRootElement.Open(projectPath);
Debug.Assert(root != null);
var folderNormalized = folder.NormalizePath();
var itemsToRemove = root.FindAllItemsInFolder(itemType, folderNormalized).ToList();
if (itemsToRemove.Count > 0)
{
foreach (var item in itemsToRemove)
item.Parent.RemoveChild(item);
root.Save();
}
}
private static string[] GetAllFilesRecursive(string rootDirectory, string mask) private static string[] GetAllFilesRecursive(string rootDirectory, string mask)
{ {
string[] files = Directory.GetFiles(rootDirectory, mask, SearchOption.AllDirectories); string[] files = Directory.GetFiles(rootDirectory, mask, SearchOption.AllDirectories);

View file

@ -96,7 +96,7 @@ namespace GodotTools.Export
if (type != Internal.CSharpLanguageType) if (type != Internal.CSharpLanguageType)
return; return;
if (Path.GetExtension(path) != $".{Internal.CSharpLanguageExtension}") if (Path.GetExtension(path) != Internal.CSharpLanguageExtension)
throw new ArgumentException($"Resource of type {Internal.CSharpLanguageType} has an invalid file extension: {path}", nameof(path)); throw new ArgumentException($"Resource of type {Internal.CSharpLanguageType} has an invalid file extension: {path}", nameof(path));
// TODO What if the source file is not part of the game's C# project // TODO What if the source file is not part of the game's C# project

View file

@ -13,6 +13,7 @@ using JetBrains.Annotations;
using static GodotTools.Internals.Globals; using static GodotTools.Internals.Globals;
using File = GodotTools.Utils.File; using File = GodotTools.Utils.File;
using OS = GodotTools.Utils.OS; using OS = GodotTools.Utils.OS;
using Path = System.IO.Path;
namespace GodotTools namespace GodotTools
{ {
@ -154,6 +155,34 @@ namespace GodotTools
Instance.BottomPanel.BuildProjectPressed(); Instance.BottomPanel.BuildProjectPressed();
} }
private void _FileSystemDockFileMoved(string file, string newFile)
{
if (Path.GetExtension(file) == Internal.CSharpLanguageExtension)
{
ProjectUtils.RenameItemInProjectChecked(GodotSharpDirs.ProjectCsProjPath, "Compile",
ProjectSettings.GlobalizePath(file), ProjectSettings.GlobalizePath(newFile));
}
}
private void _FileSystemDockFileRemoved(string file)
{
if (Path.GetExtension(file) == Internal.CSharpLanguageExtension)
ProjectUtils.RemoveItemFromProjectChecked(GodotSharpDirs.ProjectCsProjPath, "Compile",
ProjectSettings.GlobalizePath(file));
}
private void _FileSystemDockFolderMoved(string oldFolder, string newFolder)
{
ProjectUtils.RenameItemsToNewFolderInProjectChecked(GodotSharpDirs.ProjectCsProjPath, "Compile",
ProjectSettings.GlobalizePath(oldFolder), ProjectSettings.GlobalizePath(newFolder));
}
private void _FileSystemDockFolderRemoved(string oldFolder)
{
ProjectUtils.RemoveItemsInFolderFromProjectChecked(GodotSharpDirs.ProjectCsProjPath, "Compile",
ProjectSettings.GlobalizePath(oldFolder));
}
public override void _Notification(int what) public override void _Notification(int what)
{ {
base._Notification(what); base._Notification(what);
@ -168,6 +197,13 @@ namespace GodotTools
// Once shown a first time, it can be seen again via the Mono menu - it doesn't have to be exclusive from that time on. // Once shown a first time, it can be seen again via the Mono menu - it doesn't have to be exclusive from that time on.
aboutDialog.PopupExclusive = false; aboutDialog.PopupExclusive = false;
} }
var fileSystemDock = GetEditorInterface().GetFileSystemDock();
fileSystemDock.Connect("files_moved", this, nameof(_FileSystemDockFileMoved));
fileSystemDock.Connect("file_removed", this, nameof(_FileSystemDockFileRemoved));
fileSystemDock.Connect("folder_moved", this, nameof(_FileSystemDockFolderMoved));
fileSystemDock.Connect("folder_removed", this, nameof(_FileSystemDockFolderRemoved));
} }
} }

View file

@ -8,7 +8,7 @@ namespace GodotTools.Internals
public static class Internal public static class Internal
{ {
public const string CSharpLanguageType = "CSharpScript"; public const string CSharpLanguageType = "CSharpScript";
public const string CSharpLanguageExtension = "cs"; public const string CSharpLanguageExtension = ".cs";
public static string UpdateApiAssembliesFromPrebuilt(string config) => public static string UpdateApiAssembliesFromPrebuilt(string config) =>
internal_UpdateApiAssembliesFromPrebuilt(config); internal_UpdateApiAssembliesFromPrebuilt(config);