mirror of
https://github.com/goatcorp/Dalamud.git
synced 2025-12-13 12:14:16 +01:00
Make FuzzyMatcher more convenient to use
This commit is contained in:
parent
d3bd5f1dce
commit
361a0a95e9
2 changed files with 178 additions and 158 deletions
|
|
@ -1252,7 +1252,7 @@ internal class PluginInstallerWindow : Window, IDisposable
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
IEnumerable<IChangelogEntry> changelogs = null;
|
IEnumerable<IChangelogEntry>? changelogs = null;
|
||||||
if (displayDalamud && displayPlugins && this.dalamudChangelogManager.Changelogs != null)
|
if (displayDalamud && displayPlugins && this.dalamudChangelogManager.Changelogs != null)
|
||||||
{
|
{
|
||||||
changelogs = this.dalamudChangelogManager.Changelogs;
|
changelogs = this.dalamudChangelogManager.Changelogs;
|
||||||
|
|
@ -1266,10 +1266,15 @@ internal class PluginInstallerWindow : Window, IDisposable
|
||||||
changelogs = this.dalamudChangelogManager.Changelogs.OfType<PluginChangelogEntry>();
|
changelogs = this.dalamudChangelogManager.Changelogs.OfType<PluginChangelogEntry>();
|
||||||
}
|
}
|
||||||
|
|
||||||
var sortedChangelogs = changelogs?.Where(x => this.searchText.IsNullOrWhitespace() || new FuzzyMatcher(this.searchText.ToLowerInvariant(), MatchMode.FuzzyParts).Matches(x.Title.ToLowerInvariant()) > 0)
|
changelogs ??= Array.Empty<IChangelogEntry>();
|
||||||
.OrderByDescending(x => x.Date).ToList();
|
var sortedChangelogs =
|
||||||
|
this.searchText.IsNullOrWhitespace()
|
||||||
|
? changelogs.ToList()
|
||||||
|
: changelogs.Where(x => x.Title.FuzzyMatchesParts(this.searchText))
|
||||||
|
.OrderByDescending(x => x.Date)
|
||||||
|
.ToList();
|
||||||
|
|
||||||
if (sortedChangelogs == null || sortedChangelogs.Count == 0)
|
if (sortedChangelogs.Count == 0)
|
||||||
{
|
{
|
||||||
ImGui.TextColored(
|
ImGui.TextColored(
|
||||||
ImGuiColors.DalamudGrey2,
|
ImGuiColors.DalamudGrey2,
|
||||||
|
|
@ -3793,22 +3798,24 @@ internal class PluginInstallerWindow : Window, IDisposable
|
||||||
|
|
||||||
private int GetManifestSearchScore(IPluginManifest manifest)
|
private int GetManifestSearchScore(IPluginManifest manifest)
|
||||||
{
|
{
|
||||||
var searchString = this.searchText.ToLowerInvariant();
|
var loc = Localization.GetCultureInfoFromLangCode(Service<DalamudConfiguration>.Get().EffectiveLanguage);
|
||||||
var matcher = new FuzzyMatcher(searchString, MatchMode.FuzzyParts);
|
var maxScore = 0;
|
||||||
var scores = new List<int> { 0 };
|
|
||||||
|
|
||||||
if (!manifest.Name.IsNullOrEmpty())
|
if (manifest.Name.FuzzyMatches(this.searchText, FuzzyMatcher.Mode.FuzzyParts, loc, out var score))
|
||||||
scores.Add(matcher.Matches(manifest.Name.ToLowerInvariant()) * 110);
|
maxScore = Math.Max(maxScore, score * 110);
|
||||||
if (!manifest.InternalName.IsNullOrEmpty())
|
if (manifest.InternalName.FuzzyMatches(this.searchText, FuzzyMatcher.Mode.FuzzyParts, loc, out score))
|
||||||
scores.Add(matcher.Matches(manifest.InternalName.ToLowerInvariant()) * 105);
|
maxScore = Math.Max(maxScore, score * 105);
|
||||||
if (!manifest.Author.IsNullOrEmpty())
|
if (manifest.Author.FuzzyMatches(this.searchText, FuzzyMatcher.Mode.FuzzyParts, loc, out score))
|
||||||
scores.Add(matcher.Matches(manifest.Author.ToLowerInvariant()) * 100);
|
maxScore = Math.Max(maxScore, score * 100);
|
||||||
if (!manifest.Punchline.IsNullOrEmpty())
|
if (manifest.Punchline.FuzzyMatches(this.searchText, FuzzyMatcher.Mode.FuzzyParts, loc, out score))
|
||||||
scores.Add(matcher.Matches(manifest.Punchline.ToLowerInvariant()) * 100);
|
maxScore = Math.Max(maxScore, score * 100);
|
||||||
if (manifest.Tags != null)
|
foreach (var tag in manifest.Tags ?? [])
|
||||||
scores.Add(matcher.MatchesAny(manifest.Tags.ToArray()) * 100);
|
{
|
||||||
|
if (tag.FuzzyMatches(this.searchText, FuzzyMatcher.Mode.FuzzyParts, loc, out score))
|
||||||
|
maxScore = Math.Max(maxScore, score * 100);
|
||||||
|
}
|
||||||
|
|
||||||
return scores.Max();
|
return maxScore;
|
||||||
}
|
}
|
||||||
|
|
||||||
private (bool IsInstalled, LocalPlugin Plugin) IsManifestInstalled(IPluginManifest? manifest)
|
private (bool IsInstalled, LocalPlugin Plugin) IsManifestInstalled(IPluginManifest? manifest)
|
||||||
|
|
|
||||||
|
|
@ -1,168 +1,140 @@
|
||||||
#define BORDER_MATCHING
|
#define BORDER_MATCHING
|
||||||
|
|
||||||
using System.Collections.Generic;
|
using System.Globalization;
|
||||||
using System.Runtime.CompilerServices;
|
using System.Runtime.CompilerServices;
|
||||||
|
|
||||||
namespace Dalamud.Utility;
|
namespace Dalamud.Utility;
|
||||||
|
|
||||||
#pragma warning disable SA1600
|
#pragma warning disable SA1600
|
||||||
#pragma warning disable SA1602
|
#pragma warning disable SA1602
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Specify fuzzy match mode.
|
||||||
|
/// </summary>
|
||||||
internal enum MatchMode
|
internal enum MatchMode
|
||||||
{
|
{
|
||||||
Simple,
|
Simple,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// The string is considered for fuzzy matching as a whole.
|
||||||
|
/// </summary>
|
||||||
Fuzzy,
|
Fuzzy,
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Each part of the string, separated by whitespace, is considered for fuzzy matching; each part must match in a fuzzy way.
|
||||||
|
/// </summary>
|
||||||
FuzzyParts,
|
FuzzyParts,
|
||||||
}
|
}
|
||||||
|
|
||||||
internal readonly ref struct FuzzyMatcher
|
/// <summary>
|
||||||
|
/// Matches a string in a fuzzy way.
|
||||||
|
/// </summary>
|
||||||
|
internal static class FuzzyMatcher
|
||||||
{
|
{
|
||||||
private static readonly (int, int)[] EmptySegArray = Array.Empty<(int, int)>();
|
/// <summary>
|
||||||
|
/// Specify fuzzy match mode.
|
||||||
private readonly string needleString = string.Empty;
|
/// </summary>
|
||||||
private readonly ReadOnlySpan<char> needleSpan = ReadOnlySpan<char>.Empty;
|
internal enum Mode
|
||||||
private readonly int needleFinalPosition = -1;
|
|
||||||
private readonly (int Start, int End)[] needleSegments = EmptySegArray;
|
|
||||||
private readonly MatchMode mode = MatchMode.Simple;
|
|
||||||
|
|
||||||
public FuzzyMatcher(string term, MatchMode matchMode)
|
|
||||||
{
|
{
|
||||||
this.needleString = term;
|
/// <summary>
|
||||||
this.needleSpan = this.needleString.AsSpan();
|
/// The string is considered for fuzzy matching as a whole.
|
||||||
this.needleFinalPosition = this.needleSpan.Length - 1;
|
/// </summary>
|
||||||
this.mode = matchMode;
|
Fuzzy,
|
||||||
|
|
||||||
switch (matchMode)
|
/// <summary>
|
||||||
|
/// Each part of the string, separated by whitespace, is considered for fuzzy matching; each part must match in a fuzzy way.
|
||||||
|
/// </summary>
|
||||||
|
FuzzyParts,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Determines if <paramref name="needle"/> can be found in <paramref name="haystack"/> in a fuzzy way.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="haystack">The string to search from.</param>
|
||||||
|
/// <param name="needle">The substring to search for.</param>
|
||||||
|
/// <param name="mode">Fuzzy match mode.</param>
|
||||||
|
/// <param name="cultureInfo">Culture info for case insensitive matching.</param>
|
||||||
|
/// <param name="score">The score. 0 means that the string did not match. The scores are meaningful only across matches using the same <paramref name="needle"/> value.</param>
|
||||||
|
/// <returns><c>true</c> if matches.</returns>
|
||||||
|
public static bool FuzzyMatches(
|
||||||
|
this ReadOnlySpan<char> haystack,
|
||||||
|
ReadOnlySpan<char> needle,
|
||||||
|
Mode mode,
|
||||||
|
CultureInfo cultureInfo,
|
||||||
|
out int score)
|
||||||
{
|
{
|
||||||
case MatchMode.FuzzyParts:
|
score = 0;
|
||||||
this.needleSegments = FindNeedleSegments(this.needleSpan);
|
switch (mode)
|
||||||
|
{
|
||||||
|
case var _ when needle.Length == 0:
|
||||||
|
score = 0;
|
||||||
break;
|
break;
|
||||||
case MatchMode.Fuzzy:
|
|
||||||
case MatchMode.Simple:
|
case Mode.Fuzzy:
|
||||||
this.needleSegments = EmptySegArray;
|
score = GetRawScore(haystack, needle, cultureInfo);
|
||||||
break;
|
break;
|
||||||
default:
|
|
||||||
throw new ArgumentOutOfRangeException(nameof(matchMode), matchMode, null);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static (int Start, int End)[] FindNeedleSegments(ReadOnlySpan<char> span)
|
case Mode.FuzzyParts:
|
||||||
|
foreach (var needleSegment in new WordEnumerator(needle))
|
||||||
{
|
{
|
||||||
var segments = new List<(int, int)>();
|
var cur = GetRawScore(haystack, needleSegment, cultureInfo);
|
||||||
var wordStart = -1;
|
|
||||||
|
|
||||||
for (var i = 0; i < span.Length; i++)
|
|
||||||
{
|
|
||||||
if (span[i] is not ' ' and not '\u3000')
|
|
||||||
{
|
|
||||||
if (wordStart < 0)
|
|
||||||
{
|
|
||||||
wordStart = i;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (wordStart >= 0)
|
|
||||||
{
|
|
||||||
segments.Add((wordStart, i - 1));
|
|
||||||
wordStart = -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (wordStart >= 0)
|
|
||||||
{
|
|
||||||
segments.Add((wordStart, span.Length - 1));
|
|
||||||
}
|
|
||||||
|
|
||||||
return segments.ToArray();
|
|
||||||
}
|
|
||||||
|
|
||||||
#pragma warning disable SA1202
|
|
||||||
public int Matches(string value)
|
|
||||||
#pragma warning restore SA1202
|
|
||||||
{
|
|
||||||
if (this.needleFinalPosition < 0)
|
|
||||||
{
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.mode == MatchMode.Simple)
|
|
||||||
{
|
|
||||||
return value.Contains(this.needleString) ? 1 : 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
var haystack = value.AsSpan();
|
|
||||||
|
|
||||||
if (this.mode == MatchMode.Fuzzy)
|
|
||||||
{
|
|
||||||
return this.GetRawScore(haystack, 0, this.needleFinalPosition);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.mode == MatchMode.FuzzyParts)
|
|
||||||
{
|
|
||||||
if (this.needleSegments.Length < 2)
|
|
||||||
{
|
|
||||||
return this.GetRawScore(haystack, 0, this.needleFinalPosition);
|
|
||||||
}
|
|
||||||
|
|
||||||
var total = 0;
|
|
||||||
for (var i = 0; i < this.needleSegments.Length; i++)
|
|
||||||
{
|
|
||||||
var (start, end) = this.needleSegments[i];
|
|
||||||
var cur = this.GetRawScore(haystack, start, end);
|
|
||||||
if (cur == 0)
|
if (cur == 0)
|
||||||
{
|
{
|
||||||
return 0;
|
score = 0;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
total += cur;
|
score += cur;
|
||||||
}
|
}
|
||||||
|
|
||||||
return total;
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
throw new ArgumentOutOfRangeException(nameof(mode), mode, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 8;
|
return score > 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
public int MatchesAny(params string[] values)
|
/// <inheritdoc cref="FuzzyMatches(ReadOnlySpan{char},ReadOnlySpan{char},Mode,CultureInfo,out int)"/>
|
||||||
|
public static bool FuzzyMatches(
|
||||||
|
this string? haystack,
|
||||||
|
ReadOnlySpan<char> needle,
|
||||||
|
Mode mode,
|
||||||
|
CultureInfo cultureInfo,
|
||||||
|
out int score) => haystack.AsSpan().FuzzyMatches(needle, mode, cultureInfo, out score);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Determines if <paramref name="needle"/> can be found in <paramref name="haystack"/> using the mode
|
||||||
|
/// <see cref="Mode.FuzzyParts"/>.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="haystack">The string to search from.</param>
|
||||||
|
/// <param name="needle">The substring to search for.</param>
|
||||||
|
/// <returns><c>true</c> if matches.</returns>
|
||||||
|
public static bool FuzzyMatchesParts(this string? haystack, ReadOnlySpan<char> needle) =>
|
||||||
|
haystack.FuzzyMatches(needle, Mode.FuzzyParts, CultureInfo.InvariantCulture, out _);
|
||||||
|
|
||||||
|
private static int GetRawScore(ReadOnlySpan<char> haystack, ReadOnlySpan<char> needle, CultureInfo cultureInfo)
|
||||||
{
|
{
|
||||||
var max = 0;
|
var (startPos, gaps, consecutive, borderMatches, endPos) = FindForward(haystack, needle, cultureInfo);
|
||||||
for (var i = 0; i < values.Length; i++)
|
|
||||||
{
|
|
||||||
var cur = this.Matches(values[i]);
|
|
||||||
if (cur > max)
|
|
||||||
{
|
|
||||||
max = cur;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return max;
|
|
||||||
}
|
|
||||||
|
|
||||||
private int GetRawScore(ReadOnlySpan<char> haystack, int needleStart, int needleEnd)
|
|
||||||
{
|
|
||||||
var (startPos, gaps, consecutive, borderMatches, endPos) = this.FindForward(haystack, needleStart, needleEnd);
|
|
||||||
if (startPos < 0)
|
if (startPos < 0)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
var needleSize = needleEnd - needleStart + 1;
|
var score = CalculateRawScore(needle.Length, startPos, gaps, consecutive, borderMatches);
|
||||||
|
|
||||||
var score = CalculateRawScore(needleSize, startPos, gaps, consecutive, borderMatches);
|
|
||||||
// PluginLog.Debug(
|
// PluginLog.Debug(
|
||||||
// $"['{needleString.Substring(needleStart, needleEnd - needleStart + 1)}' in '{haystack}'] fwd: needleSize={needleSize} startPos={startPos} gaps={gaps} consecutive={consecutive} borderMatches={borderMatches} score={score}");
|
// $"['{needle.Substring(needleStart, needleEnd - needleStart + 1)}' in '{haystack}'] fwd: needleSize={needleSize} startPos={startPos} gaps={gaps} consecutive={consecutive} borderMatches={borderMatches} score={score}");
|
||||||
|
|
||||||
(startPos, gaps, consecutive, borderMatches) = this.FindReverse(haystack, endPos, needleStart, needleEnd);
|
(startPos, gaps, consecutive, borderMatches) = FindReverse(haystack[..(endPos + 1)], needle, cultureInfo);
|
||||||
var revScore = CalculateRawScore(needleSize, startPos, gaps, consecutive, borderMatches);
|
var revScore = CalculateRawScore(needle.Length, startPos, gaps, consecutive, borderMatches);
|
||||||
// PluginLog.Debug(
|
// PluginLog.Debug(
|
||||||
// $"['{needleString.Substring(needleStart, needleEnd - needleStart + 1)}' in '{haystack}'] rev: needleSize={needleSize} startPos={startPos} gaps={gaps} consecutive={consecutive} borderMatches={borderMatches} score={revScore}");
|
// $"['{needle.Substring(needleStart, needleEnd - needleStart + 1)}' in '{haystack}'] rev: needleSize={needleSize} startPos={startPos} gaps={gaps} consecutive={consecutive} borderMatches={borderMatches} score={revScore}");
|
||||||
|
|
||||||
return int.Max(score, revScore);
|
return int.Max(score, revScore);
|
||||||
}
|
}
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
#pragma warning disable SA1204
|
|
||||||
private static int CalculateRawScore(int needleSize, int startPos, int gaps, int consecutive, int borderMatches)
|
private static int CalculateRawScore(int needleSize, int startPos, int gaps, int consecutive, int borderMatches)
|
||||||
#pragma warning restore SA1204
|
|
||||||
{
|
{
|
||||||
var score = 100
|
var score = 100
|
||||||
+ needleSize * 3
|
+ needleSize * 3
|
||||||
|
|
@ -175,10 +147,12 @@ internal readonly ref struct FuzzyMatcher
|
||||||
return score < 1 ? 1 : score;
|
return score < 1 ? 1 : score;
|
||||||
}
|
}
|
||||||
|
|
||||||
private (int StartPos, int Gaps, int Consecutive, int BorderMatches, int HaystackIndex) FindForward(
|
private static (int StartPos, int Gaps, int Consecutive, int BorderMatches, int HaystackIndex) FindForward(
|
||||||
ReadOnlySpan<char> haystack, int needleStart, int needleEnd)
|
ReadOnlySpan<char> haystack,
|
||||||
|
ReadOnlySpan<char> needle,
|
||||||
|
CultureInfo cultureInfo)
|
||||||
{
|
{
|
||||||
var needleIndex = needleStart;
|
var needleIndex = 0;
|
||||||
var lastMatchIndex = -10;
|
var lastMatchIndex = -10;
|
||||||
|
|
||||||
var startPos = 0;
|
var startPos = 0;
|
||||||
|
|
@ -188,7 +162,7 @@ internal readonly ref struct FuzzyMatcher
|
||||||
|
|
||||||
for (var haystackIndex = 0; haystackIndex < haystack.Length; haystackIndex++)
|
for (var haystackIndex = 0; haystackIndex < haystack.Length; haystackIndex++)
|
||||||
{
|
{
|
||||||
if (haystack[haystackIndex] == this.needleSpan[needleIndex])
|
if (char.ToLower(haystack[haystackIndex], cultureInfo) == char.ToLower(needle[needleIndex], cultureInfo))
|
||||||
{
|
{
|
||||||
#if BORDER_MATCHING
|
#if BORDER_MATCHING
|
||||||
if (haystackIndex > 0)
|
if (haystackIndex > 0)
|
||||||
|
|
@ -207,7 +181,7 @@ internal readonly ref struct FuzzyMatcher
|
||||||
consecutive++;
|
consecutive++;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (needleIndex > needleEnd)
|
if (needleIndex >= needle.Length)
|
||||||
{
|
{
|
||||||
return (startPos, gaps, consecutive, borderMatches, haystackIndex);
|
return (startPos, gaps, consecutive, borderMatches, haystackIndex);
|
||||||
}
|
}
|
||||||
|
|
@ -216,7 +190,7 @@ internal readonly ref struct FuzzyMatcher
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (needleIndex > needleStart)
|
if (needleIndex > 0)
|
||||||
{
|
{
|
||||||
gaps++;
|
gaps++;
|
||||||
}
|
}
|
||||||
|
|
@ -230,19 +204,21 @@ internal readonly ref struct FuzzyMatcher
|
||||||
return (-1, 0, 0, 0, 0);
|
return (-1, 0, 0, 0, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
private (int StartPos, int Gaps, int Consecutive, int BorderMatches) FindReverse(
|
private static (int StartPos, int Gaps, int Consecutive, int BorderMatches) FindReverse(
|
||||||
ReadOnlySpan<char> haystack, int haystackLastMatchIndex, int needleStart, int needleEnd)
|
ReadOnlySpan<char> haystack,
|
||||||
|
ReadOnlySpan<char> needle,
|
||||||
|
CultureInfo cultureInfo)
|
||||||
{
|
{
|
||||||
var needleIndex = needleEnd;
|
var needleIndex = needle.Length - 1;
|
||||||
var revLastMatchIndex = haystack.Length + 10;
|
var revLastMatchIndex = haystack.Length + 10;
|
||||||
|
|
||||||
var gaps = 0;
|
var gaps = 0;
|
||||||
var consecutive = 0;
|
var consecutive = 0;
|
||||||
var borderMatches = 0;
|
var borderMatches = 0;
|
||||||
|
|
||||||
for (var haystackIndex = haystackLastMatchIndex; haystackIndex >= 0; haystackIndex--)
|
for (var haystackIndex = haystack.Length - 1; haystackIndex >= 0; haystackIndex--)
|
||||||
{
|
{
|
||||||
if (haystack[haystackIndex] == this.needleSpan[needleIndex])
|
if (char.ToLower(haystack[haystackIndex], cultureInfo) == char.ToLower(needle[needleIndex], cultureInfo))
|
||||||
{
|
{
|
||||||
#if BORDER_MATCHING
|
#if BORDER_MATCHING
|
||||||
if (haystackIndex > 0)
|
if (haystackIndex > 0)
|
||||||
|
|
@ -261,7 +237,7 @@ internal readonly ref struct FuzzyMatcher
|
||||||
consecutive++;
|
consecutive++;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (needleIndex < needleStart)
|
if (needleIndex < 0)
|
||||||
{
|
{
|
||||||
return (haystackIndex, gaps, consecutive, borderMatches);
|
return (haystackIndex, gaps, consecutive, borderMatches);
|
||||||
}
|
}
|
||||||
|
|
@ -276,7 +252,44 @@ internal readonly ref struct FuzzyMatcher
|
||||||
|
|
||||||
return (-1, 0, 0, 0);
|
return (-1, 0, 0, 0);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
#pragma warning restore SA1600
|
private ref struct WordEnumerator
|
||||||
#pragma warning restore SA1602
|
{
|
||||||
|
private readonly ReadOnlySpan<char> fullNeedle;
|
||||||
|
private int start = -1;
|
||||||
|
private int end = 0;
|
||||||
|
|
||||||
|
public WordEnumerator(ReadOnlySpan<char> fullNeedle)
|
||||||
|
{
|
||||||
|
this.fullNeedle = fullNeedle;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ReadOnlySpan<char> Current
|
||||||
|
{
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
get => this.fullNeedle[this.start..this.end];
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public bool MoveNext()
|
||||||
|
{
|
||||||
|
if (this.start >= this.fullNeedle.Length - 1)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
this.start = this.end;
|
||||||
|
|
||||||
|
// Skip the spaces
|
||||||
|
while (this.start < this.fullNeedle.Length && char.IsWhiteSpace(this.fullNeedle[this.start]))
|
||||||
|
this.start++;
|
||||||
|
|
||||||
|
this.end = this.start;
|
||||||
|
while (this.end < this.fullNeedle.Length && !char.IsWhiteSpace(this.fullNeedle[this.end]))
|
||||||
|
this.end++;
|
||||||
|
|
||||||
|
return this.start != this.end;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public WordEnumerator GetEnumerator() => this;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue