Implement import & export support for "PBR extensions" in .mtl files (T101029, also fixes T86736). Newly supported parameters: - Roughness (Pr, map_Pr) - Metallic (Pm, map_Pm) - Sheen (Ps, map_Ps) - Clearcoat thickness (Pc) and roughness (Pcr) - Anisotropy (aniso) and rotation (anisor) - Transmittance (Tf / Kt) Exporter has an option to enable these additional PBR parameters export; defaults to off since not all software understands that. Exporter UI tweaked and all material-related options were put into their own separate box. Added/extended test files in Subversion repository for test coverage.
539 lines
20 KiB
C
539 lines
20 KiB
C
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
|
|
/** \file
|
|
* \ingroup editor/io
|
|
*/
|
|
|
|
#ifdef WITH_IO_WAVEFRONT_OBJ
|
|
|
|
# include "DNA_space_types.h"
|
|
|
|
# include "BKE_context.h"
|
|
# include "BKE_main.h"
|
|
# include "BKE_report.h"
|
|
|
|
# include "BLI_path_util.h"
|
|
# include "BLI_string.h"
|
|
# include "BLI_utildefines.h"
|
|
|
|
# include "BLT_translation.h"
|
|
|
|
# include "ED_fileselect.h"
|
|
# include "ED_outliner.h"
|
|
|
|
# include "MEM_guardedalloc.h"
|
|
|
|
# include "RNA_access.h"
|
|
# include "RNA_define.h"
|
|
|
|
# include "UI_interface.h"
|
|
# include "UI_resources.h"
|
|
|
|
# include "WM_api.h"
|
|
# include "WM_types.h"
|
|
|
|
# include "DEG_depsgraph.h"
|
|
|
|
# include "IO_orientation.h"
|
|
# include "IO_path_util_types.h"
|
|
# include "IO_wavefront_obj.h"
|
|
# include "io_obj.h"
|
|
|
|
static const EnumPropertyItem io_obj_export_evaluation_mode[] = {
|
|
{DAG_EVAL_RENDER, "DAG_EVAL_RENDER", 0, "Render", "Export objects as they appear in render"},
|
|
{DAG_EVAL_VIEWPORT,
|
|
"DAG_EVAL_VIEWPORT",
|
|
0,
|
|
"Viewport",
|
|
"Export objects as they appear in the viewport"},
|
|
{0, NULL, 0, NULL, NULL}};
|
|
|
|
static const EnumPropertyItem io_obj_path_mode[] = {
|
|
{PATH_REFERENCE_AUTO, "AUTO", 0, "Auto", "Use Relative paths with subdirectories only"},
|
|
{PATH_REFERENCE_ABSOLUTE, "ABSOLUTE", 0, "Absolute", "Always write absolute paths"},
|
|
{PATH_REFERENCE_RELATIVE, "RELATIVE", 0, "Relative", "Write relative paths where possible"},
|
|
{PATH_REFERENCE_MATCH, "MATCH", 0, "Match", "Match Absolute/Relative setting with input path"},
|
|
{PATH_REFERENCE_STRIP, "STRIP", 0, "Strip", "Write filename only"},
|
|
{PATH_REFERENCE_COPY, "COPY", 0, "Copy", "Copy the file to the destination path"},
|
|
{0, NULL, 0, NULL, NULL}};
|
|
|
|
static int wm_obj_export_invoke(bContext *C, wmOperator *op, const wmEvent *UNUSED(event))
|
|
{
|
|
ED_fileselect_ensure_default_filepath(C, op, ".obj");
|
|
|
|
WM_event_add_fileselect(C, op);
|
|
return OPERATOR_RUNNING_MODAL;
|
|
}
|
|
|
|
static int wm_obj_export_exec(bContext *C, wmOperator *op)
|
|
{
|
|
if (!RNA_struct_property_is_set_ex(op->ptr, "filepath", false)) {
|
|
BKE_report(op->reports, RPT_ERROR, "No filename given");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
struct OBJExportParams export_params;
|
|
export_params.file_base_for_tests[0] = '\0';
|
|
RNA_string_get(op->ptr, "filepath", export_params.filepath);
|
|
export_params.blen_filepath = CTX_data_main(C)->filepath;
|
|
export_params.export_animation = RNA_boolean_get(op->ptr, "export_animation");
|
|
export_params.start_frame = RNA_int_get(op->ptr, "start_frame");
|
|
export_params.end_frame = RNA_int_get(op->ptr, "end_frame");
|
|
|
|
export_params.forward_axis = RNA_enum_get(op->ptr, "forward_axis");
|
|
export_params.up_axis = RNA_enum_get(op->ptr, "up_axis");
|
|
export_params.scaling_factor = RNA_float_get(op->ptr, "scaling_factor");
|
|
export_params.apply_modifiers = RNA_boolean_get(op->ptr, "apply_modifiers");
|
|
export_params.export_eval_mode = RNA_enum_get(op->ptr, "export_eval_mode");
|
|
|
|
export_params.export_selected_objects = RNA_boolean_get(op->ptr, "export_selected_objects");
|
|
export_params.export_uv = RNA_boolean_get(op->ptr, "export_uv");
|
|
export_params.export_normals = RNA_boolean_get(op->ptr, "export_normals");
|
|
export_params.export_colors = RNA_boolean_get(op->ptr, "export_colors");
|
|
export_params.export_materials = RNA_boolean_get(op->ptr, "export_materials");
|
|
export_params.path_mode = RNA_enum_get(op->ptr, "path_mode");
|
|
export_params.export_triangulated_mesh = RNA_boolean_get(op->ptr, "export_triangulated_mesh");
|
|
export_params.export_curves_as_nurbs = RNA_boolean_get(op->ptr, "export_curves_as_nurbs");
|
|
export_params.export_pbr_extensions = RNA_boolean_get(op->ptr, "export_pbr_extensions");
|
|
|
|
export_params.export_object_groups = RNA_boolean_get(op->ptr, "export_object_groups");
|
|
export_params.export_material_groups = RNA_boolean_get(op->ptr, "export_material_groups");
|
|
export_params.export_vertex_groups = RNA_boolean_get(op->ptr, "export_vertex_groups");
|
|
export_params.export_smooth_groups = RNA_boolean_get(op->ptr, "export_smooth_groups");
|
|
export_params.smooth_groups_bitflags = RNA_boolean_get(op->ptr, "smooth_group_bitflags");
|
|
|
|
OBJ_export(C, &export_params);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
static void ui_obj_export_settings(uiLayout *layout, PointerRNA *imfptr)
|
|
{
|
|
const bool export_animation = RNA_boolean_get(imfptr, "export_animation");
|
|
const bool export_smooth_groups = RNA_boolean_get(imfptr, "export_smooth_groups");
|
|
const bool export_materials = RNA_boolean_get(imfptr, "export_materials");
|
|
|
|
uiLayoutSetPropSep(layout, true);
|
|
uiLayoutSetPropDecorate(layout, false);
|
|
|
|
uiLayout *box, *col, *sub, *row;
|
|
|
|
/* Object Transform options. */
|
|
box = uiLayoutBox(layout);
|
|
col = uiLayoutColumn(box, false);
|
|
sub = uiLayoutColumnWithHeading(col, false, IFACE_("Limit to"));
|
|
uiItemR(sub, imfptr, "export_selected_objects", 0, IFACE_("Selected Only"), ICON_NONE);
|
|
uiItemR(sub, imfptr, "scaling_factor", 0, NULL, ICON_NONE);
|
|
|
|
row = uiLayoutRow(box, false);
|
|
uiItemR(row, imfptr, "forward_axis", UI_ITEM_R_EXPAND, IFACE_("Forward Axis"), ICON_NONE);
|
|
row = uiLayoutRow(box, false);
|
|
uiItemR(row, imfptr, "up_axis", UI_ITEM_R_EXPAND, IFACE_("Up Axis"), ICON_NONE);
|
|
|
|
col = uiLayoutColumn(box, false);
|
|
sub = uiLayoutColumn(col, false);
|
|
sub = uiLayoutColumnWithHeading(col, false, IFACE_("Objects"));
|
|
uiItemR(sub, imfptr, "apply_modifiers", 0, IFACE_("Apply Modifiers"), ICON_NONE);
|
|
uiItemR(sub, imfptr, "export_eval_mode", 0, IFACE_("Properties"), ICON_NONE);
|
|
|
|
/* Geometry options. */
|
|
box = uiLayoutBox(layout);
|
|
col = uiLayoutColumn(box, false);
|
|
sub = uiLayoutColumnWithHeading(col, false, IFACE_("Geometry"));
|
|
uiItemR(sub, imfptr, "export_uv", 0, IFACE_("UV Coordinates"), ICON_NONE);
|
|
uiItemR(sub, imfptr, "export_normals", 0, IFACE_("Normals"), ICON_NONE);
|
|
uiItemR(sub, imfptr, "export_colors", 0, IFACE_("Colors"), ICON_NONE);
|
|
uiItemR(sub, imfptr, "export_triangulated_mesh", 0, IFACE_("Triangulated Mesh"), ICON_NONE);
|
|
uiItemR(sub, imfptr, "export_curves_as_nurbs", 0, IFACE_("Curves as NURBS"), ICON_NONE);
|
|
|
|
/* Material options. */
|
|
box = uiLayoutBox(layout);
|
|
col = uiLayoutColumn(box, false);
|
|
sub = uiLayoutColumnWithHeading(col, false, IFACE_("Materials"));
|
|
uiItemR(sub, imfptr, "export_materials", 0, IFACE_("Export"), ICON_NONE);
|
|
sub = uiLayoutColumn(sub, false);
|
|
uiLayoutSetEnabled(sub, export_materials);
|
|
uiItemR(sub, imfptr, "export_pbr_extensions", 0, IFACE_("PBR Extensions"), ICON_NONE);
|
|
uiItemR(sub, imfptr, "path_mode", 0, IFACE_("Path Mode"), ICON_NONE);
|
|
|
|
/* Grouping options. */
|
|
box = uiLayoutBox(layout);
|
|
col = uiLayoutColumn(box, false);
|
|
sub = uiLayoutColumnWithHeading(col, false, IFACE_("Grouping"));
|
|
uiItemR(sub, imfptr, "export_object_groups", 0, IFACE_("Object Groups"), ICON_NONE);
|
|
uiItemR(sub, imfptr, "export_material_groups", 0, IFACE_("Material Groups"), ICON_NONE);
|
|
uiItemR(sub, imfptr, "export_vertex_groups", 0, IFACE_("Vertex Groups"), ICON_NONE);
|
|
uiItemR(sub, imfptr, "export_smooth_groups", 0, IFACE_("Smooth Groups"), ICON_NONE);
|
|
sub = uiLayoutColumn(sub, false);
|
|
uiLayoutSetEnabled(sub, export_smooth_groups);
|
|
uiItemR(sub, imfptr, "smooth_group_bitflags", 0, IFACE_("Smooth Group Bitflags"), ICON_NONE);
|
|
|
|
/* Animation options. */
|
|
box = uiLayoutBox(layout);
|
|
col = uiLayoutColumn(box, false);
|
|
sub = uiLayoutColumnWithHeading(col, false, IFACE_("Animation"));
|
|
uiItemR(sub, imfptr, "export_animation", 0, IFACE_("Export"), ICON_NONE);
|
|
sub = uiLayoutColumn(sub, true);
|
|
uiLayoutSetEnabled(sub, export_animation);
|
|
uiItemR(sub, imfptr, "start_frame", 0, IFACE_("Frame Start"), ICON_NONE);
|
|
uiItemR(sub, imfptr, "end_frame", 0, IFACE_("End"), ICON_NONE);
|
|
}
|
|
|
|
static void wm_obj_export_draw(bContext *UNUSED(C), wmOperator *op)
|
|
{
|
|
PointerRNA ptr;
|
|
RNA_pointer_create(NULL, op->type->srna, op->properties, &ptr);
|
|
ui_obj_export_settings(op->layout, &ptr);
|
|
}
|
|
|
|
/**
|
|
* Return true if any property in the UI is changed.
|
|
*/
|
|
static bool wm_obj_export_check(bContext *C, wmOperator *op)
|
|
{
|
|
char filepath[FILE_MAX];
|
|
Scene *scene = CTX_data_scene(C);
|
|
bool changed = false;
|
|
RNA_string_get(op->ptr, "filepath", filepath);
|
|
|
|
if (!BLI_path_extension_check(filepath, ".obj")) {
|
|
BLI_path_extension_ensure(filepath, FILE_MAX, ".obj");
|
|
RNA_string_set(op->ptr, "filepath", filepath);
|
|
changed = true;
|
|
}
|
|
|
|
{
|
|
int start = RNA_int_get(op->ptr, "start_frame");
|
|
int end = RNA_int_get(op->ptr, "end_frame");
|
|
/* Set the defaults. */
|
|
if (start == INT_MIN) {
|
|
start = scene->r.sfra;
|
|
changed = true;
|
|
}
|
|
if (end == INT_MAX) {
|
|
end = scene->r.efra;
|
|
changed = true;
|
|
}
|
|
/* Fix user errors. */
|
|
if (end < start) {
|
|
end = start;
|
|
changed = true;
|
|
}
|
|
RNA_int_set(op->ptr, "start_frame", start);
|
|
RNA_int_set(op->ptr, "end_frame", end);
|
|
}
|
|
return changed;
|
|
}
|
|
|
|
/* Both forward and up axes cannot be along the same direction. */
|
|
static void forward_axis_update(struct Main *UNUSED(main),
|
|
struct Scene *UNUSED(scene),
|
|
struct PointerRNA *ptr)
|
|
{
|
|
int forward = RNA_enum_get(ptr, "forward_axis");
|
|
int up = RNA_enum_get(ptr, "up_axis");
|
|
if ((forward % 3) == (up % 3)) {
|
|
RNA_enum_set(ptr, "up_axis", (up + 1) % 6);
|
|
}
|
|
}
|
|
|
|
static void up_axis_update(struct Main *UNUSED(main),
|
|
struct Scene *UNUSED(scene),
|
|
struct PointerRNA *ptr)
|
|
{
|
|
int forward = RNA_enum_get(ptr, "forward_axis");
|
|
int up = RNA_enum_get(ptr, "up_axis");
|
|
if ((forward % 3) == (up % 3)) {
|
|
RNA_enum_set(ptr, "forward_axis", (forward + 1) % 6);
|
|
}
|
|
}
|
|
|
|
void WM_OT_obj_export(struct wmOperatorType *ot)
|
|
{
|
|
PropertyRNA *prop;
|
|
|
|
ot->name = "Export Wavefront OBJ";
|
|
ot->description = "Save the scene to a Wavefront OBJ file";
|
|
ot->idname = "WM_OT_obj_export";
|
|
|
|
ot->invoke = wm_obj_export_invoke;
|
|
ot->exec = wm_obj_export_exec;
|
|
ot->poll = WM_operator_winactive;
|
|
ot->ui = wm_obj_export_draw;
|
|
ot->check = wm_obj_export_check;
|
|
|
|
ot->flag = OPTYPE_PRESET;
|
|
|
|
WM_operator_properties_filesel(ot,
|
|
FILE_TYPE_FOLDER,
|
|
FILE_BLENDER,
|
|
FILE_SAVE,
|
|
WM_FILESEL_FILEPATH | WM_FILESEL_SHOW_PROPS,
|
|
FILE_DEFAULTDISPLAY,
|
|
FILE_SORT_DEFAULT);
|
|
|
|
/* Animation options. */
|
|
RNA_def_boolean(ot->srna,
|
|
"export_animation",
|
|
false,
|
|
"Export Animation",
|
|
"Export multiple frames instead of the current frame only");
|
|
RNA_def_int(ot->srna,
|
|
"start_frame",
|
|
INT_MIN, /* wm_obj_export_check uses this to set scene->r.sfra. */
|
|
INT_MIN,
|
|
INT_MAX,
|
|
"Start Frame",
|
|
"The first frame to be exported",
|
|
INT_MIN,
|
|
INT_MAX);
|
|
RNA_def_int(ot->srna,
|
|
"end_frame",
|
|
INT_MAX, /* wm_obj_export_check uses this to set scene->r.efra. */
|
|
INT_MIN,
|
|
INT_MAX,
|
|
"End Frame",
|
|
"The last frame to be exported",
|
|
INT_MIN,
|
|
INT_MAX);
|
|
/* Object transform options. */
|
|
prop = RNA_def_enum(
|
|
ot->srna, "forward_axis", io_transform_axis, IO_AXIS_NEGATIVE_Z, "Forward Axis", "");
|
|
RNA_def_property_update_runtime(prop, (void *)forward_axis_update);
|
|
prop = RNA_def_enum(ot->srna, "up_axis", io_transform_axis, IO_AXIS_Y, "Up Axis", "");
|
|
RNA_def_property_update_runtime(prop, (void *)up_axis_update);
|
|
RNA_def_float(ot->srna,
|
|
"scaling_factor",
|
|
1.0f,
|
|
0.001f,
|
|
10000.0f,
|
|
"Scale",
|
|
"Upscale the object by this factor",
|
|
0.01,
|
|
1000.0f);
|
|
/* File Writer options. */
|
|
RNA_def_boolean(
|
|
ot->srna, "apply_modifiers", true, "Apply Modifiers", "Apply modifiers to exported meshes");
|
|
RNA_def_enum(ot->srna,
|
|
"export_eval_mode",
|
|
io_obj_export_evaluation_mode,
|
|
DAG_EVAL_VIEWPORT,
|
|
"Object Properties",
|
|
"Determines properties like object visibility, modifiers etc., where they differ "
|
|
"for Render and Viewport");
|
|
RNA_def_boolean(ot->srna,
|
|
"export_selected_objects",
|
|
false,
|
|
"Export Selected Objects",
|
|
"Export only selected objects instead of all supported objects");
|
|
RNA_def_boolean(ot->srna, "export_uv", true, "Export UVs", "");
|
|
RNA_def_boolean(ot->srna,
|
|
"export_normals",
|
|
true,
|
|
"Export Normals",
|
|
"Export per-face normals if the face is flat-shaded, per-face-per-loop "
|
|
"normals if smooth-shaded");
|
|
RNA_def_boolean(ot->srna, "export_colors", false, "Export Colors", "Export per-vertex colors");
|
|
RNA_def_boolean(ot->srna,
|
|
"export_materials",
|
|
true,
|
|
"Export Materials",
|
|
"Export MTL library. There must be a Principled-BSDF node for image textures to "
|
|
"be exported to the MTL file");
|
|
RNA_def_boolean(ot->srna,
|
|
"export_pbr_extensions",
|
|
false,
|
|
"Export Materials with PBR Extensions",
|
|
"Export MTL library using PBR extensions (roughness, metallic, sheen, "
|
|
"clearcoat, anisotropy, transmission)");
|
|
RNA_def_enum(ot->srna,
|
|
"path_mode",
|
|
io_obj_path_mode,
|
|
PATH_REFERENCE_AUTO,
|
|
"Path Mode",
|
|
"Method used to reference paths");
|
|
RNA_def_boolean(ot->srna,
|
|
"export_triangulated_mesh",
|
|
false,
|
|
"Export Triangulated Mesh",
|
|
"All ngons with four or more vertices will be triangulated. Meshes in "
|
|
"the scene will not be affected. Behaves like Triangulate Modifier with "
|
|
"ngon-method: \"Beauty\", quad-method: \"Shortest Diagonal\", min vertices: 4");
|
|
RNA_def_boolean(ot->srna,
|
|
"export_curves_as_nurbs",
|
|
false,
|
|
"Export Curves as NURBS",
|
|
"Export curves in parametric form instead of exporting as mesh");
|
|
|
|
RNA_def_boolean(ot->srna,
|
|
"export_object_groups",
|
|
false,
|
|
"Export Object Groups",
|
|
"Append mesh name to object name, separated by a '_'");
|
|
RNA_def_boolean(ot->srna,
|
|
"export_material_groups",
|
|
false,
|
|
"Export Material Groups",
|
|
"Generate an OBJ group for each part of a geometry using a different material");
|
|
RNA_def_boolean(
|
|
ot->srna,
|
|
"export_vertex_groups",
|
|
false,
|
|
"Export Vertex Groups",
|
|
"Export the name of the vertex group of a face. It is approximated "
|
|
"by choosing the vertex group with the most members among the vertices of a face");
|
|
RNA_def_boolean(
|
|
ot->srna,
|
|
"export_smooth_groups",
|
|
false,
|
|
"Export Smooth Groups",
|
|
"Every smooth-shaded face is assigned group \"1\" and every flat-shaded face \"off\"");
|
|
RNA_def_boolean(
|
|
ot->srna, "smooth_group_bitflags", false, "Generate Bitflags for Smooth Groups", "");
|
|
|
|
/* Only show .obj or .mtl files by default. */
|
|
prop = RNA_def_string(ot->srna, "filter_glob", "*.obj;*.mtl", 0, "Extension Filter", "");
|
|
RNA_def_property_flag(prop, PROP_HIDDEN);
|
|
}
|
|
|
|
static int wm_obj_import_invoke(bContext *C, wmOperator *op, const wmEvent *UNUSED(event))
|
|
{
|
|
WM_event_add_fileselect(C, op);
|
|
return OPERATOR_RUNNING_MODAL;
|
|
}
|
|
|
|
static int wm_obj_import_exec(bContext *C, wmOperator *op)
|
|
{
|
|
struct OBJImportParams import_params;
|
|
RNA_string_get(op->ptr, "filepath", import_params.filepath);
|
|
import_params.clamp_size = RNA_float_get(op->ptr, "clamp_size");
|
|
import_params.forward_axis = RNA_enum_get(op->ptr, "forward_axis");
|
|
import_params.up_axis = RNA_enum_get(op->ptr, "up_axis");
|
|
import_params.import_vertex_groups = RNA_boolean_get(op->ptr, "import_vertex_groups");
|
|
import_params.validate_meshes = RNA_boolean_get(op->ptr, "validate_meshes");
|
|
import_params.relative_paths = ((U.flag & USER_RELPATHS) != 0);
|
|
import_params.clear_selection = true;
|
|
|
|
int files_len = RNA_collection_length(op->ptr, "files");
|
|
if (files_len) {
|
|
/* Importing multiple files: loop over them and import one by one. */
|
|
PointerRNA fileptr;
|
|
PropertyRNA *prop;
|
|
char dir_only[FILE_MAX], file_only[FILE_MAX];
|
|
|
|
RNA_string_get(op->ptr, "directory", dir_only);
|
|
prop = RNA_struct_find_property(op->ptr, "files");
|
|
for (int i = 0; i < files_len; i++) {
|
|
RNA_property_collection_lookup_int(op->ptr, prop, i, &fileptr);
|
|
RNA_string_get(&fileptr, "name", file_only);
|
|
BLI_join_dirfile(
|
|
import_params.filepath, sizeof(import_params.filepath), dir_only, file_only);
|
|
import_params.clear_selection = (i == 0);
|
|
OBJ_import(C, &import_params);
|
|
}
|
|
}
|
|
else if (RNA_struct_property_is_set_ex(op->ptr, "filepath", false)) {
|
|
/* Importing one file. */
|
|
RNA_string_get(op->ptr, "filepath", import_params.filepath);
|
|
OBJ_import(C, &import_params);
|
|
}
|
|
else {
|
|
BKE_report(op->reports, RPT_ERROR, "No filename given");
|
|
return OPERATOR_CANCELLED;
|
|
}
|
|
|
|
Scene *scene = CTX_data_scene(C);
|
|
WM_event_add_notifier(C, NC_SCENE | ND_OB_SELECT, scene);
|
|
WM_event_add_notifier(C, NC_SCENE | ND_OB_ACTIVE, scene);
|
|
WM_event_add_notifier(C, NC_SCENE | ND_LAYER_CONTENT, scene);
|
|
ED_outliner_select_sync_from_object_tag(C);
|
|
|
|
return OPERATOR_FINISHED;
|
|
}
|
|
|
|
static void ui_obj_import_settings(uiLayout *layout, PointerRNA *imfptr)
|
|
{
|
|
uiLayoutSetPropSep(layout, true);
|
|
uiLayoutSetPropDecorate(layout, false);
|
|
|
|
uiLayout *box = uiLayoutBox(layout);
|
|
uiItemL(box, IFACE_("Transform"), ICON_OBJECT_DATA);
|
|
uiLayout *col = uiLayoutColumn(box, false);
|
|
uiLayout *sub = uiLayoutColumn(col, false);
|
|
uiItemR(sub, imfptr, "clamp_size", 0, NULL, ICON_NONE);
|
|
sub = uiLayoutColumn(col, false);
|
|
|
|
uiLayout *row = uiLayoutRow(box, false);
|
|
uiItemR(row, imfptr, "forward_axis", UI_ITEM_R_EXPAND, IFACE_("Forward Axis"), ICON_NONE);
|
|
row = uiLayoutRow(box, false);
|
|
uiItemR(row, imfptr, "up_axis", UI_ITEM_R_EXPAND, IFACE_("Up Axis"), ICON_NONE);
|
|
|
|
box = uiLayoutBox(layout);
|
|
uiItemL(box, IFACE_("Options"), ICON_EXPORT);
|
|
col = uiLayoutColumn(box, false);
|
|
uiItemR(col, imfptr, "import_vertex_groups", 0, NULL, ICON_NONE);
|
|
uiItemR(col, imfptr, "validate_meshes", 0, NULL, ICON_NONE);
|
|
}
|
|
|
|
static void wm_obj_import_draw(bContext *C, wmOperator *op)
|
|
{
|
|
PointerRNA ptr;
|
|
wmWindowManager *wm = CTX_wm_manager(C);
|
|
RNA_pointer_create(&wm->id, op->type->srna, op->properties, &ptr);
|
|
ui_obj_import_settings(op->layout, &ptr);
|
|
}
|
|
|
|
void WM_OT_obj_import(struct wmOperatorType *ot)
|
|
{
|
|
PropertyRNA *prop;
|
|
|
|
ot->name = "Import Wavefront OBJ";
|
|
ot->description = "Load a Wavefront OBJ scene";
|
|
ot->idname = "WM_OT_obj_import";
|
|
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
|
|
|
|
ot->invoke = wm_obj_import_invoke;
|
|
ot->exec = wm_obj_import_exec;
|
|
ot->poll = WM_operator_winactive;
|
|
ot->ui = wm_obj_import_draw;
|
|
|
|
WM_operator_properties_filesel(ot,
|
|
FILE_TYPE_FOLDER,
|
|
FILE_BLENDER,
|
|
FILE_OPENFILE,
|
|
WM_FILESEL_FILEPATH | WM_FILESEL_SHOW_PROPS |
|
|
WM_FILESEL_DIRECTORY | WM_FILESEL_FILES,
|
|
FILE_DEFAULTDISPLAY,
|
|
FILE_SORT_DEFAULT);
|
|
RNA_def_float(
|
|
ot->srna,
|
|
"clamp_size",
|
|
0.0f,
|
|
0.0f,
|
|
1000.0f,
|
|
"Clamp Bounding Box",
|
|
"Resize the objects to keep bounding box under this value. Value 0 disables clamping",
|
|
0.0f,
|
|
1000.0f);
|
|
prop = RNA_def_enum(
|
|
ot->srna, "forward_axis", io_transform_axis, IO_AXIS_NEGATIVE_Z, "Forward Axis", "");
|
|
RNA_def_property_update_runtime(prop, (void *)forward_axis_update);
|
|
prop = RNA_def_enum(ot->srna, "up_axis", io_transform_axis, IO_AXIS_Y, "Up Axis", "");
|
|
RNA_def_property_update_runtime(prop, (void *)up_axis_update);
|
|
RNA_def_boolean(ot->srna,
|
|
"import_vertex_groups",
|
|
false,
|
|
"Vertex Groups",
|
|
"Import OBJ groups as vertex groups");
|
|
RNA_def_boolean(ot->srna,
|
|
"validate_meshes",
|
|
false,
|
|
"Validate Meshes",
|
|
"Check imported mesh objects for invalid data (slow)");
|
|
|
|
/* Only show .obj or .mtl files by default. */
|
|
prop = RNA_def_string(ot->srna, "filter_glob", "*.obj;*.mtl", 0, "Extension Filter", "");
|
|
RNA_def_property_flag(prop, PROP_HIDDEN);
|
|
}
|
|
|
|
#endif /* WITH_IO_WAVEFRONT_OBJ */
|