-
Notifications
You must be signed in to change notification settings - Fork 721
/
Copy pathaction.hpp
1242 lines (879 loc) · 38.2 KB
/
action.hpp
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
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
// ==========================================================================
// Dedmonwakeen's Raid DPS/TPS Simulator.
// Send questions to [email protected]
// ==========================================================================
#pragma once
#include "config.hpp"
#include "dbc/data_definitions.hh"
#include "player/target_specific.hpp"
#include "sc_enums.hpp"
#include "sim/cooldown_waste_data.hpp"
#include "util/format.hpp"
#include "util/generic.hpp"
#include "util/rng.hpp"
#include "util/string_view.hpp"
#include "util/timespan.hpp"
#include <array>
#include <memory>
#include <string>
#include <vector>
struct action_priority_t;
struct action_priority_list_t;
struct action_state_t;
struct spell_data_t;
struct cooldown_t;
class conduit_data_t;
struct dot_t;
struct buff_t;
struct event_t;
struct expr_t;
struct gain_t;
struct item_t;
struct option_t;
struct sim_t;
struct player_t;
struct proc_t;
struct spelleffect_data_t;
struct stats_t;
struct travel_event_t;
struct weapon_t;
namespace io {
class ofstream;
}
namespace report {
using sc_html_stream = io::ofstream;
}
// Action ===================================================================
// Container to hold base value and all permanent/passive modifiers for values that are automatically parsed from spell
// data. We need to hold these separate because flat modifiers, including those from dynamic sources, are calculated
// before percent multipliers. The final value should be properly reconstituted in the relevant method. Also includes
// operator overloads for ease of use/compatibility with operations to type previously used to store the value.
template <typename T>
struct parsed_value_t
{
T base;
T flat_add;
double pct_mul;
parsed_value_t( T value = T() ) : base( value ), flat_add(), pct_mul( 1.0 ) {}
T value() const
{
if constexpr( std::is_same_v<T, timespan_t> )
{
// timespan_t truncates to the nearest second, but in-game testing suggests aura periods & durations are rounded instead.
// if this holds for all time values, timespan_t should be adjusted instead.
return timespan_t::from_native( std::round( timespan_t::to_native( base + flat_add ) * pct_mul ) );
}
else
{
return ( base + flat_add ) * pct_mul;
}
}
operator T() const
{ return value(); }
parsed_value_t& operator=( T v )
{ base = v; flat_add = T(); pct_mul = 1.0; return *this; }
parsed_value_t& operator+=( T v )
{ flat_add += v; return *this; }
parsed_value_t& operator-=( T v )
{ flat_add -= v; return *this; }
template <typename U>
parsed_value_t& operator*=( U v )
{ pct_mul *= v; return *this; }
template <typename U>
parsed_value_t& operator/=( U v )
{ pct_mul /= v; return *this; }
friend void sc_format_to( const parsed_value_t<T>& v, fmt::format_context::iterator out )
{ fmt::format_to( out, "{}", v.value() ); }
// additional operator overrides for timespan_t, as it is used quite often. these are unnecessary for POD
template <typename U = T, typename = std::enable_if_t<std::is_same_v<U, timespan_t>>>
bool operator==( const timespan_t& t ) const
{ return value() == t; }
template <typename U = T, typename = std::enable_if_t<std::is_same_v<U, timespan_t>>>
bool operator<( const timespan_t& t ) const
{ return value() < t; }
template <typename U = T, typename = std::enable_if_t<std::is_same_v<U, timespan_t>>>
bool operator>( const timespan_t& t ) const
{ return value() > t; }
template <typename U = T, typename = std::enable_if_t<std::is_same_v<U, timespan_t>>>
bool operator<=( const timespan_t& t ) const
{ return value() <= t; }
template <typename U = T, typename = std::enable_if_t<std::is_same_v<U, timespan_t>>>
bool operator>=( const timespan_t& t ) const
{ return value() >= t; }
template <typename U = T, typename = std::enable_if_t<std::is_same_v<U, timespan_t>>>
double total_seconds() const
{ return value().total_seconds(); }
template <typename U = T, typename = std::enable_if_t<std::is_same_v<U, timespan_t>>>
time_t total_millis() const
{ return value().total_millis(); }
};
struct action_t : private noncopyable
{
public:
const spell_data_t* s_data;
const spell_data_t* s_data_reporting;
sim_t* sim;
const action_e type;
std::string name_str;
std::string name_str_reporting;
player_t* const player;
player_t* target;
/// Item back-pointer for trinket-sourced actions so we can show proper tooltips in reports
const item_t* item;
/// Weapon used for this ability. If set extra weapon damage is calculated.
weapon_t* weapon;
/**
* Default target is needed, otherwise there's a chance that cycle_targets
* option will _MAJORLY_ mess up the action list for the actor, as there's no
* guarantee cycle_targets will end up on the "initial target" when an
* iteration ends.
*/
player_t* default_target;
/// What type of damage this spell does.
school_e school;
/// What base school components this spell has
std::vector<school_e> base_schools;
/// Spell school prior to set_school_override()
school_e original_school;
/// Spell id if available, 0 otherwise
unsigned id;
/**
* @brief player & action-name unique id.
*
* Every action -- even actions without spelldata -- is given an internal_id
*/
int internal_id;
/// What resource does this ability use.
resource_e resource_current;
/// The amount of targets that an ability impacts on. -1 will hit all targets.
int aoe;
/// If set to true, this action will not be counted toward total amount of executes in reporting. Useful for abilities with parent/children attacks.
bool dual;
/// enables/disables proc callback system on the action, like trinkets, enchants, rppm. this supercedes all dbc-parsed flags.
/// * damage done callbacks require both ( callbacks == true && caster_callbacks == true )
/// * damage taken callbacks require both ( callbacks == true && target_callbacks == true )
bool callbacks, caster_callbacks, target_callbacks;
/// if true, does not trigger callbacks on caster/target.
bool suppress_caster_procs, suppress_target_procs;
/// can trigger callbacks on caster even if suppress_caster_proc is true, as long as the callback has can_proc_from_suppressed = true.
/// TODO: determine if equivalent for suppressed target procs is needed.
bool enable_proc_from_suppressed;
/// Allows triggering of procs marked to only proc from class abilities.
bool allow_class_ability_procs;
/// Specifies that a spell is not a proc and can be considered for triggering only proc from class abilities procs even if it is a background ability.
bool not_a_proc;
/// Whether or not the spell uses the yellow attack hit table.
bool special;
/// Tells the sim to not perform any other actions, as the ability is channeled.
bool channeled;
/// mark this as a sequence_t action
bool sequence;
/**
* @brief Disables/enables reporting of this action.
*
* When set to true, action will not show up in raid report or count towards executes.
*/
bool quiet;
/**
* @brief Enables/Disables direct execution of the ability in an action list.
*
* Background actions cannot be executed via action list, but can be triggered by other actions.
* Background actions do not count for executes.
* Abilities with background = true can still be called upon by other actions,
* example being deep wounds and how it is activated by devastate.
*/
bool background;
/**
* @brief Check if action is executable between GCD's
*
* When set to true, will check every 100 ms to see if this action needs to be used,
* rather than waiting until the next gcd.
* Slows simulation down significantly.
*/
bool use_off_gcd;
/// True if ability should be used while casting another spell.
bool use_while_casting;
/// True if ability is usable while casting another spell
bool usable_while_casting;
/// False if channeled action does not reschedule autoattacks, used on abilities such as bladestorm.
bool interrupt_auto_attack;
/// True if channeled action fully resets the autoattack timer rather than simply delaying it
bool reset_auto_attack;
/// Used for actions that will do awful things to the sim when a "false positive" skill roll happens.
bool ignore_false_positive;
/// Skill is now done per ability, with the default being set to the player option.
double action_skill;
/// Used with DoT Drivers, tells simc that the direct hit is actually a tick.
bool direct_tick;
/// Used with direct damage effects that trigger periodic proc flags
bool treat_as_periodic;
/// Used with psudo-DoT effects, tells us to ignore armor even if the physical damage is direct
bool ignores_armor;
/// Used for abilities that repeat themselves without user interaction, only used on autoattacks.
bool repeating;
/**
* Simplified: Will the ability pull the boss if used.
* Also determines whether ability can be used precombat without counting towards the 1 harmful spell limit
*/
bool harmful;
/**
* @brief Whether or not this ability is a proc.
*
* Procs do not consume resources.
*/
bool proc;
/// Whether or not the action is an interrupt (specifically triggers PF2_CAST_INTERRUPT callbacks)
bool is_interrupt;
/// Whether the action is used from the precombat action list. Will be set up automatically by action_t::init_finished
bool is_precombat;
/// Is the action initialized? (action_t::init ran successfully)
bool initialized;
/// Self explanatory.
bool may_hit, may_miss, may_dodge, may_parry, may_glance, may_block, may_crit, tick_may_crit;
/// Whether or not the ability/dot ticks immediately on usage.
bool tick_zero;
/// Whether or not the ability/dot ticks when it is first applied, but not on refresh applications
bool tick_on_application;
/**
* @brief Whether or not ticks scale with haste.
*
* Generally only used for bleeds that do not scale with haste,
* or with ability drivers that deal damage every x number of seconds.
*/
bool hasted_ticks;
/// Need to consume per tick?
bool consume_per_tick_;
/// Rolling Periodic DoTs will add damage from the previous instance into the new one on refresh
bool rolling_periodic;
/// Split damage evenly between targets
bool split_aoe_damage;
/// Reduce damage to targets when total targets is greater than value
/// Formula used is <damage per target> = sqrt( reduced_aoe_targets / <number of targets> )
double reduced_aoe_targets;
/// If reduced_aoe_targets > 0, the number of target(s) that will take full unreduced amount
int full_amount_targets;
/**
* @brief Normalize weapon speed for weapon damage calculations
*
* \sa weapon_t::get_normalized_speed()
*/
bool normalize_weapon_speed;
/// This ability leaves a ticking dot on the ground, and doesn't move when the target moves. Used with original_x and original_y
bool ground_aoe;
/// Round spell base damage to integer before using
bool round_base_dmg;
/// Used with tick_action, tells tick_action to update state on every tick.
bool dynamic_tick_action;
/// Track time spent with fully charged cooldown before the action is used.
bool track_cd_waste;
cooldown_waste_data_t* cd_waste_data;
/// Did a channel action have an interrupt_immediate used to cancel it on it
bool interrupt_immediate_occurred;
bool hit_any_target;
/// Duration of the ground area trigger
timespan_t ground_aoe_duration;
/// Type of attack power used by the ability
attack_power_type ap_type;
/**
* @brief Behavior of dot.
*
* Acceptable inputs are DOT_CLIP, DOT_EXTEND, DOT_REFRESH_PANDEMIC, DOT_REFRESH_DURATION, or DOT_NONE.
*/
dot_behavior_e dot_behavior;
/// Ability specific extra player ready delay
rng::truncated_gauss_t ability_lag;
/// The minimum gcd triggered no matter the haste.
timespan_t min_gcd;
/// Hasted GCD stat type. One of HASTE_NONE, HASTE_ATTACK, HASTE_SPELL, SPEED_ATTACK, SPEED_SPELL
gcd_haste_type gcd_type;
/// Length of unhasted gcd triggered when used.
timespan_t trigger_gcd;
/// This is how far away the target can be from the player, and still be hit or targeted.
double range;
/**
* @brief AoE ability area of influence.
*
* Typically anything that has a radius, but not a range, is based on the players location.
*/
double radius;
/// Weapon AP multiplier.
double weapon_power_mod;
/// Attack/Spell power scaling of the ability.
struct {
double direct, tick;
} attack_power_mod, spell_power_mod;
/**
* @brief full damage variation in %
*
* Amount of variation of the full raw damage (base damage + AP/SP modified).
* Example: amount_delta = 0.1 means variation between 95% and 105%.
* Parsed from spell data.
*/
double amount_delta;
/// Amount of time the ability uses to execute before modifiers.
parsed_value_t<timespan_t> base_execute_time;
/// Amount of time the ability uses between ticks.
parsed_value_t<timespan_t> base_tick_time;
/// Default full duration of dot.
parsed_value_t<timespan_t> dot_duration;
/// Whether or not the dot duration is hasted. Channeled spells generally also have this flag.
bool hasted_dot_duration;
/// Maximum number of DoT stacks.
int dot_max_stack;
/// Whether or not the dot damage ignores number of stacks.
bool dot_ignore_stack;
/// Cost of using the ability.
std::array<parsed_value_t<double>, RESOURCE_MAX> base_costs;
/// Maximum amount of additional resource that can be expended.
std::array<double, RESOURCE_MAX> secondary_costs;
/// Cost of using ability per periodic effect tick.
std::array<double, RESOURCE_MAX> base_costs_per_tick;
/// Minimum base direct damage
double base_dd_min;
/// Maximum base direct damage
double base_dd_max;
/// Base tick damage
double base_td;
/// base direct damage multiplier
double base_dd_multiplier;
/// base tick damage multiplier
double base_td_multiplier;
/// base damage multiplier (direct and tick damage)
double base_multiplier;
double base_hit;
double base_crit;
double crit_chance_multiplier;
double crit_bonus_multiplier; // action scoped Add Percent Modifier (108): Spell Critical Bonus Multiplier (15)
double crit_bonus;
double base_dd_adder;
double base_ta_adder;
/// Weapon damage for the ability.
double weapon_multiplier;
/// Damage modifier for chained/AoE, exponentially increased by number of target hit.
double chain_multiplier;
/// Damage modifier for chained/AoE, linearly increasing with each target hit.
double chain_bonus_damage;
/// Static reduction of damage for AoE
double base_aoe_multiplier;
/// Static action cooldown duration multiplier
double base_recharge_multiplier;
/// A second static action cooldown duration multiplier that also reduces the effectiveness of flat cooldown adjustments
double base_recharge_rate_multiplier;
/// Dynamically adjustable action cooldown duration multipliers. These are reset to 1.0 in action_t::reset.
double dynamic_recharge_multiplier;
double dynamic_recharge_rate_multiplier;
/// Maximum distance that the ability can travel. Used on abilities that instantly move you, or nearly instantly move you to a location.
double base_teleport_distance;
/// Missile travel speed in yards / second
double travel_speed;
/// Missile travel delay in seconds
double travel_delay;
/// Minimum travel time in seconds
double min_travel_time;
// Amount of resource for the energize to grant.
double energize_amount;
/**
* @brief Movement Direction
* @code
* movement_directionality = movement_direction::MOVEMENT_OMNI; // Can move in any direction, ex: Heroic Leap, Blink. Generally set movement skills to this.
* movement_directionality = movement_direction::MOVEMENT_TOWARDS; // Can only be used towards enemy target. ex: Charge
* movement_directionality = movement_direction::MOVEMENT_AWAY; // Can only be used away from target. Ex: ????
* @endcode
*/
movement_direction_type movement_directionality;
/// This is used to cache/track spells that have a parent driver, such as most channeled/ground aoe abilities.
dot_t* parent_dot;
std::vector< action_t* > child_action;
/// This action will execute every tick
action_t* tick_action;
/// This action will execute every execute
action_t* execute_action;
/// This action will execute every impact - Useful for AoE debuffs
action_t* impact_action;
// Gain object of the same name as the action
gain_t* gain;
// Sets the behavior for when this action's resource energize occur.
action_energize energize_type;
// Resource for the energize to grant.
resource_e energize_resource;
/**
* @brief Used to manipulate cooldown duration and charges.
*
* @code
* cooldown -> duration = timespan_t::from_seconds( 20 ); //Ability has a cooldown of 20 seconds.
* cooldown -> charges = 3; // Ability has 3 charges.
* @endcode
*/
cooldown_t* cooldown, *internal_cooldown;
/** action statistics, merged by action-name */
stats_t* stats;
/** Execute event (e.g., "casting" of the action) */
event_t* execute_event;
/** Queue delay event (for queueing cooldowned actions shortly before they execute. */
event_t* queue_event;
/** Last available, effectively used execute time */
timespan_t time_to_execute;
/** Last available, effectively used travel time */
timespan_t time_to_travel;
/** Last available, effectively used resource cost */
double last_resource_cost;
/** Last available number of targets effectively hit */
int num_targets_hit;
/** Marker for sample action priority list reporting */
char marker;
/* The last time the action was executed */
timespan_t last_used;
// Effects which affect this action applied via apply_affecting_auras()
std::vector<std::pair<const spelleffect_data_t*, double>> affecting_list;
// Options
struct options_t {
/**
* moving (default: -1), when different from -1, will flag the action as usable only when the players are moving
* (moving=1) or not moving (moving=0). When left to -1, the action will be usable anytime. The players happen to
* move either because of a "movement" raid event, or because of "start_moving" actions. Note that actions which are
* not usable while moving do not need to be flagged with "move=0", Simulationcraft is already aware of those
* restrictions.
*
*/
int moving;
int wait_on_ready;
int max_cycle_targets;
int target_number;
/** Interrupt option. If true, channeled action can get interrupted. */
bool interrupt;
/**
* Chain can be used to re-cast a channeled spell at the beginning of its last tick. This has two advantages over
* waiting for the channel to complete before re-casting: 1) the gcd finishes sooner, and 2) it avoids the roughly
* 1/4 second delay between the end of a channel and the beginning of the next cast.
*/
bool chain;
int cycle_targets;
bool cycle_players;
bool interrupt_immediate;
std::string if_expr_str;
std::string target_if_str;
std::string interrupt_if_expr_str;
std::string early_chain_if_expr_str;
std::string cancel_if_expr_str;
std::string sync_str;
std::string target_str;
options_t();
} option;
bool interrupt_global;
std::unique_ptr<expr_t> if_expr;
enum target_if_mode_e
{
TARGET_IF_NONE,
TARGET_IF_FIRST,
TARGET_IF_MIN,
TARGET_IF_MAX
} target_if_mode;
std::unique_ptr<expr_t> target_if_expr;
std::unique_ptr<expr_t> interrupt_if_expr;
std::unique_ptr<expr_t> early_chain_if_expr;
std::unique_ptr<expr_t> cancel_if_expr;
action_t* sync_action;
std::string signature_str;
target_specific_t<dot_t> target_specific_dot;
target_specific_t<buff_t> target_specific_debuff;
const spell_data_t* target_debuff;
action_priority_list_t* action_list;
/**
* @brief Resource starvation tracking.
*
* Tracking proc triggered on resource-starved ready() calls.
* Can be overridden by class modules for tracking purposes.
*/
proc_t* starved_proc;
// Tracking proc triggered when action fails to execute after being queued.
// Can be overridden by class modules for tracking purposes.
proc_t* queue_failed_proc;
uint_least64_t total_executions;
/**
* @brief Cooldown for specific APL line.
*
* Tied to a action_t object, and not shared by action-name,
* this cooldown helps articifally restricting usage of a specific line
* in the APL.
*/
std::unique_ptr<cooldown_t> line_cooldown;
const action_priority_t* signature;
/// State of the last execute()
action_state_t* execute_state;
/// Optional - if defined before execute(), will be copied into execute_state
action_state_t* pre_execute_state;
unsigned snapshot_flags;
unsigned update_flags;
/**
* Target Cache System
* - list: contains the cached target pointers
* - callback: unique_Ptr to callback
* - is_valid: gets invalidated by the callback from the target list source.
* When the target list is requested in action_t::target_list(), it gets recalculated if
* flag is false, otherwise cached version is used
*/
struct target_cache_t {
std::vector< player_t* > list;
bool is_valid;
target_cache_t() : is_valid( false ) {}
} mutable target_cache;
private:
std::vector<std::unique_ptr<option_t>> options;
action_state_t* state_cache;
std::vector<travel_event_t*> travel_events;
public:
action_t( action_e type, util::string_view token, player_t* p );
action_t( action_e type, util::string_view token, player_t* p, const spell_data_t* s );
virtual ~action_t();
void add_child( action_t* child );
void add_option( std::unique_ptr<option_t> new_option );
void check_spec( specialization_e );
void check_spell( const spell_data_t* );
/**
* @brief Spell Data associated with the action.
*
* spell_data_t::nil() if no spell data is available,
* spell_data_t::not_found if spell given was not found.
*
* This means that if no spell data is available/found (eg. talent not available),
* all spell data fields will be filled with 0 and can thus be used directly
* without checking specifically for the spell_data_t::ok()
*/
const spell_data_t& data() const
{ return ( *s_data ); }
// return s_data_reporting if available, otherwise fallback to s_data
const spell_data_t& data_reporting() const;
dot_t* find_dot( player_t* target ) const;
buff_t* find_debuff( player_t* target ) const;
bool is_aoe() const
{
const int num_targets = n_targets();
return num_targets == -1 || num_targets > 0;
}
const char* name() const
{ return name_str.c_str(); }
const char* name_reporting() const;
size_t num_travel_events() const
{ return travel_events.size(); }
bool has_travel_events() const
{ return ! travel_events.empty(); }
timespan_t shortest_travel_event() const;
bool has_travel_events_for( const player_t* ) const;
/** Determine if the action can have a resulting damage/heal amount > 0 */
virtual bool has_amount_result() const
{
return attack_power_mod.direct > 0 || attack_power_mod.tick > 0
|| spell_power_mod.direct > 0 || spell_power_mod.tick > 0
|| (weapon && weapon_multiplier > 0);
}
void parse_spell_data( const spell_data_t& );
void parse_effect_direct_mods( const spelleffect_data_t& spelleffect_data, bool item_scaling );
void parse_effect_periodic_mods( const spelleffect_data_t& spelleffect_data, bool item_scaling );
void parse_effect_period( const spelleffect_data_t& );
void parse_effect_data( const spelleffect_data_t& );
void parse_target_str();
void remove_travel_event( travel_event_t* e );
void reschedule_queue_event();
rng::rng_t& rng();
rng::rng_t& rng() const;
player_t* select_target_if_target();
void apply_affecting_aura( const spell_data_t*, const spell_data_t* modifier = nullptr );
void apply_affecting_effect( const spelleffect_data_t& effect, const spelleffect_data_t* modifier = nullptr );
void apply_affecting_conduit( const conduit_data_t& conduit, int effect_num = 1 );
void apply_affecting_conduit_effect( const conduit_data_t& conduit, size_t effect_num );
action_state_t* get_state( const action_state_t* = nullptr );
void execute_on_target( player_t*, double = -1.0 );
private:
friend struct action_state_t;
void release_state( action_state_t* );
public:
// =======================
// Const virtual functions
// =======================
virtual bool verify_actor_level() const;
virtual bool verify_actor_spec() const;
virtual bool verify_actor_weapon() const;
virtual double cost() const;
virtual double cost_flat_modifier() const;
virtual double cost_pct_multiplier() const;
virtual double base_cost() const;
virtual double cost_per_tick( resource_e ) const;
virtual timespan_t cooldown_duration() const;
virtual timespan_t gcd() const;
virtual double false_positive_pct() const;
virtual double false_negative_pct() const;
virtual timespan_t execute_time() const;
virtual timespan_t execute_time_flat_modifier() const;
virtual double execute_time_pct_multiplier() const;
virtual timespan_t tick_time( const action_state_t* state ) const;
virtual timespan_t tick_time_flat_modifier( const action_state_t* state ) const;
virtual double tick_time_pct_multiplier( const action_state_t* state ) const;
virtual timespan_t travel_time() const;
virtual timespan_t distance_targeting_travel_time( action_state_t* s ) const;
virtual result_e calculate_result( action_state_t* /* state */ ) const
{
assert( false );
return RESULT_UNKNOWN;
}
virtual block_result_e calculate_block_result( action_state_t* s ) const;
virtual double calculate_direct_amount( action_state_t* state ) const;
virtual double calculate_tick_amount( action_state_t* state, double multiplier ) const;
virtual double calculate_weapon_damage( double attack_power ) const;
virtual double calculate_crit_damage_bonus( action_state_t* s ) const;
virtual double recharge_multiplier( const cooldown_t& ) const
{ return base_recharge_multiplier * dynamic_recharge_multiplier; }
virtual double recharge_rate_multiplier( const cooldown_t& ) const
{ return base_recharge_rate_multiplier * dynamic_recharge_rate_multiplier; }
/** Cooldown base duration for action based cooldowns. */
virtual timespan_t cooldown_base_duration( const cooldown_t& cd ) const;
virtual resource_e current_resource() const
{ return resource_current; }
virtual int n_targets() const
{ return aoe; }
virtual double last_tick_factor( const dot_t* d, timespan_t time_to_tick, timespan_t duration ) const;
virtual result_amount_type amount_type( const action_state_t* /* state */, bool /* periodic */ = false ) const
{ return result_amount_type::NONE; }
virtual result_amount_type report_amount_type( const action_state_t* state ) const;
virtual double miss_chance( double /* hit */, player_t* /* target */ ) const
{ return 0; }
virtual double dodge_chance( double /* expertise */, player_t* /* target */ ) const
{ return 0; }
virtual double parry_chance( double /* expertise */, player_t* /* target */ ) const
{ return 0; }
virtual double glance_chance( int /* delta_level */ ) const
{ return 0; }
virtual double block_chance( action_state_t* /* state */ ) const
{ return 0; }
virtual double crit_block_chance( action_state_t* /* state */ ) const
{ return 0; }
virtual double total_crit_bonus( const action_state_t* /* state */ ) const; // Check if we want to move this into the stateless system.
virtual int num_targets() const;
virtual size_t available_targets( std::vector< player_t* >& ) const;
virtual std::vector< player_t* >& target_list() const;
virtual player_t* find_target_by_number( int number ) const;
virtual bool execute_targeting( action_t* action ) const;
virtual std::vector<player_t*>& targets_in_range_list( std::vector< player_t* >& tl ) const;
virtual std::vector<player_t*>& check_distance_targeting( std::vector< player_t* >& tl ) const;
virtual double ppm_proc_chance( double PPM ) const;
virtual bool usable_moving() const;
virtual bool usable_precombat() const;
virtual timespan_t composite_dot_duration( const action_state_t* ) const;
virtual timespan_t dot_duration_flat_modifier( const action_state_t* ) const;
virtual double dot_duration_pct_multiplier( const action_state_t* ) const;
virtual double attack_direct_power_coefficient( const action_state_t* ) const
{ return attack_power_mod.direct; }
virtual double amount_delta_modifier( const action_state_t* ) const
{ return amount_delta; }
virtual double attack_tick_power_coefficient( const action_state_t* ) const
{ return attack_power_mod.tick; }
virtual double spell_direct_power_coefficient( const action_state_t* ) const
{ return spell_power_mod.direct; }
virtual double spell_tick_power_coefficient( const action_state_t* ) const
{ return spell_power_mod.tick; }
virtual double base_da_min( const action_state_t* ) const
{ return base_dd_min; }
virtual double base_da_max( const action_state_t* ) const
{ return base_dd_max; }
virtual double base_ta( const action_state_t* ) const
{ return base_td; }
virtual double bonus_da( const action_state_t* ) const
{ return base_dd_adder; }
virtual double bonus_ta( const action_state_t* ) const
{ return base_ta_adder; }
virtual double range_() const
{ return range; }
virtual double radius_() const
{ return radius; }
virtual double action_multiplier() const
{ return base_multiplier; }
virtual double action_da_multiplier() const
{ return base_dd_multiplier; }
virtual double action_ta_multiplier() const
{ return base_td_multiplier; }
virtual double composite_hit() const
{ return base_hit; }
virtual double composite_crit_chance() const
{ return base_crit; }
virtual double composite_crit_chance_multiplier() const
{ return crit_chance_multiplier; }
virtual double composite_crit_damage_bonus_multiplier() const
{ return crit_bonus_multiplier; }
virtual double composite_haste() const
{ return 1.0; }
virtual attack_power_type get_attack_power_type() const
{ return ap_type; }
virtual double composite_total_attack_power() const;
virtual double composite_total_spell_power() const;
virtual double composite_target_armor( player_t* ) const;
virtual double composite_target_crit_chance( player_t* ) const;
virtual double composite_target_crit_damage_bonus_multiplier( player_t* ) const
{ return 1.0; }
virtual double composite_target_multiplier( player_t* ) const;
virtual double composite_target_damage_vulnerability( player_t* ) const;
virtual double composite_versatility( const action_state_t* ) const
{ return 1.0; }
virtual double composite_leech( const action_state_t* ) const;
virtual double composite_run_speed() const;
virtual double composite_avoidance() const;
virtual double composite_corruption() const;