/* Copyright (c) 2002-2012 Croteam Ltd. 
This program is free software; you can redistribute it and/or modify
it under the terms of version 2 of the GNU General Public License as published by
the Free Software Foundation


This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License along
with this program; if not, write to the Free Software Foundation, Inc.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */

507
%{
#include "EntitiesMP/StdH/StdH.h"
#include "ModelsMP/Enemies/AirElemental/Twister.h"

#define ECF_TWISTER ( \
  ((ECBI_BRUSH|ECBI_MODEL|ECBI_CORPSE|ECBI_ITEM|ECBI_PROJECTILE_MAGIC|ECBI_PROJECTILE_SOLID)<<ECB_TEST) |\
  ((ECBI_MODEL|ECBI_CORPSE|ECBI_ITEM|ECBI_PROJECTILE_MAGIC|ECBI_PROJECTILE_SOLID)<<ECB_PASS) |\
  ((ECBI_MODEL)<<ECB_IS))
#define EPF_TWISTER ( \
  EPF_ONBLOCK_CLIMBORSLIDE|EPF_ORIENTEDBYGRAVITY|\
  EPF_TRANSLATEDBYGRAVITY|EPF_MOVABLE|EPF_ABSOLUTETRANSLATE)
%}


uses "EntitiesMP/AirElemental";
uses "EntitiesMP/Elemental";
uses "EntitiesMP/Spinner";


// input parameter for twister
event ETwister {
  CEntityPointer penOwner,        // entity which owns it
  FLOAT fSize,                    // twister size
  FLOAT fDuration,                // twister duration
  INDEX sgnSpinDir,               // spin direction
  BOOL  bGrow,                    // grow from miniature twister to full size?
  BOOL  bMovingAllowed,           // if moving is allowed
};


%{
static EntityInfo eiTwister = {
  EIBT_AIR, 0.0f,
  0.0f, 1.0f, 0.0f,
  0.0f, 0.75f, 0.0f,
};


#define MOVE_FREQUENCY 0.1f
#define ROTATE_SPEED 10000.0f
#define MOVE_SPEED 7.5f

void CTwister_OnPrecache(CDLLEntityClass *pdec, INDEX iUser) 
{
  pdec->PrecacheClass(CLASS_SPINNER);
  pdec->PrecacheModel(MODEL_TWISTER);
  pdec->PrecacheTexture(TEXTURE_TWISTER);
  pdec->PrecacheSound(SOUND_SPIN);
}
%}

class CTwister : CMovableModelEntity {
name      "Twister";
thumbnail "";
features  "ImplementsOnPrecache";

properties:
  1 CEntityPointer m_penOwner,                  // entity which owns it
  2 FLOAT   m_fSize = 1.0f,                     // size
  3 FLOAT3D m_vSpeed = FLOAT3D(0,0,0),          // current speed
  4 INDEX   m_sgnSpinDir = 1,                   // spin clockwise
  5 BOOL    m_bGrow = TRUE,                     // grow to full size?
  6 FLOAT   m_tmLastMove = 0.0f,                // when moving has started
  7 FLOAT3D m_aSpeedRotation = FLOAT3D(0,0,0),
  8 BOOL    m_bMoving = FALSE,
  9 BOOL    m_bMovingAllowed = TRUE,
  
 // internal -> do not use
 10 FLOAT3D m_vDesiredPosition = FLOAT3D(0,0,0),
 11 FLOAT3D m_vDesiredAngle = FLOAT3D(0,0,0),
 12 FLOAT m_fStopTime = 0.0f,
 13 FLOAT m_fActionRadius = 0.0f,
 14 FLOAT m_fActionTime = 0.0f,
 15 FLOAT m_fDiffMultiply = 0.0f,
 16 FLOAT m_fUpMultiply = 0.0f,
 20 BOOL  m_bFadeOut = FALSE,
 21 FLOAT m_fFadeStartTime = 1e6,
 22 FLOAT m_fFadeTime = 2.0f,
 23 FLOAT m_fStartTime = 0.0f,

 50 CSoundObject m_soSpin,  // sound channel for spinning

components:

  1 class   CLASS_SPINNER         "Classes\\Spinner.ecl",
 
 10 model   MODEL_TWISTER         "ModelsMP\\Enemies\\AirElemental\\Twister.mdl",
 11 texture TEXTURE_TWISTER       "ModelsMP\\Enemies\\AirElemental\\Twister.tex",

200 sound   SOUND_SPIN            "ModelsMP\\Enemies\\AirElemental\\Sounds\\TwisterSpin.wav",

functions:
  /* Entity info */
  void *GetEntityInfo(void) {
    return &eiTwister;
  };

  /* Receive damage */
  void ReceiveDamage(CEntity *penInflictor, enum DamageType dmtType,
    FLOAT fDamageAmmount, const FLOAT3D &vHitPoint, const FLOAT3D &vDirection) 
  {
    return;
  };

