mirror of
https://github.com/microsoft/PowerToys.git
synced 2026-04-04 18:26:39 +02:00
116 lines
4.6 KiB
C#
116 lines
4.6 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Data.OleDb;
|
|
using Microsoft.Search.Interop;
|
|
|
|
namespace Wox.Plugin.Indexer.SearchHelper
|
|
{
|
|
public class WindowsSearchAPI
|
|
{
|
|
public OleDbConnection conn;
|
|
public OleDbCommand command;
|
|
public OleDbDataReader WDSResults;
|
|
private readonly object _lock = new object();
|
|
|
|
|
|
public List<SearchResult> ExecuteQuery(ISearchQueryHelper queryHelper, string keyword)
|
|
{
|
|
List<SearchResult> _Result = new List<SearchResult>();
|
|
// Generate SQL from our parameters, converting the userQuery from AQS->WHERE clause
|
|
string sqlQuery = queryHelper.GenerateSQLFromUserQuery(keyword);
|
|
|
|
// --- Perform the query ---
|
|
// create an OleDbConnection object which connects to the indexer provider with the windows application
|
|
using (conn = new OleDbConnection(queryHelper.ConnectionString))
|
|
{
|
|
// open the connection
|
|
conn.Open();
|
|
|
|
// now create an OleDB command object with the query we built above and the connection we just opened.
|
|
using (command = new OleDbCommand(sqlQuery, conn))
|
|
{
|
|
// execute the command, which returns the results as an OleDbDataReader.
|
|
using (WDSResults = command.ExecuteReader())
|
|
{
|
|
if(WDSResults.HasRows)
|
|
{
|
|
while (WDSResults.Read())
|
|
{
|
|
if(WDSResults.GetValue(0) != DBNull.Value && WDSResults.GetValue(1) != DBNull.Value)
|
|
{
|
|
var result = new SearchResult
|
|
{
|
|
Path = WDSResults.GetString(0),
|
|
Title = WDSResults.GetString(1)
|
|
};
|
|
_Result.Add(result);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return _Result;
|
|
}
|
|
|
|
|
|
public void ModifyQueryHelper(ref ISearchQueryHelper queryHelper, string pattern)
|
|
{
|
|
// convert file pattern if it is not '*'. Don't create restriction for '*' as it includes all files.
|
|
if (pattern != "*")
|
|
{
|
|
pattern = pattern.Replace("*", "%");
|
|
pattern = pattern.Replace("?", "_");
|
|
|
|
if (pattern.Contains("%") || pattern.Contains("_"))
|
|
{
|
|
queryHelper.QueryWhereRestrictions += " AND System.FileName LIKE '" + pattern + "' ";
|
|
}
|
|
else
|
|
{
|
|
// if there are no wildcards we can use a contains which is much faster as it uses the index
|
|
queryHelper.QueryWhereRestrictions += " AND Contains(System.FileName, '" + pattern + "') ";
|
|
}
|
|
}
|
|
}
|
|
|
|
public void InitQueryHelper(out ISearchQueryHelper queryHelper, int maxCount)
|
|
{
|
|
// This uses the Microsoft.Search.Interop assembly
|
|
CSearchManager manager = new CSearchManager();
|
|
|
|
// SystemIndex catalog is the default catalog in Windows
|
|
ISearchCatalogManager catalogManager = manager.GetCatalog("SystemIndex");
|
|
|
|
// Get the ISearchQueryHelper which will help us to translate AQS --> SQL necessary to query the indexer
|
|
queryHelper = catalogManager.GetQueryHelper();
|
|
|
|
// Set the number of results we want. Don't set this property if all results are needed.
|
|
queryHelper.QueryMaxResults = maxCount;
|
|
|
|
// Set list of columns we want to display, getting the path presently
|
|
queryHelper.QuerySelectColumns = "System.ItemPathDisplay";
|
|
|
|
// Set additional query restriction
|
|
queryHelper.QueryWhereRestrictions = "AND scope='file:'";
|
|
|
|
// To filter based on title for now
|
|
queryHelper.QueryContentProperties = "System.Title";
|
|
|
|
// Set sorting order
|
|
queryHelper.QuerySorting = "System.DateModified DESC";
|
|
}
|
|
|
|
public IEnumerable<SearchResult> Search(string keyword, string pattern = "*", int maxCount = 100)
|
|
{
|
|
lock(_lock){
|
|
ISearchQueryHelper queryHelper;
|
|
InitQueryHelper(out queryHelper, maxCount);
|
|
ModifyQueryHelper(ref queryHelper, pattern);
|
|
return ExecuteQuery(queryHelper, keyword);
|
|
}
|
|
}
|
|
}
|
|
}
|