ksys/res: Implement BoneControl

This commit is contained in:
Léo Lam 2021-03-20 15:49:25 +01:00
parent 737a9b522b
commit 5737a6ea7c
No known key found for this signature in database
GPG Key ID: 0DF30F9081000741
4 changed files with 422 additions and 25 deletions

View File

@ -96693,31 +96693,31 @@
0x00000071012f7028,sub_71012F7028,340,_ZThn664_N4ksys3res9AwarenessD1Ev
0x00000071012f717c,sub_71012F717C,328,_ZThn664_N4ksys3res9AwarenessD0Ev
0x00000071012f72c4,sub_71012F72C4,372,_ZN4ksys3res9Awareness33checkDerivedRuntimeTypeInfoStaticEPKN4sead15RuntimeTypeInfo9InterfaceE
0x00000071012f7438,sub_71012F7438,520,
0x00000071012f7640,sub_71012F7640,940,
0x00000071012f79ec,sub_71012F79EC,8,
0x00000071012f79f4,sub_71012F79F4,8,
0x00000071012f79fc,sub_71012F79FC,36,
0x00000071012f7a20,sub_71012F7A20,40,
0x00000071012f7a48,sub_71012F7A48,40,
0x00000071012f7a70,nullsub_4835,4,
0x00000071012f7a74,nullsub_4836,4,
0x00000071012f7a78,sub_71012F7A78,7664,
0x00000071012f9868,sub_71012F9868,312,
0x00000071012f99a0,sub_71012F99A0,8,
0x00000071012f99a8,sub_71012F99A8,452,
0x00000071012f9b6c,sub_71012F9B6C,8,
0x00000071012f9b74,sub_71012F9B74,8,
0x00000071012f9b7c,sub_71012F9B7C,92,
0x00000071012f9bd8,sub_71012F9BD8,8,
0x00000071012f9be0,sub_71012F9BE0,8,
0x00000071012f9be8,sub_71012F9BE8,92,
0x00000071012f9c44,sub_71012F9C44,8,
0x00000071012f9c4c,sub_71012F9C4C,252,
0x00000071012f9d48,sub_71012F9D48,296,
0x00000071012f9e70,sub_71012F9E70,372,
0x00000071012f9fe4,sub_71012F9FE4,384,
0x00000071012fa164,sub_71012FA164,368,
0x00000071012f7438,sub_71012F7438,520,_ZN4ksys3res11BoneControlC1Ev
0x00000071012f7640,sub_71012F7640,940,_ZN4ksys3res11BoneControlD1Ev
0x00000071012f79ec,sub_71012F79EC,8,_ZThn632_N4ksys3res11BoneControlD1Ev
0x00000071012f79f4,sub_71012F79F4,8,_ZThn664_N4ksys3res11BoneControlD1Ev
0x00000071012f79fc,sub_71012F79FC,36,_ZN4ksys3res11BoneControlD0Ev
0x00000071012f7a20,sub_71012F7A20,40,_ZThn632_N4ksys3res11BoneControlD0Ev
0x00000071012f7a48,sub_71012F7A48,40,_ZThn664_N4ksys3res11BoneControlD0Ev
0x00000071012f7a70,nullsub_4835,4,_ZN4ksys3res11BoneControl9doCreate_EPhjPN4sead4HeapE
0x00000071012f7a74,nullsub_4836,4,_ZThn632_N4ksys3res11BoneControl9doCreate_EPhjPN4sead4HeapE
0x00000071012f7a78,sub_71012F7A78,7664,_ZN4ksys3res11BoneControl6parse_EPhmPN4sead4HeapE?
0x00000071012f9868,sub_71012F9868,312,_ZN4ksys3res15allocBoneGroupsERN4sead6BufferINS0_11BoneControl9BoneGroupEEEiPNS1_4HeapEi
0x00000071012f99a0,sub_71012F99A0,8,_ZThn632_N4ksys3res11BoneControl6parse_EPhmPN4sead4HeapE
0x00000071012f99a8,sub_71012F99A8,452,_ZNK4ksys3res11BoneControl12getBoneGroupERKN4sead14SafeStringBaseIcEE
0x00000071012f9b6c,sub_71012F9B6C,8,_ZN4ksys3res11BoneControl10ParamIO_m0Ev
0x00000071012f9b74,sub_71012F9B74,8,_ZNK4ksys3res11BoneControl27checkDerivedRuntimeTypeInfoEPKN4sead15RuntimeTypeInfo9InterfaceE
0x00000071012f9b7c,sub_71012F9B7C,92,_ZNK4ksys3res11BoneControl18getRuntimeTypeInfoEv
0x00000071012f9bd8,sub_71012F9BD8,8,_ZNK4ksys3res11BoneControl10needsParseEv
0x00000071012f9be0,sub_71012F9BE0,8,_ZThn632_NK4ksys3res11BoneControl27checkDerivedRuntimeTypeInfoEPKN4sead15RuntimeTypeInfo9InterfaceE
0x00000071012f9be8,sub_71012F9BE8,92,_ZThn632_NK4ksys3res11BoneControl18getRuntimeTypeInfoEv
0x00000071012f9c44,sub_71012F9C44,8,_ZThn632_NK4ksys3res11BoneControl10needsParseEv
0x00000071012f9c4c,sub_71012F9C4C,252,_ZN4ksys3res11BoneControl16FootIkControllerC2Ev
0x00000071012f9d48,sub_71012F9D48,296,_ZN4ksys3res11BoneControl5SpineC2Ev
0x00000071012f9e70,sub_71012F9E70,372,_ZN4ksys3res11BoneControl33checkDerivedRuntimeTypeInfoStaticEPKN4sead15RuntimeTypeInfo9InterfaceE
0x00000071012f9fe4,sub_71012F9FE4,384,_ZN4ksys3res11BoneControl6EyeSetC2Ev
0x00000071012fa164,sub_71012FA164,368,_ZN4ksys3res11BoneControl9SpineNodeC2Ev
0x00000071012fa2d4,sub_71012FA2D4,272,
0x00000071012fa3e4,nullsub_4837,4,
0x00000071012fa3e8,nullsub_4838,4,

