我错过了什么或者我的代码有什么问题吗?
以下链接可能会有所帮助:
我写了一个 .smd 模型导入器,由于某种原因,除了静止帧之外的所有帧都搞砸了。
首先,我从文件中加载所有信息并将其存储到数组列表中。然后我调用这个方法并计算所有的渲染:
private void CalculateMatrices(Model model) {
for(Bone bone : model.bones){
if(bone.boneParentsID >= 0) {
bone.parent = model.bones.get(bone.boneParentsID);
}
Matrix4f defaultBoneTransform = new Matrix4f();
defaultBoneTransform.rotate(bone.defaultRotation.x, new Vector3f(1.0f, 0.0f, 0.0f));
defaultBoneTransform.rotate(bone.defaultRotation.y, new Vector3f(0.0f, 1.0f, 0.0f));
defaultBoneTransform.rotate(bone.defaultRotation.z, new Vector3f(0.0f, 0.0f, 1.0f));
defaultBoneTransform.translate(bone.defaultPosition);
bone.defaultBoneTransform = defaultBoneTransform;
for(Frame frame : bone.frames){
Matrix4f boneTransform = new Matrix4f();
boneTransform.rotate(frame.rotation.x, new Vector3f(1.0f, 0.0f, 0.0f));
boneTransform.rotate(frame.rotation.y, new Vector3f(0.0f, 1.0f, 0.0f));
boneTransform.rotate(frame.rotation.z, new Vector3f(0.0f, 0.0f, 1.0f));
boneTransform.translate(frame.position);
bone.boneTransforms.add(boneTransform);
}
}
for(Bone bone : model.bones){
Matrix4f atRestTransform = new Matrix4f();
if(bone.boneParentsID == -1) {
atRestTransform = new Matrix4f(bone.defaultBoneTransform);
} else {
for (Bone current = bone; current != null; current = current.parent) {
Matrix4f.mul(current.defaultBoneTransform, atRestTransform, atRestTransform);
}
}
bone.boneToWorldDefault = new Matrix4f(atRestTransform);
atRestTransform.invert();
bone.atRestTransform = new Matrix4f(atRestTransform);
for(Frame frame : bone.frames){
Matrix4f boneToWorldTransform = new Matrix4f();
if(bone.boneParentsID == -1) {
boneToWorldTransform = new Matrix4f(bone.boneTransforms.get(frame.time));
} else {
for (Bone current = bone; current != null; current = current.parent) {
Matrix4f.mul(current.boneTransforms.get(frame.time), boneToWorldTransform, boneToWorldTransform);
}
}
bone.boneToWorldTransforms.add(new Matrix4f(boneToWorldTransform));
}
}
for (Vertex vertex : model.vertices) {
for (Frame frame : model.bones.get(0).frames) {
List<Vector4f> vertexPosAnimatedList = new ArrayList<Vector4f>();
for (float boneID : vertex.boneIDs) {
Vector4f vertexPosAnimated = new Vector4f();
Bone bone = model.bones.get((int) boneID);
Matrix4f mul = new Matrix4f();
Matrix4f.mul(bone.boneToWorldTransforms.get(frame.time), bone.atRestTransform, mul);
Vector4f vecPosMod = new Vector4f(vertex.position.x, vertex.position.y, vertex.position.z, 1f);
Matrix4f.transform(mul, vecPosMod, vertexPosAnimated);
vertexPosAnimated.scale(vertex.boneW.get(vertex.boneIDs.indexOf(boneID)));
vertexPosAnimatedList.add(vertexPosAnimated);
}
Vector4f vertexPosAnimated1 = new Vector4f();
for(int i = 0; i < vertexPosAnimatedList.size(); i++) {
if(i == 0) {
vertexPosAnimated1 = vertexPosAnimatedList.get(0);
} else {
Vector4f.add(vertexPosAnimated1, vertexPosAnimatedList.get(i), vertexPosAnimated1);
}
}
Vector3f animatedPos = new Vector3f(vertexPosAnimated1.x, vertexPosAnimated1.y, vertexPosAnimated1.z);
vertex.positionOnFrames.add(animatedPos);
}
}
}
和渲染:
GL11.glBegin(GL11.GL_TRIANGLES);
for (Vertex vertex : model.vertices) {
Vector3f pos = vertex.positionOnFrames.get(curFrame);
GL11.glNormal3f(model.normals.get(model.vertices.indexOf(vertex)).x, model.normals.get(model.vertices.indexOf(vertex)).y, model.normals.get(model.vertices.indexOf(vertex)).z);
GL11.glTexCoord2f(model.textures.get(model.vertices.indexOf(vertex)).x, model.textures.get(model.vertices.indexOf(vertex)).y);
GL11.glVertex3f(pos.x, pos.y, pos.z);
}
GL11.glEnd();
我想问题出在CalculateMatrices
方法的某个地方。
在那个方法中:
- 为每个骨骼和每个帧创建一个变换矩阵,其中包括骨骼旋转和平移。
- 计算骨骼atrestposition,以及它在每一帧上的世界位置。
- 做更多的计算,比如为一个顶点设置多个骨骼。
- 将每个顶点及其动画位置存储到数组列表中。