using Ryujinx.Common; using Ryujinx.Common.Configuration; using ShellLink; using SixLabors.ImageSharp; using SixLabors.ImageSharp.PixelFormats; using System; using System.Collections.Generic; using System.Drawing; using System.Drawing.Drawing2D; using System.Drawing.Imaging; using System.IO; using System.Runtime.Versioning; using Image = System.Drawing.Image; namespace Ryujinx.Ui.Common.Helper { public static class ShortcutHelper { [SupportedOSPlatform("windows")] private static void CreateShortcutWindows(string applicationFilePath, byte[] iconData, string iconPath, string cleanedAppName, string desktopPath) { string basePath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, AppDomain.CurrentDomain.FriendlyName + ".exe"); iconPath += ".ico"; MemoryStream iconDataStream = new(iconData); using Image image = Image.FromStream(iconDataStream); using Bitmap bitmap = new(128, 128); using System.Drawing.Graphics graphic = System.Drawing.Graphics.FromImage(bitmap); graphic.InterpolationMode = InterpolationMode.HighQualityBicubic; graphic.DrawImage(image, 0, 0, 128, 128); SaveBitmapAsIcon(bitmap, iconPath); var shortcut = Shortcut.CreateShortcut(basePath, GetArgsString(basePath, applicationFilePath), iconPath, 0); shortcut.StringData.NameString = cleanedAppName; shortcut.WriteToFile(Path.Combine(desktopPath, cleanedAppName + ".lnk")); } [SupportedOSPlatform("linux")] private static void CreateShortcutLinux(string applicationFilePath, byte[] iconData, string iconPath, string desktopPath, string cleanedAppName) { string basePath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "Ryujinx.sh"); var desktopFile = EmbeddedResources.ReadAllText("Ryujinx.Ui.Common/shortcut-template.desktop"); iconPath += ".png"; var image = SixLabors.ImageSharp.Image.Load(iconData); image.SaveAsPng(iconPath); using StreamWriter outputFile = new(Path.Combine(desktopPath, cleanedAppName + ".desktop")); outputFile.Write(desktopFile, cleanedAppName, iconPath, GetArgsString(basePath, applicationFilePath)); } [SupportedOSPlatform("macos")] private static void CreateShortcutMacos(string appFilePath, byte[] iconData, string desktopPath, string cleanedAppName) { string basePath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, AppDomain.CurrentDomain.FriendlyName); var plistFile = EmbeddedResources.ReadAllText("Ryujinx.Ui.Common/shortcut-template.plist"); // Macos .App folder string contentFolderPath = Path.Combine(desktopPath, cleanedAppName + ".app", "Contents"); string scriptFolderPath = Path.Combine(contentFolderPath, "MacOS"); if (!Directory.Exists(scriptFolderPath)) { Directory.CreateDirectory(scriptFolderPath); } // Runner script const string ScriptName = "runner.sh"; string scriptPath = Path.Combine(scriptFolderPath, ScriptName); using StreamWriter scriptFile = new(scriptPath); scriptFile.WriteLine("#!/bin/sh"); scriptFile.WriteLine(GetArgsString(basePath, appFilePath)); // Set execute permission FileInfo fileInfo = new(scriptPath); fileInfo.UnixFileMode |= UnixFileMode.UserExecute; // img string resourceFolderPath = Path.Combine(contentFolderPath, "Resources"); if (!Directory.Exists(resourceFolderPath)) { Directory.CreateDirectory(resourceFolderPath); } const string IconName = "icon.png"; var image = SixLabors.ImageSharp.Image.Load(iconData); image.SaveAsPng(Path.Combine(resourceFolderPath, IconName)); // plist file using StreamWriter outputFile = new(Path.Combine(contentFolderPath, "Info.plist")); outputFile.Write(plistFile, ScriptName, cleanedAppName, IconName); } public static void CreateAppShortcut(string applicationFilePath, string applicationName, string applicationId, byte[] iconData) { string desktopPath = Environment.GetFolderPath(Environment.SpecialFolder.DesktopDirectory); string cleanedAppName = string.Join("_", applicationName.Split(Path.GetInvalidFileNameChars())); if (OperatingSystem.IsWindows()) { string iconPath = Path.Combine(AppDataManager.BaseDirPath, "games", applicationId, "app"); CreateShortcutWindows(applicationFilePath, iconData, iconPath, cleanedAppName, desktopPath); return; } if (OperatingSystem.IsLinux()) { string iconPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.UserProfile), ".local", "share", "icons", "Ryujinx"); Directory.CreateDirectory(iconPath); CreateShortcutLinux(applicationFilePath, iconData, Path.Combine(iconPath, applicationId), desktopPath, cleanedAppName); return; } if (OperatingSystem.IsMacOS()) { CreateShortcutMacos(applicationFilePath, iconData, desktopPath, cleanedAppName); return; } throw new NotImplementedException("Shortcut support has not been implemented yet for this OS."); } private static string GetArgsString(string basePath, string appFilePath) { // args are first defined as a list, for easier adjustments in the future var argsList = new List { basePath, }; if (!string.IsNullOrEmpty(CommandLineState.BaseDirPathArg)) { argsList.Add("--root-data-dir"); argsList.Add($"\"{CommandLineState.BaseDirPathArg}\""); } argsList.Add($"\"{appFilePath}\""); return String.Join(" ", argsList); } /// /// Creates a Icon (.ico) file using the source bitmap image at the specified file path. /// /// The source bitmap image that will be saved as an .ico file /// The location that the new .ico file will be saved too (Make sure to include '.ico' in the path). [SupportedOSPlatform("windows")] private static void SaveBitmapAsIcon(Bitmap source, string filePath) { // Code Modified From https://stackoverflow.com/a/11448060/368354 by Benlitz byte[] header = { 0, 0, 1, 0, 1, 0, 0, 0, 0, 0, 1, 0, 32, 0, 0, 0, 0, 0, 22, 0, 0, 0 }; using FileStream fs = new(filePath, FileMode.Create); fs.Write(header); // Writing actual data source.Save(fs, ImageFormat.Png); // Getting data length (file length minus header) long dataLength = fs.Length - header.Length; // Write it in the correct place fs.Seek(14, SeekOrigin.Begin); fs.WriteByte((byte)dataLength); fs.WriteByte((byte)(dataLength >> 8)); } } }