mirror of
https://github.com/microsoft/PowerToys.git
synced 2025-12-15 19:27:56 +01: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>
186 lines
6.6 KiB
C#
186 lines
6.6 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.
|
|
|
|
using System;
|
|
using System.Buffers;
|
|
using System.Collections.Generic;
|
|
using System.Collections.Immutable;
|
|
using System.Globalization;
|
|
using System.IO.Abstractions;
|
|
using System.Linq;
|
|
using ManagedCommon;
|
|
using Microsoft.Plugin.Folder.Sources.Result;
|
|
|
|
namespace Microsoft.Plugin.Folder.Sources
|
|
{
|
|
public class QueryInternalDirectory : IQueryInternalDirectory
|
|
{
|
|
private static readonly SearchValues<char> PathChars = SearchValues.Create("\\/");
|
|
private readonly FolderSettings _settings;
|
|
private readonly IQueryFileSystemInfo _queryFileSystemInfo;
|
|
private readonly IDirectory _directory;
|
|
|
|
private static readonly HashSet<char> SpecialSearchChars = new HashSet<char>
|
|
{
|
|
'?', '*', '>',
|
|
};
|
|
|
|
private static string _warningIconPath;
|
|
|
|
public QueryInternalDirectory(FolderSettings folderSettings, IQueryFileSystemInfo queryFileSystemInfo, IDirectory directory)
|
|
{
|
|
_settings = folderSettings;
|
|
_queryFileSystemInfo = queryFileSystemInfo;
|
|
_directory = directory;
|
|
}
|
|
|
|
private static bool HasSpecialChars(string search)
|
|
{
|
|
return search.Any(c => SpecialSearchChars.Contains(c));
|
|
}
|
|
|
|
public static bool RecursiveSearch(string query)
|
|
{
|
|
// give the ability to search all folder when it contains a >
|
|
return query.Any(c => c.Equals('>'));
|
|
}
|
|
|
|
private (string Search, string IncompleteName) Process(string search)
|
|
{
|
|
string incompleteName = string.Empty;
|
|
if (HasSpecialChars(search) || !_directory.Exists($@"{search}\"))
|
|
{
|
|
// if folder doesn't exist, we want to take the last part and use it afterwards to help the user
|
|
// find the right folder.
|
|
int index = search.AsSpan().LastIndexOfAny(PathChars);
|
|
|
|
// No slashes found, so probably not a folder
|
|
if (index <= 0 || index >= search.Length - 1)
|
|
{
|
|
return default;
|
|
}
|
|
|
|
// Remove everything after the last \ and add *
|
|
// Using InvariantCulture since this is internal
|
|
incompleteName = search.Substring(index + 1)
|
|
.ToLower(CultureInfo.InvariantCulture) + "*";
|
|
search = search.Substring(0, index + 1);
|
|
if (!_directory.Exists(search))
|
|
{
|
|
return default;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// folder exist, add \ at the end of doesn't exist
|
|
// Using Ordinal since this is internal and is used for a symbol
|
|
if (!search.EndsWith(@"\", StringComparison.Ordinal))
|
|
{
|
|
search += @"\";
|
|
}
|
|
}
|
|
|
|
return (search, incompleteName);
|
|
}
|
|
|
|
public IEnumerable<IItemResult> Query(string search)
|
|
{
|
|
ArgumentNullException.ThrowIfNull(search);
|
|
|
|
var processed = Process(search);
|
|
|
|
if (processed == default)
|
|
{
|
|
yield break;
|
|
}
|
|
|
|
var (querySearch, incompleteName) = processed;
|
|
var isRecursive = RecursiveSearch(incompleteName);
|
|
|
|
if (isRecursive)
|
|
{
|
|
// match everything before and after search term using supported wildcard '*', ie. *searchterm*
|
|
if (string.IsNullOrEmpty(incompleteName))
|
|
{
|
|
incompleteName = "*";
|
|
}
|
|
else
|
|
{
|
|
incompleteName = string.Concat("*", incompleteName.AsSpan(1));
|
|
}
|
|
}
|
|
|
|
yield return new CreateOpenCurrentFolderResult(querySearch);
|
|
|
|
// Note: Take 1000 is so that you don't search the whole system before you discard
|
|
var lookup = _queryFileSystemInfo.MatchFileSystemInfo(querySearch, incompleteName, isRecursive)
|
|
.Take(1000)
|
|
.ToLookup(r => r.Type);
|
|
|
|
var folderList = lookup[DisplayType.Directory].ToImmutableArray();
|
|
var fileList = lookup[DisplayType.File].ToImmutableArray();
|
|
|
|
var fileSystemResult = GenerateFolderResults(querySearch, folderList)
|
|
.Concat<IItemResult>(GenerateFileResults(querySearch, fileList))
|
|
.ToImmutableArray();
|
|
|
|
foreach (var result in fileSystemResult)
|
|
{
|
|
yield return result;
|
|
}
|
|
|
|
// Show warning message if result has been truncated
|
|
if (folderList.Length > _settings.MaxFolderResults || fileList.Length > _settings.MaxFileResults)
|
|
{
|
|
yield return GenerateTruncatedItemResult(folderList.Length + fileList.Length, fileSystemResult.Length);
|
|
}
|
|
}
|
|
|
|
private IEnumerable<FileItemResult> GenerateFileResults(string search, IEnumerable<DisplayFileInfo> fileList)
|
|
{
|
|
return fileList
|
|
.Select(fileSystemInfo => new FileItemResult()
|
|
{
|
|
FilePath = fileSystemInfo.FullName,
|
|
Search = search,
|
|
})
|
|
.OrderBy(x => x.Title)
|
|
.Take(_settings.MaxFileResults);
|
|
}
|
|
|
|
private IEnumerable<FolderItemResult> GenerateFolderResults(string search, IEnumerable<DisplayFileInfo> folderList)
|
|
{
|
|
return folderList
|
|
.Select(fileSystemInfo => new FolderItemResult(fileSystemInfo)
|
|
{
|
|
Search = search,
|
|
})
|
|
.OrderBy(x => x.Title)
|
|
.Take(_settings.MaxFolderResults);
|
|
}
|
|
|
|
private static TruncatedItemResult GenerateTruncatedItemResult(int preTruncationCount, int postTruncationCount)
|
|
{
|
|
return new TruncatedItemResult()
|
|
{
|
|
PreTruncationCount = preTruncationCount,
|
|
PostTruncationCount = postTruncationCount,
|
|
WarningIconPath = _warningIconPath,
|
|
};
|
|
}
|
|
|
|
public static void SetWarningIcon(Theme theme)
|
|
{
|
|
if (theme == Theme.Light || theme == Theme.HighContrastWhite)
|
|
{
|
|
_warningIconPath = "Images/Warning.light.png";
|
|
}
|
|
else
|
|
{
|
|
_warningIconPath = "Images/Warning.dark.png";
|
|
}
|
|
}
|
|
}
|
|
}
|