Move context menu and socre into program model

This commit is contained in:
bao-qian 2016-08-22 02:21:22 +01:00
parent 2d72c58ae7
commit fb5938280e
5 changed files with 216 additions and 167 deletions

View File

@ -17,7 +17,7 @@ namespace Wox.Plugin.Program
private static Win32[] _win32s = { }; private static Win32[] _win32s = { };
private static UWP.Application[] _uwps = { }; private static UWP.Application[] _uwps = { };
private PluginInitContext _context; private static PluginInitContext _context;
private static ProgramIndexCache _cache; private static ProgramIndexCache _cache;
private static BinaryStorage<ProgramIndexCache> _cacheStorage; private static BinaryStorage<ProgramIndexCache> _cacheStorage;
@ -49,105 +49,15 @@ namespace Wox.Plugin.Program
public List<Result> Query(Query query) public List<Result> Query(Query query)
{ {
var results1 = _win32s.AsParallel() var results1 = _win32s.AsParallel()
.Where(p => Score(p, query.Search) > 0) .Select(p => p.Result(query.Search, _context.API))
.Select(ResultFromWin32); .Where(r => r.Score > 0);
var results2 = _uwps.AsParallel() var results2 = _uwps.AsParallel()
.Where(app => Score(app, query.Search) > 0) .Select(p => p.Result(query.Search, _context.API))
.Select(ResultFromUWP); .Where(r => r.Score > 0);
var result = results1.Concat(results2).ToList(); var result = results1.Concat(results2).ToList();
return result; return result;
} }
public Result ResultFromWin32(Win32 program)
{
var result = new Result
{
SubTitle = program.FullPath,
IcoPath = program.IcoPath,
Score = program.Score,
ContextData = program,
Action = e =>
{
var info = new ProcessStartInfo
{
FileName = program.FullPath,
WorkingDirectory = program.ParentDirectory
};
var hide = StartProcess(info);
return hide;
}
};
if (program.Description.Length >= program.Name.Length &&
program.Description.Substring(0, program.Name.Length) == program.Name)
{
result.Title = program.Description;
}
else if (!string.IsNullOrEmpty(program.Description))
{
result.Title = $"{program.Name}: {program.Description}";
}
else
{
result.Title = program.Name;
}
return result;
}
public Result ResultFromUWP(UWP.Application app)
{
var result = new Result
{
SubTitle = $"{app.Location}",
Icon = app.Logo,
Score = app.Score,
ContextData = app,
Action = e =>
{
app.Launch();
return true;
}
};
if (app.Description.Length >= app.DisplayName.Length &&
app.Description.Substring(0, app.DisplayName.Length) == app.DisplayName)
{
result.Title = app.Description;
}
else if (!string.IsNullOrEmpty(app.Description))
{
result.Title = $"{app.DisplayName}: {app.Description}";
}
else
{
result.Title = app.DisplayName;
}
return result;
}
private int Score(Win32 program, string query)
{
var score1 = StringMatcher.Score(program.Name, query);
var score2 = StringMatcher.ScoreForPinyin(program.Name, query);
var score3 = StringMatcher.Score(program.Description, query);
var score4 = StringMatcher.ScoreForPinyin(program.Description, query);
var score5 = StringMatcher.Score(program.ExecutableName, query);
var score = new[] { score1, score2, score3, score4, score5 }.Max();
program.Score = score;
return score;
}
private int Score(UWP.Application app, string query)
{
var score1 = StringMatcher.Score(app.DisplayName, query);
var score2 = StringMatcher.ScoreForPinyin(app.DisplayName, query);
var score3 = StringMatcher.Score(app.Description, query);
var score = new[] { score1, score2, score3 }.Max();
app.Score = score;
return score;
}
public void Init(PluginInitContext context) public void Init(PluginInitContext context)
{ {
_context = context; _context = context;
@ -176,39 +86,11 @@ namespace Wox.Plugin.Program
public List<Result> LoadContextMenus(Result selectedResult) public List<Result> LoadContextMenus(Result selectedResult)
{ {
Win32 p = selectedResult.ContextData as Win32; var program = selectedResult.ContextData as IProgram;
if (p != null) if (program != null)
{ {
List<Result> contextMenus = new List<Result> var menus = program.ContextMenus(_context.API);
{ return menus;
new Result
{
Title = _context.API.GetTranslation("wox_plugin_program_run_as_administrator"),
Action = _ =>
{
var info = new ProcessStartInfo
{
FileName = p.FullPath,
WorkingDirectory = p.ParentDirectory,
Verb = "runas"
};
var hide = StartProcess(info);
return hide;
},
IcoPath = "Images/cmd.png"
},
new Result
{
Title = _context.API.GetTranslation("wox_plugin_program_open_containing_folder"),
Action = _ =>
{
var hide = StartProcess(new ProcessStartInfo(p.ParentDirectory));
return hide;
},
IcoPath = "Images/folder.png"
}
};
return contextMenus;
} }
else else
{ {
@ -216,7 +98,7 @@ namespace Wox.Plugin.Program
} }
} }
private bool StartProcess(ProcessStartInfo info) public static bool StartProcess(ProcessStartInfo info)
{ {
bool hide; bool hide;
try try
@ -224,10 +106,10 @@ namespace Wox.Plugin.Program
Process.Start(info); Process.Start(info);
hide = true; hide = true;
} }
catch (Win32Exception) catch (Exception)
{ {
var name = $"Plugin: {_context.CurrentPluginMetadata.Name}"; var name = "Plugin: Program";
var message = "Can't open this file"; var message = $"Can't start: {info.FileName}";
_context.API.ShowMsg(name, message, string.Empty); _context.API.ShowMsg(name, message, string.Empty);
hide = false; hide = false;
} }

View File

@ -0,0 +1,14 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace Wox.Plugin.Program.Programs
{
public interface IProgram
{
List<Result> ContextMenus(IPublicAPI api);
Result Result(string query, IPublicAPI api);
}
}

View File

@ -1,10 +1,10 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Diagnostics;
using System.IO; using System.IO;
using System.Linq; using System.Linq;
using System.Runtime.InteropServices; using System.Runtime.InteropServices;
using System.Security.Principal; using System.Security.Principal;
using System.Threading.Tasks;
using System.Windows.Media; using System.Windows.Media;
using System.Windows.Media.Imaging; using System.Windows.Media.Imaging;
using Windows.ApplicationModel; using Windows.ApplicationModel;
@ -190,7 +190,7 @@ namespace Wox.Plugin.Program.Programs
} }
public class Application public class Application : IProgram
{ {
public string DisplayName { get; set; } public string DisplayName { get; set; }
public string Description { get; set; } public string Description { get; set; }
@ -201,12 +201,69 @@ namespace Wox.Plugin.Program.Programs
public string BackgroundColor { get; set; } public string BackgroundColor { get; set; }
public string LogoPath { get; set; } public string LogoPath { get; set; }
public int Score { get; set; }
public string Location { get; set; } public string Location { get; set; }
private int Score(string query)
{
var score1 = StringMatcher.Score(DisplayName, query);
var score2 = StringMatcher.ScoreForPinyin(DisplayName, query);
var score3 = StringMatcher.Score(Description, query);
var score = new[] { score1, score2, score3 }.Max();
return score;
}
// todo: wrap with try exception public Result Result(string query, IPublicAPI api)
public void Launch() {
var result = new Result
{
SubTitle = Location,
Icon = Logo,
Score = Score(query),
ContextData = this,
Action = e =>
{
Launch(api);
return true;
}
};
if (Description.Length >= DisplayName.Length &&
Description.Substring(0, DisplayName.Length) == DisplayName)
{
result.Title = Description;
}
else if (!string.IsNullOrEmpty(Description))
{
result.Title = $"{DisplayName}: {Description}";
}
else
{
result.Title = DisplayName;
}
return result;
}
public List<Result> ContextMenus(IPublicAPI api)
{
var contextMenus = new List<Result>
{
new Result
{
Title = api.GetTranslation("wox_plugin_program_open_containing_folder"),
Action = _ =>
{
var hide = Main.StartProcess(new ProcessStartInfo(Location));
return hide;
},
IcoPath = "Images/folder.png"
}
};
return contextMenus;
}
private void Launch(IPublicAPI api)
{
try
{ {
var appManager = new ApplicationActivationManager(); var appManager = new ApplicationActivationManager();
uint unusedPid; uint unusedPid;
@ -214,6 +271,13 @@ namespace Wox.Plugin.Program.Programs
const ACTIVATEOPTIONS noFlags = ACTIVATEOPTIONS.AO_NONE; const ACTIVATEOPTIONS noFlags = ACTIVATEOPTIONS.AO_NONE;
appManager.ActivateApplication(UserModelId, noArgs, noFlags, out unusedPid); appManager.ActivateApplication(UserModelId, noArgs, noFlags, out unusedPid);
} }
catch (Exception)
{
var name = "Plugin: Program";
var message = $"Can't start UWP: {DisplayName}";
api.ShowMsg(name, message, string.Empty);
}
}
public ImageSource Logo() public ImageSource Logo()

View File

@ -1,17 +1,19 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.ComponentModel;
using System.Diagnostics; using System.Diagnostics;
using System.IO; using System.IO;
using System.Linq; using System.Linq;
using System.Text; using System.Text;
using Microsoft.Win32; using Microsoft.Win32;
using Shell; using Shell;
using Wox.Infrastructure;
using Wox.Infrastructure.Logger; using Wox.Infrastructure.Logger;
namespace Wox.Plugin.Program.Programs namespace Wox.Plugin.Program.Programs
{ {
[Serializable] [Serializable]
public class Win32 public class Win32 : IProgram
{ {
public string Name { get; set; } public string Name { get; set; }
public string IcoPath { get; set; } public string IcoPath { get; set; }
@ -19,12 +21,97 @@ namespace Wox.Plugin.Program.Programs
public string ParentDirectory { get; set; } public string ParentDirectory { get; set; }
public string ExecutableName { get; set; } public string ExecutableName { get; set; }
public string Description { get; set; } public string Description { get; set; }
public int Score { get; set; }
private const string ShortcutExtension = "lnk"; private const string ShortcutExtension = "lnk";
private const string ApplicationReferenceExtension = "appref-ms"; private const string ApplicationReferenceExtension = "appref-ms";
private const string ExeExtension = "exe"; private const string ExeExtension = "exe";
private int Score(string query)
{
var score1 = StringMatcher.Score(Name, query);
var score2 = StringMatcher.ScoreForPinyin(Name, query);
var score3 = StringMatcher.Score(Description, query);
var score4 = StringMatcher.ScoreForPinyin(Description, query);
var score5 = StringMatcher.Score(ExecutableName, query);
var score = new[] { score1, score2, score3, score4, score5 }.Max();
return score;
}
public Result Result(string query, IPublicAPI api)
{
var result = new Result
{
SubTitle = FullPath,
IcoPath = IcoPath,
Score = Score(query),
ContextData = this,
Action = e =>
{
var info = new ProcessStartInfo
{
FileName = FullPath,
WorkingDirectory = ParentDirectory
};
var hide = Main.StartProcess(info);
return hide;
}
};
if (Description.Length >= Name.Length &&
Description.Substring(0, Name.Length) == Name)
{
result.Title = Description;
}
else if (!string.IsNullOrEmpty(Description))
{
result.Title = $"{Name}: {Description}";
}
else
{
result.Title = Name;
}
return result;
}
public List<Result> ContextMenus(IPublicAPI api)
{
var contextMenus = new List<Result>
{
new Result
{
Title = api.GetTranslation("wox_plugin_program_run_as_administrator"),
Action = _ =>
{
var info = new ProcessStartInfo
{
FileName = FullPath,
WorkingDirectory = ParentDirectory,
Verb = "runas"
};
var hide = Main.StartProcess(info);
return hide;
},
IcoPath = "Images/cmd.png"
},
new Result
{
Title = api.GetTranslation("wox_plugin_program_open_containing_folder"),
Action = _ =>
{
var hide = Main.StartProcess(new ProcessStartInfo(ParentDirectory));
return hide;
},
IcoPath = "Images/folder.png"
}
};
return contextMenus;
}
public override string ToString() public override string ToString()
{ {
return ExecutableName; return ExecutableName;
@ -211,30 +298,30 @@ namespace Wox.Plugin.Program.Programs
return new Win32(); return new Win32();
} }
private static Win32 ScoreFilter(Win32 p) //private static Win32 ScoreFilter(Win32 p)
{ //{
var start = new[] { "启动", "start" }; // var start = new[] { "启动", "start" };
var doc = new[] { "帮助", "help", "文档", "documentation" }; // var doc = new[] { "帮助", "help", "文档", "documentation" };
var uninstall = new[] { "卸载", "uninstall" }; // var uninstall = new[] { "卸载", "uninstall" };
var contained = start.Any(s => p.Name.ToLower().Contains(s)); // var contained = start.Any(s => p.Name.ToLower().Contains(s));
if (contained) // if (contained)
{ // {
p.Score += 10; // p.Score += 10;
} // }
contained = doc.Any(d => p.Name.ToLower().Contains(d)); // contained = doc.Any(d => p.Name.ToLower().Contains(d));
if (contained) // if (contained)
{ // {
p.Score -= 10; // p.Score -= 10;
} // }
contained = uninstall.Any(u => p.Name.ToLower().Contains(u)); // contained = uninstall.Any(u => p.Name.ToLower().Contains(u));
if (contained) // if (contained)
{ // {
p.Score -= 20; // p.Score -= 20;
} // }
return p; // return p;
} //}
public static Win32[] All(Settings settings) public static Win32[] All(Settings settings)
{ {
@ -250,7 +337,8 @@ namespace Wox.Plugin.Program.Programs
programs = programs.Concat(startMenu); programs = programs.Concat(startMenu);
} }
var unregistered = UnregisteredPrograms(settings.ProgramSources, settings.ProgramSuffixes); var unregistered = UnregisteredPrograms(settings.ProgramSources, settings.ProgramSuffixes);
programs = programs.Concat(unregistered).Select(ScoreFilter); programs = programs.Concat(unregistered);
//.Select(ScoreFilter);
return programs.ToArray(); return programs.ToArray();
} }
} }

View File

@ -71,6 +71,7 @@
<DependentUpon>AddProgramSource.xaml</DependentUpon> <DependentUpon>AddProgramSource.xaml</DependentUpon>
</Compile> </Compile>
<Compile Include="FileChangeWatcher.cs" /> <Compile Include="FileChangeWatcher.cs" />
<Compile Include="Programs\IProgram.cs" />
<Compile Include="Programs\UWP.cs" /> <Compile Include="Programs\UWP.cs" />
<Compile Include="Programs\Win32.cs" /> <Compile Include="Programs\Win32.cs" />
<Compile Include="SuffixesConverter.cs" /> <Compile Include="SuffixesConverter.cs" />