libanim
changeset 21:5993f405a1cb
implemented multiple animations per node, and blending between two animations
author | John Tsiombikas <nuclear@member.fsf.org> |
---|---|
date | Fri, 27 Dec 2013 06:28:43 +0200 (2013-12-27) |
parents | 3c2428cb38f7 |
children | 9758004136f8 |
files | example/test.c src/anim.c src/anim.h |
diffstat | 3 files changed, 511 insertions(+), 90 deletions(-) [+] |
line diff
1.1 --- a/example/test.c Mon Dec 09 04:06:30 2013 +0200 1.2 +++ b/example/test.c Fri Dec 27 06:28:43 2013 +0200 1.3 @@ -48,6 +48,8 @@ 1.4 }; 1.5 1.6 int init(void); 1.7 +static void set_walk_animation(int idx); 1.8 +static void set_jump_animation(int idx); 1.9 void disp(void); 1.10 void idle(void); 1.11 void reshape(int x, int y); 1.12 @@ -60,6 +62,9 @@ 1.13 1.14 struct anm_node *nodes[NUM_NODES]; 1.15 1.16 +int cur_anim = 0, next_anim = 0; 1.17 +unsigned int trans_start_tm; 1.18 + 1.19 int main(int argc, char **argv) 1.20 { 1.21 glutInitWindowSize(800, 600); 1.22 @@ -92,15 +97,11 @@ 1.23 glEnable(GL_LIGHTING); 1.24 glEnable(GL_LIGHT0); 1.25 1.26 - root = nodes[0]; 1.27 - 1.28 for(i=0; i<NUM_NODES; i++) { 1.29 nodes[i] = anm_create_node(); 1.30 - 1.31 anm_set_pivot(nodes[i], parts[i].pivot); 1.32 - anm_set_position(nodes[i], parts[i].pos, 0); 1.33 - anm_set_extrapolator(nodes[i], ANM_EXTRAP_REPEAT); 1.34 } 1.35 + root = nodes[0]; 1.36 1.37 anm_link_node(nodes[NODE_TORSO], nodes[NODE_HEAD]); 1.38 anm_link_node(nodes[NODE_TORSO], nodes[NODE_LEFT_UPPER_LEG]); 1.39 @@ -112,6 +113,27 @@ 1.40 anm_link_node(nodes[NODE_LEFT_UPPER_ARM], nodes[NODE_LEFT_LOWER_ARM]); 1.41 anm_link_node(nodes[NODE_RIGHT_UPPER_ARM], nodes[NODE_RIGHT_LOWER_ARM]); 1.42 1.43 + set_walk_animation(0); 1.44 + 1.45 + anm_add_animation(root); /* recursively add another animation slot to all nodes */ 1.46 + set_jump_animation(1); 1.47 + 1.48 + anm_use_animation(root, cur_anim); 1.49 + 1.50 + return 0; 1.51 +} 1.52 + 1.53 +static void set_walk_animation(int idx) 1.54 +{ 1.55 + int i; 1.56 + 1.57 + anm_use_animation(root, idx); 1.58 + 1.59 + for(i=0; i<NUM_NODES; i++) { 1.60 + anm_set_position(nodes[i], parts[i].pos, 0); 1.61 + anm_set_extrapolator(nodes[i], ANM_EXTRAP_REPEAT); 1.62 + } 1.63 + 1.64 /* upper leg animation */ 1.65 anm_set_rotation(nodes[NODE_LEFT_UPPER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(-15), 1, 0, 0), 0); 1.66 anm_set_rotation(nodes[NODE_LEFT_UPPER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(45), 1, 0, 0), 1000); 1.67 @@ -163,8 +185,53 @@ 1.68 anm_set_rotation(nodes[NODE_RIGHT_LOWER_ARM], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 500); 1.69 anm_set_rotation(nodes[NODE_RIGHT_LOWER_ARM], quat_rotate(quat_identity(), DEG_TO_RAD(40), 1, 0, 0), 1000); 1.70 anm_set_rotation(nodes[NODE_RIGHT_LOWER_ARM], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 2000); 1.71 +} 1.72 1.73 - return 0; 1.74 +static void set_jump_animation(int idx) 1.75 +{ 1.76 + int i; 1.77 + 1.78 + anm_use_animation(root, idx); 1.79 + 1.80 + for(i=0; i<NUM_NODES; i++) { 1.81 + anm_set_position(nodes[i], parts[i].pos, 0); 1.82 + anm_set_extrapolator(nodes[i], ANM_EXTRAP_REPEAT); 1.83 + } 1.84 + 1.85 + anm_set_rotation(nodes[NODE_LEFT_UPPER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 0); 1.86 + anm_set_rotation(nodes[NODE_LEFT_UPPER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(40), 1, 0, 0), 1000); 1.87 + anm_set_rotation(nodes[NODE_LEFT_UPPER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 1500); 1.88 + anm_set_rotation(nodes[NODE_LEFT_UPPER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 2000); 1.89 + anm_set_rotation(nodes[NODE_RIGHT_UPPER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 0); 1.90 + anm_set_rotation(nodes[NODE_RIGHT_UPPER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(40), 1, 0, 0), 1000); 1.91 + anm_set_rotation(nodes[NODE_RIGHT_UPPER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 1500); 1.92 + anm_set_rotation(nodes[NODE_RIGHT_UPPER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 2000); 1.93 + 1.94 + anm_set_rotation(nodes[NODE_LEFT_LOWER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 0); 1.95 + anm_set_rotation(nodes[NODE_LEFT_LOWER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(-80), 1, 0, 0), 1000); 1.96 + anm_set_rotation(nodes[NODE_LEFT_LOWER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 1500); 1.97 + anm_set_rotation(nodes[NODE_LEFT_LOWER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 2000); 1.98 + anm_set_rotation(nodes[NODE_RIGHT_LOWER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 0); 1.99 + anm_set_rotation(nodes[NODE_RIGHT_LOWER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(-80), 1, 0, 0), 1000); 1.100 + anm_set_rotation(nodes[NODE_RIGHT_LOWER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 1500); 1.101 + anm_set_rotation(nodes[NODE_RIGHT_LOWER_LEG], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 2000); 1.102 + 1.103 + anm_set_position(nodes[NODE_TORSO], parts[NODE_TORSO].pos, 0); 1.104 + anm_set_position(nodes[NODE_TORSO], v3_add(parts[NODE_TORSO].pos, v3_cons(0, -1, 0)), 1000); 1.105 + anm_set_position(nodes[NODE_TORSO], v3_add(parts[NODE_TORSO].pos, v3_cons(0, 2, 0)), 1500); 1.106 + anm_set_position(nodes[NODE_TORSO], parts[NODE_TORSO].pos, 2000); 1.107 + 1.108 + anm_set_rotation(nodes[NODE_LEFT_UPPER_ARM], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 0); 1.109 + anm_set_rotation(nodes[NODE_LEFT_UPPER_ARM], quat_rotate(quat_identity(), DEG_TO_RAD(-20), 1, 0, 0), 1000); 1.110 + anm_set_rotation(nodes[NODE_LEFT_UPPER_ARM], quat_rotate(quat_identity(), DEG_TO_RAD(20), 1, 0, 0), 1200); 1.111 + anm_set_rotation(nodes[NODE_LEFT_UPPER_ARM], quat_rotate(quat_identity(), DEG_TO_RAD(170), 1, 0, 0), 1500); 1.112 + anm_set_rotation(nodes[NODE_LEFT_UPPER_ARM], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 2000); 1.113 + 1.114 + anm_set_rotation(nodes[NODE_RIGHT_UPPER_ARM], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 0); 1.115 + anm_set_rotation(nodes[NODE_RIGHT_UPPER_ARM], quat_rotate(quat_identity(), DEG_TO_RAD(-20), 1, 0, 0), 1000); 1.116 + anm_set_rotation(nodes[NODE_RIGHT_UPPER_ARM], quat_rotate(quat_identity(), DEG_TO_RAD(20), 1, 0, 0), 1200); 1.117 + anm_set_rotation(nodes[NODE_RIGHT_UPPER_ARM], quat_rotate(quat_identity(), DEG_TO_RAD(170), 1, 0, 0), 1500); 1.118 + anm_set_rotation(nodes[NODE_RIGHT_UPPER_ARM], quat_rotate(quat_identity(), DEG_TO_RAD(0), 1, 0, 0), 2000); 1.119 } 1.120 1.121 void disp(void) 1.122 @@ -184,6 +251,18 @@ 1.123 glRotatef(cam_phi, 1, 0, 0); 1.124 glRotatef(cam_theta, 0, 1, 0); 1.125 1.126 + if(cur_anim != next_anim) { 1.127 + float t = (msec - trans_start_tm) / 1000.0; 1.128 + 1.129 + if(t >= 1.0) { 1.130 + t = 1.0; 1.131 + cur_anim = next_anim; 1.132 + anm_use_animation(root, cur_anim); 1.133 + } else { 1.134 + anm_use_animations(root, cur_anim, next_anim, t); 1.135 + } 1.136 + } 1.137 + 1.138 /* first render a character with bottom-up lazy matrix calculation */ 1.139 glPushMatrix(); 1.140 glTranslatef(-2.5, 0, 0); 1.141 @@ -264,6 +343,11 @@ 1.142 switch(key) { 1.143 case 27: 1.144 exit(0); 1.145 + 1.146 + case ' ': 1.147 + next_anim = (cur_anim + 1) % 2; 1.148 + trans_start_tm = glutGet(GLUT_ELAPSED_TIME); 1.149 + break; 1.150 } 1.151 } 1.152
2.1 --- a/src/anim.c Mon Dec 09 04:06:30 2013 +0200 2.2 +++ b/src/anim.c Fri Dec 27 06:28:43 2013 +0200 2.3 @@ -8,7 +8,7 @@ 2.4 2.5 static void invalidate_cache(struct anm_node *node); 2.6 2.7 -int anm_init_node(struct anm_node *node) 2.8 +int anm_init_animation(struct anm_animation *anim) 2.9 { 2.10 int i, j; 2.11 static const float defaults[] = { 2.12 @@ -17,20 +17,49 @@ 2.13 1.0f, 1.0f, 1.0f /* default scale factor */ 2.14 }; 2.15 2.16 + anim->name = 0; 2.17 + 2.18 + for(i=0; i<ANM_NUM_TRACKS; i++) { 2.19 + if(anm_init_track(anim->tracks + i) == -1) { 2.20 + for(j=0; j<i; j++) { 2.21 + anm_destroy_track(anim->tracks + i); 2.22 + } 2.23 + } 2.24 + anm_set_track_default(anim->tracks + i, defaults[i]); 2.25 + } 2.26 + return 0; 2.27 +} 2.28 + 2.29 +void anm_destroy_animation(struct anm_animation *anim) 2.30 +{ 2.31 + int i; 2.32 + for(i=0; i<ANM_NUM_TRACKS; i++) { 2.33 + anm_destroy_track(anim->tracks + i); 2.34 + } 2.35 + free(anim->name); 2.36 +} 2.37 + 2.38 +/* ---- node implementation ----- */ 2.39 + 2.40 +int anm_init_node(struct anm_node *node) 2.41 +{ 2.42 memset(node, 0, sizeof *node); 2.43 2.44 + node->cur_anim[1] = -1; 2.45 + node->cur_mix = 0; 2.46 + 2.47 + if(!(node->animations = dynarr_alloc(1, sizeof *node->animations))) { 2.48 + return -1; 2.49 + } 2.50 + if(anm_init_animation(node->animations) == -1) { 2.51 + dynarr_free(node->animations); 2.52 + return -1; 2.53 + } 2.54 + 2.55 /* initialize thread-local matrix cache */ 2.56 pthread_key_create(&node->cache_key, 0); 2.57 pthread_mutex_init(&node->cache_list_lock, 0); 2.58 2.59 - for(i=0; i<ANM_NUM_TRACKS; i++) { 2.60 - if(anm_init_track(node->tracks + i) == -1) { 2.61 - for(j=0; j<i; j++) { 2.62 - anm_destroy_track(node->tracks + i); 2.63 - } 2.64 - } 2.65 - anm_set_track_default(node->tracks + i, defaults[i]); 2.66 - } 2.67 return 0; 2.68 } 2.69 2.70 @@ -40,8 +69,9 @@ 2.71 free(node->name); 2.72 2.73 for(i=0; i<ANM_NUM_TRACKS; i++) { 2.74 - anm_destroy_track(node->tracks + i); 2.75 + anm_destroy_animation(node->animations + i); 2.76 } 2.77 + dynarr_free(node->animations); 2.78 2.79 /* destroy thread-specific cache */ 2.80 pthread_key_delete(node->cache_key); 2.81 @@ -123,26 +153,6 @@ 2.82 return node->name ? node->name : ""; 2.83 } 2.84 2.85 -void anm_set_interpolator(struct anm_node *node, enum anm_interpolator in) 2.86 -{ 2.87 - int i; 2.88 - 2.89 - for(i=0; i<ANM_NUM_TRACKS; i++) { 2.90 - anm_set_track_interpolator(node->tracks + i, in); 2.91 - } 2.92 - invalidate_cache(node); 2.93 -} 2.94 - 2.95 -void anm_set_extrapolator(struct anm_node *node, enum anm_extrapolator ex) 2.96 -{ 2.97 - int i; 2.98 - 2.99 - for(i=0; i<ANM_NUM_TRACKS; i++) { 2.100 - anm_set_track_extrapolator(node->tracks + i, ex); 2.101 - } 2.102 - invalidate_cache(node); 2.103 -} 2.104 - 2.105 void anm_link_node(struct anm_node *p, struct anm_node *c) 2.106 { 2.107 c->next = p->child; 2.108 @@ -175,40 +185,284 @@ 2.109 return -1; 2.110 } 2.111 2.112 +void anm_set_pivot(struct anm_node *node, vec3_t piv) 2.113 +{ 2.114 + node->pivot = piv; 2.115 +} 2.116 + 2.117 +vec3_t anm_get_pivot(struct anm_node *node) 2.118 +{ 2.119 + return node->pivot; 2.120 +} 2.121 + 2.122 + 2.123 +/* animation management */ 2.124 + 2.125 +int anm_use_node_animation(struct anm_node *node, int aidx) 2.126 +{ 2.127 + if(aidx == node->cur_anim[0] && node->cur_anim[1] == -1) { 2.128 + return 0; /* no change, no invalidation */ 2.129 + } 2.130 + 2.131 + if(aidx < 0 || aidx >= anm_get_animation_count(node)) { 2.132 + return -1; 2.133 + } 2.134 + 2.135 + node->cur_anim[0] = aidx; 2.136 + node->cur_anim[1] = -1; 2.137 + node->cur_mix = 0; 2.138 + 2.139 + invalidate_cache(node); 2.140 + return 0; 2.141 +} 2.142 + 2.143 +int anm_use_node_animations(struct anm_node *node, int aidx, int bidx, float t) 2.144 +{ 2.145 + int num_anim; 2.146 + 2.147 + if(node->cur_anim[0] == aidx && node->cur_anim[1] == bidx && 2.148 + fabs(t - node->cur_mix) < 1e-6) { 2.149 + return 0; /* no change, no invalidation */ 2.150 + } 2.151 + 2.152 + num_anim = anm_get_animation_count(node); 2.153 + if(aidx < 0 || aidx >= num_anim) { 2.154 + return anm_use_animation(node, bidx); 2.155 + } 2.156 + if(bidx < 0 || bidx >= num_anim) { 2.157 + return anm_use_animation(node, aidx); 2.158 + } 2.159 + node->cur_anim[0] = aidx; 2.160 + node->cur_anim[1] = bidx; 2.161 + node->cur_mix = t; 2.162 + 2.163 + invalidate_cache(node); 2.164 + return 0; 2.165 +} 2.166 + 2.167 +int anm_use_animation(struct anm_node *node, int aidx) 2.168 +{ 2.169 + struct anm_node *child; 2.170 + 2.171 + if(anm_use_node_animation(node, aidx) == -1) { 2.172 + return -1; 2.173 + } 2.174 + 2.175 + child = node->child; 2.176 + while(child) { 2.177 + if(anm_use_animation(child, aidx) == -1) { 2.178 + return -1; 2.179 + } 2.180 + child = child->next; 2.181 + } 2.182 + return 0; 2.183 +} 2.184 + 2.185 +int anm_use_animations(struct anm_node *node, int aidx, int bidx, float t) 2.186 +{ 2.187 + struct anm_node *child; 2.188 + 2.189 + if(anm_use_node_animations(node, aidx, bidx, t) == -1) { 2.190 + return -1; 2.191 + } 2.192 + 2.193 + child = node->child; 2.194 + while(child) { 2.195 + if(anm_use_animations(child, aidx, bidx, t) == -1) { 2.196 + return -1; 2.197 + } 2.198 + child = child->next; 2.199 + } 2.200 + return 0; 2.201 + 2.202 +} 2.203 + 2.204 +int anm_get_active_animation_index(struct anm_node *node, int which) 2.205 +{ 2.206 + if(which < 0 || which >= 2) return -1; 2.207 + return node->cur_anim[which]; 2.208 +} 2.209 + 2.210 +struct anm_animation *anm_get_active_animation(struct anm_node *node, int which) 2.211 +{ 2.212 + int idx = anm_get_active_animation_index(node, which); 2.213 + if(idx < 0 || idx >= anm_get_animation_count(node)) { 2.214 + return 0; 2.215 + } 2.216 + return node->animations + idx; 2.217 +} 2.218 + 2.219 +float anm_get_active_animation_mix(struct anm_node *node) 2.220 +{ 2.221 + return node->cur_mix; 2.222 +} 2.223 + 2.224 +int anm_get_animation_count(struct anm_node *node) 2.225 +{ 2.226 + return dynarr_size(node->animations); 2.227 +} 2.228 + 2.229 +int anm_add_node_animation(struct anm_node *node) 2.230 +{ 2.231 + struct anm_animation newanim; 2.232 + anm_init_animation(&newanim); 2.233 + 2.234 + node->animations = dynarr_push(node->animations, &newanim); 2.235 + return 0; 2.236 +} 2.237 + 2.238 +int anm_remove_node_animation(struct anm_node *node, int idx) 2.239 +{ 2.240 + fprintf(stderr, "anm_remove_animation: unimplemented!"); 2.241 + abort(); 2.242 + return 0; 2.243 +} 2.244 + 2.245 +int anm_add_animation(struct anm_node *node) 2.246 +{ 2.247 + struct anm_node *child; 2.248 + 2.249 + if(anm_add_node_animation(node) == -1) { 2.250 + return -1; 2.251 + } 2.252 + 2.253 + child = node->child; 2.254 + while(child) { 2.255 + if(anm_add_animation(child)) { 2.256 + return -1; 2.257 + } 2.258 + child = child->next; 2.259 + } 2.260 + return 0; 2.261 +} 2.262 + 2.263 +int anm_remove_animation(struct anm_node *node, int idx) 2.264 +{ 2.265 + struct anm_node *child; 2.266 + 2.267 + if(anm_remove_node_animation(node, idx) == -1) { 2.268 + return -1; 2.269 + } 2.270 + 2.271 + child = node->child; 2.272 + while(child) { 2.273 + if(anm_remove_animation(child, idx) == -1) { 2.274 + return -1; 2.275 + } 2.276 + child = child->next; 2.277 + } 2.278 + return 0; 2.279 +} 2.280 + 2.281 +struct anm_animation *anm_get_animation(struct anm_node *node, int idx) 2.282 +{ 2.283 + if(idx < 0 || idx > anm_get_animation_count(node)) { 2.284 + return 0; 2.285 + } 2.286 + return node->animations + idx; 2.287 +} 2.288 + 2.289 +struct anm_animation *anm_get_animation_by_name(struct anm_node *node, const char *name) 2.290 +{ 2.291 + return anm_get_animation(node, anm_find_animation(node, name)); 2.292 +} 2.293 + 2.294 +int anm_find_animation(struct anm_node *node, const char *name) 2.295 +{ 2.296 + int i, count = anm_get_animation_count(node); 2.297 + for(i=0; i<count; i++) { 2.298 + if(strcmp(node->animations[i].name, name) == 0) { 2.299 + return i; 2.300 + } 2.301 + } 2.302 + return -1; 2.303 +} 2.304 + 2.305 +/* all the rest act on the current animation(s) */ 2.306 + 2.307 +void anm_set_interpolator(struct anm_node *node, enum anm_interpolator in) 2.308 +{ 2.309 + int i; 2.310 + struct anm_animation *anim = anm_get_active_animation(node, 0); 2.311 + if(!anim) return; 2.312 + 2.313 + for(i=0; i<ANM_NUM_TRACKS; i++) { 2.314 + anm_set_track_interpolator(anim->tracks + i, in); 2.315 + } 2.316 + invalidate_cache(node); 2.317 +} 2.318 + 2.319 +void anm_set_extrapolator(struct anm_node *node, enum anm_extrapolator ex) 2.320 +{ 2.321 + int i; 2.322 + struct anm_animation *anim = anm_get_active_animation(node, 0); 2.323 + if(!anim) return; 2.324 + 2.325 + for(i=0; i<ANM_NUM_TRACKS; i++) { 2.326 + anm_set_track_extrapolator(anim->tracks + i, ex); 2.327 + } 2.328 + invalidate_cache(node); 2.329 +} 2.330 + 2.331 void anm_set_position(struct anm_node *node, vec3_t pos, anm_time_t tm) 2.332 { 2.333 - anm_set_value(node->tracks + ANM_TRACK_POS_X, tm, pos.x); 2.334 - anm_set_value(node->tracks + ANM_TRACK_POS_Y, tm, pos.y); 2.335 - anm_set_value(node->tracks + ANM_TRACK_POS_Z, tm, pos.z); 2.336 + struct anm_animation *anim = anm_get_active_animation(node, 0); 2.337 + if(!anim) return; 2.338 + 2.339 + anm_set_value(anim->tracks + ANM_TRACK_POS_X, tm, pos.x); 2.340 + anm_set_value(anim->tracks + ANM_TRACK_POS_Y, tm, pos.y); 2.341 + anm_set_value(anim->tracks + ANM_TRACK_POS_Z, tm, pos.z); 2.342 invalidate_cache(node); 2.343 } 2.344 2.345 vec3_t anm_get_node_position(struct anm_node *node, anm_time_t tm) 2.346 { 2.347 vec3_t v; 2.348 - v.x = anm_get_value(node->tracks + ANM_TRACK_POS_X, tm); 2.349 - v.y = anm_get_value(node->tracks + ANM_TRACK_POS_Y, tm); 2.350 - v.z = anm_get_value(node->tracks + ANM_TRACK_POS_Z, tm); 2.351 + struct anm_animation *anim0 = anm_get_active_animation(node, 0); 2.352 + struct anm_animation *anim1 = anm_get_active_animation(node, 1); 2.353 + 2.354 + if(!anim0) { 2.355 + return v3_cons(0, 0, 0); 2.356 + } 2.357 + 2.358 + v.x = anm_get_value(anim0->tracks + ANM_TRACK_POS_X, tm); 2.359 + v.y = anm_get_value(anim0->tracks + ANM_TRACK_POS_Y, tm); 2.360 + v.z = anm_get_value(anim0->tracks + ANM_TRACK_POS_Z, tm); 2.361 + 2.362 + if(anim1) { 2.363 + vec3_t v1; 2.364 + v1.x = anm_get_value(anim1->tracks + ANM_TRACK_POS_X, tm); 2.365 + v1.y = anm_get_value(anim1->tracks + ANM_TRACK_POS_Y, tm); 2.366 + v1.z = anm_get_value(anim1->tracks + ANM_TRACK_POS_Z, tm); 2.367 + 2.368 + v.x = v.x + (v1.x - v.x) * node->cur_mix; 2.369 + v.y = v.y + (v1.y - v.y) * node->cur_mix; 2.370 + v.z = v.z + (v1.z - v.z) * node->cur_mix; 2.371 + } 2.372 + 2.373 return v; 2.374 } 2.375 2.376 void anm_set_rotation(struct anm_node *node, quat_t rot, anm_time_t tm) 2.377 { 2.378 - anm_set_value(node->tracks + ANM_TRACK_ROT_X, tm, rot.x); 2.379 - anm_set_value(node->tracks + ANM_TRACK_ROT_Y, tm, rot.y); 2.380 - anm_set_value(node->tracks + ANM_TRACK_ROT_Z, tm, rot.z); 2.381 - anm_set_value(node->tracks + ANM_TRACK_ROT_W, tm, rot.w); 2.382 + struct anm_animation *anim = anm_get_active_animation(node, 0); 2.383 + if(!anim) return; 2.384 + 2.385 + anm_set_value(anim->tracks + ANM_TRACK_ROT_X, tm, rot.x); 2.386 + anm_set_value(anim->tracks + ANM_TRACK_ROT_Y, tm, rot.y); 2.387 + anm_set_value(anim->tracks + ANM_TRACK_ROT_Z, tm, rot.z); 2.388 + anm_set_value(anim->tracks + ANM_TRACK_ROT_W, tm, rot.w); 2.389 invalidate_cache(node); 2.390 } 2.391 2.392 -quat_t anm_get_node_rotation(struct anm_node *node, anm_time_t tm) 2.393 +static quat_t get_node_rotation(struct anm_node *node, anm_time_t tm, struct anm_animation *anim) 2.394 { 2.395 #ifndef ROT_USE_SLERP 2.396 quat_t q; 2.397 - q.x = anm_get_value(node->tracks + ANM_TRACK_ROT_X, tm); 2.398 - q.y = anm_get_value(node->tracks + ANM_TRACK_ROT_Y, tm); 2.399 - q.z = anm_get_value(node->tracks + ANM_TRACK_ROT_Z, tm); 2.400 - q.w = anm_get_value(node->tracks + ANM_TRACK_ROT_W, tm); 2.401 + q.x = anm_get_value(anim->tracks + ANM_TRACK_ROT_X, tm); 2.402 + q.y = anm_get_value(anim->tracks + ANM_TRACK_ROT_Y, tm); 2.403 + q.z = anm_get_value(anim->tracks + ANM_TRACK_ROT_Z, tm); 2.404 + q.w = anm_get_value(anim->tracks + ANM_TRACK_ROT_W, tm); 2.405 return q; 2.406 #else 2.407 int idx0, idx1, last_idx; 2.408 @@ -217,10 +471,10 @@ 2.409 struct anm_track *track_x, *track_y, *track_z, *track_w; 2.410 quat_t q, q1, q2; 2.411 2.412 - track_x = node->tracks + ANM_TRACK_ROT_X; 2.413 - track_y = node->tracks + ANM_TRACK_ROT_Y; 2.414 - track_z = node->tracks + ANM_TRACK_ROT_Z; 2.415 - track_w = node->tracks + ANM_TRACK_ROT_W; 2.416 + track_x = anim->tracks + ANM_TRACK_ROT_X; 2.417 + track_y = anim->tracks + ANM_TRACK_ROT_Y; 2.418 + track_z = anim->tracks + ANM_TRACK_ROT_Z; 2.419 + track_w = anim->tracks + ANM_TRACK_ROT_W; 2.420 2.421 if(!track_x->count) { 2.422 q.x = track_x->def_val; 2.423 @@ -277,20 +531,62 @@ 2.424 #endif 2.425 } 2.426 2.427 +quat_t anm_get_node_rotation(struct anm_node *node, anm_time_t tm) 2.428 +{ 2.429 + quat_t q; 2.430 + struct anm_animation *anim0 = anm_get_active_animation(node, 0); 2.431 + struct anm_animation *anim1 = anm_get_active_animation(node, 1); 2.432 + 2.433 + if(!anim0) { 2.434 + return quat_identity(); 2.435 + } 2.436 + 2.437 + q = get_node_rotation(node, tm, anim0); 2.438 + 2.439 + if(anim1) { 2.440 + quat_t q1 = get_node_rotation(node, tm, anim1); 2.441 + 2.442 + q = quat_slerp(q, q1, node->cur_mix); 2.443 + } 2.444 + return q; 2.445 +} 2.446 + 2.447 void anm_set_scaling(struct anm_node *node, vec3_t scl, anm_time_t tm) 2.448 { 2.449 - anm_set_value(node->tracks + ANM_TRACK_SCL_X, tm, scl.x); 2.450 - anm_set_value(node->tracks + ANM_TRACK_SCL_Y, tm, scl.y); 2.451 - anm_set_value(node->tracks + ANM_TRACK_SCL_Z, tm, scl.z); 2.452 + struct anm_animation *anim = anm_get_active_animation(node, 0); 2.453 + if(!anim) return; 2.454 + 2.455 + anm_set_value(anim->tracks + ANM_TRACK_SCL_X, tm, scl.x); 2.456 + anm_set_value(anim->tracks + ANM_TRACK_SCL_Y, tm, scl.y); 2.457 + anm_set_value(anim->tracks + ANM_TRACK_SCL_Z, tm, scl.z); 2.458 invalidate_cache(node); 2.459 } 2.460 2.461 vec3_t anm_get_node_scaling(struct anm_node *node, anm_time_t tm) 2.462 { 2.463 vec3_t v; 2.464 - v.x = anm_get_value(node->tracks + ANM_TRACK_SCL_X, tm); 2.465 - v.y = anm_get_value(node->tracks + ANM_TRACK_SCL_Y, tm); 2.466 - v.z = anm_get_value(node->tracks + ANM_TRACK_SCL_Z, tm); 2.467 + struct anm_animation *anim0 = anm_get_active_animation(node, 0); 2.468 + struct anm_animation *anim1 = anm_get_active_animation(node, 1); 2.469 + 2.470 + if(!anim0) { 2.471 + return v3_cons(1, 1, 1); 2.472 + } 2.473 + 2.474 + v.x = anm_get_value(anim0->tracks + ANM_TRACK_SCL_X, tm); 2.475 + v.y = anm_get_value(anim0->tracks + ANM_TRACK_SCL_Y, tm); 2.476 + v.z = anm_get_value(anim0->tracks + ANM_TRACK_SCL_Z, tm); 2.477 + 2.478 + if(anim1) { 2.479 + vec3_t v1; 2.480 + v1.x = anm_get_value(anim1->tracks + ANM_TRACK_SCL_X, tm); 2.481 + v1.y = anm_get_value(anim1->tracks + ANM_TRACK_SCL_Y, tm); 2.482 + v1.z = anm_get_value(anim1->tracks + ANM_TRACK_SCL_Z, tm); 2.483 + 2.484 + v.x = v.x + (v1.x - v.x) * node->cur_mix; 2.485 + v.y = v.y + (v1.y - v.y) * node->cur_mix; 2.486 + v.z = v.z + (v1.z - v.z) * node->cur_mix; 2.487 + } 2.488 + 2.489 return v; 2.490 } 2.491 2.492 @@ -334,16 +630,6 @@ 2.493 return v3_mul(s, ps); 2.494 } 2.495 2.496 -void anm_set_pivot(struct anm_node *node, vec3_t piv) 2.497 -{ 2.498 - node->pivot = piv; 2.499 -} 2.500 - 2.501 -vec3_t anm_get_pivot(struct anm_node *node) 2.502 -{ 2.503 - return node->pivot; 2.504 -} 2.505 - 2.506 void anm_get_node_matrix(struct anm_node *node, mat4_t mat, anm_time_t tm) 2.507 { 2.508 int i; 2.509 @@ -463,15 +749,20 @@ 2.510 2.511 anm_time_t anm_get_start_time(struct anm_node *node) 2.512 { 2.513 - int i; 2.514 + int i, j; 2.515 struct anm_node *c; 2.516 anm_time_t res = LONG_MAX; 2.517 2.518 - for(i=0; i<ANM_NUM_TRACKS; i++) { 2.519 - if(node->tracks[i].count) { 2.520 - anm_time_t tm = node->tracks[i].keys[0].time; 2.521 - if(tm < res) { 2.522 - res = tm; 2.523 + for(j=0; j<2; j++) { 2.524 + struct anm_animation *anim = anm_get_active_animation(node, j); 2.525 + if(!anim) break; 2.526 + 2.527 + for(i=0; i<ANM_NUM_TRACKS; i++) { 2.528 + if(anim->tracks[i].count) { 2.529 + anm_time_t tm = anim->tracks[i].keys[0].time; 2.530 + if(tm < res) { 2.531 + res = tm; 2.532 + } 2.533 } 2.534 } 2.535 } 2.536 @@ -489,15 +780,20 @@ 2.537 2.538 anm_time_t anm_get_end_time(struct anm_node *node) 2.539 { 2.540 - int i; 2.541 + int i, j; 2.542 struct anm_node *c; 2.543 anm_time_t res = LONG_MIN; 2.544 2.545 - for(i=0; i<ANM_NUM_TRACKS; i++) { 2.546 - if(node->tracks[i].count) { 2.547 - anm_time_t tm = node->tracks[i].keys[node->tracks[i].count - 1].time; 2.548 - if(tm > res) { 2.549 - res = tm; 2.550 + for(j=0; j<2; j++) { 2.551 + struct anm_animation *anim = anm_get_active_animation(node, j); 2.552 + if(!anim) break; 2.553 + 2.554 + for(i=0; i<ANM_NUM_TRACKS; i++) { 2.555 + if(anim->tracks[i].count) { 2.556 + anm_time_t tm = anim->tracks[i].keys[anim->tracks[i].count - 1].time; 2.557 + if(tm > res) { 2.558 + res = tm; 2.559 + } 2.560 } 2.561 } 2.562 }
3.1 --- a/src/anim.h Mon Dec 09 04:06:30 2013 +0200 3.2 +++ b/src/anim.h Fri Dec 27 06:28:43 2013 +0200 3.3 @@ -27,10 +27,18 @@ 3.4 ANM_NUM_TRACKS 3.5 }; 3.6 3.7 +struct anm_animation { 3.8 + char *name; 3.9 + struct anm_track tracks[ANM_NUM_TRACKS]; 3.10 +}; 3.11 + 3.12 struct anm_node { 3.13 char *name; 3.14 3.15 - struct anm_track tracks[ANM_NUM_TRACKS]; 3.16 + int cur_anim[2]; 3.17 + float cur_mix; 3.18 + 3.19 + struct anm_animation *animations; 3.20 vec3_t pivot; 3.21 3.22 /* matrix cache */ 3.23 @@ -54,6 +62,9 @@ 3.24 extern "C" { 3.25 #endif 3.26 3.27 +int anm_init_animation(struct anm_animation *anim); 3.28 +void anm_destroy_animation(struct anm_animation *anim); 3.29 + 3.30 /* node constructor and destructor */ 3.31 int anm_init_node(struct anm_node *node); 3.32 void anm_destroy_node(struct anm_node *node); 3.33 @@ -74,13 +85,46 @@ 3.34 int anm_set_node_name(struct anm_node *node, const char *name); 3.35 const char *anm_get_node_name(struct anm_node *node); 3.36 3.37 -void anm_set_interpolator(struct anm_node *node, enum anm_interpolator in); 3.38 -void anm_set_extrapolator(struct anm_node *node, enum anm_extrapolator ex); 3.39 - 3.40 /* link and unlink nodes with parent/child relations */ 3.41 void anm_link_node(struct anm_node *parent, struct anm_node *child); 3.42 int anm_unlink_node(struct anm_node *parent, struct anm_node *child); 3.43 3.44 +void anm_set_pivot(struct anm_node *node, vec3_t pivot); 3.45 +vec3_t anm_get_pivot(struct anm_node *node); 3.46 + 3.47 +/* set active animation(s) */ 3.48 +int anm_use_node_animation(struct anm_node *node, int aidx); 3.49 +int anm_use_node_animations(struct anm_node *node, int aidx, int bidx, float t); 3.50 +/* recursive variants */ 3.51 +int anm_use_animation(struct anm_node *node, int aidx); 3.52 +int anm_use_animations(struct anm_node *node, int aidx, int bidx, float t); 3.53 + 3.54 +/* returns the requested current animation index, which can be 0 or 1 */ 3.55 +int anm_get_active_animation_index(struct anm_node *node, int which); 3.56 +/* returns the requested current animation, which can be 0 or 1 */ 3.57 +struct anm_animation *anm_get_active_animation(struct anm_node *node, int which); 3.58 +float anm_get_active_animation_mix(struct anm_node *node); 3.59 + 3.60 +int anm_get_animation_count(struct anm_node *node); 3.61 + 3.62 +/* add/remove an animation to the specified node */ 3.63 +int anm_add_node_animation(struct anm_node *node); 3.64 +int anm_remove_node_animation(struct anm_node *node, int idx); 3.65 + 3.66 +/* add/remove an animation to the specified node and all it's descendants */ 3.67 +int anm_add_animation(struct anm_node *node); 3.68 +int anm_remove_animation(struct anm_node *node, int idx); 3.69 + 3.70 +struct anm_animation *anm_get_animation(struct anm_node *node, int idx); 3.71 +struct anm_animation *anm_get_animation_by_name(struct anm_node *node, const char *name); 3.72 + 3.73 +int anm_find_animation(struct anm_node *node, const char *name); 3.74 + 3.75 +/* set the interpolator for the (first) currently active animation */ 3.76 +void anm_set_interpolator(struct anm_node *node, enum anm_interpolator in); 3.77 +/* set the extrapolator for the (first) currently active animation */ 3.78 +void anm_set_extrapolator(struct anm_node *node, enum anm_extrapolator ex); 3.79 + 3.80 void anm_set_position(struct anm_node *node, vec3_t pos, anm_time_t tm); 3.81 vec3_t anm_get_node_position(struct anm_node *node, anm_time_t tm); 3.82 3.83 @@ -95,9 +139,6 @@ 3.84 quat_t anm_get_rotation(struct anm_node *node, anm_time_t tm); 3.85 vec3_t anm_get_scaling(struct anm_node *node, anm_time_t tm); 3.86 3.87 -void anm_set_pivot(struct anm_node *node, vec3_t pivot); 3.88 -vec3_t anm_get_pivot(struct anm_node *node); 3.89 - 3.90 /* those return the start and end times of the whole tree */ 3.91 anm_time_t anm_get_start_time(struct anm_node *node); 3.92 anm_time_t anm_get_end_time(struct anm_node *node);