I've noticed that while getObj(), getProp() and setProp() are supported by the Dialogue System (and I'm really thankful they are), they don't map as cleanly to DS Lua functions once you involve object references.
Context
I'll briefly explain what I was trying to accomplish to give some context.
Basically, the player character has a set of personality traits which increases/decreases with certain dialogue choices and can affect the future dialogue option (unlock, lock, add some flavour). Changing the numeric value is simple (I wrote a custom sequence for that).
data:image/s3,"s3://crabby-images/efdc1/efdc15458c1468174b2be05ad3f83907eb034793" alt="Image"
The problem is preventing the player from increasing the traits repeatedly if the conversation can be started again (e.g. it's a part of a dialogue with a party member, or it's echoed back to you in a dream). Giving XP for talking with NPCs only once is a similar thing.
SimStatus seems like a good approach, but since I need it only for a small fraction of the whole dialogue database, I took the documentation's advice and set up a small, local version fulfilling the same purpose.
data:image/s3,"s3://crabby-images/1f58a/1f58a7c4d07bb7d65bb8aa1f169b863723c5d4de" alt="Image"
The choice hub has an initially empty property which is filled with the option you pick. And once you make your choice, the other options will be blocked (and the AddTrait sequence will just pass through with no effect).
I wrote a plugin script with articy macro devkit which automatically generates the appropriate condition/instruction when you link an option to the choice hub.
Ideally, I'd like to end up with something which works in articy (so I can simulate a dialogue via a Journey, and the syntax checker doesn't complain when it stumbles upon non-expresso code) and can be imported in a Dialogue Database without edits (because subsequent reimports would overwrite all changes).
Basically, the player character has a set of personality traits which increases/decreases with certain dialogue choices and can affect the future dialogue option (unlock, lock, add some flavour). Changing the numeric value is simple (I wrote a custom sequence for that).
data:image/s3,"s3://crabby-images/efdc1/efdc15458c1468174b2be05ad3f83907eb034793" alt="Image"
The problem is preventing the player from increasing the traits repeatedly if the conversation can be started again (e.g. it's a part of a dialogue with a party member, or it's echoed back to you in a dream). Giving XP for talking with NPCs only once is a similar thing.
SimStatus seems like a good approach, but since I need it only for a small fraction of the whole dialogue database, I took the documentation's advice and set up a small, local version fulfilling the same purpose.
data:image/s3,"s3://crabby-images/1f58a/1f58a7c4d07bb7d65bb8aa1f169b863723c5d4de" alt="Image"
The choice hub has an initially empty property which is filled with the option you pick. And once you make your choice, the other options will be blocked (and the AddTrait sequence will just pass through with no effect).
I wrote a plugin script with articy macro devkit which automatically generates the appropriate condition/instruction when you link an option to the choice hub.
Ideally, I'd like to end up with something which works in articy (so I can simulate a dialogue via a Journey, and the syntax checker doesn't complain when it stumbles upon non-expresso code) and can be imported in a Dialogue Database without edits (because subsequent reimports would overwrite all changes).
In articy's expresso, it looks like this (the ID belongs to the hub):
Code: Select all
getProp(getObj("0x01000000000636EC"), "Selected.Fragment") == "0_0"
||
getProp(getObj("0x01000000000636EC"), "Selected.Fragment") == self
The second part allows you to pick the option if it's referenced by the hub.
The output pin of the fragment adds itself to the choice hub's property:
Code: Select all
setProp(getObj("0x01000000000636EC"), "Selected.Fragment", self)
getObj for Flow
Self vs Speaker
In articy:draft, speaker refers to the person currently talking (so does speaker in DS), but self refers to the current dialogue fragment – yet DS points it to the speaker as well.
data:image/s3,"s3://crabby-images/3f4e8/3f4e8e7884a4e22dd1671b391338a3b9f0d6c2b4" alt="Image"
So in my scenario, it originally went to look up something like Actor["Player"].SimStatus instead of the current dialogue's entry sim status.
getObj
It's not difficult to find the current dialogue entry in C# – but I didn't find a built-in way to do it in Lua, neither did I find a way to get to the dialogue entry's fields. I thought there would be something like "Dialog[thisID].{field}" table, but that seems to be used for SimStatus alone (when the option is enabled). Unless I missed something obvious, of course.
Perhaps that's why getObj() also fails when you're looking for flow elements:
data:image/s3,"s3://crabby-images/7e12d/7e12de3cd4b40917939e46feaeb2b10496b37fde" alt="Image"
Again, not hard to add a custom line which finds it, but accessing the fields in Lua is the problem.
data:image/s3,"s3://crabby-images/ef283/ef2832760a6d56afb01f54214332d75223d5dfd6" alt="Image"
String ID
As a side note, here is a use case so minimal that I'm quite confident no one in the universe will ever encounter it again.
Anyway…
getObj() in articy:draft seems to parse the ID back to number and then do a comparison. Which means that as far as articy is concerned "0x01000000000636EC" (HEX, copy-pasted from object properties) and "72057594038335212" (DEC, auto-inserted via articy devkit plugin) point to the same object.
Dialogue System, on the other hand, performs a string equality check, so the format must match exactly.
That's just to point out another difference, once I spotted the problem, it wasn't too hard to fix it.
In articy:draft, speaker refers to the person currently talking (so does speaker in DS), but self refers to the current dialogue fragment – yet DS points it to the speaker as well.
data:image/s3,"s3://crabby-images/3f4e8/3f4e8e7884a4e22dd1671b391338a3b9f0d6c2b4" alt="Image"
So in my scenario, it originally went to look up something like Actor["Player"].SimStatus instead of the current dialogue's entry sim status.
getObj
It's not difficult to find the current dialogue entry in C# – but I didn't find a built-in way to do it in Lua, neither did I find a way to get to the dialogue entry's fields. I thought there would be something like "Dialog[thisID].{field}" table, but that seems to be used for SimStatus alone (when the option is enabled). Unless I missed something obvious, of course.
Perhaps that's why getObj() also fails when you're looking for flow elements:
data:image/s3,"s3://crabby-images/7e12d/7e12de3cd4b40917939e46feaeb2b10496b37fde" alt="Image"
Again, not hard to add a custom line which finds it, but accessing the fields in Lua is the problem.
data:image/s3,"s3://crabby-images/ef283/ef2832760a6d56afb01f54214332d75223d5dfd6" alt="Image"
String ID
As a side note, here is a use case so minimal that I'm quite confident no one in the universe will ever encounter it again.
data:image/s3,"s3://crabby-images/e942b/e942b1cac328defdf8026ab3d6aed7ad99b4e1d2" alt="Very Happy :D"
Anyway…
data:image/s3,"s3://crabby-images/7e45e/7e45e28ee44888d65578361dfd49d7ad2275d65d" alt="Razz :P"
getObj() in articy:draft seems to parse the ID back to number and then do a comparison. Which means that as far as articy is concerned "0x01000000000636EC" (HEX, copy-pasted from object properties) and "72057594038335212" (DEC, auto-inserted via articy devkit plugin) point to the same object.
Dialogue System, on the other hand, performs a string equality check, so the format must match exactly.
That's just to point out another difference, once I spotted the problem, it wasn't too hard to fix it.
data:image/s3,"s3://crabby-images/53344/53344662b4abda948d0c1a29200f2c4c59fdcda0" alt="Image"
getProp
getProp has one other issue as well. Articy makes you look up the properties as getProp(obj, "Feature.Property") – but Dialogue System flattens this to a single table which means the code won't work anymore.
data:image/s3,"s3://crabby-images/808b1/808b1535429940d0a2f7ac6daa68aafcca205267" alt="Image"
Perhaps the property name ought to be split and only the last part after the final dot used.
data:image/s3,"s3://crabby-images/808b1/808b1535429940d0a2f7ac6daa68aafcca205267" alt="Image"
Perhaps the property name ought to be split and only the last part after the final dot used.
ObjectReferences after import
Eventually, I found a bit ugly way to extend the getObj and getProp method to get around the above issues and to the dialogue entry fields.
Unfortunately, object references in DS appear like this:
data:image/s3,"s3://crabby-images/38d69/38d69ca7f06939098b050ce92a45910c912f7363" alt="Image"
which is great if you need a human-readable title, but bad if you have a condition comparing this to self, which didn't originally exist either but you made it reference DS's dialogue entry ID; not the Articy Id, so I'd have to find a way to convert another dialogue entry's title to its Articy Id (otherwise there's a Lua exception when it tries to evaluate this: return Dirk: "[-1 Honest]: A lie.) or find other objects using the title and pray that there's no conflict, while keeping in mind that DS is probably not going to understand "0_0" as null (or 'nil') and getProp(obj, "Id") will also need to change to getProp(obj, "Articy Id") after import –
Unfortunately, object references in DS appear like this:
data:image/s3,"s3://crabby-images/38d69/38d69ca7f06939098b050ce92a45910c912f7363" alt="Image"
which is great if you need a human-readable title, but bad if you have a condition comparing this to self, which didn't originally exist either but you made it reference DS's dialogue entry ID; not the Articy Id, so I'd have to find a way to convert another dialogue entry's title to its Articy Id (otherwise there's a Lua exception when it tries to evaluate this: return Dirk: "[-1 Honest]: A lie.) or find other objects using the title and pray that there's no conflict, while keeping in mind that DS is probably not going to understand "0_0" as null (or 'nil') and getProp(obj, "Id") will also need to change to getProp(obj, "Articy Id") after import –
data:image/s3,"s3://crabby-images/e942b/e942b1cac328defdf8026ab3d6aed7ad99b4e1d2" alt="Very Happy :D"
I opted for another solution for my initial problem which seemed like too much micromanagement at first, but with each new extension method I had to add, it became increasingly viable.
data:image/s3,"s3://crabby-images/27d13/27d13c984aa1382b1dbd4953c4c4eaafee84acd1" alt="Wink ;)"
Context conclusion
I figured out the best way to do the permanent, one-time-only choice would most likely be to create a new variable set, e.g. "Choice" and then create an integer global variable marking the index of a chosen option.
For example Choice.TutorialConversation – starting at 0, allowing every option, then setting it to e.g. 3 (index of the chosen option) and blocking everything once the variable's value is greater than 0.
For example Choice.TutorialConversation – starting at 0, allowing every option, then setting it to e.g. 3 (index of the chosen option) and blocking everything once the variable's value is greater than 0.
Please consider looking into some of these problems where it's viable – if not, it'd be great if they could at least get a mention in their relevant pages (articy, Lua).
I'm mostly asking to know where the limits of these functions are in the Dialogue System and I bet other people in the future would be glad to know as well.
I think at least the getProp's "Feature.Variable" deserves a fix, because otherwise you can only retrieve the basic object's variables like "Id" or "MenuText", not your custom properties in templates, which kinda defeats the purpose.
Also, did I overlook a simple way to access the dialogue entry's fields in Lua, or is there a reason why it's not there?
Thank you very much.
data:image/s3,"s3://crabby-images/27d13/27d13c984aa1382b1dbd4953c4c4eaafee84acd1" alt="Wink ;)"