2024-06-18 21:56:24 +00:00
|
|
|
package funkin.ui.freeplay.charselect;
|
|
|
|
|
|
|
|
import funkin.data.IRegistryEntry;
|
|
|
|
import funkin.data.freeplay.player.PlayerData;
|
|
|
|
import funkin.data.freeplay.player.PlayerRegistry;
|
2024-07-04 02:50:39 +00:00
|
|
|
import funkin.play.scoring.Scoring.ScoringRank;
|
2024-06-18 21:56:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* An object used to retrieve data about a playable character (also known as "weeks").
|
|
|
|
* Can be scripted to override each function, for custom behavior.
|
|
|
|
*/
|
2024-09-01 07:22:34 +00:00
|
|
|
@:nullSafety
|
2024-06-18 21:56:24 +00:00
|
|
|
class PlayableCharacter implements IRegistryEntry<PlayerData>
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* The ID of the playable character.
|
|
|
|
*/
|
|
|
|
public final id:String;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Playable character data as parsed from the JSON file.
|
|
|
|
*/
|
2024-09-01 07:22:34 +00:00
|
|
|
public final _data:Null<PlayerData>;
|
2024-06-18 21:56:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param id The ID of the JSON file to parse.
|
|
|
|
*/
|
|
|
|
public function new(id:String)
|
|
|
|
{
|
|
|
|
this.id = id;
|
|
|
|
_data = _fetchData(id);
|
|
|
|
|
|
|
|
if (_data == null)
|
|
|
|
{
|
|
|
|
throw 'Could not parse playable character data for id: $id';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the readable name of the playable character.
|
|
|
|
*/
|
|
|
|
public function getName():String
|
|
|
|
{
|
|
|
|
// TODO: Maybe add localization support?
|
2024-09-01 07:22:34 +00:00
|
|
|
return _data?.name ?? "Unknown";
|
2024-06-18 21:56:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the list of stage character IDs associated with this playable character.
|
|
|
|
* @return The list of associated character IDs
|
|
|
|
*/
|
|
|
|
public function getOwnedCharacterIds():Array<String>
|
|
|
|
{
|
2024-09-01 07:22:34 +00:00
|
|
|
return _data?.ownedChars ?? [];
|
2024-06-18 21:56:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return `true` if, when this character is selected in Freeplay,
|
|
|
|
* songs unassociated with a specific character should appear.
|
|
|
|
*/
|
|
|
|
public function shouldShowUnownedChars():Bool
|
|
|
|
{
|
2024-09-01 07:22:34 +00:00
|
|
|
return _data?.showUnownedChars ?? false;
|
2024-06-18 21:56:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public function shouldShowCharacter(id:String):Bool
|
|
|
|
{
|
2024-09-01 07:22:34 +00:00
|
|
|
if (getOwnedCharacterIds().contains(id))
|
2024-06-18 21:56:24 +00:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2024-09-01 07:22:34 +00:00
|
|
|
if (shouldShowUnownedChars())
|
2024-06-18 21:56:24 +00:00
|
|
|
{
|
|
|
|
var result = !PlayerRegistry.instance.isCharacterOwned(id);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-09-01 07:22:34 +00:00
|
|
|
public function getFreeplayStyleID():String
|
2024-06-19 00:07:27 +00:00
|
|
|
{
|
2024-09-01 07:22:34 +00:00
|
|
|
return _data?.freeplayStyle ?? Constants.DEFAULT_FREEPLAY_STYLE;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getFreeplayDJData():Null<PlayerFreeplayDJData>
|
|
|
|
{
|
|
|
|
return _data?.freeplayDJ;
|
2024-06-19 00:07:27 +00:00
|
|
|
}
|
|
|
|
|
2024-06-20 20:17:53 +00:00
|
|
|
public function getFreeplayDJText(index:Int):String
|
|
|
|
{
|
2024-09-01 07:22:34 +00:00
|
|
|
// Silly little placeholder
|
|
|
|
return _data?.freeplayDJ?.getFreeplayDJText(index) ?? 'GET FREAKY ON A FRIDAY';
|
2024-06-20 20:17:53 +00:00
|
|
|
}
|
|
|
|
|
2024-09-01 07:22:34 +00:00
|
|
|
public function getCharSelectData():Null<PlayerCharSelectData>
|
2024-08-28 10:09:14 +00:00
|
|
|
{
|
2024-09-01 07:22:34 +00:00
|
|
|
return _data?.charSelect;
|
2024-08-28 10:09:14 +00:00
|
|
|
}
|
|
|
|
|
2024-07-04 02:50:39 +00:00
|
|
|
/**
|
|
|
|
* @param rank Which rank to get info for
|
|
|
|
* @return An array of animations. For example, BF Great has two animations, one for BF and one for GF
|
|
|
|
*/
|
|
|
|
public function getResultsAnimationDatas(rank:ScoringRank):Array<PlayerResultsAnimationData>
|
|
|
|
{
|
2024-09-01 07:22:34 +00:00
|
|
|
if (_data == null || _data.results == null)
|
2024-07-04 02:50:39 +00:00
|
|
|
{
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (rank)
|
|
|
|
{
|
|
|
|
case PERFECT | PERFECT_GOLD:
|
|
|
|
return _data.results.perfect;
|
|
|
|
case EXCELLENT:
|
|
|
|
return _data.results.excellent;
|
|
|
|
case GREAT:
|
|
|
|
return _data.results.great;
|
|
|
|
case GOOD:
|
|
|
|
return _data.results.good;
|
|
|
|
case SHIT:
|
|
|
|
return _data.results.loss;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-18 21:56:24 +00:00
|
|
|
/**
|
|
|
|
* Returns whether this character is unlocked.
|
|
|
|
*/
|
|
|
|
public function isUnlocked():Bool
|
|
|
|
{
|
2024-09-01 07:22:34 +00:00
|
|
|
return _data?.unlocked ?? true;
|
2024-06-18 21:56:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Called when the character is destroyed.
|
|
|
|
* TODO: Document when this gets called
|
|
|
|
*/
|
|
|
|
public function destroy():Void {}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
return 'PlayableCharacter($id)';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve and parse the JSON data for a playable character by ID.
|
|
|
|
* @param id The ID of the character
|
|
|
|
* @return The parsed player data, or null if not found or invalid
|
|
|
|
*/
|
|
|
|
static function _fetchData(id:String):Null<PlayerData>
|
|
|
|
{
|
|
|
|
return PlayerRegistry.instance.parseEntryDataWithMigration(id, PlayerRegistry.instance.fetchEntryVersion(id));
|
|
|
|
}
|
|
|
|
}
|