mirror of
https://github.com/ppy/osu.git
synced 2024-12-17 22:22:54 +08:00
Implement interval list
This commit is contained in:
parent
cc0bcea6e5
commit
91b3aa2914
128
osu.Game/Lists/IntervalList.cs
Normal file
128
osu.Game/Lists/IntervalList.cs
Normal file
@ -0,0 +1,128 @@
|
||||
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
||||
// See the LICENCE file in the repository root for full licence text.
|
||||
|
||||
using System;
|
||||
using System.Collections;
|
||||
using System.Collections.Generic;
|
||||
using osu.Framework.Lists;
|
||||
|
||||
namespace osu.Game.Lists
|
||||
{
|
||||
/// <summary>
|
||||
/// Represents a list of intervals that can be used for whether a specific value falls into one of them.
|
||||
/// </summary>
|
||||
/// <typeparam name="T">The type of interval values.</typeparam>
|
||||
public class IntervalList<T> : ICollection<Interval<T>>, IReadOnlyList<Interval<T>>
|
||||
{
|
||||
private static readonly IComparer<T> type_comparer = Comparer<T>.Default;
|
||||
|
||||
private readonly SortedList<Interval<T>> intervals = new SortedList<Interval<T>>((x, y) => type_comparer.Compare(x.Start, y.Start));
|
||||
|
||||
/// <summary>
|
||||
/// The index of the nearest interval from last <see cref="IsInAnyInterval"/> call.
|
||||
/// </summary>
|
||||
protected int NearestIntervalIndex;
|
||||
|
||||
/// <summary>
|
||||
/// Whether the provided value is in any interval added to this list.
|
||||
/// </summary>
|
||||
/// <param name="value">The value to check for.</param>
|
||||
public bool IsInAnyInterval(T value)
|
||||
{
|
||||
if (intervals.Count == 0)
|
||||
return false;
|
||||
|
||||
// Clamp the nearest index in case there were intervals
|
||||
// removed from the list causing the index to go out of range.
|
||||
NearestIntervalIndex = Math.Clamp(NearestIntervalIndex, 0, Count - 1);
|
||||
|
||||
if (type_comparer.Compare(value, this[NearestIntervalIndex].End) > 0)
|
||||
{
|
||||
while (type_comparer.Compare(value, this[NearestIntervalIndex].End) > 0 && NearestIntervalIndex < Count - 1)
|
||||
NearestIntervalIndex++;
|
||||
}
|
||||
else
|
||||
{
|
||||
while (type_comparer.Compare(value, this[NearestIntervalIndex].Start) < 0 && NearestIntervalIndex > 0)
|
||||
NearestIntervalIndex--;
|
||||
}
|
||||
|
||||
var nearestInterval = this[NearestIntervalIndex];
|
||||
|
||||
return type_comparer.Compare(value, nearestInterval.Start) >= 0 &&
|
||||
type_comparer.Compare(value, nearestInterval.End) <= 0;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Adds a new interval to the list.
|
||||
/// </summary>
|
||||
/// <param name="start">The start value of the interval.</param>
|
||||
/// <param name="end">The end value of the interval.</param>
|
||||
public void Add(T start, T end) => Add(new Interval<T>(start, end));
|
||||
|
||||
#region ICollection<Interval<T>>
|
||||
|
||||
public int Count => intervals.Count;
|
||||
|
||||
bool ICollection<Interval<T>>.IsReadOnly => false;
|
||||
|
||||
/// <summary>
|
||||
/// Adds a new interval to the list
|
||||
/// </summary>
|
||||
/// <param name="interval">The interval to add.</param>
|
||||
public void Add(Interval<T> interval) => intervals.Add(interval);
|
||||
|
||||
/// <summary>
|
||||
/// Removes an existing interval from the list.
|
||||
/// </summary>
|
||||
/// <param name="interval">The interval to remove.</param>
|
||||
/// <returns>Whether the provided interval exists in the list and has been removed.</returns>
|
||||
public bool Remove(Interval<T> interval) => intervals.Remove(interval);
|
||||
|
||||
/// <summary>
|
||||
/// Removes all intervals from the list.
|
||||
/// </summary>
|
||||
public void Clear() => intervals.Clear();
|
||||
|
||||
public void CopyTo(Interval<T>[] array, int arrayIndex) => intervals.CopyTo(array, arrayIndex);
|
||||
|
||||
public bool Contains(Interval<T> item) => intervals.Contains(item);
|
||||
|
||||
public IEnumerator<Interval<T>> GetEnumerator() => intervals.GetEnumerator();
|
||||
|
||||
IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
|
||||
|
||||
#endregion
|
||||
|
||||
#region IReadOnlyList<Interval<T>>
|
||||
|
||||
public Interval<T> this[int index]
|
||||
{
|
||||
get => intervals[index];
|
||||
set => intervals[index] = value;
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
|
||||
public readonly struct Interval<T>
|
||||
{
|
||||
/// <summary>
|
||||
/// The start value of this interval.
|
||||
/// </summary>
|
||||
public readonly T Start;
|
||||
|
||||
/// <summary>
|
||||
/// The end value of this interval.
|
||||
/// </summary>
|
||||
public readonly T End;
|
||||
|
||||
public Interval(T start, T end)
|
||||
{
|
||||
bool startLessThanEnd = Comparer<T>.Default.Compare(start, end) < 0;
|
||||
|
||||
Start = startLessThanEnd ? start : end;
|
||||
End = startLessThanEnd ? end : start;
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user