6e218cac4295521ed8055d6ecaf680faf1ab3a8e
[blender.git] / source / blender / alembic / intern / abc_transform.cc
1 /*
2  * ***** BEGIN GPL LICENSE BLOCK *****
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License
6  * as published by the Free Software Foundation; either version 2
7  * of the License, or (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  * GNU General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software Foundation,
16  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17  *
18  * Contributor(s): Esteban Tovagliari, Cedric Paille, Kevin Dietrich
19  *
20  * ***** END GPL LICENSE BLOCK *****
21  */
22
23 #include "abc_transform.h"
24
25 #include <OpenEXR/ImathBoxAlgo.h>
26
27 #include "abc_util.h"
28
29 extern "C" {
30 #include "DNA_object_types.h"
31
32 #include "BLI_math.h"
33
34 #include "BKE_object.h"
35 }
36
37 using Alembic::AbcGeom::OObject;
38 using Alembic::AbcGeom::OXform;
39
40 /* ************************************************************************** */
41
42 static bool has_parent_camera(Object *ob)
43 {
44         if (!ob->parent) {
45                 return false;
46         }
47
48         Object *parent = ob->parent;
49
50         if (parent->type == OB_CAMERA) {
51                 return true;
52         }
53
54         return has_parent_camera(parent);
55 }
56
57 /* ************************************************************************** */
58
59 AbcTransformWriter::AbcTransformWriter(Object *ob,
60                                        const OObject &abc_parent,
61                                        AbcTransformWriter *parent,
62                                        unsigned int time_sampling,
63                                        ExportSettings &settings)
64     : AbcObjectWriter(NULL, ob, time_sampling, settings, parent)
65     , m_proxy_from(NULL)
66 {
67         m_is_animated = hasAnimation(m_object);
68
69         if (!m_is_animated) {
70                 time_sampling = 0;
71         }
72
73         m_xform = OXform(abc_parent, get_id_name(m_object), time_sampling);
74         m_schema = m_xform.getSchema();
75
76         /* Blender objects can't have a parent without inheriting the transform. */
77         m_inherits_xform = parent != NULL;
78 }
79
80 void AbcTransformWriter::do_write()
81 {
82         if (m_first_frame) {
83                 m_visibility = Alembic::AbcGeom::CreateVisibilityProperty(m_xform, m_xform.getSchema().getTimeSampling());
84         }
85
86         m_visibility.set(!(m_object->restrictflag & OB_RESTRICT_VIEW));
87
88         if (!m_first_frame && !m_is_animated) {
89                 return;
90         }
91
92         float yup_mat[4][4];
93         create_transform_matrix(m_object, yup_mat,
94                                 m_inherits_xform ? ABC_MATRIX_LOCAL : ABC_MATRIX_WORLD,
95                                 m_proxy_from);
96
97         /* Only apply rotation to root camera, parenting will propagate it. */
98         if (m_object->type == OB_CAMERA && (!m_inherits_xform || !has_parent_camera(m_object))) {
99                 float rot_mat[4][4];
100                 axis_angle_to_mat4_single(rot_mat, 'X', -M_PI_2);
101                 mul_m4_m4m4(yup_mat, yup_mat, rot_mat);
102         }
103
104         if (!m_object->parent || !m_inherits_xform) {
105                 /* Only apply scaling to root objects, parenting will propagate it. */
106                 float scale_mat[4][4];
107                 scale_m4_fl(scale_mat, m_settings.global_scale);
108                 scale_mat[3][3] = m_settings.global_scale;  /* also scale translation */
109                 mul_m4_m4m4(yup_mat, yup_mat, scale_mat);
110         }
111
112         m_matrix = convert_matrix(yup_mat);
113         m_sample.setMatrix(m_matrix);
114         m_sample.setInheritsXforms(m_inherits_xform);
115         m_schema.set(m_sample);
116 }
117
118 Imath::Box3d AbcTransformWriter::bounds()
119 {
120         Imath::Box3d bounds;
121
122         for (int i = 0; i < m_children.size(); ++i) {
123                 Imath::Box3d box(m_children[i]->bounds());
124                 bounds.extendBy(box);
125         }
126
127         return Imath::transform(bounds, m_matrix);
128 }
129
130 bool AbcTransformWriter::hasAnimation(Object * /*ob*/) const
131 {
132         /* TODO(kevin): implement this. */
133         return true;
134 }
135
136 /* ************************************************************************** */
137
138 AbcEmptyReader::AbcEmptyReader(const Alembic::Abc::IObject &object, ImportSettings &settings)
139     : AbcObjectReader(object, settings)
140 {
141         /* Empties have no data. It makes the import of Alembic files easier to
142          * understand when we name the empty after its name in Alembic. */
143         m_object_name = object.getName();
144
145         Alembic::AbcGeom::IXform xform(object, Alembic::AbcGeom::kWrapExisting);
146         m_schema = xform.getSchema();
147
148         get_min_max_time(m_iobject, m_schema, m_min_time, m_max_time);
149 }
150
151 bool AbcEmptyReader::valid() const
152 {
153         return m_schema.valid();
154 }
155
156 void AbcEmptyReader::readObjectData(Main *bmain, float /*time*/)
157 {
158         m_object = BKE_object_add_only_object(bmain, OB_EMPTY,
159                                               m_object_name.c_str());
160         m_object->data = NULL;
161 }