How to use display verbs and inventory verbs effectively

The Pixie
When you play a text adventure using Quest, there will usually be a set of panels on the right. As well as the compass rose, there will be lists of objects in the current location and in your inventory. If you click on an object, buttons will appear giving short-cuts to commands with the object. These are display verbs and inventory verbs.


Adding And Removing Verbs

There is a simple way to change the list of verbs for an object. On the Object tab, at the bottom, is a section called "Display verbs". You can add and remove as appropriate. For example, if an object cannot be picked up, remove the "Take" entry from the display verbs.

By the way you can add anything you like here, even if it makes no sense to Quest. It is, therefore, a good idea when playing through your game to click on each verb for every object to see how Quest responds (if it can be picked up, do it for both in the inventory and in the room).

Changing the object type on the Setup tab will also change the verbs. Changing it to a male character, for example, will change the displayer verbs to "Look at" and "Speak to", rather than "Look at" and "Take".

When you add a verb to an object via the Verb tab, Quest will automatically add that verb to both the display verbs and the inventory verb. You can stop it doing that by unticking the "Automatically generate object display verb lists" box on the Room Descriptions tab (I do not know why either) of the game object. I prefer to do this, as it gives you full control over the verbs that will be shown. You can also stop verbs being generated automatically for a specific item by ticking the box on the Object tab for that object.


Adding And Removing Verbs On The Fly

Note: I have edited this (28/Jan/16) as it was pointed out the old method does not work consistently when playing on-line.

So far so good, but what if you want verbs to change during the course of the game? Say there is a hat that can be worn, so you want a "Wear" verb, but when put on you want a "Remove" verb instead. The verbs are held in two string list attributes, displayverbs and inventoryverbs.

There are issues to be aware of. Firstly, automatically generated verbs are not in that list (another good reason to turn the feature off).

Secondly, your object will only have those attributes if you have modified the lists on the Object tab. If you go to the Attributes tab, look for displayverbs in the list at the bottom. If it is in grey, your object is getting its list from its type, and if you try to add or remove something in the list during play, you will get this helpful message:

Error running script: Cannot modify the contents of this list as it is defined by an inherited type. Clone it before attempting to modify.

Thirdly, the web player is not great at spotting when the list has changed, so might not update the display anyway.


Coding...

So how do you actually add and remove verbs? Let us suppose we have an object called "hat", and we want to add a "Wear" verb to the inventory list. The safe approach is to create a new list each time. This is easily done with the Split function. This takes two strings, the first being a list of verbs, seperated by semi-colons, the second just a semi-colon, telling Quest what to break the first list on.
hat.inventoryverbs = Split("look at;Drop;Wear", ";")

Then when the hat is worn:
hat.inventoryverbs = Split("look at;Remove", ";")


Alternatively, here are a set of functions you can paste into your game in File - Code view. These will let you add and remove single verbs.
  <function name="AddDisplayVerb" parameters="c, s">
if (not ListContains(c.displayverbs, s)) {
s2 = Join(c.displayverbs, "|") + "|" + s
c.displayverbs = Split(s2, "|")
}
</function>

<function name="RemoveDisplayVerb" parameters="c, s">
a = NewStringList()
foreach (s2, c.displayverbs) {
if (not s2 = s) {
list add (a, s2)
}
}
c.displayverbs = a
</function>

<function name="AddInventoryVerb" parameters="c, s">
if (not ListContains(c.inventoryverbs, s)) {
s2 = Join(c.inventoryverbs, "|") + "|" + s
c.inventoryverbs= Split(s2, "|")
}
</function>

<function name="RemoveInventoryVerb" parameters="c, s">
a = NewStringList()
foreach (s2, c.inventoryverbs) {
if (not s2 = s) {
list add (a, s2)
}
}
c.inventoryverbs= a
</function>

Now you can do:
AddInventoryVerb(hat, "Wear")

Using this technique you can have a game where an object's verbs are dynamically assigned and always make sense, making life much easier, and therefore more enjoyable, for the player.

Silver
Thanks for that. It often puzzled me when playing games with clickable objects that you could click one of the verbs and it would return a negative.

Not related to this but similarly to being able to remove a verb that isn't there I found it very helpful that you can stop timers that aren't running; which is useful if a script employs several timers sequentially but you want the one running to stop but have no way of checking which it is.

That sentence was probably longer than it needed to be.

Quantus
The above code works for me, but I have to drop the item and pick it back up for the new verbs to show up. How do I get it to change the verbs list while it's in my inventory without dropping it?

The Pixie
Quantus, ivestigating further, it is clear that there is an issue here. I have updated the OP with a solution, which essentially is to give the object a new list, rather than modifying the existing one.

Deckrect
Ok. But does anything changes if I plan using it only on objects in room which cannot be taken and case by case?

The Pixie
You are better off using it on each object individually, and you can use it on any object.

This topic is now closed. Topics are closed after 14 days of inactivity.

Support

Forums