Can't render this file because it is too large.

View File

@ -1 +1,244 @@
#include "KingSystem/Resource/resResourceBoneControl.h"
namespace ksys::res {
BoneControl::BoneControl() : ParamIO("bonectrl", 0) {}
BoneControl::~BoneControl() {
mEyeSets.freeBuffer();
mSpine.spineNodes.freeBuffer();
for (auto& group : mBoneGroups)
group.bones.freeBuffer();
mBoneGroups.freeBuffer();
}
void BoneControl::doCreate_(u8* buffer, u32 buffer_size, sead::Heap* heap) {}
bool allocBoneGroups(sead::Buffer<BoneControl::BoneGroup>& buffer, int size, sead::Heap* heap,
int align = sizeof(void*)) {
return buffer.tryAllocBuffer(size, heap, align);
}
// NON_MATCHING: mFootIkController.isInvalidFt (???)
bool BoneControl::parse_(u8* data, size_t size, sead::Heap* heap) {
if (!data)
return true;
agl::utl::ResParameterArchive archive{data};
const auto root = archive.getRootList();
mWhole.neckAndEyeRatio.init(0.0, "neckAndEyeRatio", "首向けと眼球制御の比率", &mWhole.obj);
mWhole.isFaceCtrlInvalid.init(true, "isFaceCtrlInvalid", "顔全体無効", &mWhole.obj);
addObj(&mWhole.obj, "Whole");
mEyeBall.isEyeBallCtrlInvalid.init(false, "isEyeBallCtrlInvalid", "無効にする", &mEyeBall.obj);
mEyeBall.isEyeBallRotWorldAxis.init(false, "isEyeBallRotWorldAxis", "ワールド軸で回転する",
&mEyeBall.obj);
mEyeBall.eyeBallSRTName.init("", "eyeBallSRTName", "眼球SRT名", &mEyeBall.obj);
mEyeBall.eyeRotRateLR.init(0.0, "eyeRotRateLR", "左右回転比率", &mEyeBall.obj);
mEyeBall.eyeRotRateUD.init(0.0, "eyeRotRateUD", "上下回転比率", &mEyeBall.obj);
mEyeBall.eyeMinRotPerFrame.init(0.5, "eyeMinRotPerFrame", "フレーム毎の最小回転量",
&mEyeBall.obj);
mEyeBall.eyeMaxRotPerFrame.init(6.0, "eyeMaxRotPerFrame", "フレーム毎の最大回転量",
&mEyeBall.obj);
mEyeBall.eyeSetNum.init(0, "eyeSetNum", "眼球セット数", &mEyeBall.obj);
addObj(&mEyeBall.obj, "EyeBall");
applyResParameterArchive(agl::utl::ResParameterArchive{data});
const int eye_set_num = mEyeBall.eyeSetNum.ref();
if (eye_set_num > 0) {
if (!mEyeSets.tryAllocBuffer(eye_set_num, heap))
return false;
for (int i = 0; i < eye_set_num; ++i) {
sead::FormatFixedSafeString<64> name("");
name.format("EyeSet_%02d", i);
mEyeSets[i].isControlTexture.init(false, "isControlTexture", "テクスチャ制御する",
&mEyeSets[i].obj);
mEyeSets[i].materialName.init("", "materialName", "マテリアル名", &mEyeSets[i].obj);
mEyeSets[i].boneName.init("", "boneName", "ボーン名", &mEyeSets[i].obj);
mEyeSets[i].forwardBoneName.init("", "forwardBoneName", "前方方向ボーン名",
&mEyeSets[i].obj);
mEyeSets[i].forwardAxis.init(1, "forwardAxis", "前方方向とする軸", &mEyeSets[i].obj);
mEyeSets[i].axisLR.init(0, "axisLR", "左右回転軸", &mEyeSets[i].obj);
mEyeSets[i].axisUD.init(0, "axisUD", "上下回転軸", &mEyeSets[i].obj);
mEyeSets[i].lTransLimit.init(0.0, "lTransLimit", "左移動量上限", &mEyeSets[i].obj);
mEyeSets[i].rTransLimit.init(0.0, "rTransLimit", "右移動量上限", &mEyeSets[i].obj);
mEyeSets[i].dTransLimit.init(0.0, "dTransLimit", "下移動量上限", &mEyeSets[i].obj);
mEyeSets[i].uTransLimit.init(0.0, "uTransLimit", "上移動量上限", &mEyeSets[i].obj);
mEyeSets[i].isCorrectForward.init(false, "isCorrectForward", "前方方向を補正する",
&mEyeSets[i].obj);
mEyeSets[i].axisCorrect.init(0, "axisCorrect", "補正軸", &mEyeSets[i].obj);
mEyeSets[i].correctRot.init(0.0, "correctRot", "補正量", &mEyeSets[i].obj);
mEyeSets[i].lRotLimit.init(0.0, "lRotLimit", "左向き角上限", &mEyeSets[i].obj);
mEyeSets[i].rRotLimit.init(0.0, "rRotLimit", "右向き角上限", &mEyeSets[i].obj);
mEyeSets[i].dRotLimit.init(0.0, "dRotLimit", "下向き角上限", &mEyeSets[i].obj);
mEyeSets[i].uRotLimit.init(0.0, "uRotLimit", "上向き角上限", &mEyeSets[i].obj);
mEyeSets[i].offset.init({0.0, 0.0, 0.0}, "offset", "オフセット", &mEyeSets[i].obj);
addObj(&mEyeSets[i].obj, name);
}
}
mSpine.isInvalid.init(false, "isInvalid", "無効にする", &mSpine.obj);
mSpine.isBasisSelfPosNeckLR.init(false, "isBasisSelfPosNeckLR",
"左右計算を自分の位置基準にする", &mSpine.obj);
mSpine.isBasisSelfPosNeckUD.init(false, "isBasisSelfPosNeckUD",
"上下計算を自分の位置基準にする", &mSpine.obj);
mSpine.isBattleNeckRecalcUD.init(false, "isBattleNeckRecalcUD",
"戦闘時の首向け上下角を再計算する", &mSpine.obj);
mSpine.spineDisableBaseDirAlongXZ.init(false, "spineDisableBaseDirAlongXZ",
"基準方向をXZ平面に沿わせない", &mSpine.obj);
mSpine.spineRotRate.init(0.0, "spineRotRate", "回転比率", &mSpine.obj);
mSpine.spineRetRotRate.init(0.0, "spineRetRotRate", "戻り回転比率", &mSpine.obj);
mSpine.spineNeckBaseBone.init("", "spineNeckBaseBone", "首向け基準位置ボーン名", &mSpine.obj);
mSpine.neckPosOffset.init(sead::Vector3f::zero, "neckPosOffset", "オフセット", &mSpine.obj);
mSpine.spineNodeNum.init(0, "spineNodeNum", "背骨ノード数", &mSpine.obj);
mSpine.spineNeckNodeNum.init(0, "spineNeckNodeNum", "首とみなすノード数", &mSpine.obj);
addObj(&mSpine.obj, "Spine");
applyResParameterArchive(agl::utl::ResParameterArchive{data});
const auto spine_node_num = mSpine.spineNodeNum.ref();
if (spine_node_num > 0) {
if (!mSpine.spineNodes.tryAllocBuffer(spine_node_num, heap))
return false;
for (int i = 0; i < spine_node_num; ++i) {
sead::FormatFixedSafeString<64> name("");
name.format("SpineNode_%02d", i);
mSpine.spineNodes[i].boneName.init("", "boneName", "ボーン名",
&mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].isRotWorldAxis.init(
false, "isRotWorldAxis", "ワールド軸で回転する", &mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].axisLR.init(0, "axisLR", "左右回転軸", &mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].axisUD.init(0, "axisUD", "上下回転軸", &mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].lLimit.init(0.0, "lLimit", "左向き角上限",
&mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].rLimit.init(0.0, "rLimit", "右向き角上限",
&mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].dLimit.init(0.0, "dLimit", "下向き角上限",
&mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].uLimit.init(0.0, "uLimit", "上向き角上限",
&mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].lBattleLimit.init(0.0, "lBattleLimit", "左向き角上限(戦闘時)",
&mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].rBattleLimit.init(0.0, "rBattleLimit", "右向き角上限(戦闘時)",
&mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].dBattleLimit.init(0.0, "dBattleLimit", "下向き角上限(戦闘時)",
&mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].uBattleLimit.init(0.0, "uBattleLimit", "上向き角上限(戦闘時)",
&mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].minRotPerFrame.init(
0.5, "minRotPerFrame", "フレーム毎の最小回転量", &mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].maxRotPerFrame.init(
6.0, "maxRotPerFrame", "フレーム毎の最大回転量", &mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].isEnableCorrect.init(
false, "isEnableCorrect", "左右回転時に補正する", &mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].axisCorrect.init(0, "axisCorrect", "補正回転軸",
&mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].lCorrect.init(0.0, "lCorrect", "左向き補正回転量",
&mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].rCorrect.init(0.0, "rCorrect", "右向き補正回転量",
&mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].lBattleCorrect.init(
0.0, "lBattleCorrect", "左向き補正回転量(戦闘時)", &mSpine.spineNodes[i].obj);
mSpine.spineNodes[i].rBattleCorrect.init(
0.0, "rBattleCorrect", "右向き補正回転量(戦闘時)", &mSpine.spineNodes[i].obj);
addObj(&mSpine.spineNodes[i].obj, name);
}
}
mFootIkController.isInvalidFt.init(true, "isInvalidFt", "無効", &mFootIkController.obj);
mFootIkController.calculateTypeFt.init(1, "calculateTypeFt", "計算タイプ",
&mFootIkController.obj);
mFootIkController.ankleOffsetYFt.init(0.125, "ankleOffsetYFt", "地面から足首(Ankle)までの高さ",
&mFootIkController.obj);
mFootIkController.ankleOffsetAngleDegFt.init(-27.0, "ankleOffsetAngleDegFt",
"足首オフセット角度(Deg)", &mFootIkController.obj);
mFootIkController.ankleAngleLimitUpDegFt.init(
90.0, "ankleAngleLimitUpDegFt", "上方向への足首回転最大角度(Deg)", &mFootIkController.obj);
mFootIkController.ankleAngleLimitDownDegFt.init(-90.0, "ankleAngleLimitDownDegFt",
"下方向への足首回転最大角度(Deg)",
&mFootIkController.obj);
mFootIkController.ankleHeightLimitRateFt.init(
0.8, "ankleHeightLimitRateFt", "地面に対して足位置の制限比率", &mFootIkController.obj);
mFootIkController.waistDownRateFt.init(0.7, "waistDownRateFt", "腰を落とす最長比率",
&mFootIkController.obj);
mFootIkController.kneeRotateAgnleMinDegFt.init(
0.0, "kneeRotateAgnleMinDegFt", "ヒザの最小回転角度(Deg)", &mFootIkController.obj);
mFootIkController.kneeRotateAgnleMaxDegFt.init(
180.0, "kneeRotateAgnleMaxDegFt", "ヒザの最大回転角度(Deg)", &mFootIkController.obj);
mFootIkController.enableLimitThighAngleFt.init(
false, "enableLimitThighAngleFt", "モモの角度制限を行なうか?", &mFootIkController.obj);
mFootIkController.thighRotateAngleMinDegFt.init(
-180.0, "thighRotateAngleMinDegFt", "モモの最小回転角度(Deg)", &mFootIkController.obj);
mFootIkController.thighRotateAngleMaxDegFt.init(
180.0, "thighRotateAngleMaxDegFt", "モモの最大回転角度(Deg)", &mFootIkController.obj);
addObj(&mFootIkController.obj, "FootIkController");
const auto bone_groups = agl::utl::getResParameterList(root, "BoneGroups");
if (bone_groups.ptr() && bone_groups.getResParameterListNum() != 0) {
if (!allocBoneGroups(mBoneGroups, bone_groups.getResParameterListNum(), heap))
return false;
sead::FixedSafeString<32> bone_group_name{"BoneGroup_"};
const auto bone_group_name_base_len = bone_group_name.calcLength();
sead::FixedSafeString<32> bone_name{"Bone_"};
const auto bone_name_base_len = bone_name.calcLength();
for (auto it = mBoneGroups.begin(), end = mBoneGroups.end(); it != end; ++it) {
const auto list = bone_groups.getResParameterList(it.getIndex());
if (!list.ptr())
continue;
const auto bones = agl::utl::getResParameterObj(list, "Bones");
if (!bones.ptr())
continue;
it->groupName.init("", "GroupName", "グループ名", &it->paramObj);
const auto num_bones = bones.getNum();
if (num_bones != 0 && !it->bones.tryAllocBuffer(num_bones, heap))
return false;
auto& bones_obj = it->bonesObj;
for (auto b = it->bones.begin(), bone_end = it->bones.end(); b != bone_end; ++b) {
bone_name.trim(bone_name_base_len);
bone_name.appendWithFormat("%d", b.getIndex());
b->name.init("", bone_name, "ボーン名", &bones_obj);
}
it->list.addObj(&bones_obj, "Bones");
it->list.addObj(&it->paramObj, "Param");
bone_group_name.trim(bone_group_name_base_len);
bone_group_name.appendWithFormat("%d", it.getIndex());
mBoneGroupsList.addList(&it->list, bone_group_name);
}
addList(&mBoneGroupsList, "BoneGroups");
}
applyResParameterArchive(agl::utl::ResParameterArchive{data});
return true;
}
const BoneControl::BoneGroup* BoneControl::getBoneGroup(const sead::SafeString& name) const {
const auto idx = mBoneGroups.binarySearch(
name, +[](const BoneGroup& group, const sead::SafeString& key) {
return group.groupName.ref().compare(key);
});
if (idx == -1)
return nullptr;
return &mBoneGroups[idx];
}
} // namespace ksys::res

