sm64coopdx/src/goddard/skin_movement.c

159 lines
4.7 KiB
C
Raw Normal View History

2020-06-02 16:44:34 +00:00
#include <PR/ultratypes.h>
2019-08-25 04:46:40 +00:00
#include "debug_utils.h"
2020-06-02 16:44:34 +00:00
#include "gd_math.h"
#include "gd_types.h"
2019-08-25 04:46:40 +00:00
#include "joints.h"
2020-06-02 16:44:34 +00:00
#include "macros.h"
#include "objects.h"
2019-08-25 04:46:40 +00:00
#include "skin.h"
2020-06-02 16:44:34 +00:00
#include "skin_movement.h"
2019-08-25 04:46:40 +00:00
/* bss */
struct ObjWeight *sResetCurWeight;
2019-08-25 04:46:40 +00:00
static Mat4f D_801B9EA8; // TODO: rename to sHead2Mtx?
static struct ObjJoint *D_801B9EE8; // set but not used
2019-08-25 04:46:40 +00:00
/* @ 22FDB0 for 0x180 */
2020-06-02 16:44:34 +00:00
void func_801815E0(Mat4f *mtx) {
2019-08-25 04:46:40 +00:00
struct GdVec3f scratchVec;
scratchVec.x = (*mtx)[0][0];
scratchVec.y = (*mtx)[0][1];
scratchVec.z = (*mtx)[0][2];
2020-04-03 18:57:26 +00:00
gd_normalize_vec3f(&scratchVec);
2019-08-25 04:46:40 +00:00
(*mtx)[0][0] = scratchVec.x;
(*mtx)[0][1] = scratchVec.y;
(*mtx)[0][2] = scratchVec.z;
scratchVec.x = (*mtx)[1][0];
scratchVec.y = (*mtx)[1][1];
scratchVec.z = (*mtx)[1][2];
2020-04-03 18:57:26 +00:00
gd_normalize_vec3f(&scratchVec);
2019-08-25 04:46:40 +00:00
(*mtx)[1][0] = scratchVec.x;
(*mtx)[1][1] = scratchVec.y;
(*mtx)[1][2] = scratchVec.z;
scratchVec.x = (*mtx)[2][0];
scratchVec.y = (*mtx)[2][1];
scratchVec.z = (*mtx)[2][2];
2020-04-03 18:57:26 +00:00
gd_normalize_vec3f(&scratchVec);
2019-08-25 04:46:40 +00:00
(*mtx)[2][0] = scratchVec.x;
(*mtx)[2][1] = scratchVec.y;
(*mtx)[2][2] = scratchVec.z;
}
/* @ 22FF30 for 0xDC */
/* called with ObjNext->unk1A8 (variable obj ptr?) ->unk20 or ->unk24 ptr*/
// TODO: figure out the proper object type for a0
void scale_verts(struct ObjGroup *a0) {
2019-08-25 04:46:40 +00:00
register f32 sp1C;
register struct ListNode *link;
2019-11-03 19:36:27 +00:00
struct ObjVertex *vtx;
2019-08-25 04:46:40 +00:00
for (link = a0->firstMember; link != NULL; link = link->next) {
2019-11-03 19:36:27 +00:00
vtx = (struct ObjVertex *) link->obj;
2019-08-25 04:46:40 +00:00
2019-11-03 19:36:27 +00:00
if ((sp1C = vtx->scaleFactor) != 0.0f) {
vtx->pos.x = vtx->initPos.x * sp1C;
vtx->pos.y = vtx->initPos.y * sp1C;
vtx->pos.z = vtx->initPos.z * sp1C;
2019-09-01 19:50:50 +00:00
} else {
2019-11-03 19:36:27 +00:00
vtx->pos.x = vtx->pos.y = vtx->pos.z = 0.0f;
2019-09-01 19:50:50 +00:00
}
2019-08-25 04:46:40 +00:00
}
}
/* @ 23000C for 0x58; orig name: func8018183C*/
void move_skin(struct ObjNet *net) {
UNUSED u8 pad1C[8];
if (net->shapePtr != NULL) {
scale_verts(net->shapePtr->scaledVtxGroup);
2019-09-01 19:50:50 +00:00
}
2019-08-25 04:46:40 +00:00
}
/* @ 230064 for 0x13C*/
void func_80181894(struct ObjJoint *joint) {
register struct ObjGroup *weightGroup; // baseGroup? weights Only?
struct GdVec3f stackVec;
register struct ObjWeight *curWeight;
register struct ObjVertex *connectedVtx;
register struct ListNode *link;
2019-08-25 04:46:40 +00:00
register f32 scaleFactor;
struct GdObj *linkedObj;
weightGroup = joint->weightGrp;
2019-08-25 04:46:40 +00:00
if (weightGroup != NULL) {
for (link = weightGroup->firstMember; link != NULL; link = link->next) {
2019-08-25 04:46:40 +00:00
linkedObj = link->obj;
curWeight = (struct ObjWeight *) linkedObj;
if (curWeight->weightVal > 0.0) //? 0.0f
2019-08-25 04:46:40 +00:00
{
stackVec.x = curWeight->vec20.x;
stackVec.y = curWeight->vec20.y;
stackVec.z = curWeight->vec20.z;
2020-04-03 18:57:26 +00:00
gd_rotate_and_translate_vec3f(&stackVec, &joint->matE8);
2019-08-25 04:46:40 +00:00
connectedVtx = curWeight->vtx;
scaleFactor = curWeight->weightVal;
2019-08-25 04:46:40 +00:00
connectedVtx->pos.x += stackVec.x * scaleFactor;
connectedVtx->pos.y += stackVec.y * scaleFactor;
connectedVtx->pos.z += stackVec.z * scaleFactor;
}
}
}
}
/* @ 2301A0 for 0x110 */
void reset_weight_vtx(struct ObjVertex *vtx) {
2019-08-25 04:46:40 +00:00
struct GdVec3f localVec;
UNUSED u8 pad24[0x10];
if (sResetWeightVtxNum++ == sResetCurWeight->vtxId) { // found matching vertex
sResetCurWeight->vtx = vtx;
2019-08-25 04:46:40 +00:00
localVec.x = vtx->pos.x;
localVec.y = vtx->pos.y;
localVec.z = vtx->pos.z;
2020-04-03 18:57:26 +00:00
gd_rotate_and_translate_vec3f(&localVec, &D_801B9EA8);
sResetCurWeight->vec20.x = localVec.x;
sResetCurWeight->vec20.y = localVec.y;
sResetCurWeight->vec20.z = localVec.z;
2019-08-25 04:46:40 +00:00
vtx->scaleFactor -= sResetCurWeight->weightVal;
2019-08-25 04:46:40 +00:00
}
}
void reset_weight(struct ObjWeight *weight) {
UNUSED u32 vtxCount;
UNUSED u32 pad20;
struct ObjGroup *skinGroup;
sResetCurWeight = weight;
sResetWeightVtxNum = 0;
2019-09-01 19:50:50 +00:00
if ((skinGroup = gGdSkinNet->skinGrp) != NULL) {
// Go through every vertex in the skin group, and reset the weight if the vertex is managed by the weight
2019-08-25 04:46:40 +00:00
vtxCount =
apply_to_obj_types_in_group(OBJ_TYPE_VERTICES, (applyproc_t) reset_weight_vtx, skinGroup);
2019-09-01 19:50:50 +00:00
} else {
2019-08-25 04:46:40 +00:00
fatal_printf("reset_weight(): Skin net has no SkinGroup");
2019-09-01 19:50:50 +00:00
}
2019-08-25 04:46:40 +00:00
if (weight->vtx == NULL) {
fatal_printf("reset_weight(): Skin vertex ID %d not found", weight->vtxId);
2019-09-01 19:50:50 +00:00
}
2019-08-25 04:46:40 +00:00
}
void reset_joint_weights(struct ObjJoint *joint) {
2019-08-25 04:46:40 +00:00
struct ObjGroup *group;
2020-04-03 18:57:26 +00:00
gd_inverse_mat4f(&joint->matE8, &D_801B9EA8);
2019-08-25 04:46:40 +00:00
D_801B9EE8 = joint;
if ((group = joint->weightGrp) != NULL) {
2019-08-25 04:46:40 +00:00
apply_to_obj_types_in_group(OBJ_TYPE_WEIGHTS, (applyproc_t) reset_weight, group);
2019-09-01 19:50:50 +00:00
}
2019-08-25 04:46:40 +00:00
}