mirror of
https://github.com/Flow-Launcher/Flow.Launcher.git
synced 2026-03-11 08:54:32 +00:00
* Merge pull request #1061 from Flow-Launcher/remove_winget_ci * Merge pull request #991 from Flow-Launcher/context_menu_plugin_site * Merge pull request #1080 from gissehel/caret-position-fix * Caret position fix : Include PR #1074 * Merge pull request #1283 from nachmore/dev * Merge pull request #1296 from nachmore/bug_1284 * Merge pull request #1294 from Flow-Launcher/pluginInfoMultipleActionKeyword * Merge pull request #1299 from nachmore/bug_1269 * Plugin Exception Draft (#1147) * Merge pull request #1355 from onesounds/LimitWidth * Merge pull request #1088 from Flow-Launcher/add_spanish_latin_america * Merge pull request #1387 from Flow-Launcher/fix_exception_duplicate_url_opening * Merge pull request #1390 from Flow-Launcher/issue_1371 * Merge pull request #1391 from Flow-Launcher/issue_1366
391 lines
15 KiB
C#
391 lines
15 KiB
C#
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 System.Windows.Media.Effects;
|
|
using Flow.Launcher.Infrastructure;
|
|
using Flow.Launcher.Infrastructure.Logger;
|
|
using Flow.Launcher.Infrastructure.UserSettings;
|
|
|
|
namespace Flow.Launcher.Core.Resource
|
|
{
|
|
public class Theme
|
|
{
|
|
private const int ShadowExtraMargin = 12;
|
|
|
|
private readonly List<string> _themeDirectories = new List<string>();
|
|
private ResourceDictionary _oldResource;
|
|
private string _oldTheme;
|
|
public Settings Settings { get; set; }
|
|
private const string Folder = Constant.Themes;
|
|
private const string Extension = ".xaml";
|
|
private string DirectoryPath => Path.Combine(Constant.ProgramDirectory, Folder);
|
|
private string UserDirectoryPath => Path.Combine(DataLocation.DataDirectory(), Folder);
|
|
|
|
public bool BlurEnabled { get; set; }
|
|
|
|
private double mainWindowWidth;
|
|
|
|
public Theme()
|
|
{
|
|
_themeDirectories.Add(DirectoryPath);
|
|
_themeDirectories.Add(UserDirectoryPath);
|
|
MakesureThemeDirectoriesExist();
|
|
|
|
var dicts = Application.Current.Resources.MergedDictionaries;
|
|
_oldResource = dicts.First(d =>
|
|
{
|
|
if (d.Source == null)
|
|
return false;
|
|
|
|
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 = Constant.DefaultTheme;
|
|
|
|
string path = GetThemePath(theme);
|
|
try
|
|
{
|
|
if (string.IsNullOrEmpty(path))
|
|
throw new DirectoryNotFoundException("Theme path can't be found <{path}>");
|
|
|
|
Settings.Theme = theme;
|
|
|
|
// reload all resources even if the theme itself hasn't changed in order to pickup changes
|
|
// to things like fonts
|
|
UpdateResourceDictionary(GetResourceDictionary());
|
|
|
|
//always allow re-loading default theme, in case of failure of switching to a new theme from default theme
|
|
if (_oldTheme != theme || theme == defaultTheme)
|
|
{
|
|
_oldTheme = Path.GetFileNameWithoutExtension(_oldResource.Source.AbsolutePath);
|
|
}
|
|
|
|
BlurEnabled = IsBlurTheme();
|
|
|
|
if (Settings.UseDropShadowEffect && !BlurEnabled)
|
|
AddDropShadowEffectToCurrentTheme();
|
|
|
|
SetBlurForWindow();
|
|
}
|
|
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;
|
|
}
|
|
|
|
private void UpdateResourceDictionary(ResourceDictionary dictionaryToUpdate)
|
|
{
|
|
var dicts = Application.Current.Resources.MergedDictionaries;
|
|
|
|
dicts.Remove(_oldResource);
|
|
dicts.Add(dictionaryToUpdate);
|
|
_oldResource = dictionaryToUpdate;
|
|
}
|
|
|
|
private ResourceDictionary CurrentThemeResourceDictionary()
|
|
{
|
|
var uri = GetThemePath(Settings.Theme);
|
|
var dict = new ResourceDictionary
|
|
{
|
|
Source = new Uri(uri, UriKind.Absolute)
|
|
};
|
|
|
|
return dict;
|
|
}
|
|
|
|
public ResourceDictionary GetResourceDictionary()
|
|
{
|
|
var dict = CurrentThemeResourceDictionary();
|
|
|
|
if (dict["QueryBoxStyle"] is Style queryBoxStyle &&
|
|
dict["QuerySuggestionBoxStyle"] is Style querySuggestionBoxStyle)
|
|
{
|
|
var fontFamily = new FontFamily(Settings.QueryBoxFont);
|
|
var fontStyle = FontHelper.GetFontStyleFromInvariantStringOrNormal(Settings.QueryBoxFontStyle);
|
|
var fontWeight = FontHelper.GetFontWeightFromInvariantStringOrNormal(Settings.QueryBoxFontWeight);
|
|
var fontStretch = FontHelper.GetFontStretchFromInvariantStringOrNormal(Settings.QueryBoxFontStretch);
|
|
|
|
queryBoxStyle.Setters.Add(new Setter(TextBox.FontFamilyProperty, fontFamily));
|
|
queryBoxStyle.Setters.Add(new Setter(TextBox.FontStyleProperty, fontStyle));
|
|
queryBoxStyle.Setters.Add(new Setter(TextBox.FontWeightProperty, fontWeight));
|
|
queryBoxStyle.Setters.Add(new Setter(TextBox.FontStretchProperty, fontStretch));
|
|
|
|
var caretBrushPropertyValue = queryBoxStyle.Setters.OfType<Setter>().Any(x => x.Property.Name == "CaretBrush");
|
|
var foregroundPropertyValue = queryBoxStyle.Setters.OfType<Setter>().Where(x => x.Property.Name == "Foreground")
|
|
.Select(x => x.Value).FirstOrDefault();
|
|
if (!caretBrushPropertyValue && foregroundPropertyValue != null) //otherwise BaseQueryBoxStyle will handle styling
|
|
queryBoxStyle.Setters.Add(new Setter(TextBox.CaretBrushProperty, foregroundPropertyValue));
|
|
|
|
// Query suggestion box's font style is aligned with query box
|
|
querySuggestionBoxStyle.Setters.Add(new Setter(TextBox.FontFamilyProperty, fontFamily));
|
|
querySuggestionBoxStyle.Setters.Add(new Setter(TextBox.FontStyleProperty, fontStyle));
|
|
querySuggestionBoxStyle.Setters.Add(new Setter(TextBox.FontWeightProperty, fontWeight));
|
|
querySuggestionBoxStyle.Setters.Add(new Setter(TextBox.FontStretchProperty, fontStretch));
|
|
}
|
|
|
|
if (dict["ItemTitleStyle"] is Style resultItemStyle &&
|
|
dict["ItemSubTitleStyle"] is Style resultSubItemStyle &&
|
|
dict["ItemSubTitleSelectedStyle"] is Style resultSubItemSelectedStyle &&
|
|
dict["ItemTitleSelectedStyle"] is Style resultItemSelectedStyle &&
|
|
dict["ItemHotkeyStyle"] is Style resultHotkeyItemStyle &&
|
|
dict["ItemHotkeySelectedStyle"] is Style resultHotkeyItemSelectedStyle)
|
|
{
|
|
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, resultHotkeyItemStyle, resultHotkeyItemSelectedStyle }, o
|
|
=> Array.ForEach(setters, p => o.Setters.Add(p)));
|
|
}
|
|
/* Ignore Theme Window Width and use setting */
|
|
var windowStyle = dict["WindowStyle"] as Style;
|
|
var width = Settings.WindowSize;
|
|
windowStyle.Setters.Add(new Setter(Window.WidthProperty, width));
|
|
mainWindowWidth = (double)width;
|
|
return dict;
|
|
}
|
|
|
|
public List<string> LoadAvailableThemes()
|
|
{
|
|
List<string> themes = new List<string>();
|
|
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;
|
|
}
|
|
|
|
public void AddDropShadowEffectToCurrentTheme()
|
|
{
|
|
var dict = GetResourceDictionary();
|
|
|
|
var windowBorderStyle = dict["WindowBorderStyle"] as Style;
|
|
|
|
var effectSetter = new Setter
|
|
{
|
|
Property = Border.EffectProperty,
|
|
Value = new DropShadowEffect
|
|
{
|
|
Opacity = 0.4,
|
|
ShadowDepth = 2,
|
|
BlurRadius = 15
|
|
}
|
|
};
|
|
|
|
var marginSetter = windowBorderStyle.Setters.FirstOrDefault(setterBase => setterBase is Setter setter && setter.Property == Border.MarginProperty) as Setter;
|
|
if (marginSetter == null)
|
|
{
|
|
marginSetter = new Setter()
|
|
{
|
|
Property = Border.MarginProperty,
|
|
Value = new Thickness(ShadowExtraMargin),
|
|
};
|
|
windowBorderStyle.Setters.Add(marginSetter);
|
|
}
|
|
else
|
|
{
|
|
var baseMargin = (Thickness)marginSetter.Value;
|
|
var newMargin = new Thickness(
|
|
baseMargin.Left + ShadowExtraMargin,
|
|
baseMargin.Top + ShadowExtraMargin,
|
|
baseMargin.Right + ShadowExtraMargin,
|
|
baseMargin.Bottom + ShadowExtraMargin);
|
|
marginSetter.Value = newMargin;
|
|
}
|
|
|
|
windowBorderStyle.Setters.Add(effectSetter);
|
|
|
|
UpdateResourceDictionary(dict);
|
|
}
|
|
|
|
public void RemoveDropShadowEffectFromCurrentTheme()
|
|
{
|
|
var dict = CurrentThemeResourceDictionary();
|
|
var windowBorderStyle = dict["WindowBorderStyle"] as Style;
|
|
|
|
var effectSetter = windowBorderStyle.Setters.FirstOrDefault(setterBase => setterBase is Setter setter && setter.Property == Border.EffectProperty) as Setter;
|
|
var marginSetter = windowBorderStyle.Setters.FirstOrDefault(setterBase => setterBase is Setter setter && setter.Property == Border.MarginProperty) as Setter;
|
|
|
|
if (effectSetter != null)
|
|
{
|
|
windowBorderStyle.Setters.Remove(effectSetter);
|
|
}
|
|
if (marginSetter != null)
|
|
{
|
|
var currentMargin = (Thickness)marginSetter.Value;
|
|
var newMargin = new Thickness(
|
|
currentMargin.Left - ShadowExtraMargin,
|
|
currentMargin.Top - ShadowExtraMargin,
|
|
currentMargin.Right - ShadowExtraMargin,
|
|
currentMargin.Bottom - ShadowExtraMargin);
|
|
marginSetter.Value = newMargin;
|
|
}
|
|
|
|
UpdateResourceDictionary(dict);
|
|
}
|
|
|
|
#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);
|
|
|
|
/// <summary>
|
|
/// Sets the blur for a window via SetWindowCompositionAttribute
|
|
/// </summary>
|
|
public void SetBlurForWindow()
|
|
{
|
|
if (BlurEnabled)
|
|
{
|
|
SetWindowAccent(Application.Current.MainWindow, AccentState.ACCENT_ENABLE_BLURBEHIND);
|
|
}
|
|
else
|
|
{
|
|
SetWindowAccent(Application.Current.MainWindow, AccentState.ACCENT_DISABLED);
|
|
}
|
|
}
|
|
|
|
private bool IsBlurTheme()
|
|
{
|
|
if (Environment.OSVersion.Version >= new Version(6, 2))
|
|
{
|
|
var resource = Application.Current.TryFindResource("ThemeBlurEnabled");
|
|
|
|
if (resource is bool)
|
|
return (bool)resource;
|
|
|
|
return false;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private void SetWindowAccent(Window w, AccentState state)
|
|
{
|
|
var windowHelper = new WindowInteropHelper(w);
|
|
|
|
windowHelper.EnsureHandle();
|
|
|
|
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
|
|
}
|
|
}
|