[395] | 1 | // Copyright (C) 2011-2015 ChaosForge Ltd
|
---|
| 2 | // http://chaosforge.org/
|
---|
| 3 | //
|
---|
| 4 | // This file is part of Nova libraries.
|
---|
| 5 | // For conditions of distribution and use, see copying.txt file in root folder.
|
---|
[227] | 6 |
|
---|
| 7 | #include "nv/gfx/skeletal_mesh.hh"
|
---|
| 8 |
|
---|
| 9 | #include "nv/interface/context.hh"
|
---|
| 10 | #include "nv/interface/device.hh"
|
---|
[392] | 11 | #include "nv/stl/unordered_map.hh"
|
---|
[227] | 12 |
|
---|
[475] | 13 | #include "nv/core/logging.hh"
|
---|
[287] | 14 |
|
---|
[477] | 15 | void nv::skeletal_animation_entry::update_skeleton( skeleton_instance& data, uint32 a_ms_time ) const
|
---|
[283] | 16 | {
|
---|
[470] | 17 | float fframe = ( a_ms_time * 0.001f ) * m_fps;
|
---|
[471] | 18 | uint32 frame = uint32( math::floor( fframe ) );
|
---|
[470] | 19 | float reminder = fframe - static_cast<float>( frame );
|
---|
| 20 | uint32 duration = get_frame_count();
|
---|
| 21 | if ( duration == 0 )
|
---|
| 22 | {
|
---|
| 23 | frame = get_start_frame();
|
---|
| 24 | fframe = static_cast<float>( frame );
|
---|
| 25 | }
|
---|
| 26 | else if ( frame >= duration )
|
---|
| 27 | {
|
---|
| 28 | if ( is_looping() )
|
---|
| 29 | {
|
---|
| 30 | frame = frame % duration;
|
---|
| 31 | fframe = static_cast<float>( frame ) + reminder;
|
---|
| 32 | }
|
---|
| 33 | else
|
---|
| 34 | {
|
---|
| 35 | frame = get_end_frame();
|
---|
| 36 | fframe = static_cast<float>( frame );
|
---|
| 37 | }
|
---|
| 38 | }
|
---|
[287] | 39 |
|
---|
[477] | 40 | if ( data.size() == 0 )
|
---|
| 41 | data.initialize( m_temp_anim->size() );
|
---|
| 42 | data.animate( m_temp_anim, m_data, fframe );
|
---|
[283] | 43 | }
|
---|
| 44 |
|
---|
[467] | 45 | void nv::skeletal_animation_entry::prepare( const mesh_nodes_data* bones )
|
---|
[283] | 46 | {
|
---|
[477] | 47 | m_data.prepare( m_temp_anim, bones ? bones : m_temp_anim );
|
---|
[283] | 48 | }
|
---|
| 49 |
|
---|
[467] | 50 | nv::skeletal_mesh::skeletal_mesh( context* a_context, const data_channel_set* a_mesh, const mesh_nodes_data* a_bone_data )
|
---|
[477] | 51 | : m_skeleton( a_bone_data ? a_bone_data->size() : 0 ), m_context( a_context ), m_bone_data( a_bone_data ), m_index_count( 0 ), m_parent_id(-1)
|
---|
[287] | 52 | {
|
---|
[458] | 53 | if ( a_mesh )
|
---|
| 54 | {
|
---|
| 55 | m_va = a_context->create_vertex_array( a_mesh, nv::STATIC_DRAW );
|
---|
| 56 | m_index_count = a_mesh->get_channel_size( slot::INDEX );
|
---|
[470] | 57 | m_parent_id = a_mesh->get_parent_id();
|
---|
[458] | 58 | }
|
---|
[287] | 59 | }
|
---|
| 60 |
|
---|
[477] | 61 | void nv::skeletal_mesh::update_animation( animation_entry& a_anim, uint32 a_anim_time )
|
---|
[283] | 62 | {
|
---|
[477] | 63 | skeletal_animation_entry& anim = static_cast<skeletal_animation_entry&>( a_anim );
|
---|
| 64 | anim.prepare( m_bone_data );
|
---|
| 65 | anim.update_skeleton( m_skeleton, a_anim_time );
|
---|
[283] | 66 | }
|
---|
| 67 |
|
---|
[477] | 68 | void nv::skeletal_mesh::update_program( program a_program )
|
---|
[283] | 69 | {
|
---|
[477] | 70 | m_context->get_device()->set_opt_uniform_array( a_program, "nv_m_bones", m_skeleton.transforms(), m_skeleton.size() );
|
---|
[287] | 71 | }
|
---|
| 72 |
|
---|
[467] | 73 | nv::transform nv::skeletal_mesh::get_node_transform( uint32 node_id ) const
|
---|
[287] | 74 | {
|
---|
[477] | 75 | return transform( get_node_matrix( node_id ) );
|
---|
[287] | 76 | }
|
---|
| 77 |
|
---|
[467] | 78 | nv::mat4 nv::skeletal_mesh::get_node_matrix( uint32 node_id ) const
|
---|
[287] | 79 | {
|
---|
[477] | 80 | return m_skeleton.transforms()[ node_id ];
|
---|
[287] | 81 | }
|
---|