Editing Menus: Difference between revisions

From PMDOWiki
MistressNebula (talk | contribs)
Preparations edited in advance with for future code changes
MistressNebula (talk | contribs)
Connect to service tutorial pages
 
(12 intermediate revisions by the same user not shown)
Line 1: Line 1:
As of 0.8.3, a new service callback has been added to the game, called AddMenu. This callback can be used by modders to edit menus on the fly right before they appear on screen. This guide exists as a way to explain how this callback works and how to use it.
<small>''This guide does not apply to the [[Adding new Settings|Settings Menu]]. Please refer to its specific page if that's what you are looking for.''</small>
 
As of 0.8.3, a new service callback has been added to the game, called [[Service Events#EngineServiceEvents.AddMenu|AddMenu]]. This callback can be used by modders to edit menus on the fly right before they appear on screen. This guide exists as a way to explain how this callback works and how to use it.


== Prerequisites ==
== Prerequisites ==
This guide assumes that the user is capable of writing [[Service Callbacks|Services]] and [[Creating Menus]]. Please refer to these pages if anything is unclear.
This guide assumes that the user is capable of writing [[Services]] and [[Creating Menus]]. Please refer to these pages if anything is unclear.
The user is also expected to be able to navigate the RogueEssence repository, specifically looking into its [https://github.com/RogueCollab/RogueEssence/tree/master/RogueEssence/Menu Menu] folder and its subfolders.
The user is also expected to be able to navigate the RogueEssence repository, specifically looking into its [https://github.com/RogueCollab/RogueEssence/tree/master/RogueEssence/Menu Menu] folder and its subfolders.


=== Preparations ===
With the advent of 0.8.3, the callback AddMenu has been added to the game. This message is sent to services just before a menu is displayed so that modders can add, remove or modify its elements at their heart's content. What can be edited and how depends on the menu itself. Feel free to check the [[Menu Editing Reference]] page if you ever need more details on a specific function.
With the advent of 0.8.3, the callback AddMenu has been added to the game. This message is sent to services just before a menu is displayed so that modders can add, remove or modify its content at their heart's content. What can be edited depends on the menu itself.


There are three broader types of Menus. Each of them has one Scriptable counterpart and they are [https://github.com/RogueCollab/RogueEssence/blob/master/RogueEssence/Menu/InteractableMenu.cs InteractableMenu], [https://github.com/RogueCollab/RogueEssence/blob/master/RogueEssence/Menu/ChoiceMenu.cs ChoiceMenu] and [https://github.com/RogueCollab/RogueEssence/blob/master/RogueEssence/Menu/MultiPageMenu.cs MultiPageMenu]. Being able to differentiate between them is the key to know how to interact with the menu you're looking to edit. Furthermore, we will need its Label. A Label is a little string property that serves to identify a menu. Lucky for us, the game already guves us everything we need to know: if a menu has a label, its Menu Type and its Label will be displayed in the dev mode's output console every time you open it. So... let's do that! Here's what happens when we open the OthersMenu, for example:
There are three broader types of Menus. Each of them has one Scriptable counterpart and they are [[Menu Editing Reference#InteractableMenu|InteractableMenu]], [[Menu Editing Reference#ChoiceMenu|ChoiceMenu]] and [[Menu Editing Reference#MultiPageMenu|MultiPageMenu]]. Being able to differentiate between them is key to know how to interact with the menu you're looking to edit. Furthermore, we will need the menu's Label. A Label is a string property that serves to identify a menu. Lucky for us, the game already gives us everything we need to know by just opening the F1 overlay and hovering over the menu itself with your mouse. Here's the OthersMenu, for example:


[[File:Others_label.png]]
[[File:LabelDisplay.png]]


Great! Now we have all the pieces necessary to start working on our changes.
This is already everything we need. We can now start working on our changes.


== How to edit a menu ==
== How to edit a menu ==
Now that we have our menu type and our label, we need to write our AddMenu callback.
Say that we want to reproduce what the menu_tools service does in the base game. Since we already have our menu type and our label, we can start writing our AddMenu callback.
Since the AddMenu event message contains an argument, you will need to pass that argument on to your callback like this:<br>
The AddMenu event message contains an argument, so you will need to pass that argument on to your callback like this:<br>
<code>med:Subscribe("MenuTools", EngineServiceEvents.AddMenu, function(_, args) self:OnAddMenu(args[0]) end )</code>
<code>med:Subscribe("MenuTools", EngineServiceEvents.AddMenu, function(_, args) self:OnAddMenu(args[0]) end )</code>


Line 27: Line 28:
         local index = menu:GetChoiceIndexByLabel("OTH_SETTINGS")
         local index = menu:GetChoiceIndexByLabel("OTH_SETTINGS")
         if index <0 then index = math.min(1, menu.Choices.Count) end
         if index <0 then index = math.min(1, menu.Choices.Count) end
         menu.Choices:Insert(index, RogueEssence.Menu.MenuTextChoice("OTH_RECRUIT",
         local choices = menu:ExportChoices()
                RogueEssence.StringKey("MENU_RECRUITMENT"):ToLocal(),
        choices:Insert(index, RogueEssence.Menu.MenuTextChoice("OTH_RECRUIT", RogueEssence.StringKey("MENU_RECRUITMENT"):ToLocal(), function () _MENU:AddMenu(RecruitmentListMenu:new().menu, false) end))
                function () _MENU:AddMenu(RecruitmentListMenu:new().menu, false) end))
         menu:ImportChoices(choices)
         menu:InitMenu()
     end
     end
end
end
</pre>
</pre>


Now, what does this all mean?
Now, what does this all mean? Let's analyze it one step at a time.
Let's analyze it one step at a time.
 
First thinkgs first, this new option must only appear while inside a dungeon, so that's what the first part of the if statement does. We must then check if the menu that has been opened has a label, and, if it does, check if it's the label we're looking for.
First things first, this new option must only appear while inside a dungeon, so the first part of the if statement makes sure that's the case. We must then check if the menu that has been opened has a label, and, if it does, check if it's the label we're looking for.
 
Now that we know that this is the right menu, we want to add our new option just above Settings. Since menu elements can also have labels, we ask the menu for the index of its OTH_SETTINGS option and save that value. The function then checks if the number is valid, and defaults to slot 1 if it isn't, or even 0 if the menu is empty for some reason.
If you ever need an option's label, you can either hover it while displaying the F1 overlay or look them up inside of the [https://github.com/RogueCollab/RogueEssence/blob/master/RogueEssence/Menu/ILabeled.cs ILabeled.cs] file.


Now that we know this is the right menu, we want to add our new option just above Settings. Since menu elements also have labels, we ask the menu for the index of its OTH_SETTINGS option and save that value. The fubnction then checks if the number is valid, and defaults to either slot 1 or 0 if it isn't.
Finally, we apply the change: We first export the current list of choices, then add our new MenuTextChoice in the spot we chose and then load our edited list using ImportChoices.
Finally, we add our new MenuTextChoice in that spot and reload the Choice list by calling InitMenu.


== Functions ==
And it's done! The game will now display the Recruitment Search option every time we open the Others menu while we're inside a dungeon.
Now that we know how editing a menu works in practice, it's time to explore all of the tools offered by the engine to help modders.


''This article is currently still work in progress''
This process can be applied to any menu that has a label. The only exception is SettingsMenu, that has [[Adding new Settings|its own page]] describing how to interact with it.

Latest revision as of 15:55, 13 March 2025

This guide does not apply to the Settings Menu. Please refer to its specific page if that's what you are looking for.

As of 0.8.3, a new service callback has been added to the game, called AddMenu. This callback can be used by modders to edit menus on the fly right before they appear on screen. This guide exists as a way to explain how this callback works and how to use it.

Prerequisites

This guide assumes that the user is capable of writing Services and Creating Menus. Please refer to these pages if anything is unclear. The user is also expected to be able to navigate the RogueEssence repository, specifically looking into its Menu folder and its subfolders.

With the advent of 0.8.3, the callback AddMenu has been added to the game. This message is sent to services just before a menu is displayed so that modders can add, remove or modify its elements at their heart's content. What can be edited and how depends on the menu itself. Feel free to check the Menu Editing Reference page if you ever need more details on a specific function.

There are three broader types of Menus. Each of them has one Scriptable counterpart and they are InteractableMenu, ChoiceMenu and MultiPageMenu. Being able to differentiate between them is key to know how to interact with the menu you're looking to edit. Furthermore, we will need the menu's Label. A Label is a string property that serves to identify a menu. Lucky for us, the game already gives us everything we need to know by just opening the F1 overlay and hovering over the menu itself with your mouse. Here's the OthersMenu, for example:

This is already everything we need. We can now start working on our changes.

How to edit a menu

Say that we want to reproduce what the menu_tools service does in the base game. Since we already have our menu type and our label, we can start writing our AddMenu callback. The AddMenu event message contains an argument, so you will need to pass that argument on to your callback like this:
med:Subscribe("MenuTools", EngineServiceEvents.AddMenu, function(_, args) self:OnAddMenu(args[0]) end )

Now we write the function. Here's what menu_tools does:

function MenuTools:OnAddMenu(menu)
    if RogueEssence.GameManager.Instance.CurrentScene == RogueEssence.Dungeon.DungeonScene.Instance
            and menu:HasLabel() and menu.Label == "OTHERS_MENU" then
        local index = menu:GetChoiceIndexByLabel("OTH_SETTINGS")
        if index <0 then index = math.min(1, menu.Choices.Count) end
        local choices = menu:ExportChoices()
        choices:Insert(index, RogueEssence.Menu.MenuTextChoice("OTH_RECRUIT", RogueEssence.StringKey("MENU_RECRUITMENT"):ToLocal(), function () _MENU:AddMenu(RecruitmentListMenu:new().menu, false) end))
        menu:ImportChoices(choices)
    end
end

Now, what does this all mean? Let's analyze it one step at a time.

First things first, this new option must only appear while inside a dungeon, so the first part of the if statement makes sure that's the case. We must then check if the menu that has been opened has a label, and, if it does, check if it's the label we're looking for.

Now that we know that this is the right menu, we want to add our new option just above Settings. Since menu elements can also have labels, we ask the menu for the index of its OTH_SETTINGS option and save that value. The function then checks if the number is valid, and defaults to slot 1 if it isn't, or even 0 if the menu is empty for some reason. If you ever need an option's label, you can either hover it while displaying the F1 overlay or look them up inside of the ILabeled.cs file.

Finally, we apply the change: We first export the current list of choices, then add our new MenuTextChoice in the spot we chose and then load our edited list using ImportChoices.

And it's done! The game will now display the Recruitment Search option every time we open the Others menu while we're inside a dungeon.

This process can be applied to any menu that has a label. The only exception is SettingsMenu, that has its own page describing how to interact with it.