View File

@ -1,13 +1,167 @@
#pragma once
#include <agl/Utils/aglParameter.h>
#include <agl/Utils/aglParameterList.h>
#include <agl/Utils/aglParameterObj.h>
#include <container/seadBuffer.h>
#include <math/seadVector.h>
#include <prim/seadSafeString.h>
#include "KingSystem/Resource/resResource.h"
#include "KingSystem/Utils/ParamIO.h"
#include "KingSystem/Utils/Types.h"
namespace ksys::res {
// TODO
class BoneControl : public ParamIO, public Resource {
SEAD_RTTI_OVERRIDE(BoneControl, Resource)
public:
struct Whole {
agl::utl::ParameterObj obj;
agl::utl::Parameter<bool> isFaceCtrlInvalid;
agl::utl::Parameter<float> neckAndEyeRatio;
};
KSYS_CHECK_SIZE_NX150(Whole, 0x70);
struct EyeBall {
agl::utl::ParameterObj obj;
agl::utl::Parameter<bool> isEyeBallCtrlInvalid;
agl::utl::Parameter<bool> isEyeBallRotWorldAxis;
agl::utl::Parameter<sead::SafeString> eyeBallSRTName;
agl::utl::Parameter<float> eyeRotRateLR;
agl::utl::Parameter<float> eyeRotRateUD;
agl::utl::Parameter<float> eyeMinRotPerFrame;
agl::utl::Parameter<float> eyeMaxRotPerFrame;
agl::utl::Parameter<int> eyeSetNum;
};
KSYS_CHECK_SIZE_NX150(EyeBall, 0x138);
struct EyeSet {
agl::utl::ParameterObj obj;
agl::utl::Parameter<sead::SafeString> boneName;
agl::utl::Parameter<bool> isControlTexture;
agl::utl::Parameter<sead::SafeString> materialName;
agl::utl::Parameter<sead::SafeString> forwardBoneName;
agl::utl::Parameter<int> forwardAxis;
agl::utl::Parameter<int> axisLR;
agl::utl::Parameter<int> axisUD;
agl::utl::Parameter<float> lTransLimit;
agl::utl::Parameter<float> rTransLimit;
agl::utl::Parameter<float> dTransLimit;
agl::utl::Parameter<float> uTransLimit;
agl::utl::Parameter<bool> isCorrectForward;
agl::utl::Parameter<int> axisCorrect;
agl::utl::Parameter<float> correctRot;
agl::utl::Parameter<float> lRotLimit;
agl::utl::Parameter<float> rRotLimit;
agl::utl::Parameter<float> dRotLimit;
agl::utl::Parameter<float> uRotLimit;
agl::utl::Parameter<sead::Vector3f> offset;
};
KSYS_CHECK_SIZE_NX150(EyeSet, 0x2b0);
struct Bone {
agl::utl::Parameter<sead::SafeString> name;
};
KSYS_CHECK_SIZE_NX150(Bone, 0x28);
struct BoneGroup {
agl::utl::ParameterList list;
agl::utl::ParameterObj paramObj;
agl::utl::ParameterObj bonesObj;
agl::utl::Parameter<sead::SafeString> groupName;
sead::Buffer<Bone> bones;
};
KSYS_CHECK_SIZE_NX150(BoneGroup, 0xe0);
struct SpineNode {
agl::utl::ParameterObj obj;
agl::utl::Parameter<sead::SafeString> boneName;
agl::utl::Parameter<bool> isRotWorldAxis;
agl::utl::Parameter<int> axisLR;
agl::utl::Parameter<int> axisUD;
agl::utl::Parameter<float> lLimit;
agl::utl::Parameter<float> rLimit;
agl::utl::Parameter<float> dLimit;
agl::utl::Parameter<float> uLimit;
agl::utl::Parameter<float> lBattleLimit;
agl::utl::Parameter<float> rBattleLimit;
agl::utl::Parameter<float> dBattleLimit;
agl::utl::Parameter<float> uBattleLimit;
agl::utl::Parameter<bool> isEnableCorrect;
agl::utl::Parameter<int> axisCorrect;
agl::utl::Parameter<float> lCorrect;
agl::utl::Parameter<float> rCorrect;
agl::utl::Parameter<float> lBattleCorrect;
agl::utl::Parameter<float> rBattleCorrect;
agl::utl::Parameter<float> minRotPerFrame;
agl::utl::Parameter<float> maxRotPerFrame;
};
KSYS_CHECK_SIZE_NX150(SpineNode, 0x2b8);
struct Spine {
agl::utl::ParameterObj obj;
agl::utl::ParameterList list;
agl::utl::Parameter<bool> isInvalid;
agl::utl::Parameter<bool> isBasisSelfPosNeckLR;
agl::utl::Parameter<bool> isBasisSelfPosNeckUD;
agl::utl::Parameter<bool> isBattleNeckRecalcUD;
agl::utl::Parameter<bool> spineDisableBaseDirAlongXZ;
agl::utl::Parameter<float> spineRotRate;
agl::utl::Parameter<float> spineRetRotRate;
agl::utl::Parameter<sead::SafeString> spineNeckBaseBone;
agl::utl::Parameter<sead::Vector3f> neckPosOffset;
agl::utl::Parameter<int> spineNeckNodeNum;
agl::utl::Parameter<int> spineNodeNum;
sead::Buffer<SpineNode> spineNodes;
};
KSYS_CHECK_SIZE_NX150(Spine, 0x1f8);
struct FootIkController {
agl::utl::ParameterObj obj;
agl::utl::Parameter<bool> isInvalidFt;
agl::utl::Parameter<int> calculateTypeFt;
agl::utl::Parameter<float> ankleOffsetYFt;
agl::utl::Parameter<float> ankleOffsetAngleDegFt;
agl::utl::Parameter<float> ankleAngleLimitUpDegFt;
agl::utl::Parameter<float> ankleAngleLimitDownDegFt;
agl::utl::Parameter<float> ankleHeightLimitRateFt;
agl::utl::Parameter<float> waistDownRateFt;
agl::utl::Parameter<float> kneeRotateAgnleMinDegFt;
agl::utl::Parameter<float> kneeRotateAgnleMaxDegFt;
agl::utl::Parameter<bool> enableLimitThighAngleFt;
agl::utl::Parameter<float> thighRotateAngleMinDegFt;
agl::utl::Parameter<float> thighRotateAngleMaxDegFt;
};
KSYS_CHECK_SIZE_NX150(FootIkController, 0x1d0);
BoneControl();
~BoneControl() override;
void doCreate_(u8* buffer, u32 buffer_size, sead::Heap* heap) override;
bool parse_(u8* data, size_t size, sead::Heap* heap) override;
bool ParamIO_m0() override { return true; }
bool needsParse() const override { return true; }
const Whole& getWhole() const { return mWhole; }
const Spine& getSpine() const { return mSpine; }
const EyeBall& getEyeBall() const { return mEyeBall; }
const sead::Buffer<EyeSet>& getEyeSets() const { return mEyeSets; }
const FootIkController& getFootIkController() const { return mFootIkController; }
const sead::Buffer<BoneGroup>& getBoneGroups() const { return mBoneGroups; }
const BoneGroup* getBoneGroup(const sead::SafeString& name) const;
private:
Whole mWhole;
Spine mSpine;
EyeBall mEyeBall;
agl::utl::ParameterList _650;
sead::Buffer<EyeSet> mEyeSets;
FootIkController mFootIkController;
agl::utl::ParameterList mBoneGroupsList;
sead::Buffer<BoneGroup> mBoneGroups;
};
KSYS_CHECK_SIZE_NX150(BoneControl, 0x8d0);
} // namespace ksys::res