Standard Menu UI/Number Setting Example

Below is the code for the Wrye Leveling options command token. This is fairly compact code for presenting several menus and handling the selected options. The point of this page is to demonstrate some techniques and standards for arranging hierachical menu code, and handling menu selections.

Background (Cobl Option Token handling):

  • This short section is a bit OT. Just FYI, explains the part of the code that's concerned with physical movement of the option tokens.
  • Option tokens are stored in a special options container. The player can then select options from the container, which then triggers the corresponding menu.
  • To prevent the player from using the container as free storage, all container contents are removed to the player when the container is closed.
  • Hence, each token, when added to the player will: 1) add a new copy of itself back into the options container, 2) (if not in cleanup mode), show the the options menu, and 3) self-destruct (removeMe).
  • Because of the way this is structured, it's guaranteed that everything that is menu display/handling related will all occure while in menu mode.

Menu Display/Handling:

  • Note that the menu display code is actually the last thing that happens. Even the onAdd block doesn't show a menu.
  • The current menu is indicated by menu variable. If this variable is negative, it means "Skip the menu handling section." So to trigger the top menu the first time, I set menu to -1; that gets inverted at the top of the handling section, and then falls through to the menu display section.
  • Menu handling is together, separate and above menu display. This means that menu handling can carry out operations and then change the current menu number as necessary. I'm starting to use the standard "menu == 100" means exit.

Button Handling Check out the menu handling section beginning with "elseif menu == 4 ;--Final Limits". There are a couple of things I need to do here:

  • Recognize the Back button.
  • Allow numbers to be incremented up or down.
  • Set restrictions on how much numbers can be incremented up or down.
  • Set different increment amounts for different buttons. Level goes up/down by 5 units at a time, while attrs and skill limit goes up/down by 10 units at a time.

So, what I do is:

  • First determine whether increment is up or down. Delta is thus set to 1 or -1 depending on whether shift key is pressed.
  • Set temp local variables to what value for each button will be if change is applied to it.
  • Finally, do the if block which:
    • Handles buttons and upper/lower limits for each item.
    • Handles back button.
    • Displays "Can't go up." or "Can't go down." message if the above blocks fail.

Note that there's a lot of similarity in the lines of the code. Coding was speeded by writing first example of a set of lines, then duplicating that several times, then doing search and replace.

scn wrLevOptOS
;--Wrye Levling: Options
short button
short menu

;--Temp values
long key
long delta
long costBase   ;--Levelup cost = costBase + nextLevel*costPerLevel
long costPerLevel 
long attrPoints  ;--Attribute points available per levelup
long skillPoints ;--Skill points available per levelup
long maxLevel   ;--Maximum level player is allowed to reach
long maxAttr     ;--Maximum value an attribute can have
long maxSkill   ;--Maximum value a skill can have
long maxAttrGain ;--Maximum attribute increase per levelup
long maxLuckGain ;--Maximum luck increase per levelup
long maxSkillGain;--Maximum skill increase per levelup
long daysBetween ;--Minimum days between levelups

begin onAdd player;-----------------------------------------------------------
;--Restock the options box
    cobOptChestRef.addItem wrLevOpt 1 

;--Ignore if added by optionsbox purge.
    if cobOptChestRef.state != 20 
        removeMe
    endif

    set menu to -1
end;onAdd

