Merge branch 'dev' into highlight-how-results-matched

This commit is contained in:
SysC0mp
2019-12-11 16:50:17 +01:00
12 changed files with 239 additions and 185 deletions

View File

@@ -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()

View File

@@ -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,9 +117,9 @@ 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);

View File

@@ -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 =>
{ {

View File

@@ -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;

View File

@@ -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;

View File

@@ -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()
{ {

View File

@@ -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; }
} }

View File

@@ -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>

View File

@@ -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;
});
}
} }
} }

View File

@@ -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;
} }

View File

@@ -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))

View File

@@ -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