Alembic export: write curve/NURBS as mesh
[blender.git] / source / blender / editors / io / io_alembic.c
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  * The Original Code is Copyright (C) 2016 Blender Foundation.
19  * All rights reserved.
20  *
21  * ***** END GPL LICENSE BLOCK *****
22  *
23  */
24
25 /** \file blender/editors/io/io_alembic.c
26  *  \ingroup editor/io
27  */
28
29 #ifdef WITH_ALEMBIC
30
31 /* needed for directory lookup */
32 #ifndef WIN32
33 #  include <dirent.h>
34 #else
35 #  include "BLI_winstuff.h"
36 #endif
37
38 #include <string.h>
39 #include <errno.h>
40
41 #include "MEM_guardedalloc.h"
42
43 #include "DNA_mesh_types.h"
44 #include "DNA_modifier_types.h"
45 #include "DNA_object_types.h"
46 #include "DNA_scene_types.h"
47 #include "DNA_space_types.h"
48
49 #include "BKE_context.h"
50 #include "BKE_global.h"
51 #include "BKE_main.h"
52 #include "BKE_report.h"
53
54 #include "BLI_listbase.h"
55 #include "BLI_math_vector.h"
56 #include "BLI_path_util.h"
57 #include "BLI_string.h"
58 #include "BLI_utildefines.h"
59
60 #include "BLT_translation.h"
61
62 #include "RNA_access.h"
63 #include "RNA_define.h"
64 #include "RNA_enum_types.h"
65
66 #include "ED_object.h"
67
68 #include "UI_interface.h"
69 #include "UI_resources.h"
70
71 #include "WM_api.h"
72 #include "WM_types.h"
73
74 #include "io_alembic.h"
75
76 #include "ABC_alembic.h"
77
78 static int wm_alembic_export_invoke(bContext *C, wmOperator *op, const wmEvent *event)
79 {
80         if (!RNA_struct_property_is_set(op->ptr, "as_background_job")) {
81                 RNA_boolean_set(op->ptr, "as_background_job", true);
82         }
83
84         RNA_boolean_set(op->ptr, "init_scene_frame_range", true);
85
86         if (!RNA_struct_property_is_set(op->ptr, "filepath")) {
87                 Main *bmain = CTX_data_main(C);
88                 char filepath[FILE_MAX];
89
90                 if (BKE_main_blendfile_path(bmain)[0] == '\0') {
91                         BLI_strncpy(filepath, "untitled", sizeof(filepath));
92                 }
93                 else {
94                         BLI_strncpy(filepath, BKE_main_blendfile_path(bmain), sizeof(filepath));
95                 }
96
97                 BLI_path_extension_replace(filepath, sizeof(filepath), ".abc");
98                 RNA_string_set(op->ptr, "filepath", filepath);
99         }
100
101         WM_event_add_fileselect(C, op);
102
103         return OPERATOR_RUNNING_MODAL;
104
105         UNUSED_VARS(event);
106 }
107
108 static int wm_alembic_export_exec(bContext *C, wmOperator *op)
109 {
110         if (!RNA_struct_property_is_set(op->ptr, "filepath")) {
111                 BKE_report(op->reports, RPT_ERROR, "No filename given");
112                 return OPERATOR_CANCELLED;
113         }
114
115         char filename[FILE_MAX];
116         RNA_string_get(op->ptr, "filepath", filename);
117
118         struct AlembicExportParams params = {
119             .frame_start = RNA_int_get(op->ptr, "start"),
120             .frame_end = RNA_int_get(op->ptr, "end"),
121
122             .frame_samples_xform = RNA_int_get(op->ptr, "xsamples"),
123             .frame_samples_shape = RNA_int_get(op->ptr, "gsamples"),
124
125             .shutter_open = RNA_float_get(op->ptr, "sh_open"),
126             .shutter_close = RNA_float_get(op->ptr, "sh_close"),
127
128             .selected_only = RNA_boolean_get(op->ptr, "selected"),
129             .uvs = RNA_boolean_get(op->ptr, "uvs"),
130             .normals = RNA_boolean_get(op->ptr, "normals"),
131             .vcolors = RNA_boolean_get(op->ptr, "vcolors"),
132             .apply_subdiv = RNA_boolean_get(op->ptr, "apply_subdiv"),
133             .curves_as_mesh = RNA_boolean_get(op->ptr, "curves_as_mesh"),
134             .flatten_hierarchy = RNA_boolean_get(op->ptr, "flatten"),
135             .visible_layers_only = RNA_boolean_get(op->ptr, "visible_layers_only"),
136             .renderable_only = RNA_boolean_get(op->ptr, "renderable_only"),
137             .face_sets = RNA_boolean_get(op->ptr, "face_sets"),
138             .use_subdiv_schema = RNA_boolean_get(op->ptr, "subdiv_schema"),
139             .export_hair = RNA_boolean_get(op->ptr, "export_hair"),
140             .export_particles = RNA_boolean_get(op->ptr, "export_particles"),
141             .compression_type = RNA_enum_get(op->ptr, "compression_type"),
142             .packuv = RNA_boolean_get(op->ptr, "packuv"),
143             .triangulate = RNA_boolean_get(op->ptr, "triangulate"),
144             .quad_method = RNA_enum_get(op->ptr, "quad_method"),
145             .ngon_method = RNA_enum_get(op->ptr, "ngon_method"),
146
147             .global_scale = RNA_float_get(op->ptr, "global_scale"),
148         };
149
150         /* Take some defaults from the scene, if not specified explicitly. */
151         Scene *scene = CTX_data_scene(C);
152         if (params.frame_start == INT_MIN) {
153                 params.frame_start = SFRA;
154         }
155         if (params.frame_end == INT_MIN) {
156                 params.frame_end = EFRA;
157         }
158
159         const bool as_background_job = RNA_boolean_get(op->ptr, "as_background_job");
160         bool ok = ABC_export(scene, C, filename, &params, as_background_job);
161
162         return as_background_job || ok ? OPERATOR_FINISHED : OPERATOR_CANCELLED;
163 }
164
165 static void ui_alembic_export_settings(uiLayout *layout, PointerRNA *imfptr)
166 {
167         uiLayout *box;
168         uiLayout *row;
169         uiLayout *col;
170
171 #ifdef WITH_ALEMBIC_HDF5
172         box = uiLayoutBox(layout);
173         row = uiLayoutRow(box, false);
174         uiItemL(row, IFACE_("Archive Options:"), ICON_NONE);
175
176         row = uiLayoutRow(box, false);
177         uiItemR(row, imfptr, "compression_type", 0, NULL, ICON_NONE);
178 #endif
179
180         box = uiLayoutBox(layout);
181         row = uiLayoutRow(box, false);
182         uiItemL(row, IFACE_("Manual Transform:"), ICON_NONE);
183
184         row = uiLayoutRow(box, false);
185         uiItemR(row, imfptr, "global_scale", 0, NULL, ICON_NONE);
186
187         /* Scene Options */
188         box = uiLayoutBox(layout);
189         row = uiLayoutRow(box, false);
190         uiItemL(row, IFACE_("Scene Options:"), ICON_SCENE_DATA);
191
192         row = uiLayoutRow(box, false);
193         uiItemR(row, imfptr, "start", 0, NULL, ICON_NONE);
194
195         row = uiLayoutRow(box, false);
196         uiItemR(row, imfptr, "end", 0, NULL, ICON_NONE);
197
198         row = uiLayoutRow(box, false);
199         uiItemR(row, imfptr, "xsamples", 0, NULL, ICON_NONE);
200
201         row = uiLayoutRow(box, false);
202         uiItemR(row, imfptr, "gsamples", 0, NULL, ICON_NONE);
203
204         row = uiLayoutRow(box, false);
205         uiItemR(row, imfptr, "sh_open", 0, NULL, ICON_NONE);
206
207         row = uiLayoutRow(box, false);
208         uiItemR(row, imfptr, "sh_close", 0, NULL, ICON_NONE);
209
210         row = uiLayoutRow(box, false);
211         uiItemR(row, imfptr, "selected", 0, NULL, ICON_NONE);
212
213         row = uiLayoutRow(box, false);
214         uiItemR(row, imfptr, "renderable_only", 0, NULL, ICON_NONE);
215
216         row = uiLayoutRow(box, false);
217         uiItemR(row, imfptr, "visible_layers_only", 0, NULL, ICON_NONE);
218
219         row = uiLayoutRow(box, false);
220         uiItemR(row, imfptr, "flatten", 0, NULL, ICON_NONE);
221
222         /* Object Data */
223         box = uiLayoutBox(layout);
224         row = uiLayoutRow(box, false);
225         uiItemL(row, IFACE_("Object Options:"), ICON_OBJECT_DATA);
226
227         row = uiLayoutRow(box, false);
228         uiItemR(row, imfptr, "uvs", 0, NULL, ICON_NONE);
229
230         row = uiLayoutRow(box, false);
231         uiItemR(row, imfptr, "packuv", 0, NULL, ICON_NONE);
232         uiLayoutSetEnabled(row, RNA_boolean_get(imfptr, "uvs"));
233
234         row = uiLayoutRow(box, false);
235         uiItemR(row, imfptr, "normals", 0, NULL, ICON_NONE);
236
237         row = uiLayoutRow(box, false);
238         uiItemR(row, imfptr, "vcolors", 0, NULL, ICON_NONE);
239
240         row = uiLayoutRow(box, false);
241         uiItemR(row, imfptr, "face_sets", 0, NULL, ICON_NONE);
242
243         row = uiLayoutRow(box, false);
244         uiItemR(row, imfptr, "subdiv_schema", 0, NULL, ICON_NONE);
245
246         row = uiLayoutRow(box, false);
247         uiItemR(row, imfptr, "apply_subdiv", 0, NULL, ICON_NONE);
248
249         row = uiLayoutRow(box, false);
250         uiItemR(row, imfptr, "curves_as_mesh", 0, NULL, ICON_NONE);
251
252         row = uiLayoutRow(box, false);
253         uiItemR(row, imfptr, "triangulate", 0, NULL, ICON_NONE);
254
255         const bool triangulate = RNA_boolean_get(imfptr, "triangulate");
256
257         row = uiLayoutRow(box, false);
258         uiLayoutSetEnabled(row, triangulate);
259         uiItemR(row, imfptr, "quad_method", 0, NULL, ICON_NONE);
260
261         row = uiLayoutRow(box, false);
262         uiLayoutSetEnabled(row, triangulate);
263         uiItemR(row, imfptr, "ngon_method", 0, NULL, ICON_NONE);
264
265         /* Object Data */
266         box = uiLayoutBox(layout);
267         row = uiLayoutRow(box, false);
268         uiItemL(row, IFACE_("Particle Systems:"), ICON_PARTICLE_DATA);
269
270         col = uiLayoutColumn(box, true);
271         uiItemR(col, imfptr, "export_hair", 0, NULL, ICON_NONE);
272         uiItemR(col, imfptr, "export_particles", 0, NULL, ICON_NONE);
273 }
274
275 static void wm_alembic_export_draw(bContext *C, wmOperator *op)
276 {
277         PointerRNA ptr;
278
279         RNA_pointer_create(NULL, op->type->srna, op->properties, &ptr);
280
281         /* Conveniently set start and end frame to match the scene's frame range. */
282         Scene *scene = CTX_data_scene(C);
283
284         if (scene != NULL && RNA_boolean_get(&ptr, "init_scene_frame_range")) {
285                 RNA_int_set(&ptr, "start", SFRA);
286                 RNA_int_set(&ptr, "end", EFRA);
287
288                 RNA_boolean_set(&ptr, "init_scene_frame_range", false);
289         }
290
291         ui_alembic_export_settings(op->layout, &ptr);
292 }
293
294 static bool wm_alembic_export_check(bContext *UNUSED(C), wmOperator *op)
295 {
296         char filepath[FILE_MAX];
297         RNA_string_get(op->ptr, "filepath", filepath);
298
299         if (!BLI_path_extension_check(filepath, ".abc")) {
300                 BLI_path_extension_ensure(filepath, FILE_MAX, ".abc");
301                 RNA_string_set(op->ptr, "filepath", filepath);
302                 return true;
303         }
304
305         return false;
306 }
307
308 void WM_OT_alembic_export(wmOperatorType *ot)
309 {
310         ot->name = "Export Alembic";
311         ot->description = "Export current scene in an Alembic archive";
312         ot->idname = "WM_OT_alembic_export";
313
314         ot->invoke = wm_alembic_export_invoke;
315         ot->exec = wm_alembic_export_exec;
316         ot->poll = WM_operator_winactive;
317         ot->ui = wm_alembic_export_draw;
318         ot->check = wm_alembic_export_check;
319
320         WM_operator_properties_filesel(ot, FILE_TYPE_FOLDER | FILE_TYPE_ALEMBIC,
321                                        FILE_BLENDER, FILE_SAVE, WM_FILESEL_FILEPATH,
322                                        FILE_DEFAULTDISPLAY, FILE_SORT_ALPHA);
323
324         RNA_def_int(ot->srna, "start", INT_MIN, INT_MIN, INT_MAX,
325                     "Start Frame",
326                     "Start frame of the export, use the default value to "
327                     "take the start frame of the current scene",
328                     INT_MIN, INT_MAX);
329
330         RNA_def_int(ot->srna, "end", INT_MIN, INT_MIN, INT_MAX,
331                     "End Frame",
332                     "End frame of the export, use the default value to "
333                     "take the end frame of the current scene",
334                     INT_MIN, INT_MAX);
335
336         RNA_def_int(ot->srna, "xsamples", 1, 1, 128,
337                     "Transform Samples", "Number of times per frame transformations are sampled", 1, 128);
338
339         RNA_def_int(ot->srna, "gsamples", 1, 1, 128,
340                     "Geometry Samples", "Number of times per frame object data are sampled", 1, 128);
341
342         RNA_def_float(ot->srna, "sh_open", 0.0f, -1.0f, 1.0f,
343                       "Shutter Open", "Time at which the shutter is open", -1.0f, 1.0f);
344
345         RNA_def_float(ot->srna, "sh_close", 1.0f, -1.0f, 1.0f,
346                       "Shutter Close", "Time at which the shutter is closed", -1.0f, 1.0f);
347
348         RNA_def_boolean(ot->srna, "selected", 0,
349                         "Selected Objects Only", "Export only selected objects");
350
351         RNA_def_boolean(ot->srna, "renderable_only", 1,
352                         "Renderable Objects Only",
353                         "Export only objects marked renderable in the outliner");
354
355         RNA_def_boolean(ot->srna, "visible_layers_only", 0,
356                         "Visible Layers Only", "Export only objects in visible layers");
357
358         RNA_def_boolean(ot->srna, "flatten", 0,
359                         "Flatten Hierarchy",
360                         "Do not preserve objects' parent/children relationship");
361
362         RNA_def_boolean(ot->srna, "uvs", 1, "UVs", "Export UVs");
363
364         RNA_def_boolean(ot->srna, "packuv", 1, "Pack UV Islands",
365                         "Export UVs with packed island");
366
367         RNA_def_boolean(ot->srna, "normals", 1, "Normals", "Export normals");
368
369         RNA_def_boolean(ot->srna, "vcolors", 0, "Vertex Colors", "Export vertex colors");
370
371         RNA_def_boolean(ot->srna, "face_sets", 0, "Face Sets", "Export per face shading group assignments");
372
373         RNA_def_boolean(ot->srna, "subdiv_schema", 0,
374                         "Use Subdivision Schema",
375                         "Export meshes using Alembic's subdivision schema");
376
377         RNA_def_boolean(ot->srna, "apply_subdiv", 0,
378                         "Apply Subsurf", "Export subdivision surfaces as meshes");
379
380         RNA_def_boolean(ot->srna, "curves_as_mesh", false,
381                         "Curves as Mesh", "Export curves and NURBS surfaces as meshes");
382
383         RNA_def_enum(ot->srna, "compression_type", rna_enum_abc_compression_items,
384                      ABC_ARCHIVE_OGAWA, "Compression", "");
385
386         RNA_def_float(ot->srna, "global_scale", 1.0f, 0.0001f, 1000.0f, "Scale",
387                       "Value by which to enlarge or shrink the objects with respect to the world's origin",
388                       0.0001f, 1000.0f);
389
390         RNA_def_boolean(ot->srna, "triangulate", false, "Triangulate",
391                         "Export Polygons (Quads & NGons) as Triangles");
392
393         RNA_def_enum(ot->srna, "quad_method", rna_enum_modifier_triangulate_quad_method_items,
394                      MOD_TRIANGULATE_QUAD_SHORTEDGE, "Quad Method", "Method for splitting the quads into triangles");
395
396         RNA_def_enum(ot->srna, "ngon_method", rna_enum_modifier_triangulate_quad_method_items,
397                      MOD_TRIANGULATE_NGON_BEAUTY, "Polygon Method", "Method for splitting the polygons into triangles");
398
399         RNA_def_boolean(ot->srna, "export_hair", 1, "Export Hair", "Exports hair particle systems as animated curves");
400         RNA_def_boolean(ot->srna, "export_particles", 1, "Export Particles", "Exports non-hair particle systems");
401
402         RNA_def_boolean(ot->srna, "as_background_job", false, "Run as Background Job",
403                         "Enable this to run the import in the background, disable to block Blender while importing. "
404                         "This option is deprecated; EXECUTE this operator to run in the foreground, and INVOKE it "
405                         "to run as a background job");
406
407         /* This dummy prop is used to check whether we need to init the start and
408          * end frame values to that of the scene's, otherwise they are reset at
409          * every change, draw update. */
410         RNA_def_boolean(ot->srna, "init_scene_frame_range", false, "", "");
411 }
412
413 /* ************************************************************************** */
414
415 /* TODO(kevin): check on de-duplicating all this with code in image_ops.c */
416
417 typedef struct CacheFrame {
418         struct CacheFrame *next, *prev;
419         int framenr;
420 } CacheFrame;
421
422 static int cmp_frame(const void *a, const void *b)
423 {
424         const CacheFrame *frame_a = a;
425         const CacheFrame *frame_b = b;
426
427         if (frame_a->framenr < frame_b->framenr) return -1;
428         if (frame_a->framenr > frame_b->framenr) return 1;
429         return 0;
430 }
431
432 static int get_sequence_len(char *filename, int *ofs)
433 {
434         int frame;
435         int numdigit;
436
437         if (!BLI_path_frame_get(filename, &frame, &numdigit)) {
438                 return 1;
439         }
440
441         char path[FILE_MAX];
442         BLI_path_abs(filename, BKE_main_blendfile_path_from_global());
443         BLI_split_dir_part(filename, path, FILE_MAX);
444
445         if (path[0] == '\0') {
446                 /* The filename had no path, so just use the blend file path. */
447                 BLI_split_dir_part(BKE_main_blendfile_path_from_global(), path, FILE_MAX);
448         }
449
450         DIR *dir = opendir(path);
451         if (dir == NULL) {
452                 fprintf(stderr, "Error opening directory '%s': %s\n",
453                         path, errno ? strerror(errno) : "unknown error");
454                 return -1;
455         }
456
457         const char *ext = ".abc";
458         const char *basename = BLI_path_basename(filename);
459         const int len = strlen(basename) - (numdigit + strlen(ext));
460
461         ListBase frames;
462         BLI_listbase_clear(&frames);
463
464         struct dirent *fname;
465         while ((fname = readdir(dir)) != NULL) {
466                 /* do we have the right extension? */
467                 if (!strstr(fname->d_name, ext)) {
468                         continue;
469                 }
470
471                 if (!STREQLEN(basename, fname->d_name, len)) {
472                         continue;
473                 }
474
475                 CacheFrame *cache_frame = MEM_callocN(sizeof(CacheFrame), "abc_frame");
476
477                 BLI_path_frame_get(fname->d_name, &cache_frame->framenr, &numdigit);
478
479                 BLI_addtail(&frames, cache_frame);
480         }
481
482         closedir(dir);
483
484         BLI_listbase_sort(&frames, cmp_frame);
485
486         CacheFrame *cache_frame = frames.first;
487
488         if (cache_frame) {
489                 int frame_curr = cache_frame->framenr;
490                 (*ofs) = frame_curr;
491
492                 while (cache_frame && (cache_frame->framenr == frame_curr)) {
493                         ++frame_curr;
494                         cache_frame = cache_frame->next;
495                 }
496
497                 BLI_freelistN(&frames);
498
499                 return frame_curr - (*ofs);
500         }
501
502         return 1;
503 }
504
505 /* ************************************************************************** */
506
507 static void ui_alembic_import_settings(uiLayout *layout, PointerRNA *imfptr)
508 {
509         uiLayout *box = uiLayoutBox(layout);
510         uiLayout *row = uiLayoutRow(box, false);
511         uiItemL(row, IFACE_("Manual Transform:"), ICON_NONE);
512
513         row = uiLayoutRow(box, false);
514         uiItemR(row, imfptr, "scale", 0, NULL, ICON_NONE);
515
516         box = uiLayoutBox(layout);
517         row = uiLayoutRow(box, false);
518         uiItemL(row, IFACE_("Options:"), ICON_NONE);
519
520         row = uiLayoutRow(box, false);
521         uiItemR(row, imfptr, "set_frame_range", 0, NULL, ICON_NONE);
522
523         row = uiLayoutRow(box, false);
524         uiItemR(row, imfptr, "is_sequence", 0, NULL, ICON_NONE);
525
526         row = uiLayoutRow(box, false);
527         uiItemR(row, imfptr, "validate_meshes", 0, NULL, ICON_NONE);
528 }
529
530 static void wm_alembic_import_draw(bContext *UNUSED(C), wmOperator *op)
531 {
532         PointerRNA ptr;
533
534         RNA_pointer_create(NULL, op->type->srna, op->properties, &ptr);
535         ui_alembic_import_settings(op->layout, &ptr);
536 }
537
538
539 /* op->invoke, opens fileselect if path property not set, otherwise executes */
540 static int wm_alembic_import_invoke(bContext *C, wmOperator *op, const wmEvent *event)
541 {
542         if (!RNA_struct_property_is_set(op->ptr, "as_background_job")) {
543                 RNA_boolean_set(op->ptr, "as_background_job", true);
544         }
545         return WM_operator_filesel(C, op, event);
546 }
547
548
549 static int wm_alembic_import_exec(bContext *C, wmOperator *op)
550 {
551         if (!RNA_struct_property_is_set(op->ptr, "filepath")) {
552                 BKE_report(op->reports, RPT_ERROR, "No filename given");
553                 return OPERATOR_CANCELLED;
554         }
555
556         char filename[FILE_MAX];
557         RNA_string_get(op->ptr, "filepath", filename);
558
559         const float scale = RNA_float_get(op->ptr, "scale");
560         const bool is_sequence = RNA_boolean_get(op->ptr, "is_sequence");
561         const bool set_frame_range = RNA_boolean_get(op->ptr, "set_frame_range");
562         const bool validate_meshes = RNA_boolean_get(op->ptr, "validate_meshes");
563         const bool as_background_job = RNA_boolean_get(op->ptr, "as_background_job");
564
565         int offset = 0;
566         int sequence_len = 1;
567
568         if (is_sequence) {
569                 sequence_len = get_sequence_len(filename, &offset);
570                 if (sequence_len < 0) {
571                         BKE_report(op->reports, RPT_ERROR, "Unable to determine ABC sequence length");
572                         return OPERATOR_CANCELLED;
573                 }
574         }
575
576         /* Switch out of edit mode to avoid being stuck in it (T54326). */
577         Object *obedit = CTX_data_edit_object(C);
578         if (obedit) {
579                 ED_object_mode_toggle(C, OB_MODE_EDIT);
580         }
581
582         bool ok = ABC_import(C, filename, scale, is_sequence, set_frame_range,
583                              sequence_len, offset, validate_meshes,
584                              as_background_job);
585
586         return as_background_job || ok ? OPERATOR_FINISHED : OPERATOR_CANCELLED;
587 }
588
589 void WM_OT_alembic_import(wmOperatorType *ot)
590 {
591         ot->name = "Import Alembic";
592         ot->description = "Load an Alembic archive";
593         ot->idname = "WM_OT_alembic_import";
594
595         ot->invoke = wm_alembic_import_invoke;
596         ot->exec = wm_alembic_import_exec;
597         ot->poll = WM_operator_winactive;
598         ot->ui = wm_alembic_import_draw;
599
600         WM_operator_properties_filesel(ot, FILE_TYPE_FOLDER | FILE_TYPE_ALEMBIC,
601                                        FILE_BLENDER, FILE_SAVE, WM_FILESEL_FILEPATH,
602                                        FILE_DEFAULTDISPLAY, FILE_SORT_ALPHA);
603
604         RNA_def_float(ot->srna, "scale", 1.0f, 0.0001f, 1000.0f, "Scale",
605                       "Value by which to enlarge or shrink the objects with respect to the world's origin",
606                       0.0001f, 1000.0f);
607
608         RNA_def_boolean(ot->srna, "set_frame_range", true,
609                         "Set Frame Range",
610                         "If checked, update scene's start and end frame to match those of the Alembic archive");
611
612         RNA_def_boolean(ot->srna, "validate_meshes", 0,
613                         "Validate Meshes", "Check imported mesh objects for invalid data (slow)");
614
615         RNA_def_boolean(ot->srna, "is_sequence", false, "Is Sequence",
616                         "Set to true if the cache is split into separate files");
617
618         RNA_def_boolean(ot->srna, "as_background_job", false, "Run as Background Job",
619                         "Enable this to run the export in the background, disable to block Blender while exporting. "
620                         "This option is deprecated; EXECUTE this operator to run in the foreground, and INVOKE it "
621                         "to run as a background job");
622 }
623
624 #endif