/** * @file d_a_obj_wchain.cpp * */ #include "d/actor/d_a_obj_wchain.h" #include "d/d_com_inf_game.h" #include "d/actor/d_a_player.h" #include "d/d_procname.h" #include "SSystem/SComponent/c_math.h" #include "JSystem/J3DGraphBase/J3DMaterial.h" #include "JSystem/J3DGraphBase/J3DDrawBuffer.h" /* 80D31924-80D3192C 000000 0007+01 8/8 0/0 0/0 .rodata l_arcName */ static char const l_arcName[7] = "Wchain"; /* 80D2FEF8-80D2FF9C 000078 00A4+00 1/1 0/0 0/0 .text createHeap__13daObjWchain_cFv */ int daObjWchain_c::createHeap() { J3DModelData* handle_model_data = (J3DModelData*)dComIfG_getObjectRes(l_arcName, 4); mpHandleModel = mDoExt_J3DModel__create(handle_model_data, 0x80000, 0x11000084); if (mpHandleModel == NULL) { return 0; } mpChainModelData = (J3DModelData*)dComIfG_getObjectRes(l_arcName, 3); mShape.setUserArea((u32)this); return 1; } /* 80D2FF9C-80D2FFBC 00011C 0020+00 1/1 0/0 0/0 .text daObjWchain_createHeap__FP10fopAc_ac_c */ static int daObjWchain_createHeap(fopAc_ac_c* i_this) { return static_cast(i_this)->createHeap(); } /* 80D2FFBC-80D30394 00013C 03D8+00 1/1 0/0 0/0 .text create__13daObjWchain_cFv */ cPhs__Step daObjWchain_c::create() { fopAcM_SetupActor(this, daObjWchain_c); mSw = fopAcM_GetParam(this) & 0xff; cPhs__Step step = (cPhs__Step)dComIfG_resLoad(&mPhaseReq, l_arcName); if (step == cPhs_COMPLEATE_e) { mRepeatable = (fopAcM_GetParam(this) >> 8) & 0xf; if (mRepeatable == 0xf) { mRepeatable = 0; } if (!fopAcM_entrySolidHeap(this, daObjWchain_createHeap, 0x820)) { return cPhs_ERROR_e; } tevStr.room_no = dStage_roomControl_c::mStayNo; attention_info.position = current.pos; eyePos = attention_info.position; attention_info.distances[fopAc_attn_LOCK_e] = 0x1c; fopAcM_SetMtx(this, mpHandleModel->getBaseTRMtx()); fopAcM_SetMin(this, -200.0f, -45.0f, -200.0f); fopAcM_SetMax(this, 200.0f, 800.0f, 200.0f); mTopPos = current.pos; current.pos.y += 53.75f; mRoofPos.set(current.pos.x, current.pos.y + 280.0f - 100.0f, current.pos.z); mRealRoofY = mRoofPos.y + 250.0f; mInitOutLength = mRoofPos.y - home.pos.y; if (!mRepeatable && fopAcM_isSwitch(this, mSw)) { current.pos.y -= 100.0f; mEnd = true; mPullLength = 100.0f; field_0x7a8 = 600.0f; } else { mPullLength = 0.0f; } shape_angle.x = 0x4000; gravity = -7.0f; cXyz* chain_pos = &mChainPos[0xf]; cXyz* chain_speed = &mChainSpeed[0xf]; csXyz* chain_angle = &mChainAngle[0xf]; s16 ang_z = cM_rndFX(2048.0f) + 16384.0f; f32 pos_y = current.pos.y; for (int i = 0xf; i >= 0; chain_pos--, chain_speed--, chain_angle--, i--) { chain_pos->set(current.pos.x, pos_y, current.pos.z); *chain_speed = cXyz::Zero; chain_angle->set(0x4000, shape_angle.y, ang_z); ang_z += cM_rndFX(2048.0f) + 16384.0f; pos_y += 17.5f; } speed = cXyz::Zero; setMatrix(); } return step; } /* 80D30414-80D30434 000594 0020+00 1/0 0/0 0/0 .text daObjWchain_Create__FP10fopAc_ac_c */ static cPhs__Step daObjWchain_Create(fopAc_ac_c* i_this) { return static_cast(i_this)->create(); } /* 80D30434-80D3050C 0005B4 00D8+00 1/1 0/0 0/0 .text __dt__13daObjWchain_cFv */ daObjWchain_c::~daObjWchain_c() { dComIfG_resDelete(&mPhaseReq, l_arcName); } /* 80D3050C-80D30534 00068C 0028+00 1/0 0/0 0/0 .text daObjWchain_Delete__FP13daObjWchain_c */ static int daObjWchain_Delete(daObjWchain_c* i_this) { i_this->~daObjWchain_c(); return 1; } /* 80D30534-80D305E4 0006B4 00B0+00 2/2 0/0 0/0 .text setMatrix__13daObjWchain_cFv */ void daObjWchain_c::setMatrix() { mDoMtx_stack_c::transS(current.pos.x, current.pos.y, current.pos.z); mDoMtx_stack_c::ZrotM(mHandleRotation); mDoMtx_stack_c::ZXYrotM(shape_angle.x, shape_angle.y, shape_angle.z); mDoMtx_stack_c::transM(0.0f, 0.0f, 8.75f); mpHandleModel->setBaseTRMtx(mDoMtx_stack_c::get()); static Vec const eyeOffset = {0.0f, 0.0f, 53.75f}; mDoMtx_stack_c::multVec(&eyeOffset, &eyePos); } /* 80D305E4-80D3080C 000764 0228+00 1/1 0/0 0/0 .text getChainAngleZ__13daObjWchain_cFP4cXyzi */ s16 daObjWchain_c::getChainAngleZ(cXyz* param_0, int param_1) { cXyz vec(param_0->x, 0.0f, param_0->z); f32 len = vec.abs(); if (param_1 > 0x5000) { return -len * (cM_rndF(0.5f) + 0.5f) * 512.0f; } else if (param_1 < 0x3000) { return len * (cM_rndF(0.5f) + 0.5f) * 512.0f; } else if (cM_rnd() < 0.5f) { return -len * (cM_rndF(0.5f) + 0.5f) * 512.0f; } else { return len * (cM_rndF(0.5f) + 0.5f) * 512.0f; } } /* 80D3080C-80D310AC 00098C 08A0+00 1/1 0/0 0/0 .text setChainPos__13daObjWchain_cFv */ // NONMATCHING regalloc, instruction ordering void daObjWchain_c::setChainPos() { cXyz prev_pos, vec1; if (mRide) { shape_angle.y = daPy_getLinkPlayerActorClass()->shape_angle.y; shape_angle.z = 0; prev_pos = mTopPos; mTopPos = daPy_getLinkPlayerActorClass()->current.pos; speed = (mTopPos - prev_pos) * 0.75f; vec1 = mTopPos - mRoofPos; mDoMtx_stack_c::YrotS(-shape_angle.y); mDoMtx_stack_c::multVec(&vec1, &vec1); shape_angle.x = cM_atan2s(-vec1.z, -vec1.y) + 0x4000; mHandleRotation = cM_atan2s(vec1.x, JMAFastSqrt(vec1.y * vec1.y + vec1.z * vec1.z)); mDoMtx_stack_c::transS(mTopPos.x, mTopPos.y, mTopPos.z); mDoMtx_stack_c::ZrotM(mHandleRotation); mDoMtx_stack_c::ZXYrotM(shape_angle.x, shape_angle.y, shape_angle.z); static Vec const currentOffset = {0.0f, 0.0f, -53.75f}; mDoMtx_stack_c::multVec(¤tOffset, ¤t.pos); cXyz* chain_pos = &mChainPos[0xf]; csXyz* chain_angle = &mChainAngle[0xf]; cXyz* chain_speed = &mChainSpeed[0xf]; s16* chain_rotation = &mChainRotation[0xf]; prev_pos = current.pos; int svar7 = shape_angle.z; for (int i = 0xf; i >= 0; i--, chain_pos--, chain_angle--, chain_speed--, chain_rotation--) { chain_angle->z += getChainAngleZ(chain_speed, abs((s16)(chain_angle->z - svar7))); *chain_speed = (prev_pos - *chain_pos) * 0.75; *chain_pos = prev_pos; chain_angle->x = shape_angle.x; *chain_rotation = mHandleRotation; mDoMtx_stack_c::transS(prev_pos.x, prev_pos.y, prev_pos.z); mDoMtx_stack_c::ZrotM(*chain_rotation); mDoMtx_stack_c::ZXYrotM(chain_angle->x, shape_angle.y, chain_angle->z); static Vec const chainOffset = {0.0f, 0.0f, -17.5f}; mDoMtx_stack_c::multVec(&chainOffset, &prev_pos); svar7 = chain_angle->z; } } else { if (mReset) { if (cLib_chaseF(&mPullLength, 0.0f, 1.0f)) { mReset = false; mEnd = false; } } else if (mRepeatable || !fopAcM_isSwitch(this, mSw)) { cLib_chaseF(&mPullLength, 0.0f, 5.0f); } else { mPullLength = 100.0f; mEnd = true; if (!mRepeatable) { cLib_chaseF(&field_0x7a8, 600.0f, 30.0f); } } f32 fvar2 = mInitOutLength + mPullLength - 53.75f; int local_68 = fvar2 * (1.0f / 17.5f); int ivar5 = local_68 <= 0xf ? local_68 + 1 : 0x10; int chain_no = 0x10 - ivar5; cXyz* chain_pos = &mChainPos[chain_no]; csXyz* chain_angle = &mChainAngle[chain_no]; s16* chain_rotation = &mChainRotation[chain_no]; chain_pos->set( mRoofPos.x, field_0x7a8 + (mRoofPos.y - (17.5f - (ivar5 * 17.5f - fvar2))), mRoofPos.z ); chain_angle->x = 0x4000; *chain_rotation = 0; chain_pos = mChainPos + 1 + chain_no; chain_angle++; cXyz* chain_speed = mChainSpeed + 1 + chain_no; chain_rotation++; mDoMtx_stack_c::YrotS(-shape_angle.y); f32 prob = 0.2f; cXyz local_90; if (!mEnd && cM_rnd() < prob) { f32 ang = cM_rnd() * 6.283185f; local_90.set(cM_fsin(ang), 0.0f, cM_fcos(ang)); } else { local_90 = cXyz::Zero; } for (int i = chain_no + 1; i < 0x10; i++, chain_pos++, chain_angle++, chain_speed++, chain_rotation++) { prev_pos = *chain_pos; vec1 = *chain_pos - chain_pos[-1]; if (chain_speed->abs2XZ() < 0.04f && cM_rnd() < prob) { vec1 += local_90; prob *= 0.5f; } vec1.y += gravity; vec1 += *chain_speed; chain_angle->z += getChainAngleZ(chain_speed, abs((s16)(chain_angle->z - chain_angle[-1].z))); vec1.normalizeZP(); *chain_pos = chain_pos[-1] + vec1 * 17.5f; *chain_speed = (*chain_pos - prev_pos) * 0.75f; mDoMtx_stack_c::multVec(&vec1, &vec1); chain_angle->x = cM_atan2s(-vec1.z, -vec1.y) + 0x4000; *chain_rotation = cM_atan2s(vec1.x, JMAFastSqrt(vec1.y * vec1.y + vec1.z * vec1.z)); } current.pos = mChainPos[0xf]; shape_angle.z += getChainAngleZ(&speed, abs((s16)(shape_angle.z - mChainAngle[0xf].z))); prev_pos = mTopPos; vec1 = mTopPos - current.pos; vec1.y += gravity; vec1 += speed; vec1.normalizeZP(); mTopPos = current.pos + vec1 * 53.75f; speed = (mTopPos - prev_pos) * 0.75f; mDoMtx_stack_c::multVec(&vec1, &vec1); shape_angle.x = cM_atan2s(-vec1.z, -vec1.y) + 0x4000; mHandleRotation = cM_atan2s(vec1.x, JMAFastSqrt(vec1.y * vec1.y + vec1.z * vec1.z)); } } /* 80D310AC-80D313F8 00122C 034C+00 1/1 0/0 0/0 .text execute__13daObjWchain_cFv */ int daObjWchain_c::execute() { if (!mRidePrev && mRide) { fopAcM_seStartCurrent(this, Z2SE_OBJ_GNAW_CHAIN_SW, 0); } else if (mRidePrev && !mRide) { fopAcM_seStartCurrent(this, Z2SE_OBJ_GNAW_CHAIN_SW, 0); } if (!mRide) { mDown = false; } if (fopAcM_rc_c::roofCheck(&home.pos)) { mRealRoofY = fopAcM_rc_c::getRoofY(); if (mRoofPos.y > mRealRoofY) { mRoofPos.y = mRealRoofY; mInitOutLength = mRoofPos.y - home.pos.y; } mRealRoofY += 250.0f; } setChainPos(); if (daPy_py_c::i_checkNowWolf() && !mRide && mPullLength < 0.1f) { attention_info.flags |= 1; } else { attention_info.flags &= ~1; } attention_info.position = current.pos; attention_info.position.y += 150.0f; setMatrix(); if (mNowSwitch) { fopAcM_seStartCurrent(this, Z2SE_OBJ_STOP_CHAIN_SW, 0); mNowSwitch = false; if (mRepeatable) { mReset = true; if (fopAcM_isSwitch(this, mSw)) { fopAcM_offSwitch(this, mSw); } else { fopAcM_onSwitch(this, mSw); } } else { fopAcM_onSwitch(this, mSw); } field_0x77e = 20; } else if (mRide && !mDown) { fopAcM_seStartCurrentLevel(this, Z2SE_OBJ_PULLDOWN_CHAIN_SW, 0); } mRidePrev = mRide; return 1; } /* 80D313F8-80D31418 001578 0020+00 1/0 0/0 0/0 .text daObjWchain_Execute__FP13daObjWchain_c */ static int daObjWchain_Execute(daObjWchain_c* i_this) { return i_this->execute(); } /* 80D31418-80D31810 001598 03F8+00 1/0 0/0 0/0 .text draw__19daObjWchain_shape_cFv */ void daObjWchain_shape_c::draw() { daObjWchain_c* chain = (daObjWchain_c*)getUserArea(); cXyz* pos = chain->getChainPos(); csXyz* angle = chain->getChainAngle(); s16* rotation = chain->getChainAngleZ(); J3DModelData* model_data = chain->getChainModelData(); J3DMaterial* material = model_data->getMaterialNodePointer(0); dKy_tevstr_c& tevstr = chain->tevStr; j3dSys.setVtxPos(model_data->getVtxPosArray()); j3dSys.setVtxNrm(model_data->getVtxNrmArray()); j3dSys.setVtxCol(model_data->getVtxColorArray(0)); J3DShape::resetVcdVatCache(); material->loadSharedDL(); material->getShape()->loadPreDrawSetting(); GXColor amb_color; amb_color.r = tevstr.AmbCol.r; amb_color.g = tevstr.AmbCol.g; amb_color.b = tevstr.AmbCol.b; amb_color.a = tevstr.AmbCol.a; GXSetChanAmbColor(GX_COLOR0A0, amb_color); GXSetChanMatColor(GX_COLOR0A0, g_whiteColor); dKy_setLight_again(); dKy_GxFog_tevstr_set(&tevstr); GXLoadLightObjImm(&tevstr.mLightObj.mLightObj, GX_LIGHT0); for (int i = 0; i < 0x10; pos++, angle++, rotation++, i++) { mDoMtx_stack_c::copy(j3dSys.getViewMtx()); mDoMtx_stack_c::transM(*pos); mDoMtx_stack_c::ZrotM(*rotation); mDoMtx_stack_c::ZXYrotM(angle->x, chain->shape_angle.y, angle->z); mDoMtx_stack_c::transM(0.0f, 0.0f, -8.75f); GXLoadPosMtxImm(mDoMtx_stack_c::get(), 0); GXLoadNrmMtxImm(mDoMtx_stack_c::get(), 0); material->getShape()->simpleDrawCache(); } cXyz roof_pos(chain->getRoofPos().x, chain->getRealRoofY(), chain->getRoofPos().z); cXyz delta = roof_pos - *chain->getChainPos(); f32 len = delta.abs(); if (len > 17.5f) { cXyz pos = *chain->getChainPos(); csXyz angle( delta.atan2sY_XZ(), chain->getChainAngle()->y, chain->getChainAngle()->z + 0x3000 ); delta *= (17.5f / len); for (; len > 17.5f; len -= 17.5f, pos += delta, angle.z += 0x3000) { mDoMtx_stack_c::copy(j3dSys.getViewMtx()); mDoMtx_stack_c::transM(pos); mDoMtx_stack_c::ZXYrotM(angle); mDoMtx_stack_c::transM(0.0f, 0.0f, 8.75f); GXLoadPosMtxImm(mDoMtx_stack_c::get(), 0); GXLoadNrmMtxImm(mDoMtx_stack_c::get(), 0); material->getShape()->simpleDrawCache(); } } } /* 80D31810-80D318A0 001990 0090+00 1/1 0/0 0/0 .text draw__13daObjWchain_cFv */ int daObjWchain_c::draw() { g_env_light.settingTevStruct(0, ¤t.pos, &tevStr); g_env_light.setLightTevColorType_MAJI(mpHandleModel, &tevStr); mDoExt_modelUpdateDL(mpHandleModel); g_env_light.setLightTevColorType_MAJI(mpChainModelData, &tevStr); dComIfGd_getOpaList()->entryImm(&mShape, 0); return 1; } /* 80D318A0-80D318C0 001A20 0020+00 1/0 0/0 0/0 .text daObjWchain_Draw__FP13daObjWchain_c */ static int daObjWchain_Draw(daObjWchain_c* i_this) { return i_this->draw(); } /* 80D319E4-80D31A04 -00001 0020+00 1/0 0/0 0/0 .data l_daObjWchain_Method */ static actor_method_class l_daObjWchain_Method = { (process_method_func)daObjWchain_Create, (process_method_func)daObjWchain_Delete, (process_method_func)daObjWchain_Execute, NULL, (process_method_func)daObjWchain_Draw, }; /* 80D31A04-80D31A34 -00001 0030+00 0/0 0/0 1/0 .data g_profile_Obj_Wchain */ extern actor_process_profile_definition g_profile_Obj_Wchain = { fpcLy_CURRENT_e, // mLayerID 7, // mListID fpcPi_CURRENT_e, // mListPrio PROC_Obj_Wchain, // mProcName &g_fpcLf_Method.base, // sub_method 0x000007BC, // mSize 0, // mSizeOther 0, // mParameters &g_fopAc_Method.base, // sub_method 247, // mPriority &l_daObjWchain_Method, // sub_method 0x00060100, // mStatus fopAc_ENV_e, // mActorType fopAc_CULLBOX_CUSTOM_e, // cullType };