forked from CleverRaven/Cataclysm-DDA
-
Notifications
You must be signed in to change notification settings - Fork 0
/
dialogue.h
577 lines (522 loc) · 22.6 KB
/
dialogue.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
#pragma once
#ifndef CATA_SRC_DIALOGUE_H
#define CATA_SRC_DIALOGUE_H
#include <functional>
#include <iosfwd>
#include <memory>
#include <set>
#include <string>
#include <type_traits>
#include <utility>
#include <vector>
#include "dialogue_win.h"
#include "global_vars.h"
#include "npc.h"
#include "talker.h"
#include "translations.h"
#include "type_id.h"
class JsonArray;
class JsonObject;
class martialart;
class mission;
struct dialogue;
struct input_event;
enum talk_trial_type : unsigned char {
TALK_TRIAL_NONE, // No challenge here!
TALK_TRIAL_LIE, // Straight up lying
TALK_TRIAL_PERSUADE, // Convince them
TALK_TRIAL_INTIMIDATE, // Physical intimidation
TALK_TRIAL_CONDITION, // Some other condition
NUM_TALK_TRIALS
};
enum dialogue_consequence : unsigned char {
none = 0,
hostile,
helpless,
action
};
using talkfunction_ptr = std::add_pointer<void ( npc & )>::type;
using dialogue_fun_ptr = std::add_pointer<void( npc & )>::type;
using trial_mod = std::pair<std::string, int>;
/**
* If not TALK_TRIAL_NONE, it defines how to decide whether the responses succeeds (e.g. the
* NPC believes the lie). The difficulty is a 0...100 percent chance of success (!), 100 means
* always success, 0 means never. It is however affected by mutations/traits/bionics/etc. of
* the player character.
*/
struct talk_trial {
talk_trial_type type = TALK_TRIAL_NONE;
int difficulty = 0;
std::function<bool( const dialogue & )> condition;
int calc_chance( const dialogue &d ) const;
/**
* Returns a user-friendly representation of @ref type
*/
std::string name() const;
std::vector<trial_mod> modifiers;
explicit operator bool() const {
return type != TALK_TRIAL_NONE;
}
/**
* Roll for success or failure of this trial.
*/
bool roll( dialogue &d ) const;
talk_trial() = default;
explicit talk_trial( const JsonObject & );
};
struct talk_topic {
explicit talk_topic( const std::string &i ) : id( i ) { }
std::string id;
/** If we're talking about an item, this should be its type. */
itype_id item_type = itype_id::NULL_ID();
/** Reason for denying a request. */
std::string reason;
};
struct talk_effect_fun_t {
private:
std::function<void( const dialogue &d )> function;
std::vector<std::pair<int, itype_id>> likely_rewards;
public:
talk_effect_fun_t() = default;
explicit talk_effect_fun_t( const talkfunction_ptr & );
explicit talk_effect_fun_t( const std::function<void( npc & )> & );
explicit talk_effect_fun_t( const std::function<void( const dialogue &d )> & );
void set_companion_mission( const std::string &role_id );
void set_add_effect( const JsonObject &jo, const std::string &member, bool is_npc = false );
void set_remove_effect( const JsonObject &jo, const std::string &member, bool is_npc = false );
void set_add_trait( const JsonObject &jo, const std::string &member, bool is_npc = false );
void set_remove_trait( const JsonObject &jo, const std::string &member, bool is_npc = false );
void set_add_bionic( const JsonObject &jo, const std::string &member, bool is_npc = false );
void set_lose_bionic( const JsonObject &jo, const std::string &member, bool is_npc = false );
void set_message( const JsonObject &jo, const std::string &member, bool is_npc = false );
void set_add_wet( const JsonObject &jo, const std::string &member, bool is_npc );
void set_assign_activity( const JsonObject &jo, const std::string &member, bool is_npc = false );
void set_assign_mission( const JsonObject &jo, const std::string &member );
void set_finish_mission( const JsonObject &jo, const std::string &member );
void set_make_sound( const JsonObject &jo, const std::string &member, bool is_npc );
void set_queue_effect_on_condition( const JsonObject &jo, const std::string &member,
bool is_npc = false );
void set_weighted_list_eocs( const JsonObject &jo, const std::string &member );
void set_mod_healthy( const JsonObject &jo, const std::string &member, bool is_npc );
void set_cast_spell( const JsonObject &jo, const std::string &member, bool is_npc );
void set_lightning();
void set_next_weather();
void set_sound_effect( const JsonObject &jo, const std::string &member );
void set_mod_fatigue( const JsonObject &jo, const std::string &member, bool is_npc = false );
void set_add_var( const JsonObject &jo, const std::string &member, bool is_npc = false );
void set_remove_var( const JsonObject &jo, const std::string &member, bool is_npc = false );
void set_adjust_var( const JsonObject &jo, const std::string &member, bool is_npc = false );
void set_u_spawn_item( const itype_id &item_name, int count, const std::string &container_name );
void set_u_buy_item( const itype_id &item_name, int cost, int count,
const std::string &container_name );
void set_u_spend_cash( int amount );
void set_u_sell_item( const itype_id &item_name, int cost, int count );
void set_consume_item( const JsonObject &jo, const std::string &member, int count,
bool is_npc = false );
void set_remove_item_with( const JsonObject &jo, const std::string &member, bool is_npc = false );
void set_npc_change_faction( const std::string &faction_name );
void set_npc_change_class( const std::string &class_name );
void set_change_faction_rep( int rep_change );
void set_add_debt( const std::vector<trial_mod> &debt_modifiers );
void set_toggle_npc_rule( const std::string &rule );
void set_set_npc_rule( const std::string &rule );
void set_clear_npc_rule( const std::string &rule );
void set_npc_engagement_rule( const std::string &setting );
void set_npc_aim_rule( const std::string &setting );
void set_npc_cbm_reserve_rule( const std::string &setting );
void set_npc_cbm_recharge_rule( const std::string &setting );
void set_location_variable( const JsonObject &jo, const std::string &member, bool is_npc );
void set_transform_radius( const JsonObject &jo, const std::string &member, bool is_npc );
void set_mapgen_update( const JsonObject &jo, const std::string &member );
void set_npc_goal( const JsonObject &jo, const std::string &member );
void set_bulk_trade_accept( bool is_trade, int quantity, bool is_npc = false );
void set_npc_gets_item( bool to_use );
void set_add_mission( const std::string &mission_id );
const std::vector<std::pair<int, itype_id>> &get_likely_rewards() const;
void set_u_buy_monster( const std::string &monster_type_id, int cost, int count, bool pacified,
const translation &name );
void set_u_learn_recipe( const std::string &learned_recipe_id );
void set_npc_first_topic( const std::string &chat_topic );
void set_add_morale( const JsonObject &jo, const std::string &member, bool is_npc );
void set_lose_morale( const JsonObject &jo, const std::string &member, bool is_npc );
void set_add_faction_trust( const JsonObject &jo, const std::string &member );
void set_lose_faction_trust( const JsonObject &jo, const std::string &member );
void set_arithmetic( const JsonObject &jo, const std::string &member );
void set_custom_light_level( const JsonObject &jo, const std::string &member );
void set_spawn_monster( const JsonObject &jo, const std::string &member, bool is_npc );
void set_field( const JsonObject &jo, const std::string &member, bool is_npc );
void set_teleport( const JsonObject &jo, const std::string &member, bool is_npc );
void set_open_dialogue();
void set_take_control();
void set_take_control_menu();
void operator()( const dialogue &d ) const {
if( !function ) {
return;
}
return function( d );
}
};
/**
* Defines what happens when the trial succeeds or fails. If trial is
* TALK_TRIAL_NONE it always succeeds.
*/
struct talk_effect_t {
/**
* How (if at all) the NPCs opinion of the player character (@ref npc::op_of_u)
* will change.
*/
npc_opinion opinion;
/**
* How (if at all) the NPCs opinion of the player character (@ref npc::op_of_u)
* will change. These values are divisors of the mission value.
*/
npc_opinion mission_opinion;
/**
* Topic to switch to. TALK_DONE ends the talking, TALK_NONE keeps the current topic.
*/
talk_topic next_topic = talk_topic( "TALK_NONE" );
talk_topic apply( dialogue &d ) const;
dialogue_consequence get_consequence( const dialogue &d ) const;
/**
* Sets an effect and consequence based on function pointer.
*/
void set_effect( talkfunction_ptr );
void set_effect( const talk_effect_fun_t & );
/**
* Sets an effect to a function object and consequence to explicitly given one.
*/
void set_effect_consequence( const talk_effect_fun_t &fun, dialogue_consequence con );
void set_effect_consequence( const std::function<void( npc &p )> &ptr, dialogue_consequence con );
void load_effect( const JsonObject &jo, const std::string &member_name );
void parse_sub_effect( const JsonObject &jo );
void parse_string_effect( const std::string &effect_id, const JsonObject &jo );
talk_effect_t() = default;
explicit talk_effect_t( const JsonObject &, const std::string & );
/**
* Functions that are called when the response is chosen.
*/
std::vector<talk_effect_fun_t> effects;
private:
dialogue_consequence guaranteed_consequence = dialogue_consequence::none;
};
/**
* This defines possible responses from the player character.
*/
struct talk_response {
/**
* What the player character says (literally). Should already be translated and will be
* displayed.
*/
std::string text;
/*
* Optional responses from a true/false test that defaults to true.
*/
translation truetext;
translation falsetext;
std::function<bool( const dialogue & )> truefalse_condition;
talk_trial trial;
/**
* The following values are forwarded to the chatbin of the NPC (see @ref npc_chatbin).
*/
mission *mission_selected = nullptr;
skill_id skill = skill_id();
matype_id style = matype_id();
spell_id dialogue_spell = spell_id();
proficiency_id proficiency = proficiency_id();
talk_effect_t success;
talk_effect_t failure;
talk_data create_option_line( const dialogue &d, const input_event &hotkey );
std::set<dialogue_consequence> get_consequences( const dialogue &d ) const;
talk_response();
explicit talk_response( const JsonObject & );
};
struct dialogue {
/**
* If true, we are done talking and the dialog ends.
*/
bool done = false;
std::vector<talk_topic> topic_stack;
/** Missions that have been assigned by this npc to the player they currently speak to. */
std::vector<mission *> missions_assigned;
talk_topic opt( dialogue_window &d_win, const talk_topic &topic );
dialogue() = default;
dialogue( std::unique_ptr<talker> alpha_in, std::unique_ptr<talker> beta_in );
talker *actor( const bool is_beta ) const;
mutable itype_id cur_item;
mutable std::string reason;
std::string dynamic_line( const talk_topic &topic ) const;
void apply_speaker_effects( const talk_topic &the_topic );
/** This dialogue is happening over a radio */
bool by_radio = false;
/**
* Possible responses from the player character, filled in @ref gen_responses.
*/
std::vector<talk_response> responses;
void gen_responses( const talk_topic &topic );
void add_topic( const std::string &topic );
void add_topic( const talk_topic &topic );
bool has_beta;
bool has_alpha;
private:
/**
* The talker that speaks (almost certainly representing the avatar, ie get_avatar() )
*/
std::unique_ptr<talker> alpha;
/**
* The talker responded to alpha, usually a talker_npc.
*/
std::unique_ptr<talker> beta;
/**
* Add a simple response that switches the topic to the new one. If first == true, force
* this topic to the front of the responses.
*/
talk_response &add_response( const std::string &text, const std::string &r,
bool first = false );
/**
* Add a response with the result TALK_DONE.
*/
talk_response &add_response_done( const std::string &text );
/**
* Add a response with the result TALK_NONE.
*/
talk_response &add_response_none( const std::string &text );
/**
* Add a simple response that switches the topic to the new one and executes the given
* action. The response always succeeds. Consequence is based on function used.
*/
talk_response &add_response( const std::string &text, const std::string &r,
const dialogue_fun_ptr &effect_success, bool first = false );
/**
* Add a simple response that switches the topic to the new one and executes the given
* action. The response always succeeds. Consequence must be explicitly specified.
*/
talk_response &add_response( const std::string &text, const std::string &r,
const std::function<void( npc & )> &effect_success,
dialogue_consequence consequence, bool first = false );
/**
* Add a simple response that switches the topic to the new one and sets the currently
* talked about mission to the given one. The mission pointer must be valid.
*/
talk_response &add_response( const std::string &text, const std::string &r, mission *miss,
bool first = false );
/**
* Add a simple response that switches the topic to the new one and sets the currently
* talked about skill to the given one.
*/
talk_response &add_response( const std::string &text, const std::string &r, const skill_id &skill,
bool first = false );
/**
* Add a simple response that switches the topic to the new one and sets the currently
* talked about proficiency to the given one.
*/
talk_response &add_response( const std::string &text, const std::string &r,
const proficiency_id &proficiency,
bool first = false );
/**
* Add a simple response that switches the topic to the new one and sets the currently
* talked about magic spell to the given one.
*/
talk_response &add_response( const std::string &text, const std::string &r,
const spell_id &sp, bool first = false );
/**
* Add a simple response that switches the topic to the new one and sets the currently
* talked about martial art style to the given one.
*/
talk_response &add_response( const std::string &text, const std::string &r,
const martialart &style, bool first = false );
/**
* Add a simple response that switches the topic to the new one and sets the currently
* talked about item type to the given one.
*/
talk_response &add_response( const std::string &text, const std::string &r,
const itype_id &item_type, bool first = false );
int get_best_quit_response() const;
};
/**
* A dynamically generated line, spoken by the NPC.
* This struct only adds the constructors which will load the data from json
* into a lambda, stored in the std::function object.
* Invoking the function operator with a dialog reference (so the function can access the NPC)
* returns the actual line.
*/
struct dynamic_line_t {
private:
std::function<std::string( const dialogue & )> function;
public:
dynamic_line_t() = default;
explicit dynamic_line_t( const translation &line );
explicit dynamic_line_t( const JsonObject &jo );
explicit dynamic_line_t( const JsonArray &ja );
static dynamic_line_t from_member( const JsonObject &jo, const std::string &member_name );
std::string operator()( const dialogue &d ) const {
if( !function ) {
return std::string{};
}
return function( d );
}
};
struct var_info {
var_info( var_type in_type, std::string in_name, std::string in_default_val ): type( in_type ),
name( in_name ), default_val( in_default_val ) {}
var_type type;
std::string name;
std::string default_val;
};
static std::string read_var_value( var_type type, std::string name, talker *talk )
{
global_variables &globvars = get_globals();
switch( type ) {
case var_type::global:
return globvars.get_global_value( name );
break;
case var_type::u:
case var_type::npc:
return talk->get_value( name );
break;
case var_type::faction:
debugmsg( "Not implemented yet." );
break;
case var_type::party:
debugmsg( "Not implemented yet." );
break;
default:
debugmsg( "Invalid type." );
break;
}
return "";
}
struct int_or_var {
cata::optional<int> int_val;
cata::optional<std::string> var_val;
cata::optional<int> default_val;
var_type type = var_type::u;
bool is_npc() const {
return type == var_type::npc;
}
int evaluate( talker *talk ) const {
if( int_val.has_value() ) {
return int_val.value();
} else if( var_val.has_value() ) {
std::string val = read_var_value( type, var_val.value(), talk );
if( !val.empty() ) {
return std::stoi( val );
}
return default_val.value();
} else {
debugmsg( "No valid value." );
return 0;
}
}
};
struct duration_or_var {
cata::optional<time_duration> dur_val;
cata::optional<std::string> var_val;
cata::optional<time_duration> default_val;
var_type type = var_type::u;
bool is_npc() const {
return type == var_type::npc;
}
time_duration evaluate( talker *talk ) const {
if( dur_val.has_value() ) {
return dur_val.value();
} else if( var_val.has_value() ) {
std::string val = read_var_value( type, var_val.value(), talk );
if( !val.empty() ) {
time_duration ret_val;
ret_val = time_duration::from_turns( std::stoi( val ) );
return ret_val;
}
return default_val.value();
} else {
debugmsg( "No valid value." );
return 0_seconds;
}
}
};
/**
* An extended response. It contains the response itself and a condition, so we can include the
* response if, and only if the condition is met.
*/
class json_talk_response
{
private:
talk_response actual_response;
std::function<bool( const dialogue & )> condition;
bool has_condition_ = false;
bool is_switch = false;
bool is_default = false;
void load_condition( const JsonObject &jo );
bool test_condition( const dialogue &d ) const;
public:
json_talk_response() = default;
explicit json_talk_response( const JsonObject &jo );
const talk_response &get_actual_response() const;
bool has_condition() const {
return has_condition_;
}
/**
* Callback from @ref json_talk_topic::gen_responses, see there.
*/
bool gen_responses( dialogue &d, bool switch_done ) const;
bool gen_repeat_response( dialogue &d, const itype_id &item_id, bool switch_done ) const;
};
/**
* A structure for generating repeated responses
*/
class json_talk_repeat_response
{
public:
json_talk_repeat_response() = default;
explicit json_talk_repeat_response( const JsonObject &jo );
bool is_npc = false;
bool include_containers = false;
std::vector<itype_id> for_item;
std::vector<item_category_id> for_category;
json_talk_response response;
};
class json_dynamic_line_effect
{
private:
std::function<bool( const dialogue & )> condition;
talk_effect_t effect;
public:
json_dynamic_line_effect( const JsonObject &jo, const std::string &id );
bool test_condition( const dialogue &d ) const;
void apply( dialogue &d ) const;
};
/**
* Talk topic definitions load from json.
*/
class json_talk_topic
{
private:
bool replace_built_in_responses = false;
std::vector<json_talk_response> responses;
dynamic_line_t dynamic_line;
std::vector<json_dynamic_line_effect> speaker_effects;
std::vector<json_talk_repeat_response> repeat_responses;
public:
json_talk_topic() = default;
/**
* Load data from json.
* This will append responses (not change existing ones).
* It will override dynamic_line and replace_built_in_responses if those entries
* exist in the input, otherwise they will not be changed at all.
*/
void load( const JsonObject &jo );
std::string get_dynamic_line( const dialogue &d ) const;
std::vector<json_dynamic_line_effect> get_speaker_effects() const;
void check_consistency() const;
/**
* Callback from @ref dialogue::gen_responses, it should add the response from here
* into the list of possible responses (that will be presented to the player).
* It may add an arbitrary number of responses (including none at all).
* @return true if built in response should excluded (not added). If false, built in
* responses will be added (behind those added here).
*/
bool gen_responses( dialogue &d ) const;
cata::flat_set<std::string> get_directly_reachable_topics( bool only_unconditional ) const;
};
void unload_talk_topics();
void load_talk_topic( const JsonObject &jo );
#endif // CATA_SRC_DIALOGUE_H