diff --git a/xs/CMakeLists.txt b/xs/CMakeLists.txt
index b7d99ed696..1f2a93375e 100644
--- a/xs/CMakeLists.txt
+++ b/xs/CMakeLists.txt
@@ -177,6 +177,7 @@ add_library(libslic3r STATIC
${LIBDIR}/libslic3r/SurfaceCollection.hpp
${LIBDIR}/libslic3r/SVG.cpp
${LIBDIR}/libslic3r/SVG.hpp
+ ${LIBDIR}/libslic3r/Technologies.hpp
${LIBDIR}/libslic3r/TriangleMesh.cpp
${LIBDIR}/libslic3r/TriangleMesh.hpp
${LIBDIR}/libslic3r/SLABasePool.hpp
diff --git a/xs/src/libslic3r/Format/3mf.cpp b/xs/src/libslic3r/Format/3mf.cpp
index 464cbb8e67..ddd3de69e5 100644
--- a/xs/src/libslic3r/Format/3mf.cpp
+++ b/xs/src/libslic3r/Format/3mf.cpp
@@ -1252,9 +1252,17 @@ namespace Slic3r {
// we extract from the given matrix only the values currently used
// translation
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ Vec3d offset(transform(0, 3), transform(1, 3), transform(2, 3));
+#else
+//################################################################################################################################
double offset_x = transform(0, 3);
double offset_y = transform(1, 3);
double offset_z = transform(2, 3);
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
// scale
double sx = ::sqrt(sqr(transform(0, 0)) + sqr(transform(1, 0)) + sqr(transform(2, 0)));
@@ -1287,8 +1295,16 @@ namespace Slic3r {
double angle_z = (rotation.axis() == Vec3d::UnitZ()) ? rotation.angle() : -rotation.angle();
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ instance.set_offset(offset);
+#else
+//################################################################################################################################
instance.offset(0) = offset_x;
instance.offset(1) = offset_y;
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
instance.scaling_factor = sx;
instance.rotation = angle_z;
}
diff --git a/xs/src/libslic3r/Format/AMF.cpp b/xs/src/libslic3r/Format/AMF.cpp
index 81617ad729..b783f4973f 100644
--- a/xs/src/libslic3r/Format/AMF.cpp
+++ b/xs/src/libslic3r/Format/AMF.cpp
@@ -29,7 +29,16 @@
// VERSION NUMBERS
// 0 : .amf, .amf.xml and .zip.amf files saved by older slic3r. No version definition in them.
// 1 : Introduction of amf versioning. No other change in data saved into amf files.
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+// 2 : Added z component of offset.
+const unsigned int VERSION_AMF = 2;
+#else
+//################################################################################################################################
const unsigned int VERSION_AMF = 1;
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
const char* SLIC3RPE_AMF_VERSION = "slic3rpe_amf_version";
const char* SLIC3R_CONFIG_TYPE = "slic3rpe_config";
@@ -119,19 +128,39 @@ struct AMFParserContext
NODE_TYPE_INSTANCE, // amf/constellation/instance
NODE_TYPE_DELTAX, // amf/constellation/instance/deltax
NODE_TYPE_DELTAY, // amf/constellation/instance/deltay
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ NODE_TYPE_DELTAZ, // amf/constellation/instance/deltaz
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
NODE_TYPE_RZ, // amf/constellation/instance/rz
NODE_TYPE_SCALE, // amf/constellation/instance/scale
NODE_TYPE_METADATA, // anywhere under amf/*/metadata
};
struct Instance {
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ Instance() : deltax_set(false), deltay_set(false), deltaz_set(false), rz_set(false), scale_set(false) {}
+#else
+//################################################################################################################################
Instance() : deltax_set(false), deltay_set(false), rz_set(false), scale_set(false) {}
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
// Shift in the X axis.
float deltax;
bool deltax_set;
// Shift in the Y axis.
float deltay;
bool deltay_set;
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ // Shift in the Z axis.
+ float deltaz;
+ bool deltaz_set;
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
// Rotation around the Z axis.
float rz;
bool rz_set;
@@ -254,6 +283,12 @@ void AMFParserContext::startElement(const char *name, const char **atts)
node_type_new = NODE_TYPE_DELTAX;
else if (strcmp(name, "deltay") == 0)
node_type_new = NODE_TYPE_DELTAY;
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ else if (strcmp(name, "deltaz") == 0)
+ node_type_new = NODE_TYPE_DELTAZ;
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
else if (strcmp(name, "rz") == 0)
node_type_new = NODE_TYPE_RZ;
else if (strcmp(name, "scale") == 0)
@@ -314,7 +349,19 @@ void AMFParserContext::characters(const XML_Char *s, int len)
{
switch (m_path.size()) {
case 4:
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ if (m_path.back() == NODE_TYPE_DELTAX ||
+ m_path.back() == NODE_TYPE_DELTAY ||
+ m_path.back() == NODE_TYPE_DELTAZ ||
+ m_path.back() == NODE_TYPE_RZ ||
+ m_path.back() == NODE_TYPE_SCALE)
+#else
+//################################################################################################################################
if (m_path.back() == NODE_TYPE_DELTAX || m_path.back() == NODE_TYPE_DELTAY || m_path.back() == NODE_TYPE_RZ || m_path.back() == NODE_TYPE_SCALE)
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
m_value[0].append(s, len);
break;
case 6:
@@ -354,6 +401,16 @@ void AMFParserContext::endElement(const char * /* name */)
m_instance->deltay_set = true;
m_value[0].clear();
break;
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ case NODE_TYPE_DELTAZ:
+ assert(m_instance);
+ m_instance->deltaz = float(atof(m_value[0].c_str()));
+ m_instance->deltaz_set = true;
+ m_value[0].clear();
+ break;
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
case NODE_TYPE_RZ:
assert(m_instance);
m_instance->rz = float(atof(m_value[0].c_str()));
@@ -498,8 +555,16 @@ void AMFParserContext::endDocument()
for (const Instance &instance : object.second.instances)
if (instance.deltax_set && instance.deltay_set) {
ModelInstance *mi = m_model.objects[object.second.idx]->add_instance();
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ mi->set_offset(Vec3d((double)instance.deltax, (double)instance.deltay, (double)instance.deltaz));
+#else
+//################################################################################################################################
mi->offset(0) = instance.deltax;
mi->offset(1) = instance.deltay;
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
mi->rotation = instance.rz_set ? instance.rz : 0.f;
mi->scaling_factor = instance.scale_set ? instance.scale : 1.f;
}
@@ -800,12 +865,27 @@ bool store_amf(const char *path, Model *model, Print* print, bool export_print_c
" \n"
" %lf\n"
" %lf\n"
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ " %lf\n"
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
" %lf\n"
" %lf\n"
" \n",
object_id,
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ instance->get_offset(X),
+ instance->get_offset(Y),
+ instance->get_offset(Z),
+#else
+//################################################################################################################################
instance->offset(0),
instance->offset(1),
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
instance->rotation,
instance->scaling_factor);
//FIXME missing instance->scaling_factor
diff --git a/xs/src/libslic3r/Format/PRUS.cpp b/xs/src/libslic3r/Format/PRUS.cpp
index 3cf3fc075d..0f3807c7a9 100644
--- a/xs/src/libslic3r/Format/PRUS.cpp
+++ b/xs/src/libslic3r/Format/PRUS.cpp
@@ -166,7 +166,15 @@ bool load_prus(const char *path, Model *model)
float trafo[3][4] = { 0 };
double instance_rotation = 0.;
double instance_scaling_factor = 1.f;
- Vec2d instance_offset(0., 0.);
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ Vec3d instance_offset = Vec3d::Zero();
+#else
+//################################################################################################################################
+ Vec2d instance_offset(0., 0.);
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
bool trafo_set = false;
unsigned int group_id = (unsigned int)-1;
unsigned int extruder_id = (unsigned int)-1;
@@ -207,8 +215,16 @@ bool load_prus(const char *path, Model *model)
for (size_t c = 0; c < 3; ++ c)
trafo[r][c] += mat_trafo(r, c);
}
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ instance_offset = Vec3d((double)(position[0] - zero[0]), (double)(position[1] - zero[1]), (double)(position[2] - zero[2]));
+#else
+//################################################################################################################################
instance_offset(0) = position[0] - zero[0];
instance_offset(1) = position[1] - zero[1];
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
trafo[2][3] = position[2] / instance_scaling_factor;
trafo_set = true;
}
@@ -360,8 +376,16 @@ bool load_prus(const char *path, Model *model)
ModelInstance *instance = model_object->add_instance();
instance->rotation = instance_rotation;
instance->scaling_factor = instance_scaling_factor;
- instance->offset = instance_offset;
- ++ num_models;
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ instance->set_offset(instance_offset);
+#else
+//################################################################################################################################
+ instance->offset = instance_offset;
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
+ ++num_models;
if (group_id != (size_t)-1)
group_to_model_object[group_id] = model_object;
} else {
diff --git a/xs/src/libslic3r/Model.cpp b/xs/src/libslic3r/Model.cpp
index 19c474cad9..f7d9df79b5 100644
--- a/xs/src/libslic3r/Model.cpp
+++ b/xs/src/libslic3r/Model.cpp
@@ -242,10 +242,27 @@ void Model::center_instances_around_point(const Vec2d &point)
for (size_t i = 0; i < o->instances.size(); ++ i)
bb.merge(o->instance_bounding_box(i, false));
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ Vec2d shift2 = point - to_2d(bb.center());
+ Vec3d shift3 = Vec3d(shift2(0), shift2(1), 0.0);
+#else
+//################################################################################################################################
Vec2d shift = point - 0.5 * to_2d(bb.size()) - to_2d(bb.min);
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
for (ModelObject *o : this->objects) {
for (ModelInstance *i : o->instances)
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ i->set_offset(i->get_offset() + shift3);
+#else
+//################################################################################################################################
i->offset += shift;
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
o->invalidate_bounding_box();
}
}
@@ -311,8 +328,17 @@ bool Model::arrange_objects(coordf_t dist, const BoundingBoxf* bb)
size_t idx = 0;
for (ModelObject *o : this->objects) {
for (ModelInstance *i : o->instances) {
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ Vec2d offset_xy = positions[idx] - instance_centers[idx];
+ i->set_offset(Vec3d(offset_xy(0), offset_xy(1), i->get_offset(Z)));
+#else
+//################################################################################################################################
i->offset = positions[idx] - instance_centers[idx];
- ++ idx;
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
+ ++idx;
}
o->invalidate_bounding_box();
}
@@ -336,7 +362,15 @@ void Model::duplicate(size_t copies_num, coordf_t dist, const BoundingBoxf* bb)
for (const ModelInstance *i : instances) {
for (const Vec2d &pos : positions) {
ModelInstance *instance = o->add_instance(*i);
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ instance->set_offset(instance->get_offset() + Vec3d(pos(0), pos(1), 0.0));
+#else
+//################################################################################################################################
instance->offset += pos;
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
}
}
o->invalidate_bounding_box();
@@ -366,13 +400,29 @@ void Model::duplicate_objects_grid(size_t x, size_t y, coordf_t dist)
ModelObject* object = this->objects.front();
object->clear_instances();
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ Vec3d ext_size = object->bounding_box().size() + dist * Vec3d::Ones();
+#else
+//################################################################################################################################
Vec3d size = object->bounding_box().size();
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
for (size_t x_copy = 1; x_copy <= x; ++x_copy) {
for (size_t y_copy = 1; y_copy <= y; ++y_copy) {
ModelInstance* instance = object->add_instance();
- instance->offset(0) = (size(0) + dist) * (x_copy-1);
- instance->offset(1) = (size(1) + dist) * (y_copy-1);
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ instance->set_offset(Vec3d(ext_size(0) * (double)(x_copy - 1), ext_size(1) * (double)(y_copy - 1), 0.0));
+#else
+//################################################################################################################################
+ instance->offset(0) = (size(0) + dist) * (x_copy - 1);
+ instance->offset(1) = (size(1) + dist) * (y_copy - 1);
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
}
}
}
@@ -676,12 +726,27 @@ void ModelObject::center_around_origin()
this->translate(shift);
this->origin_translation += shift;
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ // set z to zero, translation in z has already been done within the mesh
+ shift(2) = 0.0;
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
+
if (!this->instances.empty()) {
for (ModelInstance *i : this->instances) {
// apply rotation and scaling to vector as well before translating instance,
// in order to leave final position unaltered
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ i->set_offset(i->get_offset() + i->transform_vector(-shift, true));
+#else
+//################################################################################################################################
Vec3d i_shift = i->world_matrix(true) * shift;
i->offset -= to_2d(i_shift);
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
}
this->invalidate_bounding_box();
}
@@ -1004,8 +1069,17 @@ BoundingBoxf3 ModelInstance::transform_mesh_bounding_box(const TriangleMesh* mes
}
// Translate the bounding box.
if (! dont_translate) {
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ bbox.min += this->m_offset;
+ bbox.max += this->m_offset;
+#else
+//################################################################################################################################
Eigen::Map(bbox.min.data()) += this->offset;
Eigen::Map(bbox.max.data()) += this->offset;
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
}
}
return bbox;
@@ -1032,7 +1106,15 @@ Transform3d ModelInstance::world_matrix(bool dont_translate, bool dont_rotate, b
Transform3d m = Transform3d::Identity();
if (!dont_translate)
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ m.translate(m_offset);
+#else
+//################################################################################################################################
m.translate(Vec3d(offset(0), offset(1), 0.0));
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
if (!dont_rotate)
m.rotate(Eigen::AngleAxisd(rotation, Vec3d::UnitZ()));
diff --git a/xs/src/libslic3r/Model.hpp b/xs/src/libslic3r/Model.hpp
index 8a8af481c5..39633162de 100644
--- a/xs/src/libslic3r/Model.hpp
+++ b/xs/src/libslic3r/Model.hpp
@@ -224,15 +224,42 @@ public:
friend class ModelObject;
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+private:
+ Vec3d m_offset; // in unscaled coordinates
+
+public:
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
+
double rotation; // Rotation around the Z axis, in radians around mesh center point
double scaling_factor;
+//################################################################################################################################
+#if !ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
Vec2d offset; // in unscaled coordinates
-
+//################################################################################################################################
+#endif // !ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
+
// flag showing the position of this instance with respect to the print volume (set by Print::validate() using ModelObject::check_instances_print_volume_state())
EPrintVolumeState print_volume_state;
ModelObject* get_object() const { return this->object; }
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ const Vec3d& get_offset() const { return m_offset; }
+ double get_offset(Axis axis) const { return m_offset(axis); }
+
+ void set_offset(const Vec3d& offset) { m_offset = offset; }
+ void set_offset(Axis axis, double offset) { m_offset(axis) = offset; }
+
+// void add_to_offset(const Vec3d& shift) { m_offset += shift; }
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
+
// To be called on an external mesh
void transform_mesh(TriangleMesh* mesh, bool dont_translate = false) const;
// Calculate a bounding box of a transformed mesh. To be called on an external mesh.
@@ -252,9 +279,19 @@ private:
// Parent object, owning this instance.
ModelObject* object;
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ ModelInstance(ModelObject *object) : rotation(0), scaling_factor(1), m_offset(Vec3d::Zero()), object(object), print_volume_state(PVS_Inside) {}
+ ModelInstance(ModelObject *object, const ModelInstance &other) :
+ rotation(other.rotation), scaling_factor(other.scaling_factor), m_offset(other.m_offset), object(object), print_volume_state(PVS_Inside) {}
+#else
+//################################################################################################################################
ModelInstance(ModelObject *object) : rotation(0), scaling_factor(1), offset(Vec2d::Zero()), object(object), print_volume_state(PVS_Inside) {}
ModelInstance(ModelObject *object, const ModelInstance &other) :
rotation(other.rotation), scaling_factor(other.scaling_factor), offset(other.offset), object(object), print_volume_state(PVS_Inside) {}
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
};
diff --git a/xs/src/libslic3r/ModelArrange.hpp b/xs/src/libslic3r/ModelArrange.hpp
index a8e7936d61..68d8427229 100644
--- a/xs/src/libslic3r/ModelArrange.hpp
+++ b/xs/src/libslic3r/ModelArrange.hpp
@@ -517,8 +517,17 @@ ShapeData2D projectModelFromTop(const Slic3r::Model &model) {
if(item.vertexCount() > 3) {
item.rotation(objinst->rotation);
item.translation( {
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ ClipperLib::cInt(objinst->get_offset(X) / SCALING_FACTOR),
+ ClipperLib::cInt(objinst->get_offset(Y) / SCALING_FACTOR)
+#else
+//################################################################################################################################
ClipperLib::cInt(objinst->offset(0)/SCALING_FACTOR),
ClipperLib::cInt(objinst->offset(1)/SCALING_FACTOR)
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
});
ret.emplace_back(objinst, item);
}
@@ -655,11 +664,27 @@ void applyResult(
// appropriately
auto off = item.translation();
Radians rot = item.rotation();
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ Vec3d foff(off.X*SCALING_FACTOR + batch_offset, off.Y*SCALING_FACTOR, 0.0);
+#else
+//################################################################################################################################
Vec2d foff(off.X*SCALING_FACTOR + batch_offset, off.Y*SCALING_FACTOR);
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
// write the tranformation data into the model instance
inst_ptr->rotation = rot;
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ inst_ptr->set_offset(foff);
+#else
+//################################################################################################################################
inst_ptr->offset = foff;
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
}
}
diff --git a/xs/src/libslic3r/PrintObject.cpp b/xs/src/libslic3r/PrintObject.cpp
index d03b994168..fac653d08c 100644
--- a/xs/src/libslic3r/PrintObject.cpp
+++ b/xs/src/libslic3r/PrintObject.cpp
@@ -102,8 +102,20 @@ bool PrintObject::reload_model_instances()
copies.reserve(this->_model_object->instances.size());
for (const ModelInstance *mi : this->_model_object->instances)
{
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ if (mi->is_printable())
+ {
+ const Vec3d& offset = mi->get_offset();
+ copies.emplace_back(Point::new_scale(offset(0), offset(1)));
+ }
+#else
+//################################################################################################################################
if (mi->is_printable())
copies.emplace_back(Point::new_scale(mi->offset(0), mi->offset(1)));
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
}
return this->set_copies(copies);
}
diff --git a/xs/src/libslic3r/Technologies.hpp b/xs/src/libslic3r/Technologies.hpp
new file mode 100644
index 0000000000..5c4f8617de
--- /dev/null
+++ b/xs/src/libslic3r/Technologies.hpp
@@ -0,0 +1,12 @@
+#ifndef _technologies_h_
+#define _technologies_h_
+
+// 1.42.0 techs
+#define ENABLE_1_42_0 1
+
+// Add z coordinate to model instances' offset
+#define ENABLE_MODELINSTANCE_3D_OFFSET (1 && ENABLE_1_42_0)
+
+#endif // _technologies_h_
+
+
diff --git a/xs/src/libslic3r/libslic3r.h b/xs/src/libslic3r/libslic3r.h
index 925e93031c..2a96682f0f 100644
--- a/xs/src/libslic3r/libslic3r.h
+++ b/xs/src/libslic3r/libslic3r.h
@@ -13,6 +13,10 @@
#include
#include
+//################################################################################################################################
+#include "Technologies.hpp"
+//################################################################################################################################
+
#define SLIC3R_FORK_NAME "Slic3r Prusa Edition"
#define SLIC3R_VERSION "1.41.0"
#define SLIC3R_BUILD "UNKNOWN"
diff --git a/xs/src/slic3r/GUI/3DScene.cpp b/xs/src/slic3r/GUI/3DScene.cpp
index ed2f8690d7..a25e299286 100644
--- a/xs/src/slic3r/GUI/3DScene.cpp
+++ b/xs/src/slic3r/GUI/3DScene.cpp
@@ -684,7 +684,15 @@ std::vector GLVolumeCollection::load_object(
}
v.is_modifier = model_volume->modifier;
v.shader_outside_printer_detection_enabled = !model_volume->modifier;
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ v.set_offset(instance->get_offset());
+#else
+//################################################################################################################################
v.set_offset(Vec3d(instance->offset(0), instance->offset(1), 0.0));
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
v.set_rotation(instance->rotation);
v.set_scaling_factor(instance->scaling_factor);
}
diff --git a/xs/src/slic3r/GUI/GLCanvas3D.cpp b/xs/src/slic3r/GUI/GLCanvas3D.cpp
index fdee693ee0..b037e57eea 100644
--- a/xs/src/slic3r/GUI/GLCanvas3D.cpp
+++ b/xs/src/slic3r/GUI/GLCanvas3D.cpp
@@ -2375,7 +2375,15 @@ void GLCanvas3D::update_gizmos_data()
ModelInstance* model_instance = model_object->instances[0];
if (model_instance != nullptr)
{
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ m_gizmos.set_position(model_instance->get_offset());
+#else
+//################################################################################################################################
m_gizmos.set_position(Vec3d(model_instance->offset(0), model_instance->offset(1), 0.0));
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
m_gizmos.set_scale(model_instance->scaling_factor);
m_gizmos.set_angle_z(model_instance->rotation);
m_gizmos.set_flattening_data(model_object);
@@ -5355,8 +5363,16 @@ void GLCanvas3D::_on_move(const std::vector& volume_idxs)
ModelObject* model_object = m_model->objects[obj_idx];
if (model_object != nullptr)
{
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ model_object->instances[instance_idx]->set_offset(volume->get_offset());
+#else
+//################################################################################################################################
const Vec3d& offset = volume->get_offset();
model_object->instances[instance_idx]->offset = Vec2d(offset(0), offset(1));
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
model_object->invalidate_bounding_box();
update_position_values();
object_moved = true;
diff --git a/xs/src/slic3r/GUI/GLGizmo.cpp b/xs/src/slic3r/GUI/GLGizmo.cpp
index 9c209035b5..892ae5ca48 100644
--- a/xs/src/slic3r/GUI/GLGizmo.cpp
+++ b/xs/src/slic3r/GUI/GLGizmo.cpp
@@ -1233,10 +1233,27 @@ void GLGizmoFlatten::on_render(const BoundingBoxf3& box) const
else
::glColor4f(0.9f, 0.9f, 0.9f, 0.5f);
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ for (Vec3d offset : m_instances_positions) {
+ offset += dragged_offset;
+#else
+//################################################################################################################################
for (Vec2d offset : m_instances_positions) {
offset += to_2d(dragged_offset);
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
::glPushMatrix();
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ ::glTranslated(offset(0), offset(1), offset(2));
+#else
+//################################################################################################################################
::glTranslatef((GLfloat)offset(0), (GLfloat)offset(1), 0.0f);
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
::glBegin(GL_POLYGON);
for (const Vec3d& vertex : m_planes[i].vertices)
::glVertex3f((GLfloat)vertex(0), (GLfloat)vertex(1), (GLfloat)vertex(2));
@@ -1255,9 +1272,25 @@ void GLGizmoFlatten::on_render_for_picking(const BoundingBoxf3& box) const
for (unsigned int i = 0; i < m_planes.size(); ++i)
{
::glColor3f(1.0f, 1.0f, picking_color_component(i));
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ for (const Vec3d& offset : m_instances_positions) {
+#else
+//################################################################################################################################
for (const Vec2d& offset : m_instances_positions) {
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
::glPushMatrix();
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ ::glTranslated(offset(0), offset(1), offset(2));
+#else
+//################################################################################################################################
::glTranslatef((GLfloat)offset(0), (GLfloat)offset(1), 0.0f);
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
::glBegin(GL_POLYGON);
for (const Vec3d& vertex : m_planes[i].vertices)
::glVertex3f((GLfloat)vertex(0), (GLfloat)vertex(1), (GLfloat)vertex(2));
@@ -1275,7 +1308,15 @@ void GLGizmoFlatten::set_flattening_data(const ModelObject* model_object)
if (m_model_object && !m_model_object->instances.empty()) {
m_instances_positions.clear();
for (const auto* instance : m_model_object->instances)
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ m_instances_positions.emplace_back(instance->get_offset());
+#else
+//################################################################################################################################
m_instances_positions.emplace_back(instance->offset);
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
}
if (is_plane_update_necessary())
diff --git a/xs/src/slic3r/GUI/GLGizmo.hpp b/xs/src/slic3r/GUI/GLGizmo.hpp
index fc2912d3ac..41ccb4d77f 100644
--- a/xs/src/slic3r/GUI/GLGizmo.hpp
+++ b/xs/src/slic3r/GUI/GLGizmo.hpp
@@ -328,7 +328,15 @@ private:
SourceDataSummary m_source_data;
std::vector m_planes;
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ Pointf3s m_instances_positions;
+#else
+//################################################################################################################################
std::vector m_instances_positions;
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
Vec3d m_starting_center;
const ModelObject* m_model_object = nullptr;
diff --git a/xs/src/slic3r/GUI/GUI_ObjectParts.cpp b/xs/src/slic3r/GUI/GUI_ObjectParts.cpp
index af57db8ed8..1570a4ccf1 100644
--- a/xs/src/slic3r/GUI/GUI_ObjectParts.cpp
+++ b/xs/src/slic3r/GUI/GUI_ObjectParts.cpp
@@ -1564,9 +1564,19 @@ void update_position_values()
auto og = get_optgroup(ogFrequentlyObjectSettings);
auto instance = (*m_objects)[m_selected_object_id]->instances.front();
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ og->set_value("position_x", int(instance->get_offset(X)));
+ og->set_value("position_y", int(instance->get_offset(Y)));
+ og->set_value("position_z", int(instance->get_offset(Z)));
+#else
+//################################################################################################################################
og->set_value("position_x", int(instance->offset(0)));
og->set_value("position_y", int(instance->offset(1)));
og->set_value("position_z", 0);
+//################################################################################################################################
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
}
void update_position_values(const Vec3d& position)
diff --git a/xs/xsp/Model.xsp b/xs/xsp/Model.xsp
index a5925775df..9032323c7f 100644
--- a/xs/xsp/Model.xsp
+++ b/xs/xsp/Model.xsp
@@ -358,15 +358,32 @@ ModelMaterial::attributes()
%code%{ RETVAL = THIS->rotation; %};
double scaling_factor()
%code%{ RETVAL = THIS->scaling_factor; %};
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ Vec2d* offset()
+ %code%{ RETVAL = new Vec2d(THIS->get_offset(X), THIS->get_offset(Y)); %};
+#else
Ref offset()
%code%{ RETVAL = &THIS->offset; %};
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
void set_rotation(double val)
%code%{ THIS->rotation = val; THIS->get_object()->invalidate_bounding_box(); %};
void set_scaling_factor(double val)
%code%{ THIS->scaling_factor = val; THIS->get_object()->invalidate_bounding_box(); %};
+//################################################################################################################################
+#if ENABLE_MODELINSTANCE_3D_OFFSET
+ void set_offset(Vec2d *offset)
+ %code%{
+ THIS->set_offset(X, (*offset)(0));
+ THIS->set_offset(Y, (*offset)(1));
+ %};
+#else
void set_offset(Vec2d *offset)
%code%{ THIS->offset = *offset; %};
+#endif // ENABLE_MODELINSTANCE_3D_OFFSET
+//################################################################################################################################
void transform_mesh(TriangleMesh* mesh, bool dont_translate = false) const;
void transform_polygon(Polygon* polygon) const;