mirror of
https://github.com/Flow-Launcher/Flow.Launcher.git
synced 2026-03-11 08:54:32 +00:00
550 lines
18 KiB
C#
550 lines
18 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Net;
|
|
using System.Threading.Tasks;
|
|
using System.Windows;
|
|
using System.Windows.Controls;
|
|
using System.Windows.Media;
|
|
using System.Windows.Media.Imaging;
|
|
using Flow.Launcher.Core;
|
|
using Flow.Launcher.Core.Configuration;
|
|
using Flow.Launcher.Core.ExternalPlugins;
|
|
using Flow.Launcher.Core.Plugin;
|
|
using Flow.Launcher.Core.Resource;
|
|
using Flow.Launcher.Helper;
|
|
using Flow.Launcher.Infrastructure;
|
|
using Flow.Launcher.Infrastructure.Storage;
|
|
using Flow.Launcher.Infrastructure.UserSettings;
|
|
using Flow.Launcher.Plugin;
|
|
using Flow.Launcher.Plugin.SharedModels;
|
|
|
|
namespace Flow.Launcher.ViewModel
|
|
{
|
|
public class SettingWindowViewModel : BaseModel
|
|
{
|
|
private readonly Updater _updater;
|
|
private readonly IPortable _portable;
|
|
private readonly FlowLauncherJsonStorage<Settings> _storage;
|
|
|
|
public SettingWindowViewModel(Updater updater, IPortable portable)
|
|
{
|
|
_updater = updater;
|
|
_portable = portable;
|
|
_storage = new FlowLauncherJsonStorage<Settings>();
|
|
Settings = _storage.Load();
|
|
Settings.PropertyChanged += (s, e) =>
|
|
{
|
|
switch (e.PropertyName)
|
|
{
|
|
case nameof(Settings.ActivateTimes):
|
|
OnPropertyChanged(nameof(ActivatedTimes));
|
|
break;
|
|
}
|
|
};
|
|
}
|
|
|
|
public Settings Settings { get; set; }
|
|
|
|
public async void UpdateApp()
|
|
{
|
|
await _updater.UpdateAppAsync(App.API, false);
|
|
}
|
|
|
|
public bool AutoUpdates
|
|
{
|
|
get => Settings.AutoUpdates;
|
|
set
|
|
{
|
|
Settings.AutoUpdates = value;
|
|
|
|
if (value)
|
|
UpdateApp();
|
|
}
|
|
}
|
|
|
|
// This is only required to set at startup. When portable mode enabled/disabled a restart is always required
|
|
private bool _portableMode = DataLocation.PortableDataLocationInUse();
|
|
public bool PortableMode
|
|
{
|
|
get => _portableMode;
|
|
set
|
|
{
|
|
if (!_portable.CanUpdatePortability())
|
|
return;
|
|
|
|
if (DataLocation.PortableDataLocationInUse())
|
|
{
|
|
_portable.DisablePortableMode();
|
|
}
|
|
else
|
|
{
|
|
_portable.EnablePortableMode();
|
|
}
|
|
}
|
|
}
|
|
|
|
public void Save()
|
|
{
|
|
foreach (var vm in PluginViewModels)
|
|
{
|
|
var id = vm.PluginPair.Metadata.ID;
|
|
|
|
Settings.PluginSettings.Plugins[id].Disabled = vm.PluginPair.Metadata.Disabled;
|
|
Settings.PluginSettings.Plugins[id].Priority = vm.Priority;
|
|
}
|
|
|
|
PluginManager.Save();
|
|
_storage.Save();
|
|
}
|
|
|
|
#region general
|
|
|
|
// todo a better name?
|
|
public class LastQueryMode
|
|
{
|
|
public string Display { get; set; }
|
|
public Infrastructure.UserSettings.LastQueryMode Value { get; set; }
|
|
}
|
|
public List<LastQueryMode> LastQueryModes
|
|
{
|
|
get
|
|
{
|
|
List<LastQueryMode> modes = new List<LastQueryMode>();
|
|
var enums = (Infrastructure.UserSettings.LastQueryMode[])Enum.GetValues(typeof(Infrastructure.UserSettings.LastQueryMode));
|
|
foreach (var e in enums)
|
|
{
|
|
var key = $"LastQuery{e}";
|
|
var display = _translater.GetTranslation(key);
|
|
var m = new LastQueryMode { Display = display, Value = e, };
|
|
modes.Add(m);
|
|
}
|
|
return modes;
|
|
}
|
|
}
|
|
|
|
public string Language
|
|
{
|
|
get
|
|
{
|
|
return Settings.Language;
|
|
}
|
|
set
|
|
{
|
|
InternationalizationManager.Instance.ChangeLanguage(value);
|
|
|
|
if (InternationalizationManager.Instance.PromptShouldUsePinyin(value))
|
|
ShouldUsePinyin = true;
|
|
}
|
|
}
|
|
|
|
public bool ShouldUsePinyin
|
|
{
|
|
get
|
|
{
|
|
return Settings.ShouldUsePinyin;
|
|
}
|
|
set
|
|
{
|
|
Settings.ShouldUsePinyin = value;
|
|
}
|
|
}
|
|
|
|
public List<string> QuerySearchPrecisionStrings
|
|
{
|
|
get
|
|
{
|
|
var precisionStrings = new List<string>();
|
|
|
|
var enumList = Enum.GetValues(typeof(SearchPrecisionScore)).Cast<SearchPrecisionScore>().ToList();
|
|
|
|
enumList.ForEach(x => precisionStrings.Add(x.ToString()));
|
|
|
|
return precisionStrings;
|
|
}
|
|
}
|
|
|
|
public List<string> OpenResultModifiersList => new List<string> { KeyConstant.Alt, KeyConstant.Ctrl, $"{KeyConstant.Ctrl}+{KeyConstant.Alt}" };
|
|
private Internationalization _translater => InternationalizationManager.Instance;
|
|
public List<Language> Languages => _translater.LoadAvailableLanguages();
|
|
public IEnumerable<int> MaxResultsRange => Enumerable.Range(2, 16);
|
|
|
|
public string TestProxy()
|
|
{
|
|
var proxyServer = Settings.Proxy.Server;
|
|
var proxyUserName = Settings.Proxy.UserName;
|
|
if (string.IsNullOrEmpty(proxyServer))
|
|
{
|
|
return InternationalizationManager.Instance.GetTranslation("serverCantBeEmpty");
|
|
}
|
|
if (Settings.Proxy.Port <= 0)
|
|
{
|
|
return InternationalizationManager.Instance.GetTranslation("portCantBeEmpty");
|
|
}
|
|
|
|
HttpWebRequest request = (HttpWebRequest)WebRequest.Create(_updater.GitHubRepository);
|
|
|
|
if (string.IsNullOrEmpty(proxyUserName) || string.IsNullOrEmpty(Settings.Proxy.Password))
|
|
{
|
|
request.Proxy = new WebProxy(proxyServer, Settings.Proxy.Port);
|
|
}
|
|
else
|
|
{
|
|
request.Proxy = new WebProxy(proxyServer, Settings.Proxy.Port)
|
|
{
|
|
Credentials = new NetworkCredential(proxyUserName, Settings.Proxy.Password)
|
|
};
|
|
}
|
|
try
|
|
{
|
|
var response = (HttpWebResponse)request.GetResponse();
|
|
if (response.StatusCode == HttpStatusCode.OK)
|
|
{
|
|
return InternationalizationManager.Instance.GetTranslation("proxyIsCorrect");
|
|
}
|
|
else
|
|
{
|
|
return InternationalizationManager.Instance.GetTranslation("proxyConnectFailed");
|
|
}
|
|
}
|
|
catch
|
|
{
|
|
return InternationalizationManager.Instance.GetTranslation("proxyConnectFailed");
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region startup
|
|
|
|
public void SetStartup()
|
|
{
|
|
AutoStartup.Enable();
|
|
}
|
|
|
|
public void RemoveStartup()
|
|
{
|
|
AutoStartup.Disable();
|
|
}
|
|
#endregion
|
|
|
|
#region plugin
|
|
|
|
public static string Plugin => @"https://github.com/Flow-Launcher/Flow.Launcher.PluginsManifest";
|
|
public PluginViewModel SelectedPlugin { get; set; }
|
|
|
|
public IList<PluginViewModel> PluginViewModels
|
|
{
|
|
get
|
|
{
|
|
var metadatas = PluginManager.AllPlugins
|
|
.OrderBy(x => x.Metadata.Disabled)
|
|
.ThenBy(y => y.Metadata.Name)
|
|
.Select(p => new PluginViewModel { PluginPair = p })
|
|
.ToList();
|
|
return metadatas;
|
|
}
|
|
}
|
|
|
|
public IList<UserPlugin> ExternalPlugins
|
|
{
|
|
get
|
|
{
|
|
return PluginsManifest.UserPlugins;
|
|
}
|
|
}
|
|
|
|
public Control SettingProvider
|
|
{
|
|
get
|
|
{
|
|
var settingProvider = SelectedPlugin.PluginPair.Plugin as ISettingProvider;
|
|
if (settingProvider != null)
|
|
{
|
|
var control = settingProvider.CreateSettingPanel();
|
|
control.HorizontalAlignment = HorizontalAlignment.Stretch;
|
|
control.VerticalAlignment = VerticalAlignment.Stretch;
|
|
return control;
|
|
}
|
|
else
|
|
{
|
|
return new Control();
|
|
}
|
|
}
|
|
}
|
|
|
|
public async Task RefreshExternalPluginsAsync()
|
|
{
|
|
await PluginsManifest.UpdateManifestAsync();
|
|
OnPropertyChanged(nameof(ExternalPlugins));
|
|
}
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
#region theme
|
|
|
|
public static string Theme => @"https://flowlauncher.com/docs/#/how-to-create-a-theme";
|
|
|
|
public string SelectedTheme
|
|
{
|
|
get { return Settings.Theme; }
|
|
set
|
|
{
|
|
Settings.Theme = value;
|
|
ThemeManager.Instance.ChangeTheme(value);
|
|
|
|
if (ThemeManager.Instance.BlurEnabled && Settings.UseDropShadowEffect)
|
|
DropShadowEffect = false;
|
|
}
|
|
}
|
|
|
|
public List<string> Themes
|
|
=> ThemeManager.Instance.LoadAvailableThemes().Select(Path.GetFileNameWithoutExtension).ToList();
|
|
|
|
public bool DropShadowEffect
|
|
{
|
|
get { return Settings.UseDropShadowEffect; }
|
|
set
|
|
{
|
|
if (ThemeManager.Instance.BlurEnabled && value)
|
|
{
|
|
MessageBox.Show(InternationalizationManager.Instance.GetTranslation("shadowEffectNotAllowed"));
|
|
return;
|
|
}
|
|
|
|
if (value)
|
|
{
|
|
ThemeManager.Instance.AddDropShadowEffectToCurrentTheme();
|
|
}
|
|
else
|
|
{
|
|
ThemeManager.Instance.RemoveDropShadowEffectFromCurrentTheme();
|
|
}
|
|
|
|
Settings.UseDropShadowEffect = value;
|
|
}
|
|
}
|
|
|
|
public class ColorScheme
|
|
{
|
|
public string Display { get; set; }
|
|
public ColorSchemes Value { get; set; }
|
|
}
|
|
|
|
public List<ColorScheme> ColorSchemes
|
|
{
|
|
get
|
|
{
|
|
List<ColorScheme> modes = new List<ColorScheme>();
|
|
var enums = (ColorSchemes[])Enum.GetValues(typeof(ColorSchemes));
|
|
foreach (var e in enums)
|
|
{
|
|
var key = $"ColorScheme{e}";
|
|
var display = _translater.GetTranslation(key);
|
|
var m = new ColorScheme { Display = display, Value = e, };
|
|
modes.Add(m);
|
|
}
|
|
return modes;
|
|
}
|
|
}
|
|
|
|
public double WindowWidthSize
|
|
{
|
|
get => Settings.WindowSize;
|
|
set => Settings.WindowSize = value;
|
|
}
|
|
|
|
public bool UseGlyphIcons
|
|
{
|
|
get => Settings.UseGlyphIcons;
|
|
set => Settings.UseGlyphIcons = value;
|
|
}
|
|
|
|
public bool UseAnimation
|
|
{
|
|
get => Settings.UseAnimation;
|
|
set => Settings.UseAnimation = value;
|
|
}
|
|
|
|
public bool UseSound
|
|
{
|
|
get => Settings.UseSound;
|
|
set => Settings.UseSound = value;
|
|
}
|
|
|
|
public Brush PreviewBackground
|
|
{
|
|
get
|
|
{
|
|
var wallpaper = WallpaperPathRetrieval.GetWallpaperPath();
|
|
if (wallpaper != null && File.Exists(wallpaper))
|
|
{
|
|
var memStream = new MemoryStream(File.ReadAllBytes(wallpaper));
|
|
var bitmap = new BitmapImage();
|
|
bitmap.BeginInit();
|
|
bitmap.StreamSource = memStream;
|
|
bitmap.EndInit();
|
|
var brush = new ImageBrush(bitmap) { Stretch = Stretch.UniformToFill };
|
|
return brush;
|
|
}
|
|
else
|
|
{
|
|
var wallpaperColor = WallpaperPathRetrieval.GetWallpaperColor();
|
|
var brush = new SolidColorBrush(wallpaperColor);
|
|
return brush;
|
|
}
|
|
}
|
|
}
|
|
|
|
public ResultsViewModel PreviewResults
|
|
{
|
|
get
|
|
{
|
|
var results = new List<Result>
|
|
{
|
|
new Result
|
|
{
|
|
Title = "Explorer",
|
|
SubTitle = "Search for files, folders and file contents",
|
|
IcoPath = Path.Combine(Constant.ProgramDirectory, @"Plugins\Flow.Launcher.Plugin.Explorer\Images\explorer.png")
|
|
},
|
|
new Result
|
|
{
|
|
Title = "WebSearch",
|
|
SubTitle = "Search the web with different search engine support",
|
|
IcoPath =Path.Combine(Constant.ProgramDirectory, @"Plugins\Flow.Launcher.Plugin.WebSearch\Images\web_search.png")
|
|
},
|
|
new Result
|
|
{
|
|
Title = "Program",
|
|
SubTitle = "Launch programs as admin or a different user",
|
|
IcoPath =Path.Combine(Constant.ProgramDirectory, @"Plugins\Flow.Launcher.Plugin.Program\Images\program.png")
|
|
},
|
|
new Result
|
|
{
|
|
Title = "ProcessKiller",
|
|
SubTitle = "Terminate unwanted processes",
|
|
IcoPath =Path.Combine(Constant.ProgramDirectory, @"Plugins\Flow.Launcher.Plugin.ProcessKiller\Images\app.png")
|
|
}
|
|
};
|
|
var vm = new ResultsViewModel(Settings);
|
|
vm.AddResults(results, "PREVIEW");
|
|
return vm;
|
|
}
|
|
}
|
|
|
|
public FontFamily SelectedQueryBoxFont
|
|
{
|
|
get
|
|
{
|
|
if (Fonts.SystemFontFamilies.Count(o =>
|
|
o.FamilyNames.Values != null &&
|
|
o.FamilyNames.Values.Contains(Settings.QueryBoxFont)) > 0)
|
|
{
|
|
var font = new FontFamily(Settings.QueryBoxFont);
|
|
return font;
|
|
}
|
|
else
|
|
{
|
|
var font = new FontFamily("Segoe UI");
|
|
return font;
|
|
}
|
|
}
|
|
set
|
|
{
|
|
Settings.QueryBoxFont = value.ToString();
|
|
ThemeManager.Instance.ChangeTheme(Settings.Theme);
|
|
}
|
|
}
|
|
|
|
public FamilyTypeface SelectedQueryBoxFontFaces
|
|
{
|
|
get
|
|
{
|
|
var typeface = SyntaxSugars.CallOrRescueDefault(
|
|
() => SelectedQueryBoxFont.ConvertFromInvariantStringsOrNormal(
|
|
Settings.QueryBoxFontStyle,
|
|
Settings.QueryBoxFontWeight,
|
|
Settings.QueryBoxFontStretch
|
|
));
|
|
return typeface;
|
|
}
|
|
set
|
|
{
|
|
Settings.QueryBoxFontStretch = value.Stretch.ToString();
|
|
Settings.QueryBoxFontWeight = value.Weight.ToString();
|
|
Settings.QueryBoxFontStyle = value.Style.ToString();
|
|
ThemeManager.Instance.ChangeTheme(Settings.Theme);
|
|
}
|
|
}
|
|
|
|
public FontFamily SelectedResultFont
|
|
{
|
|
get
|
|
{
|
|
if (Fonts.SystemFontFamilies.Count(o =>
|
|
o.FamilyNames.Values != null &&
|
|
o.FamilyNames.Values.Contains(Settings.ResultFont)) > 0)
|
|
{
|
|
var font = new FontFamily(Settings.ResultFont);
|
|
return font;
|
|
}
|
|
else
|
|
{
|
|
var font = new FontFamily("Segoe UI");
|
|
return font;
|
|
}
|
|
}
|
|
set
|
|
{
|
|
Settings.ResultFont = value.ToString();
|
|
ThemeManager.Instance.ChangeTheme(Settings.Theme);
|
|
}
|
|
}
|
|
|
|
public FamilyTypeface SelectedResultFontFaces
|
|
{
|
|
get
|
|
{
|
|
var typeface = SyntaxSugars.CallOrRescueDefault(
|
|
() => SelectedResultFont.ConvertFromInvariantStringsOrNormal(
|
|
Settings.ResultFontStyle,
|
|
Settings.ResultFontWeight,
|
|
Settings.ResultFontStretch
|
|
));
|
|
return typeface;
|
|
}
|
|
set
|
|
{
|
|
Settings.ResultFontStretch = value.Stretch.ToString();
|
|
Settings.ResultFontWeight = value.Weight.ToString();
|
|
Settings.ResultFontStyle = value.Style.ToString();
|
|
ThemeManager.Instance.ChangeTheme(Settings.Theme);
|
|
}
|
|
}
|
|
|
|
public string ThemeImage => Constant.QueryTextBoxIconImagePath;
|
|
|
|
#endregion
|
|
|
|
#region hotkey
|
|
|
|
public CustomPluginHotkey SelectedCustomPluginHotkey { get; set; }
|
|
|
|
#endregion
|
|
|
|
#region about
|
|
|
|
public string Website => Constant.Website;
|
|
public string ReleaseNotes => _updater.GitHubRepository + @"/releases/latest";
|
|
public string Documentation => Constant.Documentation;
|
|
public string Docs => Constant.Docs;
|
|
public string Github => Constant.GitHub;
|
|
public static string Version => Constant.Version;
|
|
public string ActivatedTimes => string.Format(_translater.GetTranslation("about_activate_times"), Settings.ActivateTimes);
|
|
#endregion
|
|
}
|
|
}
|