Mercurial > mm7
view SpriteObject.cpp @ 1668:d17247968bd6
m
author | Ritor1 |
---|---|
date | Tue, 17 Sep 2013 22:55:34 +0600 |
parents | 2d9c8f609a3c |
children | 7182930263b3 |
line wrap: on
line source
#include "BSPModel.h" #include "SpriteObject.h" #include "Party.h" #include "TurnEngine.h" #include "MapInfo.h" #include "Math.h" #include "ObjectList.h" #include "Indoor.h" #include "Outdoor.h" #include "ParticleEngine.h" #include "Time.h" #include "Game.h" #include "LOD.h" #include "Actor.h" #include "Events.h" #include "mm7_data.h" #include "MM7.h" size_t uNumSpriteObjects; std::array<SpriteObject, MAX_SPRITE_OBJECTS> pSpriteObjects; //----- (00404828) -------------------------------------------------------- SpriteObject::SpriteObject() { field_22_glow_radius_multiplier = 1; uSoundID = 0; uFacing = 0; vVelocity.z = 0; vVelocity.y = 0; vVelocity.x = 0; uType = 0; uObjectDescID = 0; field_61 = 0; field_60_distance_related_prolly_lod = 0; field_20 = 0; uSpriteFrameID = 0; spell_skill = 0; spell_level = 0; spell_id = 0; field_54 = 0; } //----- (0042F5ED) -------------------------------------------------------- int SpriteObject::Create(int yaw, int pitch, int a4, int a5) { signed int v6; // ebx@2 int v13; // ST2C_4@20 Vec3_int_ v17; // [sp-20h] [bp-30h]@11 int angle; // [sp+Ch] [bp-4h]@1 int a5a; // [sp+20h] [bp+10h]@20 angle = yaw; if (!uObjectDescID) return -1; v6 = 1000; for (uint i = 0; i < MAX_SPRITE_OBJECTS; ++i) if (!pSpriteObjects[i].uObjectDescID) { v6 = i; break; } if ( v6 >= 1000 ) return -1; field_64.x = vPosition.x; field_64.y = vPosition.y; field_64.z = vPosition.z; assert(sizeof(SpriteObject) == 0x70); memcpy(&pSpriteObjects[v6], this, sizeof(*this)); if ( a5 == 0 ) { pSpriteObjects[v6].vVelocity.z = 0; if ( a4 ) { v13 = fixpoint_sub0(stru_5C6E00->Cos(angle), stru_5C6E00->Cos(pitch)); a5a = fixpoint_sub0(stru_5C6E00->Sin(angle), stru_5C6E00->Cos(pitch)); pSpriteObjects[v6].vVelocity.x = fixpoint_sub0(v13, a4); pSpriteObjects[v6].vVelocity.y = fixpoint_sub0(a5a, a4); pSpriteObjects[v6].vVelocity.z = fixpoint_sub0(stru_5C6E00->Sin(pitch), a4); } else { pSpriteObjects[v6].vVelocity.y = 0; pSpriteObjects[v6].vVelocity.x = 0; } if ( v6 >= (signed int)uNumSpriteObjects ) uNumSpriteObjects = v6 + 1; return v6; } if ( a5 == 1 ) { v17.x = vPosition.x; v17.y = vPosition.y; v17.z = vPosition.z; Vec3_int_::Rotate(24, stru_5C6E00->uIntegerHalfPi + pSpriteObjects[v6].uFacing, 0, v17, &pSpriteObjects[v6].vPosition.x, &pSpriteObjects[v6].vPosition.y, &pSpriteObjects[v6].vPosition.z); pSpriteObjects[v6].vVelocity.z = 0; if ( a4 ) { v13 = fixpoint_sub0(stru_5C6E00->Cos(angle), stru_5C6E00->Cos(pitch)); a5a = fixpoint_sub0(stru_5C6E00->Sin(angle), stru_5C6E00->Cos(pitch)); pSpriteObjects[v6].vVelocity.x = fixpoint_sub0(v13, a4); pSpriteObjects[v6].vVelocity.y = fixpoint_sub0(a5a, a4); pSpriteObjects[v6].vVelocity.z = fixpoint_sub0(stru_5C6E00->Sin(pitch), a4); } else { pSpriteObjects[v6].vVelocity.y = 0; pSpriteObjects[v6].vVelocity.x = 0; } if ( v6 >= (signed int)uNumSpriteObjects ) uNumSpriteObjects = v6 + 1; return v6; } if ( a5 == 2 ) { v17.x = vPosition.x; v17.y = vPosition.y; v17.z = vPosition.z; Vec3_int_::Rotate(8, stru_5C6E00->uIntegerHalfPi + pSpriteObjects[v6].uFacing, 0, v17, &pSpriteObjects[v6].vPosition.x, &pSpriteObjects[v6].vPosition.y, &pSpriteObjects[v6].vPosition.z); pSpriteObjects[v6].vVelocity.z = 0; if ( a4 ) { v13 = fixpoint_sub0(stru_5C6E00->Cos(angle), stru_5C6E00->Cos(pitch)) >> 16; a5a = fixpoint_sub0(stru_5C6E00->Sin(angle), stru_5C6E00->Cos(pitch)) >> 16; pSpriteObjects[v6].vVelocity.x = fixpoint_sub0(v13, a4); pSpriteObjects[v6].vVelocity.y = fixpoint_sub0(a5a, a4); pSpriteObjects[v6].vVelocity.z = fixpoint_sub0(stru_5C6E00->Sin(pitch), a4); } else { pSpriteObjects[v6].vVelocity.y = 0; pSpriteObjects[v6].vVelocity.x = 0; } if ( v6 >= (signed int)uNumSpriteObjects ) uNumSpriteObjects = v6 + 1; return v6; } if ( a5 == 3 ) { v17.x = vPosition.x; v17.y = vPosition.y; v17.z = vPosition.z; Vec3_int_::Rotate(8, pSpriteObjects[v6].uFacing - stru_5C6E00->uIntegerHalfPi, 0, v17, &pSpriteObjects[v6].vPosition.x, &pSpriteObjects[v6].vPosition.y, &pSpriteObjects[v6].vPosition.z); pSpriteObjects[v6].vVelocity.z = 0; if ( a4 ) { v13 = fixpoint_sub0(stru_5C6E00->Cos(angle), stru_5C6E00->Cos(pitch)); a5a = fixpoint_sub0(stru_5C6E00->Sin(angle), stru_5C6E00->Cos(pitch)); pSpriteObjects[v6].vVelocity.x = fixpoint_sub0(v13, a4); pSpriteObjects[v6].vVelocity.y = fixpoint_sub0(a5a, a4); pSpriteObjects[v6].vVelocity.z = fixpoint_sub0(stru_5C6E00->Sin(pitch), a4); } else { pSpriteObjects[v6].vVelocity.y = 0; pSpriteObjects[v6].vVelocity.x = 0; } if ( v6 >= (signed int)uNumSpriteObjects ) uNumSpriteObjects = v6 + 1; return v6; } if ( a5 == 4 ) { v17.x = vPosition.x; v17.y = vPosition.y; v17.z = vPosition.z; Vec3_int_::Rotate(24, pSpriteObjects[v6].uFacing - stru_5C6E00->uIntegerHalfPi, 0, v17, &pSpriteObjects[v6].vPosition.x, &pSpriteObjects[v6].vPosition.y, &pSpriteObjects[v6].vPosition.z); pSpriteObjects[v6].vVelocity.z = 0; if ( a4 ) { v13 = fixpoint_sub0(stru_5C6E00->Cos(angle), stru_5C6E00->Cos(pitch)); a5a = fixpoint_sub0(stru_5C6E00->Sin(angle), stru_5C6E00->Cos(pitch)); pSpriteObjects[v6].vVelocity.x = fixpoint_sub0(v13, a4); pSpriteObjects[v6].vVelocity.y = fixpoint_sub0(a5a, a4); pSpriteObjects[v6].vVelocity.z = fixpoint_sub0(stru_5C6E00->Sin(pitch), a4); } else { pSpriteObjects[v6].vVelocity.y = 0; pSpriteObjects[v6].vVelocity.x = 0; } if ( v6 >= (signed int)uNumSpriteObjects ) uNumSpriteObjects = v6 + 1; return v6; } assert(false); return 0; } //----- (00471C03) -------------------------------------------------------- void SpriteObject::UpdateObject_fn0_ODM(unsigned int uLayingItemID) { SpriteObject *v1; // esi@1 ObjectDesc *v2; // ebx@1 signed int v3; // edx@1 int v4; // ecx@1 int v5; // ST04_4@1 int v6; // eax@1 int v7; // ecx@1 int v8; // edi@1 int v9; // eax@4 __int16 v10; // ax@7 int v11; // edx@11 int v12; // ecx@11 signed int v13; // edx@14 signed int v14; // edx@16 int v15; // eax@24 int v16; // eax@25 int v17; // ST10_4@25 signed int v18; // eax@25 signed int v19; // eax@28 Actor *v20; // edi@31 int v21; // eax@41 int v22; // ecx@43 __int16 v23; // bx@45 char v24; // al@46 signed int i; // edi@50 int v26; // edi@52 int v27; // eax@52 __int16 v28; // cx@55 int v29; // eax@55 signed int v30; // edi@59 BSPModel *v31; // ecx@61 ODMFace *v32; // edi@61 int v33; // eax@62 int v34; // ecx@62 int v35; // eax@63 int v36; // ecx@67 __int16 v37; // ax@67 int v38; // eax@72 int v39; // eax@72 unsigned __int64 v40; // qax@72 int v41; // eax@72 unsigned __int8 v42; // sf@74 unsigned __int8 v43; // of@74 int v44; // eax@77 __int16 v45; // bx@81 int v46; // eax@85 const char *v47; // [sp-8h] [bp-B0h]@83 enum TEXTURE_TYPE v48; // [sp-4h] [bp-ACh]@46 int v49; // [sp+Ch] [bp-9Ch]@52 int v50; // [sp+10h] [bp-98h]@52 Vec3_int_ v51; // [sp+14h] [bp-94h]@11 Particle_sw Dst; // [sp+20h] [bp-88h]@45 unsigned int uLayingItemID_; // [sp+88h] [bp-20h]@1 int v54; // [sp+8Ch] [bp-1Ch]@1 int v55; // [sp+90h] [bp-18h]@1 int v56; // [sp+94h] [bp-14h]@11 int v57; // [sp+98h] [bp-10h]@1 int v58; // [sp+9Ch] [bp-Ch]@1 int on_water; // [sp+A0h] [bp-8h]@1 int v60; // [sp+A4h] [bp-4h]@11 uLayingItemID_ = uLayingItemID; v1 = &pSpriteObjects[uLayingItemID]; v58 = 0; v2 = &pObjectList->pObjects[v1->uObjectDescID]; v57 = IsTerrainSlopeTooHigh(v1->vPosition.x, v1->vPosition.y); v3 = v1->vPosition.y; v4 = v1->vPosition.x; v5 = v2->uHeight; v55 = 0; v6 = ODM_GetFloorLevel(v4, v3, v1->vPosition.z, v5, &on_water, &v55, 0); v7 = v6; v54 = v6; v8 = v6 + 1; if ( v1->vPosition.z <= v6 + 1 ) { if ( on_water ) { v9 = v6 + 60; if ( v55 ) v9 = v7 + 30; sub_42F960_create_object(v1->vPosition.x, v1->vPosition.y, v9); SpriteObject::OnInteraction(uLayingItemID_); } } else { v58 = 1; } v10 = v2->uFlags; if ( !(v2->uFlags & OBJECT_DESC_NO_GRAVITY) ) { if ( v58 ) { v1->vVelocity.z -= LOWORD(pEventTimer->uTimeElapsed) * GetGravityStrength(); goto LABEL_13; } if ( v57 ) { v11 = v1->vPosition.y; v12 = v1->vPosition.x; v1->vPosition.z = v8; ODM_GetTerrainNormalAt(v12, v11, &v51); v1->vVelocity.z -= LOWORD(pEventTimer->uTimeElapsed) * GetGravityStrength(); v56 = abs(v51.y * v1->vVelocity.y + v51.z * v1->vVelocity.z + v51.x * v1->vVelocity.x) >> 16; //v60 = ((unsigned __int64)(v56 * (signed __int64)v51.x) >> 16); v1->vVelocity.x += fixpoint_sub0(v56, v51.x); //v60 = ((unsigned __int64)(v56 * (signed __int64)v51.y) >> 16); v1->vVelocity.y += fixpoint_sub0(v56, v51.y); //v60 = ((unsigned __int64)(v56 * (signed __int64)v51.z) >> 16); v1->vVelocity.z += fixpoint_sub0(v56, v51.z); LABEL_12: v7 = v54; goto LABEL_13; } if ( v10 & 0x40 ) { if ( v1->vPosition.z < v7 ) v1->vPosition.z = v8; if ( !_46BFFA_check_object_intercept(uLayingItemID_, 0) ) return; } v1->vPosition.z = v8; if ( !(v2->uFlags & OBJECT_DESC_BOUNCE) || (v21 = -v1->vVelocity.z >> 1, v1->vVelocity.z = v21, (signed __int16)v21 < 10) ) v1->vVelocity.z = 0; v1->vVelocity.x = fixpoint_sub0(58500, v1->vVelocity.x); v1->vVelocity.y = fixpoint_sub0(58500, v1->vVelocity.y); v1->vVelocity.z = fixpoint_sub0(58500, v1->vVelocity.z); if ( (v1->vVelocity.y * v1->vVelocity.y + v1->vVelocity.x * v1->vVelocity.x) < 400 ) { v1->vVelocity.y = 0; v1->vVelocity.x = 0; memset(&Dst, 0, 0x68u); Dst.x = (double)v1->vPosition.x; Dst.y = (double)v1->vPosition.y; Dst.z = (double)v1->vPosition.z; Dst.r = 0.0; Dst.g = 0.0; Dst.b = 0.0; if (v2->uFlags & OBJECT_DESC_TRIAL_FIRE ) { Dst.type = ParticleType_Bitmap | ParticleType_Rotating | ParticleType_8; Dst.uDiffuse = 0xFF3C1E; Dst.timeToLive = (unsigned __int8)(rand() & 0x80) + 128; Dst.uTextureID = pBitmaps_LOD->LoadTexture("effpar01", TEXTURE_DEFAULT); Dst.flt_28 = 1.0; pGame->pParticleEngine->AddParticle(&Dst); } else if ( v2->uFlags & OBJECT_DESC_TRIAL_LINE) { Dst.type = ParticleType_Line; Dst.uDiffuse = rand(); Dst.timeToLive = 64; Dst.uTextureID = 0; Dst.flt_28 = 1.0; pGame->pParticleEngine->AddParticle(&Dst); } else if ( v2->uFlags & OBJECT_DESC_TRIAL_PARTICLE ) { Dst.type = ParticleType_Bitmap | ParticleType_8; Dst.uDiffuse = rand(); Dst.timeToLive = (unsigned __int8)(rand() & 0x80) + 128; Dst.uTextureID = pBitmaps_LOD->LoadTexture("effpar03", TEXTURE_DEFAULT); Dst.flt_28 = 1.0; pGame->pParticleEngine->AddParticle(&Dst); } return; } } LABEL_13: if ( v1->vPosition.z > v7 && (v13 = v1->vPosition.x, v13 >= -0x8000) && v13 <= 0x8000 && (v14 = v1->vPosition.y, v14 >= -0x8000) && v14 <= 0x8000 && v1->vPosition.z <= 13000 || !(v2->uFlags & OBJECT_DESC_INTERACTABLE) ) goto LABEL_92; if ( v1->vPosition.z < v7 ) v1->vPosition.z = v8; if ( _46BFFA_check_object_intercept(uLayingItemID_, 0) ) { LABEL_92: stru_721530.field_0 = 0; v55 = 0; stru_721530.prolly_normal_d = v2->uRadius; stru_721530.height = v2->uHeight; stru_721530.field_8_radius = 0; stru_721530.field_70 = 0; while ( 1 ) { stru_721530.position.x = v1->vPosition.x; stru_721530.normal.x = stru_721530.position.x; v15 = v1->vPosition.y; stru_721530.uSectorID = 0; stru_721530.position.y = v15; stru_721530.normal.y = v15; stru_721530.position.z = v1->vPosition.z + stru_721530.prolly_normal_d + 1; stru_721530.normal.z = stru_721530.position.z; stru_721530.velocity.x = v1->vVelocity.x; stru_721530.velocity.y = v1->vVelocity.y; stru_721530.velocity.z = v1->vVelocity.z; if ( stru_721530._47050A(0) ) return; _46E889_collide_against_bmodels(0); v16 = WorldPosToGridCellZ(v1->vPosition.y); v18 = WorldPosToGridCellX(v1->vPosition.x); _46E26D_collide_against_sprites(v18, v16); if (PID_TYPE(v1->spell_caster_pid) != OBJECT_Player) _46EF01_collision_chech_player(0); if (PID_TYPE(v1->spell_caster_pid) == OBJECT_Actor) { v19 = PID_ID(v1->spell_caster_pid); if (( v19 >= 0 )&&( v19 < (signed int)(uNumActors - 1) )) { v20 = &pActors[v19]; for (v56 =0; v56 < uNumActors; ++v56) { if ( v20->GetActorsRelation(&pActors[v56]) ) _46DF1A_collide_against_actor(v56, 0); } } } else { for ( i = 0; i < (signed int)uNumActors; ++i ) _46DF1A_collide_against_actor(i, 0); } v26 = stru_721530.normal2.z - stru_721530.prolly_normal_d - 1; v27 = ODM_GetFloorLevel( stru_721530.normal2.x, stru_721530.normal2.y, stru_721530.normal2.z - stru_721530.prolly_normal_d - 1, v2->uHeight, &v49, &v50, 0); if ( on_water && v26 < v27 + 60 ) { if ( v50 ) v44 = v27 + 30; else v44 = v54 + 60; sub_42F960_create_object(v1->vPosition.x, v1->vPosition.y, v44); SpriteObject::OnInteraction(uLayingItemID_); return; } if ( stru_721530.field_7C >= stru_721530.field_6C ) { v1->vPosition.x = stru_721530.normal2.x; v1->vPosition.y = stru_721530.normal2.y; v1->vPosition.z = stru_721530.normal2.z - stru_721530.prolly_normal_d - 1; v1->uSectorID = LOWORD(stru_721530.uSectorID); memset(&Dst, 0, 0x68u); Dst.x = (double)v1->vPosition.x; Dst.y = (double)v1->vPosition.y; Dst.z = (double)v1->vPosition.z; Dst.r = 0.0; Dst.g = 0.0; Dst.b = 0.0; if ( v2->uFlags & OBJECT_DESC_TRIAL_FIRE ) { Dst.type = ParticleType_Bitmap | ParticleType_Rotating | ParticleType_8; Dst.uDiffuse = 0xFF3C1E; Dst.timeToLive = (unsigned __int8)( rand() & 0x80) + 128; Dst.uTextureID = pBitmaps_LOD->LoadTexture("effpar01", TEXTURE_DEFAULT); Dst.flt_28 = 1.0; pGame->pParticleEngine->AddParticle(&Dst); return; } else if ( v2->uFlags & OBJECT_DESC_TRIAL_LINE ) { Dst.type = ParticleType_Line; Dst.uTextureID = 0; Dst.uDiffuse = rand(); Dst.timeToLive = 64; Dst.flt_28 = 1.0; pGame->pParticleEngine->AddParticle(&Dst); return; } else if ( v2->uFlags & OBJECT_DESC_TRIAL_PARTICLE) { Dst.type = ParticleType_Bitmap | ParticleType_8; Dst.uDiffuse = rand(); Dst.timeToLive = (unsigned __int8)(rand() & 0x80) + 128; Dst.uTextureID = pBitmaps_LOD->LoadTexture("effpar03", TEXTURE_DEFAULT); Dst.flt_28 = 1.0; pGame->pParticleEngine->AddParticle(&Dst); } return; } //v60 = ((unsigned __int64)(stru_721530.field_7C * (signed __int64)stru_721530.direction.x) >> 16); v1->vPosition.x += fixpoint_sub0(stru_721530.field_7C, stru_721530.direction.x); //v60 = ((unsigned __int64)(stru_721530.field_7C * (signed __int64)stru_721530.direction.y) >> 16); v1->vPosition.y += fixpoint_sub0(stru_721530.field_7C, stru_721530.direction.y); //v60 = ((unsigned __int64)(stru_721530.field_7C * (signed __int64)stru_721530.direction.z) >> 16); v28 = LOWORD(stru_721530.uSectorID); v1->vPosition.z += fixpoint_sub0(stru_721530.field_7C, stru_721530.direction.z); v29 = v1->vPosition.z; v1->uSectorID = v28; stru_721530.field_70 += stru_721530.field_7C; if ( v2->uFlags & OBJECT_DESC_INTERACTABLE ) { if ( v29 < v54 ) v1->vPosition.z = v54 + 1; if ( !_46BFFA_check_object_intercept(uLayingItemID_, stru_721530.uFaceID) ) return; } v30 = (signed int)PID_ID(stru_721530.uFaceID); if (PID_TYPE(stru_721530.uFaceID) == OBJECT_Decoration) break; if (PID_TYPE(stru_721530.uFaceID) == OBJECT_BModel) { v31 = &pOutdoor->pBModels[(signed int)stru_721530.uFaceID >> 9]; v32 = &v31->pFaces[v30 & 0x3F]; if ( v32->uPolygonType != 3 ) { v56 = abs(v32->pFacePlane.vNormal.z * v1->vVelocity.z + v32->pFacePlane.vNormal.y * v1->vVelocity.y + v32->pFacePlane.vNormal.x * v1->vVelocity.x) >> 16; if ( (stru_721530.speed >> 3) > v56 ) v56 = stru_721530.speed >> 3; v57 = v32->pFacePlane.vNormal.x; v57 = (unsigned __int64)(v56 * (signed __int64)v57) >> 16; v58 = v32->pFacePlane.vNormal.y; v58 = (unsigned __int64)(v56 * (signed __int64)v58) >> 16; v60 = ((unsigned __int64)(v56 * (signed __int64)v32->pFacePlane.vNormal.z) >> 16); v1->vVelocity.x += 2 * v57; v1->vVelocity.y += 2 * v58; if ( v32->pFacePlane.vNormal.z <= 32000 ) { v37 = 2 * (short)v60; } else { v36 = v60; v1->vVelocity.z += (signed __int16)v60; v58 = (unsigned __int64)(32000 * (signed __int64)(signed int)v36) >> 16; v37 = (unsigned int)(32000 * v36) >> 16; } v1->vVelocity.z += v37; LABEL_70: if ( BYTE3(v32->uAttributes) & 0x10 ) EventProcessor(v32->sCogTriggeredID, 0, 1); goto LABEL_74; } v33 = v31->pVertices.pVertices[v32->pVertexIDs[0]].z; v34 = v1->vVelocity.x; v1->vPosition.z = v33 + 1; if ( v1->vVelocity.y * v1->vVelocity.y + v34 * v34 >= 400 ) goto LABEL_70; LOWORD(v35) = 0; v1->vVelocity.z = 0; v1->vVelocity.x = 0; goto LABEL_73; } LABEL_74: v1->vVelocity.x = (signed __int16)((unsigned __int64)(58500i64 * (signed __int64)(signed int)v1->vVelocity.x) >> 16); v1->vVelocity.y = (signed __int16)((unsigned __int64)(58500i64 * (signed __int64)(signed int)v1->vVelocity.y) >> 16); v1->vVelocity.z = (signed __int16)((unsigned __int64)(58500i64 * (signed __int64)(signed int)v1->vVelocity.z) >> 16); ++v55; //v43 = __OFSUB__(v55, 100); // v42 = v55 - 100 < 0; if (v55>=100 )//!(v42 ^ v43) return; } v57 = integer_sqrt(v1->vVelocity.y * v1->vVelocity.y + v1->vVelocity.x * v1->vVelocity.x); v38 = stru_5C6E00->Atan2( v1->vPosition.x - pLevelDecorations[v30].vPosition.x, v1->vPosition.y - pLevelDecorations[v30].vPosition.y); v56 = v38; v39 = stru_5C6E00->Cos(v38); // v60 = v39; v40 = v39 * (signed __int64)v57; v58 = v40 >> 16; v1->vVelocity.x = WORD1(v40); v41 = stru_5C6E00->Sin(v56 - stru_5C6E00->uIntegerHalfPi); // v60 = v41; v35 = (unsigned __int64)(v41 * (signed __int64)v57) >> 16; v58 = v35; LABEL_73: v1->vVelocity.y = v35; goto LABEL_74; } } //----- (0047136C) -------------------------------------------------------- void SpriteObject::UpdateObject_fn0_BLV(unsigned int uLayingItemID) { SpriteObject *pSpriteObject; // esi@1 ObjectDesc *pObject; // edi@1 //int v3; // ST08_4@1 //__int16 v4; // ax@5 __int16 v5; // ax@7 BLVFace *v6; // ecx@11 BLVFace *v7; // eax@11 signed int v8; // ebx@12 int v9; // ecx@16 __int16 v10; // di@18 char v11; // al@19 //int v12; // eax@25 int v13; // eax@31 int v14; // ebx@34 signed int v15; // ebx@46 BLVFace *v16; // edi@48 int v17; // eax@50 int v18; // eax@52 int v19; // ecx@52 Vec3_short_ *v20; // ecx@53 //int v21; // ecx@57 __int16 v22; // ax@57 int v23; // edi@62 //int v24; // edi@62 //int v25; // eax@62 //unsigned __int64 v26; // qax@62 unsigned __int8 v27; // sf@64 unsigned __int8 v28; // of@64 __int16 v29; // di@67 char v30; // al@68 const char *v31; // [sp-8h] [bp-98h]@19 const char *v32; // [sp-8h] [bp-98h]@68 enum TEXTURE_TYPE v33; // [sp-4h] [bp-94h]@19 enum TEXTURE_TYPE v34; // [sp-4h] [bp-94h]@68 Particle_sw Dst; // [sp+Ch] [bp-84h]@18 //unsigned int uLayingItemID_; // [sp+74h] [bp-1Ch]@1 //ObjectDesc *v37; // [sp+78h] [bp-18h]@1 unsigned int uFaceID; // [sp+7Ch] [bp-14h]@4 int v39; // [sp+80h] [bp-10h]@33 Actor *v39b; int v40; // [sp+84h] [bp-Ch]@28 //int v41; // [sp+88h] [bp-8h]@34 int v42; // [sp+8Ch] [bp-4h]@4 //uLayingItemID_ = uLayingItemID; pSpriteObject = &pSpriteObjects[uLayingItemID]; pObject = &pObjectList->pObjects[pSpriteObject->uObjectDescID]; //v3 = pSpriteObject->vPosition.x; //v37 = &pObjectList->pObjects[pSpriteObject->uObjectDescID]; pSpriteObject->uSectorID = pIndoor->GetSector(pSpriteObject->vPosition.x, pSpriteObject->vPosition.y, pSpriteObject->vPosition.z); v42 = BLV_GetFloorLevel(pSpriteObject->vPosition.x, pSpriteObject->vPosition.y, pSpriteObject->vPosition.z, pSpriteObject->uSectorID, &uFaceID); if ( abs(pSpriteObject->vPosition.x) > 32767 || abs(pSpriteObject->vPosition.y) > 32767 || abs(pSpriteObject->vPosition.z) > 20000 || v42 <= -30000 && (pSpriteObject->uSectorID == 0)) // || (v42 = _46CEC3_get_floor_level(pSpriteObject->vPosition.x, pSpriteObject->vPosition.y, pSpriteObject->vPosition.z, v4, &uFaceID), v42 == -30000)) ) { SpriteObject::OnInteraction(uLayingItemID); return; } v5 = pObject->uFlags; if ( v5 & 0x20 ) { LABEL_24: v8 = 0; LABEL_25: stru_721530.field_0 = v8; uFaceID = v8; stru_721530.prolly_normal_d = pObject->uRadius; stru_721530.field_84 = -1; stru_721530.height = pObject->uHeight; stru_721530.field_8_radius = v8; stru_721530.field_70 = v8; while ( 1 ) { stru_721530.position.x = pSpriteObject->vPosition.x; stru_721530.normal.x = stru_721530.position.x; stru_721530.position.y = pSpriteObject->vPosition.y; stru_721530.normal.y = stru_721530.position.y; stru_721530.position.z = stru_721530.prolly_normal_d + pSpriteObject->vPosition.z + 1; stru_721530.normal.z = stru_721530.position.z; stru_721530.velocity.x = pSpriteObject->vVelocity.x; stru_721530.velocity.y = pSpriteObject->vVelocity.y; stru_721530.velocity.z = pSpriteObject->vVelocity.z; stru_721530.uSectorID = pSpriteObject->uSectorID; if ( stru_721530._47050A(v8) ) return; v40 = v8; do { _46E44E_collide_against_faces_and_portals(0); _46E0B2_collide_against_decorations(); if (PID_TYPE(pSpriteObject->spell_caster_pid) != OBJECT_Player) _46EF01_collision_chech_player(1); v13 = pSpriteObject->spell_caster_pid; v42 = v8; if (PID_TYPE(v13) == OBJECT_Actor) { if ( (signed int)uNumActors > v8 ) { v39b = pActors.data();//[0].word_000086_some_monster_id; do { //v41 = pActors[v1->field_58 >> 3].pMonsterInfo.uID - 1; //v14 = (signed __int64)((double)v41 * 0.3333333333333333); //v41 = *(short *)(v39 - 38) - 1; //if ( v14 != (unsigned int)(signed __int64)((double)v41 * 0.3333333333333333) ) if( pActors[pSpriteObject->spell_caster_pid >> 3].pMonsterInfo.uID != v39b->pMonsterInfo.uID ) //not sure: pMonsterList->pMonsters[v39b->word_000086_some_monster_id-1].uToHitRadius _46DF1A_collide_against_actor(v42, pMonsterList->pMonsters[v39b->word_000086_some_monster_id-1].uToHitRadius); ++v42; ++v39b;// += 836; } while ( v42 < (signed int)uNumActors ); v8 = 0; } } else { if ( (signed int)uNumActors > v8 ) { v39b = pActors.data();//[0].word_000086_some_monster_id; do { _46DF1A_collide_against_actor(v42++, pMonsterList->pMonsters[v39b->word_000086_some_monster_id-1].uToHitRadius); ++v39b; } while ( v42 < (signed int)uNumActors ); } } if ( _46F04E_collide_against_portals() ) break; ++v40; } while ( v40 < 100 ); if ( stru_721530.field_7C >= stru_721530.field_6C ) { pSpriteObject->vPosition.x = stru_721530.normal2.x; pSpriteObject->vPosition.y = stru_721530.normal2.y; pSpriteObject->vPosition.z = stru_721530.normal2.z - stru_721530.prolly_normal_d - 1; pSpriteObject->uSectorID = LOWORD(stru_721530.uSectorID); if ( !(HIBYTE(pObject->uFlags) & 1) ) return; memset(&Dst, v8, 0x68u); v29 = pObject->uFlags; Dst.x = (double)pSpriteObject->vPosition.x; Dst.y = (double)pSpriteObject->vPosition.y; Dst.z = (double)pSpriteObject->vPosition.z; Dst.r = 0.0; Dst.g = 0.0; Dst.b = 0.0; if ( v29 & 0x200 ) { Dst.type = ParticleType_Bitmap | ParticleType_Rotating | ParticleType_8; Dst.uDiffuse = 0xFF3C1E; v30 = rand(); v34 = (TEXTURE_TYPE)v8; v32 = "effpar01"; } else { if ( v29 & 0x400 ) { Dst.type = ParticleType_Line; Dst.uDiffuse = rand(); Dst.timeToLive = 64; Dst.uTextureID = v8; Dst.flt_28 = 1.0; pGame->pParticleEngine->AddParticle(&Dst); return; } Dst.type = ParticleType_Bitmap | ParticleType_8; Dst.uDiffuse = rand(); v30 = rand(); v34 = (TEXTURE_TYPE)v8; v32 = "effpar03"; } Dst.timeToLive = (unsigned __int8)(v30 & 0x80) + 128; Dst.uTextureID = pBitmaps_LOD->LoadTexture(v32, v34); Dst.flt_28 = 1.0; pGame->pParticleEngine->AddParticle(&Dst); return; } //v40 = (unsigned __int64)(stru_721530.field_7C * (signed __int64)stru_721530.direction.x) >> 16; pSpriteObject->vPosition.x += fixpoint_sub0(stru_721530.field_7C, stru_721530.direction.x); //v40 = (unsigned __int64)(stru_721530.field_7C * (signed __int64)stru_721530.direction.y) >> 16; pSpriteObject->vPosition.y += fixpoint_sub0(stru_721530.field_7C, stru_721530.direction.y); //v40 = (unsigned __int64)(stru_721530.field_7C * (signed __int64)stru_721530.direction.z) >> 16; pSpriteObject->vPosition.z += fixpoint_sub0(stru_721530.field_7C, stru_721530.direction.z); pSpriteObject->uSectorID = stru_721530.uSectorID; stru_721530.field_70 += stru_721530.field_7C; if ( pObject->uFlags & 0x40 && !_46BFFA_check_object_intercept(uLayingItemID, stru_721530.uFaceID) ) return; v15 = (signed int)stru_721530.uFaceID >> 3; if (PID_TYPE(stru_721530.uFaceID) == OBJECT_Decoration) { v40 = integer_sqrt(pSpriteObject->vVelocity.x * pSpriteObject->vVelocity.x + pSpriteObject->vVelocity.y * pSpriteObject->vVelocity.y); v23 = stru_5C6E00->Atan2(pSpriteObject->vPosition.x - pLevelDecorations[v15].vPosition.x, pSpriteObject->vPosition.y - pLevelDecorations[v15].vPosition.y); pSpriteObject->vVelocity.x = fixpoint_sub0(stru_5C6E00->Cos(v23), v40); pSpriteObject->vVelocity.y = fixpoint_sub0(stru_5C6E00->Sin(v23), v40); } else { if (PID_TYPE(stru_721530.uFaceID) != OBJECT_BModel) goto LABEL_64; stru_721530.field_84 = (signed int)PID_ID(stru_721530.uFaceID); v16 = &pIndoor->pFaces[v15]; if ( v16->uPolygonType != 3 ) { v42 = abs(v16->pFacePlane_old.vNormal.x * pSpriteObject->vVelocity.x + v16->pFacePlane_old.vNormal.z * pSpriteObject->vVelocity.z + v16->pFacePlane_old.vNormal.y * pSpriteObject->vVelocity.y) >> 16; if ( (stru_721530.speed >> 3) > v42 ) v42 = stru_721530.speed >> 3; pSpriteObject->vVelocity.x += 2 * fixpoint_sub0(v42, v16->pFacePlane_old.vNormal.x); pSpriteObject->vVelocity.y += 2 * fixpoint_sub0(v42, v16->pFacePlane_old.vNormal.y); v39 = fixpoint_sub0(v42, v16->pFacePlane_old.vNormal.z); if ( v16->pFacePlane_old.vNormal.z <= 32000 ) { v22 = 2 * v39; } else { pSpriteObject->vVelocity.z += v39; v22 = fixpoint_sub0(32000, v39); } pSpriteObject->vVelocity.z += v22; if ( BYTE3(v16->uAttributes) & 0x10 ) EventProcessor(pIndoor->pFaceExtras[v16->uFaceExtraID].uEventID, 0, 1); goto LABEL_63; } if ( pObject->uFlags & 0x80 ) { v17 = -pSpriteObject->vVelocity.z >> 1; pSpriteObject->vVelocity.z = v17; if ( (signed __int16)v17 < 10 ) pSpriteObject->vVelocity.z = 0; if ( BYTE3(v16->uAttributes) & 0x10 ) EventProcessor(pIndoor->pFaceExtras[v16->uFaceExtraID].uEventID, 0, 1); goto LABEL_63; } v18 = pSpriteObject->vVelocity.y; v19 = pSpriteObject->vVelocity.x; pSpriteObject->vVelocity.z = 0; if ( v19 * v19 + v18 * v18 >= 400 ) { if ( BYTE3(v16->uAttributes) & 0x10 ) EventProcessor(pIndoor->pFaceExtras[v16->uFaceExtraID].uEventID, 0, 1); goto LABEL_63; } v20 = pIndoor->pVertices; pSpriteObject->vVelocity.z = 0; pSpriteObject->vVelocity.y = 0; pSpriteObject->vVelocity.x = 0; pSpriteObject->vPosition.z = v20[*v16->pVertexIDs].z + 1; } LABEL_63: //v2 = v37; LABEL_64: pSpriteObject->vVelocity.x = fixpoint_sub0(58500, pSpriteObject->vVelocity.x); pSpriteObject->vVelocity.y = fixpoint_sub0(58500, pSpriteObject->vVelocity.y); pSpriteObject->vVelocity.z = fixpoint_sub0(58500, pSpriteObject->vVelocity.z); ++uFaceID; v28 = __OFSUB__(uFaceID, 100); v27 = uFaceID - 100 < 0; if ( !(v27 ^ v28) ) return; v8 = 0; } } if ( v42 <= pSpriteObject->vPosition.z - 3 ) { pSpriteObject->vVelocity.z -= LOWORD(pEventTimer->uTimeElapsed) * GetGravityStrength(); goto LABEL_24; } if ( !(v5 & 0x40) || _46BFFA_check_object_intercept(uLayingItemID, 0) ) { v6 = pIndoor->pFaces; pSpriteObject->vPosition.z = v42 + 1; v7 = &v6[uFaceID]; if ( v7->uPolygonType == 3 ) { v8 = 0; pSpriteObject->vVelocity.z = 0; } else { if ( v7->pFacePlane_old.vNormal.z < 45000 ) pSpriteObject->vVelocity.z -= LOWORD(pEventTimer->uTimeElapsed) * GetGravityStrength(); v8 = 0; } pSpriteObject->vVelocity.x = fixpoint_sub0(58500, pSpriteObject->vVelocity.x); pSpriteObject->vVelocity.y = fixpoint_sub0(58500, pSpriteObject->vVelocity.y); pSpriteObject->vVelocity.z = fixpoint_sub0(58500, pSpriteObject->vVelocity.z); v9 = pSpriteObject->vVelocity.x; if ( v9 * v9 + pSpriteObject->vVelocity.y * pSpriteObject->vVelocity.y < 400 ) { pSpriteObject->vVelocity.z = v8; pSpriteObject->vVelocity.y = v8; pSpriteObject->vVelocity.x = v8; if ( !(pObject->uFlags & 1) ) return; memset(&Dst, v8, 0x68u); v10 = pObject->uFlags; Dst.x = (double)pSpriteObject->vPosition.x; Dst.y = (double)pSpriteObject->vPosition.y; Dst.z = (double)pSpriteObject->vPosition.z; Dst.r = 0.0; Dst.g = 0.0; Dst.b = 0.0; if ( v10 & 0x200 ) { Dst.type = ParticleType_Bitmap | ParticleType_Rotating | ParticleType_8; Dst.uDiffuse = 0xFF3C1E; Dst.flt_28 = 1.0; v11 = rand(); v33 = (TEXTURE_TYPE)v8; v31 = "effpar01"; } else { if ( v10 & 0x400 ) { Dst.type = ParticleType_Line; Dst.uDiffuse = rand(); Dst.timeToLive = 64; Dst.uTextureID = v8; Dst.flt_28 = 1.0; pGame->pParticleEngine->AddParticle(&Dst); return; } Dst.type = ParticleType_Bitmap | ParticleType_8; Dst.uDiffuse = rand(); Dst.flt_28 = 1.0; v11 = rand(); v33 = (TEXTURE_TYPE)v8; v31 = "effpar03"; } Dst.timeToLive = (unsigned __int8)(v11 & 0x80) + 128; Dst.uTextureID = pBitmaps_LOD->LoadTexture(v31, v33); pGame->pParticleEngine->AddParticle(&Dst); return; } goto LABEL_25; } } // 46DF1A: using guessed type int __fastcall 46DF1A_collide_against_actor(int, int); //----- (00438E35) -------------------------------------------------------- void SpriteObject::ExplosionTraps() { MapInfo *pMapInfo; // esi@1 int dir_x; // ebx@1 int v7; // edx@2 unsigned int v10; // eax@7 signed int v11; // ebx@8 signed int v13; // edi@20 int dir_y; // [sp+Ch] [bp-Ch]@1 int dir_z; // [sp+10h] [bp-8h]@1 DAMAGE_TYPE pDamageType; // [sp+14h] [bp-4h]@14 pMapInfo = &pMapStats->pInfos[pMapStats->GetMapInfo(pCurrentMapName)]; dir_x = abs(pParty->vPosition.x - this->vPosition.x); dir_y = abs(pParty->vPosition.y - this->vPosition.y); dir_z = abs(pParty->vPosition.z + pParty->sEyelevel - this->vPosition.z); if ( dir_x < dir_y ) { v7 = dir_x; dir_x = dir_y; dir_y = v7; } if ( dir_x < dir_z ) { v7 = dir_x; dir_x = dir_z; dir_z = v7; } if ( dir_y < dir_z ) { v7 = dir_z; dir_z = dir_y; dir_y = v7; } v10 = ((unsigned int)(11 * dir_y) >> 5) + (dir_z / 4) + dir_x; if ( (signed int)v10 <= 768 ) { v11 = 5; if ( pMapInfo->Trap_D20 ) { for ( uint i = 0; i < pMapInfo->Trap_D20; ++i ) v11 += rand() % 20 + 1; } switch ( this->uType ) { case 811: pDamageType = DMGT_FIRE; break; case 812: pDamageType = DMGT_ELECTR; break; case 813: pDamageType = DMGT_COLD; break; case 814: pDamageType = DMGT_BODY; break; default: return; } for ( uint i = 1; i <= 4; ++i ) { if ( pPlayers[i]->CanAct() && (v13 = pPlayers[i]->GetPerception() + 20, rand() % v13 > 20) ) pPlayers[i]->PlaySound(SPEECH_6, 0); else pPlayers[i]->ReceiveDamage(v11, pDamageType); } } } //----- (0042F933) -------------------------------------------------------- void SpriteObject::OnInteraction(unsigned int uLayingItemID) { pSpriteObjects[uLayingItemID].uObjectDescID = 0; if ( pParty->bTurnBasedModeOn == 1 ) { if (pSpriteObjects[uLayingItemID].uAttributes & 4 ) { pSpriteObjects[uLayingItemID].uAttributes &= 0xFFFB; --pTurnEngine->pending_actions; } } } //----- (0042FA22) -------------------------------------------------------- void CompactLayingItemsList() { int new_obj_pos=0; for (int i=0; i<MAX_SPRITE_OBJECTS; ++i) { if (pSpriteObjects[i].uObjectDescID) memcpy(&pSpriteObjects[new_obj_pos++], &pSpriteObjects[i],sizeof(SpriteObject)); } uNumSpriteObjects = new_obj_pos; } //----- (00408896) -------------------------------------------------------- void InitializeSpriteObjects() { for (uint i = 0; i < uNumSpriteObjects; ++i) { auto item = &pSpriteObjects[i]; if (item->uType && (item->uSoundID & 8 || pObjectList->pObjects[item->uType].uFlags & 0x10)) SpriteObject::OnInteraction(i); } } //----- (0046BEF1) -------------------------------------------------------- void SpriteObject::_46BEF1_apply_spells_aoe() { SpriteObject *v1; // edi@1 Actor *v2; // esi@2 __int16 v3; // fps@4 unsigned __int8 v4; // c0@4 unsigned __int8 v5; // c3@4 signed int v6; // [sp+8h] [bp-4h]@1 int v7,v8,v9,v10,v11; v6 = 0; v1 = this; if ( (signed int)uNumActors > 0 ) { v2 = pActors.data();//[0].vPosition.y; do { if ( v2->CanAct() ) { //UNDEF(v3); //.text:0046BF26 movsx eax, word ptr [esi-2] //.text:0046BF2A sub eax, [edi+4] //.text:0046BF31 mov [ebp+var_8], eax //.text:0046BF37 fild [ebp+var_8] // v7 pushed to stack v7 = v2->vPosition.x - this->vPosition.x; //.text:0046BF2D movsx ecx, word ptr [esi+2] v8 = v2->vPosition.z; //.text:0046BF34 movsx eax, word ptr [esi] //.text:0046BF3A sub eax, [edi+8] //.text:0046BF3D mov [ebp+var_8], eax //.text:0046BF44 fild [ebp+var_8] // v9 pushed to stack v9 = v2->vPosition.y - this->vPosition.y; //.text:0046BF40 movsx eax, word ptr [esi-6] //.text:0046BF47 sar eax, 1 //.text:0046BF49 add eax, ecx //.text:0046BF4B sub eax, [edi+0Ch] //.text:0046BF4E mov [ebp+var_8], eax //.text:0046BF51 fild [ebp+var_8] //.text:0046BF58 fld st // v10 pushed to stack, two times v10 = v2->uActorHeight / 2 + v8 - this->vVelocity.y; //.text:0046BF54 movsx eax, word ptr [esi-8] //.text:0046BF5A add eax, 100h //.text:0046BF63 mov ecx, eax v11 = this->vVelocity.x; //.text:0046BF5F fmul st, st(1) // stack: v10*v10, v10, v9, v7 //.text:0046BF61 fld st(2) // stack: v7, v10*v10, v10, v9, v7 //.text:0046BF65 fmul st, st(3) // stack: v7*v9, v10*v10, v10, v9, v7 //.text:0046BF67 imul ecx, eax v11 = v11 * v11; //.text:0046BF6A faddp st(1), st // stack: v10*v10+v7*v9, v10, v9, v7 //.text:0046BF6C fld st(3) // stack: v7, v10*v10+v7*v9, v10, v9, v7 //.text:0046BF6E fmul st, st(4) // stack: v7*v7, v10*v10+v7*v9, v10, v9, v7 //.text:0046BF70 faddp st(1), st // stack: v10*v10+v7*v9+v7*v7, v10, v9, v7 //.text:0046BF72 mov [ebp+var_8], ecx //.text:0046BF75 fild [ebp+var_8] // v11 pushed to stack //.text:0046BF78 fcompp // if ( v11 > v10*v10+v7*v9+v7*v7 ) // stack: v10, v9, v7 //.text:0046BF7A fstp st // stack: v9, v7 //.text:0046BF7C fnstsw ax //.text:0046BF7E fstp st // stack: v7 //.text:0046BF80 test ah, 41h //.text:0046BF83 fstp st //.text:0046BF85 jnz short loc_46BFDD if ( v11 >= v10*v10+v7*v9+v7*v7 ) { if ( stru_50C198.GetMagicalResistance(v2, 0xAu) ) { v2->pActorBuffs[v1->spell_id].Apply( pParty->uTimePlayed + (signed int)(signed __int64)((double)(v1->spell_level << 7) * 0.033333335), v1->spell_skill, 4u, 0, 0); HIWORD(v2->uAttributes) |= 8u; } } } ++v6; ++v2; } while ( v6 < (signed int)uNumActors ); } }