2023-10-26 09:46:22 +00:00
|
|
|
package funkin.ui.debug.charting.components;
|
2023-07-23 00:16:43 +00:00
|
|
|
|
|
|
|
import funkin.play.notes.Strumline;
|
|
|
|
import funkin.data.notestyle.NoteStyleRegistry;
|
2024-06-02 20:11:17 +00:00
|
|
|
import funkin.play.notes.notestyle.NoteStyle;
|
2023-07-23 00:16:43 +00:00
|
|
|
import flixel.FlxObject;
|
|
|
|
import flixel.FlxSprite;
|
|
|
|
import flixel.graphics.frames.FlxFramesCollection;
|
|
|
|
import flixel.graphics.frames.FlxTileFrames;
|
|
|
|
import flixel.math.FlxPoint;
|
|
|
|
import funkin.play.notes.SustainTrail;
|
2023-09-08 21:46:44 +00:00
|
|
|
import funkin.data.song.SongData.SongNoteData;
|
2023-11-21 20:10:18 +00:00
|
|
|
import flixel.math.FlxMath;
|
2023-07-23 00:16:43 +00:00
|
|
|
|
|
|
|
/**
|
2023-10-26 09:46:22 +00:00
|
|
|
* A sprite that can be used to display the trail of a hold note in a chart.
|
2023-07-23 00:16:43 +00:00
|
|
|
* Designed to be used and reused efficiently. Has no gameplay functionality.
|
|
|
|
*/
|
2024-06-02 20:11:17 +00:00
|
|
|
@:access(funkin.ui.debug.charting.ChartEditorState)
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2023-07-23 00:16:43 +00:00
|
|
|
class ChartEditorHoldNoteSprite extends SustainTrail
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* The ChartEditorState this note belongs to.
|
|
|
|
*/
|
|
|
|
public var parentState:ChartEditorState;
|
|
|
|
|
2024-06-02 20:11:17 +00:00
|
|
|
@:isVar
|
|
|
|
public var noteStyle(get, set):Null<String>;
|
|
|
|
|
|
|
|
function get_noteStyle():Null<String>
|
|
|
|
{
|
|
|
|
return this.noteStyle ?? this.parentState.currentSongNoteStyle;
|
|
|
|
}
|
|
|
|
|
|
|
|
@:nullSafety(Off)
|
|
|
|
function set_noteStyle(value:Null<String>):Null<String>
|
|
|
|
{
|
|
|
|
this.noteStyle = value;
|
|
|
|
this.updateHoldNoteGraphic();
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2023-07-23 00:16:43 +00:00
|
|
|
public function new(parent:ChartEditorState)
|
|
|
|
{
|
|
|
|
var noteStyle = NoteStyleRegistry.instance.fetchDefault();
|
|
|
|
|
|
|
|
super(0, 100, noteStyle);
|
|
|
|
|
|
|
|
this.parentState = parent;
|
|
|
|
}
|
|
|
|
|
2024-06-02 20:11:17 +00:00
|
|
|
@:nullSafety(Off)
|
|
|
|
function updateHoldNoteGraphic():Void
|
|
|
|
{
|
|
|
|
var bruhStyle:NoteStyle = NoteStyleRegistry.instance.fetchEntry(noteStyle);
|
2024-06-02 21:31:49 +00:00
|
|
|
setupHoldNoteGraphic(bruhStyle);
|
2024-06-05 17:47:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
override function setupHoldNoteGraphic(noteStyle:NoteStyle):Void
|
|
|
|
{
|
|
|
|
loadGraphic(noteStyle.getHoldNoteAssetPath());
|
|
|
|
|
|
|
|
antialiasing = true;
|
|
|
|
|
|
|
|
this.isPixel = noteStyle.isHoldNotePixel();
|
|
|
|
if (isPixel)
|
|
|
|
{
|
|
|
|
endOffset = bottomClip = 1;
|
|
|
|
antialiasing = false;
|
|
|
|
}
|
2024-06-11 18:52:08 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
endOffset = 0.5;
|
|
|
|
bottomClip = 0.9;
|
|
|
|
}
|
2024-06-02 20:11:17 +00:00
|
|
|
|
|
|
|
zoom = 1.0;
|
2024-06-05 17:47:34 +00:00
|
|
|
zoom *= noteStyle.fetchHoldNoteScale();
|
2024-06-02 20:11:17 +00:00
|
|
|
zoom *= 0.7;
|
|
|
|
zoom *= ChartEditorState.GRID_SIZE / Strumline.STRUMLINE_SIZE;
|
|
|
|
|
2024-06-05 17:47:34 +00:00
|
|
|
graphicWidth = graphic.width / 8 * zoom; // amount of notes * 2
|
|
|
|
graphicHeight = sustainLength * 0.45; // sustainHeight
|
|
|
|
|
2024-06-02 20:11:17 +00:00
|
|
|
flipY = false;
|
|
|
|
|
2024-06-05 17:47:34 +00:00
|
|
|
alpha = 1.0;
|
|
|
|
|
|
|
|
updateColorTransform();
|
2024-06-02 22:03:47 +00:00
|
|
|
|
2024-06-05 17:47:34 +00:00
|
|
|
updateClipping();
|
|
|
|
|
|
|
|
setup();
|
2024-06-02 20:11:17 +00:00
|
|
|
}
|
|
|
|
|
2024-01-12 11:13:34 +00:00
|
|
|
public override function updateHitbox():Void
|
|
|
|
{
|
|
|
|
// Expand the clickable hitbox to the full column width, then nudge to the left to re-center it.
|
|
|
|
width = ChartEditorState.GRID_SIZE;
|
|
|
|
height = graphicHeight;
|
|
|
|
|
|
|
|
var xOffset = (ChartEditorState.GRID_SIZE - graphicWidth) / 2;
|
|
|
|
offset.set(-xOffset, 0);
|
|
|
|
origin.set(width * 0.5, height * 0.5);
|
|
|
|
}
|
|
|
|
|
2023-07-23 00:16:43 +00:00
|
|
|
/**
|
|
|
|
* Set the height directly, to a value in pixels.
|
|
|
|
* @param h The desired height in pixels.
|
|
|
|
*/
|
2024-01-06 06:06:10 +00:00
|
|
|
public function setHeightDirectly(h:Float, lerp:Bool = false)
|
2023-07-23 00:16:43 +00:00
|
|
|
{
|
2024-01-06 06:06:10 +00:00
|
|
|
if (lerp)
|
|
|
|
{
|
2024-05-09 16:51:03 +00:00
|
|
|
sustainLength = FlxMath.lerp(sustainLength, h / (getBaseScrollSpeed() * Constants.PIXELS_PER_MS), 0.25);
|
2024-01-06 06:06:10 +00:00
|
|
|
}
|
2023-11-21 20:10:18 +00:00
|
|
|
else
|
2024-01-06 06:06:10 +00:00
|
|
|
{
|
2024-05-09 16:51:03 +00:00
|
|
|
sustainLength = h / (getBaseScrollSpeed() * Constants.PIXELS_PER_MS);
|
2024-01-06 06:06:10 +00:00
|
|
|
}
|
2023-11-21 20:10:18 +00:00
|
|
|
|
2023-07-23 00:16:43 +00:00
|
|
|
fullSustainLength = sustainLength;
|
|
|
|
}
|
|
|
|
|
2024-01-13 00:50:13 +00:00
|
|
|
#if FLX_DEBUG
|
2024-01-12 11:13:34 +00:00
|
|
|
/**
|
|
|
|
* Call this to override how debug bounding boxes are drawn for this sprite.
|
|
|
|
*/
|
|
|
|
public override function drawDebugOnCamera(camera:flixel.FlxCamera):Void
|
|
|
|
{
|
|
|
|
if (!camera.visible || !camera.exists || !isOnScreen(camera)) return;
|
|
|
|
|
|
|
|
var rect = getBoundingBox(camera);
|
|
|
|
trace('hold note bounding box: ' + rect.x + ', ' + rect.y + ', ' + rect.width + ', ' + rect.height);
|
|
|
|
|
|
|
|
var gfx = beginDrawDebug(camera);
|
|
|
|
debugBoundingBoxColor = 0xffFF66FF;
|
|
|
|
gfx.lineStyle(2, color, 0.5); // thickness, color, alpha
|
|
|
|
gfx.drawRect(rect.x, rect.y, rect.width, rect.height);
|
|
|
|
endDrawDebug(camera);
|
|
|
|
}
|
2024-01-13 00:50:13 +00:00
|
|
|
#end
|
2024-01-12 11:13:34 +00:00
|
|
|
|
2023-07-23 00:16:43 +00:00
|
|
|
function setup():Void
|
|
|
|
{
|
|
|
|
strumTime = 999999999;
|
|
|
|
missedNote = false;
|
|
|
|
hitNote = false;
|
2023-07-26 03:11:12 +00:00
|
|
|
active = true;
|
2023-07-23 00:16:43 +00:00
|
|
|
visible = true;
|
|
|
|
alpha = 1.0;
|
2024-01-12 11:13:34 +00:00
|
|
|
graphicWidth = graphic.width / 8 * zoom; // amount of notes * 2
|
|
|
|
|
|
|
|
updateHitbox();
|
2023-07-23 00:16:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public override function revive():Void
|
|
|
|
{
|
|
|
|
super.revive();
|
|
|
|
|
|
|
|
setup();
|
|
|
|
}
|
|
|
|
|
2023-07-26 03:11:12 +00:00
|
|
|
public override function kill():Void
|
|
|
|
{
|
|
|
|
super.kill();
|
|
|
|
|
|
|
|
active = false;
|
|
|
|
visible = false;
|
|
|
|
noteData = null;
|
|
|
|
strumTime = 999999999;
|
|
|
|
noteDirection = 0;
|
|
|
|
sustainLength = 0;
|
|
|
|
fullSustainLength = 0;
|
|
|
|
}
|
|
|
|
|
2023-07-23 00:16:43 +00:00
|
|
|
/**
|
|
|
|
* Return whether this note is currently visible.
|
|
|
|
*/
|
|
|
|
public function isHoldNoteVisible(viewAreaBottom:Float, viewAreaTop:Float):Bool
|
|
|
|
{
|
|
|
|
// True if the note is above the view area.
|
|
|
|
var aboveViewArea = (this.y + this.height < viewAreaTop);
|
|
|
|
|
|
|
|
// True if the note is below the view area.
|
|
|
|
var belowViewArea = (this.y > viewAreaBottom);
|
|
|
|
|
|
|
|
return !aboveViewArea && !belowViewArea;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return whether a hold note, if placed in the scene, would be visible.
|
|
|
|
*/
|
|
|
|
public static function wouldHoldNoteBeVisible(viewAreaBottom:Float, viewAreaTop:Float, noteData:SongNoteData, ?origin:FlxObject):Bool
|
|
|
|
{
|
2023-09-13 18:51:12 +00:00
|
|
|
var noteHeight:Float = noteData.getStepLength() * ChartEditorState.GRID_SIZE;
|
|
|
|
var stepTime:Float = inline noteData.getStepTime();
|
|
|
|
var notePosY:Float = stepTime * ChartEditorState.GRID_SIZE;
|
2023-07-23 00:16:43 +00:00
|
|
|
if (origin != null) notePosY += origin.y;
|
|
|
|
|
|
|
|
// True if the note is above the view area.
|
|
|
|
var aboveViewArea = (notePosY + noteHeight < viewAreaTop);
|
|
|
|
|
|
|
|
// True if the note is below the view area.
|
|
|
|
var belowViewArea = (notePosY > viewAreaBottom);
|
|
|
|
|
|
|
|
return !aboveViewArea && !belowViewArea;
|
|
|
|
}
|
|
|
|
|
2023-08-31 22:47:23 +00:00
|
|
|
public function updateHoldNotePosition(?origin:FlxObject):Void
|
2023-07-23 00:16:43 +00:00
|
|
|
{
|
2023-08-31 22:47:23 +00:00
|
|
|
if (this.noteData == null) return;
|
|
|
|
|
2023-07-23 00:16:43 +00:00
|
|
|
var cursorColumn:Int = this.noteData.data;
|
|
|
|
|
|
|
|
if (cursorColumn < 0) cursorColumn = 0;
|
|
|
|
if (cursorColumn >= (ChartEditorState.STRUMLINE_SIZE * 2 + 1))
|
|
|
|
{
|
|
|
|
cursorColumn = (ChartEditorState.STRUMLINE_SIZE * 2 + 1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Invert player and opponent columns.
|
|
|
|
if (cursorColumn >= ChartEditorState.STRUMLINE_SIZE)
|
|
|
|
{
|
|
|
|
cursorColumn -= ChartEditorState.STRUMLINE_SIZE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cursorColumn += ChartEditorState.STRUMLINE_SIZE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.x = cursorColumn * ChartEditorState.GRID_SIZE;
|
|
|
|
|
|
|
|
// Notes far in the song will start far down, but the group they belong to will have a high negative offset.
|
2023-09-13 18:51:12 +00:00
|
|
|
// noteData.getStepTime() returns a calculated value which accounts for BPM changes
|
|
|
|
var stepTime:Float =
|
|
|
|
inline this.noteData.getStepTime();
|
|
|
|
if (stepTime >= 0)
|
2023-07-23 00:16:43 +00:00
|
|
|
{
|
2023-09-11 23:42:10 +00:00
|
|
|
// Add epsilon to fix rounding issues?
|
|
|
|
// var roundedStepTime:Float = Math.floor((stepTime + 0.01) / noteSnapRatio) * noteSnapRatio;
|
|
|
|
this.y = stepTime * ChartEditorState.GRID_SIZE;
|
2023-07-23 00:16:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
this.x += ChartEditorState.GRID_SIZE / 2;
|
2024-01-12 11:13:34 +00:00
|
|
|
this.x -= this.graphicWidth / 2;
|
2023-07-23 00:16:43 +00:00
|
|
|
|
|
|
|
this.y += ChartEditorState.GRID_SIZE / 2;
|
|
|
|
|
|
|
|
if (origin != null)
|
|
|
|
{
|
|
|
|
this.x += origin.x;
|
|
|
|
this.y += origin.y;
|
|
|
|
}
|
2024-01-12 11:13:34 +00:00
|
|
|
|
|
|
|
// Account for expanded clickable hitbox.
|
|
|
|
this.x += this.offset.x;
|
2023-07-23 00:16:43 +00:00
|
|
|
}
|
|
|
|
}
|