/* transform.c General transform handling Copyright (C) 2021 Bill Currke This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to: Free Software Foundation, Inc. 59 Temple Place - Suite 330 Boston, MA 02111-1307, USA */ #ifdef HAVE_CONFIG_H # include "config.h" #endif #ifdef HAVE_STRING_H # include #endif #ifdef HAVE_STRINGS_H # include #endif #define IMPLEMENT_TRANSFORM_Funcs #include "QF/scene/component.h" #include "QF/scene/hierarchy.h" #include "QF/scene/scene.h" #include "QF/scene/transform.h" #include "scn_internal.h" static void transform_mat4f_identity (void *_mat) { vec4f_t *mat = _mat; mat4fidentity (mat); } static void transform_rotation_identity (void *_rot) { vec4f_t *rot = _rot; *rot = (vec4f_t) { 0, 0, 0, 1 }; } static void transform_scale_identity (void *_scale) { vec4f_t *scale = _scale; *scale = (vec4f_t) { 1, 1, 1, 1 }; } static void transform_modified_init (void *_modified) { byte *modified = _modified; *modified = 1; } static const component_t transform_components[transform_type_count] = { [transform_type_name] = { .size = sizeof (char *), .name = "Name", }, [transform_type_tag] = { .size = sizeof (uint32_t), .name = "Tag", }, [transform_type_modified] = { .size = sizeof (byte), .create = transform_modified_init, .name = "Modified", }, [transform_type_localMatrix] = { .size = sizeof (mat4f_t), .create = transform_mat4f_identity, .name = "Local Matrix", }, [transform_type_localInverse] = { .size = sizeof (mat4f_t), .create = transform_mat4f_identity, .name = "Local Inverse", }, [transform_type_worldMatrix] = { .size = sizeof (mat4f_t), .create = transform_mat4f_identity, .name = "World Matrix", }, [transform_type_worldInverse] = { .size = sizeof (mat4f_t), .create = transform_mat4f_identity, .name = "World Inverse", }, [transform_type_localRotation] = { .size = sizeof (vec4f_t), .create = transform_rotation_identity, .name = "Local Rotation", }, [transform_type_localScale] = { .size = sizeof (vec4f_t), .create = transform_scale_identity, .name = "Local Scale", }, [transform_type_worldRotation] = { .size = sizeof (vec4f_t), .create = transform_rotation_identity, .name = "World Rotation", }, [transform_type_worldScale] = { .size = sizeof (vec4f_t), .create = transform_scale_identity, .name = "World Scale", }, }; static const hierarchy_type_t transform_type = { .num_components = transform_type_count, .components = transform_components, }; transform_t * __transform_alloc (scene_t *scene) { scene_resources_t *res = scene->resources; transform_t *transform = PR_RESNEW_NC (res->transforms); transform->scene = scene; transform->id = PR_RESINDEX (res->transforms, transform); transform->hierarchy = 0; transform->index = 0; return transform; } static void transform_calcLocalInverse (hierarchy_t *h, uint32_t index) { mat4f_t *localMatrix = h->components[transform_type_localMatrix]; mat4f_t *localInverse = h->components[transform_type_localInverse]; // This takes advantage of the fact that localMatrix is a simple // homogenous scale/rotate/translate matrix with no shear vec4f_t x = localMatrix[index][0]; vec4f_t y = localMatrix[index][1]; vec4f_t z = localMatrix[index][2]; vec4f_t t = localMatrix[index][3]; // "one" is to ensure both the scalar and translation have 1 in their // fourth components vec4f_t one = { 0, 0, 0, 1 }; vec4f_t nx = { x[0], y[0], z[0], 0 }; vec4f_t ny = { x[1], y[1], z[1], 0 }; vec4f_t nz = { x[2], y[2], z[2], 0 }; vec4f_t nt = one - t[0] * nx - t[1] * ny - t[2] * nz; // vertical dot product!!! vec4f_t s = 1 / (nx * nx + ny * ny + nz * nz + one); localInverse[index][0] = nx * s; localInverse[index][1] = ny * s; localInverse[index][2] = nz * s; localInverse[index][3] = nt * s; } static void Transform_UpdateMatrices (hierarchy_t *h) { mat4f_t *localMatrix = h->components[transform_type_localMatrix]; mat4f_t *localInverse = h->components[transform_type_localInverse]; mat4f_t *worldMatrix = h->components[transform_type_worldMatrix]; mat4f_t *worldInverse = h->components[transform_type_worldInverse]; vec4f_t *localRotation = h->components[transform_type_localRotation]; vec4f_t *localScale = h->components[transform_type_localScale]; vec4f_t *worldRotation = h->components[transform_type_worldRotation]; vec4f_t *worldScale = h->components[transform_type_worldScale]; byte *modified = h->components[transform_type_modified]; for (uint32_t i = 0; i < h->num_objects; i++) { if (modified[i]) { transform_calcLocalInverse (h, i); } } if (modified[0]) { memcpy (worldMatrix[0], localMatrix[0], sizeof (mat4_t)); memcpy (worldInverse[0], localInverse[0], sizeof (mat4_t)); worldRotation[0] = localRotation[0]; worldScale[0] = localScale[0]; } for (size_t i = 1; i < h->num_objects; i++) { uint32_t parent = h->parentIndex[i]; if (modified[i] || modified[parent]) { mmulf (worldMatrix[i], worldMatrix[parent], localMatrix[i]); modified[i] = 1; } } for (size_t i = 1; i < h->num_objects; i++) { uint32_t parent = h->parentIndex[i]; if (modified[i] || modified[parent]) { mmulf (worldInverse[i], localInverse[i], worldInverse[parent]); } } for (size_t i = 1; i < h->num_objects; i++) { uint32_t parent = h->parentIndex[i]; if (modified[i] || modified[parent]) { worldRotation[i] = qmulf (worldRotation[parent], localRotation[i]); } } for (size_t i = 1; i < h->num_objects; i++) { uint32_t parent = h->parentIndex[i]; if (modified[i] || modified[parent]) { worldScale[i] = m3vmulf (worldMatrix[parent], localScale[i]); } } memset (modified, 0, h->num_objects); } transform_t * Transform_New (scene_t *scene, transform_t *parent) { transform_t *transform = __transform_alloc (scene); if (parent) { transform->hierarchy = parent->hierarchy; transform->index = Hierarchy_InsertHierarchy (parent->hierarchy, 0, parent->index, 0); } else { transform->hierarchy = Hierarchy_New (scene, &transform_type, 1); transform->index = 0; } transform->hierarchy->transform[transform->index] = transform; Transform_UpdateMatrices (transform->hierarchy); return transform; } void Transform_Delete (transform_t *transform) { if (transform->index != 0) { // The transform is not the root, so pull it out of its current // hierarchy so deleting it is easier Transform_SetParent (transform, 0); } // Takes care of freeing the transforms Hierarchy_Delete (transform->hierarchy); } transform_t * Transform_NewNamed (scene_t *scene, transform_t *parent, const char *name) { transform_t *transform = Transform_New (scene, parent); Transform_SetName (transform, name); return transform; } void Transform_SetParent (transform_t *transform, transform_t *parent) { if (parent) { hierarchy_t *hierarchy = transform->hierarchy; uint32_t index = transform->index; Hierarchy_InsertHierarchy (parent->hierarchy, hierarchy, parent->index, index); Hierarchy_RemoveHierarchy (hierarchy, index); if (!hierarchy->num_objects) { Hierarchy_Delete (hierarchy); } } else { // null parent -> make transform root if (!transform->index) { // already root return; } hierarchy_t *hierarchy = transform->hierarchy; uint32_t index = transform->index; hierarchy_t *new_hierarchy = Hierarchy_New (transform->scene, &transform_type, 0); Hierarchy_InsertHierarchy (new_hierarchy, hierarchy, null_transform, index); Hierarchy_RemoveHierarchy (hierarchy, index); } hierarchy_t *h = transform->hierarchy; byte *modified = h->components[transform_type_modified]; modified[transform->index] = 1; Transform_UpdateMatrices (h); } void Transform_SetName (transform_t *transform, const char *_name) { hierarchy_t *h = transform->hierarchy; char **name = h->components[transform_type_name]; //FIXME create a string pool (similar to qfcc's, or even move that to util) if (name[transform->index]) { free (name[transform->index]); } name[transform->index] = strdup (_name); } void Transform_SetTag (transform_t *transform, uint32_t _tag) { hierarchy_t *h = transform->hierarchy; uint32_t *tag = h->components[transform_type_tag]; tag[transform->index] = _tag; } void Transform_SetLocalPosition (transform_t *transform, vec4f_t position) { hierarchy_t *h = transform->hierarchy; mat4f_t *localMatrix = h->components[transform_type_localMatrix]; byte *modified = h->components[transform_type_modified]; localMatrix[transform->index][3] = position; modified[transform->index] = 1; Transform_UpdateMatrices (h); } void Transform_SetLocalRotation (transform_t *transform, vec4f_t rotation) { hierarchy_t *h = transform->hierarchy; mat4f_t *localMatrix = h->components[transform_type_localMatrix]; vec4f_t *localRotation = h->components[transform_type_localRotation]; vec4f_t *localScale = h->components[transform_type_localScale]; byte *modified = h->components[transform_type_modified]; vec4f_t scale = localScale[transform->index]; mat4f_t mat; mat4fquat (mat, rotation); localRotation[transform->index] = rotation; localMatrix[transform->index][0] = mat[0] * scale[0]; localMatrix[transform->index][1] = mat[1] * scale[1]; localMatrix[transform->index][2] = mat[2] * scale[2]; modified[transform->index] = 1; Transform_UpdateMatrices (h); } void Transform_SetLocalScale (transform_t *transform, vec4f_t scale) { hierarchy_t *h = transform->hierarchy; mat4f_t *localMatrix = h->components[transform_type_localMatrix]; vec4f_t *localRotation = h->components[transform_type_localRotation]; vec4f_t *localScale = h->components[transform_type_localScale]; byte *modified = h->components[transform_type_modified]; vec4f_t rotation = localRotation[transform->index]; mat4f_t mat; mat4fquat (mat, rotation); localScale[transform->index] = scale; localMatrix[transform->index][0] = mat[0] * scale[0]; localMatrix[transform->index][1] = mat[1] * scale[1]; localMatrix[transform->index][2] = mat[2] * scale[2]; modified[transform->index] = 1; Transform_UpdateMatrices (h); } void Transform_SetWorldPosition (transform_t *transform, vec4f_t position) { if (transform->index) { hierarchy_t *h = transform->hierarchy; mat4f_t *worldInverse = h->components[transform_type_worldInverse]; uint32_t parent = h->parentIndex[transform->index]; position = mvmulf (worldInverse[parent], position); } Transform_SetLocalPosition (transform, position); } void Transform_SetWorldRotation (transform_t *transform, vec4f_t rotation) { if (transform->index) { hierarchy_t *h = transform->hierarchy; vec4f_t *worldRotation = h->components[transform_type_worldRotation]; uint32_t parent = h->parentIndex[transform->index]; rotation = qmulf (qconjf (worldRotation[parent]), rotation); } Transform_SetLocalRotation (transform, rotation); } void Transform_SetLocalTransform (transform_t *transform, vec4f_t scale, vec4f_t rotation, vec4f_t position) { hierarchy_t *h = transform->hierarchy; mat4f_t *localMatrix = h->components[transform_type_localMatrix]; vec4f_t *localRotation = h->components[transform_type_localRotation]; vec4f_t *localScale = h->components[transform_type_localScale]; byte *modified = h->components[transform_type_modified]; mat4f_t mat; mat4fquat (mat, rotation); position[3] = 1; localRotation[transform->index] = rotation; localScale[transform->index] = scale; localMatrix[transform->index][0] = mat[0] * scale[0]; localMatrix[transform->index][1] = mat[1] * scale[1]; localMatrix[transform->index][2] = mat[2] * scale[2]; localMatrix[transform->index][3] = position; modified[transform->index] = 1; Transform_UpdateMatrices (h); }