A dictionary is a data set where each entry is accessed by a string, called a key. If you think about an actual dictionary, it contains words linked to definitions. To find a specific definition, you use the word to look it up. Quest dictionaries are the same, you use a word, called a key, to look up your information, called a value.
A dictionary can never have two entries with the same key, just as a list can never have two entries with the same position. However, unlike a list, there is no order to the entries in a dictionary (in theory anyway).
As with lists, there are various types, but with dictionaries you also get a version specifically for scripts. Note that whatever the type, the key must be a string.
mystringdict = NewStringDictionary()
myobjectdict = NewObjectDictionary()
myscriptdict = NewScriptDictionary()
mydict = NewDictionary()
As with lists, you are better using a specific dictionary type, rather than a general one, if possible.
If using the off-line editor, you can add a dictionary to an object on the attributes page, but you are restricted to string and script dictionaries.
To add items to a dictionary, use the dictionary add
command. To remove something from a dictionary, use dictionary remove
. Unlike the list versions of these commands, we now need to provide a key.
d = NewObjectDictionary()
dictionary add (d, "table", fancy table)
dictionary add (d, "table2", fancy table)
dictionary add (d, "pov", player)
dictionary remove (d, "table")
As with lists, you can add the same thing as many times as you like, however, the key must be unique. If you try to add an item with a key that is already in the dictionary you will get an error. If you try to remove a key that does not exist you will again get an error.
As of Quest 5.8, the functions DictionaryAdd
and DictionaryRemove
offer a safe way to add and remove keys. DictionaryAdd
will replace the existing key, if it is present. DictionaryRemove
will just do nothing if the key does not exist.
You can access an item in a dictionary using the DictionaryItem
function. This takes the dictionary and the key as parameters. If you are using a specific dictionary type, it is again better to use the function for that type:
StringDictionaryItem
ScriptDictionaryItem
ObjectDictionaryItem
Note that Quest will throw an error if the key is not found in the dictionary, so testing before hand is usually advised.
Often you will want to go through each member of a dictionary, and as with a list we can use the foreach
command to do this. It takes two parameters, the first being a variable to store a key in, and the second being the dictionary. It also requires a script.
This example will output each member of the dictionary. The script will be run once for each entry in the dictionary d
, and when it runs key
will have the key for that entry.
foreach(key, d) {
msg("Entry: " + key + "=" + DictionaryItem(d, key))
}
Changing a dictionary whilst in a foreach loop (i.e., adding or removing entries) will cause an error.
Dictionaries are fussy things that will throw an error if you try to add a key that is already there, if you try to delete a key that is not, or try to retrieve a key that is not (unlike lists). The DictionaryContains
function, then, is extremely useful as it will tell you if the dictionary already contains the given key (there is no function that will tell you if the entry is already in the dictionary).
To check a key is not already in use before adding to the dictionary:
if (not DictionaryContains(dict, obj.name)) {
dictionary add(dict, obj.name, obj)
}
To check a key IS already in use before removing from the dictionary:
if (DictionaryContains(dict, obj.name)) {
dictionary remove(dict, obj.name, obj)
}
Alternatively we can use the in
operator, which is rather less typing!
if (not obj.name in dict) {
dictionary add(dict, obj.name, obj)
}
if (obj.name in dict) {
dictionary remove(dict, obj.name, obj)
}
By the way, the reason there is a DictionaryContains
function is to make it easier to use dictionaries via the GUI interface.
The DictionaryCount
function will return the number of entries in the dictionary.
msg("My dictionary has " + DictionaryCount(myDict) + " things in it.")
You can use a string dictionary with the ShowMenu
function or show menu
command. This allows you to give a set of options, and to handle them as a corresponding set of strings. This could, for example, allow you to run a script, depending on the option chosen. Suppose “getthing”, “jump” and “vomit” are all scripts on the player object:
options = NewStringDictionary()
dictionary add (options, "getthing", "Get the thing")
dictionary add (options, "jump", "Jump as high as you can")
dictionary add (options, "vomit", "Try to vomit the poison up")
ShowMenu ("Choose", options, false) {
do(player, result)
}
You can also use dictionaries to pass values to scripts. The key will become the name of a local variable, while the value will be its value.
vars = NewDictionary()
dictionary add (vars, "weapon", weapon)
dictionary add (vars, "success", "You hit the monster")
dictionary add (vars, "failure", "You missed!")
do (monster, "attack", vars)
In the attack script, you can now use three local variables; weapon
, which is an object, and success
and failure
, two strings.