mirror of
https://github.com/microsoft/PowerToys.git
synced 2024-12-13 19:19:23 +08:00
dc3b01dc15
1. Fix can't find Result.ctor bug for plugin introduced in c0889de1f9ae460b2cc189eb59e5bd90ddb7d17e 2. use %APPDATA% for all data, part of #389 3. MISC
624 lines
20 KiB
C#
624 lines
20 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics;
|
|
using System.Threading;
|
|
using System.Threading.Tasks;
|
|
using System.Windows;
|
|
using System.Windows.Input;
|
|
using Wox.Core.Plugin;
|
|
using Wox.Core.Resource;
|
|
using Wox.Core.Updater;
|
|
using Wox.Core.UserSettings;
|
|
using Wox.Helper;
|
|
using Wox.Infrastructure;
|
|
using Wox.Infrastructure.Hotkey;
|
|
using Wox.Infrastructure.Storage;
|
|
using Wox.Plugin;
|
|
using Wox.Storage;
|
|
|
|
namespace Wox.ViewModel
|
|
{
|
|
public class MainViewModel : BaseViewModel
|
|
{
|
|
#region Private Fields
|
|
|
|
private bool _isProgressBarTooltipVisible;
|
|
private double _left;
|
|
private double _top;
|
|
|
|
private Visibility _contextMenuVisibility;
|
|
private Visibility _progressBarVisibility;
|
|
private Visibility _resultListBoxVisibility;
|
|
private Visibility _mainWindowVisibility;
|
|
|
|
private bool _queryHasReturn;
|
|
private Query _lastQuery;
|
|
private bool _ignoreTextChange;
|
|
private string _queryTextBeforeLoadContextMenu;
|
|
private string _queryText;
|
|
|
|
private readonly JsonStrorage<Settings> _settingsStorage;
|
|
private readonly JsonStrorage<QueryHistory> _queryHistoryStorage;
|
|
private readonly JsonStrorage<UserSelectedRecord> _userSelectedRecordStorage;
|
|
private readonly JsonStrorage<TopMostRecord> _topMostRecordStorage;
|
|
// todo happlebao this field should be private in the future
|
|
public readonly Settings _settings;
|
|
private readonly QueryHistory _queryHistory;
|
|
private readonly UserSelectedRecord _userSelectedRecord;
|
|
private readonly TopMostRecord _topMostRecord;
|
|
|
|
private CancellationTokenSource _updateSource;
|
|
private CancellationToken _updateToken;
|
|
|
|
#endregion
|
|
|
|
#region Constructor
|
|
|
|
public MainViewModel()
|
|
{
|
|
_queryTextBeforeLoadContextMenu = "";
|
|
_queryText = "";
|
|
_lastQuery = new Query();
|
|
|
|
_settingsStorage = new JsonStrorage<Settings>();
|
|
_settings = _settingsStorage.Load();
|
|
|
|
// happlebao todo temp fix for instance code logic
|
|
HttpProxy.Instance.Settings = _settings;
|
|
UpdaterManager.Instance.Settings = _settings;
|
|
InternationalizationManager.Instance.Settings = _settings;
|
|
ThemeManager.Instance.Settings = _settings;
|
|
|
|
_queryHistoryStorage = new JsonStrorage<QueryHistory>();
|
|
_userSelectedRecordStorage = new JsonStrorage<UserSelectedRecord>();
|
|
_topMostRecordStorage = new JsonStrorage<TopMostRecord>();
|
|
_queryHistory = _queryHistoryStorage.Load();
|
|
_userSelectedRecord = _userSelectedRecordStorage.Load();
|
|
_topMostRecord = _topMostRecordStorage.Load();
|
|
|
|
InitializeResultListBox();
|
|
InitializeContextMenu();
|
|
InitializeKeyCommands();
|
|
}
|
|
|
|
~MainViewModel()
|
|
{
|
|
_settingsStorage.Save();
|
|
_queryHistoryStorage.Save();
|
|
_userSelectedRecordStorage.Save();
|
|
_topMostRecordStorage.Save();
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region ViewModel Properties
|
|
|
|
public ResultsViewModel Results { get; private set; }
|
|
|
|
public ResultsViewModel ContextMenu { get; private set; }
|
|
|
|
public string QueryText
|
|
{
|
|
get
|
|
{
|
|
return _queryText;
|
|
}
|
|
set
|
|
{
|
|
_queryText = value;
|
|
OnPropertyChanged();
|
|
|
|
if (_ignoreTextChange)
|
|
{
|
|
_ignoreTextChange = false;
|
|
}
|
|
else
|
|
{
|
|
HandleQueryTextUpdated();
|
|
}
|
|
}
|
|
}
|
|
|
|
public double Left
|
|
{
|
|
get
|
|
{
|
|
return _left;
|
|
}
|
|
set
|
|
{
|
|
_left = value;
|
|
OnPropertyChanged();
|
|
}
|
|
}
|
|
|
|
public double Top
|
|
{
|
|
get
|
|
{
|
|
return _top;
|
|
}
|
|
set
|
|
{
|
|
_top = value;
|
|
OnPropertyChanged();
|
|
}
|
|
}
|
|
|
|
public Visibility ContextMenuVisibility
|
|
|
|
{
|
|
get
|
|
{
|
|
return _contextMenuVisibility;
|
|
}
|
|
set
|
|
{
|
|
_contextMenuVisibility = value;
|
|
OnPropertyChanged();
|
|
|
|
_ignoreTextChange = true;
|
|
if (!value.IsVisible())
|
|
{
|
|
QueryText = _queryTextBeforeLoadContextMenu;
|
|
ResultListBoxVisibility = Visibility.Visible;
|
|
OnCursorMovedToEnd();
|
|
}
|
|
else
|
|
{
|
|
_queryTextBeforeLoadContextMenu = QueryText;
|
|
QueryText = "";
|
|
ResultListBoxVisibility = Visibility.Collapsed;
|
|
}
|
|
}
|
|
}
|
|
|
|
public Visibility ProgressBarVisibility
|
|
{
|
|
get
|
|
{
|
|
return _progressBarVisibility;
|
|
}
|
|
set
|
|
{
|
|
_progressBarVisibility = value;
|
|
OnPropertyChanged();
|
|
}
|
|
}
|
|
|
|
public Visibility ResultListBoxVisibility
|
|
{
|
|
get
|
|
{
|
|
return _resultListBoxVisibility;
|
|
}
|
|
set
|
|
{
|
|
_resultListBoxVisibility = value;
|
|
OnPropertyChanged();
|
|
}
|
|
}
|
|
|
|
public Visibility MainWindowVisibility
|
|
{
|
|
get
|
|
{
|
|
return _mainWindowVisibility;
|
|
}
|
|
set
|
|
{
|
|
_mainWindowVisibility = value;
|
|
OnPropertyChanged();
|
|
MainWindowVisibilityChanged?.Invoke(this, new EventArgs());
|
|
}
|
|
}
|
|
|
|
public ICommand EscCommand { get; set; }
|
|
public ICommand SelectNextItemCommand { get; set; }
|
|
public ICommand SelectPrevItemCommand { get; set; }
|
|
public ICommand DisplayNextQueryCommand { get; set; }
|
|
public ICommand DisplayPrevQueryCommand { get; set; }
|
|
public ICommand SelectNextPageCommand { get; set; }
|
|
public ICommand SelectPrevPageCommand { get; set; }
|
|
public ICommand StartHelpCommand { get; set; }
|
|
public ICommand LoadContextMenuCommand { get; set; }
|
|
public ICommand OpenResultCommand { get; set; }
|
|
public ICommand BackCommand { get; set; }
|
|
#endregion
|
|
|
|
#region Private Methods
|
|
|
|
private void InitializeKeyCommands()
|
|
{
|
|
EscCommand = new RelayCommand(_ =>
|
|
{
|
|
if (ContextMenuVisibility.IsVisible())
|
|
{
|
|
ContextMenuVisibility = Visibility.Collapsed;
|
|
}
|
|
else
|
|
{
|
|
MainWindowVisibility = Visibility.Collapsed;
|
|
}
|
|
});
|
|
|
|
SelectNextItemCommand = new RelayCommand(_ =>
|
|
{
|
|
if (ContextMenuVisibility.IsVisible())
|
|
{
|
|
ContextMenu.SelectNextResult();
|
|
}
|
|
else
|
|
{
|
|
Results.SelectNextResult();
|
|
}
|
|
});
|
|
|
|
SelectPrevItemCommand = new RelayCommand(_ =>
|
|
{
|
|
if (ContextMenuVisibility.IsVisible())
|
|
{
|
|
ContextMenu.SelectPrevResult();
|
|
}
|
|
else
|
|
{
|
|
Results.SelectPrevResult();
|
|
}
|
|
});
|
|
|
|
|
|
|
|
DisplayNextQueryCommand = new RelayCommand(_ =>
|
|
{
|
|
var nextQuery = _queryHistory.Next();
|
|
DisplayQueryHistory(nextQuery);
|
|
});
|
|
|
|
DisplayPrevQueryCommand = new RelayCommand(_ =>
|
|
{
|
|
var prev = _queryHistory.Previous();
|
|
DisplayQueryHistory(prev);
|
|
});
|
|
|
|
SelectNextPageCommand = new RelayCommand(_ =>
|
|
{
|
|
Results.SelectNextPage();
|
|
});
|
|
|
|
SelectPrevPageCommand = new RelayCommand(_ =>
|
|
{
|
|
Results.SelectPrevPage();
|
|
});
|
|
|
|
StartHelpCommand = new RelayCommand(_ =>
|
|
{
|
|
Process.Start("http://doc.getwox.com");
|
|
});
|
|
|
|
OpenResultCommand = new RelayCommand(o =>
|
|
{
|
|
var results = ContextMenuVisibility.IsVisible() ? ContextMenu : Results;
|
|
|
|
if (o != null)
|
|
{
|
|
var index = int.Parse(o.ToString());
|
|
results.SelectResult(index);
|
|
}
|
|
|
|
var result = results.SelectedResult.RawResult;
|
|
bool hideWindow = result.Action(new ActionContext
|
|
{
|
|
SpecialKeyState = GlobalHotkey.Instance.CheckModifiers()
|
|
});
|
|
if (hideWindow)
|
|
{
|
|
MainWindowVisibility = Visibility.Collapsed;
|
|
}
|
|
|
|
if (!ContextMenuVisibility.IsVisible())
|
|
{
|
|
_userSelectedRecord.Add(result);
|
|
_queryHistory.Add(result.OriginQuery.RawQuery);
|
|
}
|
|
});
|
|
|
|
LoadContextMenuCommand = new RelayCommand(_ =>
|
|
{
|
|
if (!ContextMenuVisibility.IsVisible())
|
|
{
|
|
var result = Results.SelectedResult.RawResult;
|
|
var id = result.PluginID;
|
|
|
|
var menus = PluginManager.GetContextMenusForPlugin(result);
|
|
menus.Add(ContextMenuTopMost(result));
|
|
menus.Add(ContextMenuPluginInfo(id));
|
|
|
|
ContextMenu.Clear();
|
|
ContextMenu.AddResults(menus, id);
|
|
ContextMenuVisibility = Visibility.Visible;
|
|
}
|
|
else
|
|
{
|
|
ContextMenuVisibility = Visibility.Collapsed;
|
|
}
|
|
});
|
|
|
|
BackCommand = new RelayCommand(_ =>
|
|
{
|
|
ListeningKeyPressed?.Invoke(this, new ListeningKeyPressedEventArgs(_ as KeyEventArgs));
|
|
});
|
|
}
|
|
|
|
private void InitializeResultListBox()
|
|
{
|
|
Results = new ResultsViewModel(_settings, _topMostRecord);
|
|
ResultListBoxVisibility = Visibility.Collapsed;
|
|
}
|
|
|
|
|
|
private void InitializeContextMenu()
|
|
{
|
|
ContextMenu = new ResultsViewModel(_settings, _topMostRecord);
|
|
ContextMenuVisibility = Visibility.Collapsed;
|
|
}
|
|
|
|
private void HandleQueryTextUpdated()
|
|
{
|
|
ProgressBarVisibility = Visibility.Hidden;
|
|
_updateSource?.Cancel();
|
|
_updateSource = new CancellationTokenSource();
|
|
_updateToken = _updateSource.Token;
|
|
|
|
if (ContextMenuVisibility.IsVisible())
|
|
{
|
|
QueryContextMenu();
|
|
}
|
|
else
|
|
{
|
|
string query = QueryText.Trim();
|
|
if (!string.IsNullOrEmpty(query))
|
|
{
|
|
Query(query);
|
|
//reset query history index after user start new query
|
|
ResetQueryHistoryIndex();
|
|
}
|
|
else
|
|
{
|
|
Results.Clear();
|
|
ResultListBoxVisibility = Visibility.Collapsed;
|
|
}
|
|
}
|
|
}
|
|
|
|
private void QueryContextMenu()
|
|
{
|
|
var contextMenuId = "Context Menu Id";
|
|
var query = QueryText.ToLower();
|
|
if (!string.IsNullOrEmpty(query))
|
|
{
|
|
|
|
List<Result> filterResults = new List<Result>();
|
|
foreach (var contextMenu in ContextMenu.Results)
|
|
{
|
|
if (StringMatcher.IsMatch(contextMenu.Title, query)
|
|
|| StringMatcher.IsMatch(contextMenu.SubTitle, query))
|
|
{
|
|
filterResults.Add(contextMenu.RawResult);
|
|
}
|
|
}
|
|
ContextMenu.Clear();
|
|
ContextMenu.AddResults(filterResults, contextMenuId);
|
|
}
|
|
}
|
|
|
|
private void Query(string text)
|
|
{
|
|
_queryHasReturn = false;
|
|
var query = PluginManager.QueryInit(text);
|
|
if (query != null)
|
|
{
|
|
// handle the exclusiveness of plugin using action keyword
|
|
string lastKeyword = _lastQuery.ActionKeyword;
|
|
string keyword = query.ActionKeyword;
|
|
if (string.IsNullOrEmpty(lastKeyword))
|
|
{
|
|
if (!string.IsNullOrEmpty(keyword))
|
|
{
|
|
Results.RemoveResultsExcept(PluginManager.NonGlobalPlugins[keyword].Metadata);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (string.IsNullOrEmpty(keyword))
|
|
{
|
|
Results.RemoveResultsFor(PluginManager.NonGlobalPlugins[lastKeyword].Metadata);
|
|
}
|
|
else if (lastKeyword != keyword)
|
|
{
|
|
Results.RemoveResultsExcept(PluginManager.NonGlobalPlugins[keyword].Metadata);
|
|
}
|
|
}
|
|
|
|
_lastQuery = query;
|
|
Task.Delay(200, _updateToken).ContinueWith(_ =>
|
|
{
|
|
if (query.RawQuery == _lastQuery.RawQuery && !_queryHasReturn)
|
|
{
|
|
ProgressBarVisibility = Visibility.Visible;
|
|
}
|
|
}, _updateToken);
|
|
|
|
var plugins = PluginManager.ValidPluginsForQuery(query);
|
|
foreach (var plugin in plugins)
|
|
{
|
|
var config = _settings.PluginSettings[plugin.Metadata.ID];
|
|
if (!config.Disabled)
|
|
{
|
|
Task.Factory.StartNew(() =>
|
|
{
|
|
var results = PluginManager.QueryForPlugin(plugin, query);
|
|
UpdateResultView(results, plugin.Metadata, query);
|
|
}, _updateToken);
|
|
}
|
|
}
|
|
|
|
|
|
}
|
|
}
|
|
|
|
private void ResetQueryHistoryIndex()
|
|
{
|
|
Results.RemoveResultsFor(QueryHistory.MetaData);
|
|
_queryHistory.Reset();
|
|
}
|
|
|
|
private void UpdateResultViewInternal(List<Result> list, PluginMetadata metadata)
|
|
{
|
|
Results.AddResults(list, metadata.ID);
|
|
}
|
|
|
|
private void DisplayQueryHistory(HistoryItem history)
|
|
{
|
|
if (history != null)
|
|
{
|
|
var historyMetadata = QueryHistory.MetaData;
|
|
|
|
QueryText = history.Query;
|
|
OnTextBoxSelected();
|
|
|
|
var executeQueryHistoryTitle = InternationalizationManager.Instance.GetTranslation("executeQuery");
|
|
var lastExecuteTime = InternationalizationManager.Instance.GetTranslation("lastExecuteTime");
|
|
Results.RemoveResultsExcept(historyMetadata);
|
|
UpdateResultViewInternal(new List<Result>
|
|
{
|
|
new Result
|
|
{
|
|
Title = string.Format(executeQueryHistoryTitle,history.Query),
|
|
SubTitle = string.Format(lastExecuteTime,history.ExecutedDateTime),
|
|
IcoPath = "Images\\history.png",
|
|
PluginDirectory = Infrastructure.Wox.ProgramPath,
|
|
Action = _ =>{
|
|
QueryText = history.Query;
|
|
OnTextBoxSelected();
|
|
return false;
|
|
}
|
|
}
|
|
}, historyMetadata);
|
|
}
|
|
}
|
|
private Result ContextMenuTopMost(Result result)
|
|
{
|
|
Result menu;
|
|
if (_topMostRecord.IsTopMost(result))
|
|
{
|
|
menu = new Result
|
|
{
|
|
Title = InternationalizationManager.Instance.GetTranslation("cancelTopMostInThisQuery"),
|
|
IcoPath = "Images\\down.png",
|
|
PluginDirectory = Infrastructure.Wox.ProgramPath,
|
|
Action = _ =>
|
|
{
|
|
_topMostRecord.Remove(result);
|
|
App.API.ShowMsg("Succeed");
|
|
return false;
|
|
}
|
|
};
|
|
}
|
|
else
|
|
{
|
|
menu = new Result
|
|
{
|
|
Title = InternationalizationManager.Instance.GetTranslation("setAsTopMostInThisQuery"),
|
|
IcoPath = "Images\\up.png",
|
|
PluginDirectory = Infrastructure.Wox.ProgramPath,
|
|
Action = _ =>
|
|
{
|
|
_topMostRecord.AddOrUpdate(result);
|
|
App.API.ShowMsg("Succeed");
|
|
return false;
|
|
}
|
|
};
|
|
}
|
|
return menu;
|
|
}
|
|
|
|
private Result ContextMenuPluginInfo(string id)
|
|
{
|
|
var metadata = PluginManager.GetPluginForId(id).Metadata;
|
|
var translator = InternationalizationManager.Instance;
|
|
|
|
var author = translator.GetTranslation("author");
|
|
var website = translator.GetTranslation("website");
|
|
var version = translator.GetTranslation("version");
|
|
var plugin = translator.GetTranslation("plugin");
|
|
var title = $"{plugin}: {metadata.Name}";
|
|
var icon = metadata.IcoPath;
|
|
var subtitle = $"{author}: {metadata.Author}, {website}: {metadata.Website} {version}: {metadata.Version}";
|
|
|
|
var menu = new Result
|
|
{
|
|
Title = title,
|
|
IcoPath = icon,
|
|
SubTitle = subtitle,
|
|
PluginDirectory = metadata.PluginDirectory,
|
|
Action = _ => false
|
|
};
|
|
return menu;
|
|
}
|
|
#endregion
|
|
|
|
#region Public Methods
|
|
|
|
public void UpdateResultView(List<Result> list, PluginMetadata metadata, Query originQuery)
|
|
{
|
|
_queryHasReturn = true;
|
|
ProgressBarVisibility = Visibility.Hidden;
|
|
|
|
list.ForEach(o =>
|
|
{
|
|
o.Score += _userSelectedRecord.GetSelectedCount(o) * 5;
|
|
});
|
|
if (originQuery.RawQuery == _lastQuery.RawQuery)
|
|
{
|
|
Application.Current.Dispatcher.Invoke(() =>
|
|
{
|
|
UpdateResultViewInternal(list, metadata);
|
|
});
|
|
}
|
|
|
|
if (list.Count > 0 && !ResultListBoxVisibility.IsVisible())
|
|
{
|
|
ResultListBoxVisibility = Visibility.Visible;
|
|
}
|
|
|
|
}
|
|
|
|
#endregion
|
|
|
|
public event EventHandler<ListeningKeyPressedEventArgs> ListeningKeyPressed;
|
|
public event EventHandler MainWindowVisibilityChanged;
|
|
|
|
public event EventHandler CursorMovedToEnd;
|
|
public void OnCursorMovedToEnd()
|
|
{
|
|
CursorMovedToEnd?.Invoke(this, new EventArgs());
|
|
}
|
|
|
|
public event EventHandler TextBoxSelected;
|
|
public void OnTextBoxSelected()
|
|
{
|
|
TextBoxSelected?.Invoke(this, new EventArgs());
|
|
}
|
|
}
|
|
|
|
public class ListeningKeyPressedEventArgs : EventArgs
|
|
{
|
|
public KeyEventArgs KeyEventArgs { get; private set; }
|
|
|
|
public ListeningKeyPressedEventArgs(KeyEventArgs keyEventArgs)
|
|
{
|
|
KeyEventArgs = keyEventArgs;
|
|
}
|
|
}
|
|
}
|