begin menuMode;---------------------------------------------------------------
    ;--Quick return if not in menu displaying mode.
    if menu == 0 
        return
    endif

    set button to getButtonPressed
    if menu < 0
        set menu to -menu
    
    elseif button == -1
        return

    elseif menu == 1;--Main Menu
        if button < 3
            set menu to button + 2
        else
            set menu to 100
        endif

    elseif menu == 2;--Cost
        set key to getKeyPress 0
        set delta to 1 - 2 * (key == 42 || key == 54)
        set costBase to wrLevZ.costBase + 25*delta
        set costPerLevel to wrLevZ.costPerLevel + 25*delta
        if button == 0 && costBase >= 0 && costBase <= 10000
            set wrLevZ.costBase to costBase
        elseif button == 1 && costPerLevel >= 50 && costPerLevel <= 10000
            set wrLevZ.costPerLevel to costPerLevel
        elseif button > 1
            set menu to 1
        elseif delta > 0
            message "That cannot be raised further."
        else
            message "That cannot be lowered further."
        endif
    
    elseif menu == 3;--Round Limits
        set key to getKeyPress 0
        set delta to 1 - 2 * (key == 42 || key == 54)
        set attrPoints   to wrLevZ.attrPoints   + delta
        set skillPoints  to wrLevZ.skillPoints  + delta
        set maxAttrGain  to wrLevZ.maxAttrGain  + delta
        set maxLuckGain  to wrLevZ.maxLuckGain  + delta
        set maxSkillGain to wrLevZ.maxSkillGain + delta
        set daysBetween  to wrLevZ.daysBetween  + delta
        if button == 0 && attrPoints >= 3 && attrPoints <= 15
            set wrLevZ.attrPoints to attrPoints
        elseif button == 1 && skillPoints >= 10 && skillPoints <= 20
            set wrLevZ.skillPoints to skillPoints
        elseif button == 2 && maxAttrGain >= 3 && maxAttrGain <= 15
            set wrLevZ.maxAttrGain to maxAttrGain
        elseif button == 3 && maxLuckGain >= 1 && maxLuckGain <= 5
            set wrLevZ.maxLuckGain to maxLuckGain
        elseif button == 4 && maxSkillGain >= 5 && maxSkillGain <= 20
            set wrLevZ.maxSkillGain to maxSkillGain
        elseif button == 5 && daysBetween >= 0 && daysBetween <= 90
            set wrLevZ.daysBetween to daysBetween
        elseif button > 5
            set menu to 1
        elseif delta > 0
            message "That cannot be raised further."
        else
            message "That cannot be lowered further."
        endif

    elseif menu == 4;--Final Limits
        set key to getKeyPress 0
        set delta to 1 - 2 * (key == 42 || key == 54)
        set maxLevel to wrLevZ.maxLevel + 5*delta
        set maxAttr  to wrLevZ.maxAttr  + 10*delta
        set maxSkill to wrLevZ.maxSkill + 10*delta
        if button == 0 && maxLevel >= 30 && maxLevel <= 100
            set wrLevZ.maxLevel to maxLevel
        elseif button == 1 && maxAttr >= 50 && maxAttr <= 250
            set wrLevZ.maxAttr to maxAttr
        elseif button == 2 && maxSkill >= 50 && maxSkill <= 250
            set wrLevZ.maxSkill to maxSkill
        elseif button > 2
            set menu to 1
        elseif delta > 0
            message "That cannot be raised further."
        else
            message "That cannot be lowered further."
        endif

    endif

;--Menus ------------------------------------------------------------------
    if menu == 1;--Main Menu
        messageBoxEx "Wrye Leveling configuration.|Cost >|Round Limits >|Final Limits >|Done"

    elseif menu == 2
        messageBoxEx "Cost = CostBase + level*CostPerLevel. Click or shift-click to modify.|CostBase %g|CostPerLevel %g|< Back" wrLevZ.costBase wrLevZ.costPerLevel
    
    elseif menu == 3
        messageBoxEx "Configure levelup round limits. Click or shift-click to modify.|Attr Points %g|Skill Points %g|Max Attr Gain %g|Max Luck Gain %g|Max Skill Gain %g|Days Between %g|< Back" wrLevZ.attrPoints wrLevZ.skillPoints wrLevZ.maxAttrGain wrLevZ.maxLuckGain wrLevZ.maxSkillGain wrLevZ.daysBetween

    elseif menu == 4
        messageBoxEx "Configure level, attribute and skills limits. Click or shift-click to modify.|Level Max %g|Attr Max %g|Skill Max %g|< Back" wrLevZ.maxLevel wrLevZ.maxAttr wrLevz.maxSkill

    elseif menu == 100;--Done
        removeMe

    endif

end;--menuMode