… | |
… | |
11 | =cut |
11 | =cut |
12 | |
12 | |
13 | package NPC_Dialogue; |
13 | package NPC_Dialogue; |
14 | |
14 | |
15 | use strict; |
15 | use strict; |
16 | |
|
|
17 | sub has_dialogue($) { |
|
|
18 | my ($ob) = @_; |
|
|
19 | |
|
|
20 | $ob->msg =~ /^\@match /; |
|
|
21 | } |
|
|
22 | |
16 | |
23 | sub parse_message($) { |
17 | sub parse_message($) { |
24 | map [split /\n/, $_, 2], |
18 | map [split /\n/, $_, 2], |
25 | grep length, |
19 | grep length, |
26 | split /^\@match /m, |
20 | split /^\@match /m, |
… | |
… | |
102 | |
96 | |
103 | =item $flag - A hashref that stores flags associated with the player and |
97 | =item $flag - A hashref that stores flags associated with the player and |
104 | can be seen by all NPCs (so better name your flags uniquely). This is |
98 | can be seen by all NPCs (so better name your flags uniquely). This is |
105 | useful for storing e.g. quest information. See C<@setflag> and C<@ifflag>. |
99 | useful for storing e.g. quest information. See C<@setflag> and C<@ifflag>. |
106 | |
100 | |
|
|
101 | =item @find - see @find, below. |
|
|
102 | |
107 | =back |
103 | =back |
108 | |
104 | |
109 | The environment is that standard "map scripting environment", which is |
105 | The environment is that standard "map scripting environment", which is |
110 | limited in the type of constructs allowed (no loops, for example). |
106 | limited in the type of constructs allowed (no loops, for example). |
111 | |
107 | |
… | |
… | |
121 | |
117 | |
122 | You may want to change the C<name> method there to something like C<title>, |
118 | You may want to change the C<name> method there to something like C<title>, |
123 | C<slaying> or any other method that is allowed to be called on a |
119 | C<slaying> or any other method that is allowed to be called on a |
124 | C<cf::object> here. |
120 | C<cf::object> here. |
125 | |
121 | |
|
|
122 | =item B<matching for an item name and removing the matched item> |
|
|
123 | |
|
|
124 | @match found earhorn |
|
|
125 | @cond grep $_->slaying =~ /Gramp's walking stick/, $who->inv |
|
|
126 | @eval my @g = grep { $_->slaying =~ /Gramp's walking stick/ } $who->inv; $g[0]->decrease; |
|
|
127 | Thanks for the earhorn! |
|
|
128 | |
|
|
129 | This example is a bit more complex. The C<@eval> statement will search |
|
|
130 | the players inventory for the same term as the C<@cond> and then |
|
|
131 | decreases the number of objects used there. |
|
|
132 | |
|
|
133 | (See also the map: C<scorn/houses/cornerbrook.map> for an example how this is |
|
|
134 | used in the real world :-) |
|
|
135 | |
126 | =back |
136 | =back |
127 | |
137 | |
128 | =item @eval perl |
138 | =item @eval perl |
129 | |
139 | |
130 | Like C<@cond>, but proceed regardless of the outcome. |
140 | Like C<@cond>, but proceed regardless of the outcome. |
131 | |
141 | |
132 | =item @msg perl |
142 | =item @msg perl |
133 | |
143 | |
134 | Like C<@cond>, but the return value will be stringified and prepended to |
144 | Like C<@cond>, but the return value will be stringified and prepended to |
135 | the message. |
145 | the reply message. |
|
|
146 | |
|
|
147 | =item @check match expression |
|
|
148 | |
|
|
149 | Executes a match expression (see |
|
|
150 | http://pod.tst.eu/http://cvs.schmorp.de/deliantra/server/lib/cf/match.pm) |
|
|
151 | to see if it matches. |
|
|
152 | |
|
|
153 | C<self> is the npc object, C<object>, C<source> and C<originator> are the |
|
|
154 | player communicating with the NPC. |
|
|
155 | |
|
|
156 | If the check fails, the match is skipped. |
|
|
157 | |
|
|
158 | =item @find match expression |
|
|
159 | |
|
|
160 | Like C<@check> in that it executes a match expression, but instead of |
|
|
161 | failing, it gathers all objects matched into the C<@find> array variable. |
|
|
162 | |
|
|
163 | When you want to skip the match when no objects have been found, combine |
|
|
164 | C<@find> with C<@cond>: |
|
|
165 | |
|
|
166 | @match see my spellbook |
|
|
167 | @find type=SPELLBOOK in inv |
|
|
168 | @cond @find |
|
|
169 | It looks dirty. |
|
|
170 | @match see my spellbook |
|
|
171 | I can't see any, where do you have it? |
136 | |
172 | |
137 | =item @setstate state value |
173 | =item @setstate state value |
138 | |
174 | |
139 | Sets the named state C<state> to the given C<value>. State values are |
175 | Sets the named state C<state> to the given C<value>. State values are |
140 | associated with a specific player-NPC pair, so each NPC has its own state |
176 | associated with a specific player-NPC pair, so each NPC has its own state |
… | |
… | |
193 | When the state argument is omitted the trigger is stateful and retains an |
229 | When the state argument is omitted the trigger is stateful and retains an |
194 | internal state per connected-id. There is a limitation to the use of this: The |
230 | internal state per connected-id. There is a limitation to the use of this: The |
195 | state won't be changed when the connection is triggered by other triggers. So |
231 | state won't be changed when the connection is triggered by other triggers. So |
196 | be careful when triggering the connection from other objects. |
232 | be careful when triggering the connection from other objects. |
197 | |
233 | |
198 | When a state argument is given it should be either 0 or 1. 1 will 'push' the connection |
234 | When a state argument is given it should be a positive integer. Any value |
199 | and 0 will 'release' the connection. This is useful for example when you want to |
235 | C<!= 0> will 'push' the connection (in general, you should specify C<1> |
200 | let a npc control a door. |
236 | for this) and C<0> will 'release' the connection. This is useful for |
|
|
237 | example when you want to let an NPC control a door. |
201 | |
238 | |
202 | Trigger all objects with the given connected-id by 'releasing' the connection. |
239 | Trigger all objects with the given connected-id by 'releasing' the connection. |
|
|
240 | |
|
|
241 | =item @playersound face-name |
|
|
242 | |
|
|
243 | Plays the given sound face (either an alias or sound file path) so that |
|
|
244 | only the player talking to the npc can hear it. |
|
|
245 | |
|
|
246 | =item @npcsound face-name |
|
|
247 | |
|
|
248 | Plays the given sound face (either an alias or sound file path) as if |
|
|
249 | the npc had made that sound, i.e. it will be located at the npc and all |
|
|
250 | players near enough can hear it. |
203 | |
251 | |
204 | =item @addtopic topic |
252 | =item @addtopic topic |
205 | |
253 | |
206 | Adds the given topic names (separated by C<|>) to the list of topics |
254 | Adds the given topic names (separated by C<|>) to the list of topics |
207 | returned. |
255 | returned. |
… | |
… | |
225 | my @replies; |
273 | my @replies; |
226 | my @match; # @match/@parse command results |
274 | my @match; # @match/@parse command results |
227 | |
275 | |
228 | my $state = $self->{npc}{$self->{ob}->name}{dialog_state} ||= {}; |
276 | my $state = $self->{npc}{$self->{ob}->name}{dialog_state} ||= {}; |
229 | my $flag = $self->{ob}{dialog_flag} ||= {}; |
277 | my $flag = $self->{ob}{dialog_flag} ||= {}; |
|
|
278 | |
|
|
279 | my @find; |
230 | |
280 | |
231 | my %vars = ( |
281 | my %vars = ( |
232 | who => $self->{ob}, |
282 | who => $self->{ob}, |
233 | npc => $self->{npc}, |
283 | npc => $self->{npc}, |
234 | state => $state, |
284 | state => $state, |
… | |
… | |
249 | or next topic; |
299 | or next topic; |
250 | |
300 | |
251 | } elsif ($cmd eq "comment") { |
301 | } elsif ($cmd eq "comment") { |
252 | # nop |
302 | # nop |
253 | |
303 | |
|
|
304 | } elsif ($cmd eq "playersound") { |
|
|
305 | $self->{ob}->contr->play_sound (cf::sound::find $args); |
|
|
306 | |
|
|
307 | } elsif ($cmd eq "npcsound") { |
|
|
308 | $self->{npc}->play_sound (cf::sound::find $args); |
|
|
309 | |
254 | } elsif ($cmd eq "cond") { |
310 | } elsif ($cmd eq "cond") { |
255 | cf::safe_eval $args, %vars |
311 | cf::safe_eval $args, %vars |
256 | or next topic; |
312 | or next topic; |
257 | |
313 | |
258 | } elsif ($cmd eq "eval") { |
314 | } elsif ($cmd eq "eval") { |
259 | cf::safe_eval $args, %vars; |
315 | cf::safe_eval $args, %vars; |
260 | warn "\@eval evaluation error: $@\n" if $@; |
316 | warn "\@eval evaluation error: $@\n" if $@; |
|
|
317 | |
|
|
318 | } elsif ($cmd eq "check") { |
|
|
319 | eval { |
|
|
320 | cf::match::match $args, $self->{ob}, $self->{npc}, $self->{ob} |
|
|
321 | or next topic; |
|
|
322 | }; |
|
|
323 | warn "\@check evaluation error: $@\n" if $@; |
|
|
324 | |
|
|
325 | } elsif ($cmd eq "find") { |
|
|
326 | @find = eval { |
|
|
327 | cf::match::match $args, $self->{ob}, $self->{npc}, $self->{ob} |
|
|
328 | }; |
|
|
329 | warn "\@find evaluation error: $@\n" if $@; |
261 | |
330 | |
262 | } elsif ($cmd eq "msg") { |
331 | } elsif ($cmd eq "msg") { |
263 | push @replies, [$self->{npc}, (scalar cf::safe_eval $args, %vars)]; |
332 | push @replies, [$self->{npc}, (scalar cf::safe_eval $args, %vars)]; |
264 | |
333 | |
265 | } elsif ($cmd eq "setflag") { |
334 | } elsif ($cmd eq "setflag") { |
… | |
… | |
282 | $state->{$name} eq $value |
351 | $state->{$name} eq $value |
283 | or next topic; |
352 | or next topic; |
284 | |
353 | |
285 | } elsif ($cmd eq "trigger") { |
354 | } elsif ($cmd eq "trigger") { |
286 | my ($con, $state) = split /\s+/, $args, 2; |
355 | my ($con, $state) = split /\s+/, $args, 2; |
287 | $con = $con * 1; |
|
|
288 | |
356 | |
289 | if (defined $state) { |
357 | if (defined $state) { |
290 | $self->{npc}->map->trigger ($args, $state); |
358 | $self->{npc}->map->trigger ($con, $state, $self->{npc}, $self->{ob}); |
291 | } else { |
359 | } else { |
292 | my $rvalue = \$self->{npc}{dialog_trigger}{$con}; |
360 | my $rvalue = \$self->{npc}{dialog_trigger}{$con+0}; |
293 | $self->{npc}->map->trigger ($con, $$rvalue = !$$rvalue); |
361 | $self->{npc}->map->trigger ($con, $$rvalue = !$$rvalue, $self->{npc}, $self->{ob}); |
294 | } |
362 | } |
295 | |
363 | |
296 | } elsif ($cmd eq "addtopic") { |
364 | } elsif ($cmd eq "addtopic") { |
297 | push @kw, split /\|/, $args; |
365 | push @kw, split /\|/, $args; |
298 | $self->{add_topic}->(split /\s*\|\s*/, $args) if $self->{add_topic}; |
366 | $self->{add_topic}->(split /\s*\|\s*/, $args) if $self->{add_topic}; |
… | |
… | |
306 | } |
374 | } |
307 | } |
375 | } |
308 | |
376 | |
309 | delete $self->{npc}{$self->{ob}->name}{dialog_state} unless %$state; |
377 | delete $self->{npc}{$self->{ob}->name}{dialog_state} unless %$state; |
310 | delete $self->{ob}{dialog_flag} unless %$flag; |
378 | delete $self->{ob}{dialog_flag} unless %$flag; |
311 | |
|
|
312 | # combine lines into paragraphs |
|
|
313 | $reply =~ s/(?<=\S)\n(?=\w)/ /g; |
|
|
314 | $reply =~ s/\n\n/\n/g; |
|
|
315 | |
379 | |
316 | # ignores flags and npc from replies |
380 | # ignores flags and npc from replies |
317 | $reply = join "\n", (map $_->[1], @replies), $reply; |
381 | $reply = join "\n", (map $_->[1], @replies), $reply; |
318 | |
382 | |
319 | # now mark up all matching keywords |
383 | # now mark up all matching keywords |
… | |
… | |
324 | last; |
388 | last; |
325 | } |
389 | } |
326 | } |
390 | } |
327 | } |
391 | } |
328 | |
392 | |
|
|
393 | $self->{npc}->use_trigger ($self->{ob}) |
|
|
394 | if $self->{npc}->type == cf::MAGIC_EAR; |
|
|
395 | |
329 | return wantarray ? ($reply, @kw) : $reply; |
396 | return wantarray ? ($reply, @kw) : $reply; |
330 | } |
397 | } |
331 | } |
398 | } |
332 | } |
399 | } |
333 | |
400 | |