  // render burning particles
  void RenderParticles(void)
  {
    if(m_bMovingAllowed)
    {
      Particles_Twister(this, m_fSize/15.0f, m_fStartTime, m_fFadeStartTime, 1.0f);
    }
    else
    {
      CEntity *penParent=GetParent();
      FLOAT fStretch=1.0f;
      if(penParent!=NULL)
      {
        CAirElemental *penAir=(CAirElemental *)penParent;
        FLOAT fStretchRatio=penAir->GetCurrentStretchRatio();
        fStretch=1.0f+(fStretchRatio)*6.0f;
      }
      Particles_Twister(this, m_fSize/15.0f*fStretch, m_fStartTime, m_fFadeStartTime, 0.5f*fStretch);
    }
  }

/************************************************************
 *                   FADE OUT & MOVING                      *
 ************************************************************/
  BOOL AdjustShadingParameters(FLOAT3D &vLightDirection, COLOR &colLight, COLOR &colAmbient) {
    // fading out
    if (m_bFadeOut) {
      FLOAT fTimeRemain = m_fFadeStartTime + m_fFadeTime - _pTimer->CurrentTick();
      if (fTimeRemain < 0.0f) { fTimeRemain = 0.0f; }
      COLOR colAlpha = GetModelObject()->mo_colBlendColor;
      colAlpha = (colAlpha&0xffffff00) + (COLOR(fTimeRemain/m_fFadeTime*0xff)&0xff);
      GetModelObject()->mo_colBlendColor = colAlpha;
    }
    return CMovableModelEntity::AdjustShadingParameters(vLightDirection, colLight, colAmbient);
  };


/************************************************************
 *                   ATTACK SPECIFIC                        *
 ************************************************************/
  
  void SpinEntity(CEntity *pen) {
    
    // don't spin air elemental and other twisters and any items
    if (IsOfClass(pen, "AirElemental") || IsOfClass(pen, "Twister")
        || IsDerivedFromClass(pen, "Item")) {
      return;
    }
    // don't spin air elementals wind blast
    if (IsOfClass(pen, "Projectile")) {
      if (((CProjectile *)&*pen)->m_prtType==PRT_AIRELEMENTAL_WIND)
      {
        return; 
      }
    }
    


    if (pen->GetPhysicsFlags()&EPF_MOVABLE) {
      // if any other spinner affects the target, skip this spinner
      BOOL bNoSpinner = TRUE;
      {FOREACHINLIST( CEntity, en_lnInParent, pen->en_lhChildren, iten) {
        if (IsOfClass(iten, "Spinner"))
        {
          bNoSpinner = FALSE;
          return;      
        }
      }}
      if (bNoSpinner) {    
        ESpinnerInit esi;
        CEntityPointer penSpinner;
        esi.penParent = pen;
        esi.penTwister = this;
        esi.bImpulse = FALSE;

        // spin projectiles a bit longer but not so high
        if (IsOfClass(pen, "Projectile"))
        {
          switch(((CProjectile &)*pen).m_prtType) {
          case PRT_GRENADE:
          case PRT_HEADMAN_BOMBERMAN:
          case PRT_DEMON_FIREBALL:
          case PRT_SHOOTER_FIREBALL:
          case PRT_BEAST_PROJECTILE:
          case PRT_BEAST_BIG_PROJECTILE:
          case PRT_LAVA_COMET:
            esi.tmSpinTime = 2.5f;
            esi.vRotationAngle = ANGLE3D(-m_sgnSpinDir*250.0f, 0, 0);
            esi.fUpSpeed = m_fDiffMultiply*0.75;
            break;
          default:
            esi.tmSpinTime = 1.5f;
            esi.vRotationAngle = ANGLE3D(-m_sgnSpinDir*180.0f, 0, 0);
            esi.fUpSpeed = m_fDiffMultiply/5.0f;
            break;
          }
        // cannon ball - short but powerfull
        } else if (IsOfClass(pen, "Cannon ball")){
          esi.tmSpinTime = 0.2f;
          esi.vRotationAngle = ANGLE3D(-m_sgnSpinDir*500.0f, 0, 0);
          esi.fUpSpeed = m_fDiffMultiply*3.0f;          
        // don't take it easy with players
        } else if (IsOfClass(pen, "Player")){
          esi.tmSpinTime = 3.0f;
          esi.vRotationAngle = ANGLE3D(-m_sgnSpinDir*220.0f, 0, 0);
          esi.bImpulse = TRUE;
          esi.fUpSpeed = m_fDiffMultiply*(0.4f + FRnd()*0.4f);
          esi.tmImpulseDuration = 1.4f + FRnd()*0.5f;
        // everything else
        } else {
          esi.tmSpinTime = 0.5f;
          esi.vRotationAngle = ANGLE3D(-m_sgnSpinDir*180.0f, 0, 0);
          esi.fUpSpeed = m_fDiffMultiply;
        }
        penSpinner = CreateEntity(pen->GetPlacement(), CLASS_SPINNER);
        penSpinner->Initialize(esi);
        penSpinner->SetParent(pen);
      }
      // damage
      FLOAT3D vDirection;
      AnglesToDirectionVector(GetPlacement().pl_OrientationAngle, vDirection);
      InflictDirectDamage(pen, m_penOwner, DMT_IMPACT, 2.0f, GetPlacement().pl_PositionVector, vDirection);
    }
    
  };
  
