VRCMelonAssistant/ModAssistant/Classes/Themes.cs

248 lines
11 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
using System.Linq;
using System.Windows;
using System.IO;
using System.Windows.Media;
2020-02-08 12:55:01 +13:00
using ModAssistant.Pages;
2020-02-14 16:49:43 +13:00
using System.Reflection;
using Microsoft.Win32;
2020-02-18 18:21:21 +13:00
using System.Windows.Media.Imaging;
namespace ModAssistant
{
public class Themes
{
2020-02-14 18:18:59 +13:00
public static string LoadedTheme { get; private set; } //Currently loaded theme
public static List<string> LoadedThemes { get => loadedThemes.Keys.ToList(); } //String of themes that can be loaded
public static string ThemeDirectory => $"{Environment.CurrentDirectory}/Themes"; //Self explanatory.
2020-02-14 18:18:59 +13:00
//Local dictionary of ResourceDictionarys mapped by their names.
private static Dictionary<string, ResourceDictionary> loadedThemes = new Dictionary<string, ResourceDictionary>();
2020-02-17 14:12:38 +13:00
private static List<string> preInstalledThemes = new List<string> { "Light", "Dark", "Light Pink" }; //These themes will always be available to use.
2020-02-08 14:09:25 +13:00
/// <summary>
/// Load all themes from local Themes subfolder and from embedded resources.
/// This also refreshes the Themes dropdown in the Options screen.
/// </summary>
public static void LoadThemes()
{
loadedThemes.Clear();
2020-02-08 14:09:25 +13:00
foreach (string localTheme in preInstalledThemes) //Load local themes (Light and Dark)
{
ResourceDictionary theme = LoadTheme(localTheme, true);
loadedThemes.Add(localTheme, theme);
}
2020-02-08 14:09:25 +13:00
if (Directory.Exists(ThemeDirectory)) //Load themes from Themes subfolder if it exists.
{
2020-02-08 12:55:01 +13:00
foreach (string file in Directory.EnumerateFiles(ThemeDirectory))
{
FileInfo info = new FileInfo(file);
2020-02-17 07:59:45 +13:00
//FileInfo includes the extension in its Name field, so we have to select only the actual name.
string name = Path.GetFileNameWithoutExtension(info.Name);
//Ignore Themes without the xaml extension and ignore themes with the same names as others.
//If requests are made I can instead make a Theme class that splits the pre-installed themes from
//user-made ones so that one more user-made Light/Dark theme can be added.
2020-02-08 14:09:25 +13:00
if (info.Extension.ToLower().Contains("xaml") && !loadedThemes.ContainsKey(name))
{
ResourceDictionary theme = LoadTheme(name);
if (theme != null)
{
loadedThemes.Add(name, theme);
}
}
}
}
2020-02-08 14:09:25 +13:00
if (Options.Instance != null && Options.Instance.ApplicationThemeComboBox != null) //Refresh Themes dropdown in Options screen.
{
2020-02-08 12:55:01 +13:00
Options.Instance.ApplicationThemeComboBox.ItemsSource = LoadedThemes;
Options.Instance.ApplicationThemeComboBox.SelectedIndex = LoadedThemes.IndexOf(LoadedTheme);
}
}
2020-02-17 13:39:52 +13:00
/// <summary>
/// Runs once at the start of the program, performs settings checking.
/// </summary>
/// <param name="savedTheme">Theme name retrieved from the settings file.</param>
public static void FirstLoad(string savedTheme)
{
2020-02-18 18:21:21 +13:00
if (string.IsNullOrEmpty(savedTheme))
2020-02-17 13:39:52 +13:00
{
Themes.ApplyWindowsTheme();
return;
}
try
{
Themes.ApplyTheme(savedTheme, false);
}
catch (ArgumentException)
{
Themes.ApplyWindowsTheme();
2020-02-17 15:16:22 +13:00
MainWindow.Instance.MainText = (string)Application.Current.FindResource("Themes:ThemeNotFound");
2020-02-17 13:39:52 +13:00
}
}
2020-02-08 14:09:25 +13:00
/// <summary>
/// Applies a loaded theme to ModAssistant.
/// </summary>
/// <param name="theme">Name of the theme.</param>
/// <param name="sendMessage">Send message to MainText (default: true).</param>
public static void ApplyTheme(string theme, bool sendMessage = true)
{
2020-02-08 14:26:06 +13:00
if (loadedThemes.TryGetValue(theme, out ResourceDictionary newTheme))
{
Application.Current.Resources.MergedDictionaries.RemoveAt(2); //We might want to change this to a static integer or search by name.
LoadedTheme = theme;
Application.Current.Resources.MergedDictionaries.Insert(2, newTheme); //Insert our new theme into the same spot as last time.
Properties.Settings.Default.SelectedTheme = theme;
Properties.Settings.Default.Save();
if (sendMessage)
{
2020-02-17 15:16:22 +13:00
MainWindow.Instance.MainText = string.Format((string)Application.Current.FindResource("Themes:ThemeSet"), theme);
}
2020-02-18 19:26:54 +13:00
LoadWaifus(theme);
2020-02-08 14:09:25 +13:00
ReloadIcons();
}
2020-02-17 15:16:22 +13:00
else
{
throw new ArgumentException(string.Format((string)Application.Current.FindResource("Themes:ThemeMissing"), theme));
}
}
2020-02-08 14:09:25 +13:00
/// <summary>
/// Writes a local theme to disk. You cannot write a theme loaded from the Themes subfolder to disk.
/// </summary>
/// <param name="themeName">Name of local theme.</param>
2020-02-08 13:46:20 +13:00
public static void WriteThemeToDisk(string themeName)
{
if (!Directory.Exists(ThemeDirectory))
{
Directory.CreateDirectory(ThemeDirectory);
}
if (!File.Exists($@"{ThemeDirectory}\\{themeName}.xaml"))
{
2020-02-14 18:18:59 +13:00
/*
* Light Pink theme is set to build as an Embedded Resource instead of the default Page.
* This is so that we can grab its exact content from Manifest, shown below.
* Writing it as is instead of using XAMLWriter keeps the source as is with comments, spacing, and organization.
* Using XAMLWriter would compress it into an unorganized mess.
*/
2020-02-14 16:49:43 +13:00
using (Stream s = Assembly.GetExecutingAssembly().GetManifestResourceStream($"ModAssistant.Themes.{themeName}.xaml"))
using (FileStream writer = new FileStream($@"{ThemeDirectory}\\{themeName}.xaml", FileMode.Create))
{
byte[] buffer = new byte[s.Length];
int read = s.Read(buffer, 0, (int)s.Length);
writer.Write(buffer, 0, buffer.Length);
}
2020-02-17 15:16:22 +13:00
MainWindow.Instance.MainText = string.Format((string)Application.Current.FindResource("Themes:SavedTemplateTheme"), themeName);
}
else
{
MessageBox.Show((string)Application.Current.FindResource("Themes:TemplateThemeExists"));
2020-02-08 13:46:20 +13:00
}
}
2020-02-08 14:09:25 +13:00
/// <summary>
/// Finds the theme set on Windows and applies it.
/// </summary>
public static void ApplyWindowsTheme()
{
using (RegistryKey key = Registry.CurrentUser
.OpenSubKey("Software").OpenSubKey("Microsoft")
.OpenSubKey("Windows").OpenSubKey("CurrentVersion")
.OpenSubKey("Themes").OpenSubKey("Personalize"))
{
object registryValueObject = key?.GetValue("AppsUseLightTheme");
if (registryValueObject != null)
{
if ((int)registryValueObject <= 0)
{
ApplyTheme("Dark", false);
return;
}
}
ApplyTheme("Light", false);
}
}
2020-02-08 14:09:25 +13:00
/// <summary>
/// Loads a ResourceDictionary from either Embedded Resources or from a file location.
/// </summary>
/// <param name="name">ResourceDictionary file name.</param>
2020-02-14 18:18:59 +13:00
/// <param name="localUri">Specifies whether or not to search locally or in the Themes subfolder.</param>
2020-02-08 14:09:25 +13:00
/// <returns></returns>
private static ResourceDictionary LoadTheme(string name, bool localUri = false)
{
string location = $"{Environment.CurrentDirectory}/Themes/{name}.xaml";
2020-02-14 18:18:59 +13:00
if (!File.Exists(location) && !localUri) //Return null if we're looking for an item in the Themes subfolder that doesn't exist.
{
return null;
}
if (localUri) //Change the location of the theme since we're not looking in a directory but rather in ModAssistant itself.
{
location = $"Themes/{name}.xaml";
}
2020-02-08 14:09:25 +13:00
Uri uri = new Uri(location, localUri ? UriKind.Relative : UriKind.Absolute);
ResourceDictionary dictionary = new ResourceDictionary();
2020-02-08 14:26:06 +13:00
try
{
dictionary.Source = uri;
}
2020-02-17 14:57:58 +13:00
catch (Exception ex)
2020-02-08 14:26:06 +13:00
{
MessageBox.Show($"Could not load {name}.\n\n{ex.Message}\n\nIgnoring...");
return null;
}
2020-02-08 14:09:25 +13:00
return dictionary;
}
2020-02-18 19:26:54 +13:00
/// <summary>
/// Loads the waifus from the Themes folder if they exist.
/// </summary>
/// <param name="name">Theme's name.</param>
private static void LoadWaifus(string name)
2020-02-18 18:21:21 +13:00
{
string location = Path.Combine(Environment.CurrentDirectory, "Themes");
BitmapImage background = null;
BitmapImage sidebar = null;
if (File.Exists(Path.Combine(location, name + ".png")))
{
background = new BitmapImage(new Uri(Path.Combine(location, name + ".png")));
}
if (File.Exists(Path.Combine(location, name + ".side.png")))
{
sidebar = new BitmapImage(new Uri(Path.Combine(location, name + ".side.png")));
}
MainWindow.Instance.BackgroundImage.ImageSource = background;
MainWindow.Instance.SideImage.ImageSource = sidebar;
}
2020-02-08 14:09:25 +13:00
/// <summary>
/// Reload the icon colors for the About, Info, Options, and Mods buttons from the currently loaded theme.
/// </summary>
private static void ReloadIcons()
{
ResourceDictionary icons = Application.Current.Resources.MergedDictionaries.First(x => x.Source.ToString() == "Resources/Icons.xaml");
ChangeColor(icons, "AboutIconColor", "heartDrawingGroup");
ChangeColor(icons, "InfoIconColor", "info_circleDrawingGroup");
ChangeColor(icons, "OptionsIconColor", "cogDrawingGroup");
ChangeColor(icons, "ModsIconColor", "microchipDrawingGroup");
}
/// <summary>
/// Change the color of an image from the loaded theme.
/// </summary>
/// <param name="icons">ResourceDictionary that contains the image.</param>
/// <param name="ResourceColorName">Resource name of the color to change.</param>
/// <param name="DrawingGroupName">DrawingGroup name for the image.</param>
private static void ChangeColor(ResourceDictionary icons, string ResourceColorName, string DrawingGroupName)
{
Application.Current.Resources[ResourceColorName] = loadedThemes[LoadedTheme][ResourceColorName];
((GeometryDrawing)((DrawingGroup)icons[DrawingGroupName]).Children[0]).Brush = (Brush)Application.Current.Resources[ResourceColorName];
}
}
}