1 # ##### BEGIN GPL LICENSE BLOCK #####
3 # This program is free software; you can redistribute it and/or
4 # modify it under the terms of the GNU General Public License
5 # as published by the Free Software Foundation; either version 2
6 # of the License, or (at your option) any later version.
8 # This program is distributed in the hope that it will be useful,
9 # but WITHOUT ANY WARRANTY; without even the implied warranty of
10 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 # GNU General Public License for more details.
13 # You should have received a copy of the GNU General Public License
14 # along with this program; if not, write to the Free Software Foundation,
15 # Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17 # ##### END GPL LICENSE BLOCK #####
20 "name": "Export Pointcache Format(.pc2)",
21 "author": "Florian Meyer (tstscr)",
23 "blender": (2, 71, 0),
24 "location": "File > Export > Pointcache (.pc2)",
25 "description": "Export mesh Pointcache data (.pc2)",
27 "wiki_url": "http://wiki.blender.org/index.php/Extensions:2.6/Py/"
28 "Scripts/Import-Export/PC2_Pointcache_export",
29 "category": "Import-Export"}
33 https://developer.blender.org/T34456
34 https://developer.blender.org/T25408
39 cacheFile -pc2 1 -pcf "<insert filepath of source>" -f "<insert target filename w/o extension>" -dir "<insert directory path for target>" -format "OneFile";
44 from bpy
.props
import BoolProperty
, IntProperty
, EnumProperty
46 from bpy_extras
.io_utils
import ExportHelper
53 def get_sampled_frames(start
, end
, sampling
):
54 return [math
.modf(start
+ x
* sampling
) for x
in range(int((end
- start
) / sampling
) + 1)]
56 def do_export(context
, props
, filepath
):
57 mat_x90
= mathutils
.Matrix
.Rotation(-math
.pi
/2, 4, 'X')
58 ob
= context
.active_object
60 start
= props
.range_start
62 sampling
= float(props
.sampling
)
63 apply_modifiers
= props
.apply_modifiers
64 me
= ob
.to_mesh(sc
, apply_modifiers
, 'PREVIEW')
65 vertCount
= len(me
.vertices
)
66 sampletimes
= get_sampled_frames(start
, end
, sampling
)
67 sampleCount
= len(sampletimes
)
70 headerFormat
='<12siiffi'
71 headerStr
= struct
.pack(headerFormat
, b
'POINTCACHE2\0',
72 1, vertCount
, start
, sampling
, sampleCount
)
74 file = open(filepath
, "wb")
77 for frame
in sampletimes
:
78 sc
.frame_set(int(frame
[1]), frame
[0]) # stupid modf() gives decimal part first!
79 me
= ob
.to_mesh(sc
, apply_modifiers
, 'PREVIEW')
81 if len(me
.vertices
) != vertCount
:
82 bpy
.data
.meshes
.remove(me
, do_unlink
=True)
87 empty
= open(filepath
, 'w')
88 empty
.write('DUMMIFILE - export failed\n')
90 print('Export failed. Vertexcount of Object is not constant')
94 me
.transform(ob
.matrix_world
)
99 thisVertex
= struct
.pack('<fff', float(v
.co
[0]),
102 file.write(thisVertex
)
104 bpy
.data
.meshes
.remove(me
, do_unlink
=True)
112 ###### EXPORT OPERATOR #######
113 class Export_pc2(bpy
.types
.Operator
, ExportHelper
):
114 """Export the active Object as a .pc2 Pointcache file"""
115 bl_idname
= "export_shape.pc2"
116 bl_label
= "Export Pointcache (.pc2)"
118 filename_ext
= ".pc2"
120 rot_x90
= BoolProperty(name
="Convert to Y-up",
121 description
="Rotate 90 degrees around X to convert to y-up",
124 world_space
= BoolProperty(name
="Export into Worldspace",
125 description
="Transform the Vertexcoordinates into Worldspace",
128 apply_modifiers
= BoolProperty(name
="Apply Modifiers",
129 description
="Applies the Modifiers",
132 range_start
= IntProperty(name
='Start Frame',
133 description
='First frame to use for Export',
136 range_end
= IntProperty(name
='End Frame',
137 description
='Last frame to use for Export',
140 sampling
= EnumProperty(name
='Sampling',
141 description
='Sampling --> frames per sample (0.1 yields 10 samples per frame)',
142 items
=(('0.01', '0.01', ''),
143 ('0.05', '0.05', ''),
146 ('0.25', '0.25', ''),
159 def poll(cls
, context
):
160 obj
= context
.active_object
163 obj
.type in {'MESH', 'CURVE', 'SURFACE', 'FONT'}
166 def execute(self
, context
):
167 start_time
= time
.time()
168 print('\n_____START_____')
169 props
= self
.properties
170 filepath
= self
.filepath
171 filepath
= bpy
.path
.ensure_ext(filepath
, self
.filename_ext
)
173 exported
= do_export(context
, props
, filepath
)
176 print('finished export in %s seconds' %((time
.time() - start_time
)))
181 def invoke(self
, context
, event
):
182 wm
= context
.window_manager
186 wm
.fileselect_add(self
) # will run self.execute()
187 return {'RUNNING_MODAL'}
190 wm
.invoke_search_popup(self
)
191 return {'RUNNING_MODAL'}
194 return wm
.invoke_props_popup(self
, event
)
196 return self
.execute(context
)
201 def menu_func(self
, context
):
202 self
.layout
.operator(Export_pc2
.bl_idname
, text
="Pointcache (.pc2)")
206 bpy
.utils
.register_module(__name__
)
208 bpy
.types
.TOPBAR_MT_file_export
.append(menu_func
)
209 #bpy.types.VIEW3D_PT_tools_objectmode.prepend(menu_func)
212 bpy
.utils
.unregister_module(__name__
)
214 bpy
.types
.TOPBAR_MT_file_export
.remove(menu_func
)
215 #bpy.types.VIEW3D_PT_tools_objectmode.remove(menu_func)
217 if __name__
== "__main__":