mirror of
https://github.com/FunkinCrew/Funkin.git
synced 2025-01-22 07:30:13 -05:00
70 lines
2.3 KiB
Haxe
70 lines
2.3 KiB
Haxe
|
package funkin.util.tools;
|
||
|
|
||
|
import funkin.data.song.SongData.SongNoteData;
|
||
|
|
||
|
/**
|
||
|
* A static extension which provides utility functions for `Array<SongNoteData>`s.
|
||
|
*/
|
||
|
class SongNoteDataArrayTools
|
||
|
{
|
||
|
/**
|
||
|
* Queries whether the provided `SongNoteData` is contained in the provided array.
|
||
|
* The input array must be already sorted by `time`.
|
||
|
* Vastly more efficient than `array.indexOf`.
|
||
|
* This is not crazy or premature optimization, I'm writing this because `ChartEditorState.handleNoteDisplay` is using like 71% of its CPU time on this.
|
||
|
* @param arr The array to search.
|
||
|
* @param note The note to search for.
|
||
|
* @param predicate
|
||
|
* @return The index of the note in the array, or `-1` if it is not present.
|
||
|
*/
|
||
|
public static function fastIndexOf(input:Array<SongNoteData>, note:SongNoteData):Int
|
||
|
{
|
||
|
// I would have made this use a generic/predicate, but that would have made it slower!
|
||
|
|
||
|
// Prefix with some simple checks to save time.
|
||
|
if (input.length == 0) return -1;
|
||
|
if (note.time < input[0].time || note.time > input[input.length - 1].time) return -1;
|
||
|
|
||
|
// Thank you Github Copilot for suggesting a binary search!
|
||
|
var lowIndex:Int = 0;
|
||
|
var highIndex:Int = input.length - 1;
|
||
|
|
||
|
// When lowIndex overtakes highIndex
|
||
|
while (lowIndex <= highIndex)
|
||
|
{
|
||
|
// Get the middle index of the range.
|
||
|
var midIndex = Std.int((lowIndex + highIndex) / 2);
|
||
|
|
||
|
// Compare the middle note of the range to the note we're looking for.
|
||
|
// If it matches, return the index, else halve the range and try again.
|
||
|
var midNote = input[midIndex];
|
||
|
if (midNote.time < note.time)
|
||
|
{
|
||
|
// Search the upper half of the range.
|
||
|
lowIndex = midIndex + 1;
|
||
|
}
|
||
|
else if (midNote.time > note.time)
|
||
|
{
|
||
|
// Search the lower half of the range.
|
||
|
highIndex = midIndex - 1;
|
||
|
}
|
||
|
// Found it? Do a more thorough check.
|
||
|
else if (midNote == note)
|
||
|
{
|
||
|
return midIndex;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// We may be close, so constrain the range (but only a little) and try again.
|
||
|
highIndex -= 1;
|
||
|
}
|
||
|
}
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
public static inline function fastContains(input:Array<SongNoteData>, note:SongNoteData):Bool
|
||
|
{
|
||
|
return fastIndexOf(input, note) != -1;
|
||
|
}
|
||
|
}
|