mirror of
https://github.com/microsoft/PowerToys.git
synced 2026-04-04 10:16:24 +02:00
* Upgraded projects to target .NET 8 * Updated .NET runtime package targets to use latest .NET 8 build * Updated PowerToys Interop to target .NET 8 * Switch to use ArgumentNullException.ThrowIfNull * ArgumentNullException.ThrowIfNull for CropAndLockViewModel * Switching to ObjectDisposedException.ThrowIf * Upgrade System.ComponentModel.Composition to 8.0 * ArgumentNullException.ThrowIfNull in Helper * Switch to StartsWith using StringComparison.Ordinal * Disabled CA1859, CA1716, SYSLIB1096 analyzers * Update RIDs to reflect breaking changes in .NET 8 * Updated Microsoft NuGet packages to RC1 * Updated Analyzer package to latest .NET 8 preview package * CA1854: Use TryGetValue instead of ContainsKey * [Build] Update TFM to .NET 8 for publish profiles * [Analyzers] Remove CA1309, CA1860-CA1865, CA1869, CA2208 from warning. * [Analyzers] Fix for C26495 * [Analyzers] Disable CS1615, CS9191 * [CI] Target .NET 8 in YAML * [CI] Add .NET preview version flag temporarily. * [FileLocksmith] Update TFM to .NET 8 * [CI] Switch to preview agent * [CI] Update NOTICE.md * [CI] Update Release to target .NET 8 and use Preview agent * [Analyzers] Disable CA1854 * Fix typo * Updated Microsoft.CodeAnalysis.NetAnalyzers to latest preview Updated packages to rc2 * [Analyzers][CPP] Turn off warning for 5271 * [Analyzers][CPP] Turn off warning for 26493 * [KeyboardListener] Add mutex include to resolve error * [PT Run][Folder] Use static SearchValues to resolve CA1870 * [PowerLauncher] Fix TryGetValue * [MouseJumpSettings] Use ArgumentNullException.ThrowIfNull * [Build] Disable parallel dotnet tool restore * [Build] No cache of dotnet tool packages * [Build] Temporarily move .NET 8 SDK task before XAML formatting * [Build][Temp] Try using .NET 7 prior to XAML formatting and then switch to .NET 8 after * [Build] Use .NET 6 for XAML Styler * [CI] Updated NOTICE.md * [FancyZones] Update TFM to .NET 8 * [EnvVar] Update TFM to .NET 8 and update RID * [EnvVar] Use ArgumentNullException.ThrowIfNull * [Dev] Updated packages to .NET 8 RTM version * [Dev] Updated Microsoft.CodeAnalysis.NetAnalyzers to latest * [CI] Updated NOTICE.md with latest package versions * Fix new utility target fameworks and runtimeids * Don't use preview images anymore * [CI] Add script to update VCToolsVersion environment variable * [CI] Add Step to Verify VCToolsVersion * [CI] Use latest flag for vswhere to set proper VCToolsVersion * Add VCToolsVersion checking to release.yml * Remove net publishing from local/ PR CI builds * Revert "Remove net publishing from local/ PR CI builds" This reverts commitf469778996. * Only publish necessary projects * Add verbosity to release pipelines builds of PowerTOys * Set VCToolsVersion for publish.cmd when called from installer * [Installer] Moved project publish logic to MSBuild Task * [CI] Revert using publish.cmd * [CI] Set VCToolsVersion and unset ClearDevCommandPromptEnvVars property * Installer publishes for x64 too * Revert "Add verbosity to release pipelines builds of PowerTOys" This reverts commit654d4a7f78. * [Dev] Update CodeAnalysis library to non-preview package * Remove unneeded warning removal * Fix Notice.md * Rename VCToolsVersion file and task name * Remove unneeded mutex header include --------- Co-authored-by: Jaime Bernardo <jaime@janeasystems.com>
136 lines
5.1 KiB
C#
136 lines
5.1 KiB
C#
// Copyright (c) Microsoft Corporation
|
|
// The Microsoft Corporation licenses this file to you under the MIT license.
|
|
// See the LICENSE file in the project root for more information.
|
|
|
|
// Code forked from Betsegaw Tadele's https://github.com/betsegaw/windowwalker/
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
using System.Linq;
|
|
|
|
namespace Microsoft.Plugin.WindowWalker.Components
|
|
{
|
|
/// <summary>
|
|
/// Class housing fuzzy matching methods
|
|
/// </summary>
|
|
internal static class FuzzyMatching
|
|
{
|
|
/// <summary>
|
|
/// Finds the best match (the one with the most
|
|
/// number of letters adjacent to each other) and
|
|
/// returns the index location of each of the letters
|
|
/// of the matches
|
|
/// </summary>
|
|
/// <param name="text">The text to search inside of</param>
|
|
/// <param name="searchText">the text to search for</param>
|
|
/// <returns>returns the index location of each of the letters of the matches</returns>
|
|
internal static List<int> FindBestFuzzyMatch(string text, string searchText)
|
|
{
|
|
ArgumentNullException.ThrowIfNull(searchText);
|
|
|
|
ArgumentNullException.ThrowIfNull(text);
|
|
|
|
// Using CurrentCulture since this is user facing
|
|
searchText = searchText.ToLower(CultureInfo.CurrentCulture);
|
|
text = text.ToLower(CultureInfo.CurrentCulture);
|
|
|
|
// Create a grid to march matches like
|
|
// eg.
|
|
// a b c a d e c f g
|
|
// a x x
|
|
// c x x
|
|
bool[,] matches = new bool[text.Length, searchText.Length];
|
|
for (int firstIndex = 0; firstIndex < text.Length; firstIndex++)
|
|
{
|
|
for (int secondIndex = 0; secondIndex < searchText.Length; secondIndex++)
|
|
{
|
|
matches[firstIndex, secondIndex] =
|
|
searchText[secondIndex] == text[firstIndex] ?
|
|
true :
|
|
false;
|
|
}
|
|
}
|
|
|
|
// use this table to get all the possible matches
|
|
List<List<int>> allMatches = GetAllMatchIndexes(matches);
|
|
|
|
// return the score that is the max
|
|
int maxScore = allMatches.Count > 0 ? CalculateScoreForMatches(allMatches[0]) : 0;
|
|
List<int> bestMatch = allMatches.Count > 0 ? allMatches[0] : new List<int>();
|
|
|
|
foreach (var match in allMatches)
|
|
{
|
|
int score = CalculateScoreForMatches(match);
|
|
if (score > maxScore)
|
|
{
|
|
bestMatch = match;
|
|
maxScore = score;
|
|
}
|
|
}
|
|
|
|
return bestMatch;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets all the possible matches to the search string with in the text
|
|
/// </summary>
|
|
/// <param name="matches"> a table showing the matches as generated by
|
|
/// a two dimensional array with the first dimension the text and the second
|
|
/// one the search string and each cell marked as an intersection between the two</param>
|
|
/// <returns>a list of the possible combinations that match the search text</returns>
|
|
internal static List<List<int>> GetAllMatchIndexes(bool[,] matches)
|
|
{
|
|
ArgumentNullException.ThrowIfNull(matches);
|
|
|
|
List<List<int>> results = new List<List<int>>();
|
|
|
|
for (int secondIndex = 0; secondIndex < matches.GetLength(1); secondIndex++)
|
|
{
|
|
for (int firstIndex = 0; firstIndex < matches.GetLength(0); firstIndex++)
|
|
{
|
|
if (secondIndex == 0 && matches[firstIndex, secondIndex])
|
|
{
|
|
results.Add(new List<int> { firstIndex });
|
|
}
|
|
else if (matches[firstIndex, secondIndex])
|
|
{
|
|
var tempList = results.Where(x => x.Count == secondIndex && x[x.Count - 1] < firstIndex).Select(x => x.ToList()).ToList();
|
|
|
|
foreach (var pathSofar in tempList)
|
|
{
|
|
pathSofar.Add(firstIndex);
|
|
}
|
|
|
|
results.AddRange(tempList);
|
|
}
|
|
}
|
|
|
|
results = results.Where(x => x.Count == secondIndex + 1).ToList();
|
|
}
|
|
|
|
return results.Where(x => x.Count == matches.GetLength(1)).ToList();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Calculates the score for a string
|
|
/// </summary>
|
|
/// <param name="matches">the index of the matches</param>
|
|
/// <returns>an integer representing the score</returns>
|
|
internal static int CalculateScoreForMatches(List<int> matches)
|
|
{
|
|
ArgumentNullException.ThrowIfNull(matches);
|
|
|
|
var score = 0;
|
|
|
|
for (int currentIndex = 1; currentIndex < matches.Count; currentIndex++)
|
|
{
|
|
var previousIndex = currentIndex - 1;
|
|
|
|
score -= matches[currentIndex] - matches[previousIndex];
|
|
}
|
|
|
|
return score == 0 ? -10000 : score;
|
|
}
|
|
}
|
|
}
|