  void PreMoving(void) {
    // moving - rotate speed direction
    if (m_bMoving) {
      FLOATmatrix3D m;
      ANGLE3D aRotation;
      aRotation = m_aSpeedRotation*(_pTimer->CurrentTick()-m_tmLastMove);
      MakeRotationMatrix(m, aRotation);
      m_vSpeed = m_vSpeed*m;
      SetDesiredTranslation(m_vSpeed);
      m_tmLastMove = _pTimer->CurrentTick();
    }
    CMovableModelEntity::PreMoving();
  };
 
  
/************************************************************
 *                   P R O C E D U R E S                    *
 ************************************************************/
procedures:
  // --->>> MAIN
  Main(ETwister et) {
    // remember the initial parameters
    ASSERT(et.penOwner!=NULL);
    m_penOwner = et.penOwner;
    m_sgnSpinDir = et.sgnSpinDir;
    if (m_sgnSpinDir==0) { m_sgnSpinDir=1; };
    m_fSize = et.fSize;
    m_fStopTime = _pTimer->CurrentTick() + et.fDuration;
    m_bGrow = et.bGrow;
    m_bMovingAllowed = et.bMovingAllowed;
    
    // initialization
    InitAsEditorModel();
    SetPhysicsFlags(EPF_TWISTER);
    SetCollisionFlags(ECF_TWISTER);
    SetFlags(GetFlags() | ENF_SEETHROUGH);
    SetModel(MODEL_TWISTER);
    SetModelMainTexture(TEXTURE_TWISTER);

    // some twister parameters
    m_fActionRadius = pow(m_fSize, 0.33333f)*10.0f;
    m_fActionTime = m_fActionRadius;
    m_fUpMultiply = m_fActionRadius/2.0f;
    m_fDiffMultiply = sqrt(m_fSize);
    GetModelObject()->StretchModel(FLOAT3D(m_fSize, m_fSize, m_fSize));
    ModelChangeNotify();

    m_fStartTime=_pTimer->CurrentTick();
    
    //wait for some randome time
    autowait(FRnd()*0.25f);
 
    m_soSpin.Set3DParameters(50.0f, 10.0f, 1.0f, 1.0f);
    PlaySound(m_soSpin, SOUND_SPIN, SOF_3D|SOF_LOOP);
    
    // immediately rotate
    SetDesiredRotation(ANGLE3D(m_sgnSpinDir*(FRnd()*50.0f+50.0f), 0.0f, 0.0f));
    
    if (m_bGrow) {
      StartModelAnim(TWISTER_ANIM_GROWING, AOF_SMOOTHCHANGE|AOF_NORESTART);
    }
    autowait(GetModelObject()->GetAnimLength(TWISTER_ANIM_GROWING));
    
    // beginning random speed
    FLOAT fR = FRndIn(5.0f, 10.0f);
    FLOAT fA = FRnd()*360.0f;
    m_vSpeed = FLOAT3D(CosFast(fA)*fR, 0, SinFast(fA)*fR);
    m_bMoving = m_bMovingAllowed;

    // move in range
    while(_pTimer->CurrentTick() < m_fStopTime) {
      FLOAT fMoveTime = FRndIn(2.0f, 4.0f);
      m_aSpeedRotation = FLOAT3D(FRndIn(8.0f, 16.0f), 0.0f, 0.0f);
      m_tmLastMove = _pTimer->CurrentTick();
      // NOTE: fMoveTime will cause the twister to stop existing
      // a bit later then the m_fStopTime, but what the heck?
      wait(fMoveTime) {
        on (EBegin) : { resume; }
        on (ETimer) : { stop; }
        on (EPass ep) : {
          if (ep.penOther->GetRenderType()&RT_MODEL &&
            ep.penOther->GetPhysicsFlags()&EPF_MOVABLE &&
            !IsOfClass(ep.penOther, "Twister")) {
            SpinEntity(ep.penOther);
          }
          resume;
        }
      }     
    }

    // fade out
    m_fFadeStartTime = _pTimer->CurrentTick();
    m_bFadeOut = TRUE;
    m_fFadeTime = 2.0f;
    autowait(m_fFadeTime);

    // cease to exist
    Destroy();

    return;
  }
};