/* Copyright (C) 1996-2008 by Jan Eric Kyprianidis All rights reserved. This program is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation, either version 2.1 of the License, or (at your option) any later version. Thisprogram 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this program; If not, see . */ #include "lib3ds_impl.h" /*! * Create and return a new node object. * * The node is returned with an identity matrix. All other fields * are zero. * * \return Lib3dsNode */ Lib3dsNode* lib3ds_node_new(Lib3dsNodeType type) { Lib3dsNode *node; switch (type) { case LIB3DS_NODE_AMBIENT_COLOR: { Lib3dsAmbientColorNode *n = (Lib3dsAmbientColorNode*)calloc(sizeof(Lib3dsAmbientColorNode), 1); node = (Lib3dsNode*)n; strcpy(node->name, "$AMBIENT$"); n->color_track.type = LIB3DS_TRACK_VECTOR; break; } case LIB3DS_NODE_MESH_INSTANCE: { Lib3dsMeshInstanceNode *n = (Lib3dsMeshInstanceNode*)calloc(sizeof(Lib3dsMeshInstanceNode), 1); node = (Lib3dsNode*)n; strcpy(node->name, "$$$DUMMY"); n->pos_track.type = LIB3DS_TRACK_VECTOR; n->scl_track.type = LIB3DS_TRACK_VECTOR; n->rot_track.type = LIB3DS_TRACK_QUAT; n->hide_track.type = LIB3DS_TRACK_BOOL; break; } case LIB3DS_NODE_CAMERA: { Lib3dsCameraNode *n = (Lib3dsCameraNode*)calloc(sizeof(Lib3dsCameraNode), 1); node = (Lib3dsNode*)n; n->pos_track.type = LIB3DS_TRACK_VECTOR; n->fov_track.type = LIB3DS_TRACK_FLOAT; n->roll_track.type = LIB3DS_TRACK_FLOAT; break; } case LIB3DS_NODE_CAMERA_TARGET: { Lib3dsTargetNode *n = (Lib3dsTargetNode*)calloc(sizeof(Lib3dsTargetNode), 1); node = (Lib3dsNode*)n; n->pos_track.type = LIB3DS_TRACK_VECTOR; break; } case LIB3DS_NODE_OMNILIGHT: { Lib3dsOmnilightNode *n = (Lib3dsOmnilightNode*)calloc(sizeof(Lib3dsOmnilightNode), 1); node = (Lib3dsNode*)n; n->pos_track.type = LIB3DS_TRACK_VECTOR; n->color_track.type = LIB3DS_TRACK_VECTOR; break; } case LIB3DS_NODE_SPOTLIGHT: { Lib3dsSpotlightNode *n = (Lib3dsSpotlightNode*)calloc(sizeof(Lib3dsSpotlightNode), 1); node = (Lib3dsNode*)n; n->pos_track.type = LIB3DS_TRACK_VECTOR; n->color_track.type = LIB3DS_TRACK_VECTOR; n->hotspot_track.type = LIB3DS_TRACK_FLOAT; n->falloff_track.type = LIB3DS_TRACK_FLOAT; n->roll_track.type = LIB3DS_TRACK_FLOAT; break; } case LIB3DS_NODE_SPOTLIGHT_TARGET: { Lib3dsTargetNode *n = (Lib3dsTargetNode*)calloc(sizeof(Lib3dsTargetNode), 1); node = (Lib3dsNode*)n; n->pos_track.type = LIB3DS_TRACK_VECTOR; break; } default: assert(0); return NULL; } node->type = type; node->node_id = 65535; node->user_id = 65535; lib3ds_matrix_identity(node->matrix); return node; } Lib3dsAmbientColorNode* lib3ds_node_new_ambient_color(float color0[3]) { Lib3dsNode *node; Lib3dsAmbientColorNode *n; node = lib3ds_node_new(LIB3DS_NODE_AMBIENT_COLOR); n = (Lib3dsAmbientColorNode*)node; lib3ds_track_resize(&n->color_track, 1); if (color0) { lib3ds_vector_copy(n->color_track.keys[0].value, color0); } else { lib3ds_vector_zero(n->color_track.keys[0].value); } return n; } Lib3dsMeshInstanceNode* lib3ds_node_new_mesh_instance(Lib3dsMesh *mesh, const char *instance_name, float pos0[3], float scl0[3], float rot0[4]) { Lib3dsNode *node; Lib3dsMeshInstanceNode *n; int i; node = lib3ds_node_new(LIB3DS_NODE_MESH_INSTANCE); if (mesh) { strcpy(node->name, mesh->name); } else { strcpy(node->name, "$$$DUMMY"); } n = (Lib3dsMeshInstanceNode*)node; if (instance_name) { strcpy(n->instance_name, instance_name); } lib3ds_track_resize(&n->pos_track, 1); if (pos0) { lib3ds_vector_copy(n->pos_track.keys[0].value, pos0); } lib3ds_track_resize(&n->scl_track, 1); if (scl0) { lib3ds_vector_copy(n->scl_track.keys[0].value, scl0); } else { lib3ds_vector_make(n->scl_track.keys[0].value, 1, 1, 1); } lib3ds_track_resize(&n->rot_track, 1); if (rot0) { for (i = 0; i < 4; ++i) n->rot_track.keys[0].value[i] = rot0[i]; } else { for (i = 0; i < 4; ++i) n->rot_track.keys[0].value[i] = 0; } return n; } Lib3dsCameraNode* lib3ds_node_new_camera(Lib3dsCamera *camera) { Lib3dsNode *node; Lib3dsCameraNode *n; assert(camera); node = lib3ds_node_new(LIB3DS_NODE_CAMERA); strcpy(node->name, camera->name); n = (Lib3dsCameraNode*)node; lib3ds_track_resize(&n->pos_track, 1); lib3ds_vector_copy(n->pos_track.keys[0].value, camera->position); lib3ds_track_resize(&n->fov_track, 1); n->fov_track.keys[0].value[0] = camera->fov; lib3ds_track_resize(&n->roll_track, 1); n->roll_track.keys[0].value[0] = camera->roll; return n; } Lib3dsTargetNode* lib3ds_node_new_camera_target(Lib3dsCamera *camera) { Lib3dsNode *node; Lib3dsTargetNode *n; assert(camera); node = lib3ds_node_new(LIB3DS_NODE_CAMERA_TARGET); strcpy(node->name, camera->name); n = (Lib3dsTargetNode*)node; lib3ds_track_resize(&n->pos_track, 1); lib3ds_vector_copy(n->pos_track.keys[0].value, camera->target); return n; } Lib3dsOmnilightNode* lib3ds_node_new_omnilight(Lib3dsLight *light) { Lib3dsNode *node; Lib3dsOmnilightNode *n; assert(light); node = lib3ds_node_new(LIB3DS_NODE_OMNILIGHT); strcpy(node->name, light->name); n = (Lib3dsOmnilightNode*)node; lib3ds_track_resize(&n->pos_track, 1); lib3ds_vector_copy(n->pos_track.keys[0].value, light->position); lib3ds_track_resize(&n->color_track, 1); lib3ds_vector_copy(n->color_track.keys[0].value, light->color); return n; } Lib3dsSpotlightNode* lib3ds_node_new_spotlight(Lib3dsLight *light) { Lib3dsNode *node; Lib3dsSpotlightNode *n; assert(light); node = lib3ds_node_new(LIB3DS_NODE_SPOTLIGHT); strcpy(node->name, light->name); n = (Lib3dsSpotlightNode*)node; lib3ds_track_resize(&n->pos_track, 1); lib3ds_vector_copy(n->pos_track.keys[0].value, light->position); lib3ds_track_resize(&n->color_track, 1); lib3ds_vector_copy(n->color_track.keys[0].value, light->color); lib3ds_track_resize(&n->hotspot_track, 1); n->hotspot_track.keys[0].value[0] = light->hotspot; lib3ds_track_resize(&n->falloff_track, 1); n->falloff_track.keys[0].value[0] = light->falloff; lib3ds_track_resize(&n->roll_track, 1); n->roll_track.keys[0].value[0] = light->roll; return n; } Lib3dsTargetNode* lib3ds_node_new_spotligf_target(Lib3dsLight *light) { Lib3dsNode *node; Lib3dsTargetNode *n; assert(light); node = lib3ds_node_new(LIB3DS_NODE_SPOTLIGHT_TARGET); strcpy(node->name, light->name); n = (Lib3dsTargetNode*)node; lib3ds_track_resize(&n->pos_track, 1); lib3ds_vector_copy(n->pos_track.keys[0].value, light->target); return n; } static void free_node_and_childs(Lib3dsNode *node) { assert(node); switch (node->type) { case LIB3DS_NODE_AMBIENT_COLOR: { Lib3dsAmbientColorNode *n = (Lib3dsAmbientColorNode*)node; lib3ds_track_resize(&n->color_track, 0); break; } case LIB3DS_NODE_MESH_INSTANCE: { Lib3dsMeshInstanceNode *n = (Lib3dsMeshInstanceNode*)node; lib3ds_track_resize(&n->pos_track, 0); lib3ds_track_resize(&n->rot_track, 0); lib3ds_track_resize(&n->scl_track, 0); lib3ds_track_resize(&n->hide_track, 0); break; } case LIB3DS_NODE_CAMERA: { Lib3dsCameraNode *n = (Lib3dsCameraNode*)node; lib3ds_track_resize(&n->pos_track, 0); lib3ds_track_resize(&n->fov_track, 0); lib3ds_track_resize(&n->roll_track, 0); break; } case LIB3DS_NODE_CAMERA_TARGET: { Lib3dsTargetNode *n = (Lib3dsTargetNode*)node; lib3ds_track_resize(&n->pos_track, 0); break; } case LIB3DS_NODE_OMNILIGHT: { Lib3dsOmnilightNode *n = (Lib3dsOmnilightNode*)node; lib3ds_track_resize(&n->pos_track, 0); lib3ds_track_resize(&n->color_track, 0); break; } case LIB3DS_NODE_SPOTLIGHT: { Lib3dsSpotlightNode *n = (Lib3dsSpotlightNode*)node; lib3ds_track_resize(&n->pos_track, 0); lib3ds_track_resize(&n->color_track, 0); lib3ds_track_resize(&n->hotspot_track, 0); lib3ds_track_resize(&n->falloff_track, 0); lib3ds_track_resize(&n->roll_track, 0); break; } case LIB3DS_NODE_SPOTLIGHT_TARGET: { Lib3dsTargetNode *n = (Lib3dsTargetNode*)node; lib3ds_track_resize(&n->pos_track, 0); break; } } { Lib3dsNode *p, *q; for (p = node->childs; p; p = q) { q = p->next; free_node_and_childs(p); } } free(node); } /*! * Free a node and all of its resources. * * \param node Lib3dsNode object to be freed. */ void lib3ds_node_free(Lib3dsNode *node) { assert(node); free_node_and_childs(node); } /*! * Evaluate an animation node. * * Recursively sets node and its children to their appropriate values * for this point in the animation. * * \param node Node to be evaluated. * \param t time value, between 0. and file->frames */ void lib3ds_node_eval(Lib3dsNode *node, float t) { assert(node); switch (node->type) { case LIB3DS_NODE_AMBIENT_COLOR: { Lib3dsAmbientColorNode *n = (Lib3dsAmbientColorNode*)node; if (node->parent) { lib3ds_matrix_copy(node->matrix, node->parent->matrix); } else { lib3ds_matrix_identity(node->matrix); } lib3ds_track_eval_vector(&n->color_track, n->color, t); break; } case LIB3DS_NODE_MESH_INSTANCE: { float M[4][4]; Lib3dsMeshInstanceNode *n = (Lib3dsMeshInstanceNode*)node; lib3ds_track_eval_vector(&n->pos_track, n->pos, t); lib3ds_track_eval_quat(&n->rot_track, n->rot, t); if (n->scl_track.nkeys) { lib3ds_track_eval_vector(&n->scl_track, n->scl, t); } else { n->scl[0] = n->scl[1] = n->scl[2] = 1.0f; } lib3ds_track_eval_bool(&n->hide_track, &n->hide, t); lib3ds_matrix_identity(M); lib3ds_matrix_translate(M, n->pos[0], n->pos[1], n->pos[2]); lib3ds_matrix_rotate_quat(M, n->rot); lib3ds_matrix_scale(M, n->scl[0], n->scl[1], n->scl[2]); if (node->parent) { lib3ds_matrix_mult(node->matrix, node->parent->matrix, M); } else { lib3ds_matrix_copy(node->matrix, M); } break; } case LIB3DS_NODE_CAMERA: { Lib3dsCameraNode *n = (Lib3dsCameraNode*)node; lib3ds_track_eval_vector(&n->pos_track, n->pos, t); lib3ds_track_eval_float(&n->fov_track, &n->fov, t); lib3ds_track_eval_float(&n->roll_track, &n->roll, t); if (node->parent) { lib3ds_matrix_copy(node->matrix, node->parent->matrix); } else { lib3ds_matrix_identity(node->matrix); } lib3ds_matrix_translate(node->matrix, n->pos[0], n->pos[1], n->pos[2]); break; } case LIB3DS_NODE_CAMERA_TARGET: { Lib3dsTargetNode *n = (Lib3dsTargetNode*)node; lib3ds_track_eval_vector(&n->pos_track, n->pos, t); if (node->parent) { lib3ds_matrix_copy(node->matrix, node->parent->matrix); } else { lib3ds_matrix_identity(node->matrix); } lib3ds_matrix_translate(node->matrix, n->pos[0], n->pos[1], n->pos[2]); break; } case LIB3DS_NODE_OMNILIGHT: { Lib3dsOmnilightNode *n = (Lib3dsOmnilightNode*)node; lib3ds_track_eval_vector(&n->pos_track, n->pos, t); lib3ds_track_eval_vector(&n->color_track, n->color, t); if (node->parent) { lib3ds_matrix_copy(node->matrix, node->parent->matrix); } else { lib3ds_matrix_identity(node->matrix); } lib3ds_matrix_translate(node->matrix, n->pos[0], n->pos[1], n->pos[2]); break; } case LIB3DS_NODE_SPOTLIGHT: { Lib3dsSpotlightNode *n = (Lib3dsSpotlightNode*)node; lib3ds_track_eval_vector(&n->pos_track, n->pos, t); lib3ds_track_eval_vector(&n->color_track, n->color, t); lib3ds_track_eval_float(&n->hotspot_track, &n->hotspot, t); lib3ds_track_eval_float(&n->falloff_track, &n->falloff, t); lib3ds_track_eval_float(&n->roll_track, &n->roll, t); if (node->parent) { lib3ds_matrix_copy(node->matrix, node->parent->matrix); } else { lib3ds_matrix_identity(node->matrix); } lib3ds_matrix_translate(node->matrix, n->pos[0], n->pos[1], n->pos[2]); break; } case LIB3DS_NODE_SPOTLIGHT_TARGET: { Lib3dsTargetNode *n = (Lib3dsTargetNode*)node; lib3ds_track_eval_vector(&n->pos_track, n->pos, t); if (node->parent) { lib3ds_matrix_copy(node->matrix, node->parent->matrix); } else { lib3ds_matrix_identity(node->matrix); } lib3ds_matrix_translate(node->matrix, n->pos[0], n->pos[1], n->pos[2]); break; } } { Lib3dsNode *p; for (p = node->childs; p != 0; p = p->next) { lib3ds_node_eval(p, t); } } } /*! * Return a node object by name and type. * * This function performs a recursive search for the specified node. * Both name and type must match. * * \param node The parent node for the search * \param name The target node name. * \param type The target node type * * \return A pointer to the first matching node, or NULL if not found. */ Lib3dsNode* lib3ds_node_by_name(Lib3dsNode *node, const char* name, Lib3dsNodeType type) { Lib3dsNode *p, *q; for (p = node->childs; p != 0; p = p->next) { if ((p->type == type) && (strcmp(p->name, name) == 0)) { return(p); } q = lib3ds_node_by_name(p, name, type); if (q) { return(q); } } return(0); } /*! * Return a node object by id. * * This function performs a recursive search for the specified node. * * \param node The parent node for the search * \param node_id The target node id. * * \return A pointer to the first matching node, or NULL if not found. */ Lib3dsNode* lib3ds_node_by_id(Lib3dsNode *node, uint16_t node_id) { Lib3dsNode *p, *q; for (p = node->childs; p != 0; p = p->next) { if (p->node_id == node_id) { return(p); } q = lib3ds_node_by_id(p, node_id); if (q) { return(q); } } return(0); } void lib3ds_node_read(Lib3dsNode *node, Lib3dsIo *io) { Lib3dsChunk c; uint16_t chunk; assert(node); lib3ds_chunk_read_start(&c, 0, io); switch (c.chunk) { case CHK_AMBIENT_NODE_TAG: case CHK_OBJECT_NODE_TAG: case CHK_CAMERA_NODE_TAG: case CHK_TARGET_NODE_TAG: case CHK_LIGHT_NODE_TAG: case CHK_SPOTLIGHT_NODE_TAG: case CHK_L_TARGET_NODE_TAG: break; default: return; } while ((chunk = lib3ds_chunk_read_next(&c, io)) != 0) { switch (chunk) { case CHK_NODE_ID: { node->node_id = lib3ds_io_read_word(io); lib3ds_io_log_indent(io, 1); lib3ds_io_log(io, LIB3DS_LOG_INFO, "ID=%d", (short)node->node_id); lib3ds_io_log_indent(io, -1); break; } case CHK_NODE_HDR: { lib3ds_io_read_string(io, node->name, 64); node->flags = lib3ds_io_read_word(io); node->flags |= ((uint32_t)lib3ds_io_read_word(io)) << 16; node->user_id = lib3ds_io_read_word(io); lib3ds_io_log_indent(io, 1); lib3ds_io_log(io, LIB3DS_LOG_INFO, "NAME=%s", node->name); lib3ds_io_log(io, LIB3DS_LOG_INFO, "PARENT=%d", (short)node->user_id); lib3ds_io_log_indent(io, -1); break; } case CHK_PIVOT: { if (node->type == LIB3DS_NODE_MESH_INSTANCE) { Lib3dsMeshInstanceNode *n = (Lib3dsMeshInstanceNode*)node; lib3ds_io_read_vector(io, n->pivot); } else { lib3ds_chunk_unknown(chunk, io); } break; } case CHK_INSTANCE_NAME: { if (node->type == LIB3DS_NODE_MESH_INSTANCE) { Lib3dsMeshInstanceNode *n = (Lib3dsMeshInstanceNode*)node; lib3ds_io_read_string(io, n->instance_name, 64); } else { lib3ds_chunk_unknown(chunk, io); } break; } case CHK_BOUNDBOX: { if (node->type == LIB3DS_NODE_MESH_INSTANCE) { Lib3dsMeshInstanceNode *n = (Lib3dsMeshInstanceNode*)node; lib3ds_io_read_vector(io, n->bbox_min); lib3ds_io_read_vector(io, n->bbox_max); } else { lib3ds_chunk_unknown(chunk, io); } break; } case CHK_COL_TRACK_TAG: { Lib3dsTrack *track = 0; switch (node->type) { case LIB3DS_NODE_AMBIENT_COLOR: { Lib3dsAmbientColorNode *n = (Lib3dsAmbientColorNode*)node; track = &n->color_track; break; } case LIB3DS_NODE_OMNILIGHT: { Lib3dsOmnilightNode *n = (Lib3dsOmnilightNode*)node; track = &n->color_track; break; } case LIB3DS_NODE_SPOTLIGHT: { Lib3dsSpotlightNode *n = (Lib3dsSpotlightNode*)node; track = &n->color_track; break; } default: lib3ds_chunk_unknown(chunk, io); } if (track) { lib3ds_track_read(track, io); } break; } case CHK_POS_TRACK_TAG: { Lib3dsTrack *track = 0; switch (node->type) { case LIB3DS_NODE_MESH_INSTANCE: { Lib3dsMeshInstanceNode *n = (Lib3dsMeshInstanceNode*)node; track = &n->pos_track; break; } case LIB3DS_NODE_CAMERA: { Lib3dsCameraNode *n = (Lib3dsCameraNode*)node; track = &n->pos_track; break; } case LIB3DS_NODE_CAMERA_TARGET: { Lib3dsTargetNode *n = (Lib3dsTargetNode*)node; track = &n->pos_track; break; } case LIB3DS_NODE_OMNILIGHT: { Lib3dsOmnilightNode *n = (Lib3dsOmnilightNode*)node; track = &n->pos_track; break; } case LIB3DS_NODE_SPOTLIGHT: { Lib3dsSpotlightNode *n = (Lib3dsSpotlightNode*)node; track = &n->pos_track; break; } case LIB3DS_NODE_SPOTLIGHT_TARGET: { Lib3dsTargetNode *n = (Lib3dsTargetNode*)node; track = &n->pos_track; break; } default: lib3ds_chunk_unknown(chunk, io); } if (track) { lib3ds_track_read(track, io); } break; } case CHK_ROT_TRACK_TAG: { if (node->type == LIB3DS_NODE_MESH_INSTANCE) { Lib3dsMeshInstanceNode *n = (Lib3dsMeshInstanceNode*)node; n->rot_track.type = LIB3DS_TRACK_QUAT; lib3ds_track_read(&n->rot_track, io); } else { lib3ds_chunk_unknown(chunk, io); } break; } case CHK_SCL_TRACK_TAG: { if (node->type == LIB3DS_NODE_MESH_INSTANCE) { Lib3dsMeshInstanceNode *n = (Lib3dsMeshInstanceNode*)node; n->scl_track.type = LIB3DS_TRACK_VECTOR; lib3ds_track_read(&n->scl_track, io); } else { lib3ds_chunk_unknown(chunk, io); } break; } case CHK_FOV_TRACK_TAG: { if (node->type == LIB3DS_NODE_CAMERA) { Lib3dsCameraNode *n = (Lib3dsCameraNode*)node; n->fov_track.type = LIB3DS_TRACK_FLOAT; lib3ds_track_read(&n->fov_track, io); } else { lib3ds_chunk_unknown(chunk, io); } break; } case CHK_HOT_TRACK_TAG: { if (node->type == LIB3DS_NODE_SPOTLIGHT) { Lib3dsSpotlightNode *n = (Lib3dsSpotlightNode*)node; n->hotspot_track.type = LIB3DS_TRACK_FLOAT; lib3ds_track_read(&n->hotspot_track, io); } else { lib3ds_chunk_unknown(chunk, io); } break; } case CHK_FALL_TRACK_TAG: { if (node->type == LIB3DS_NODE_SPOTLIGHT) { Lib3dsSpotlightNode *n = (Lib3dsSpotlightNode*)node; n->falloff_track.type= LIB3DS_TRACK_FLOAT; lib3ds_track_read(&n->falloff_track, io); } else { lib3ds_chunk_unknown(chunk, io); } break; } case CHK_ROLL_TRACK_TAG: { switch (node->type) { case LIB3DS_NODE_CAMERA: { Lib3dsCameraNode *n = (Lib3dsCameraNode*)node; n->roll_track.type = LIB3DS_TRACK_FLOAT; lib3ds_track_read(&n->roll_track, io); break; } case LIB3DS_NODE_SPOTLIGHT: { Lib3dsSpotlightNode *n = (Lib3dsSpotlightNode*)node; n->roll_track.type = LIB3DS_TRACK_FLOAT; lib3ds_track_read(&n->roll_track, io); break; } default: lib3ds_chunk_unknown(chunk, io); } break; } case CHK_HIDE_TRACK_TAG: { if (node->type == LIB3DS_NODE_MESH_INSTANCE) { Lib3dsMeshInstanceNode *n = (Lib3dsMeshInstanceNode*)node; n->hide_track.type = LIB3DS_TRACK_BOOL; lib3ds_track_read(&n->hide_track, io); } else { lib3ds_chunk_unknown(chunk, io); } break; } case CHK_MORPH_SMOOTH: { if (node->type == LIB3DS_NODE_MESH_INSTANCE) { Lib3dsMeshInstanceNode *n = (Lib3dsMeshInstanceNode*)node; n->morph_smooth = lib3ds_io_read_float(io); } else { lib3ds_chunk_unknown(chunk, io); } } break; /* case LIB3DS_MORPH_TRACK_TAG: { if (node->type == LIB3DS_NODE_MESH_INSTANCE) { Lib3dsMeshInstanceNode *n = (Lib3dsMeshInstanceNode*)node; n->morph_track = lib3ds_track_new(node, LIB3DS_TRACK_MORPH, 0); lib3ds_track_read(n->morph_track, io); } else { lib3ds_chunk_unknown(chunk, io); } } break; */ default: lib3ds_chunk_unknown(chunk, io); } } lib3ds_chunk_read_end(&c, io); } void lib3ds_node_write(Lib3dsNode *node, uint16_t node_id, uint16_t parent_id, Lib3dsIo *io) { Lib3dsChunk c; switch (node->type) { case LIB3DS_NODE_AMBIENT_COLOR: c.chunk = CHK_AMBIENT_NODE_TAG; break; case LIB3DS_NODE_MESH_INSTANCE: c.chunk = CHK_OBJECT_NODE_TAG; break; case LIB3DS_NODE_CAMERA: c.chunk = CHK_CAMERA_NODE_TAG; break; case LIB3DS_NODE_CAMERA_TARGET: c.chunk = CHK_TARGET_NODE_TAG; break; case LIB3DS_NODE_OMNILIGHT: c.chunk = CHK_LIGHT_NODE_TAG; break; case LIB3DS_NODE_SPOTLIGHT: c.chunk = CHK_SPOTLIGHT_NODE_TAG; break; case LIB3DS_NODE_SPOTLIGHT_TARGET: c.chunk = CHK_L_TARGET_NODE_TAG; break; default: assert(0); return; } lib3ds_chunk_write_start(&c, io); { /*---- CHK_NODE_ID ----*/ Lib3dsChunk c; c.chunk = CHK_NODE_ID; c.size = 8; lib3ds_chunk_write(&c, io); lib3ds_io_write_intw(io, node_id); } { /*---- CHK_NODE_HDR ----*/ Lib3dsChunk c; c.chunk = CHK_NODE_HDR; c.size = 6 + 1 + (uint32_t)strlen(node->name) + 2 + 2 + 2; lib3ds_chunk_write(&c, io); lib3ds_io_write_string(io, node->name); lib3ds_io_write_word(io, node->flags & 0xffff); lib3ds_io_write_word(io, (node->flags >> 16) & 0xffff); lib3ds_io_write_word(io, parent_id); } switch (c.chunk) { case CHK_AMBIENT_NODE_TAG: { Lib3dsAmbientColorNode *n = (Lib3dsAmbientColorNode*)node; if (n->color_track.nkeys) { /*---- CHK_COL_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_COL_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->color_track, io); lib3ds_chunk_write_end(&c, io); } break; } case CHK_OBJECT_NODE_TAG: { Lib3dsMeshInstanceNode *n = (Lib3dsMeshInstanceNode*)node; { /*---- CHK_PIVOT ----*/ Lib3dsChunk c; c.chunk = CHK_PIVOT; c.size = 18; lib3ds_chunk_write(&c, io); lib3ds_io_write_vector(io, n->pivot); } { /*---- CHK_INSTANCE_NAME ----*/ Lib3dsChunk c; const char *name; if (strlen(n->instance_name)) { name = n->instance_name; c.chunk = CHK_INSTANCE_NAME; c.size = 6 + 1 + (uint32_t)strlen(name); lib3ds_chunk_write(&c, io); lib3ds_io_write_string(io, name); } } { int i; for (i = 0; i < 3; ++i) { if ((fabs(n->bbox_min[i]) > LIB3DS_EPSILON) || (fabs(n->bbox_max[i]) > LIB3DS_EPSILON)) { break; } } if (i < 3) { /*---- CHK_BOUNDBOX ----*/ Lib3dsChunk c; c.chunk = CHK_BOUNDBOX; c.size = 30; lib3ds_chunk_write(&c, io); lib3ds_io_write_vector(io, n->bbox_min); lib3ds_io_write_vector(io, n->bbox_max); } } if (n->pos_track.nkeys) { /*---- CHK_POS_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_POS_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->pos_track, io); lib3ds_chunk_write_end(&c, io); } if (n->rot_track.nkeys) { /*---- CHK_ROT_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_ROT_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->rot_track, io); lib3ds_chunk_write_end(&c, io); } if (n->scl_track.nkeys) { /*---- LIB3DS_SCL_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_SCL_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->scl_track, io); lib3ds_chunk_write_end(&c, io); } if (n->hide_track.nkeys) { /*---- CHK_HIDE_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_HIDE_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->hide_track, io); lib3ds_chunk_write_end(&c, io); } if (fabs(n->morph_smooth) > LIB3DS_EPSILON) { /*---- CHK_MORPH_SMOOTH ----*/ Lib3dsChunk c; c.chunk = CHK_MORPH_SMOOTH; c.size = 10; lib3ds_chunk_write(&c, io); lib3ds_io_write_float(io, n->morph_smooth); } break; } case CHK_CAMERA_NODE_TAG: { Lib3dsCameraNode *n = (Lib3dsCameraNode*)node; if (n->pos_track.nkeys) { /*---- CHK_POS_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_POS_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->pos_track, io); lib3ds_chunk_write_end(&c, io); } if (n->fov_track.nkeys) { /*---- CHK_FOV_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_FOV_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->fov_track, io); lib3ds_chunk_write_end(&c, io); } if (n->roll_track.nkeys) { /*---- CHK_ROLL_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_ROLL_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->roll_track, io); lib3ds_chunk_write_end(&c, io); } break; } case CHK_TARGET_NODE_TAG: { Lib3dsTargetNode *n = (Lib3dsTargetNode*)node; if (n->pos_track.nkeys) { /*---- CHK_POS_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_POS_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->pos_track, io); lib3ds_chunk_write_end(&c, io); } break; } case CHK_LIGHT_NODE_TAG: { Lib3dsOmnilightNode *n = (Lib3dsOmnilightNode*)node; if (n->pos_track.nkeys) { /*---- CHK_POS_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_POS_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->pos_track, io); lib3ds_chunk_write_end(&c, io); } if (n->color_track.nkeys) { /*---- CHK_COL_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_COL_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->color_track, io); lib3ds_chunk_write_end(&c, io); } break; } case CHK_SPOTLIGHT_NODE_TAG: { Lib3dsSpotlightNode *n = (Lib3dsSpotlightNode*)node; if (n->pos_track.nkeys) { /*---- CHK_POS_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_POS_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->pos_track, io); lib3ds_chunk_write_end(&c, io); } if (n->color_track.nkeys) { /*---- CHK_COL_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_COL_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->color_track, io); lib3ds_chunk_write_end(&c, io); } if (n->hotspot_track.nkeys) { /*---- CHK_HOT_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_HOT_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->hotspot_track, io); lib3ds_chunk_write_end(&c, io); } if (n->falloff_track.nkeys) { /*---- CHK_FALL_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_FALL_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->falloff_track, io); lib3ds_chunk_write_end(&c, io); } if (n->roll_track.nkeys) { /*---- CHK_ROLL_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_ROLL_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->roll_track, io); lib3ds_chunk_write_end(&c, io); } break; } case CHK_L_TARGET_NODE_TAG: { Lib3dsTargetNode *n = (Lib3dsTargetNode*)node; if (n->pos_track.nkeys) { /*---- CHK_POS_TRACK_TAG ----*/ Lib3dsChunk c; c.chunk = CHK_POS_TRACK_TAG; lib3ds_chunk_write_start(&c, io); lib3ds_track_write(&n->pos_track, io); lib3ds_chunk_write_end(&c, io); } break; } default: break; } lib3ds_chunk_write_end(&c, io); }