using System; using System.Collections.Generic; using System.Diagnostics; using System.IO; using System.Runtime.InteropServices; using System.Runtime.Versioning; using System.Text; using Avalonia; using Avalonia.Controls; using Avalonia.Media; namespace SourceGit.Native { [SupportedOSPlatform("windows")] internal class Windows : OS.IBackend { [StructLayout(LayoutKind.Sequential)] internal struct RTL_OSVERSIONINFOEX { internal uint dwOSVersionInfoSize; internal uint dwMajorVersion; internal uint dwMinorVersion; internal uint dwBuildNumber; internal uint dwPlatformId; [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 128)] internal string szCSDVersion; } [StructLayout(LayoutKind.Sequential)] internal struct MARGINS { public int cxLeftWidth; public int cxRightWidth; public int cyTopHeight; public int cyBottomHeight; } [DllImport("shlwapi.dll", CharSet = CharSet.Unicode, SetLastError = false)] private static extern bool PathFindOnPath([In, Out] StringBuilder pszFile, [In] string[] ppszOtherDirs); [DllImport("ntdll")] private static extern int RtlGetVersion(ref RTL_OSVERSIONINFOEX lpVersionInformation); [DllImport("dwmapi.dll")] private static extern int DwmExtendFrameIntoClientArea(IntPtr hwnd, ref MARGINS margins); [DllImport("shell32.dll", CharSet = CharSet.Unicode, SetLastError = false)] private static extern IntPtr ILCreateFromPathW(string pszPath); [DllImport("shell32.dll", SetLastError = false)] private static extern void ILFree(IntPtr pidl); [DllImport("shell32.dll", CharSet = CharSet.Unicode, SetLastError = false)] private static extern int SHOpenFolderAndSelectItems(IntPtr pidlFolder, int cild, IntPtr apidl, int dwFlags); public Models.Shell Shell { get; set; } = Models.Shell.Default; public void SetupApp(AppBuilder builder) { builder.With(new FontManagerOptions() { DefaultFamilyName = "Microsoft YaHei UI", FontFallbacks = [new FontFallback { FontFamily = "Microsoft YaHei" }], }); // Fix drop shadow issue on Windows 10 RTL_OSVERSIONINFOEX v = new RTL_OSVERSIONINFOEX(); v.dwOSVersionInfoSize = (uint)Marshal.SizeOf(); if (RtlGetVersion(ref v) == 0 && (v.dwMajorVersion < 10 || v.dwBuildNumber < 22000)) { Window.WindowStateProperty.Changed.AddClassHandler((w, e) => ExtendWindowFrame(w)); Window.LoadedEvent.AddClassHandler((w, e) => ExtendWindowFrame(w)); } } public string FindGitExecutable() { var reg = Microsoft.Win32.RegistryKey.OpenBaseKey( Microsoft.Win32.RegistryHive.LocalMachine, Microsoft.Win32.RegistryView.Registry64); var git = reg.OpenSubKey("SOFTWARE\\GitForWindows"); if (git != null) { return Path.Combine(git.GetValue("InstallPath") as string, "bin", "git.exe"); } var builder = new StringBuilder("git.exe", 259); if (!PathFindOnPath(builder, null)) { return null; } var exePath = builder.ToString(); if (!string.IsNullOrEmpty(exePath)) { return exePath; } return null; } public List FindExternalTools() { var finder = new Models.ExternalToolsFinder(); finder.VSCode(FindVSCode); finder.VSCodeInsiders(FindVSCodeInsiders); finder.Fleet(() => $"{Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData)}\\Programs\\Fleet\\Fleet.exe"); finder.FindJetBrainsFromToolbox(() => $"{Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData)}\\JetBrains\\Toolbox"); finder.SublimeText(FindSublimeText); return finder.Founded; } public void OpenBrowser(string url) { var info = new ProcessStartInfo("cmd", $"/c start {url}"); info.CreateNoWindow = true; Process.Start(info); } public void OpenTerminal(string workdir) { if (string.IsNullOrEmpty(workdir) || !Path.Exists(workdir)) { workdir = "."; } var startInfo = new ProcessStartInfo(); startInfo.WorkingDirectory = workdir; switch (Shell) { case Models.Shell.Default: var binDir = Path.GetDirectoryName(OS.GitExecutable); var bash = Path.Combine(binDir, "bash.exe"); if (!File.Exists(bash)) { App.RaiseException(workdir, $"Can NOT found bash.exe under '{binDir}'"); return; } startInfo.FileName = bash; break; case Models.Shell.PowerShell: startInfo.FileName = ChoosePowerShell(); startInfo.Arguments = startInfo.FileName.EndsWith("pswd.exe") ? $"-WorkingDirectory \"{workdir}\" -Nologo" : "-Nologo"; break; case Models.Shell.CommandPrompt: startInfo.FileName = "cmd"; break; case Models.Shell.DefaultShellOfWindowsTerminal: var wt = FindWindowsTerminalApp(); if (!File.Exists(wt)) { App.RaiseException(workdir, $"Can NOT found wt.exe on your system!"); return; } startInfo.FileName = wt; startInfo.Arguments = $"-d \"{workdir}\""; break; default: App.RaiseException(workdir, $"Bad shell configuration!"); return; } Process.Start(startInfo); } public void OpenInFileManager(string path, bool select) { var fullpath = string.Empty; if (File.Exists(path)) { fullpath = new FileInfo(path).FullName; // For security reason, we never execute a file. // Instead, we open the folder and select it. select = true; } else { fullpath = new DirectoryInfo(path).FullName; } if (select) { // The fullpath here may be a file or a folder. OpenFolderAndSelectFile(fullpath); } else { // The fullpath here is always a folder. Process.Start(new ProcessStartInfo(fullpath) { UseShellExecute = true, CreateNoWindow = true, }); } } public void OpenWithDefaultEditor(string file) { var info = new FileInfo(file); var start = new ProcessStartInfo("cmd", $"/c start {info.FullName}"); start.CreateNoWindow = true; Process.Start(start); } private void ExtendWindowFrame(Window w) { var platformHandle = w.TryGetPlatformHandle(); if (platformHandle == null) return; var margins = new MARGINS { cxLeftWidth = 1, cxRightWidth = 1, cyTopHeight = 1, cyBottomHeight = 1 }; DwmExtendFrameIntoClientArea(platformHandle.Handle, ref margins); } // There are two versions of PowerShell : pwsh.exe (preferred) and powershell.exe (system default) private string ChoosePowerShell() { if (!string.IsNullOrEmpty(_powershellPath)) return _powershellPath; var localMachine = Microsoft.Win32.RegistryKey.OpenBaseKey( Microsoft.Win32.RegistryHive.LocalMachine, Microsoft.Win32.RegistryView.Registry64); var pwsh = localMachine.OpenSubKey(@"SOFTWARE\Microsoft\Windows\CurrentVersion\App Paths\pwsh.exe"); if (pwsh != null) { var path = pwsh.GetValue(null) as string; if (File.Exists(path)) { _powershellPath = path; return _powershellPath; } } var finder = new StringBuilder("powershell.exe", 512); if (PathFindOnPath(finder, null)) { _powershellPath = finder.ToString(); return _powershellPath; } return string.Empty; } private string FindWindowsTerminalApp() { if (!string.IsNullOrEmpty(_wtPath)) return _wtPath; var finder = new StringBuilder("wt.exe", 512); if (PathFindOnPath(finder, null)) { _wtPath = finder.ToString(); return _wtPath; } return string.Empty; } #region EXTERNAL_EDITOR_FINDER private string FindVSCode() { var localMachine = Microsoft.Win32.RegistryKey.OpenBaseKey( Microsoft.Win32.RegistryHive.LocalMachine, Microsoft.Win32.RegistryView.Registry64); // VSCode (system) var systemVScode = localMachine.OpenSubKey(@"SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\{EA457B21-F73E-494C-ACAB-524FDE069978}_is1"); if (systemVScode != null) { return systemVScode.GetValue("DisplayIcon") as string; } var currentUser = Microsoft.Win32.RegistryKey.OpenBaseKey( Microsoft.Win32.RegistryHive.CurrentUser, Microsoft.Win32.RegistryView.Registry64); // VSCode (user) var vscode = currentUser.OpenSubKey(@"SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\{771FD6B0-FA20-440A-A002-3B3BAC16DC50}_is1"); if (vscode != null) { return vscode.GetValue("DisplayIcon") as string; } return string.Empty; } private string FindVSCodeInsiders() { var localMachine = Microsoft.Win32.RegistryKey.OpenBaseKey( Microsoft.Win32.RegistryHive.LocalMachine, Microsoft.Win32.RegistryView.Registry64); // VSCode - Insiders (system) var systemVScodeInsiders = localMachine.OpenSubKey(@"SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\{1287CAD5-7C8D-410D-88B9-0D1EE4A83FF2}_is1"); if (systemVScodeInsiders != null) { return systemVScodeInsiders.GetValue("DisplayIcon") as string; } var currentUser = Microsoft.Win32.RegistryKey.OpenBaseKey( Microsoft.Win32.RegistryHive.CurrentUser, Microsoft.Win32.RegistryView.Registry64); // VSCode - Insiders (user) var vscodeInsiders = currentUser.OpenSubKey(@"SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\{217B4C08-948D-4276-BFBB-BEE930AE5A2C}_is1"); if (vscodeInsiders != null) { return vscodeInsiders.GetValue("DisplayIcon") as string; } return string.Empty; } private string FindSublimeText() { var localMachine = Microsoft.Win32.RegistryKey.OpenBaseKey( Microsoft.Win32.RegistryHive.LocalMachine, Microsoft.Win32.RegistryView.Registry64); // Sublime Text 4 var sublime = localMachine.OpenSubKey(@"SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\Sublime Text_is1"); if (sublime != null) { var icon = sublime.GetValue("DisplayIcon") as string; return Path.Combine(Path.GetDirectoryName(icon), "subl.exe"); } // Sublime Text 3 var sublime3 = localMachine.OpenSubKey(@"SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\Sublime Text 3_is1"); if (sublime3 != null) { var icon = sublime3.GetValue("DisplayIcon") as string; return Path.Combine(Path.GetDirectoryName(icon), "subl.exe"); } return string.Empty; } #endregion private void OpenFolderAndSelectFile(string folderPath) { var pidl = ILCreateFromPathW(folderPath); try { SHOpenFolderAndSelectItems(pidl, 0, 0, 0); } finally { ILFree(pidl); } } private string _powershellPath = string.Empty; private string _wtPath = string.Empty; } }