36 using namespace std::chrono_literals;
40 return cfg[
"healed_sound"].
empty() ?
"heal.wav" :
cfg[
"healed_sound"].str();
47 config result = attributes;
56 std::vector<config::const_all_children_iterator>
children;
64 : itors(
cfg.all_children_range()), branches(1), parent(nullptr)
66 branches.back().attributes.merge_attributes(
cfg);
70 : itors(
cfg.all_children_range()), branches(
p->branches), parent(
p)
76 bool previously_hits_set =
false;
77 bool previously_direction_set =
false;
78 bool previously_terrain_set =
false;
79 bool previously_value_set =
false;
80 bool previously_value_2nd_set =
false;
82 const std::string s_cfg_hits =
cfg[
"hits"];
83 const std::string s_cfg_direction =
cfg[
"direction"];
84 const std::string s_cfg_terrain =
cfg[
"terrain_types"];
85 const std::string s_cfg_value =
cfg[
"value"];
86 const std::string s_cfg_value_2nd =
cfg[
"value_2nd"];
88 for(
const auto& branch : branches) {
89 const std::string s_branch_hits = branch.attributes[
"hits"];
90 const std::string s_branch_direction = branch.attributes[
"direction"];
91 const std::string s_branch_terrain = branch.attributes[
"terrain_types"];
92 const std::string s_branch_value = branch.attributes[
"value"];
93 const std::string s_branch_value_2nd = branch.attributes[
"value_second"];
95 if(!s_branch_hits.empty() && s_branch_hits == s_cfg_hits) {
96 previously_hits_set =
true;
99 if(!s_branch_direction.empty() && s_branch_direction == s_cfg_direction) {
100 previously_direction_set =
true;
103 if(!s_branch_terrain.empty() && s_branch_terrain == s_cfg_terrain) {
104 previously_terrain_set =
true;
107 if(!s_branch_value.empty() && s_branch_value == s_cfg_value) {
108 previously_value_set =
true;
111 if(!s_branch_value_2nd.empty() && s_branch_value_2nd == s_cfg_value_2nd) {
112 previously_value_2nd_set =
true;
118 for(
auto iter = branches.begin(); iter != branches.end(); ) {
119 const std::string s_branch_hits = (*iter).attributes[
"hits"];
120 const std::string s_branch_direction = (*iter).attributes[
"direction"];
121 const std::string s_branch_terrain = (*iter).attributes[
"terrain_types"];
122 const std::string s_branch_value = (*iter).attributes[
"value"];
123 const std::string s_branch_value_2nd = (*iter).attributes[
"value_second"];
125 const bool hits_match = (previously_hits_set && s_branch_hits != s_cfg_hits);
126 const bool direction_match = (previously_direction_set && s_branch_direction != s_cfg_direction);
127 const bool terrain_match = (previously_terrain_set && s_branch_terrain != s_cfg_terrain);
128 const bool value_match = (previously_value_set && s_branch_value != s_cfg_value);
129 const bool value_2nd_match = (previously_value_2nd_set && s_branch_value_2nd != s_cfg_value_2nd);
131 if((!previously_hits_set || hits_match) &&
132 (!previously_direction_set || direction_match) &&
133 (!previously_terrain_set || terrain_match) &&
134 (!previously_value_set || value_match) &&
135 (!previously_value_2nd_set || value_2nd_match) &&
136 (hits_match || direction_match || terrain_match || value_match || value_2nd_match))
138 branches.
erase(iter++);
140 (*iter).attributes.merge_attributes(
cfg);
147 for(
auto iter = parent->branches.begin(); iter != parent->branches.end(); ) {
148 const std::string s_branch_hits = (*iter).attributes[
"hits"];
149 const std::string s_branch_direction = (*iter).attributes[
"direction"];
150 const std::string s_branch_terrain = (*iter).attributes[
"terrain_types"];
151 const std::string s_branch_value = (*iter).attributes[
"value"];
152 const std::string s_branch_value_2nd = (*iter).attributes[
"value_second"];
154 const bool hits_match = (previously_hits_set && s_branch_hits == s_cfg_hits);
155 const bool direction_match = (previously_direction_set && s_branch_direction == s_cfg_direction);
156 const bool terrain_match = (previously_terrain_set && s_branch_terrain == s_cfg_terrain);
157 const bool value_match = (previously_value_set && s_branch_value == s_cfg_value);
158 const bool value_2nd_match = (previously_value_2nd_set && s_branch_value_2nd == s_cfg_value_2nd);
160 if((!previously_hits_set || hits_match) &&
161 (!previously_direction_set || direction_match) &&
162 (!previously_terrain_set || terrain_match) &&
163 (!previously_value_set || value_match) &&
164 (!previously_value_2nd_set || value_2nd_match) &&
165 (hits_match || direction_match || terrain_match || value_match || value_2nd_match))
167 parent->branches.erase(iter++);
187 std::list<animation_cursor> anim_cursors;
188 anim_cursors.emplace_back(anim_cfg);
190 while(!anim_cursors.empty()) {
194 if(ac.
itors.empty()) {
199 anim_cursors.pop_back();
203 if(ac.
itors.front().key !=
"if") {
209 ac.
itors.pop_front();
217 anim_cursors.emplace_back(ac.
itors.front().cfg, &ac);
218 ac.
itors.pop_front();
220 }
while (!ac.
itors.empty() && ac.
itors.front().key ==
"else");
231 std::cout <<
"--branch--\n" << ab.attributes;
233 std::cout <<
"--branchcfg--\n" << ci->cfg;
240 assert(anim_cursors.size() == 1);
252 return expanded_animations;
259 , secondary_unit_filter_()
262 , base_score_(variation)
265 , primary_attack_filter_()
266 , secondary_attack_filter_()
270 , unit_anim_(start_time,builder)
273 , invalidated_(false)
274 , play_offscreen_(true)
283 , secondary_unit_filter_()
285 , frequency_(
cfg[
"frequency"].to_int())
286 , base_score_(
cfg[
"base_score"].to_int())
289 , primary_attack_filter_()
290 , secondary_attack_filter_()
294 , unit_anim_(
cfg,frame_string)
297 , invalidated_(false)
298 , play_offscreen_(true)
304 if(key == frame_string) {
308 if(key.find(
"_frame", key.size() - 6) == std::string::npos) {
321 const std::vector<std::string>& my_directions =
utils::split(
cfg[
"direction"]);
322 for(
const auto& direction : my_directions) {
347 value_.push_back(utils::from_chars<int>(v).value_or(0));
351 if(
h ==
"yes" ||
h == strike_result::hit) {
352 hits_.push_back(strike_result::type::hit);
355 if(
h ==
"no" ||
h == strike_result::miss) {
356 hits_.push_back(strike_result::type::miss);
359 if(
h ==
"yes" ||
h == strike_result::kill ) {
360 hits_.push_back(strike_result::type::kill);
365 value2_.push_back(utils::from_chars<int>(v2).value_or(0));
386 if(!event.empty() && !
event_.empty()) {
468 if(!attack->matches_filter(iter))
return MATCH_FAIL;
479 if(!second_attack->matches_filter(iter))
return MATCH_FAIL;
490 std::vector<unit_animation> animation_base;
491 for(
const auto& anim : animations) {
493 animation_base.push_back(anim);
495 animation_base.back().event_.clear();
499 const std::string default_image =
cfg[
"image"];
501 if(animation_base.empty()) {
507 frame_builder().
image(default_image).duration(300ms).blend(
"0.0~0.3:100,0.3~0.0:200", {255,255,255}),
508 "_disabled_selected_", 0));
510 for(
const auto& base : animation_base) {
511 animations.push_back(base);
512 animations.back().event_ = {
"standing" };
513 animations.back().play_offscreen_ =
false;
515 animations.push_back(base);
516 animations.back().event_ = {
"_ghosted_" };
517 animations.back().unit_anim_.override(0ms, animations.back().unit_anim_.get_animation_duration(),
particle::UNSET,
"0.9",
"", {0,0,0},
"",
"",
"~GS()");
519 animations.push_back(base);
520 animations.back().event_ = {
"_disabled_ghosted_" };
521 animations.back().unit_anim_.override(0ms, 1ms,
particle::UNSET,
"0.4",
"", {0,0,0},
"",
"",
"~GS()");
523 animations.push_back(base);
524 animations.back().event_ = {
"selected" };
525 animations.back().unit_anim_.override(0ms, 300ms,
particle::UNSET,
"",
"0.0~0.3:100,0.3~0.0:200", {255,255,255});
527 animations.push_back(base);
528 animations.back().event_ = {
"recruited" };
531 animations.push_back(base);
532 animations.back().event_ = {
"levelin" };
533 animations.back().unit_anim_.override(0ms, 600ms,
particle::NO_CYCLE,
"",
"1~0:600", {255,255,255});
535 animations.push_back(base);
536 animations.back().event_ = {
"levelout" };
537 animations.back().unit_anim_.override(0ms, 600ms,
particle::NO_CYCLE,
"",
"0~1:600,1", {255,255,255});
539 animations.push_back(base);
540 animations.back().event_ = {
"pre_movement" };
543 animations.push_back(base);
544 animations.back().event_ = {
"post_movement" };
547 animations.push_back(base);
548 animations.back().event_ = {
"movement" };
549 animations.back().unit_anim_.override(0ms, 200ms,
552 animations.push_back(base);
553 animations.back().event_ = {
"defend" };
554 animations.back().unit_anim_.override(0ms, animations.back().unit_anim_.get_animation_duration(),
556 animations.back().hits_.push_back(strike_result::type::hit);
557 animations.back().hits_.push_back(strike_result::type::kill);
559 animations.push_back(base);
560 animations.back().event_ = {
"defend" };
562 animations.push_back(base);
563 animations.back().event_ = {
"attack" };
564 animations.back().unit_anim_.override(-150ms, 300ms,
particle::NO_CYCLE,
"",
"", {0,0,0},
"0~0.6:150,0.6~0:150", std::to_string(
get_abs_frame_layer(
drawing_layer::unit_move_default)));
565 animations.back().primary_attack_filter_.emplace_back(
"range",
"melee");
567 animations.push_back(base);
568 animations.back().event_ = {
"attack" };
570 animations.back().primary_attack_filter_.emplace_back(
"range",
"ranged");
572 animations.push_back(base);
573 animations.back().event_ = {
"death" };
575 animations.back().sub_anims_[
"_death_sound"] =
particle();
576 animations.back().sub_anims_[
"_death_sound"].add_frame(1ms,
frame_builder().
sound(
cfg[
"die_sound"]),
true);
578 animations.push_back(base);
579 animations.back().event_ = {
"victory" };
580 animations.back().unit_anim_.override(0ms, animations.back().unit_anim_.get_animation_duration(),
particle::CYCLE);
582 animations.push_back(base);
584 animations.back().event_ = {
"pre_teleport" };
586 animations.push_back(base);
588 animations.back().event_ = {
"post_teleport" };
590 animations.push_back(base);
591 animations.back().event_ = {
"healing" };
593 animations.push_back(base);
594 animations.back().event_ = {
"healed" };
595 animations.back().unit_anim_.override(0ms, 300ms,
particle::NO_CYCLE,
"",
"0:30,0.5:30,0:30,0.5:30,0:30,0.5:30,0:30,0.5:30,0:30", {255,255,255});
599 animations.back().sub_anims_[
"_healed_sound"].add_frame(1ms,
frame_builder().
sound(healed_sound),
true);
601 animations.push_back(base);
602 animations.back().event_ = {
"poisoned" };
603 animations.back().unit_anim_.override(0ms, 300ms,
particle::NO_CYCLE,
"",
"0:30,0.5:30,0:30,0.5:30,0:30,0.5:30,0:30,0.5:30,0:30", {0,255,0});
604 animations.back().sub_anims_[
"_poison_sound"] =
particle();
610 const config&
cfg,
char const* tag_name,
char const* apply_to,
612 bool offscreen =
true)
616 anim[
"apply_to"] = apply_to;
620 if(v.
empty()) v =
false;
626 animations.emplace_back(anim);
633 animations.emplace_back(ab.merge());
651 anim[
"apply_to"] =
"standing";
652 anim[
"cycles"] =
true;
656 std::string sub_frame_name = ci->key;
657 std::size_t pos = sub_frame_name.find(
"_frame");
658 if(pos != std::string::npos) {
659 anim[sub_frame_name.substr(0, pos) +
"_cycles"] =
true;
663 if(anim[
"layer"].empty()) {
664 anim[
"layer"] = default_layer;
667 if(anim[
"offscreen"].empty()) {
668 anim[
"offscreen"] =
false;
671 animations.emplace_back(anim);
677 anim[
"apply_to"] =
"default";
678 anim[
"cycles"] =
true;
681 std::string sub_frame_name = ci->key;
682 std::size_t pos = sub_frame_name.find(
"_frame");
683 if(pos != std::string::npos) {
684 anim[sub_frame_name.substr(0, pos) +
"_cycles"] =
true;
688 if(anim[
"layer"].empty()) {
689 anim[
"layer"] = default_layer;
692 if(anim[
"offscreen"].empty()) {
693 anim[
"offscreen"] =
false;
696 animations.emplace_back(anim);
701 anim[
"apply_to"] =
"healing";
702 anim[
"value"] = anim[
"damage"];
704 if(anim[
"layer"].empty()) {
705 anim[
"layer"] = default_layer;
708 animations.emplace_back(anim);
713 anim[
"apply_to"] =
"healed";
714 anim[
"value"] = anim[
"healing"];
716 if(anim[
"layer"].empty()) {
717 anim[
"layer"] = default_layer;
720 animations.emplace_back(anim);
721 animations.back().sub_anims_[
"_healed_sound"] =
particle();
724 animations.back().sub_anims_[
"_healed_sound"].add_frame(1ms,
frame_builder().
sound(healed_sound),
true);
729 anim[
"apply_to"] =
"poisoned";
730 anim[
"value"] = anim[
"damage"];
732 if(anim[
"layer"].empty()) {
733 anim[
"layer"] = default_layer;
736 animations.emplace_back(anim);
737 animations.back().sub_anims_[
"_poison_sound"] =
particle();
745 anim[
"apply_to"] =
"movement";
747 if(anim[
"offset"].empty()) {
748 anim[
"offset"] =
"0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,0~1:200,";
751 if(anim[
"layer"].empty()) {
752 anim[
"layer"] = move_layer;
755 animations.emplace_back(anim);
762 anim[
"apply_to"] =
"defend";
764 if(anim[
"layer"].empty()) {
765 anim[
"layer"] = default_layer;
768 if(!anim[
"damage"].empty() && anim[
"value"].empty()) {
769 anim[
"value"] = anim[
"damage"];
772 if(anim[
"hits"].empty()) {
773 anim[
"hits"] =
false;
774 animations.emplace_back(anim);
775 animations.back().base_score_--;
778 animations.emplace_back(anim);
779 animations.back().base_score_--;
781 image::locator image_loc = animations.back().get_last_frame().end_parameters().image;
785 .blend(
"0.0,0.5:75,0.0:75,0.5:75,0.0", {255,0,0}));
787 for(
const std::string& hit_type :
utils::split(anim[
"hits"])) {
789 tmp[
"hits"] = hit_type;
791 animations.emplace_back(tmp);
793 image::locator image_loc = animations.back().get_last_frame().end_parameters().image;
794 if(hit_type ==
"yes" || hit_type == strike_result::hit || hit_type == strike_result::kill) {
798 .blend(
"0.0,0.5:75,0.0:75,0.5:75,0.0", {255,0,0}));
809 anim[
"apply_to"] =
"attack";
811 if(anim[
"layer"].empty()) {
812 anim[
"layer"] = move_layer;
816 if(anim[
"offset"].empty() && missile_fs.
empty()) {
817 anim[
"offset"] =
"0~0.6,0.6~0";
820 if(!missile_fs.empty()) {
821 if(anim[
"missile_offset"].empty()) {
822 anim[
"missile_offset"] =
"0~0.8";
825 if(anim[
"missile_layer"].empty()) {
826 anim[
"missile_layer"] = missile_layer;
836 animations.emplace_back(anim);
841 anim[
"apply_to"] =
"death";
843 if(anim[
"layer"].empty()) {
844 anim[
"layer"] = default_layer;
847 animations.emplace_back(anim);
848 image::locator image_loc = animations.back().get_last_frame().end_parameters().image;
853 .highlight(
"1~0:600"));
855 if(!
cfg[
"die_sound"].empty()) {
856 animations.back().sub_anims_[
"_death_sound"] =
particle();
857 animations.back().sub_anims_[
"_death_sound"].add_frame(1ms,
frame_builder().
sound(
cfg[
"die_sound"]),
true);
865 anim[
"apply_to"] = anim[
"flag"];
867 if(anim[
"layer"].empty()) {
868 anim[
"layer"] = default_layer;
871 animations.emplace_back(anim);
876 if(anim[
"layer"].empty()) {
877 anim[
"layer"] = default_layer;
880 anim[
"apply_to"] =
"pre_teleport";
881 animations.emplace_back(anim);
882 animations.back().unit_anim_.set_end_time(0ms);
884 anim[
"apply_to"] =
"post_teleport";
885 animations.emplace_back(anim);
886 animations.back().unit_anim_.remove_frames_until(0ms);
891 ,
const std::chrono::milliseconds& duration
893 ,
const std::string& highlight
894 ,
const std::string& blend_ratio
896 ,
const std::string& offset
897 ,
const std::string& layer
898 ,
const std::string& modifiers)
920 if(parameters_.need_update())
return true;
934 , last_frame_begin_time_(0)
940 if(!range.empty() &&
cfg[frame_string +
"start_time"].
empty()) {
941 for(
const config& frame : range) {
948 for(
const config& frame : range) {
953 cycles_ =
cfg[frame_string +
"cycles"].to_bool(
false);
965 if(anim.second.need_update())
return true;
980 if(anim.second.need_minimal_update())
return true;
990 if(!anim.second.animation_finished())
return false;
1000 if(!anim.second.animation_finished_potential())
return false;
1011 anim.second.update_last_draw_time(acceleration);
1019 result = std::max(result, anim.second.get_end_time());
1029 result = std::min(result, anim.second.get_begin_time());
1038 ,
const std::string& text
1040 ,
const bool accelerate)
1057 anim.second.start_animation(start_time);
1072 anim.second.pause_animation();
1081 anim.second.restart_animation();
1095 anim.second.redraw(value,
src_,
dst_, halo_man);
1104 anim.second.clear_halo();
1116 if(complete_redraw) {
1122 std::set<map_location> tmp = anim.second.get_overlaped_hex(value,
src_,
dst_);
1132 if(complete_redraw) {
1154 std::ostringstream outstream;
1156 return outstream.str();
1162 outstream <<
"[" << events_string <<
"]\n";
1164 outstream <<
"\tstart_time=" << u_animation.
get_begin_time().count() <<
'\n';
1166 if(u_animation.
hits_.size() > 0) {
1167 std::vector<std::string> hits;
1169 outstream <<
"\thits=" <<
utils::join(hits) <<
'\n';
1173 std::vector<std::string> dirs;
1175 outstream <<
"\tdirections=" <<
utils::join(dirs) <<
'\n';
1185 outstream <<
"[filter]\n";
1190 outstream <<
"[/filter]\n";
1194 outstream <<
"[filter_second]\n";
1199 outstream <<
"[/filter_second]\n";
1203 outstream <<
"[filter_attack]\n";
1208 outstream <<
"[/filter_attack]\n";
1212 outstream <<
"[filter_second_attack]\n";
1217 outstream <<
"[/filter_second_attack]\n";
1221 outstream <<
"\t[frame]\n";
1223 outstream <<
"\t\t" << frame_string <<
"\n";
1225 outstream <<
"\t[/frame]\n";
1228 for(std::pair<std::string, unit_animation::particle>
p : u_animation.
sub_anims_) {
1229 for(std::size_t
i = 0;
i <
p.second.get_frames_count();
i++) {
1230 std::string sub_frame_name =
p.first;
1231 std::size_t pos = sub_frame_name.find(
"_frame");
1232 if(pos != std::string::npos) sub_frame_name = sub_frame_name.substr(0, pos);
1234 outstream <<
"\t" << sub_frame_name <<
"_start_time=" <<
p.second.get_begin_time().count() <<
'\n';
1235 outstream <<
"\t[" <<
p.first <<
"]\n";
1237 for(
const std::string& frame_string :
p.second.get_frame(
i).debug_strings()) {
1238 outstream <<
"\t\t" << frame_string <<
'\n';
1241 outstream <<
"\t[/" <<
p.first <<
"]\n";
1245 outstream <<
"[/" << events_string <<
"]\n";
1251 const unit_frame& current_frame = get_current_frame();
1258 if(animation_time > get_current_frame_end_time()) in_scope_of_frame =
false;
1267 current_frame.
redraw(get_current_frame_time(),
true, in_scope_of_frame,
src,
dst, halo_id_, halo_man, default_val, value);
1269 current_frame.
redraw(get_current_frame_time(),
false, in_scope_of_frame,
src,
dst, halo_id_, halo_man, default_val, value);
1280 const unit_frame& current_frame = get_current_frame();
1293 parameters_.override(get_animation_duration());
1299 ,
const std::string& event
1304 ,
const std::string& text
1311 if(!animated_unit)
return;
1314 animated_unit->anim_comp().choose_animation(
src, event,
dst, value, hit_type, attack, second_attack, value2);
1318 animated_units_.AGGREGATE_EMPLACE(std::move(animated_unit), anim, text, text_color,
src, with_bars);
1325 ,
const std::string& text
1328 if(!animated_unit || !anim)
return;
1331 animated_units_.AGGREGATE_EMPLACE(std::move(animated_unit), anim, text, text_color,
src, with_bars);
1335 ,
const std::string& event
1344 return (animated_unit && animated_unit->anim_comp().choose_animation(
src, event,
dst, value, hit_type, attack, second_attack, value2));
1348 ,
const std::string& event
1353 ,
const std::string& text
1360 if(!animated_unit)
return;
1362 if(animated_unit->anim_comp().get_animation() &&
1363 !animated_unit->anim_comp().get_animation()->animation_finished_potential() &&
1364 animated_unit->anim_comp().get_animation()->matches(
1367 animated_units_.AGGREGATE_EMPLACE(animated_unit,
nullptr, text, text_color,
src, with_bars);
1369 add_animation(animated_unit,event,
src,
dst,value,with_bars,text,text_color,hit_type,attack,second_attack,value2);
1375 auto begin_time = std::chrono::milliseconds::max();
1377 for(
const auto& anim : animated_units_) {
1378 if(anim.my_unit->anim_comp().get_animation()) {
1379 if(anim.animation) {
1380 begin_time = std::min(begin_time, anim.animation->get_begin_time());
1382 begin_time = std::min(begin_time, anim.my_unit->anim_comp().get_animation()->get_begin_time());
1387 for(
auto& anim : animated_units_) {
1388 if(anim.animation) {
1389 anim.my_unit->anim_comp().start_animation(begin_time, anim.animation, anim.with_bars, anim.text, anim.text_color);
1390 anim.animation =
nullptr;
1392 anim.my_unit->anim_comp().get_animation()->update_parameters(anim.src, anim.src.get_direction(anim.my_unit->facing()));
1399 bool finished =
true;
1400 for(
const auto& anim : animated_units_) {
1401 finished &= anim.my_unit->anim_comp().get_animation()->animation_finished_potential();
1409 if(animated_units_.empty()) {
1414 animated_units_[0].my_unit->anim_comp().get_animation()->set_max_animation_time(animation_time);
1421 using std::chrono::steady_clock;
1422 auto end_tick = animated_units_[0].my_unit->anim_comp().get_animation()->time_to_tick(animation_time);
1424 while(steady_clock::now() < end_tick - std::min(std::chrono::floor<std::chrono::milliseconds>(20ms / speed), 20ms)) {
1425 auto rest = std::chrono::floor<std::chrono::milliseconds>((animation_time -
get_animation_time()) * speed);
1426 std::this_thread::sleep_for(std::clamp(rest, 0ms, 10ms));
1429 end_tick = animated_units_[0].my_unit->anim_comp().get_animation()->time_to_tick(animation_time);
1432 auto rest = std::max<steady_clock::duration>(0ms, end_tick - steady_clock::now() + 5ms);
1433 std::this_thread::sleep_for(rest);
1436 animated_units_[0].my_unit->anim_comp().get_animation()->set_max_animation_time(0ms);
1441 bool finished =
false;
1445 std::this_thread::sleep_for(10ms);
1448 for(
const auto& anim : animated_units_) {
1449 finished &= anim.my_unit->anim_comp().get_animation()->animation_finished_potential();
1456 if(animated_units_.empty()) {
1459 return animated_units_[0].my_unit->anim_comp().get_animation()->get_animation_time() ;
1464 if(animated_units_.empty()) {
1467 return animated_units_[0].my_unit->anim_comp().get_animation()->get_animation_time_potential() ;
1472 auto end_time = std::chrono::milliseconds::min();
1473 for(
const auto& anim : animated_units_) {
1474 if(anim.my_unit->anim_comp().get_animation()) {
1475 end_time = std::max(end_time, anim.my_unit->anim_comp().get_animation()->get_end_time());
1484 for(
const auto& anim : animated_units_) {
1485 if(anim.my_unit->anim_comp().get_animation()) {
1486 anim.my_unit->anim_comp().get_animation()->pause_animation();
1493 for(
const auto& anim : animated_units_) {
1494 if(anim.my_unit->anim_comp().get_animation()) {
1495 anim.my_unit->anim_comp().get_animation()->restart_animation();
1502 for(
const auto& anim : animated_units_) {
1503 anim.my_unit->anim_comp().set_standing();
void new_animation_frame()
std::list< animation_branch > animation_branches
static void prepare_single_animation(const config &anim_cfg, animation_branches &expanded_anims)
static std::string get_heal_sound(const config &cfg)
static void add_simple_anim(std::vector< unit_animation > &animations, const config &cfg, char const *tag_name, char const *apply_to, drawing_layer layer=drawing_layer::unit_default, bool offscreen=true)
static animation_branches prepare_animation(const config &cfg, const std::string &animation_tag)
std::unique_ptr< PangoAttribute, void(*)(PangoAttribute *)> value_
const T & get_frame(std::size_t n) const
std::chrono::milliseconds get_begin_time() const
void start_animation(const std::chrono::milliseconds &start_time, bool cycles=false)
Starts an animation cycle.
bool animation_finished_potential() const
void add_frame(const std::chrono::milliseconds &duration, const unit_frame &value, bool force_change=false)
Adds a frame to an animation.
void update_last_draw_time(double acceleration=0)
void set_end_time(const std::chrono::milliseconds &ending_time)
const unit_frame & get_last_frame() const
bool animation_finished() const
Returns true if the current animation was finished.
std::chrono::milliseconds starting_frame_time_
std::size_t get_frames_count() const
void set_begin_time(const std::chrono::milliseconds &new_begin_time)
std::chrono::milliseconds get_animation_duration() const
std::chrono::milliseconds get_end_time() const
Variant for storing WML attributes.
bool empty() const
Tests for an attribute that either was never set or was set to "".
A config object defines a single node in a WML file, with access to child nodes.
config & add_child(std::string_view key)
all_children_iterator erase(const all_children_iterator &i)
boost::iterator_range< const_all_children_iterator > const_all_children_itors
auto all_children_view() const
In-order iteration over all children.
child_itors child_range(std::string_view key)
std::string debug() const
config & add_child_at(std::string_view key, const config &val, std::size_t index)
boost::iterator_range< const_child_iterator > const_child_itors
virtual void play_slice()
virtual const gamemap & map() const =0
virtual const unit_map & units() const =0
Sort-of-Singleton that many classes, both GUI and non-GUI, use to access the game data.
bool invalidate(const map_location &loc)
Function to invalidate a specific tile for redrawing.
double turbo_speed() const
bool propagate_invalidation(const std::set< map_location > &locs)
If this set is partially invalidated, invalidate all its hexes.
bool tile_nearly_on_screen(const map_location &loc) const
Checks if location loc or one of the adjacent tiles is visible on screen.
const display_context & context() const
static display * get_singleton()
Returns the display object if a display object exists.
Easily build frame parameters with the serialized constructors.
Keep most parameters in a separate class to simplify the handling of the large number of parameters b...
void override(const std::chrono::milliseconds &duration, const std::string &highlight="", const std::string &blend_ratio="", color_t blend_color={0, 0, 0}, const std::string &offset="", const std::string &layer="", const std::string &modifiers="")
bool does_not_change() const
terrain_code get_terrain(const map_location &loc) const
Looks up terrain at a particular location.
Generic locator abstracting the location of an image.
const std::string & get_filename() const
const std::string & get_modifications() const
static rng & default_instance()
std::set< map_location > get_overlaped_hex(const frame_parameters &value, const map_location &src, const map_location &dst)
void start_animation(const std::chrono::milliseconds &start_time)
frame_parsed_parameters parameters_
bool need_minimal_update() const
particle(const std::chrono::milliseconds &start_time=std::chrono::milliseconds{0}, const frame_builder &builder=frame_builder())
void override(const std::chrono::milliseconds &start_time, const std::chrono::milliseconds &duration, const cycle_state cycles, const std::string &highlight="", const std::string &blend_ratio="", color_t blend_color={0, 0, 0}, const std::string &offset="", const std::string &layer="", const std::string &modifiers="")
void redraw(const frame_parameters &value, const map_location &src, const map_location &dst, halo::manager &halo_man)
bool invalidate(frame_parameters &value)
std::vector< config > primary_attack_filter_
std::vector< map_location::direction > directions_
std::vector< int > value2_
bool need_minimal_update() const
bool animation_finished_potential() const
std::vector< config > secondary_unit_filter_
t_translation::ter_list terrain_types_
void add_frame(const std::chrono::milliseconds &duration, const unit_frame &value, bool force_change=false)
void start_animation(const std::chrono::milliseconds &start_time, const map_location &src=map_location::null_location(), const map_location &dst=map_location::null_location(), const std::string &text="", const color_t text_color={0, 0, 0}, const bool accelerate=true)
std::chrono::milliseconds get_begin_time() const
std::vector< std::string > event_
int matches(const map_location &loc, const map_location &second_loc, const unit_const_ptr &my_unit, const std::string &event="", const int value=0, strike_result::type hit=strike_result::type::invalid, const const_attack_ptr &attack=nullptr, const const_attack_ptr &second_attack=nullptr, int value2=0) const
std::vector< int > value_
std::string debug() const
void update_last_draw_time()
std::map< std::string, particle > sub_anims_
void update_parameters(const map_location &src, const map_location &dst)
std::chrono::milliseconds get_end_time() const
auto get_current_frame_begin_time() const
void redraw(frame_parameters &value, halo::manager &halo_man)
std::set< map_location > overlaped_hex_
auto get_animation_time() const
friend std::ostream & operator<<(std::ostream &outstream, const unit_animation &u_animation)
static void fill_initial_animations(std::vector< unit_animation > &animations, const config &cfg)
std::vector< config > secondary_attack_filter_
std::vector< config > unit_filter_
static void add_anims(std::vector< unit_animation > &animations, const config &cfg)
std::vector< strike_result::type > hits_
bool animation_finished() const
std::chrono::milliseconds get_animation_time_potential() const
std::chrono::milliseconds get_animation_time() const
bool has_animation(const unit_const_ptr &animated_unit, const std::string &event, const map_location &src=map_location::null_location(), const map_location &dst=map_location::null_location(), const int value=0, const strike_result::type hit_type=strike_result::type::invalid, const const_attack_ptr &attack=nullptr, const const_attack_ptr &second_attack=nullptr, int value2=0) const
has_animation : return an boolean value if animated unit present and have animation specified,...
void wait_for_end() const
void replace_anim_if_invalid(const unit_const_ptr &animated_unit, const std::string &event, const map_location &src=map_location::null_location(), const map_location &dst=map_location::null_location(), const int value=0, bool with_bars=false, const std::string &text="", const color_t text_color={0, 0, 0}, const strike_result::type hit_type=strike_result::type::invalid, const const_attack_ptr &attack=nullptr, const const_attack_ptr &second_attack=nullptr, int value2=0)
void add_animation(unit_const_ptr animated_unit, const unit_animation *animation, const map_location &src=map_location::null_location(), bool with_bars=false, const std::string &text="", const color_t text_color={0, 0, 0})
std::chrono::milliseconds get_end_time() const
void wait_until(const std::chrono::milliseconds &animation_time) const
Describes a unit's animation sequence.
void redraw(const std::chrono::milliseconds &frame_time, bool on_start_time, bool in_scope_of_frame, const map_location &src, const map_location &dst, halo::handle &halo_id, halo::manager &halo_man, const frame_parameters &animation_val, const frame_parameters &engine_val) const
const std::chrono::milliseconds & duration() const
bool does_not_change() const
std::vector< std::string > debug_strings() const
std::set< map_location > get_overlaped_hex(const std::chrono::milliseconds &frame_time, const map_location &src, const map_location &dst, const frame_parameters &animation_val, const frame_parameters &engine_val) const
unit_iterator find(std::size_t id)
This class represents a single unit of a specific type.
A variable-expanding proxy for the config class.
map_display and display: classes which take care of displaying the map and game-data on the screen.
@ unit_missile_default
Default layer for missile frames.
@ unit_default
Default layer for drawing units.
@ unit_move_default
Default layer for drawing moving units.
constexpr int get_abs_frame_layer(drawing_layer layer)
Functions to load and save images from/to disk.
play_controller * controller
bool terrain_matches(const terrain_code &src, const terrain_code &dest)
Tests whether a specific terrain matches an expression, for matching rules see above.
ter_list read_list(std::string_view str, const ter_layer filler)
Reads a list of terrains from a string, when reading the.
bool contains(const Container &container, const Value &value)
Returns true iff value is found in container.
std::string join(const T &v, const std::string &s=",")
Generates a new string joining container items in a list.
std::vector< std::string > split(const config_attribute_value &val)
std::shared_ptr< const unit > unit_const_ptr
std::shared_ptr< const attack_type > const_attack_ptr
rect dst
Location on the final composed sheet.
rect src
Non-transparent portion of the surface to compose.
std::vector< config::const_all_children_iterator > children
animation_branches branches
animation_cursor(const config &cfg)
animation_cursor(const config &cfg, animation_cursor *p)
config::const_all_children_itors itors
animation_cursor * parent
The basic class for representing 8-bit RGB or RGBA colour values.
All parameters from a frame at a given instant.
boost::tribool primary_frame
Encapsulates the map of the game.
static std::string write_direction(direction dir)
direction
Valid directions which can be moved in our hexagonal world.
static direction parse_direction(const std::string &str)
static std::string get_string(enum_type key)
Converts a enum to its string equivalent.