2023-11-07 04:04:22 -05:00
|
|
|
package funkin.ui.mainmenu;
|
2020-10-31 21:11:14 -04:00
|
|
|
|
2024-04-05 01:24:03 -04:00
|
|
|
import funkin.graphics.FunkinSprite;
|
2024-03-16 22:20:22 -04:00
|
|
|
import flixel.addons.transition.FlxTransitionableState;
|
2023-06-15 00:29:54 -04:00
|
|
|
import funkin.ui.debug.DebugMenuSubState;
|
2020-10-31 21:11:14 -04:00
|
|
|
import flixel.FlxObject;
|
|
|
|
import flixel.FlxSprite;
|
2021-02-15 17:04:08 -05:00
|
|
|
import flixel.FlxState;
|
2021-02-12 01:20:20 -05:00
|
|
|
import flixel.addons.transition.FlxTransitionableState;
|
2020-10-31 21:11:14 -04:00
|
|
|
import flixel.effects.FlxFlicker;
|
|
|
|
import flixel.graphics.frames.FlxAtlasFrames;
|
2024-02-05 19:46:11 -05:00
|
|
|
import flixel.util.typeLimit.NextState;
|
2020-10-31 21:11:14 -04:00
|
|
|
import flixel.group.FlxGroup.FlxTypedGroup;
|
2021-08-23 18:52:38 -04:00
|
|
|
import flixel.input.touch.FlxTouch;
|
2020-11-06 21:17:27 -05:00
|
|
|
import flixel.text.FlxText;
|
2024-03-11 23:42:32 -04:00
|
|
|
import funkin.data.song.SongData.SongMusicData;
|
2020-10-31 21:11:14 -04:00
|
|
|
import flixel.tweens.FlxEase;
|
2024-02-13 01:38:11 -05:00
|
|
|
import funkin.graphics.FunkinCamera;
|
2024-03-11 23:42:32 -04:00
|
|
|
import funkin.audio.FunkinSound;
|
2020-10-31 21:11:14 -04:00
|
|
|
import flixel.tweens.FlxTween;
|
2023-11-07 04:04:22 -05:00
|
|
|
import funkin.ui.MusicBeatState;
|
2021-02-15 17:04:08 -05:00
|
|
|
import flixel.util.FlxTimer;
|
2022-03-08 03:13:53 -05:00
|
|
|
import funkin.ui.AtlasMenuList;
|
2023-11-07 04:04:22 -05:00
|
|
|
import funkin.ui.freeplay.FreeplayState;
|
2022-03-08 03:13:53 -05:00
|
|
|
import funkin.ui.MenuList;
|
2023-08-08 15:41:25 -04:00
|
|
|
import funkin.ui.title.TitleState;
|
2023-05-17 16:42:58 -04:00
|
|
|
import funkin.ui.story.StoryMenuState;
|
2022-03-08 03:13:53 -05:00
|
|
|
import funkin.ui.Prompt;
|
2022-03-26 22:18:26 -04:00
|
|
|
import funkin.util.WindowUtil;
|
2021-03-30 00:05:21 -04:00
|
|
|
#if discord_rpc
|
2021-03-23 00:05:46 -04:00
|
|
|
import Discord.DiscordClient;
|
|
|
|
#end
|
2021-02-19 23:10:16 -05:00
|
|
|
#if newgrounds
|
2022-03-08 03:13:53 -05:00
|
|
|
import funkin.ui.NgPrompt;
|
2022-04-15 17:11:01 -04:00
|
|
|
import io.newgrounds.NG;
|
2021-02-19 23:10:16 -05:00
|
|
|
#end
|
|
|
|
|
2020-10-31 21:11:14 -04:00
|
|
|
class MainMenuState extends MusicBeatState
|
|
|
|
{
|
2023-01-22 22:25:45 -05:00
|
|
|
var menuItems:MenuTypedList<AtlasMenuItem>;
|
|
|
|
|
|
|
|
var magenta:FlxSprite;
|
|
|
|
var camFollow:FlxObject;
|
|
|
|
|
2024-05-31 05:39:53 -04:00
|
|
|
var overrideMusic:Bool = false;
|
|
|
|
|
2024-05-18 18:29:46 -04:00
|
|
|
static var rememberedSelectedIndex:Int = 0;
|
|
|
|
|
2024-05-31 05:39:53 -04:00
|
|
|
public function new(?_overrideMusic:Bool = false)
|
|
|
|
{
|
|
|
|
super();
|
|
|
|
overrideMusic = _overrideMusic;
|
|
|
|
}
|
|
|
|
|
2024-04-24 19:45:17 -04:00
|
|
|
override function create():Void
|
2023-01-22 22:25:45 -05:00
|
|
|
{
|
|
|
|
#if discord_rpc
|
|
|
|
// Updating Discord Rich Presence
|
|
|
|
DiscordClient.changePresence("In the Menus", null);
|
|
|
|
#end
|
|
|
|
|
2024-05-20 15:52:48 -04:00
|
|
|
FlxG.cameras.reset(new FunkinCamera('mainMenu'));
|
|
|
|
|
2023-01-22 22:25:45 -05:00
|
|
|
transIn = FlxTransitionableState.defaultTransIn;
|
|
|
|
transOut = FlxTransitionableState.defaultTransOut;
|
|
|
|
|
2024-06-06 20:15:11 -04:00
|
|
|
if (overrideMusic == false) playMenuMusic();
|
2023-01-22 22:25:45 -05:00
|
|
|
|
2024-04-30 16:30:22 -04:00
|
|
|
// We want the state to always be able to begin with being able to accept inputs and show the anims of the menu items.
|
|
|
|
persistentUpdate = true;
|
|
|
|
persistentDraw = true;
|
2023-01-22 22:25:45 -05:00
|
|
|
|
|
|
|
var bg:FlxSprite = new FlxSprite(Paths.image('menuBG'));
|
|
|
|
bg.scrollFactor.x = 0;
|
|
|
|
bg.scrollFactor.y = 0.17;
|
|
|
|
bg.setGraphicSize(Std.int(bg.width * 1.2));
|
|
|
|
bg.updateHitbox();
|
|
|
|
bg.screenCenter();
|
|
|
|
add(bg);
|
|
|
|
|
|
|
|
camFollow = new FlxObject(0, 0, 1, 1);
|
|
|
|
add(camFollow);
|
|
|
|
|
2024-04-16 20:19:20 -04:00
|
|
|
magenta = new FlxSprite(Paths.image('menuBGMagenta'));
|
2023-01-22 22:25:45 -05:00
|
|
|
magenta.scrollFactor.x = bg.scrollFactor.x;
|
|
|
|
magenta.scrollFactor.y = bg.scrollFactor.y;
|
|
|
|
magenta.setGraphicSize(Std.int(bg.width));
|
|
|
|
magenta.updateHitbox();
|
|
|
|
magenta.x = bg.x;
|
|
|
|
magenta.y = bg.y;
|
|
|
|
magenta.visible = false;
|
2023-10-17 00:38:28 -04:00
|
|
|
|
|
|
|
// TODO: Why doesn't this line compile I'm going fucking feral
|
|
|
|
|
|
|
|
if (Preferences.flashingLights) add(magenta);
|
2023-01-22 22:25:45 -05:00
|
|
|
|
|
|
|
menuItems = new MenuTypedList<AtlasMenuItem>();
|
|
|
|
add(menuItems);
|
|
|
|
menuItems.onChange.add(onMenuItemChange);
|
2023-03-16 00:55:25 -04:00
|
|
|
menuItems.onAcceptPress.add(function(_) {
|
2023-01-22 22:25:45 -05:00
|
|
|
if (_.name == 'freeplay')
|
|
|
|
{
|
|
|
|
magenta.visible = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
FlxFlicker.flicker(magenta, 1.1, 0.15, false, true);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
menuItems.enabled = true; // can move on intro
|
2024-02-05 19:46:11 -05:00
|
|
|
createMenuItem('storymode', 'mainmenu/storymode', function() startExitState(() -> new StoryMenuState()));
|
2023-03-16 00:55:25 -04:00
|
|
|
createMenuItem('freeplay', 'mainmenu/freeplay', function() {
|
2023-01-22 22:25:45 -05:00
|
|
|
persistentDraw = true;
|
|
|
|
persistentUpdate = false;
|
2023-08-02 11:00:23 -04:00
|
|
|
// Freeplay has its own custom transition
|
2024-03-16 22:20:22 -04:00
|
|
|
FlxTransitionableState.skipNextTransIn = true;
|
|
|
|
FlxTransitionableState.skipNextTransOut = true;
|
2024-04-29 23:31:55 -04:00
|
|
|
|
2023-01-22 22:25:45 -05:00
|
|
|
openSubState(new FreeplayState());
|
|
|
|
});
|
|
|
|
|
|
|
|
#if CAN_OPEN_LINKS
|
2024-03-28 02:57:51 -04:00
|
|
|
// In order to prevent popup blockers from triggering,
|
|
|
|
// we need to open the link as an immediate result of a keypress event,
|
|
|
|
// so we can't wait for the flicker animation to complete.
|
2023-01-22 22:25:45 -05:00
|
|
|
var hasPopupBlocker = #if web true #else false #end;
|
2024-03-28 02:57:51 -04:00
|
|
|
createMenuItem('merch', 'mainmenu/merch', selectMerch, hasPopupBlocker);
|
2023-01-22 22:25:45 -05:00
|
|
|
#end
|
|
|
|
|
2023-03-16 00:55:25 -04:00
|
|
|
createMenuItem('options', 'mainmenu/options', function() {
|
2024-02-05 19:46:11 -05:00
|
|
|
startExitState(() -> new funkin.ui.options.OptionsState());
|
2023-01-22 22:25:45 -05:00
|
|
|
});
|
|
|
|
|
2024-03-28 02:57:51 -04:00
|
|
|
createMenuItem('credits', 'mainmenu/credits', function() {
|
2024-03-26 12:33:54 -04:00
|
|
|
startExitState(() -> new funkin.ui.credits.CreditsState());
|
|
|
|
});
|
|
|
|
|
2023-01-22 22:25:45 -05:00
|
|
|
// Reset position of menu items.
|
|
|
|
var spacing = 160;
|
|
|
|
var top = (FlxG.height - (spacing * (menuItems.length - 1))) / 2;
|
|
|
|
for (i in 0...menuItems.length)
|
|
|
|
{
|
|
|
|
var menuItem = menuItems.members[i];
|
|
|
|
menuItem.x = FlxG.width / 2;
|
|
|
|
menuItem.y = top + spacing * i;
|
2024-03-26 12:33:54 -04:00
|
|
|
menuItem.scrollFactor.x = 0.0;
|
|
|
|
// This one affects how much the menu items move when you scroll between them.
|
|
|
|
menuItem.scrollFactor.y = 0.4;
|
2023-01-22 22:25:45 -05:00
|
|
|
}
|
|
|
|
|
2024-05-18 18:29:46 -04:00
|
|
|
menuItems.selectItem(rememberedSelectedIndex);
|
|
|
|
|
2023-03-16 00:55:25 -04:00
|
|
|
resetCamStuff();
|
|
|
|
|
2024-06-24 23:43:34 -04:00
|
|
|
// reset camera when debug menu is closed
|
|
|
|
subStateClosed.add(_ -> resetCamStuff(false));
|
|
|
|
|
2023-03-16 00:55:25 -04:00
|
|
|
subStateOpened.add(sub -> {
|
|
|
|
if (Type.getClass(sub) == FreeplayState)
|
|
|
|
{
|
|
|
|
new FlxTimer().start(0.5, _ -> {
|
|
|
|
magenta.visible = false;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2023-01-22 22:25:45 -05:00
|
|
|
// FlxG.camera.setScrollBounds(bg.x, bg.x + bg.width, bg.y, bg.y + bg.height * 1.2);
|
|
|
|
|
|
|
|
super.create();
|
|
|
|
|
|
|
|
// This has to come AFTER!
|
|
|
|
this.leftWatermarkText.text = Constants.VERSION;
|
|
|
|
// this.rightWatermarkText.text = "blablabla test";
|
|
|
|
|
|
|
|
// NG.core.calls.event.logEvent('swag').send();
|
|
|
|
}
|
|
|
|
|
2024-03-11 23:42:32 -04:00
|
|
|
function playMenuMusic():Void
|
|
|
|
{
|
2024-03-23 17:50:48 -04:00
|
|
|
FunkinSound.playMusic('freakyMenu',
|
|
|
|
{
|
|
|
|
overrideExisting: true,
|
|
|
|
restartTrack: false
|
|
|
|
});
|
2024-03-11 23:42:32 -04:00
|
|
|
}
|
|
|
|
|
2024-06-24 23:43:34 -04:00
|
|
|
function resetCamStuff(?snap:Bool = true):Void
|
2023-03-16 00:55:25 -04:00
|
|
|
{
|
|
|
|
FlxG.camera.follow(camFollow, null, 0.06);
|
2024-06-24 23:43:34 -04:00
|
|
|
|
|
|
|
if (snap) FlxG.camera.snapToTarget();
|
2023-03-16 00:55:25 -04:00
|
|
|
}
|
|
|
|
|
2023-01-22 22:25:45 -05:00
|
|
|
function createMenuItem(name:String, atlas:String, callback:Void->Void, fireInstantly:Bool = false):Void
|
|
|
|
{
|
|
|
|
var item = new AtlasMenuItem(name, Paths.getSparrowAtlas(atlas), callback);
|
|
|
|
item.fireInstantly = fireInstantly;
|
|
|
|
item.ID = menuItems.length;
|
|
|
|
|
|
|
|
item.scrollFactor.set();
|
|
|
|
|
|
|
|
// Set the offset of the item so the sprite is centered on the origin.
|
|
|
|
item.centered = true;
|
|
|
|
item.changeAnim('idle');
|
|
|
|
|
|
|
|
menuItems.addItem(name, item);
|
|
|
|
}
|
|
|
|
|
2024-04-24 19:45:17 -04:00
|
|
|
override function closeSubState():Void
|
2023-01-22 22:25:45 -05:00
|
|
|
{
|
|
|
|
magenta.visible = false;
|
|
|
|
|
|
|
|
super.closeSubState();
|
|
|
|
}
|
|
|
|
|
2024-04-24 19:45:17 -04:00
|
|
|
override function finishTransIn():Void
|
2023-01-22 22:25:45 -05:00
|
|
|
{
|
|
|
|
super.finishTransIn();
|
|
|
|
|
|
|
|
// menuItems.enabled = true;
|
|
|
|
|
|
|
|
// #if newgrounds
|
|
|
|
// if (NGio.savedSessionFailed)
|
|
|
|
// showSavedSessionFailed();
|
|
|
|
// #end
|
|
|
|
}
|
|
|
|
|
2023-11-07 04:04:22 -05:00
|
|
|
function onMenuItemChange(selected:MenuListItem)
|
2023-01-22 22:25:45 -05:00
|
|
|
{
|
|
|
|
camFollow.setPosition(selected.getGraphicMidpoint().x, selected.getGraphicMidpoint().y);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if CAN_OPEN_LINKS
|
|
|
|
function selectDonate()
|
|
|
|
{
|
|
|
|
WindowUtil.openURL(Constants.URL_ITCH);
|
|
|
|
}
|
2024-03-28 02:57:51 -04:00
|
|
|
|
|
|
|
function selectMerch()
|
|
|
|
{
|
|
|
|
WindowUtil.openURL(Constants.URL_MERCH);
|
|
|
|
}
|
2023-01-22 22:25:45 -05:00
|
|
|
#end
|
|
|
|
|
|
|
|
#if newgrounds
|
|
|
|
function selectLogin()
|
|
|
|
{
|
|
|
|
openNgPrompt(NgPrompt.showLogin());
|
|
|
|
}
|
|
|
|
|
|
|
|
function selectLogout()
|
|
|
|
{
|
|
|
|
openNgPrompt(NgPrompt.showLogout());
|
|
|
|
}
|
|
|
|
|
|
|
|
function showSavedSessionFailed()
|
|
|
|
{
|
|
|
|
openNgPrompt(NgPrompt.showSavedSessionFailed());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calls openPrompt and redraws the login/logout button
|
2023-06-08 16:30:45 -04:00
|
|
|
* @param prompt
|
|
|
|
* @param onClose
|
2023-01-22 22:25:45 -05:00
|
|
|
*/
|
|
|
|
public function openNgPrompt(prompt:Prompt, ?onClose:Void->Void)
|
|
|
|
{
|
|
|
|
var onPromptClose = checkLoginStatus;
|
|
|
|
if (onClose != null)
|
|
|
|
{
|
2023-03-16 00:55:25 -04:00
|
|
|
onPromptClose = function() {
|
2023-01-22 22:25:45 -05:00
|
|
|
checkLoginStatus();
|
|
|
|
onClose();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
openPrompt(prompt, onPromptClose);
|
|
|
|
}
|
|
|
|
|
|
|
|
function checkLoginStatus()
|
|
|
|
{
|
|
|
|
var prevLoggedIn = menuItems.has("logout");
|
|
|
|
if (prevLoggedIn && !NGio.isLoggedIn) menuItems.resetItem("login", "logout", selectLogout);
|
|
|
|
else if (!prevLoggedIn && NGio.isLoggedIn) menuItems.resetItem("logout", "login", selectLogin);
|
|
|
|
}
|
|
|
|
#end
|
|
|
|
|
2024-04-24 19:45:17 -04:00
|
|
|
public function openPrompt(prompt:Prompt, onClose:Void->Void):Void
|
2023-01-22 22:25:45 -05:00
|
|
|
{
|
|
|
|
menuItems.enabled = false;
|
2024-04-29 23:31:55 -04:00
|
|
|
persistentUpdate = false;
|
|
|
|
|
2023-03-16 00:55:25 -04:00
|
|
|
prompt.closeCallback = function() {
|
2023-01-22 22:25:45 -05:00
|
|
|
menuItems.enabled = true;
|
|
|
|
if (onClose != null) onClose();
|
|
|
|
}
|
|
|
|
|
|
|
|
openSubState(prompt);
|
|
|
|
}
|
|
|
|
|
2024-04-24 19:45:17 -04:00
|
|
|
function startExitState(state:NextState):Void
|
2023-01-22 22:25:45 -05:00
|
|
|
{
|
|
|
|
menuItems.enabled = false; // disable for exit
|
2024-05-18 18:29:46 -04:00
|
|
|
rememberedSelectedIndex = menuItems.selectedIndex;
|
|
|
|
|
2023-01-22 22:25:45 -05:00
|
|
|
var duration = 0.4;
|
2023-03-16 00:55:25 -04:00
|
|
|
menuItems.forEach(function(item) {
|
2023-01-22 22:25:45 -05:00
|
|
|
if (menuItems.selectedIndex != item.ID)
|
|
|
|
{
|
|
|
|
FlxTween.tween(item, {alpha: 0}, duration, {ease: FlxEase.quadOut});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
item.visible = false;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
new FlxTimer().start(duration, function(_) FlxG.switchState(state));
|
|
|
|
}
|
|
|
|
|
2024-04-24 19:45:17 -04:00
|
|
|
override function update(elapsed:Float):Void
|
2023-01-22 22:25:45 -05:00
|
|
|
{
|
|
|
|
super.update(elapsed);
|
|
|
|
|
|
|
|
if (FlxG.onMobile)
|
|
|
|
{
|
|
|
|
var touch:FlxTouch = FlxG.touches.getFirst();
|
|
|
|
|
|
|
|
if (touch != null)
|
|
|
|
{
|
|
|
|
for (item in menuItems)
|
|
|
|
{
|
|
|
|
if (touch.overlaps(item))
|
|
|
|
{
|
|
|
|
if (menuItems.selectedIndex == item.ID && touch.justPressed) menuItems.accept();
|
|
|
|
else
|
|
|
|
menuItems.selectItem(item.ID);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-16 00:02:42 -05:00
|
|
|
// Open the debug menu, defaults to ` / ~
|
2024-04-24 16:00:50 -04:00
|
|
|
#if CHART_EDITOR_SUPPORTED
|
2023-11-16 00:02:42 -05:00
|
|
|
if (controls.DEBUG_MENU)
|
2023-06-15 00:29:54 -04:00
|
|
|
{
|
2024-04-29 23:31:55 -04:00
|
|
|
persistentUpdate = false;
|
|
|
|
|
2023-06-15 00:29:54 -04:00
|
|
|
FlxG.state.openSubState(new DebugMenuSubState());
|
|
|
|
}
|
2024-04-24 16:00:50 -04:00
|
|
|
#end
|
2023-01-22 22:25:45 -05:00
|
|
|
|
2024-04-21 17:23:48 -04:00
|
|
|
#if (debug || FORCE_DEBUG_VERSION)
|
|
|
|
if (FlxG.keys.pressed.CONTROL && FlxG.keys.pressed.ALT && FlxG.keys.pressed.SHIFT && FlxG.keys.justPressed.W)
|
|
|
|
{
|
|
|
|
// Give the user a score of 1 point on Weekend 1 story mode.
|
|
|
|
// This makes the level count as cleared and displays the songs in Freeplay.
|
|
|
|
funkin.save.Save.instance.setLevelScore('weekend1', 'easy',
|
|
|
|
{
|
|
|
|
score: 1,
|
|
|
|
tallies:
|
|
|
|
{
|
|
|
|
sick: 0,
|
|
|
|
good: 0,
|
|
|
|
bad: 0,
|
|
|
|
shit: 0,
|
|
|
|
missed: 0,
|
|
|
|
combo: 0,
|
|
|
|
maxCombo: 0,
|
|
|
|
totalNotesHit: 0,
|
|
|
|
totalNotes: 0,
|
2024-05-17 20:26:34 -04:00
|
|
|
}
|
2024-04-21 17:23:48 -04:00
|
|
|
});
|
|
|
|
}
|
2024-06-11 00:40:43 -04:00
|
|
|
|
|
|
|
if (FlxG.keys.pressed.CONTROL && FlxG.keys.pressed.ALT && FlxG.keys.pressed.SHIFT && FlxG.keys.justPressed.R)
|
|
|
|
{
|
|
|
|
// Give the user a hypothetical overridden score,
|
|
|
|
// and see if we can maintain that golden P rank.
|
|
|
|
funkin.save.Save.instance.setSongScore('tutorial', 'easy',
|
|
|
|
{
|
|
|
|
score: 1234567,
|
|
|
|
tallies:
|
|
|
|
{
|
|
|
|
sick: 0,
|
|
|
|
good: 0,
|
|
|
|
bad: 0,
|
|
|
|
shit: 1,
|
|
|
|
missed: 0,
|
|
|
|
combo: 0,
|
|
|
|
maxCombo: 0,
|
|
|
|
totalNotesHit: 1,
|
|
|
|
totalNotes: 10,
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FlxG.keys.pressed.CONTROL && FlxG.keys.pressed.ALT && FlxG.keys.pressed.SHIFT && FlxG.keys.justPressed.E)
|
|
|
|
{
|
|
|
|
funkin.save.Save.instance.debug_dumpSave();
|
|
|
|
}
|
2024-04-21 17:23:48 -04:00
|
|
|
#end
|
|
|
|
|
2024-04-30 13:34:34 -04:00
|
|
|
if (FlxG.sound.music != null && FlxG.sound.music.volume < 0.8)
|
2023-01-22 22:25:45 -05:00
|
|
|
{
|
|
|
|
FlxG.sound.music.volume += 0.5 * elapsed;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_exiting) menuItems.enabled = false;
|
|
|
|
|
|
|
|
if (controls.BACK && menuItems.enabled && !menuItems.busy)
|
|
|
|
{
|
2024-03-23 17:50:48 -04:00
|
|
|
FunkinSound.playOnce(Paths.sound('cancelMenu'));
|
2024-02-05 19:46:11 -05:00
|
|
|
FlxG.switchState(() -> new TitleState());
|
2023-01-22 22:25:45 -05:00
|
|
|
}
|
|
|
|
}
|
2020-10-31 21:11:14 -04:00
|
|
|
}
|