2022-03-21 04:19:05 +00:00
|
|
|
package funkin.play.character;
|
|
|
|
|
2024-01-14 14:18:32 +00:00
|
|
|
import flixel.graphics.frames.FlxAtlasFrames;
|
2022-05-29 23:13:21 +00:00
|
|
|
import flixel.graphics.frames.FlxFramesCollection;
|
2022-03-21 04:19:05 +00:00
|
|
|
import funkin.modding.events.ScriptEvent;
|
|
|
|
import funkin.util.assets.FlxAnimationUtil;
|
2023-02-22 01:58:15 +00:00
|
|
|
import funkin.play.character.CharacterData.CharacterRenderType;
|
2022-03-21 04:19:05 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* For some characters which use Sparrow atlases, the spritesheets need to be split
|
2024-01-14 14:18:32 +00:00
|
|
|
* into multiple files. This character renderer concatenates these together into a single sprite.
|
2023-06-08 20:30:45 +00:00
|
|
|
*
|
2022-03-21 04:19:05 +00:00
|
|
|
* Examples in base game include BF Holding GF (most of the sprites are in one file
|
|
|
|
* but the death animation is in a separate file).
|
|
|
|
* Only example I can think of in mods is Tricky (which has a separate file for each animation).
|
2023-06-08 20:30:45 +00:00
|
|
|
*
|
2024-01-14 14:18:32 +00:00
|
|
|
* BaseCharacter has game logic, MultiSparrowCharacter has only rendering logic.
|
2022-03-21 04:19:05 +00:00
|
|
|
* KEEP THEM SEPARATE!
|
|
|
|
*/
|
|
|
|
class MultiSparrowCharacter extends BaseCharacter
|
|
|
|
{
|
2023-01-23 00:55:30 +00:00
|
|
|
public function new(id:String)
|
|
|
|
{
|
2023-02-22 01:58:15 +00:00
|
|
|
super(id, CharacterRenderType.MultiSparrow);
|
2023-01-23 00:55:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
override function onCreate(event:ScriptEvent):Void
|
|
|
|
{
|
|
|
|
trace('Creating Multi-Sparrow character: ' + this.characterId);
|
|
|
|
|
|
|
|
buildSprites();
|
|
|
|
super.onCreate(event);
|
|
|
|
}
|
|
|
|
|
2023-02-22 01:58:15 +00:00
|
|
|
function buildSprites():Void
|
2023-01-23 00:55:30 +00:00
|
|
|
{
|
2024-01-14 14:18:32 +00:00
|
|
|
buildSpritesheet();
|
2023-01-23 00:55:30 +00:00
|
|
|
buildAnimations();
|
|
|
|
|
|
|
|
if (_data.isPixel)
|
|
|
|
{
|
2023-06-15 04:29:18 +00:00
|
|
|
this.isPixel = true;
|
2023-01-23 00:55:30 +00:00
|
|
|
this.antialiasing = false;
|
2024-06-20 03:47:11 +00:00
|
|
|
pixelPerfectRender = true;
|
|
|
|
pixelPerfectPosition = true;
|
2023-01-23 00:55:30 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-06-15 04:29:18 +00:00
|
|
|
this.isPixel = false;
|
2023-01-23 00:55:30 +00:00
|
|
|
this.antialiasing = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-01-14 14:18:32 +00:00
|
|
|
function buildSpritesheet():Void
|
2023-01-23 00:55:30 +00:00
|
|
|
{
|
2024-01-14 14:18:32 +00:00
|
|
|
var assetList = [];
|
2023-01-23 00:55:30 +00:00
|
|
|
for (anim in _data.animations)
|
|
|
|
{
|
|
|
|
if (anim.assetPath != null && !assetList.contains(anim.assetPath))
|
|
|
|
{
|
|
|
|
assetList.push(anim.assetPath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-07-16 09:52:51 +00:00
|
|
|
var texture:FlxAtlasFrames = Paths.getSparrowAtlas(_data.assetPath);
|
2024-01-14 14:18:32 +00:00
|
|
|
|
|
|
|
if (texture == null)
|
|
|
|
{
|
|
|
|
trace('Multi-Sparrow atlas could not load PRIMARY texture: ${_data.assetPath}');
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
trace('Creating multi-sparrow atlas: ${_data.assetPath}');
|
|
|
|
texture.parent.destroyOnNoUse = false;
|
|
|
|
}
|
|
|
|
|
2023-01-23 00:55:30 +00:00
|
|
|
for (asset in assetList)
|
|
|
|
{
|
2024-07-16 09:52:51 +00:00
|
|
|
var subTexture:FlxAtlasFrames = Paths.getSparrowAtlas(asset);
|
2023-01-23 00:55:30 +00:00
|
|
|
// If we don't do this, the unused textures will be removed as soon as they're loaded.
|
|
|
|
|
2024-01-14 14:18:32 +00:00
|
|
|
if (subTexture == null)
|
2023-01-23 00:55:30 +00:00
|
|
|
{
|
2024-01-14 14:18:32 +00:00
|
|
|
trace('Multi-Sparrow atlas could not load subtexture: ${asset}');
|
2023-01-23 00:55:30 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2024-01-14 14:18:32 +00:00
|
|
|
trace('Concatenating multi-sparrow atlas: ${asset}');
|
|
|
|
subTexture.parent.destroyOnNoUse = false;
|
2023-01-23 00:55:30 +00:00
|
|
|
}
|
|
|
|
|
2024-01-14 14:18:32 +00:00
|
|
|
texture.addAtlas(subTexture);
|
2023-01-23 00:55:30 +00:00
|
|
|
}
|
|
|
|
|
2024-01-14 14:18:32 +00:00
|
|
|
this.frames = texture;
|
|
|
|
this.setScale(_data.scale);
|
2023-01-23 00:55:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function buildAnimations()
|
|
|
|
{
|
|
|
|
trace('[MULTISPARROWCHAR] Loading ${_data.animations.length} animations for ${characterId}');
|
|
|
|
|
|
|
|
// We need to swap to the proper frame collection before adding the animations, I think?
|
|
|
|
for (anim in _data.animations)
|
|
|
|
{
|
|
|
|
FlxAnimationUtil.addAtlasAnimation(this, anim);
|
|
|
|
|
|
|
|
if (anim.offsets == null)
|
|
|
|
{
|
|
|
|
setAnimationOffsets(anim.name, 0, 0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
setAnimationOffsets(anim.name, anim.offsets[0], anim.offsets[1]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var animNames = this.animation.getNameList();
|
|
|
|
trace('[MULTISPARROWCHAR] Successfully loaded ${animNames.length} animations for ${characterId}');
|
|
|
|
}
|
|
|
|
|
2023-08-28 19:03:29 +00:00
|
|
|
public override function playAnimation(name:String, restart:Bool = false, ignoreOther:Bool = false, reverse:Bool = false):Void
|
2023-01-23 00:55:30 +00:00
|
|
|
{
|
|
|
|
// Make sure we ignore other animations if we're currently playing a forced one,
|
|
|
|
// unless we're forcing a new animation.
|
2023-01-23 03:25:45 +00:00
|
|
|
if (!this.canPlayOtherAnims && !ignoreOther) return;
|
2023-01-23 00:55:30 +00:00
|
|
|
|
2023-05-26 21:10:08 +00:00
|
|
|
super.playAnimation(name, restart, ignoreOther, reverse);
|
2023-01-23 00:55:30 +00:00
|
|
|
}
|
2022-03-21 04:19:05 +00:00
|
|
|
}
|