// Copyright (c) 2012-2021 Wojciech Figat. All rights reserved. #include "RiderCodeEditor.h" #include "Engine/Platform/FileSystem.h" #include "Engine/Core/Log.h" #include "Editor/Editor.h" #include "Editor/ProjectInfo.h" #include "Editor/Scripting/ScriptsBuilder.h" #include "Engine/Engine/Globals.h" #if PLATFORM_WINDOWS #include "Engine/Core/Collections/Sorting.h" #include "Engine/Platform/File.h" #include "Engine/Platform/Win32/IncludeWindowsHeaders.h" #include "Engine/Serialization/Json.h" namespace { struct RiderInstallation { String path; String version; RiderInstallation(const String& path_, const String& version_) : path(path_), version(version_) { } }; bool FindRegistryKeyItems(HKEY hKey, Array& results) { Char nameBuffer[256]; for (int32 i = 0;; i++) { const LONG result = RegEnumKeyW(hKey, i, nameBuffer, ARRAY_COUNT(nameBuffer)); if (result == ERROR_NO_MORE_ITEMS) break; if (result != ERROR_SUCCESS) return false; results.Add(nameBuffer); } return true; } void SearchDirectory(Array* installations, const String& directory) { if (!FileSystem::DirectoryExists(directory)) return; // Load product info Array productInfoData; const String productInfoPath = directory / TEXT("product-info.json"); if (File::ReadAllBytes(productInfoPath, productInfoData)) return; rapidjson_flax::Document document; document.Parse((char*)productInfoData.Get(), productInfoData.Count()); if (document.HasParseError()) return; // Find version auto versionMember = document.FindMember("version"); if (versionMember == document.MemberEnd()) return; // Find executable file path auto launchMember = document.FindMember("launch"); if (launchMember == document.MemberEnd() || !launchMember->value.IsArray() || launchMember->value.Size() == 0) return; auto launcherPathMember = launchMember->value[0].FindMember("launcherPath"); if (launcherPathMember == launchMember->value[0].MemberEnd()) return; auto launcherPath = launcherPathMember->value.GetText(); auto exePath = directory / launcherPath; if (!launcherPath.HasChars() || !FileSystem::FileExists(exePath)) return; installations->Add(New(exePath, versionMember->value.GetText())); } void SearchRegistry(Array* installations, HKEY root, const Char* key, const Char* valueName = TEXT("")) { // Open key HKEY keyH; if (RegOpenKeyExW(root, key, 0, KEY_READ, &keyH) != ERROR_SUCCESS) return; // Iterate over subkeys Array subKeys; if (FindRegistryKeyItems(keyH, subKeys)) { for (auto& subKey : subKeys) { HKEY subKeyH; if (RegOpenKeyExW(keyH, *subKey, 0, KEY_READ, &subKeyH) != ERROR_SUCCESS) continue; // Read subkey value DWORD type; DWORD cbData; if (RegQueryValueExW(subKeyH, valueName, nullptr, &type, nullptr, &cbData) != ERROR_SUCCESS || type != REG_SZ) { RegCloseKey(subKeyH); continue; } Array data; data.Resize((int32)cbData / sizeof(Char)); if (RegQueryValueExW(subKeyH, valueName, nullptr, nullptr, reinterpret_cast(data.Get()), &cbData) != ERROR_SUCCESS) { RegCloseKey(subKeyH); continue; } // Check if it's a valid installation path String path(data.Get(), data.Count() - 1); SearchDirectory(installations, path); RegCloseKey(subKeyH); } } RegCloseKey(keyH); } } bool sortInstallations(RiderInstallation* const& i1, RiderInstallation* const& i2) { Array values1, values2; i1->version.Split('.', values1); i2->version.Split('.', values2); int32 version1[3] = { 0 }; int32 version2[3] = { 0 }; StringUtils::Parse(values1[0].Get(), &version1[0]); StringUtils::Parse(values1[1].Get(), &version1[1]); StringUtils::Parse(values1[2].Get(), &version1[2]); StringUtils::Parse(values2[0].Get(), &version2[0]); StringUtils::Parse(values2[1].Get(), &version2[1]); StringUtils::Parse(values2[2].Get(), &version2[2]); // Compare by MAJOR.MINOR.BUILD if (version1[0] == version2[0]) { if (version1[1] == version2[1]) return version1[2] > version2[2]; return version1[1] > version2[1]; } return version1[0] > version2[0]; } #endif RiderCodeEditor::RiderCodeEditor(const String& execPath) : _execPath(execPath) , _solutionPath(Globals::ProjectFolder / Editor::Project->Name + TEXT(".sln")) { } void RiderCodeEditor::FindEditors(Array* output) { #if PLATFORM_WINDOWS Array installations; // For versions 2021 or later SearchRegistry(&installations, HKEY_CURRENT_USER, TEXT("SOFTWARE\\JetBrains\\Rider"), TEXT("InstallDir")); // For versions 2020 or earlier SearchRegistry(&installations, HKEY_CURRENT_USER, TEXT("SOFTWARE\\WOW6432Node\\JetBrains\\JetBrains Rider")); SearchRegistry(&installations, HKEY_LOCAL_MACHINE, TEXT("SOFTWARE\\WOW6432Node\\JetBrains\\JetBrains Rider")); // Sort found installations by version number Sorting::QuickSort(installations.Get(), installations.Count(), &sortInstallations); for (RiderInstallation* installation : installations) { output->Add(New(installation->path)); Delete(installation); } #endif } CodeEditorTypes RiderCodeEditor::GetType() const { return CodeEditorTypes::Rider; } String RiderCodeEditor::GetName() const { return TEXT("Rider"); } void RiderCodeEditor::OpenFile(const String& path, int32 line) { // Generate project files if solution is missing if (!FileSystem::FileExists(_solutionPath)) { ScriptsBuilder::GenerateProject(TEXT("-vs2019")); } // Open file line = line > 0 ? line : 1; const String args = String::Format(TEXT("\"{0}\" --line {2} \"{1}\""), _solutionPath, path, line); Platform::StartProcess(_execPath, args, StringView::Empty); } void RiderCodeEditor::OpenSolution() { // Generate project files if solution is missing if (!FileSystem::FileExists(_solutionPath)) { ScriptsBuilder::GenerateProject(TEXT("-vs2019")); } // Open solution const String args = String::Format(TEXT("\"{0}\""), _solutionPath); Platform::StartProcess(_execPath, args, StringView::Empty); } void RiderCodeEditor::OnFileAdded(const String& path) { ScriptsBuilder::GenerateProject(); }