using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Runtime.InteropServices; using System.Windows; using System.Windows.Controls; using System.Windows.Interop; using System.Windows.Markup; using System.Windows.Media; using Wox.Infrastructure; using Wox.Infrastructure.Logger; using Wox.Infrastructure.UserSettings; namespace Wox.Core.Resource { public class Theme { private readonly List _themeDirectories = new List(); private ResourceDictionary _oldResource; private string _oldTheme; public Settings Settings { get; set; } private const string Folder = "Themes"; private const string Extension = ".xaml"; private string DirectoryPath => Path.Combine(Constant.ProgramDirectory, Folder); private string UserDirectoryPath => Path.Combine(Constant.DataDirectory, Folder); public Theme() { _themeDirectories.Add(DirectoryPath); _themeDirectories.Add(UserDirectoryPath); MakesureThemeDirectoriesExist(); var dicts = Application.Current.Resources.MergedDictionaries; _oldResource = dicts.First(d => { var p = d.Source.AbsolutePath; var dir = Path.GetDirectoryName(p).NonNull(); var info = new DirectoryInfo(dir); var f = info.Name; var e = Path.GetExtension(p); var found = f == Folder && e == Extension; return found; }); _oldTheme = Path.GetFileNameWithoutExtension(_oldResource.Source.AbsolutePath); } private void MakesureThemeDirectoriesExist() { foreach (string dir in _themeDirectories) { if (!Directory.Exists(dir)) { try { Directory.CreateDirectory(dir); } catch (Exception e) { Log.Exception($"|Theme.MakesureThemeDirectoriesExist|Exception when create directory <{dir}>", e); } } } } public bool ChangeTheme(string theme) { const string defaultTheme = "Dark"; string path = GetThemePath(theme); try { if (string.IsNullOrEmpty(path)) throw new DirectoryNotFoundException("Theme path can't be found <{path}>"); Settings.Theme = theme; var dicts = Application.Current.Resources.MergedDictionaries; //always allow re-loading default theme, in case of failure of switching to a new theme from default theme if (_oldTheme != theme || theme == defaultTheme) { dicts.Remove(_oldResource); var newResource = GetResourceDictionary(); dicts.Add(newResource); _oldResource = newResource; _oldTheme = Path.GetFileNameWithoutExtension(_oldResource.Source.AbsolutePath); } } catch (DirectoryNotFoundException e) { Log.Error($"|Theme.ChangeTheme|Theme <{theme}> path can't be found"); if (theme != defaultTheme) { MessageBox.Show(string.Format(InternationalizationManager.Instance.GetTranslation("theme_load_failure_path_not_exists"), theme)); ChangeTheme(defaultTheme); } return false; } catch (XamlParseException e) { Log.Error($"|Theme.ChangeTheme|Theme <{theme}> fail to parse"); if (theme != defaultTheme) { MessageBox.Show(string.Format(InternationalizationManager.Instance.GetTranslation("theme_load_failure_parse_error"), theme)); ChangeTheme(defaultTheme); } return false; } return true; } public ResourceDictionary GetResourceDictionary() { var uri = GetThemePath(Settings.Theme); var dict = new ResourceDictionary { Source = new Uri(uri, UriKind.Absolute) }; Style queryBoxStyle = dict["QueryBoxStyle"] as Style; if (queryBoxStyle != null) { queryBoxStyle.Setters.Add(new Setter(TextBox.FontFamilyProperty, new FontFamily(Settings.QueryBoxFont))); queryBoxStyle.Setters.Add(new Setter(TextBox.FontStyleProperty, FontHelper.GetFontStyleFromInvariantStringOrNormal(Settings.QueryBoxFontStyle))); queryBoxStyle.Setters.Add(new Setter(TextBox.FontWeightProperty, FontHelper.GetFontWeightFromInvariantStringOrNormal(Settings.QueryBoxFontWeight))); queryBoxStyle.Setters.Add(new Setter(TextBox.FontStretchProperty, FontHelper.GetFontStretchFromInvariantStringOrNormal(Settings.QueryBoxFontStretch))); } Style resultItemStyle = dict["ItemTitleStyle"] as Style; Style resultSubItemStyle = dict["ItemSubTitleStyle"] as Style; Style resultItemSelectedStyle = dict["ItemTitleSelectedStyle"] as Style; Style resultSubItemSelectedStyle = dict["ItemSubTitleSelectedStyle"] as Style; if (resultItemStyle != null && resultSubItemStyle != null && resultSubItemSelectedStyle != null && resultItemSelectedStyle != null) { Setter fontFamily = new Setter(TextBlock.FontFamilyProperty, new FontFamily(Settings.ResultFont)); Setter fontStyle = new Setter(TextBlock.FontStyleProperty, FontHelper.GetFontStyleFromInvariantStringOrNormal(Settings.ResultFontStyle)); Setter fontWeight = new Setter(TextBlock.FontWeightProperty, FontHelper.GetFontWeightFromInvariantStringOrNormal(Settings.ResultFontWeight)); Setter fontStretch = new Setter(TextBlock.FontStretchProperty, FontHelper.GetFontStretchFromInvariantStringOrNormal(Settings.ResultFontStretch)); Setter[] setters = { fontFamily, fontStyle, fontWeight, fontStretch }; Array.ForEach(new[] { resultItemStyle, resultSubItemStyle, resultItemSelectedStyle, resultSubItemSelectedStyle }, o => Array.ForEach(setters, p => o.Setters.Add(p))); } return dict; } public List LoadAvailableThemes() { List themes = new List(); foreach (var themeDirectory in _themeDirectories) { themes.AddRange( Directory.GetFiles(themeDirectory) .Where(filePath => filePath.EndsWith(Extension) && !filePath.EndsWith("Base.xaml")) .ToList()); } return themes.OrderBy(o => o).ToList(); } private string GetThemePath(string themeName) { foreach (string themeDirectory in _themeDirectories) { string path = Path.Combine(themeDirectory, themeName + Extension); if (File.Exists(path)) { return path; } } return string.Empty; } #region Blur Handling /* Found on https://github.com/riverar/sample-win10-aeroglass */ private enum AccentState { ACCENT_DISABLED = 0, ACCENT_ENABLE_GRADIENT = 1, ACCENT_ENABLE_TRANSPARENTGRADIENT = 2, ACCENT_ENABLE_BLURBEHIND = 3, ACCENT_INVALID_STATE = 4 } [StructLayout(LayoutKind.Sequential)] private struct AccentPolicy { public AccentState AccentState; public int AccentFlags; public int GradientColor; public int AnimationId; } [StructLayout(LayoutKind.Sequential)] private struct WindowCompositionAttributeData { public WindowCompositionAttribute Attribute; public IntPtr Data; public int SizeOfData; } private enum WindowCompositionAttribute { WCA_ACCENT_POLICY = 19 } [DllImport("user32.dll")] private static extern int SetWindowCompositionAttribute(IntPtr hwnd, ref WindowCompositionAttributeData data); /// /// Sets the blur for a window via SetWindowCompositionAttribute /// public void SetBlurForWindow() { // Exception of FindResource can't be cathed if global exception handle is set if (Environment.OSVersion.Version >= new Version(6, 2)) { var resource = Application.Current.TryFindResource("ThemeBlurEnabled"); bool blur; if (resource is bool) { blur = (bool)resource; } else { blur = false; } if (blur) { SetWindowAccent(Application.Current.MainWindow, AccentState.ACCENT_ENABLE_BLURBEHIND); } else { SetWindowAccent(Application.Current.MainWindow, AccentState.ACCENT_DISABLED); } } } private void SetWindowAccent(Window w, AccentState state) { var windowHelper = new WindowInteropHelper(w); var accent = new AccentPolicy { AccentState = state }; var accentStructSize = Marshal.SizeOf(accent); var accentPtr = Marshal.AllocHGlobal(accentStructSize); Marshal.StructureToPtr(accent, accentPtr, false); var data = new WindowCompositionAttributeData { Attribute = WindowCompositionAttribute.WCA_ACCENT_POLICY, SizeOfData = accentStructSize, Data = accentPtr }; SetWindowCompositionAttribute(windowHelper.Handle, ref data); Marshal.FreeHGlobal(accentPtr); } #endregion } }