Skip to main content
Glama
McpSettingsSection.cs8.59 kB
using System; using System.IO; using System.Runtime.InteropServices; using MCPForUnity.Editor.Constants; using MCPForUnity.Editor.Helpers; using MCPForUnity.Editor.Services; using UnityEditor; using UnityEditor.UIElements; using UnityEngine; using UnityEngine.UIElements; namespace MCPForUnity.Editor.Windows.Components.Settings { /// <summary> /// Controller for the Settings section of the MCP For Unity editor window. /// Handles version display, debug logs, validation level, and advanced path overrides. /// </summary> public class McpSettingsSection { // UI Elements private Label versionLabel; private Toggle debugLogsToggle; private EnumField validationLevelField; private Label validationDescription; private Foldout advancedSettingsFoldout; private TextField uvxPathOverride; private Button browseUvxButton; private Button clearUvxButton; private VisualElement uvxPathStatus; private TextField gitUrlOverride; private Button clearGitUrlButton; // Data private ValidationLevel currentValidationLevel = ValidationLevel.Standard; // Events public event Action OnGitUrlChanged; public event Action OnHttpServerCommandUpdateRequested; // Validation levels private enum ValidationLevel { Basic, Standard, Comprehensive, Strict } public VisualElement Root { get; private set; } public McpSettingsSection(VisualElement root) { Root = root; CacheUIElements(); InitializeUI(); RegisterCallbacks(); } private void CacheUIElements() { versionLabel = Root.Q<Label>("version-label"); debugLogsToggle = Root.Q<Toggle>("debug-logs-toggle"); validationLevelField = Root.Q<EnumField>("validation-level"); validationDescription = Root.Q<Label>("validation-description"); advancedSettingsFoldout = Root.Q<Foldout>("advanced-settings-foldout"); uvxPathOverride = Root.Q<TextField>("uv-path-override"); browseUvxButton = Root.Q<Button>("browse-uv-button"); clearUvxButton = Root.Q<Button>("clear-uv-button"); uvxPathStatus = Root.Q<VisualElement>("uv-path-status"); gitUrlOverride = Root.Q<TextField>("git-url-override"); clearGitUrlButton = Root.Q<Button>("clear-git-url-button"); } private void InitializeUI() { UpdateVersionLabel(); bool debugEnabled = EditorPrefs.GetBool(EditorPrefKeys.DebugLogs, false); debugLogsToggle.value = debugEnabled; McpLog.SetDebugLoggingEnabled(debugEnabled); validationLevelField.Init(ValidationLevel.Standard); int savedLevel = EditorPrefs.GetInt(EditorPrefKeys.ValidationLevel, 1); currentValidationLevel = (ValidationLevel)Mathf.Clamp(savedLevel, 0, 3); validationLevelField.value = currentValidationLevel; UpdateValidationDescription(); advancedSettingsFoldout.value = false; gitUrlOverride.value = EditorPrefs.GetString(EditorPrefKeys.GitUrlOverride, ""); } private void RegisterCallbacks() { debugLogsToggle.RegisterValueChangedCallback(evt => { McpLog.SetDebugLoggingEnabled(evt.newValue); }); validationLevelField.RegisterValueChangedCallback(evt => { currentValidationLevel = (ValidationLevel)evt.newValue; EditorPrefs.SetInt(EditorPrefKeys.ValidationLevel, (int)currentValidationLevel); UpdateValidationDescription(); }); browseUvxButton.clicked += OnBrowseUvxClicked; clearUvxButton.clicked += OnClearUvxClicked; gitUrlOverride.RegisterValueChangedCallback(evt => { string url = evt.newValue?.Trim(); if (string.IsNullOrEmpty(url)) { EditorPrefs.DeleteKey(EditorPrefKeys.GitUrlOverride); } else { EditorPrefs.SetString(EditorPrefKeys.GitUrlOverride, url); } OnGitUrlChanged?.Invoke(); OnHttpServerCommandUpdateRequested?.Invoke(); }); clearGitUrlButton.clicked += () => { gitUrlOverride.value = string.Empty; EditorPrefs.DeleteKey(EditorPrefKeys.GitUrlOverride); OnGitUrlChanged?.Invoke(); OnHttpServerCommandUpdateRequested?.Invoke(); }; } public void UpdatePathOverrides() { var pathService = MCPServiceLocator.Paths; bool hasOverride = pathService.HasUvxPathOverride; string uvxPath = hasOverride ? pathService.GetUvxPath() : null; uvxPathOverride.value = hasOverride ? (uvxPath ?? "(override set but invalid)") : "uvx (uses PATH)"; uvxPathStatus.RemoveFromClassList("valid"); uvxPathStatus.RemoveFromClassList("invalid"); if (hasOverride) { if (!string.IsNullOrEmpty(uvxPath) && File.Exists(uvxPath)) { uvxPathStatus.AddToClassList("valid"); } else { uvxPathStatus.AddToClassList("invalid"); } } else { uvxPathStatus.AddToClassList("valid"); } gitUrlOverride.value = EditorPrefs.GetString(EditorPrefKeys.GitUrlOverride, ""); } private void UpdateVersionLabel() { string currentVersion = AssetPathUtility.GetPackageVersion(); versionLabel.text = $"v{currentVersion}"; var updateCheck = MCPServiceLocator.Updates.CheckForUpdate(currentVersion); if (updateCheck.UpdateAvailable && !string.IsNullOrEmpty(updateCheck.LatestVersion)) { versionLabel.text = $"\u2191 v{currentVersion} (Update available: v{updateCheck.LatestVersion})"; versionLabel.style.color = new Color(1f, 0.7f, 0f); versionLabel.tooltip = $"Version {updateCheck.LatestVersion} is available. Update via Package Manager.\n\nGit URL: https://github.com/CoplayDev/unity-mcp.git?path=/MCPForUnity"; } else { versionLabel.style.color = StyleKeyword.Null; versionLabel.tooltip = $"Current version: {currentVersion}"; } } private void UpdateValidationDescription() { validationDescription.text = GetValidationLevelDescription((int)currentValidationLevel); } private string GetValidationLevelDescription(int index) { return index switch { 0 => "Only basic syntax checks (braces, quotes, comments)", 1 => "Syntax checks + Unity best practices and warnings", 2 => "All checks + semantic analysis and performance warnings", 3 => "Full semantic validation with namespace/type resolution (requires Roslyn)", _ => "Standard validation" }; } private void OnBrowseUvxClicked() { string suggested = RuntimeInformation.IsOSPlatform(OSPlatform.OSX) ? "/opt/homebrew/bin" : Environment.GetFolderPath(Environment.SpecialFolder.ProgramFiles); string picked = EditorUtility.OpenFilePanel("Select uv Executable", suggested, ""); if (!string.IsNullOrEmpty(picked)) { try { MCPServiceLocator.Paths.SetUvxPathOverride(picked); UpdatePathOverrides(); McpLog.Info($"uv path override set to: {picked}"); } catch (Exception ex) { EditorUtility.DisplayDialog("Invalid Path", ex.Message, "OK"); } } } private void OnClearUvxClicked() { MCPServiceLocator.Paths.ClearUvxPathOverride(); UpdatePathOverrides(); McpLog.Info("uv path override cleared"); } } }

Latest Blog Posts

MCP directory API

We provide all the information about MCP servers via our MCP API.

curl -X GET 'https://glama.ai/api/mcp/v1/servers/CoplayDev/unity-mcp'

If you have feedback or need assistance with the MCP directory API, please join our Discord server