mirror of
https://github.com/FunkinCrew/Funkin.git
synced 2024-11-27 10:05:41 -05:00
Merge pull request #17 from FunkinCrew/feature/bf-freeplay-dad
Added confirm and AFK animations to BF on freeplay menu
This commit is contained in:
commit
19c380988e
3 changed files with 197 additions and 30 deletions
|
@ -65,6 +65,8 @@ class FreeplayState extends MusicBeatSubstate
|
||||||
private var grpCapsules:FlxTypedGroup<SongMenuItem>;
|
private var grpCapsules:FlxTypedGroup<SongMenuItem>;
|
||||||
private var curPlaying:Bool = false;
|
private var curPlaying:Bool = false;
|
||||||
|
|
||||||
|
private var dj:DJBoyfriend;
|
||||||
|
|
||||||
private var iconArray:Array<HealthIcon> = [];
|
private var iconArray:Array<HealthIcon> = [];
|
||||||
|
|
||||||
override function create()
|
override function create()
|
||||||
|
@ -178,7 +180,7 @@ class FreeplayState extends MusicBeatSubstate
|
||||||
funnyScroll3.speed = -0.8;
|
funnyScroll3.speed = -0.8;
|
||||||
grpTxtScrolls.add(funnyScroll3);
|
grpTxtScrolls.add(funnyScroll3);
|
||||||
|
|
||||||
var dj:DJBoyfriend = new DJBoyfriend(0, -100);
|
dj = new DJBoyfriend(0, -100);
|
||||||
add(dj);
|
add(dj);
|
||||||
|
|
||||||
var bgDad:FlxSprite = new FlxSprite(pinkBack.width * 0.75, 0).loadGraphic(Paths.image('freeplay/freeplayBGdad'));
|
var bgDad:FlxSprite = new FlxSprite(pinkBack.width * 0.75, 0).loadGraphic(Paths.image('freeplay/freeplayBGdad'));
|
||||||
|
@ -231,7 +233,7 @@ class FreeplayState extends MusicBeatSubstate
|
||||||
fp.visible = false;
|
fp.visible = false;
|
||||||
add(fp);
|
add(fp);
|
||||||
|
|
||||||
dj.animHITsignal.add(function()
|
dj.onIntroDone.add(function()
|
||||||
{
|
{
|
||||||
FlxTween.tween(grpDifficulties, {x: 90}, 0.6, {ease: FlxEase.quartOut});
|
FlxTween.tween(grpDifficulties, {x: 90}, 0.6, {ease: FlxEase.quartOut});
|
||||||
|
|
||||||
|
@ -483,17 +485,32 @@ class FreeplayState extends MusicBeatSubstate
|
||||||
#end
|
#end
|
||||||
|
|
||||||
if (upP)
|
if (upP)
|
||||||
|
{
|
||||||
|
dj.resetAFKTimer();
|
||||||
changeSelection(-1);
|
changeSelection(-1);
|
||||||
|
}
|
||||||
if (downP)
|
if (downP)
|
||||||
|
{
|
||||||
|
dj.resetAFKTimer();
|
||||||
changeSelection(1);
|
changeSelection(1);
|
||||||
|
}
|
||||||
|
|
||||||
if (FlxG.mouse.wheel != 0)
|
if (FlxG.mouse.wheel != 0)
|
||||||
|
{
|
||||||
|
dj.resetAFKTimer();
|
||||||
changeSelection(-Math.round(FlxG.mouse.wheel / 4));
|
changeSelection(-Math.round(FlxG.mouse.wheel / 4));
|
||||||
|
}
|
||||||
|
|
||||||
if (controls.UI_LEFT_P)
|
if (controls.UI_LEFT_P)
|
||||||
|
{
|
||||||
|
dj.resetAFKTimer();
|
||||||
changeDiff(-1);
|
changeDiff(-1);
|
||||||
|
}
|
||||||
if (controls.UI_RIGHT_P)
|
if (controls.UI_RIGHT_P)
|
||||||
|
{
|
||||||
|
dj.resetAFKTimer();
|
||||||
changeDiff(1);
|
changeDiff(1);
|
||||||
|
}
|
||||||
|
|
||||||
if (controls.BACK)
|
if (controls.BACK)
|
||||||
{
|
{
|
||||||
|
@ -540,7 +557,15 @@ class FreeplayState extends MusicBeatSubstate
|
||||||
|
|
||||||
PlayState.storyWeek = songs[curSelected].week;
|
PlayState.storyWeek = songs[curSelected].week;
|
||||||
trace(' CUR WEEK ' + PlayState.storyWeek);
|
trace(' CUR WEEK ' + PlayState.storyWeek);
|
||||||
LoadingState.loadAndSwitchState(new PlayState());
|
|
||||||
|
// Visual and audio effects.
|
||||||
|
FlxG.sound.play(Paths.sound('confirmMenu'));
|
||||||
|
dj.confirm();
|
||||||
|
|
||||||
|
new FlxTimer().start(1, function(tmr:FlxTimer)
|
||||||
|
{
|
||||||
|
LoadingState.loadAndSwitchState(new PlayState(), true);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -2,46 +2,168 @@ package funkin.freeplayStuff;
|
||||||
|
|
||||||
import flixel.FlxSprite;
|
import flixel.FlxSprite;
|
||||||
import flixel.util.FlxSignal;
|
import flixel.util.FlxSignal;
|
||||||
|
import funkin.util.assets.FlxAnimationUtil;
|
||||||
|
|
||||||
class DJBoyfriend extends FlxSprite
|
class DJBoyfriend extends FlxSprite
|
||||||
{
|
{
|
||||||
public var animHITsignal:FlxSignal;
|
// Represents the sprite's current status.
|
||||||
|
// Without state machines I would have driven myself crazy years ago.
|
||||||
|
public var currentState:DJBoyfriendState = Intro;
|
||||||
|
|
||||||
|
// A callback activated when the intro animation finishes.
|
||||||
|
public var onIntroDone:FlxSignal = new FlxSignal();
|
||||||
|
|
||||||
|
// A callback activated when Boyfriend gets spooked.
|
||||||
|
public var onSpook:FlxSignal = new FlxSignal();
|
||||||
|
|
||||||
|
// playAnim stolen from Character.hx, cuz im lazy lol!
|
||||||
|
// TODO: Switch this class to use SwagSprite instead.
|
||||||
|
public var animOffsets:Map<String, Array<Dynamic>>;
|
||||||
|
|
||||||
|
static final SPOOK_PERIOD:Float = 180.0;
|
||||||
|
|
||||||
|
// Time since dad last SPOOKED you.
|
||||||
|
var timeSinceSpook:Float = 0;
|
||||||
|
|
||||||
public function new(x:Float, y:Float)
|
public function new(x:Float, y:Float)
|
||||||
{
|
{
|
||||||
super(x, y);
|
super(x, y);
|
||||||
|
|
||||||
animHITsignal = new FlxSignal();
|
|
||||||
|
|
||||||
animOffsets = new Map<String, Array<Dynamic>>();
|
animOffsets = new Map<String, Array<Dynamic>>();
|
||||||
|
|
||||||
frames = Paths.getSparrowAtlas('freeplay/bfFreeplay');
|
setupAnimations();
|
||||||
animation.addByPrefix('intro', "boyfriend dj intro", 24, false);
|
|
||||||
animation.addByPrefix('idle', "Boyfriend DJ0", 24);
|
|
||||||
animation.addByPrefix('confirm', "Boyfriend DJ confirm", 24);
|
|
||||||
|
|
||||||
addOffset('intro', 0, 0);
|
animation.finishCallback = onFinishAnim;
|
||||||
addOffset('idle', -4, -426);
|
|
||||||
|
|
||||||
playAnimation('intro');
|
|
||||||
animation.finishCallback = function(anim)
|
|
||||||
{
|
|
||||||
switch (anim)
|
|
||||||
{
|
|
||||||
case "intro":
|
|
||||||
animHITsignal.dispatch();
|
|
||||||
playAnimation('idle'); // plays idle anim after playing intro
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// playAnim stolen from Character.hx, cuz im lazy lol!
|
public override function update(elapsed:Float):Void
|
||||||
public var animOffsets:Map<String, Array<Dynamic>>;
|
{
|
||||||
|
super.update(elapsed);
|
||||||
|
|
||||||
|
if (FlxG.keys.justPressed.LEFT)
|
||||||
|
{
|
||||||
|
animOffsets["confirm"] = [animOffsets["confirm"][0] + 1, animOffsets["confirm"][1]];
|
||||||
|
applyAnimOffset();
|
||||||
|
}
|
||||||
|
else if (FlxG.keys.justPressed.RIGHT)
|
||||||
|
{
|
||||||
|
animOffsets["confirm"] = [animOffsets["confirm"][0] - 1, animOffsets["confirm"][1]];
|
||||||
|
applyAnimOffset();
|
||||||
|
}
|
||||||
|
else if (FlxG.keys.justPressed.UP)
|
||||||
|
{
|
||||||
|
animOffsets["confirm"] = [animOffsets["confirm"][0], animOffsets["confirm"][1] + 1];
|
||||||
|
applyAnimOffset();
|
||||||
|
}
|
||||||
|
else if (FlxG.keys.justPressed.DOWN)
|
||||||
|
{
|
||||||
|
animOffsets["confirm"] = [animOffsets["confirm"][0], animOffsets["confirm"][1] - 1];
|
||||||
|
applyAnimOffset();
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (currentState)
|
||||||
|
{
|
||||||
|
case Intro:
|
||||||
|
// Play the intro animation then leave this state immediately.
|
||||||
|
if (getCurrentAnimation() != 'intro')
|
||||||
|
playAnimation('intro', true);
|
||||||
|
timeSinceSpook = 0;
|
||||||
|
case Idle:
|
||||||
|
// We are in this state the majority of the time.
|
||||||
|
if (getCurrentAnimation() != 'idle' || animation.finished)
|
||||||
|
{
|
||||||
|
if (timeSinceSpook > SPOOK_PERIOD)
|
||||||
|
{
|
||||||
|
currentState = Spook;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
playAnimation('idle', false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
timeSinceSpook += elapsed;
|
||||||
|
case Confirm:
|
||||||
|
if (getCurrentAnimation() != 'confirm')
|
||||||
|
playAnimation('confirm', false);
|
||||||
|
timeSinceSpook = 0;
|
||||||
|
case Spook:
|
||||||
|
if (getCurrentAnimation() != 'spook')
|
||||||
|
{
|
||||||
|
onSpook.dispatch();
|
||||||
|
playAnimation('spook', false);
|
||||||
|
}
|
||||||
|
timeSinceSpook = 0;
|
||||||
|
default:
|
||||||
|
// I shit myself.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function onFinishAnim(name:String):Void
|
||||||
|
{
|
||||||
|
switch (name)
|
||||||
|
{
|
||||||
|
case "intro":
|
||||||
|
trace('Finished intro');
|
||||||
|
currentState = Idle;
|
||||||
|
onIntroDone.dispatch();
|
||||||
|
case "idle":
|
||||||
|
trace('Finished idle');
|
||||||
|
case "spook":
|
||||||
|
trace('Finished spook');
|
||||||
|
currentState = Idle;
|
||||||
|
case "confirm":
|
||||||
|
trace('Finished confirm');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function resetAFKTimer():Void
|
||||||
|
{
|
||||||
|
timeSinceSpook = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
function setupAnimations():Void
|
||||||
|
{
|
||||||
|
frames = FlxAnimationUtil.combineFramesCollections(Paths.getSparrowAtlas('freeplay/bfFreeplay'), Paths.getSparrowAtlas('freeplay/bf-freeplay-afk'));
|
||||||
|
|
||||||
|
animation.addByPrefix('intro', "boyfriend dj intro", 24, false);
|
||||||
|
addOffset('intro', 0, 0);
|
||||||
|
|
||||||
|
animation.addByPrefix('idle', "Boyfriend DJ0", 24, false);
|
||||||
|
addOffset('idle', -4, -426);
|
||||||
|
|
||||||
|
animation.addByPrefix('confirm', "Boyfriend DJ confirm", 24, false);
|
||||||
|
addOffset('confirm', 40, -451);
|
||||||
|
|
||||||
|
animation.addByPrefix('spook', "bf dj afk0", 24, false);
|
||||||
|
addOffset('spook', -3, -272);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function confirm():Void
|
||||||
|
{
|
||||||
|
currentState = Confirm;
|
||||||
|
}
|
||||||
|
|
||||||
|
public inline function addOffset(name:String, x:Float = 0, y:Float = 0)
|
||||||
|
{
|
||||||
|
animOffsets[name] = [x, y];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getCurrentAnimation():String
|
||||||
|
{
|
||||||
|
if (this.animation == null || this.animation.curAnim == null)
|
||||||
|
return "";
|
||||||
|
return this.animation.curAnim.name;
|
||||||
|
}
|
||||||
|
|
||||||
public function playAnimation(AnimName:String, Force:Bool = false, Reversed:Bool = false, Frame:Int = 0):Void
|
public function playAnimation(AnimName:String, Force:Bool = false, Reversed:Bool = false, Frame:Int = 0):Void
|
||||||
{
|
{
|
||||||
animation.play(AnimName, Force, Reversed, Frame);
|
animation.play(AnimName, Force, Reversed, Frame);
|
||||||
|
applyAnimOffset();
|
||||||
|
}
|
||||||
|
|
||||||
|
function applyAnimOffset()
|
||||||
|
{
|
||||||
|
var AnimName = getCurrentAnimation();
|
||||||
var daOffset = animOffsets.get(AnimName);
|
var daOffset = animOffsets.get(AnimName);
|
||||||
if (animOffsets.exists(AnimName))
|
if (animOffsets.exists(AnimName))
|
||||||
{
|
{
|
||||||
|
@ -50,9 +172,12 @@ class DJBoyfriend extends FlxSprite
|
||||||
else
|
else
|
||||||
offset.set(0, 0);
|
offset.set(0, 0);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
public function addOffset(name:String, x:Float = 0, y:Float = 0)
|
|
||||||
{
|
enum DJBoyfriendState
|
||||||
animOffsets[name] = [x, y];
|
{
|
||||||
}
|
Intro;
|
||||||
|
Idle;
|
||||||
|
Confirm;
|
||||||
|
Spook;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
package funkin.util.assets;
|
package funkin.util.assets;
|
||||||
|
|
||||||
import flixel.FlxSprite;
|
import flixel.FlxSprite;
|
||||||
|
import flixel.graphics.frames.FlxFramesCollection;
|
||||||
import funkin.play.AnimationData;
|
import funkin.play.AnimationData;
|
||||||
|
|
||||||
class FlxAnimationUtil
|
class FlxAnimationUtil
|
||||||
|
@ -39,4 +40,20 @@ class FlxAnimationUtil
|
||||||
addAtlasAnimation(target, anim);
|
addAtlasAnimation(target, anim);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static function combineFramesCollections(a:FlxFramesCollection, b:FlxFramesCollection):FlxFramesCollection
|
||||||
|
{
|
||||||
|
var result:FlxFramesCollection = new FlxFramesCollection(null, ATLAS, null);
|
||||||
|
|
||||||
|
for (frame in a.frames)
|
||||||
|
{
|
||||||
|
result.pushFrame(frame);
|
||||||
|
}
|
||||||
|
for (frame in b.frames)
|
||||||
|
{
|
||||||
|
result.pushFrame(frame);
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue