Animations now have an easing/ease_rate setting that will do a dynamic scaling effect on them during the animation sequence.
This commit is contained in:
		
							parent
							
								
									6e363ba78d
								
							
						
					
					
						commit
						033358749f
					
				
					 8 changed files with 42 additions and 14 deletions
				
			
		|  | @ -53,7 +53,7 @@ | ||||||
|   "player": { |   "player": { | ||||||
|   }, |   }, | ||||||
|   "worldgen": { |   "worldgen": { | ||||||
|     "enemy_probability": 50, |     "enemy_probability": 80, | ||||||
|     "empty_room_probability": 10, |     "empty_room_probability": 10, | ||||||
|     "device_probability": 10 |     "device_probability": 10 | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  | @ -20,7 +20,7 @@ | ||||||
|       {"_type": "Combat", "hp": 20, "max_hp": 20, "damage": 1, "dead": false}, |       {"_type": "Combat", "hp": 20, "max_hp": 20, "damage": 1, "dead": false}, | ||||||
|       {"_type": "Motion", "dx": 0, "dy": 0, "random": false}, |       {"_type": "Motion", "dx": 0, "dy": 0, "random": false}, | ||||||
|       {"_type": "EnemyConfig", "hearing_distance": 5}, |       {"_type": "EnemyConfig", "hearing_distance": 5}, | ||||||
|       {"_type": "Animation", "scale": 0.1, "simple": true, "frames": 10, "speed": 0.3}, |       {"_type": "Animation", "easing": 1, "ease_rate": 0.2,  "scale": 0.1, "simple": true, "frames": 10, "speed": 0.3}, | ||||||
|       {"_type": "Sprite", "name": "armored_knight"}, |       {"_type": "Sprite", "name": "armored_knight"}, | ||||||
|       {"_type": "Sound", "attack": "Sword_Hit_2", "death": "Humanoid_Death_1"} |       {"_type": "Sound", "attack": "Sword_Hit_2", "death": "Humanoid_Death_1"} | ||||||
|     ] |     ] | ||||||
|  | @ -35,7 +35,7 @@ | ||||||
|       {"_type": "Motion", "dx": 0, "dy": 0, "random": true}, |       {"_type": "Motion", "dx": 0, "dy": 0, "random": true}, | ||||||
|       {"_type": "EnemyConfig", "hearing_distance": 5}, |       {"_type": "EnemyConfig", "hearing_distance": 5}, | ||||||
|       {"_type": "Sprite", "name": "axe_ranger"}, |       {"_type": "Sprite", "name": "axe_ranger"}, | ||||||
|       {"_type": "Animation", "scale": 0.1, "simple": false, "frames": 10, "speed": 0.6}, |       {"_type": "Animation", "easing": 3, "ease_rate": 0.5,  "scale": 0.1, "simple": false, "frames": 10, "speed": 0.6}, | ||||||
|       {"_type": "Sound", "attack": "Sword_Hit_2", "death": "Ranger_1"} |       {"_type": "Sound", "attack": "Sword_Hit_2", "death": "Ranger_1"} | ||||||
|     ] |     ] | ||||||
|   }, |   }, | ||||||
|  | @ -49,7 +49,7 @@ | ||||||
|       {"_type": "Motion", "dx": 0, "dy": 0, "random": false}, |       {"_type": "Motion", "dx": 0, "dy": 0, "random": false}, | ||||||
|       {"_type": "EnemyConfig", "hearing_distance": 5}, |       {"_type": "EnemyConfig", "hearing_distance": 5}, | ||||||
|       {"_type": "Sprite", "name": "evil_eye"}, |       {"_type": "Sprite", "name": "evil_eye"}, | ||||||
|       {"_type": "Animation", "scale": 0.1, "simple": false, "frames": 10, "speed": 0.3}, |       {"_type": "Animation", "easing": 3, "ease_rate": 0.1,  "scale": 0.1, "simple": false, "frames": 10, "speed": 0.3}, | ||||||
|       {"_type": "Sound", "attack": "Evil_Eye_Sound_2", "death": "Evil_Eye_Sound_1"} |       {"_type": "Sound", "attack": "Evil_Eye_Sound_2", "death": "Evil_Eye_Sound_1"} | ||||||
|     ] |     ] | ||||||
|   }, |   }, | ||||||
|  | @ -62,7 +62,7 @@ | ||||||
|       {"_type": "Combat", "hp": 20, "max_hp": 20, "damage": 20, "dead": false}, |       {"_type": "Combat", "hp": 20, "max_hp": 20, "damage": 20, "dead": false}, | ||||||
|       {"_type": "Motion", "dx": 0, "dy": 0, "random": false}, |       {"_type": "Motion", "dx": 0, "dy": 0, "random": false}, | ||||||
|       {"_type": "EnemyConfig", "hearing_distance": 10}, |       {"_type": "EnemyConfig", "hearing_distance": 10}, | ||||||
|       {"_type": "Animation", "scale": 0.1, "simple": true, "frames": 10, "speed": 1.0}, |       {"_type": "Animation", "easing": 3, "ease_rate": 0.5,  "scale": 0.1, "simple": true, "frames": 10, "speed": 1.0}, | ||||||
|       {"_type": "Sprite", "name": "rat_with_sword"}, |       {"_type": "Sprite", "name": "rat_with_sword"}, | ||||||
|       {"_type": "Sound", "attack": "Small_Rat", "death": "Creature_Death_1"} |       {"_type": "Sound", "attack": "Small_Rat", "death": "Creature_Death_1"} | ||||||
|     ] |     ] | ||||||
|  | @ -76,7 +76,7 @@ | ||||||
|       {"_type": "Combat", "hp": 50, "max_hp": 50, "damage": 50, "dead": false}, |       {"_type": "Combat", "hp": 50, "max_hp": 50, "damage": 50, "dead": false}, | ||||||
|       {"_type": "Motion", "dx": 0, "dy": 0, "random": false}, |       {"_type": "Motion", "dx": 0, "dy": 0, "random": false}, | ||||||
|       {"_type": "EnemyConfig", "hearing_distance": 3}, |       {"_type": "EnemyConfig", "hearing_distance": 3}, | ||||||
|       {"_type": "Animation", "scale": 0.1, "simple": true, "frames": 10, "speed": 1.0}, |       {"_type": "Animation", "easing": 2, "ease_rate": 0.2,  "scale": 0.1, "simple": true, "frames": 10, "speed": 1.0}, | ||||||
|       {"_type": "Sprite", "name": "rat_king"}, |       {"_type": "Sprite", "name": "rat_king"}, | ||||||
|       {"_type": "Sound", "attack": "Medium_Rat", "death": "Creature_Death_1"} |       {"_type": "Sound", "attack": "Medium_Rat", "death": "Creature_Death_1"} | ||||||
|     ] |     ] | ||||||
|  | @ -90,7 +90,7 @@ | ||||||
|       {"_type": "Combat", "hp": 20, "max_hp": 20, "damage": 20, "dead": false}, |       {"_type": "Combat", "hp": 20, "max_hp": 20, "damage": 20, "dead": false}, | ||||||
|       {"_type": "Motion", "dx": 0, "dy": 0, "random": false}, |       {"_type": "Motion", "dx": 0, "dy": 0, "random": false}, | ||||||
|       {"_type": "EnemyConfig", "hearing_distance": 10}, |       {"_type": "EnemyConfig", "hearing_distance": 10}, | ||||||
|       {"_type": "Animation", "scale": 0.1, "simple": true, "frames": 10, "speed": 1.0}, |       {"_type": "Animation", "easing": 2, "ease_rate": 0.5,  "scale": 0.1, "simple": true, "frames": 10, "speed": 1.0}, | ||||||
|       {"_type": "Sprite", "name": "hairy_spider"}, |       {"_type": "Sprite", "name": "hairy_spider"}, | ||||||
|       {"_type": "Sound", "attack": "Spider_1", "death": "Spider_2"} |       {"_type": "Sound", "attack": "Spider_1", "death": "Spider_2"} | ||||||
|     ] |     ] | ||||||
|  |  | ||||||
|  | @ -1,5 +1,6 @@ | ||||||
| #include "components.hpp" | #include "components.hpp" | ||||||
| #include "point.hpp" | #include "point.hpp" | ||||||
|  | #include "easings.hpp" | ||||||
| 
 | 
 | ||||||
