2023-06-22 05:41:01 +00:00
|
|
|
package funkin.play.notes;
|
|
|
|
|
2023-07-14 00:27:45 +00:00
|
|
|
import funkin.play.notes.notestyle.NoteStyle;
|
2023-06-22 05:41:01 +00:00
|
|
|
import flixel.graphics.frames.FlxAtlasFrames;
|
2024-10-05 17:33:08 +00:00
|
|
|
import funkin.graphics.FunkinSprite;
|
2023-06-22 05:41:01 +00:00
|
|
|
import funkin.play.notes.NoteSprite;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The actual receptor that you see on screen.
|
|
|
|
*/
|
2024-10-05 17:33:08 +00:00
|
|
|
class StrumlineNote extends FunkinSprite
|
2023-06-22 05:41:01 +00:00
|
|
|
{
|
2024-10-05 17:33:08 +00:00
|
|
|
/**
|
|
|
|
* Whether this strumline note is on the player's side or the opponent's side.
|
|
|
|
*/
|
2023-06-22 05:41:01 +00:00
|
|
|
public var isPlayer(default, null):Bool;
|
|
|
|
|
2024-10-05 17:33:08 +00:00
|
|
|
/**
|
|
|
|
* The direction which this strumline note is facing.
|
|
|
|
*/
|
2023-06-22 05:41:01 +00:00
|
|
|
public var direction(default, set):NoteDirection;
|
|
|
|
|
|
|
|
function set_direction(value:NoteDirection):NoteDirection
|
|
|
|
{
|
|
|
|
this.direction = value;
|
|
|
|
return this.direction;
|
|
|
|
}
|
|
|
|
|
2024-10-05 17:32:58 +00:00
|
|
|
/**
|
|
|
|
* Set this flag to `true` to disable performance optimizations that cause
|
|
|
|
* the Strumline note sprite to ignore `velocity` and `acceleration`.
|
|
|
|
*/
|
|
|
|
public var forceActive:Bool = false;
|
|
|
|
|
2024-10-05 17:33:08 +00:00
|
|
|
/**
|
|
|
|
* How long to continue the hold note animation after a note is pressed.
|
|
|
|
*/
|
|
|
|
static final CONFIRM_HOLD_TIME:Float = 0.1;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* How long the hold note animation has been playing after a note is pressed.
|
|
|
|
*/
|
|
|
|
var confirmHoldTimer:Float = -1;
|
|
|
|
|
2023-07-14 00:27:45 +00:00
|
|
|
public function new(noteStyle:NoteStyle, isPlayer:Bool, direction:NoteDirection)
|
2023-06-22 05:41:01 +00:00
|
|
|
{
|
|
|
|
super(0, 0);
|
|
|
|
|
|
|
|
this.isPlayer = isPlayer;
|
|
|
|
|
|
|
|
this.direction = direction;
|
|
|
|
|
2023-07-14 00:27:45 +00:00
|
|
|
setup(noteStyle);
|
|
|
|
|
2023-06-22 05:41:01 +00:00
|
|
|
this.animation.callback = onAnimationFrame;
|
|
|
|
this.animation.finishCallback = onAnimationFinished;
|
|
|
|
|
2023-06-27 21:22:51 +00:00
|
|
|
// Must be true for animations to play.
|
2023-06-22 05:41:01 +00:00
|
|
|
this.active = true;
|
|
|
|
}
|
|
|
|
|
2024-10-05 17:33:08 +00:00
|
|
|
function onAnimationFrame(name:String, frameNumber:Int, frameIndex:Int):Void
|
|
|
|
{
|
|
|
|
// Do nothing.
|
|
|
|
}
|
2023-06-22 05:41:01 +00:00
|
|
|
|
|
|
|
function onAnimationFinished(name:String):Void
|
|
|
|
{
|
2023-07-08 05:03:46 +00:00
|
|
|
// Run a timer before we stop playing the confirm animation.
|
|
|
|
// On opponent, this prevent issues with hold notes.
|
|
|
|
// On player, this allows holding the confirm key to fall back to press.
|
|
|
|
if (name == 'confirm')
|
2023-06-22 05:41:01 +00:00
|
|
|
{
|
2023-07-08 05:03:46 +00:00
|
|
|
confirmHoldTimer = 0;
|
2023-06-22 05:41:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
override function update(elapsed:Float)
|
|
|
|
{
|
|
|
|
super.update(elapsed);
|
|
|
|
|
|
|
|
centerOrigin();
|
2023-07-08 05:03:46 +00:00
|
|
|
|
|
|
|
if (confirmHoldTimer >= 0)
|
|
|
|
{
|
|
|
|
confirmHoldTimer += elapsed;
|
|
|
|
|
|
|
|
// Ensure the opponent stops holding the key after a certain amount of time.
|
|
|
|
if (confirmHoldTimer >= CONFIRM_HOLD_TIME)
|
|
|
|
{
|
|
|
|
confirmHoldTimer = -1;
|
|
|
|
playStatic();
|
|
|
|
}
|
|
|
|
}
|
2023-06-22 05:41:01 +00:00
|
|
|
}
|
|
|
|
|
2023-07-14 00:27:45 +00:00
|
|
|
function setup(noteStyle:NoteStyle):Void
|
2023-06-22 05:41:01 +00:00
|
|
|
{
|
2024-07-28 21:10:32 +00:00
|
|
|
if (noteStyle == null)
|
|
|
|
{
|
|
|
|
// If you get an exception on this line, check the debug console.
|
|
|
|
// You probably have a parsing error in your note style's JSON file.
|
|
|
|
throw "FATAL ERROR: Attempted to initialize PlayState with an invalid NoteStyle.";
|
|
|
|
}
|
|
|
|
|
2023-07-14 00:27:45 +00:00
|
|
|
noteStyle.applyStrumlineFrames(this);
|
|
|
|
noteStyle.applyStrumlineAnimations(this, this.direction);
|
2023-06-22 05:41:01 +00:00
|
|
|
|
2024-09-17 20:47:16 +00:00
|
|
|
var scale = noteStyle.getStrumlineScale();
|
|
|
|
this.scale.set(scale, scale);
|
2023-06-22 05:41:01 +00:00
|
|
|
this.updateHitbox();
|
2023-07-14 00:27:45 +00:00
|
|
|
noteStyle.applyStrumlineOffsets(this);
|
|
|
|
|
2023-06-22 05:41:01 +00:00
|
|
|
this.playStatic();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function playAnimation(name:String = 'static', force:Bool = false, reversed:Bool = false, startFrame:Int = 0):Void
|
|
|
|
{
|
|
|
|
this.animation.play(name, force, reversed, startFrame);
|
|
|
|
|
|
|
|
centerOffsets();
|
|
|
|
centerOrigin();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function playStatic():Void
|
|
|
|
{
|
2024-10-05 17:32:58 +00:00
|
|
|
this.active = (forceActive || isAnimationDynamic('static'));
|
2023-06-22 05:41:01 +00:00
|
|
|
this.playAnimation('static', true);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function playPress():Void
|
|
|
|
{
|
2024-10-05 17:32:58 +00:00
|
|
|
this.active = (forceActive || isAnimationDynamic('press'));
|
2023-06-22 05:41:01 +00:00
|
|
|
this.playAnimation('press', true);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function playConfirm():Void
|
|
|
|
{
|
2024-10-05 17:32:58 +00:00
|
|
|
this.active = (forceActive || isAnimationDynamic('confirm'));
|
2023-06-22 05:41:01 +00:00
|
|
|
this.playAnimation('confirm', true);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function isConfirm():Bool
|
|
|
|
{
|
|
|
|
return getCurrentAnimation().startsWith('confirm');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function holdConfirm():Void
|
|
|
|
{
|
|
|
|
this.active = true;
|
|
|
|
|
2023-07-08 05:03:46 +00:00
|
|
|
if (getCurrentAnimation() == "confirm-hold")
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else if (getCurrentAnimation() == "confirm")
|
2023-06-22 05:41:01 +00:00
|
|
|
{
|
|
|
|
if (isAnimationFinished())
|
|
|
|
{
|
2023-07-08 05:03:46 +00:00
|
|
|
this.confirmHoldTimer = -1;
|
|
|
|
this.playAnimation('confirm-hold', false, false);
|
2023-06-22 05:41:01 +00:00
|
|
|
}
|
|
|
|
}
|
2023-07-08 05:03:46 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
this.playAnimation('confirm', false, false);
|
|
|
|
}
|
2023-06-22 05:41:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the name of the animation that is currently playing.
|
|
|
|
* If no animation is playing (usually this means the sprite is BROKEN!),
|
|
|
|
* returns an empty string to prevent NPEs.
|
|
|
|
*/
|
|
|
|
public function getCurrentAnimation():String
|
|
|
|
{
|
|
|
|
if (this.animation == null || this.animation.curAnim == null) return "";
|
|
|
|
return this.animation.curAnim.name;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function isAnimationFinished():Bool
|
|
|
|
{
|
|
|
|
return this.animation.finished;
|
|
|
|
}
|
|
|
|
|
|
|
|
static final DEFAULT_OFFSET:Int = 13;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adjusts the position of the sprite's graphic relative to the hitbox.
|
|
|
|
*/
|
|
|
|
function fixOffsets():Void
|
|
|
|
{
|
|
|
|
// Automatically center the bounding box within the graphic.
|
|
|
|
this.centerOffsets();
|
|
|
|
|
|
|
|
if (getCurrentAnimation() == "confirm")
|
|
|
|
{
|
|
|
|
// Move the graphic down and to the right to compensate for
|
|
|
|
// the "glow" effect on the strumline note.
|
|
|
|
this.offset.x -= DEFAULT_OFFSET;
|
|
|
|
this.offset.y -= DEFAULT_OFFSET;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this.centerOrigin();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|