# pragma region License
/*
License ( OLC - 3 )
~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~ ~
Copyright 2024 Joshua Sigona < sigonasr2 @ gmail . com >
Redistribution and use in source and binary forms , with or without modification ,
are permitted provided that the following conditions are met :
1. Redistributions or derivations of source code must retain the above copyright
notice , this list of conditions and the following disclaimer .
2. Redistributions or derivative works in binary form must reproduce the above
copyright notice . This list of conditions and the following disclaimer must be
reproduced in the documentation and / or other materials provided with the distribution .
3. Neither the name of the copyright holder nor the names of its contributors may
be used to endorse or promote products derived from this software without specific
prior written permission .
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS " AS IS " AND ANY
EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE IMPLIED WARRANTIES
OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED . IN NO EVENT
SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT , INDIRECT ,
INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT NOT LIMITED
TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE , DATA , OR PROFITS ; OR
BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY , WHETHER IN
CONTRACT , STRICT LIABILITY , OR TORT ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN
ANY WAY OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF
SUCH DAMAGE .
Portions of this software are copyright <EFBFBD> 2024 The FreeType
Project ( www . freetype . org ) . Please see LICENSE_FT . txt for more information .
All rights reserved .
*/
# pragma endregion
# include "Monster.h"
# include "MonsterStrategyHelpers.h"
# include "AdventuresInLestoria.h"
# include "DEFINES.h"
# include "util.h"
# include "BulletTypes.h"
INCLUDE_game
using A = Attribute ;
void Monster : : STRATEGY : : STONE_GOLEM ( Monster & m , float fElapsedTime , std : : string strategy ) {
enum PhaseName {
INITIALIZE ,
SPAWN_PILLAR_PREPARE ,
SPAWN_PILLAR_CAST ,
STANDARD ,
STONE_THROW_CAST ,
STONE_THROW_FINISH_ANIMATION ,
} ;
switch ( m . phase ) {
case INITIALIZE : {
m . F ( A : : RECOVERY_TIME ) = ConfigFloat ( " Beginning Phase.Pillar Cast Delay Time " ) ;
m . I ( A : : PATTERN_REPEAT_COUNT ) = ConfigInt ( " Beginning Phase.Repeat Count " ) ;
m . phase = SPAWN_PILLAR_PREPARE ;
} break ;
case SPAWN_PILLAR_PREPARE : {
m . F ( A : : RECOVERY_TIME ) - = fElapsedTime ;
if ( m . F ( A : : RECOVERY_TIME ) < = 0.f ) {
m . V ( A : : LOCKON_POS ) = game - > GetPlayer ( ) - > GetPos ( ) ;
m . PerformAnimation ( " CAST " , m . GetFacingDirectionToTarget ( m . V ( A : : LOCKON_POS ) ) ) ;
game - > AddEffect ( std : : make_unique < SpellCircle > ( m . V ( A : : LOCKON_POS ) , ConfigFloat ( " Beginning Phase.Pillar Cast Time " ) , " range_indicator.png " , " spell_insignia.png " , m . OnUpperLevel ( ) , vf2d { 1.f , 1.f } * ( MONSTER_DATA . at ( " Stone Golem Pillar " ) . GetCollisionRadius ( ) * MONSTER_DATA . at ( " Stone Golem Pillar " ) . GetSizeMult ( ) / 12.f ) * 1.25f , 0.3f , vf2d { } , ConfigPixel ( " Beginning Phase.Pillar Spell Circle Color " ) , util : : random ( 2 * PI ) , util : : degToRad ( ConfigFloat ( " Beginning Phase.Pillar Spell Circle Rotation Spd " ) ) , false , vf2d { 1.f , 1.f } * ( MONSTER_DATA . at ( " Stone Golem Pillar " ) . GetCollisionRadius ( ) * MONSTER_DATA . at ( " Stone Golem Pillar " ) . GetSizeMult ( ) / 12.f ) * 0.9f , 0.3f , vf2d { } , ConfigPixel ( " Beginning Phase.Pillar Spell Insignia Color " ) , util : : random ( 2 * PI ) , util : : degToRad ( ConfigFloat ( " Beginning Phase.Pillar Spell Insignia Rotation Spd " ) ) ) , true ) ;
m . F ( A : : CASTING_TIMER ) = ConfigFloat ( " Beginning Phase.Pillar Cast Time " ) ;
m . phase = SPAWN_PILLAR_CAST ;
}
} break ;
case SPAWN_PILLAR_CAST : {
m . F ( A : : CASTING_TIMER ) - = fElapsedTime ;
if ( m . F ( A : : CASTING_TIMER ) < = 0.f ) {
m . I ( A : : PATTERN_REPEAT_COUNT ) - - ;
game - > SpawnMonster ( m . V ( A : : LOCKON_POS ) , MONSTER_DATA . at ( " Stone Golem Pillar " ) , m . OnUpperLevel ( ) ) ;
game - > Hurt ( m . V ( A : : LOCKON_POS ) , MONSTER_DATA . at ( " Stone Golem Pillar " ) . GetCollisionRadius ( ) * MONSTER_DATA . at ( " Stone Golem Pillar " ) . GetSizeMult ( ) , MONSTER_DATA . at ( " Stone Golem Pillar " ) . GetAttack ( ) , m . OnUpperLevel ( ) , 0.f , HurtType : : PLAYER ) ;
if ( m . I ( A : : PATTERN_REPEAT_COUNT ) < = 0 ) {
m . phase = STANDARD ;
} else {
m . PerformIdleAnimation ( m . GetFacingDirectionToTarget ( game - > GetPlayer ( ) - > GetPos ( ) ) ) ;
m . F ( A : : RECOVERY_TIME ) = ConfigFloat ( " Beginning Phase.Pillar Cast Delay Time " ) ;
m . phase = SPAWN_PILLAR_PREPARE ;
}
}
} break ;
case STANDARD : {
BEAR ( m , fElapsedTime , " Bear " ) ;
//Extending the bear script's variables to read the state of it...
const bool SlamHasFinished = m . I ( A : : ATTACK_COUNT ) ! = m . I ( A : : BEAR_STOMP_COUNT ) ;
if ( SlamHasFinished ) {
const bool StoneThrowRollSucceeds = util : : random ( 100.f ) < = ConfigFloat ( " Standard Attack.Stone Throw Chance " ) ;
m . I ( A : : ATTACK_COUNT ) = m . I ( A : : BEAR_STOMP_COUNT ) ; //Make sure the slams are now reset if necessary.
if ( StoneThrowRollSucceeds ) { //The intent is one or the other attack is supposed to happen. We can't do the slam and a throw, rerolling repeatedly each tick is unncessary.
m . phase = STONE_THROW_CAST ;
m . V ( A : : LOCKON_POS ) = game - > GetPlayer ( ) - > GetPos ( ) ;
m . PerformAnimation ( " TOSS ROCK CAST " ) ;
m . F ( A : : CASTING_TIMER ) = ConfigFloat ( " Standard Attack.Stone Throw Cast Time " ) ;
game - > AddEffect ( std : : make_unique < SpellCircle > ( m . V ( A : : LOCKON_POS ) , ConfigFloat ( " Standard Attack.Stone Throw Cast Time " ) , " range_indicator.png " , " spell_insignia.png " , m . OnUpperLevel ( ) , vf2d { 1.f , 1.f } * ( ConfigPixels ( " Standard Attack.Stone Radius " ) / 12.f ) * 1.25f , 0.3f , vf2d { } , ConfigPixel ( " Standard Attack.Stone Throw Spell Circle Color " ) , util : : random ( 2 * PI ) , util : : degToRad ( ConfigFloat ( " Standard Attack.Stone Throw Spell Circle Rotation Spd " ) ) , false , vf2d { 1.f , 1.f } * ( ConfigPixels ( " Standard Attack.Stone Radius " ) / 12.f ) * 0.9f , 0.3f , vf2d { } , ConfigPixel ( " Standard Attack.Stone Throw Spell Insignia Color " ) , util : : random ( 2 * PI ) , util : : degToRad ( ConfigFloat ( " Standard Attack.Stone Throw Spell Insignia Rotation Spd " ) ) ) , true ) ;
//Use acceleration equation to determine how much time it takes for the stone to land based on gravity.
const float stoneTossTime { ConfigFloat ( " Standard Attack.Stone Throw Time " ) } ;
//Physics!! Kinematic equation from https://openstax.org/books/physics/pages/3-2-representing-acceleration-with-equations-and-graphs a=(2d)/(t^2)
const float acc { ( 2 * - ConfigFloat ( " Standard Attack.Stone Throw Height Offset " ) ) / std : : pow ( stoneTossTime , 2.f ) } ;
CreateBullet ( LargeStone ) ( m . GetPos ( ) + vf2d { 0 , ConfigFloat ( " Standard Attack.Stone Throw Height Offset " ) / 2.f } , ConfigFloat ( " Standard Attack.Stone Throw Time " ) , m . V ( A : : LOCKON_POS ) , m . F ( A : : CASTING_TIMER ) , ConfigPixels ( " Standard Attack.Stone Radius " ) , ConfigFloat ( " Standard Attack.Stone Throw Height Offset " ) , acc , ConfigInt ( " Standard Attack.Stone Damage " ) , ConfigFloat ( " Standard Attack.Stone Throw Knockback Factor " ) , m . OnUpperLevel ( ) , false , INFINITY , false , WHITE , vf2d { 1 , 1 } * m . GetSizeMult ( ) , util : : random ( 2 * PI ) ) EndBullet ;
}
}
} break ;
case STONE_THROW_CAST : {
m . F ( A : : CASTING_TIMER ) - = fElapsedTime ;
if ( m . F ( A : : CASTING_TIMER ) < = 0.f ) {
m . PerformAnimation ( " TOSS ROCK " ) ;
m . F ( A : : RECOVERY_TIME ) = m . GetCurrentAnimation ( ) . GetTotalAnimationDuration ( ) ;
m . phase = STONE_THROW_FINISH_ANIMATION ;
}
} break ;
case STONE_THROW_FINISH_ANIMATION : {
m . F ( A : : RECOVERY_TIME ) - = fElapsedTime ;
if ( m . F ( A : : RECOVERY_TIME ) < = 0.f ) {
m . phase = STANDARD ;
}
} break ;
}
}