| namespace components { | namespace components { | ||||||
|   ENROLL_COMPONENT(Position, location.x, location.y); |   ENROLL_COMPONENT(Position, location.x, location.y); | ||||||
|  | @ -7,7 +8,7 @@ namespace components { | ||||||
|   ENROLL_COMPONENT(Motion, dx, dy, random); |   ENROLL_COMPONENT(Motion, dx, dy, random); | ||||||
|   ENROLL_COMPONENT(Combat, hp, max_hp, damage, dead); |   ENROLL_COMPONENT(Combat, hp, max_hp, damage, dead); | ||||||
|   ENROLL_COMPONENT(Device, config, events); |   ENROLL_COMPONENT(Device, config, events); | ||||||
|   ENROLL_COMPONENT(Animation, scale, simple, frames, speed); |   ENROLL_COMPONENT(Animation, scale, simple, frames, speed, easing, ease_rate); | ||||||
|   ENROLL_COMPONENT(Sound, attack, death); |   ENROLL_COMPONENT(Sound, attack, death); | ||||||
| 
 | 
 | ||||||
|   void configure_entity(const ComponentMap& component_map, DinkyECS::World& world, DinkyECS::Entity ent, json& data) { |   void configure_entity(const ComponentMap& component_map, DinkyECS::World& world, DinkyECS::Entity ent, json& data) { | ||||||
|  | @ -41,11 +42,28 @@ namespace components { | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   float Animation::twitching() { | ||||||
|  |     switch(easing) { | ||||||
|  |       case ease::SINE: | ||||||
|  |         return ease::sine(float(frames) / subframe * ease_rate); | ||||||
|  |       case ease::OUT_CIRC: | ||||||
|  |         return ease::sine(ease::out_circ(float(frames) / subframe * ease_rate)); | ||||||
|  |       case ease::OUT_BOUNCE: | ||||||
|  |         return ease::sine(ease::out_bounce(float(frames) / subframe * ease_rate)); | ||||||
|  |       case ease::IN_OUT_BACK: | ||||||
|  |         return ease::sine(ease::in_out_back(float(frames) / subframe * ease_rate)); | ||||||
|  |       default: | ||||||
|  |         dbc::sentinel( | ||||||
|  |             fmt::format("Invalid easing {} given to animation", | ||||||
|  |               int(easing))); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   void Animation::step(sf::Vector2f& scale_out, sf::IntRect& rect_out) { |   void Animation::step(sf::Vector2f& scale_out, sf::IntRect& rect_out) { | ||||||
|     if(playing && current < frames) { |     if(playing && current < frames) { | ||||||
|       scale_out.x += scale; |       float tick = twitching(); | ||||||
|       scale_out.y += scale; |       scale_out.x = std::lerp(scale_out.x, scale_out.x + scale, tick); | ||||||
| 
 |       scale_out.y = std::lerp(scale_out.y, scale_out.y + scale, tick); | ||||||
|       if(!simple) { |       if(!simple) { | ||||||
|         rect_out.position.x += current * TEXTURE_WIDTH; |         rect_out.position.x += current * TEXTURE_WIDTH; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  | @ -9,6 +9,7 @@ | ||||||
| #include <functional> | #include <functional> | ||||||
| #include <nlohmann/json.hpp> | #include <nlohmann/json.hpp> | ||||||
| #include <nlohmann/json_fwd.hpp> | #include <nlohmann/json_fwd.hpp> | ||||||
|  | #include "easings.hpp" | ||||||
| 
 | 
 | ||||||
| #define ENROLL_COMPONENT(COMPONENT, ...)                         \ | #define ENROLL_COMPONENT(COMPONENT, ...)                         \ | ||||||
|     NLOHMANN_DEFINE_TYPE_NON_INTRUSIVE(COMPONENT, __VA_ARGS__);  \ |     NLOHMANN_DEFINE_TYPE_NON_INTRUSIVE(COMPONENT, __VA_ARGS__);  \ | ||||||
|  | @ -105,8 +106,11 @@ namespace components { | ||||||
|     int current = 0; |     int current = 0; | ||||||
|     bool playing = false; |     bool playing = false; | ||||||
|     float subframe = 0; |     float subframe = 0; | ||||||
|  |     ease::Style easing = ease::IN_OUT_BACK; | ||||||
|  |     float ease_rate = 0.5f; | ||||||
| 
 | 
 | ||||||
|     void play(); |     void play(); | ||||||
|  |     float twitching(); | ||||||
|     void step(sf::Vector2f& scale_out, sf::IntRect& rect_out); |     void step(sf::Vector2f& scale_out, sf::IntRect& rect_out); | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										2
									
								
								dbc.hpp
									
										
									
									
									
								
							
							
						
						
									
										2
									
								
								dbc.hpp
									
										
									
									
									
								
							|  | @ -20,7 +20,7 @@ namespace dbc { | ||||||
|   class PostCondError : public Error {}; |   class PostCondError : public Error {}; | ||||||
| 
 | 
 | ||||||
|   void log(const string &message); |   void log(const string &message); | ||||||
|   void sentinel(const string &message); |   [[noreturn]] void sentinel(const string &message); | ||||||
|   void pre(const string &message, bool test); |   void pre(const string &message, bool test); | ||||||
|   void pre(const string &message, std::function<bool()> tester); |   void pre(const string &message, std::function<bool()> tester); | ||||||
|   void post(const string &message, bool test); |   void post(const string &message, bool test); | ||||||
|  |  | ||||||
|  | @ -3,6 +3,10 @@ | ||||||
| 
 | 
 | ||||||
| namespace ease { | namespace ease { | ||||||
| 
 | 
 | ||||||
|  |   enum Style { | ||||||
|  |     SINE, OUT_CIRC, OUT_BOUNCE, IN_OUT_BACK, NONE | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|   inline double sine(double x) { |   inline double sine(double x) { | ||||||
|     return (std::sin(x) + 1.0) / 2.0; |     return (std::sin(x) + 1.0) / 2.0; | ||||||
|   } |   } | ||||||
|  |  | ||||||
							
								
								
									
										2
									
								
								main.cpp
									
										
									
									
									
								
							
							
						
						
									
										2
									
								
								main.cpp
									
										
									
									
									
								
							|  | @ -6,7 +6,7 @@ | ||||||
| int main(int argc, char* argv[]) { | int main(int argc, char* argv[]) { | ||||||
|   textures::init(); |   textures::init(); | ||||||
|   sound::init(); |   sound::init(); | ||||||
|   sound::mute(false); |   sound::mute(true); | ||||||
|   gui::FSM main; |   gui::FSM main; | ||||||
|   main.event(gui::Event::STARTED); |   main.event(gui::Event::STARTED); | ||||||
|   Autowalker walker(main); |   Autowalker walker(main); | ||||||
|  |  | ||||||
|  | @ -151,8 +151,9 @@ void Raycaster::sprite_casting(sf::RenderTarget &target) { | ||||||
|       int d = y * texture_height - $height * half_height + sprite_height * half_height; |       int d = y * texture_height - $height * half_height + sprite_height * half_height; | ||||||
|       int tex_y = ((d * texture_height) / sprite_height) / texture_height; |       int tex_y = ((d * texture_height) / sprite_height) / texture_height; | ||||||
| 
 | 
 | ||||||
|  |       sf::Vector2f origin{texture_width / 2, texture_height / 2}; | ||||||
|       sf::Vector2f scale{sprite_scale_w, sprite_scale_h}; |       sf::Vector2f scale{sprite_scale_w, sprite_scale_h}; | ||||||
|       sf::Vector2f position{x, y}; |       sf::Vector2f position{x + origin.x * scale.x, y + origin.y * scale.y}; | ||||||
|       sf::IntRect in_texture{ {tex_x, tex_y}, {tex_render_width, texture_height}}; |       sf::IntRect in_texture{ {tex_x, tex_y}, {tex_render_width, texture_height}}; | ||||||
| 
 | 
 | ||||||
|       if($level.world->has<components::Animation>(rec.second)) { |       if($level.world->has<components::Animation>(rec.second)) { | ||||||
|  | @ -160,6 +161,7 @@ void Raycaster::sprite_casting(sf::RenderTarget &target) { | ||||||
|         if(animation.playing) animation.step(scale, in_texture); |         if(animation.playing) animation.step(scale, in_texture); | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|  |       sf_sprite->setOrigin(origin); | ||||||
|       sf_sprite->setScale(scale); |       sf_sprite->setScale(scale); | ||||||
|       sf_sprite->setTextureRect(in_texture); |       sf_sprite->setTextureRect(in_texture); | ||||||
|       sf_sprite->setPosition(position); |       sf_sprite->setPosition(position); | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Zed A. Shaw
						Zed A. Shaw