mirror of
https://github.com/microsoft/PowerToys.git
synced 2026-04-08 04:07:40 +02:00
Merge branch 'dev' into highlight-how-results-matched
This commit is contained in:
@@ -1,4 +1,4 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Diagnostics;
|
using System.Diagnostics;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
@@ -22,7 +22,6 @@ namespace Wox.Plugin.Folder
|
|||||||
{
|
{
|
||||||
_storage = new PluginJsonStorage<Settings>();
|
_storage = new PluginJsonStorage<Settings>();
|
||||||
_settings = _storage.Load();
|
_settings = _storage.Load();
|
||||||
InitialDriverList();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Save()
|
public void Save()
|
||||||
@@ -38,55 +37,20 @@ namespace Wox.Plugin.Folder
|
|||||||
public void Init(PluginInitContext context)
|
public void Init(PluginInitContext context)
|
||||||
{
|
{
|
||||||
_context = context;
|
_context = context;
|
||||||
|
InitialDriverList();
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<Result> Query(Query query)
|
public List<Result> Query(Query query)
|
||||||
{
|
{
|
||||||
|
var results = GetUserFolderResults(query);
|
||||||
|
|
||||||
string search = query.Search.ToLower();
|
string search = query.Search.ToLower();
|
||||||
|
|
||||||
List<FolderLink> userFolderLinks = _settings.FolderLinks.Where(
|
|
||||||
x => x.Nickname.StartsWith(search, StringComparison.OrdinalIgnoreCase)).ToList();
|
|
||||||
List<Result> results =
|
|
||||||
userFolderLinks.Select(
|
|
||||||
item => new Result()
|
|
||||||
{
|
|
||||||
Title = item.Nickname,
|
|
||||||
IcoPath = item.Path,
|
|
||||||
SubTitle = "Ctrl + Enter to open the directory",
|
|
||||||
TitleHighlightData = StringMatcher.FuzzySearch(item.Nickname, search).MatchData,
|
|
||||||
Action = c =>
|
|
||||||
{
|
|
||||||
if (c.SpecialKeyState.CtrlPressed)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
Process.Start(item.Path);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
MessageBox.Show(ex.Message, "Could not start " + item.Path);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_context.API.ChangeQuery($"{query.ActionKeyword} {item.Path}{(item.Path.EndsWith("\\") ? "" : "\\")}");
|
|
||||||
return false;
|
|
||||||
},
|
|
||||||
ContextData = item,
|
|
||||||
}).ToList();
|
|
||||||
|
|
||||||
if (_driverNames != null && !_driverNames.Any(search.StartsWith))
|
if (_driverNames != null && !_driverNames.Any(search.StartsWith))
|
||||||
return results;
|
return results;
|
||||||
|
|
||||||
//if (!input.EndsWith("\\"))
|
|
||||||
//{
|
|
||||||
// //"c:" means "the current directory on the C drive" whereas @"c:\" means "root of the C drive"
|
|
||||||
// input = input + "\\";
|
|
||||||
//}
|
|
||||||
results.AddRange(QueryInternal_Directory_Exists(query));
|
results.AddRange(QueryInternal_Directory_Exists(query));
|
||||||
|
|
||||||
// todo temp hack for scores
|
// todo why was this hack here?
|
||||||
foreach (var result in results)
|
foreach (var result in results)
|
||||||
{
|
{
|
||||||
result.Score += 10;
|
result.Score += 10;
|
||||||
@@ -94,12 +58,56 @@ namespace Wox.Plugin.Folder
|
|||||||
|
|
||||||
return results;
|
return results;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Result CreateFolderResult(string title, string path, Query query)
|
||||||
|
{
|
||||||
|
return new Result
|
||||||
|
{
|
||||||
|
Title = title,
|
||||||
|
IcoPath = path,
|
||||||
|
SubTitle = "Ctrl + Enter to open the directory",
|
||||||
|
TitleHighlightData = StringMatcher.FuzzySearch(query.Search, title).MatchData,
|
||||||
|
Action = c =>
|
||||||
|
{
|
||||||
|
if (c.SpecialKeyState.CtrlPressed)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
Process.Start(path);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
MessageBox.Show(ex.Message, "Could not start " + path);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
string changeTo = path.EndsWith("\\") ? path : path + "\\";
|
||||||
|
_context.API.ChangeQuery(string.IsNullOrEmpty(query.ActionKeyword) ?
|
||||||
|
changeTo :
|
||||||
|
query.ActionKeyword + " " + changeTo);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<Result> GetUserFolderResults(Query query)
|
||||||
|
{
|
||||||
|
string search = query.Search.ToLower();
|
||||||
|
var userFolderLinks = _settings.FolderLinks.Where(
|
||||||
|
x => x.Nickname.StartsWith(search, StringComparison.OrdinalIgnoreCase));
|
||||||
|
var results = userFolderLinks.Select(item =>
|
||||||
|
CreateFolderResult(item.Nickname, item.Path, query)).ToList();
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
|
||||||
private void InitialDriverList()
|
private void InitialDriverList()
|
||||||
{
|
{
|
||||||
if (_driverNames == null)
|
if (_driverNames == null)
|
||||||
{
|
{
|
||||||
_driverNames = new List<string>();
|
_driverNames = new List<string>();
|
||||||
DriveInfo[] allDrives = DriveInfo.GetDrives();
|
var allDrives = DriveInfo.GetDrives();
|
||||||
foreach (DriveInfo driver in allDrives)
|
foreach (DriveInfo driver in allDrives)
|
||||||
{
|
{
|
||||||
_driverNames.Add(driver.Name.ToLower().TrimEnd('\\'));
|
_driverNames.Add(driver.Name.ToLower().TrimEnd('\\'));
|
||||||
@@ -107,121 +115,135 @@ namespace Wox.Plugin.Folder
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static readonly char[] _specialSearchChars = new char[]
|
||||||
|
{
|
||||||
|
'?', '*', '>'
|
||||||
|
};
|
||||||
|
|
||||||
private List<Result> QueryInternal_Directory_Exists(Query query)
|
private List<Result> QueryInternal_Directory_Exists(Query query)
|
||||||
{
|
{
|
||||||
var search = query.Search.ToLower();
|
var search = query.Search;
|
||||||
var results = new List<Result>();
|
var results = new List<Result>();
|
||||||
|
var hasSpecial = search.IndexOfAny(_specialSearchChars) >= 0;
|
||||||
string incompleteName = "";
|
string incompleteName = "";
|
||||||
if (!Directory.Exists(search + "\\"))
|
if (hasSpecial || !Directory.Exists(search + "\\"))
|
||||||
{
|
{
|
||||||
//if the last component of the path is incomplete,
|
// if folder doesn't exist, we want to take the last part and use it afterwards to help the user
|
||||||
//then make auto complete for it.
|
// find the right folder.
|
||||||
int index = search.LastIndexOf('\\');
|
int index = search.LastIndexOf('\\');
|
||||||
if (index > 0 && index < (search.Length - 1))
|
if (index > 0 && index < (search.Length - 1))
|
||||||
{
|
{
|
||||||
incompleteName = search.Substring(index + 1);
|
incompleteName = search.Substring(index + 1).ToLower();
|
||||||
incompleteName = incompleteName.ToLower();
|
|
||||||
search = search.Substring(0, index + 1);
|
search = search.Substring(0, index + 1);
|
||||||
if (!Directory.Exists(search))
|
if (!Directory.Exists(search))
|
||||||
|
{
|
||||||
return results;
|
return results;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
return results;
|
return results;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
// folder exist, add \ at the end of doesn't exist
|
||||||
if (!search.EndsWith("\\"))
|
if (!search.EndsWith("\\"))
|
||||||
|
{
|
||||||
search += "\\";
|
search += "\\";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
string firstResult = "Open current directory";
|
results.Add(CreateOpenCurrentFolderResult(incompleteName, search));
|
||||||
|
|
||||||
|
var searchOption = SearchOption.TopDirectoryOnly;
|
||||||
|
incompleteName += "*";
|
||||||
|
|
||||||
|
// give the ability to search all folder when starting with >
|
||||||
|
if (incompleteName.StartsWith(">"))
|
||||||
|
{
|
||||||
|
searchOption = SearchOption.AllDirectories;
|
||||||
|
incompleteName = incompleteName.Substring(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
string searchNickname = new FolderLink { Path = query.Search }.Nickname;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// search folder and add results
|
||||||
|
var directoryInfo = new DirectoryInfo(search);
|
||||||
|
var fileSystemInfos = directoryInfo.GetFileSystemInfos(incompleteName, searchOption);
|
||||||
|
|
||||||
|
foreach (var fileSystemInfo in fileSystemInfos)
|
||||||
|
{
|
||||||
|
if ((fileSystemInfo.Attributes & FileAttributes.Hidden) == FileAttributes.Hidden) continue;
|
||||||
|
|
||||||
|
var result =
|
||||||
|
fileSystemInfo is DirectoryInfo
|
||||||
|
? CreateFolderResult(fileSystemInfo.Name, fileSystemInfo.FullName, query)
|
||||||
|
: CreateFileResult(fileSystemInfo.FullName, query);
|
||||||
|
results.Add(result);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
if (e is UnauthorizedAccessException || e is ArgumentException)
|
||||||
|
{
|
||||||
|
results.Add(new Result { Title = e.Message, Score = 501 });
|
||||||
|
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw;
|
||||||
|
}
|
||||||
|
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Result CreateFileResult(string filePath, Query query)
|
||||||
|
{
|
||||||
|
var result = new Result
|
||||||
|
{
|
||||||
|
Title = Path.GetFileName(filePath),
|
||||||
|
IcoPath = filePath,
|
||||||
|
TitleHighlightData = StringMatcher.FuzzySearch(query.Search, Path.GetFileName(filePath)).MatchData,
|
||||||
|
Action = c =>
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
Process.Start(filePath);
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
MessageBox.Show(ex.Message, "Could not start " + filePath);
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Result CreateOpenCurrentFolderResult(string incompleteName, string search)
|
||||||
|
{
|
||||||
|
var firstResult = "Open current directory";
|
||||||
if (incompleteName.Length > 0)
|
if (incompleteName.Length > 0)
|
||||||
firstResult = "Open " + search;
|
firstResult = "Open " + search;
|
||||||
results.Add(new Result
|
|
||||||
|
var folderName = search.TrimEnd('\\').Split(new[] { Path.DirectorySeparatorChar }, StringSplitOptions.None).Last();
|
||||||
|
|
||||||
|
return new Result
|
||||||
{
|
{
|
||||||
Title = firstResult,
|
Title = firstResult,
|
||||||
|
SubTitle = $"Use > to search files and subfolders within {folderName}, " +
|
||||||
|
$"* to search for file extensions in {folderName} or both >* to combine the search",
|
||||||
IcoPath = search,
|
IcoPath = search,
|
||||||
Score = 10000,
|
Score = 500,
|
||||||
Action = c =>
|
Action = c =>
|
||||||
{
|
{
|
||||||
Process.Start(search);
|
Process.Start(search);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
});
|
};
|
||||||
|
|
||||||
string searchNickname = new FolderLink { Path = query.Search }.Nickname;
|
|
||||||
|
|
||||||
//Add children directories
|
|
||||||
DirectoryInfo[] dirs = new DirectoryInfo(search).GetDirectories();
|
|
||||||
foreach (DirectoryInfo dir in dirs)
|
|
||||||
{
|
|
||||||
if ((dir.Attributes & FileAttributes.Hidden) == FileAttributes.Hidden) continue;
|
|
||||||
|
|
||||||
if (incompleteName.Length != 0 && !dir.Name.ToLower().StartsWith(incompleteName))
|
|
||||||
continue;
|
|
||||||
DirectoryInfo dirCopy = dir;
|
|
||||||
var result = new Result
|
|
||||||
{
|
|
||||||
Title = dir.Name,
|
|
||||||
IcoPath = dir.FullName,
|
|
||||||
SubTitle = "Ctrl + Enter to open the directory",
|
|
||||||
TitleHighlightData = StringMatcher.FuzzySearch(dir.Name, searchNickname).MatchData,
|
|
||||||
Action = c =>
|
|
||||||
{
|
|
||||||
if (c.SpecialKeyState.CtrlPressed)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
Process.Start(dirCopy.FullName);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
MessageBox.Show(ex.Message, "Could not start " + dirCopy.FullName);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_context.API.ChangeQuery($"{query.ActionKeyword} {dirCopy.FullName}\\");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
results.Add(result);
|
|
||||||
}
|
|
||||||
|
|
||||||
//Add children files
|
|
||||||
FileInfo[] files = new DirectoryInfo(search).GetFiles();
|
|
||||||
foreach (FileInfo file in files)
|
|
||||||
{
|
|
||||||
if ((file.Attributes & FileAttributes.Hidden) == FileAttributes.Hidden) continue;
|
|
||||||
if (incompleteName.Length != 0 && !file.Name.ToLower().StartsWith(incompleteName))
|
|
||||||
continue;
|
|
||||||
string filePath = file.FullName;
|
|
||||||
var result = new Result
|
|
||||||
{
|
|
||||||
Title = Path.GetFileName(filePath),
|
|
||||||
IcoPath = filePath,
|
|
||||||
TitleHighlightData = StringMatcher.FuzzySearch(file.Name, searchNickname).MatchData,
|
|
||||||
Action = c =>
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
Process.Start(filePath);
|
|
||||||
}
|
|
||||||
catch (Exception ex)
|
|
||||||
{
|
|
||||||
MessageBox.Show(ex.Message, "Could not start " + filePath);
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
results.Add(result);
|
|
||||||
}
|
|
||||||
|
|
||||||
return results;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public string GetTranslatedPluginTitle()
|
public string GetTranslatedPluginTitle()
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ namespace Wox.Plugin.Program
|
|||||||
private static PluginInitContext _context;
|
private static PluginInitContext _context;
|
||||||
|
|
||||||
private static BinaryStorage<Win32[]> _win32Storage;
|
private static BinaryStorage<Win32[]> _win32Storage;
|
||||||
private static BinaryStorage<UWP.Application[]> _uwpStorage;
|
private static BinaryStorage<UWP.Application[]> _uwpStorage;
|
||||||
private readonly PluginJsonStorage<Settings> _settingsStorage;
|
private readonly PluginJsonStorage<Settings> _settingsStorage;
|
||||||
|
|
||||||
public Main()
|
public Main()
|
||||||
@@ -41,7 +41,7 @@ namespace Wox.Plugin.Program
|
|||||||
});
|
});
|
||||||
Log.Info($"|Wox.Plugin.Program.Main|Number of preload win32 programs <{_win32s.Length}>");
|
Log.Info($"|Wox.Plugin.Program.Main|Number of preload win32 programs <{_win32s.Length}>");
|
||||||
Log.Info($"|Wox.Plugin.Program.Main|Number of preload uwps <{_uwps.Length}>");
|
Log.Info($"|Wox.Plugin.Program.Main|Number of preload uwps <{_uwps.Length}>");
|
||||||
|
|
||||||
var a = Task.Run(() =>
|
var a = Task.Run(() =>
|
||||||
{
|
{
|
||||||
if (IsStartupIndexProgramsRequired || !_win32s.Any())
|
if (IsStartupIndexProgramsRequired || !_win32s.Any())
|
||||||
@@ -68,19 +68,25 @@ namespace Wox.Plugin.Program
|
|||||||
|
|
||||||
public List<Result> Query(Query query)
|
public List<Result> Query(Query query)
|
||||||
{
|
{
|
||||||
|
Win32[] win32;
|
||||||
|
UWP.Application[] uwps;
|
||||||
|
|
||||||
lock (IndexLock)
|
lock (IndexLock)
|
||||||
{
|
{ // just take the reference inside the lock to eliminate query time issues.
|
||||||
var results1 = _win32s.AsParallel()
|
win32 = _win32s;
|
||||||
.Where(p => p.Enabled)
|
uwps = _uwps;
|
||||||
.Select(p => p.Result(query.Search, _context.API));
|
|
||||||
|
|
||||||
var results2 = _uwps.AsParallel()
|
|
||||||
.Where(p => p.Enabled)
|
|
||||||
.Select(p => p.Result(query.Search, _context.API));
|
|
||||||
|
|
||||||
var result = results1.Concat(results2).Where(r => r.Score > 0).ToList();
|
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var results1 = win32.AsParallel()
|
||||||
|
.Where(p => p.Enabled)
|
||||||
|
.Select(p => p.Result(query.Search, _context.API));
|
||||||
|
|
||||||
|
var results2 = uwps.AsParallel()
|
||||||
|
.Where(p => p.Enabled)
|
||||||
|
.Select(p => p.Result(query.Search, _context.API));
|
||||||
|
|
||||||
|
var result = results1.Concat(results2).Where(r => r != null && r.Score > 0).ToList();
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Init(PluginInitContext context)
|
public void Init(PluginInitContext context)
|
||||||
@@ -111,14 +117,14 @@ namespace Wox.Plugin.Program
|
|||||||
|
|
||||||
public static void IndexPrograms()
|
public static void IndexPrograms()
|
||||||
{
|
{
|
||||||
var t1 = Task.Run(()=>IndexWin32Programs());
|
var t1 = Task.Run(() => IndexWin32Programs());
|
||||||
|
|
||||||
var t2 = Task.Run(()=>IndexUWPPrograms());
|
var t2 = Task.Run(() => IndexUWPPrograms());
|
||||||
|
|
||||||
Task.WaitAll(t1, t2);
|
Task.WaitAll(t1, t2);
|
||||||
|
|
||||||
_settings.LastIndexTime = DateTime.Today;
|
_settings.LastIndexTime = DateTime.Today;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Control CreateSettingPanel()
|
public Control CreateSettingPanel()
|
||||||
{
|
{
|
||||||
@@ -141,7 +147,7 @@ namespace Wox.Plugin.Program
|
|||||||
var program = selectedResult.ContextData as IProgram;
|
var program = selectedResult.ContextData as IProgram;
|
||||||
if (program != null)
|
if (program != null)
|
||||||
{
|
{
|
||||||
menuOptions = program.ContextMenus(_context.API);
|
menuOptions = program.ContextMenus(_context.API);
|
||||||
}
|
}
|
||||||
|
|
||||||
menuOptions.Add(
|
menuOptions.Add(
|
||||||
@@ -151,7 +157,7 @@ namespace Wox.Plugin.Program
|
|||||||
Action = c =>
|
Action = c =>
|
||||||
{
|
{
|
||||||
DisableProgram(program);
|
DisableProgram(program);
|
||||||
_context.API.ShowMsg(_context.API.GetTranslation("wox_plugin_program_disable_dlgtitle_success"),
|
_context.API.ShowMsg(_context.API.GetTranslation("wox_plugin_program_disable_dlgtitle_success"),
|
||||||
_context.API.GetTranslation("wox_plugin_program_disable_dlgtitle_success_message"));
|
_context.API.GetTranslation("wox_plugin_program_disable_dlgtitle_success_message"));
|
||||||
return false;
|
return false;
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -273,11 +273,17 @@ namespace Wox.Plugin.Program.Programs
|
|||||||
|
|
||||||
public Result Result(string query, IPublicAPI api)
|
public Result Result(string query, IPublicAPI api)
|
||||||
{
|
{
|
||||||
|
var score = Score(query);
|
||||||
|
if (score <= 0)
|
||||||
|
{ // no need to create result if score is 0
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
var result = new Result
|
var result = new Result
|
||||||
{
|
{
|
||||||
SubTitle = Package.Location,
|
SubTitle = Package.Location,
|
||||||
Icon = Logo,
|
Icon = Logo,
|
||||||
Score = Score(query),
|
Score = score,
|
||||||
ContextData = this,
|
ContextData = this,
|
||||||
Action = e =>
|
Action = e =>
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -43,11 +43,17 @@ namespace Wox.Plugin.Program.Programs
|
|||||||
|
|
||||||
public Result Result(string query, IPublicAPI api)
|
public Result Result(string query, IPublicAPI api)
|
||||||
{
|
{
|
||||||
|
var score = Score(query);
|
||||||
|
if (score <= 0)
|
||||||
|
{ // no need to create result if this is zero
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
var result = new Result
|
var result = new Result
|
||||||
{
|
{
|
||||||
SubTitle = FullPath,
|
SubTitle = FullPath,
|
||||||
IcoPath = IcoPath,
|
IcoPath = IcoPath,
|
||||||
Score = Score(query),
|
Score = score,
|
||||||
ContextData = this,
|
ContextData = this,
|
||||||
Action = e =>
|
Action = e =>
|
||||||
{
|
{
|
||||||
@@ -195,7 +201,7 @@ namespace Wox.Plugin.Program.Programs
|
|||||||
catch (COMException e)
|
catch (COMException e)
|
||||||
{
|
{
|
||||||
// C:\\ProgramData\\Microsoft\\Windows\\Start Menu\\Programs\\MiracastView.lnk always cause exception
|
// C:\\ProgramData\\Microsoft\\Windows\\Start Menu\\Programs\\MiracastView.lnk always cause exception
|
||||||
ProgramLogger.LogException($"|Win32|LnkProgram|{path}"+
|
ProgramLogger.LogException($"|Win32|LnkProgram|{path}" +
|
||||||
"|Error caused likely due to trying to get the description of the program", e);
|
"|Error caused likely due to trying to get the description of the program", e);
|
||||||
|
|
||||||
program.Valid = false;
|
program.Valid = false;
|
||||||
|
|||||||
@@ -172,13 +172,13 @@ namespace Wox.Core.Plugin
|
|||||||
|
|
||||||
public static List<Result> QueryForPlugin(PluginPair pair, Query query)
|
public static List<Result> QueryForPlugin(PluginPair pair, Query query)
|
||||||
{
|
{
|
||||||
var results = new List<Result>();
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
List<Result> results = null;
|
||||||
var metadata = pair.Metadata;
|
var metadata = pair.Metadata;
|
||||||
var milliseconds = Stopwatch.Debug($"|PluginManager.QueryForPlugin|Cost for {metadata.Name}", () =>
|
var milliseconds = Stopwatch.Debug($"|PluginManager.QueryForPlugin|Cost for {metadata.Name}", () =>
|
||||||
{
|
{
|
||||||
results = pair.Plugin.Query(query) ?? results;
|
results = pair.Plugin.Query(query) ?? new List<Result>();
|
||||||
UpdatePluginMetadata(results, metadata, query);
|
UpdatePluginMetadata(results, metadata, query);
|
||||||
});
|
});
|
||||||
metadata.QueryCount += 1;
|
metadata.QueryCount += 1;
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ using System.Threading.Tasks;
|
|||||||
|
|
||||||
namespace Wox.Infrastructure.Storage
|
namespace Wox.Infrastructure.Storage
|
||||||
{
|
{
|
||||||
class WoxJsonStorage<T> : JsonStrorage<T> where T : new()
|
public class WoxJsonStorage<T> : JsonStrorage<T> where T : new()
|
||||||
{
|
{
|
||||||
public WoxJsonStorage()
|
public WoxJsonStorage()
|
||||||
{
|
{
|
||||||
@@ -18,4 +18,4 @@ namespace Wox.Infrastructure.Storage
|
|||||||
FilePath = Path.Combine(directoryPath, $"{filename}{FileSuffix}");
|
FilePath = Path.Combine(directoryPath, $"{filename}{FileSuffix}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -120,7 +120,7 @@ namespace Wox.Plugin
|
|||||||
public object ContextData { get; set; }
|
public object ContextData { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Plugin ID that generate this result
|
/// Plugin ID that generated this result
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public string PluginID { get; internal set; }
|
public string PluginID { get; internal set; }
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -36,6 +36,7 @@
|
|||||||
<KeyBinding Key="P" Modifiers="Ctrl" Command="{Binding SelectPrevItemCommand}"></KeyBinding>
|
<KeyBinding Key="P" Modifiers="Ctrl" Command="{Binding SelectPrevItemCommand}"></KeyBinding>
|
||||||
<KeyBinding Key="K" Modifiers="Ctrl" Command="{Binding SelectPrevItemCommand}"></KeyBinding>
|
<KeyBinding Key="K" Modifiers="Ctrl" Command="{Binding SelectPrevItemCommand}"></KeyBinding>
|
||||||
<KeyBinding Key="U" Modifiers="Ctrl" Command="{Binding SelectPrevPageCommand}"></KeyBinding>
|
<KeyBinding Key="U" Modifiers="Ctrl" Command="{Binding SelectPrevPageCommand}"></KeyBinding>
|
||||||
|
<KeyBinding Key="Home" Modifiers="Alt" Command="{Binding SelectFirstResultCommand}"></KeyBinding>
|
||||||
<KeyBinding Key="O" Modifiers="Ctrl" Command="{Binding LoadContextMenuCommand}"></KeyBinding>
|
<KeyBinding Key="O" Modifiers="Ctrl" Command="{Binding LoadContextMenuCommand}"></KeyBinding>
|
||||||
<KeyBinding Key="H" Modifiers="Ctrl" Command="{Binding LoadHistoryCommand}"></KeyBinding>
|
<KeyBinding Key="H" Modifiers="Ctrl" Command="{Binding LoadHistoryCommand}"></KeyBinding>
|
||||||
<KeyBinding Key="Enter" Modifiers="Shift" Command="{Binding LoadContextMenuCommand}"></KeyBinding>
|
<KeyBinding Key="Enter" Modifiers="Shift" Command="{Binding LoadContextMenuCommand}"></KeyBinding>
|
||||||
|
|||||||
@@ -1,47 +1,50 @@
|
|||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using Wox.Infrastructure.Storage;
|
using Newtonsoft.Json;
|
||||||
using Wox.Plugin;
|
using Wox.Plugin;
|
||||||
|
|
||||||
namespace Wox.Storage
|
namespace Wox.Storage
|
||||||
{
|
{
|
||||||
|
// todo this class is not thread safe.... but used from multiple threads.
|
||||||
public class TopMostRecord
|
public class TopMostRecord
|
||||||
{
|
{
|
||||||
public Dictionary<string, Record> records = new Dictionary<string, Record>();
|
[JsonProperty]
|
||||||
|
private Dictionary<string, Record> records = new Dictionary<string, Record>();
|
||||||
|
|
||||||
internal bool IsTopMost(Result result)
|
internal bool IsTopMost(Result result)
|
||||||
{
|
{
|
||||||
|
if (records.Count == 0)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// since this dictionary should be very small (or empty) going over it should be pretty fast.
|
||||||
return records.Any(o => o.Value.Title == result.Title
|
return records.Any(o => o.Value.Title == result.Title
|
||||||
&& o.Value.SubTitle == result.SubTitle
|
&& o.Value.SubTitle == result.SubTitle
|
||||||
&& o.Value.PluginID == result.PluginID
|
&& o.Value.PluginID == result.PluginID
|
||||||
&& o.Key == result.OriginQuery.RawQuery);
|
&& o.Key == result.OriginQuery.RawQuery);
|
||||||
}
|
}
|
||||||
|
|
||||||
internal void Remove(Result result)
|
internal void Remove(Result result)
|
||||||
{
|
{
|
||||||
if (records.ContainsKey(result.OriginQuery.RawQuery))
|
records.Remove(result.OriginQuery.RawQuery);
|
||||||
{
|
|
||||||
records.Remove(result.OriginQuery.RawQuery);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
internal void AddOrUpdate(Result result)
|
internal void AddOrUpdate(Result result)
|
||||||
{
|
{
|
||||||
if (records.ContainsKey(result.OriginQuery.RawQuery))
|
var record = new Record
|
||||||
{
|
{
|
||||||
records[result.OriginQuery.RawQuery].Title = result.Title;
|
PluginID = result.PluginID,
|
||||||
records[result.OriginQuery.RawQuery].SubTitle = result.SubTitle;
|
Title = result.Title,
|
||||||
records[result.OriginQuery.RawQuery].PluginID = result.PluginID;
|
SubTitle = result.SubTitle
|
||||||
}
|
};
|
||||||
else
|
records[result.OriginQuery.RawQuery] = record;
|
||||||
{
|
|
||||||
records.Add(result.OriginQuery.RawQuery, new Record
|
}
|
||||||
{
|
|
||||||
PluginID = result.PluginID,
|
public void Load(Dictionary<string, Record> dictionary)
|
||||||
Title = result.Title,
|
{
|
||||||
SubTitle = result.SubTitle
|
records = dictionary;
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -12,21 +12,23 @@ namespace Wox.Storage
|
|||||||
|
|
||||||
public void Add(Result result)
|
public void Add(Result result)
|
||||||
{
|
{
|
||||||
if (records.ContainsKey(result.ToString()))
|
var key = result.ToString();
|
||||||
|
if (records.TryGetValue(key, out int value))
|
||||||
{
|
{
|
||||||
records[result.ToString()] += 1;
|
records[key] = value + 1;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
records.Add(result.ToString(), 1);
|
records.Add(key, 1);
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public int GetSelectedCount(Result result)
|
public int GetSelectedCount(Result result)
|
||||||
{
|
{
|
||||||
if (records.ContainsKey(result.ToString()))
|
if (records.TryGetValue(result.ToString(), out int value))
|
||||||
{
|
{
|
||||||
return records[result.ToString()];
|
return value;
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -126,6 +126,8 @@ namespace Wox.ViewModel
|
|||||||
SelectedResults.SelectPrevPage();
|
SelectedResults.SelectPrevPage();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
SelectFirstResultCommand = new RelayCommand(_ => SelectedResults.SelectFirstResult());
|
||||||
|
|
||||||
StartHelpCommand = new RelayCommand(_ =>
|
StartHelpCommand = new RelayCommand(_ =>
|
||||||
{
|
{
|
||||||
Process.Start("http://doc.wox.one/");
|
Process.Start("http://doc.wox.one/");
|
||||||
@@ -268,6 +270,7 @@ namespace Wox.ViewModel
|
|||||||
public ICommand SelectPrevItemCommand { get; set; }
|
public ICommand SelectPrevItemCommand { get; set; }
|
||||||
public ICommand SelectNextPageCommand { get; set; }
|
public ICommand SelectNextPageCommand { get; set; }
|
||||||
public ICommand SelectPrevPageCommand { get; set; }
|
public ICommand SelectPrevPageCommand { get; set; }
|
||||||
|
public ICommand SelectFirstResultCommand { get; set; }
|
||||||
public ICommand StartHelpCommand { get; set; }
|
public ICommand StartHelpCommand { get; set; }
|
||||||
public ICommand LoadContextMenuCommand { get; set; }
|
public ICommand LoadContextMenuCommand { get; set; }
|
||||||
public ICommand LoadHistoryCommand { get; set; }
|
public ICommand LoadHistoryCommand { get; set; }
|
||||||
@@ -415,11 +418,15 @@ namespace Wox.ViewModel
|
|||||||
var config = _settings.PluginSettings.Plugins[plugin.Metadata.ID];
|
var config = _settings.PluginSettings.Plugins[plugin.Metadata.ID];
|
||||||
if (!config.Disabled)
|
if (!config.Disabled)
|
||||||
{
|
{
|
||||||
|
|
||||||
var results = PluginManager.QueryForPlugin(plugin, query);
|
var results = PluginManager.QueryForPlugin(plugin, query);
|
||||||
UpdateResultView(results, plugin.Metadata, query);
|
UpdateResultView(results, plugin.Metadata, query);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// this should happen once after all queries are done so progress bar should continue
|
||||||
|
// until the end of all querying
|
||||||
|
_queryHasReturn = true;
|
||||||
|
ProgressBarVisibility = Visibility.Hidden;
|
||||||
}, _updateToken);
|
}, _updateToken);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -628,9 +635,6 @@ namespace Wox.ViewModel
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public void UpdateResultView(List<Result> list, PluginMetadata metadata, Query originQuery)
|
public void UpdateResultView(List<Result> list, PluginMetadata metadata, Query originQuery)
|
||||||
{
|
{
|
||||||
_queryHasReturn = true;
|
|
||||||
ProgressBarVisibility = Visibility.Hidden;
|
|
||||||
|
|
||||||
foreach (var result in list)
|
foreach (var result in list)
|
||||||
{
|
{
|
||||||
if (_topMostRecord.IsTopMost(result))
|
if (_topMostRecord.IsTopMost(result))
|
||||||
|
|||||||
@@ -109,6 +109,11 @@ namespace Wox.ViewModel
|
|||||||
SelectedIndex = NewIndex(SelectedIndex - MaxResults);
|
SelectedIndex = NewIndex(SelectedIndex - MaxResults);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void SelectFirstResult()
|
||||||
|
{
|
||||||
|
SelectedIndex = NewIndex(0);
|
||||||
|
}
|
||||||
|
|
||||||
public void Clear()
|
public void Clear()
|
||||||
{
|
{
|
||||||
Results.Clear();
|
Results.Clear();
|
||||||
@@ -157,7 +162,6 @@ namespace Wox.ViewModel
|
|||||||
// Find the same results in A (old results) and B (new newResults)
|
// Find the same results in A (old results) and B (new newResults)
|
||||||
var sameResults = oldResults
|
var sameResults = oldResults
|
||||||
.Where(t1 => newResults.Any(x => x.Result.Equals(t1.Result)))
|
.Where(t1 => newResults.Any(x => x.Result.Equals(t1.Result)))
|
||||||
.Select(t1 => t1)
|
|
||||||
.ToList();
|
.ToList();
|
||||||
|
|
||||||
// remove result of relative complement of B in A
|
// remove result of relative complement of B in A
|
||||||
|
|||||||
Reference in New Issue
Block a user