2015-01-17 23:36:21 +00:00
|
|
|
#ifndef slic3r_3DScene_hpp_
|
|
|
|
#define slic3r_3DScene_hpp_
|
|
|
|
|
2018-11-13 16:45:44 +00:00
|
|
|
#include "libslic3r/libslic3r.h"
|
|
|
|
#include "libslic3r/Point.hpp"
|
|
|
|
#include "libslic3r/Line.hpp"
|
|
|
|
#include "libslic3r/TriangleMesh.hpp"
|
|
|
|
#include "libslic3r/Utils.hpp"
|
|
|
|
#include "libslic3r/Model.hpp"
|
|
|
|
#include "slic3r/GUI/GLCanvas3DManager.hpp"
|
2015-01-17 23:36:21 +00:00
|
|
|
|
2019-01-21 09:06:51 +00:00
|
|
|
#include <functional>
|
|
|
|
|
2019-01-31 08:37:27 +00:00
|
|
|
#ifndef NDEBUG
|
|
|
|
#define HAS_GLSAFE
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef HAS_GLSAFE
|
2019-03-27 09:26:55 +00:00
|
|
|
extern void glAssertRecentCallImpl(const char *file_name, unsigned int line, const char *function_name);
|
|
|
|
inline void glAssertRecentCall() { glAssertRecentCallImpl(__FILE__, __LINE__, __FUNCTION__); }
|
|
|
|
#define glsafe(cmd) do { cmd; glAssertRecentCallImpl(__FILE__, __LINE__, __FUNCTION__); } while (false)
|
2019-01-31 08:37:27 +00:00
|
|
|
#else
|
|
|
|
inline void glAssertRecentCall() { }
|
|
|
|
#define glsafe(cmd) cmd
|
|
|
|
#endif
|
|
|
|
|
2015-01-17 23:36:21 +00:00
|
|
|
namespace Slic3r {
|
2019-03-07 10:49:00 +00:00
|
|
|
namespace GUI {
|
|
|
|
class Bed3D;
|
|
|
|
struct Camera;
|
|
|
|
class GLToolbar;
|
|
|
|
} // namespace GUI
|
2015-01-17 23:36:21 +00:00
|
|
|
|
2017-03-13 15:02:17 +00:00
|
|
|
class Print;
|
|
|
|
class PrintObject;
|
2018-11-13 16:45:44 +00:00
|
|
|
class SLAPrint;
|
|
|
|
class SLAPrintObject;
|
2018-11-13 17:44:30 +00:00
|
|
|
enum SLAPrintObjectStep : unsigned int;
|
2017-03-13 15:02:17 +00:00
|
|
|
class Model;
|
|
|
|
class ModelObject;
|
2018-03-09 09:40:42 +00:00
|
|
|
class DynamicPrintConfig;
|
2018-06-05 08:56:55 +00:00
|
|
|
class ExtrusionPath;
|
|
|
|
class ExtrusionMultiPath;
|
|
|
|
class ExtrusionLoop;
|
|
|
|
class ExtrusionEntity;
|
|
|
|
class ExtrusionEntityCollection;
|
2017-03-13 15:02:17 +00:00
|
|
|
|
2017-03-15 15:33:25 +00:00
|
|
|
// A container for interleaved arrays of 3D vertices and normals,
|
|
|
|
// possibly indexed by triangles and / or quads.
|
|
|
|
class GLIndexedVertexArray {
|
2017-03-13 15:02:17 +00:00
|
|
|
public:
|
2017-03-16 13:02:28 +00:00
|
|
|
GLIndexedVertexArray() :
|
|
|
|
vertices_and_normals_interleaved_VBO_id(0),
|
|
|
|
triangle_indices_VBO_id(0),
|
|
|
|
quad_indices_VBO_id(0)
|
|
|
|
{ this->setup_sizes(); }
|
2017-03-15 19:45:03 +00:00
|
|
|
GLIndexedVertexArray(const GLIndexedVertexArray &rhs) :
|
|
|
|
vertices_and_normals_interleaved(rhs.vertices_and_normals_interleaved),
|
|
|
|
triangle_indices(rhs.triangle_indices),
|
2017-03-16 13:02:28 +00:00
|
|
|
quad_indices(rhs.quad_indices),
|
|
|
|
vertices_and_normals_interleaved_VBO_id(0),
|
|
|
|
triangle_indices_VBO_id(0),
|
|
|
|
quad_indices_VBO_id(0)
|
|
|
|
{ this->setup_sizes(); }
|
2017-03-15 19:45:03 +00:00
|
|
|
GLIndexedVertexArray(GLIndexedVertexArray &&rhs) :
|
|
|
|
vertices_and_normals_interleaved(std::move(rhs.vertices_and_normals_interleaved)),
|
|
|
|
triangle_indices(std::move(rhs.triangle_indices)),
|
2017-03-16 13:02:28 +00:00
|
|
|
quad_indices(std::move(rhs.quad_indices)),
|
|
|
|
vertices_and_normals_interleaved_VBO_id(0),
|
|
|
|
triangle_indices_VBO_id(0),
|
|
|
|
quad_indices_VBO_id(0)
|
|
|
|
{ this->setup_sizes(); }
|
2017-03-15 19:45:03 +00:00
|
|
|
|
|
|
|
GLIndexedVertexArray& operator=(const GLIndexedVertexArray &rhs)
|
|
|
|
{
|
2017-03-16 13:02:28 +00:00
|
|
|
assert(vertices_and_normals_interleaved_VBO_id == 0);
|
|
|
|
assert(triangle_indices_VBO_id == 0);
|
|
|
|
assert(triangle_indices_VBO_id == 0);
|
2017-03-15 19:45:03 +00:00
|
|
|
this->vertices_and_normals_interleaved = rhs.vertices_and_normals_interleaved;
|
|
|
|
this->triangle_indices = rhs.triangle_indices;
|
|
|
|
this->quad_indices = rhs.quad_indices;
|
2017-03-16 13:02:28 +00:00
|
|
|
this->setup_sizes();
|
2017-03-15 19:45:03 +00:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
GLIndexedVertexArray& operator=(GLIndexedVertexArray &&rhs)
|
|
|
|
{
|
2017-03-16 13:02:28 +00:00
|
|
|
assert(vertices_and_normals_interleaved_VBO_id == 0);
|
|
|
|
assert(triangle_indices_VBO_id == 0);
|
|
|
|
assert(triangle_indices_VBO_id == 0);
|
2017-03-15 19:45:03 +00:00
|
|
|
this->vertices_and_normals_interleaved = std::move(rhs.vertices_and_normals_interleaved);
|
|
|
|
this->triangle_indices = std::move(rhs.triangle_indices);
|
|
|
|
this->quad_indices = std::move(rhs.quad_indices);
|
2017-03-16 13:02:28 +00:00
|
|
|
this->setup_sizes();
|
2017-03-15 19:45:03 +00:00
|
|
|
return *this;
|
|
|
|
}
|
2017-03-15 15:33:25 +00:00
|
|
|
|
|
|
|
// Vertices and their normals, interleaved to be used by void glInterleavedArrays(GL_N3F_V3F, 0, x)
|
|
|
|
std::vector<float> vertices_and_normals_interleaved;
|
|
|
|
std::vector<int> triangle_indices;
|
|
|
|
std::vector<int> quad_indices;
|
|
|
|
|
2017-03-16 13:02:28 +00:00
|
|
|
// When the geometry data is loaded into the graphics card as Vertex Buffer Objects,
|
|
|
|
// the above mentioned std::vectors are cleared and the following variables keep their original length.
|
|
|
|
size_t vertices_and_normals_interleaved_size;
|
|
|
|
size_t triangle_indices_size;
|
|
|
|
size_t quad_indices_size;
|
|
|
|
|
|
|
|
// IDs of the Vertex Array Objects, into which the geometry has been loaded.
|
|
|
|
// Zero if the VBOs are not used.
|
|
|
|
unsigned int vertices_and_normals_interleaved_VBO_id;
|
|
|
|
unsigned int triangle_indices_VBO_id;
|
|
|
|
unsigned int quad_indices_VBO_id;
|
|
|
|
|
2017-03-15 15:33:25 +00:00
|
|
|
void load_mesh_flat_shading(const TriangleMesh &mesh);
|
2018-03-09 09:40:42 +00:00
|
|
|
void load_mesh_full_shading(const TriangleMesh &mesh);
|
2017-03-15 15:33:25 +00:00
|
|
|
|
2017-03-16 13:02:28 +00:00
|
|
|
inline bool has_VBOs() const { return vertices_and_normals_interleaved_VBO_id != 0; }
|
|
|
|
|
2017-03-15 15:33:25 +00:00
|
|
|
inline void reserve(size_t sz) {
|
|
|
|
this->vertices_and_normals_interleaved.reserve(sz * 6);
|
|
|
|
this->triangle_indices.reserve(sz * 3);
|
|
|
|
this->quad_indices.reserve(sz * 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void push_geometry(float x, float y, float z, float nx, float ny, float nz) {
|
2017-03-28 15:09:57 +00:00
|
|
|
if (this->vertices_and_normals_interleaved.size() + 6 > this->vertices_and_normals_interleaved.capacity())
|
|
|
|
this->vertices_and_normals_interleaved.reserve(next_highest_power_of_2(this->vertices_and_normals_interleaved.size() + 6));
|
2017-03-15 15:33:25 +00:00
|
|
|
this->vertices_and_normals_interleaved.push_back(nx);
|
|
|
|
this->vertices_and_normals_interleaved.push_back(ny);
|
|
|
|
this->vertices_and_normals_interleaved.push_back(nz);
|
|
|
|
this->vertices_and_normals_interleaved.push_back(x);
|
|
|
|
this->vertices_and_normals_interleaved.push_back(y);
|
|
|
|
this->vertices_and_normals_interleaved.push_back(z);
|
2015-01-24 22:35:29 +00:00
|
|
|
};
|
2017-03-13 15:02:17 +00:00
|
|
|
|
2017-03-15 15:33:25 +00:00
|
|
|
inline void push_geometry(double x, double y, double z, double nx, double ny, double nz) {
|
|
|
|
push_geometry(float(x), float(y), float(z), float(nx), float(ny), float(nz));
|
|
|
|
}
|
|
|
|
|
2018-08-21 15:43:05 +00:00
|
|
|
inline void push_geometry(const Vec3d& p, const Vec3d& n) {
|
2018-08-17 13:53:43 +00:00
|
|
|
push_geometry(p(0), p(1), p(2), n(0), n(1), n(2));
|
2018-01-08 12:44:10 +00:00
|
|
|
}
|
|
|
|
|
2017-03-28 15:09:57 +00:00
|
|
|
inline void push_triangle(int idx1, int idx2, int idx3) {
|
|
|
|
if (this->triangle_indices.size() + 3 > this->vertices_and_normals_interleaved.capacity())
|
|
|
|
this->triangle_indices.reserve(next_highest_power_of_2(this->triangle_indices.size() + 3));
|
|
|
|
this->triangle_indices.push_back(idx1);
|
|
|
|
this->triangle_indices.push_back(idx2);
|
|
|
|
this->triangle_indices.push_back(idx3);
|
|
|
|
};
|
|
|
|
|
|
|
|
inline void push_quad(int idx1, int idx2, int idx3, int idx4) {
|
|
|
|
if (this->quad_indices.size() + 4 > this->vertices_and_normals_interleaved.capacity())
|
|
|
|
this->quad_indices.reserve(next_highest_power_of_2(this->quad_indices.size() + 4));
|
|
|
|
this->quad_indices.push_back(idx1);
|
|
|
|
this->quad_indices.push_back(idx2);
|
|
|
|
this->quad_indices.push_back(idx3);
|
|
|
|
this->quad_indices.push_back(idx4);
|
|
|
|
};
|
|
|
|
|
2017-03-16 13:02:28 +00:00
|
|
|
// Finalize the initialization of the geometry & indices,
|
|
|
|
// upload the geometry and indices to OpenGL VBO objects
|
|
|
|
// and shrink the allocated data, possibly relasing it if it has been loaded into the VBOs.
|
|
|
|
void finalize_geometry(bool use_VBOs);
|
|
|
|
// Release the geometry data, release OpenGL VBOs.
|
|
|
|
void release_geometry();
|
|
|
|
// Render either using an immediate mode, or the VBOs.
|
|
|
|
void render() const;
|
|
|
|
void render(const std::pair<size_t, size_t> &tverts_range, const std::pair<size_t, size_t> &qverts_range) const;
|
|
|
|
|
2017-03-15 15:33:25 +00:00
|
|
|
// Is there any geometry data stored?
|
2017-03-16 13:02:28 +00:00
|
|
|
bool empty() const { return vertices_and_normals_interleaved_size == 0; }
|
2017-03-15 15:33:25 +00:00
|
|
|
|
|
|
|
// Is this object indexed, or is it just a set of triangles?
|
2017-03-16 13:02:28 +00:00
|
|
|
bool indexed() const { return ! this->empty() && this->triangle_indices_size + this->quad_indices_size > 0; }
|
2017-03-15 15:33:25 +00:00
|
|
|
|
2017-03-15 19:45:03 +00:00
|
|
|
void clear() {
|
|
|
|
this->vertices_and_normals_interleaved.clear();
|
|
|
|
this->triangle_indices.clear();
|
2017-03-16 13:02:28 +00:00
|
|
|
this->quad_indices.clear();
|
|
|
|
this->setup_sizes();
|
2017-03-15 19:45:03 +00:00
|
|
|
}
|
|
|
|
|
2017-03-15 15:33:25 +00:00
|
|
|
// Shrink the internal storage to tighly fit the data stored.
|
|
|
|
void shrink_to_fit() {
|
2017-03-16 13:02:28 +00:00
|
|
|
if (! this->has_VBOs())
|
|
|
|
this->setup_sizes();
|
2017-03-15 15:33:25 +00:00
|
|
|
this->vertices_and_normals_interleaved.shrink_to_fit();
|
|
|
|
this->triangle_indices.shrink_to_fit();
|
2017-03-16 13:02:28 +00:00
|
|
|
this->quad_indices.shrink_to_fit();
|
2017-03-15 15:33:25 +00:00
|
|
|
}
|
2017-03-13 15:02:17 +00:00
|
|
|
|
|
|
|
BoundingBoxf3 bounding_box() const {
|
|
|
|
BoundingBoxf3 bbox;
|
2017-03-15 15:33:25 +00:00
|
|
|
if (! this->vertices_and_normals_interleaved.empty()) {
|
2017-09-12 16:20:06 +00:00
|
|
|
bbox.defined = true;
|
2018-08-17 13:53:43 +00:00
|
|
|
bbox.min(0) = bbox.max(0) = this->vertices_and_normals_interleaved[3];
|
|
|
|
bbox.min(1) = bbox.max(1) = this->vertices_and_normals_interleaved[4];
|
|
|
|
bbox.min(2) = bbox.max(2) = this->vertices_and_normals_interleaved[5];
|
2017-03-15 15:33:25 +00:00
|
|
|
for (size_t i = 9; i < this->vertices_and_normals_interleaved.size(); i += 6) {
|
|
|
|
const float *verts = this->vertices_and_normals_interleaved.data() + i;
|
2018-08-17 13:53:43 +00:00
|
|
|
bbox.min(0) = std::min<coordf_t>(bbox.min(0), verts[0]);
|
|
|
|
bbox.min(1) = std::min<coordf_t>(bbox.min(1), verts[1]);
|
|
|
|
bbox.min(2) = std::min<coordf_t>(bbox.min(2), verts[2]);
|
|
|
|
bbox.max(0) = std::max<coordf_t>(bbox.max(0), verts[0]);
|
|
|
|
bbox.max(1) = std::max<coordf_t>(bbox.max(1), verts[1]);
|
|
|
|
bbox.max(2) = std::max<coordf_t>(bbox.max(2), verts[2]);
|
2017-03-13 15:02:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return bbox;
|
|
|
|
}
|
2017-03-16 13:02:28 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
inline void setup_sizes() {
|
|
|
|
vertices_and_normals_interleaved_size = this->vertices_and_normals_interleaved.size();
|
|
|
|
triangle_indices_size = this->triangle_indices.size();
|
|
|
|
quad_indices_size = this->quad_indices.size();
|
|
|
|
}
|
2017-03-13 15:02:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class GLVolume {
|
|
|
|
public:
|
2018-03-09 09:40:42 +00:00
|
|
|
static const float SELECTED_COLOR[4];
|
|
|
|
static const float HOVER_COLOR[4];
|
|
|
|
static const float OUTSIDE_COLOR[4];
|
|
|
|
static const float SELECTED_OUTSIDE_COLOR[4];
|
2018-11-07 11:11:34 +00:00
|
|
|
static const float DISABLED_COLOR[4];
|
2018-11-22 10:01:57 +00:00
|
|
|
static const float SLA_SUPPORT_COLOR[4];
|
|
|
|
static const float SLA_PAD_COLOR[4];
|
2018-03-09 09:40:42 +00:00
|
|
|
|
2018-06-21 06:37:04 +00:00
|
|
|
GLVolume(float r = 1.f, float g = 1.f, float b = 1.f, float a = 1.f);
|
2017-03-13 15:02:17 +00:00
|
|
|
GLVolume(const float *rgba) : GLVolume(rgba[0], rgba[1], rgba[2], rgba[3]) {}
|
2018-11-17 16:23:56 +00:00
|
|
|
~GLVolume();
|
2017-03-13 15:02:17 +00:00
|
|
|
|
2018-06-21 06:37:04 +00:00
|
|
|
private:
|
2018-11-02 11:11:28 +00:00
|
|
|
Geometry::Transformation m_instance_transformation;
|
|
|
|
Geometry::Transformation m_volume_transformation;
|
2019-01-02 09:49:13 +00:00
|
|
|
|
2018-11-22 12:33:20 +00:00
|
|
|
// Shift in z required by sla supports+pad
|
2018-11-26 18:22:16 +00:00
|
|
|
double m_sla_shift_z;
|
2018-06-21 06:37:04 +00:00
|
|
|
// Bounding box of this volume, in unscaled coordinates.
|
|
|
|
mutable BoundingBoxf3 m_transformed_bounding_box;
|
2018-08-15 10:50:06 +00:00
|
|
|
// Whether or not is needed to recalculate the transformed bounding box.
|
|
|
|
mutable bool m_transformed_bounding_box_dirty;
|
2018-08-16 11:35:56 +00:00
|
|
|
// Pointer to convex hull of the original mesh, if any.
|
2018-11-17 16:23:56 +00:00
|
|
|
// This object may or may not own the convex hull instance based on m_convex_hull_owned
|
2018-08-16 11:35:56 +00:00
|
|
|
const TriangleMesh* m_convex_hull;
|
2018-11-17 16:23:56 +00:00
|
|
|
bool m_convex_hull_owned;
|
2018-08-15 10:50:06 +00:00
|
|
|
// Bounding box of this volume, in unscaled coordinates.
|
|
|
|
mutable BoundingBoxf3 m_transformed_convex_hull_bounding_box;
|
|
|
|
// Whether or not is needed to recalculate the transformed convex hull bounding box.
|
|
|
|
mutable bool m_transformed_convex_hull_bounding_box_dirty;
|
2017-03-13 15:02:17 +00:00
|
|
|
|
2018-06-21 06:37:04 +00:00
|
|
|
public:
|
2017-03-13 15:02:17 +00:00
|
|
|
// Bounding box of this volume, in unscaled coordinates.
|
|
|
|
BoundingBoxf3 bounding_box;
|
|
|
|
// Color of the triangles / quads held by this volume.
|
|
|
|
float color[4];
|
2018-03-09 09:40:42 +00:00
|
|
|
// Color used to render this volume.
|
|
|
|
float render_color[4];
|
2018-11-16 17:28:50 +00:00
|
|
|
struct CompositeID {
|
|
|
|
CompositeID(int object_id, int volume_id, int instance_id) : object_id(object_id), volume_id(volume_id), instance_id(instance_id) {}
|
|
|
|
CompositeID() : object_id(-1), volume_id(-1), instance_id(-1) {}
|
|
|
|
// Object ID, which is equal to the index of the respective ModelObject in Model.objects array.
|
|
|
|
int object_id;
|
|
|
|
// Volume ID, which is equal to the index of the respective ModelVolume in ModelObject.volumes array.
|
|
|
|
// If negative, it is an index of a geometry produced by the PrintObject for the respective ModelObject,
|
|
|
|
// and which has no associated ModelVolume in ModelObject.volumes. For example, SLA supports.
|
|
|
|
// Volume with a negative volume_id cannot be picked independently, it will pick the associated instance.
|
|
|
|
int volume_id;
|
|
|
|
// Instance ID, which is equal to the index of the respective ModelInstance in ModelObject.instances array.
|
|
|
|
int instance_id;
|
|
|
|
};
|
|
|
|
CompositeID composite_id;
|
|
|
|
// Fingerprint of the source geometry. For ModelVolumes, it is the ModelVolume::ID and ModelInstanceID,
|
|
|
|
// for generated volumes it is the timestamp generated by PrintState::invalidate() or PrintState::set_done(),
|
|
|
|
// and the associated ModelInstanceID.
|
|
|
|
// Valid geometry_id should always be positive.
|
|
|
|
std::pair<size_t, size_t> geometry_id;
|
2018-04-05 10:52:29 +00:00
|
|
|
// An ID containing the extruder ID (used to select color).
|
|
|
|
int extruder_id;
|
2017-03-13 15:02:17 +00:00
|
|
|
// Is this object selected?
|
|
|
|
bool selected;
|
2018-11-07 11:11:34 +00:00
|
|
|
// Is this object disabled from selection?
|
|
|
|
bool disabled;
|
2018-01-11 13:09:54 +00:00
|
|
|
// Whether or not this volume is active for rendering
|
2018-01-17 09:39:05 +00:00
|
|
|
bool is_active;
|
2018-02-19 10:28:56 +00:00
|
|
|
// Whether or not to use this volume when applying zoom_to_volumes()
|
|
|
|
bool zoom_to_volumes;
|
2018-07-26 11:12:09 +00:00
|
|
|
// Wheter or not this volume is enabled for outside print volume detection in shader.
|
|
|
|
bool shader_outside_printer_detection_enabled;
|
2018-03-09 09:40:42 +00:00
|
|
|
// Wheter or not this volume is outside print volume.
|
|
|
|
bool is_outside;
|
2017-03-13 15:02:17 +00:00
|
|
|
// Boolean: Is mouse over this object?
|
|
|
|
bool hover;
|
2018-04-05 10:52:29 +00:00
|
|
|
// Wheter or not this volume has been generated from a modifier
|
2018-06-18 13:07:17 +00:00
|
|
|
bool is_modifier;
|
2018-04-05 10:52:29 +00:00
|
|
|
// Wheter or not this volume has been generated from the wipe tower
|
2018-06-18 13:07:17 +00:00
|
|
|
bool is_wipe_tower;
|
2018-07-27 06:49:58 +00:00
|
|
|
// Wheter or not this volume has been generated from an extrusion path
|
|
|
|
bool is_extrusion_path;
|
2019-01-31 13:25:11 +00:00
|
|
|
// Wheter or not to always render this volume using its own alpha
|
|
|
|
bool force_transparent;
|
2019-03-20 13:04:20 +00:00
|
|
|
// Whether or not always use the volume's own color (not using SELECTED/HOVER/DISABLED/OUTSIDE)
|
|
|
|
bool force_native_color;
|
2017-03-13 15:02:17 +00:00
|
|
|
|
2017-03-15 15:33:25 +00:00
|
|
|
// Interleaved triangles & normals with indexed triangles & quads.
|
|
|
|
GLIndexedVertexArray indexed_vertex_array;
|
|
|
|
// Ranges of triangle and quad indices to be rendered.
|
2017-03-13 15:02:17 +00:00
|
|
|
std::pair<size_t, size_t> tverts_range;
|
2017-03-15 15:33:25 +00:00
|
|
|
std::pair<size_t, size_t> qverts_range;
|
|
|
|
|
2017-03-13 15:02:17 +00:00
|
|
|
// If the qverts or tverts contain thick extrusions, then offsets keeps pointers of the starts
|
|
|
|
// of the extrusions per layer.
|
|
|
|
std::vector<coordf_t> print_zs;
|
2017-03-14 09:11:08 +00:00
|
|
|
// Offset into qverts & tverts, or offsets into indices stored into an OpenGL name_index_buffer.
|
2017-03-13 15:02:17 +00:00
|
|
|
std::vector<size_t> offsets;
|
|
|
|
|
2018-03-09 09:40:42 +00:00
|
|
|
void set_render_color(float r, float g, float b, float a);
|
|
|
|
void set_render_color(const float* rgba, unsigned int size);
|
|
|
|
// Sets render color in dependence of current state
|
|
|
|
void set_render_color();
|
2018-12-05 11:10:09 +00:00
|
|
|
// set color according to model volume
|
|
|
|
void set_color_from_model_volume(const ModelVolume *model_volume);
|
2018-01-11 13:09:54 +00:00
|
|
|
|
2018-11-02 11:11:28 +00:00
|
|
|
const Geometry::Transformation& get_instance_transformation() const { return m_instance_transformation; }
|
|
|
|
void set_instance_transformation(const Geometry::Transformation& transformation) { m_instance_transformation = transformation; set_bounding_boxes_as_dirty(); }
|
2018-10-31 13:56:51 +00:00
|
|
|
|
2018-11-02 11:11:28 +00:00
|
|
|
const Vec3d& get_instance_offset() const { return m_instance_transformation.get_offset(); }
|
|
|
|
double get_instance_offset(Axis axis) const { return m_instance_transformation.get_offset(axis); }
|
2018-10-31 13:56:51 +00:00
|
|
|
|
2018-11-02 11:11:28 +00:00
|
|
|
void set_instance_offset(const Vec3d& offset) { m_instance_transformation.set_offset(offset); set_bounding_boxes_as_dirty(); }
|
|
|
|
void set_instance_offset(Axis axis, double offset) { m_instance_transformation.set_offset(axis, offset); set_bounding_boxes_as_dirty(); }
|
2018-10-31 13:56:51 +00:00
|
|
|
|
2018-11-02 11:11:28 +00:00
|
|
|
const Vec3d& get_instance_rotation() const { return m_instance_transformation.get_rotation(); }
|
|
|
|
double get_instance_rotation(Axis axis) const { return m_instance_transformation.get_rotation(axis); }
|
2018-10-31 13:56:51 +00:00
|
|
|
|
2018-11-02 11:11:28 +00:00
|
|
|
void set_instance_rotation(const Vec3d& rotation) { m_instance_transformation.set_rotation(rotation); set_bounding_boxes_as_dirty(); }
|
|
|
|
void set_instance_rotation(Axis axis, double rotation) { m_instance_transformation.set_rotation(axis, rotation); set_bounding_boxes_as_dirty(); }
|
2018-10-31 13:56:51 +00:00
|
|
|
|
2018-11-02 11:11:28 +00:00
|
|
|
Vec3d get_instance_scaling_factor() const { return m_instance_transformation.get_scaling_factor(); }
|
|
|
|
double get_instance_scaling_factor(Axis axis) const { return m_instance_transformation.get_scaling_factor(axis); }
|
2018-10-31 13:56:51 +00:00
|
|
|
|
2018-11-02 11:11:28 +00:00
|
|
|
void set_instance_scaling_factor(const Vec3d& scaling_factor) { m_instance_transformation.set_scaling_factor(scaling_factor); set_bounding_boxes_as_dirty(); }
|
|
|
|
void set_instance_scaling_factor(Axis axis, double scaling_factor) { m_instance_transformation.set_scaling_factor(axis, scaling_factor); set_bounding_boxes_as_dirty(); }
|
2018-10-31 13:56:51 +00:00
|
|
|
|
2018-11-02 11:11:28 +00:00
|
|
|
const Vec3d& get_instance_mirror() const { return m_instance_transformation.get_mirror(); }
|
|
|
|
double get_instance_mirror(Axis axis) const { return m_instance_transformation.get_mirror(axis); }
|
2018-10-31 13:56:51 +00:00
|
|
|
|
2018-11-02 11:11:28 +00:00
|
|
|
void set_instance_mirror(const Vec3d& mirror) { m_instance_transformation.set_mirror(mirror); set_bounding_boxes_as_dirty(); }
|
|
|
|
void set_instance_mirror(Axis axis, double mirror) { m_instance_transformation.set_mirror(axis, mirror); set_bounding_boxes_as_dirty(); }
|
|
|
|
|
|
|
|
const Geometry::Transformation& get_volume_transformation() const { return m_volume_transformation; }
|
|
|
|
void set_volume_transformation(const Geometry::Transformation& transformation) { m_volume_transformation = transformation; set_bounding_boxes_as_dirty(); }
|
|
|
|
|
|
|
|
const Vec3d& get_volume_offset() const { return m_volume_transformation.get_offset(); }
|
|
|
|
double get_volume_offset(Axis axis) const { return m_volume_transformation.get_offset(axis); }
|
|
|
|
|
|
|
|
void set_volume_offset(const Vec3d& offset) { m_volume_transformation.set_offset(offset); set_bounding_boxes_as_dirty(); }
|
|
|
|
void set_volume_offset(Axis axis, double offset) { m_volume_transformation.set_offset(axis, offset); set_bounding_boxes_as_dirty(); }
|
|
|
|
|
|
|
|
const Vec3d& get_volume_rotation() const { return m_volume_transformation.get_rotation(); }
|
|
|
|
double get_volume_rotation(Axis axis) const { return m_volume_transformation.get_rotation(axis); }
|
|
|
|
|
|
|
|
void set_volume_rotation(const Vec3d& rotation) { m_volume_transformation.set_rotation(rotation); set_bounding_boxes_as_dirty(); }
|
|
|
|
void set_volume_rotation(Axis axis, double rotation) { m_volume_transformation.set_rotation(axis, rotation); set_bounding_boxes_as_dirty(); }
|
|
|
|
|
2019-03-18 11:37:14 +00:00
|
|
|
const Vec3d& get_volume_scaling_factor() const { return m_volume_transformation.get_scaling_factor(); }
|
2018-11-02 11:11:28 +00:00
|
|
|
double get_volume_scaling_factor(Axis axis) const { return m_volume_transformation.get_scaling_factor(axis); }
|
|
|
|
|
|
|
|
void set_volume_scaling_factor(const Vec3d& scaling_factor) { m_volume_transformation.set_scaling_factor(scaling_factor); set_bounding_boxes_as_dirty(); }
|
|
|
|
void set_volume_scaling_factor(Axis axis, double scaling_factor) { m_volume_transformation.set_scaling_factor(axis, scaling_factor); set_bounding_boxes_as_dirty(); }
|
|
|
|
|
|
|
|
const Vec3d& get_volume_mirror() const { return m_volume_transformation.get_mirror(); }
|
|
|
|
double get_volume_mirror(Axis axis) const { return m_volume_transformation.get_mirror(axis); }
|
|
|
|
|
|
|
|
void set_volume_mirror(const Vec3d& mirror) { m_volume_transformation.set_mirror(mirror); set_bounding_boxes_as_dirty(); }
|
|
|
|
void set_volume_mirror(Axis axis, double mirror) { m_volume_transformation.set_mirror(axis, mirror); set_bounding_boxes_as_dirty(); }
|
2018-11-22 12:33:20 +00:00
|
|
|
|
|
|
|
double get_sla_shift_z() const { return m_sla_shift_z; }
|
|
|
|
void set_sla_shift_z(double z) { m_sla_shift_z = z; }
|
2018-09-05 07:11:58 +00:00
|
|
|
|
2018-11-17 16:23:56 +00:00
|
|
|
void set_convex_hull(const TriangleMesh *convex_hull, bool owned);
|
2018-06-21 06:37:04 +00:00
|
|
|
|
2018-11-16 17:28:50 +00:00
|
|
|
int object_idx() const { return this->composite_id.object_id; }
|
|
|
|
int volume_idx() const { return this->composite_id.volume_id; }
|
|
|
|
int instance_idx() const { return this->composite_id.instance_id; }
|
2018-06-21 06:37:04 +00:00
|
|
|
|
2019-01-21 09:06:51 +00:00
|
|
|
Transform3d world_matrix() const;
|
2019-04-02 11:47:49 +00:00
|
|
|
bool is_left_handed() const;
|
2019-01-02 09:49:13 +00:00
|
|
|
|
2018-08-28 14:08:43 +00:00
|
|
|
const BoundingBoxf3& transformed_bounding_box() const;
|
|
|
|
const BoundingBoxf3& transformed_convex_hull_bounding_box() const;
|
2017-03-13 15:02:17 +00:00
|
|
|
|
2017-03-15 15:33:25 +00:00
|
|
|
bool empty() const { return this->indexed_vertex_array.empty(); }
|
|
|
|
bool indexed() const { return this->indexed_vertex_array.indexed(); }
|
|
|
|
|
|
|
|
void set_range(coordf_t low, coordf_t high);
|
2017-03-16 13:02:28 +00:00
|
|
|
void render() const;
|
2018-06-21 06:37:04 +00:00
|
|
|
void render_VBOs(int color_id, int detection_id, int worldmatrix_id) const;
|
|
|
|
void render_legacy() const;
|
|
|
|
|
2017-03-30 08:25:52 +00:00
|
|
|
void finalize_geometry(bool use_VBOs) { this->indexed_vertex_array.finalize_geometry(use_VBOs); }
|
2017-03-16 13:02:28 +00:00
|
|
|
void release_geometry() { this->indexed_vertex_array.release_geometry(); }
|
2017-03-14 09:11:08 +00:00
|
|
|
|
2019-01-21 09:06:51 +00:00
|
|
|
void set_bounding_boxes_as_dirty() { m_transformed_bounding_box_dirty = true; m_transformed_convex_hull_bounding_box_dirty = true; }
|
2017-03-13 15:02:17 +00:00
|
|
|
};
|
|
|
|
|
2018-10-08 12:02:12 +00:00
|
|
|
typedef std::vector<GLVolume*> GLVolumePtrs;
|
|
|
|
|
2017-03-13 15:02:17 +00:00
|
|
|
class GLVolumeCollection
|
|
|
|
{
|
2018-12-21 08:56:11 +00:00
|
|
|
public:
|
|
|
|
enum ERenderType : unsigned char
|
|
|
|
{
|
|
|
|
Opaque,
|
|
|
|
Transparent,
|
|
|
|
All
|
|
|
|
};
|
|
|
|
|
|
|
|
private:
|
2018-03-09 09:40:42 +00:00
|
|
|
// min and max vertex of the print box volume
|
|
|
|
float print_box_min[3];
|
|
|
|
float print_box_max[3];
|
2018-02-12 08:04:05 +00:00
|
|
|
|
2018-11-27 13:50:57 +00:00
|
|
|
// z range for clipping in shaders
|
|
|
|
float z_range[2];
|
|
|
|
|
2017-03-13 15:02:17 +00:00
|
|
|
public:
|
2018-10-08 12:02:12 +00:00
|
|
|
GLVolumePtrs volumes;
|
|
|
|
|
2017-03-13 15:02:17 +00:00
|
|
|
GLVolumeCollection() {};
|
|
|
|
~GLVolumeCollection() { clear(); };
|
|
|
|
|
|
|
|
std::vector<int> load_object(
|
|
|
|
const ModelObject *model_object,
|
|
|
|
int obj_idx,
|
|
|
|
const std::vector<int> &instance_idxs,
|
|
|
|
const std::string &color_by,
|
2017-03-20 11:05:20 +00:00
|
|
|
bool use_VBOs);
|
2017-03-13 15:02:17 +00:00
|
|
|
|
2018-11-16 17:28:50 +00:00
|
|
|
int load_object_volume(
|
2018-11-13 16:45:44 +00:00
|
|
|
const ModelObject *model_object,
|
|
|
|
int obj_idx,
|
2018-11-16 17:28:50 +00:00
|
|
|
int volume_idx,
|
|
|
|
int instance_idx,
|
|
|
|
const std::string &color_by,
|
2018-11-13 16:45:44 +00:00
|
|
|
bool use_VBOs);
|
|
|
|
|
2018-11-16 17:28:50 +00:00
|
|
|
// Load SLA auxiliary GLVolumes (for support trees or pad).
|
2018-11-16 17:45:13 +00:00
|
|
|
void load_object_auxiliary(
|
2018-11-16 17:28:50 +00:00
|
|
|
const SLAPrintObject *print_object,
|
|
|
|
int obj_idx,
|
|
|
|
// pairs of <instance_idx, print_instance_idx>
|
|
|
|
const std::vector<std::pair<size_t, size_t>> &instances,
|
|
|
|
SLAPrintObjectStep milestone,
|
2018-11-17 16:23:56 +00:00
|
|
|
// Timestamp of the last change of the milestone
|
|
|
|
size_t timestamp,
|
2018-11-16 17:28:50 +00:00
|
|
|
bool use_VBOs);
|
|
|
|
|
2017-05-17 14:53:40 +00:00
|
|
|
int load_wipe_tower_preview(
|
2018-07-27 13:56:27 +00:00
|
|
|
int obj_idx, float pos_x, float pos_y, float width, float depth, float height, float rotation_angle, bool use_VBOs, bool size_unknown, float brim_width);
|
2017-05-17 14:53:40 +00:00
|
|
|
|
2017-03-20 11:05:20 +00:00
|
|
|
// Render the volumes by OpenGL.
|
2019-01-21 09:06:51 +00:00
|
|
|
void render_VBOs(ERenderType type, bool disable_cullface, std::function<bool(const GLVolume&)> filter_func = std::function<bool(const GLVolume&)>()) const;
|
2019-02-06 16:20:54 +00:00
|
|
|
void render_legacy(ERenderType type, bool disable_cullface, std::function<bool(const GLVolume&)> filter_func = std::function<bool(const GLVolume&)>()) const;
|
2017-03-30 08:25:52 +00:00
|
|
|
|
|
|
|
// Finalize the initialization of the geometry & indices,
|
|
|
|
// upload the geometry and indices to OpenGL VBO objects
|
|
|
|
// and shrink the allocated data, possibly relasing it if it has been loaded into the VBOs.
|
|
|
|
void finalize_geometry(bool use_VBOs) { for (auto *v : volumes) v->finalize_geometry(use_VBOs); }
|
2017-03-16 13:02:28 +00:00
|
|
|
// Release the geometry data assigned to the volumes.
|
|
|
|
// If OpenGL VBOs were allocated, an OpenGL context has to be active to release them.
|
|
|
|
void release_geometry() { for (auto *v : volumes) v->release_geometry(); }
|
|
|
|
// Clear the geometry
|
2017-03-13 15:02:17 +00:00
|
|
|
void clear() { for (auto *v : volumes) delete v; volumes.clear(); }
|
2017-03-16 13:02:28 +00:00
|
|
|
|
2017-03-13 15:02:17 +00:00
|
|
|
bool empty() const { return volumes.empty(); }
|
|
|
|
void set_range(double low, double high) { for (GLVolume *vol : this->volumes) vol->set_range(low, high); }
|
2017-03-15 19:45:03 +00:00
|
|
|
|
2018-03-09 09:40:42 +00:00
|
|
|
void set_print_box(float min_x, float min_y, float min_z, float max_x, float max_y, float max_z) {
|
|
|
|
print_box_min[0] = min_x; print_box_min[1] = min_y; print_box_min[2] = min_z;
|
|
|
|
print_box_max[0] = max_x; print_box_max[1] = max_y; print_box_max[2] = max_z;
|
|
|
|
}
|
|
|
|
|
2018-11-27 13:50:57 +00:00
|
|
|
void set_z_range(float min_z, float max_z) { z_range[0] = min_z; z_range[1] = max_z; }
|
|
|
|
|
2018-07-18 12:26:42 +00:00
|
|
|
// returns true if all the volumes are completely contained in the print volume
|
|
|
|
// returns the containment state in the given out_state, if non-null
|
|
|
|
bool check_outside_state(const DynamicPrintConfig* config, ModelInstance::EPrintVolumeState* out_state);
|
2018-04-24 07:00:33 +00:00
|
|
|
void reset_outside_state();
|
|
|
|
|
2018-04-05 10:52:29 +00:00
|
|
|
void update_colors_by_extruder(const DynamicPrintConfig* config);
|
2018-02-12 08:04:05 +00:00
|
|
|
|
2018-02-22 07:59:47 +00:00
|
|
|
// Returns a vector containing the sorted list of all the print_zs of the volumes contained in this collection
|
2018-05-18 08:14:47 +00:00
|
|
|
std::vector<double> get_current_print_zs(bool active_only) const;
|
2018-02-22 07:59:47 +00:00
|
|
|
|
2017-03-15 19:45:03 +00:00
|
|
|
private:
|
|
|
|
GLVolumeCollection(const GLVolumeCollection &other);
|
|
|
|
GLVolumeCollection& operator=(const GLVolumeCollection &);
|
2015-01-24 22:35:29 +00:00
|
|
|
};
|
|
|
|
|
2018-12-19 13:44:37 +00:00
|
|
|
class GLModel
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
GLVolume m_volume;
|
|
|
|
bool m_useVBOs;
|
2019-01-04 11:56:48 +00:00
|
|
|
std::string m_filename;
|
2018-12-19 13:44:37 +00:00
|
|
|
|
|
|
|
public:
|
|
|
|
GLModel();
|
|
|
|
virtual ~GLModel();
|
|
|
|
|
|
|
|
bool init(bool useVBOs) { return on_init(useVBOs); }
|
2019-01-04 11:56:48 +00:00
|
|
|
bool init_from_file(const std::string& filename, bool useVBOs) { return on_init_from_file(filename, useVBOs); }
|
2018-12-19 13:44:37 +00:00
|
|
|
|
2019-01-04 11:56:48 +00:00
|
|
|
void center_around(const Vec3d& center) { m_volume.set_volume_offset(center - m_volume.bounding_box.center()); }
|
2018-12-20 10:14:53 +00:00
|
|
|
void set_color(const float* color, unsigned int size);
|
|
|
|
|
|
|
|
const Vec3d& get_offset() const;
|
|
|
|
void set_offset(const Vec3d& offset);
|
|
|
|
const Vec3d& get_rotation() const;
|
|
|
|
void set_rotation(const Vec3d& rotation);
|
|
|
|
const Vec3d& get_scale() const;
|
2018-12-19 13:44:37 +00:00
|
|
|
void set_scale(const Vec3d& scale);
|
|
|
|
|
2019-01-04 11:56:48 +00:00
|
|
|
const std::string& get_filename() const { return m_filename; }
|
|
|
|
const BoundingBoxf3& get_bounding_box() const { return m_volume.bounding_box; }
|
|
|
|
|
|
|
|
void reset();
|
|
|
|
|
2018-12-19 13:44:37 +00:00
|
|
|
void render() const;
|
|
|
|
|
|
|
|
protected:
|
2019-01-04 11:56:48 +00:00
|
|
|
virtual bool on_init(bool useVBOs) { return false; }
|
|
|
|
virtual bool on_init_from_file(const std::string& filename, bool useVBOs) { return false; }
|
2018-12-19 13:44:37 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
void render_VBOs() const;
|
2018-12-20 10:42:26 +00:00
|
|
|
void render_legacy() const;
|
2018-12-19 13:44:37 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class GLArrow : public GLModel
|
|
|
|
{
|
2018-12-20 10:14:53 +00:00
|
|
|
protected:
|
|
|
|
virtual bool on_init(bool useVBOs);
|
|
|
|
};
|
|
|
|
|
|
|
|
class GLCurvedArrow : public GLModel
|
|
|
|
{
|
|
|
|
unsigned int m_resolution;
|
|
|
|
|
2018-12-19 13:44:37 +00:00
|
|
|
public:
|
2018-12-20 10:14:53 +00:00
|
|
|
explicit GLCurvedArrow(unsigned int resolution);
|
2018-12-19 13:44:37 +00:00
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual bool on_init(bool useVBOs);
|
|
|
|
};
|
|
|
|
|
2019-01-04 11:56:48 +00:00
|
|
|
class GLBed : public GLModel
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
virtual bool on_init_from_file(const std::string& filename, bool useVBOs);
|
|
|
|
};
|
|
|
|
|
2015-01-17 23:36:21 +00:00
|
|
|
class _3DScene
|
|
|
|
{
|
2018-05-09 08:47:04 +00:00
|
|
|
static GUI::GLCanvas3DManager s_canvas_mgr;
|
2018-02-06 11:43:25 +00:00
|
|
|
|
2017-03-13 15:02:17 +00:00
|
|
|
public:
|
2018-06-04 08:14:09 +00:00
|
|
|
static std::string get_gl_info(bool format_as_html, bool extensions);
|
2018-05-09 08:47:04 +00:00
|
|
|
|
2019-03-07 10:49:00 +00:00
|
|
|
static bool add_canvas(wxGLCanvas* canvas, GUI::Bed3D& bed, GUI::Camera& camera, GUI::GLToolbar& view_toolbar);
|
2018-05-09 08:47:04 +00:00
|
|
|
static bool remove_canvas(wxGLCanvas* canvas);
|
|
|
|
static void remove_all_canvases();
|
|
|
|
|
2018-06-04 11:15:28 +00:00
|
|
|
static bool init(wxGLCanvas* canvas);
|
2018-05-09 08:47:04 +00:00
|
|
|
|
2018-10-11 08:24:19 +00:00
|
|
|
static GUI::GLCanvas3D* get_canvas(wxGLCanvas* canvas);
|
2018-05-18 12:08:59 +00:00
|
|
|
|
2018-06-05 12:09:36 +00:00
|
|
|
static void thick_lines_to_verts(const Lines& lines, const std::vector<double>& widths, const std::vector<double>& heights, bool closed, double top_z, GLVolume& volume);
|
|
|
|
static void thick_lines_to_verts(const Lines3& lines, const std::vector<double>& widths, const std::vector<double>& heights, bool closed, GLVolume& volume);
|
2018-06-05 08:56:55 +00:00
|
|
|
static void extrusionentity_to_verts(const ExtrusionPath& extrusion_path, float print_z, GLVolume& volume);
|
|
|
|
static void extrusionentity_to_verts(const ExtrusionPath& extrusion_path, float print_z, const Point& copy, GLVolume& volume);
|
|
|
|
static void extrusionentity_to_verts(const ExtrusionLoop& extrusion_loop, float print_z, const Point& copy, GLVolume& volume);
|
|
|
|
static void extrusionentity_to_verts(const ExtrusionMultiPath& extrusion_multi_path, float print_z, const Point& copy, GLVolume& volume);
|
|
|
|
static void extrusionentity_to_verts(const ExtrusionEntityCollection& extrusion_entity_collection, float print_z, const Point& copy, GLVolume& volume);
|
|
|
|
static void extrusionentity_to_verts(const ExtrusionEntity* extrusion_entity, float print_z, const Point& copy, GLVolume& volume);
|
|
|
|
static void polyline3_to_verts(const Polyline3& polyline, double width, double height, GLVolume& volume);
|
2018-08-21 20:14:47 +00:00
|
|
|
static void point3_to_verts(const Vec3crd& point, double width, double height, GLVolume& volume);
|
2015-01-17 23:36:21 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|