5 Commits

Author SHA1 Message Date
boxqkrtm
0a98b387a9 Merge pull request #8 from zhust2003/master
feat: Support existing Cursor workspace detection (windows)
2025-04-24 13:41:37 +09:00
boxqkrtm
1228f59ed2 Update package.json
version change to 2.0.24
2025-04-24 13:41:23 +09:00
boxqkrtm
a24588a87a fix: Change package.json Name to Avoid Installation Warning 2025-04-24 13:33:25 +09:00
boxqkrtm
f1148f1205 fix: avoid launching cursor without folder context 2025-04-24 13:23:49 +09:00
Dalton
cac0c13658 Enhance Cursor instance management
Improve the handling of multiple Cursor instances by implementing better workspace detection and reuse of existing windows.
2025-03-20 18:40:05 +08:00
3 changed files with 139 additions and 5 deletions

View File

@@ -8,6 +8,13 @@ using System.Diagnostics;
using System.Text; using System.Text;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
using System.Runtime.InteropServices;
using System.ComponentModel;
using Debug = UnityEngine.Debug;
using System.IO;
using SimpleJSON;
using System.Collections.Generic;
using System.Linq;
namespace Microsoft.Unity.VisualStudio.Editor namespace Microsoft.Unity.VisualStudio.Editor
{ {
@@ -108,5 +115,85 @@ namespace Microsoft.Unity.VisualStudio.Editor
sb?.Append(data); sb?.Append(data);
} }
public static string[] GetProcessWorkspaces(Process process)
{
if (process == null)
return null;
try
{
var workspaces = new List<string>();
var userProfile = Environment.GetFolderPath(Environment.SpecialFolder.UserProfile);
var cursorStoragePath = Path.Combine(userProfile, "AppData", "Roaming", "cursor", "User", "workspaceStorage");
if (Directory.Exists(cursorStoragePath))
{
foreach (var workspaceDir in Directory.GetDirectories(cursorStoragePath))
{
try
{
var workspaceStatePath = Path.Combine(workspaceDir, "workspace.json");
if (File.Exists(workspaceStatePath))
{
var content = File.ReadAllText(workspaceStatePath);
if (!string.IsNullOrEmpty(content))
{
var workspace = JSONNode.Parse(content);
if (workspace != null)
{
var folder = workspace["folder"];
if (folder != null && !string.IsNullOrEmpty(folder.Value))
{
var workspacePath = folder.Value;
if (workspacePath.StartsWith("file:///"))
{
workspacePath = Uri.UnescapeDataString(workspacePath.Substring(8));
workspaces.Add(workspacePath);
}
}
}
}
}
var windowStatePath = Path.Combine(workspaceDir, "window.json");
if (File.Exists(windowStatePath))
{
var content = File.ReadAllText(windowStatePath);
if (!string.IsNullOrEmpty(content))
{
var windowState = JSONNode.Parse(content);
if (windowState != null)
{
var workspace = windowState["workspace"];
if (workspace != null && !string.IsNullOrEmpty(workspace.Value))
{
var workspacePath = workspace.Value;
if (workspacePath.StartsWith("file:///"))
{
workspacePath = Uri.UnescapeDataString(workspacePath.Substring(8));
workspaces.Add(workspacePath);
}
}
}
}
}
}
catch (Exception ex)
{
Debug.LogWarning($"[Cursor] Error reading workspace state file: {ex.Message}");
continue;
}
}
}
return workspaces.Distinct().ToArray();
}
catch (Exception ex)
{
Debug.LogError($"[Cursor] Error getting workspace directory: {ex.Message}");
return null;
}
}
} }
} }

View File

@@ -12,6 +12,7 @@ using System.Text.RegularExpressions;
using UnityEngine; using UnityEngine;
using SimpleJSON; using SimpleJSON;
using IOPath = System.IO.Path; using IOPath = System.IO.Path;
using Debug = UnityEngine.Debug;
namespace Microsoft.Unity.VisualStudio.Editor { namespace Microsoft.Unity.VisualStudio.Editor {
internal class VisualStudioCursorInstallation : VisualStudioInstallation { internal class VisualStudioCursorInstallation : VisualStudioInstallation {
@@ -448,6 +449,36 @@ namespace Microsoft.Unity.VisualStudio.Editor {
} }
} }
private Process FindRunningCursorWithSolution(string solutionPath) {
var directory = IOPath.GetDirectoryName(solutionPath);
var processes = Process.GetProcessesByName("cursor");
var normalizedTargetPath = directory.Replace('\\', '/').TrimEnd('/').ToLowerInvariant();
foreach (var process in processes) {
try {
var workspaces = ProcessRunner.GetProcessWorkspaces(process);
if (workspaces != null && workspaces.Length > 0) {
foreach (var workspace in workspaces) {
var normalizedWorkspaceDir = workspace.Replace('\\', '/').TrimEnd('/').ToLowerInvariant();
if (string.Equals(normalizedWorkspaceDir, normalizedTargetPath, StringComparison.OrdinalIgnoreCase) ||
normalizedTargetPath.StartsWith(normalizedWorkspaceDir + "/", StringComparison.OrdinalIgnoreCase) ||
normalizedWorkspaceDir.StartsWith(normalizedTargetPath + "/", StringComparison.OrdinalIgnoreCase))
{
return process;
}
}
}
}
catch (Exception ex) {
Debug.LogError($"[Cursor] Error checking process: {ex}");
continue;
}
}
return null;
}
public override bool Open(string path, int line, int column, string solution) { public override bool Open(string path, int line, int column, string solution) {
line = Math.Max(1, line); line = Math.Max(1, line);
column = Math.Max(0, column); column = Math.Max(0, column);
@@ -455,10 +486,26 @@ namespace Microsoft.Unity.VisualStudio.Editor {
var directory = IOPath.GetDirectoryName(solution); var directory = IOPath.GetDirectoryName(solution);
var application = Path; var application = Path;
ProcessRunner.Start(string.IsNullOrEmpty(path) ? var existingProcess = FindRunningCursorWithSolution(directory);
ProcessStartInfoFor(application, $"\"{directory}\"") : if (existingProcess != null) {
ProcessStartInfoFor(application, $"\"{directory}\" -g \"{path}\":{line}:{column}")); try {
var args = string.IsNullOrEmpty(path) ?
$"--reuse-window \"{directory}\"" :
$"--reuse-window -g \"{path}\":{line}:{column}";
ProcessRunner.Start(ProcessStartInfoFor(application, args));
return true;
}
catch (Exception ex) {
Debug.LogError($"[Cursor] Error using existing instance: {ex}");
}
}
var newArgs = string.IsNullOrEmpty(path) ?
$"--new-window \"{directory}\"" :
$"--new-window \"{directory}\" -g \"{path}\":{line}:{column}";
ProcessRunner.Start(ProcessStartInfoFor(application, newArgs));
return true; return true;
} }

View File

@@ -1,8 +1,8 @@
{ {
"name": "com.unity.ide.cursor", "name": "com.boxqkrtm.ide.cursor",
"displayName": "Cursor Editor", "displayName": "Cursor Editor",
"description": "Cursor editor integration for supporting Cursor as code editor for unity. Adds support for generating csproj files for intellisense purposes, auto discovery of installations, etc.", "description": "Cursor editor integration for supporting Cursor as code editor for unity. Adds support for generating csproj files for intellisense purposes, auto discovery of installations, etc.",
"version": "2.0.23", "version": "2.0.24",
"unity": "2019.4", "unity": "2019.4",
"unityRelease": "25f1", "unityRelease": "25f1",
"dependencies": { "dependencies": {