2019-03-15 11:53:15 +00:00
|
|
|
#ifndef slic3r_GLGizmoSlaSupports_hpp_
|
|
|
|
#define slic3r_GLGizmoSlaSupports_hpp_
|
|
|
|
|
|
|
|
#include "GLGizmoBase.hpp"
|
2019-03-20 12:51:25 +00:00
|
|
|
#include "GLGizmos.hpp"
|
2019-04-18 07:59:17 +00:00
|
|
|
#include "slic3r/GUI/GLSelectionRectangle.hpp"
|
2019-03-15 11:53:15 +00:00
|
|
|
|
|
|
|
// There is an L function in igl that would be overridden by our localization macro - let's undefine it...
|
|
|
|
#undef L
|
|
|
|
#include <igl/AABB.h>
|
|
|
|
#include "slic3r/GUI/I18N.hpp" // ...and redefine again when we are done with the igl code
|
|
|
|
|
|
|
|
#include "libslic3r/SLA/SLACommon.hpp"
|
|
|
|
#include "libslic3r/SLAPrint.hpp"
|
2019-04-18 05:23:51 +00:00
|
|
|
#include <wx/dialog.h>
|
2019-03-15 11:53:15 +00:00
|
|
|
|
2019-07-17 13:38:19 +00:00
|
|
|
#include <cereal/types/vector.hpp>
|
|
|
|
|
2019-03-15 11:53:15 +00:00
|
|
|
|
|
|
|
namespace Slic3r {
|
|
|
|
namespace GUI {
|
|
|
|
|
|
|
|
|
2019-04-08 08:50:10 +00:00
|
|
|
class ClippingPlane;
|
|
|
|
|
|
|
|
|
2019-03-15 11:53:15 +00:00
|
|
|
class GLGizmoSlaSupports : public GLGizmoBase
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
ModelObject* m_model_object = nullptr;
|
2019-07-29 13:08:59 +00:00
|
|
|
ObjectID m_model_object_id = 0;
|
2019-03-15 11:53:15 +00:00
|
|
|
int m_active_instance = -1;
|
2019-04-05 14:38:54 +00:00
|
|
|
float m_active_instance_bb_radius; // to cache the bb
|
|
|
|
mutable float m_z_shift = 0.f;
|
2019-03-15 11:53:15 +00:00
|
|
|
std::pair<Vec3f, Vec3f> unproject_on_mesh(const Vec2d& mouse_pos);
|
|
|
|
|
|
|
|
const float RenderPointScale = 1.f;
|
|
|
|
|
|
|
|
GLUquadricObj* m_quadric;
|
2019-06-10 20:43:21 +00:00
|
|
|
typedef Eigen::Map<const Eigen::Matrix<float, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor | Eigen::DontAlign>> MapMatrixXfUnaligned;
|
|
|
|
typedef Eigen::Map<const Eigen::Matrix<int, Eigen::Dynamic, Eigen::Dynamic, Eigen::RowMajor | Eigen::DontAlign>> MapMatrixXiUnaligned;
|
|
|
|
igl::AABB<MapMatrixXfUnaligned, 3> m_AABB;
|
2019-04-15 20:24:10 +00:00
|
|
|
const TriangleMesh* m_mesh;
|
2019-06-10 20:43:21 +00:00
|
|
|
const indexed_triangle_set* m_its;
|
2019-04-15 20:24:10 +00:00
|
|
|
mutable const TriangleMesh* m_supports_mesh;
|
2019-03-27 06:44:02 +00:00
|
|
|
mutable std::vector<Vec2f> m_triangles;
|
2019-04-15 14:20:29 +00:00
|
|
|
mutable std::vector<Vec2f> m_supports_triangles;
|
|
|
|
mutable int m_old_timestamp = -1;
|
|
|
|
mutable int m_print_object_idx = -1;
|
|
|
|
mutable int m_print_objects_count = -1;
|
2019-03-15 11:53:15 +00:00
|
|
|
|
|
|
|
class CacheEntry {
|
|
|
|
public:
|
2019-07-17 13:38:19 +00:00
|
|
|
CacheEntry() :
|
|
|
|
support_point(sla::SupportPoint()), selected(false), normal(Vec3f::Zero()) {}
|
|
|
|
|
2019-07-29 11:07:49 +00:00
|
|
|
CacheEntry(const sla::SupportPoint& point, bool sel = false, const Vec3f& norm = Vec3f::Zero()) :
|
2019-03-15 11:53:15 +00:00
|
|
|
support_point(point), selected(sel), normal(norm) {}
|
|
|
|
|
2019-07-29 11:07:49 +00:00
|
|
|
bool operator==(const CacheEntry& rhs) const {
|
|
|
|
return (support_point == rhs.support_point);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator!=(const CacheEntry& rhs) const {
|
|
|
|
return ! ((*this) == rhs);
|
|
|
|
}
|
|
|
|
|
2019-03-15 11:53:15 +00:00
|
|
|
sla::SupportPoint support_point;
|
|
|
|
bool selected; // whether the point is selected
|
|
|
|
Vec3f normal;
|
2019-07-17 13:38:19 +00:00
|
|
|
|
|
|
|
template<class Archive>
|
|
|
|
void serialize(Archive & ar)
|
|
|
|
{
|
|
|
|
ar(support_point, selected, normal);
|
|
|
|
}
|
2019-03-15 11:53:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
public:
|
|
|
|
GLGizmoSlaSupports(GLCanvas3D& parent, const std::string& icon_filename, unsigned int sprite_id);
|
|
|
|
virtual ~GLGizmoSlaSupports();
|
2019-03-19 12:30:21 +00:00
|
|
|
void set_sla_support_data(ModelObject* model_object, const Selection& selection);
|
2019-04-08 13:34:42 +00:00
|
|
|
bool gizmo_event(SLAGizmoEventType action, const Vec2d& mouse_position, bool shift_down, bool alt_down, bool control_down);
|
2019-03-15 11:53:15 +00:00
|
|
|
void delete_selected_points(bool force = false);
|
2019-04-08 08:50:10 +00:00
|
|
|
ClippingPlane get_sla_clipping_plane() const;
|
2019-03-15 11:53:15 +00:00
|
|
|
|
2019-04-24 13:43:52 +00:00
|
|
|
bool is_in_editing_mode() const { return m_editing_mode; }
|
|
|
|
bool is_selection_rectangle_dragging() const { return m_selection_rectangle.is_dragging(); }
|
|
|
|
|
2019-03-15 11:53:15 +00:00
|
|
|
private:
|
|
|
|
bool on_init();
|
2019-07-17 10:06:23 +00:00
|
|
|
void on_update(const UpdateData& data);
|
|
|
|
virtual void on_render() const;
|
|
|
|
virtual void on_render_for_picking() const;
|
2019-03-15 11:53:15 +00:00
|
|
|
|
2019-04-18 07:59:17 +00:00
|
|
|
//void render_selection_rectangle() const;
|
2019-04-17 10:24:46 +00:00
|
|
|
void render_points(const Selection& selection, bool picking = false) const;
|
|
|
|
void render_clipping_plane(const Selection& selection) const;
|
2019-03-15 11:53:15 +00:00
|
|
|
bool is_mesh_update_necessary() const;
|
|
|
|
void update_mesh();
|
|
|
|
void update_cache_entry_normal(unsigned int i) const;
|
2019-07-29 11:07:49 +00:00
|
|
|
bool unsaved_changes() const;
|
2019-03-15 11:53:15 +00:00
|
|
|
|
|
|
|
bool m_lock_unique_islands = false;
|
|
|
|
bool m_editing_mode = false; // Is editing mode active?
|
|
|
|
bool m_old_editing_state = false; // To keep track of whether the user toggled between the modes (needed for imgui refreshes).
|
|
|
|
float m_new_point_head_diameter; // Size of a new point.
|
2019-07-29 13:08:59 +00:00
|
|
|
CacheEntry m_point_before_drag; // undo/redo - so we know what state was edited
|
|
|
|
float m_old_point_head_diameter = 0.; // the same
|
2019-07-31 10:40:47 +00:00
|
|
|
float m_minimal_point_distance_stash = 0.f; // and again
|
|
|
|
float m_density_stash = 0.f; // and again
|
2019-07-29 11:07:49 +00:00
|
|
|
mutable std::vector<CacheEntry> m_editing_cache; // a support point and whether it is currently selected
|
|
|
|
std::vector<sla::SupportPoint> m_normal_cache; // to restore after discarding changes or undo/redo
|
2019-07-22 11:25:17 +00:00
|
|
|
|
2019-03-15 11:53:15 +00:00
|
|
|
float m_clipping_plane_distance = 0.f;
|
2019-03-27 06:44:02 +00:00
|
|
|
mutable float m_old_clipping_plane_distance = 0.f;
|
2019-04-17 10:24:46 +00:00
|
|
|
mutable Vec3d m_old_clipping_plane_normal;
|
|
|
|
mutable Vec3d m_clipping_plane_normal = Vec3d::Zero();
|
2019-03-15 11:53:15 +00:00
|
|
|
|
2019-05-24 10:01:26 +00:00
|
|
|
// This map holds all translated description texts, so they can be easily referenced during layout calculations
|
|
|
|
// etc. When language changes, GUI is recreated and this class constructed again, so the change takes effect.
|
|
|
|
std::map<std::string, wxString> m_desc;
|
|
|
|
|
2019-04-18 07:59:17 +00:00
|
|
|
GLSelectionRectangle m_selection_rectangle;
|
2019-04-25 07:10:03 +00:00
|
|
|
|
2019-03-20 13:04:59 +00:00
|
|
|
bool m_wait_for_up_event = false;
|
2019-03-15 11:53:15 +00:00
|
|
|
bool m_selection_empty = true;
|
|
|
|
EState m_old_state = Off; // to be able to see that the gizmo has just been closed (see on_set_state)
|
|
|
|
|
2019-03-27 06:44:02 +00:00
|
|
|
mutable std::unique_ptr<TriangleMeshSlicer> m_tms;
|
2019-04-15 14:20:29 +00:00
|
|
|
mutable std::unique_ptr<TriangleMeshSlicer> m_supports_tms;
|
2019-03-27 06:44:02 +00:00
|
|
|
|
2019-03-15 11:53:15 +00:00
|
|
|
std::vector<const ConfigOption*> get_config_options(const std::vector<std::string>& keys) const;
|
2019-04-17 10:24:46 +00:00
|
|
|
bool is_point_clipped(const Vec3d& point) const;
|
|
|
|
//void find_intersecting_facets(const igl::AABB<Eigen::MatrixXf, 3>* aabb, const Vec3f& normal, double offset, std::vector<unsigned int>& out) const;
|
2019-03-15 11:53:15 +00:00
|
|
|
|
|
|
|
// Methods that do the model_object and editing cache synchronization,
|
|
|
|
// editing mode selection, etc:
|
|
|
|
enum {
|
|
|
|
AllPoints = -2,
|
|
|
|
NoPoints,
|
|
|
|
};
|
|
|
|
void select_point(int i);
|
2019-04-03 12:44:15 +00:00
|
|
|
void unselect_point(int i);
|
2019-03-15 11:53:15 +00:00
|
|
|
void editing_mode_apply_changes();
|
|
|
|
void editing_mode_discard_changes();
|
2019-07-29 11:07:49 +00:00
|
|
|
void reload_cache();
|
2019-03-15 11:53:15 +00:00
|
|
|
void get_data_from_backend();
|
|
|
|
void auto_generate();
|
|
|
|
void switch_to_editing_mode();
|
2019-07-29 11:07:49 +00:00
|
|
|
void disable_editing_mode();
|
2019-04-17 10:24:46 +00:00
|
|
|
void reset_clipping_plane_normal() const;
|
2019-03-15 11:53:15 +00:00
|
|
|
|
|
|
|
protected:
|
|
|
|
void on_set_state() override;
|
2019-06-18 06:53:13 +00:00
|
|
|
virtual void on_set_hover_id()
|
|
|
|
{
|
2019-07-29 11:07:49 +00:00
|
|
|
if (! m_editing_mode || (int)m_editing_cache.size() <= m_hover_id)
|
2019-06-18 06:53:13 +00:00
|
|
|
m_hover_id = -1;
|
|
|
|
}
|
2019-07-17 10:06:23 +00:00
|
|
|
void on_start_dragging() override;
|
2019-07-29 11:07:49 +00:00
|
|
|
void on_stop_dragging() override;
|
2019-07-17 10:06:23 +00:00
|
|
|
virtual void on_render_input_window(float x, float y, float bottom_limit) override;
|
2019-03-15 11:53:15 +00:00
|
|
|
|
|
|
|
virtual std::string on_get_name() const;
|
2019-07-17 10:06:23 +00:00
|
|
|
virtual bool on_is_activable() const;
|
2019-03-15 11:53:15 +00:00
|
|
|
virtual bool on_is_selectable() const;
|
2019-07-17 13:38:19 +00:00
|
|
|
virtual void on_load(cereal::BinaryInputArchive& ar) override;
|
|
|
|
virtual void on_save(cereal::BinaryOutputArchive& ar) const override;
|
2019-03-15 11:53:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2019-04-18 05:23:51 +00:00
|
|
|
class SlaGizmoHelpDialog : public wxDialog
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
SlaGizmoHelpDialog();
|
|
|
|
};
|
|
|
|
|
2019-03-15 11:53:15 +00:00
|
|
|
} // namespace GUI
|
|
|
|
} // namespace Slic3r
|
|
|
|
|
|
|
|
#endif // slic3r_GLGizmoSlaSupports_hpp_
|