2023-12-12 10:24:43 +00:00
|
|
|
package funkin.ui.debug.charting.dialogs;
|
|
|
|
|
|
|
|
import flixel.math.FlxPoint;
|
|
|
|
import funkin.play.character.BaseCharacter.CharacterType;
|
|
|
|
import funkin.play.character.CharacterData;
|
|
|
|
import funkin.play.character.CharacterData.CharacterDataParser;
|
|
|
|
import funkin.play.components.HealthIcon;
|
|
|
|
import funkin.ui.debug.charting.dialogs.ChartEditorBaseDialog.DialogParams;
|
|
|
|
import funkin.util.SortUtil;
|
|
|
|
import haxe.ui.components.Label;
|
|
|
|
import haxe.ui.containers.Grid;
|
|
|
|
import haxe.ui.containers.HBox;
|
|
|
|
import haxe.ui.containers.ScrollView;
|
|
|
|
import haxe.ui.containers.ScrollView;
|
|
|
|
import haxe.ui.core.Screen;
|
2023-12-14 04:11:12 +00:00
|
|
|
import flixel.tweens.FlxTween;
|
|
|
|
import flixel.tweens.FlxEase;
|
2023-12-12 10:24:43 +00:00
|
|
|
|
|
|
|
// @:nullSafety // TODO: Fix null safety when used with HaxeUI build macros.
|
|
|
|
@:access(funkin.ui.debug.charting.ChartEditorState)
|
|
|
|
@:build(haxe.ui.ComponentBuilder.build("assets/exclude/data/ui/chart-editor/dialogs/character-icon-selector.xml"))
|
|
|
|
class ChartEditorCharacterIconSelectorMenu extends ChartEditorBaseMenu
|
|
|
|
{
|
|
|
|
public var charSelectScroll:ScrollView;
|
|
|
|
public var charIconName:Label;
|
|
|
|
|
2023-12-16 02:09:01 +00:00
|
|
|
public function new(chartEditorState2:ChartEditorState, charType:CharacterType, lockPosition:Bool = false)
|
2023-12-12 10:24:43 +00:00
|
|
|
{
|
2023-12-16 02:09:01 +00:00
|
|
|
super(chartEditorState2);
|
2023-12-12 10:24:43 +00:00
|
|
|
|
|
|
|
initialize(charType, lockPosition);
|
2023-12-14 04:11:12 +00:00
|
|
|
this.alpha = 0;
|
|
|
|
this.y -= 10;
|
|
|
|
FlxTween.tween(this, {alpha: 1, y: this.y + 10}, 0.2, {ease: FlxEase.quartOut});
|
2023-12-12 10:24:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function initialize(charType:CharacterType, lockPosition:Bool)
|
|
|
|
{
|
|
|
|
var currentCharId:String = switch (charType)
|
|
|
|
{
|
2023-12-16 02:09:01 +00:00
|
|
|
case BF: chartEditorState.currentSongMetadata.playData.characters.player;
|
|
|
|
case GF: chartEditorState.currentSongMetadata.playData.characters.girlfriend;
|
|
|
|
case DAD: chartEditorState.currentSongMetadata.playData.characters.opponent;
|
2023-12-12 10:24:43 +00:00
|
|
|
default: throw 'Invalid charType: ' + charType;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Position this menu.
|
|
|
|
var targetHealthIcon:Null<HealthIcon> = switch (charType)
|
|
|
|
{
|
2023-12-16 02:09:01 +00:00
|
|
|
case BF: chartEditorState.healthIconBF;
|
|
|
|
case DAD: chartEditorState.healthIconDad;
|
2023-12-12 10:24:43 +00:00
|
|
|
default: null;
|
|
|
|
};
|
|
|
|
|
|
|
|
if (lockPosition && targetHealthIcon != null)
|
|
|
|
{
|
|
|
|
var healthIconBottomCenter:FlxPoint = new FlxPoint(targetHealthIcon.x + targetHealthIcon.width / 2, targetHealthIcon.y + targetHealthIcon.height);
|
|
|
|
|
|
|
|
this.x = healthIconBottomCenter.x - this.width / 2;
|
|
|
|
this.y = healthIconBottomCenter.y;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this.x = Screen.instance.currentMouseX;
|
|
|
|
this.y = Screen.instance.currentMouseY;
|
|
|
|
}
|
|
|
|
|
|
|
|
var charGrid = new Grid();
|
|
|
|
charGrid.columns = 5;
|
2024-05-08 22:30:40 +00:00
|
|
|
charGrid.width = this.width;
|
2023-12-12 10:24:43 +00:00
|
|
|
charSelectScroll.addComponent(charGrid);
|
|
|
|
|
|
|
|
var charIds:Array<String> = CharacterDataParser.listCharacterIds();
|
|
|
|
charIds.sort(SortUtil.alphabetically);
|
|
|
|
|
|
|
|
var defaultText:String = '(choose a character)';
|
|
|
|
|
|
|
|
for (charIndex => charId in charIds)
|
|
|
|
{
|
|
|
|
var charData:CharacterData = CharacterDataParser.fetchCharacterData(charId);
|
|
|
|
|
|
|
|
var charButton = new haxe.ui.components.Button();
|
|
|
|
charButton.width = 70;
|
|
|
|
charButton.height = 70;
|
|
|
|
charButton.padding = 8;
|
|
|
|
charButton.iconPosition = "top";
|
|
|
|
|
|
|
|
if (charId == currentCharId)
|
|
|
|
{
|
|
|
|
// Scroll to the character if it is already selected.
|
|
|
|
charSelectScroll.hscrollPos = Math.floor(charIndex / 5) * 80;
|
|
|
|
charButton.selected = true;
|
|
|
|
|
|
|
|
defaultText = '${charData.name} [${charId}]';
|
|
|
|
}
|
|
|
|
|
|
|
|
var LIMIT = 6;
|
2024-09-17 02:04:04 +00:00
|
|
|
charButton.icon = haxe.ui.util.Variant.fromImageData(CharacterDataParser.getCharPixelIconAsset(charId));
|
2023-12-12 10:24:43 +00:00
|
|
|
charButton.text = charData.name.length > LIMIT ? '${charData.name.substr(0, LIMIT)}.' : '${charData.name}';
|
|
|
|
|
|
|
|
charButton.onClick = _ -> {
|
|
|
|
switch (charType)
|
|
|
|
{
|
2023-12-16 02:09:01 +00:00
|
|
|
case BF: chartEditorState.currentSongMetadata.playData.characters.player = charId;
|
|
|
|
case GF: chartEditorState.currentSongMetadata.playData.characters.girlfriend = charId;
|
|
|
|
case DAD: chartEditorState.currentSongMetadata.playData.characters.opponent = charId;
|
2023-12-12 10:24:43 +00:00
|
|
|
default: throw 'Invalid charType: ' + charType;
|
|
|
|
};
|
|
|
|
|
2023-12-16 02:09:01 +00:00
|
|
|
chartEditorState.healthIconsDirty = true;
|
|
|
|
chartEditorState.refreshToolbox(ChartEditorState.CHART_EDITOR_TOOLBOX_METADATA_LAYOUT);
|
2023-12-12 10:24:43 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
charButton.onMouseOver = _ -> {
|
|
|
|
charIconName.text = '${charData.name} [${charId}]';
|
|
|
|
};
|
|
|
|
charButton.onMouseOut = _ -> {
|
|
|
|
charIconName.text = defaultText;
|
|
|
|
};
|
|
|
|
charGrid.addComponent(charButton);
|
|
|
|
}
|
|
|
|
|
|
|
|
charIconName.text = defaultText;
|
|
|
|
}
|
|
|
|
|
2023-12-16 02:09:01 +00:00
|
|
|
public static function build(chartEditorState:ChartEditorState, charType:CharacterType, lockPosition:Bool = false):ChartEditorCharacterIconSelectorMenu
|
2023-12-12 10:24:43 +00:00
|
|
|
{
|
2023-12-16 02:09:01 +00:00
|
|
|
var menu = new ChartEditorCharacterIconSelectorMenu(chartEditorState, charType, lockPosition);
|
2023-12-12 10:24:43 +00:00
|
|
|
|
|
|
|
Screen.instance.addComponent(menu);
|
|
|
|
|
|
|
|
return menu;
|
|
|
|
}
|
|
|
|
}
|