diff --git a/DotRecast.sln b/DotRecast.sln
index 581b760..006fc1f 100644
--- a/DotRecast.sln
+++ b/DotRecast.sln
@@ -41,6 +41,10 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "DotRecast.Detour.TileCache.
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "DotRecast.Benchmark", "test\DotRecast.Benchmark\DotRecast.Benchmark.csproj", "{D1EFC625-D095-4208-98A2-112B73CB40B0}"
EndProject
+Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "tool", "tool", "{9C213609-BF13-4024-816E-A6ADD641DF24}"
+EndProject
+Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "DotRecast.Tool.PublishToUniRecast", "tool\DotRecast.Tool.PublishToUniRecast\DotRecast.Tool.PublishToUniRecast.csproj", "{1822BBA8-FE4E-4C5F-B9C0-3E20E6353446}"
+EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
@@ -118,6 +122,10 @@ Global
{D1EFC625-D095-4208-98A2-112B73CB40B0}.Debug|Any CPU.Build.0 = Debug|Any CPU
{D1EFC625-D095-4208-98A2-112B73CB40B0}.Release|Any CPU.ActiveCfg = Release|Any CPU
{D1EFC625-D095-4208-98A2-112B73CB40B0}.Release|Any CPU.Build.0 = Release|Any CPU
+ {1822BBA8-FE4E-4C5F-B9C0-3E20E6353446}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
+ {1822BBA8-FE4E-4C5F-B9C0-3E20E6353446}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {1822BBA8-FE4E-4C5F-B9C0-3E20E6353446}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {1822BBA8-FE4E-4C5F-B9C0-3E20E6353446}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(NestedProjects) = preSolution
{FFE40BBF-843B-41FA-8504-F4ABD166762E} = {8ED75CF7-A3D6-423D-8499-9316DD413DAD}
@@ -137,5 +145,6 @@ Global
{3CAA7306-088E-4373-A406-99755CC2B605} = {A7CB8D8B-70DA-4567-8316-0659FCAE1C73}
{10395C8F-DFBD-4263-8A20-EA3500A6E55A} = {A7CB8D8B-70DA-4567-8316-0659FCAE1C73}
{D1EFC625-D095-4208-98A2-112B73CB40B0} = {A7CB8D8B-70DA-4567-8316-0659FCAE1C73}
+ {1822BBA8-FE4E-4C5F-B9C0-3E20E6353446} = {9C213609-BF13-4024-816E-A6ADD641DF24}
EndGlobalSection
EndGlobal
diff --git a/tool/DotRecast.Tool.PublishToUniRecast/CsProj.cs b/tool/DotRecast.Tool.PublishToUniRecast/CsProj.cs
new file mode 100644
index 0000000..bd545ce
--- /dev/null
+++ b/tool/DotRecast.Tool.PublishToUniRecast/CsProj.cs
@@ -0,0 +1,15 @@
+namespace DotRecast.Tool.PublishToUniRecast;
+
+public class CsProj
+{
+ public readonly string RootPath;
+ public readonly string Name;
+ public readonly string TargetPath;
+
+ public CsProj(string rootPath, string name, string targetPath)
+ {
+ RootPath = rootPath;
+ Name = name;
+ TargetPath = targetPath;
+ }
+}
\ No newline at end of file
diff --git a/tool/DotRecast.Tool.PublishToUniRecast/DotRecast.Tool.PublishToUniRecast.csproj b/tool/DotRecast.Tool.PublishToUniRecast/DotRecast.Tool.PublishToUniRecast.csproj
new file mode 100644
index 0000000..70cfa9d
--- /dev/null
+++ b/tool/DotRecast.Tool.PublishToUniRecast/DotRecast.Tool.PublishToUniRecast.csproj
@@ -0,0 +1,9 @@
+
+
+
+ Exe
+ net6.0;net8.0
+ false
+
+
+
diff --git a/tool/DotRecast.Tool.PublishToUniRecast/Program.cs b/tool/DotRecast.Tool.PublishToUniRecast/Program.cs
new file mode 100644
index 0000000..e5300e0
--- /dev/null
+++ b/tool/DotRecast.Tool.PublishToUniRecast/Program.cs
@@ -0,0 +1,169 @@
+using System;
+using System.Collections.Generic;
+using System.Collections.Immutable;
+using System.IO;
+using System.Linq;
+
+namespace DotRecast.Tool.PublishToUniRecast;
+
+public static class Program
+{
+ public static void Main(string[] args)
+ {
+ var source = SearchDirectory("DotRecast");
+ var destination = SearchDirectory("UniRecast");
+
+ if (!Directory.Exists(source))
+ {
+ throw new Exception("not found source directory");
+ }
+
+ if (!Directory.Exists(destination))
+ {
+ throw new Exception("not found destination directory");
+ }
+
+ var ignorePaths = ImmutableArray.Create("bin", "obj");
+ var projs = ImmutableArray.Create(
+ // src
+ new CsProj("src", "DotRecast.Core", "Runtime"),
+ new CsProj("src", "DotRecast.Recast", "Runtime"),
+ new CsProj("src", "DotRecast.Detour", "Runtime"),
+ new CsProj("src", "DotRecast.Detour.Crowd", "Runtime"),
+ new CsProj("src", "DotRecast.Detour.Dynamic", "Runtime"),
+ new CsProj("src", "DotRecast.Detour.Extras", "Runtime"),
+ new CsProj("src", "DotRecast.Detour.TileCache", "Runtime"),
+ new CsProj("src", "DotRecast.Recast.Toolset", "Runtime")
+ );
+
+
+ foreach (var proj in projs)
+ {
+ var sourcePath = Path.Combine(source, proj.RootPath, $"{proj.Name}");
+ var destPath = Path.Combine(destination, $"{proj.TargetPath}", $"{proj.Name}");
+
+ SyncFiles(sourcePath, destPath, ignorePaths, "*.cs");
+ }
+
+ // // 몇몇 필요한 리소스 복사 하기
+ // string destResourcePath = destDotRecast + "/resources";
+ // if (!Directory.Exists(destResourcePath))
+ // {
+ // Directory.CreateDirectory(destResourcePath);
+ // }
+
+ // string sourceResourcePath = Path.Combine(dotRecastPath, "resources/nav_test.obj");
+ // File.Copy(sourceResourcePath, destResourcePath + "/nav_test.obj", true);
+ }
+
+ public static string SearchPath(string searchPath, int depth, out bool isDir)
+ {
+ isDir = false;
+
+ for (int i = 0; i < depth; ++i)
+ {
+ var relativePath = string.Join("", Enumerable.Range(0, i).Select(x => "../"));
+ var searchingPath = Path.Combine(relativePath, searchPath);
+ var fullSearchingPath = Path.GetFullPath(searchingPath);
+
+ if (File.Exists(fullSearchingPath))
+ {
+ return fullSearchingPath;
+ }
+
+ if (Directory.Exists(fullSearchingPath))
+ {
+ isDir = true;
+ return fullSearchingPath;
+ }
+ }
+
+ return string.Empty;
+ }
+
+ // only directory
+ public static string SearchDirectory(string dirname, int depth = 10)
+ {
+ var searchingPath = SearchPath(dirname, depth, out var isDir);
+ if (isDir)
+ {
+ return searchingPath;
+ }
+
+ var path = Path.GetDirectoryName(searchingPath) ?? string.Empty;
+ return path;
+ }
+
+ public static string SearchFile(string filename, int depth = 10)
+ {
+ var searchingPath = SearchPath(filename, depth, out var isDir);
+ if (!isDir)
+ {
+ return searchingPath;
+ }
+
+ return string.Empty;
+ }
+
+ private static void SyncFiles(string srcRootPath, string dstRootPath, IList ignoreFolders, string searchPattern = "*")
+ {
+ // 끝에서부터 이그노어 폴더일 경우 패스
+ var destLastFolderName = Path.GetFileName(dstRootPath);
+ if (ignoreFolders.Any(x => x == destLastFolderName))
+ return;
+
+ if (!Directory.Exists(dstRootPath))
+ Directory.CreateDirectory(dstRootPath);
+
+ // 소스파일 추출
+ var sourceFiles = Directory.GetFiles(srcRootPath, searchPattern).ToList();
+ var sourceFolders = Directory.GetDirectories(srcRootPath)
+ .Select(x => new DirectoryInfo(x))
+ .ToList();
+
+ // 대상 파일 추출
+ var destinationFiles = Directory.GetFiles(dstRootPath, searchPattern).ToList();
+ var destinationFolders = Directory.GetDirectories(dstRootPath)
+ .Select(x => new DirectoryInfo(x))
+ .ToList();
+
+ // 대상에 파일이 있는데, 소스에 없을 경우, 대상 파일을 삭제 한다.
+ foreach (var destinationFile in destinationFiles)
+ {
+ var destName = Path.GetFileName(destinationFile);
+ var found = sourceFiles.Any(x => Path.GetFileName(x) == destName);
+ if (found)
+ continue;
+
+ File.Delete(destinationFile);
+ Console.WriteLine($"delete file - {destinationFile}");
+ }
+
+ // 대상에 폴더가 있는데, 소스에 없을 경우, 대상 폴더를 삭제 한다.
+ foreach (var destinationFolder in destinationFolders)
+ {
+ var found = sourceFolders.Any(sourceFolder => sourceFolder.Name == destinationFolder.Name);
+ if (found)
+ continue;
+
+ Directory.Delete(destinationFolder.FullName, true);
+ Console.WriteLine($"delete folder - {destinationFolder.FullName}");
+ }
+
+ // 소스 파일을 복사 한다.
+ foreach (var sourceFile in sourceFiles)
+ {
+ var name = Path.GetFileName(sourceFile);
+ var dest = Path.Combine(dstRootPath, name);
+ File.Copy(sourceFile, dest, true);
+ Console.WriteLine($"copy - {sourceFile} => {dest}");
+ }
+
+ // 대상 폴더를 복사 한다
+ foreach (var sourceFolder in sourceFolders)
+ {
+ var dest = Path.Combine(dstRootPath, sourceFolder.Name);
+ SyncFiles(sourceFolder.FullName, dest, ignoreFolders, searchPattern);
+ }
+ }
+}
\ No newline at end of file