TriangleSelector: remerging triangles, bugfixes
This commit is contained in:
parent
bed28bb2ff
commit
fb73bb1c66
2 changed files with 140 additions and 80 deletions
|
@ -95,7 +95,10 @@ void GLGizmoFdmSupports::on_render() const
|
|||
glsafe(::glEnable(GL_DEPTH_TEST));
|
||||
|
||||
//render_triangles(selection);
|
||||
m_triangle_selector->render();
|
||||
|
||||
if (m_triangle_selector && ! m_setting_angle)
|
||||
m_triangle_selector->render(m_imgui);
|
||||
|
||||
m_c->object_clipper()->render_cut();
|
||||
render_cursor_circle();
|
||||
|
||||
|
@ -569,12 +572,6 @@ void GLGizmoFdmSupports::on_render_input_window(float x, float y, float bottom_l
|
|||
if (! m_c->selection_info()->model_object())
|
||||
return;
|
||||
|
||||
m_imgui->begin(std::string("TriangleSelector DEBUG"), ImGuiWindowFlags_AlwaysAutoResize | ImGuiWindowFlags_NoCollapse);
|
||||
static float edge_limit = 1.f;
|
||||
m_imgui->slider_float("Edge limit (mm): ", &edge_limit, 0.1f, 8.f);
|
||||
m_triangle_selector->set_edge_limit(edge_limit);
|
||||
m_imgui->end();
|
||||
|
||||
const float approx_height = m_imgui->scaled(18.0f);
|
||||
y = std::min(y, bottom_limit - approx_height);
|
||||
m_imgui->set_next_window_pos(x, y, ImGuiCond_Always);
|
||||
|
@ -740,9 +737,7 @@ CommonGizmosDataID GLGizmoFdmSupports::on_get_requirements() const
|
|||
int(CommonGizmosDataID::SelectionInfo)
|
||||
| int(CommonGizmosDataID::InstancesHider)
|
||||
| int(CommonGizmosDataID::Raycaster)
|
||||
| int(CommonGizmosDataID::HollowedMesh)
|
||||
| int(CommonGizmosDataID::ObjectClipper)
|
||||
| int(CommonGizmosDataID::SupportsClipper));
|
||||
| int(CommonGizmosDataID::ObjectClipper));
|
||||
}
|
||||
|
||||
|
||||
|
@ -872,7 +867,7 @@ void TriangleSelector::select_patch(const Vec3f& hit, int facet_start,
|
|||
while (facet_idx < int(facets_to_check.size())) {
|
||||
int facet = facets_to_check[facet_idx];
|
||||
if (! visited[facet]) {
|
||||
if (select_triangle(facet, new_state, facet == facet_start)) {
|
||||
if (select_triangle(facet, new_state)) {
|
||||
// add neighboring facets to list to be proccessed later
|
||||
for (int n=0; n<3; ++n) {
|
||||
if (faces_camera(m_mesh->stl.neighbors_start[facet].neighbor[n]))
|
||||
|
@ -891,49 +886,54 @@ void TriangleSelector::select_patch(const Vec3f& hit, int facet_start,
|
|||
// the triangle recursively, selecting just subtriangles truly inside the circle.
|
||||
// This is done by an actual recursive call. Returns false if the triangle is
|
||||
// outside the cursor.
|
||||
bool TriangleSelector::select_triangle(int facet_idx, FacetSupportType type, bool cursor_inside)
|
||||
bool TriangleSelector::select_triangle(int facet_idx, FacetSupportType type, bool recursive_call)
|
||||
{
|
||||
assert(facet_idx < int(m_triangles.size()));
|
||||
|
||||
Triangle& tr = m_triangles[facet_idx];
|
||||
if (! tr.valid)
|
||||
Triangle* tr = &m_triangles[facet_idx];
|
||||
if (! tr->valid)
|
||||
return false;
|
||||
|
||||
cursor_inside = is_pointer_in_triangle(facet_idx);
|
||||
|
||||
int num_of_inside_vertices = vertices_inside(facet_idx);
|
||||
|
||||
if (num_of_inside_vertices == 0
|
||||
&& ! cursor_inside
|
||||
&& ! is_pointer_in_triangle(facet_idx)
|
||||
&& ! is_edge_inside_cursor(facet_idx))
|
||||
return false;
|
||||
|
||||
if (vertices_inside(facet_idx) == 3) {
|
||||
if (num_of_inside_vertices == 3) {
|
||||
// dump any subdivision and select whole triangle
|
||||
undivide_triangle(facet_idx);
|
||||
tr.set_state(type);
|
||||
tr->set_state(type);
|
||||
} else {
|
||||
// the triangle is partially inside, let's recursively divide it
|
||||
// (if not already) and try selecting its children.
|
||||
|
||||
if (! tr.is_split() && tr.get_state() == type) {
|
||||
if (! tr->is_split() && tr->get_state() == type) {
|
||||
// This is leaf triangle that is already of correct type as a whole.
|
||||
// No need to split, all children would end up selected anyway.
|
||||
return true;
|
||||
}
|
||||
|
||||
split_triangle(facet_idx);
|
||||
assert(facet_idx < int(m_triangles.size()));
|
||||
int num_of_children = tr.number_of_split_sides() + 1;
|
||||
if (num_of_children != 1) {
|
||||
for (int i=0; i<num_of_children; ++i)
|
||||
select_triangle(tr.children[i], type, cursor_inside);
|
||||
}
|
||||
}
|
||||
tr = &m_triangles[facet_idx]; // might have been invalidated
|
||||
|
||||
// In case that all siblings are leafs and have the same state now,
|
||||
|
||||
int num_of_children = tr->number_of_split_sides() + 1;
|
||||
if (num_of_children != 1) {
|
||||
for (int i=0; i<num_of_children; ++i) {
|
||||
assert(i < int(tr->children.size()));
|
||||
assert(tr->children[i] < int(m_triangles.size()));
|
||||
|
||||
select_triangle(tr->children[i], type, true);
|
||||
tr = &m_triangles[facet_idx]; // might have been invalidated
|
||||
}
|
||||
}
|
||||
}
|
||||
// In case that all children are leafs and have the same state now,
|
||||
// they may be removed and substituted by the parent triangle.
|
||||
//remove_if_needless(facet_idx);
|
||||
if (! recursive_call)
|
||||
remove_useless_children(facet_idx);
|
||||
return true;
|
||||
}
|
||||
|
||||
|
@ -945,13 +945,13 @@ bool TriangleSelector::split_triangle(int facet_idx)
|
|||
return false;
|
||||
}
|
||||
|
||||
Triangle& tr = m_triangles[facet_idx];
|
||||
Triangle* tr = &m_triangles[facet_idx];
|
||||
|
||||
FacetSupportType old_type = tr.get_state();
|
||||
FacetSupportType old_type = tr->get_state();
|
||||
|
||||
const double limit_squared = m_edge_limit_sqr;
|
||||
|
||||
stl_triangle_vertex_indices& facet = tr.verts_idxs;
|
||||
stl_triangle_vertex_indices& facet = tr->verts_idxs;
|
||||
const stl_vertex* pts[3] = { &m_vertices[facet[0]], &m_vertices[facet[1]], &m_vertices[facet[2]]};
|
||||
double sides[3] = { (*pts[2]-*pts[1]).squaredNorm(), (*pts[0]-*pts[2]).squaredNorm(), (*pts[1]-*pts[0]).squaredNorm() };
|
||||
|
||||
|
@ -964,7 +964,7 @@ bool TriangleSelector::split_triangle(int facet_idx)
|
|||
side_to_keep = pt_idx;
|
||||
}
|
||||
if (sides_to_split.empty()) {
|
||||
tr.set_division(0);
|
||||
tr->set_division(0);
|
||||
return false;
|
||||
}
|
||||
|
||||
|
@ -1012,17 +1012,19 @@ bool TriangleSelector::split_triangle(int facet_idx)
|
|||
m_triangles.emplace_back(verts_idxs[1], verts_idxs[3], verts_idxs[5]);
|
||||
}
|
||||
|
||||
tr = &m_triangles[facet_idx]; // may have been invalidated
|
||||
|
||||
// Save how the triangle was split. Second argument makes sense only for one
|
||||
// or two split sides, otherwise the value is ignored.
|
||||
tr.set_division(sides_to_split.size(),
|
||||
tr->set_division(sides_to_split.size(),
|
||||
sides_to_split.size() == 2 ? side_to_keep : sides_to_split[0]);
|
||||
|
||||
// And save the children. All children should start in the same state as the triangle we just split.
|
||||
assert(! sides_to_split.empty() && int(sides_to_split.size()) <= 3);
|
||||
for (int i=0; i<=int(sides_to_split.size()); ++i) {
|
||||
tr.children[i] = m_triangles.size()-1-i;
|
||||
m_triangles[tr.children[i]].parent = facet_idx;
|
||||
m_triangles[tr.children[i]].set_state(old_type);
|
||||
tr->children[i] = m_triangles.size()-1-i;
|
||||
m_triangles[tr->children[i]].parent = facet_idx;
|
||||
m_triangles[tr->children[i]].set_state(old_type);
|
||||
}
|
||||
|
||||
return true;
|
||||
|
@ -1097,7 +1099,10 @@ bool TriangleSelector::is_edge_inside_cursor(int facet_idx) const
|
|||
Vec3f s = (b-a).normalized();
|
||||
float t = (p-a).dot(s);
|
||||
Vec3f vector = a+t*s - p;
|
||||
float dist_sqr = vector.squaredNorm();
|
||||
|
||||
// vector is 3D vector from center to the intersection. What we want to
|
||||
// measure is length of its projection onto plane perpendicular to dir.
|
||||
float dist_sqr = vector.squaredNorm() - std::pow(vector.dot(m_cursor.dir), 2.f);
|
||||
if (dist_sqr < m_cursor.radius_sqr && t>=0.f && t<=(b-a).norm())
|
||||
return true;
|
||||
}
|
||||
|
@ -1117,33 +1122,47 @@ void TriangleSelector::undivide_triangle(int facet_idx)
|
|||
undivide_triangle(tr.children[i]);
|
||||
m_triangles[tr.children[i]].valid = false;
|
||||
}
|
||||
}
|
||||
|
||||
tr.set_division(0); // not split
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void TriangleSelector::remove_if_needless(int child_facet)
|
||||
void TriangleSelector::remove_useless_children(int facet_idx)
|
||||
{
|
||||
if (m_triangles[child_facet].is_split() || ! m_triangles[child_facet].valid)
|
||||
// Check that all children are leafs of the same type. If not, try to
|
||||
// make them (recursive call). Remove them if sucessful.
|
||||
|
||||
assert(facet_idx < int(m_triangles.size()) && m_triangles[facet_idx].valid);
|
||||
Triangle& tr = m_triangles[facet_idx];
|
||||
|
||||
if (! tr.is_split()) {
|
||||
// This is a leaf, there nothing to do. This can happen during the
|
||||
// first (non-recursive call). Shouldn't otherwise.
|
||||
return;
|
||||
int parent = m_triangles[child_facet].parent;
|
||||
if (parent == -1)
|
||||
return; // root
|
||||
FacetSupportType child_type = m_triangles[child_facet].get_state();
|
||||
}
|
||||
|
||||
// Check type of all valid children, if they're same, they are needless.
|
||||
for (int i=0; i<=m_triangles[parent].number_of_split_sides(); ++i)
|
||||
if (m_triangles[m_triangles[parent].children[i]].is_split()
|
||||
|| m_triangles[m_triangles[parent].children[i]].get_state() != child_type)
|
||||
return; // not all children are the same
|
||||
// Call this for all non-leaf children.
|
||||
for (int child_idx=0; child_idx<=tr.number_of_split_sides(); ++child_idx) {
|
||||
assert(child_idx < int(m_triangles.size()) && m_triangles[child_idx].valid);
|
||||
if (m_triangles[tr.children[child_idx]].is_split())
|
||||
remove_useless_children(tr.children[child_idx]);
|
||||
}
|
||||
|
||||
// All children are the same, kill them.
|
||||
undivide_triangle(parent);
|
||||
m_triangles[parent].set_state(child_type);
|
||||
|
||||
// And now try the same for parent (which has just become leaf).
|
||||
remove_if_needless(parent);
|
||||
// Return if a child is not leaf or two children differ in type.
|
||||
FacetSupportType first_child_type;
|
||||
for (int child_idx=0; child_idx<=tr.number_of_split_sides(); ++child_idx) {
|
||||
if (m_triangles[tr.children[child_idx]].is_split())
|
||||
return;
|
||||
if (child_idx == 0)
|
||||
first_child_type = m_triangles[tr.children[0]].get_state();
|
||||
else if (m_triangles[tr.children[child_idx]].get_state() != first_child_type)
|
||||
return;
|
||||
}
|
||||
|
||||
// If we got here, the children can be removed.
|
||||
undivide_triangle(facet_idx);
|
||||
tr.set_state(first_child_type);
|
||||
}
|
||||
|
||||
|
||||
|
@ -1158,33 +1177,12 @@ TriangleSelector::TriangleSelector(const TriangleMesh& mesh)
|
|||
m_mesh = &mesh;
|
||||
}
|
||||
|
||||
|
||||
void TriangleSelector::render() const
|
||||
void TriangleSelector::render(ImGuiWrapper* imgui)
|
||||
{
|
||||
::glColor3f(0.f, 0.f, 1.f);
|
||||
::glPolygonMode( GL_FRONT_AND_BACK, GL_LINE );
|
||||
|
||||
Vec3d offset = wxGetApp().model().objects.front()->instances.front()->get_transformation().get_offset();
|
||||
::glTranslatef(offset.x(), offset.y(), offset.z());
|
||||
::glScalef(1.01f, 1.01f, 1.01f);
|
||||
::glScalef(1.005f, 1.005f, 1.005f);
|
||||
|
||||
::glBegin( GL_TRIANGLES);
|
||||
for (int tr_id=0; tr_id<int(m_triangles.size()); ++tr_id) {
|
||||
const Triangle& tr = m_triangles[tr_id];
|
||||
if (! tr.valid)
|
||||
continue;
|
||||
|
||||
if (tr_id == m_orig_size_indices-1)
|
||||
::glColor3f(1.f, 0.f, 0.f);
|
||||
|
||||
for (int i=0; i<3; ++i)
|
||||
::glVertex3f(m_vertices[tr.verts_idxs[i]][0],
|
||||
m_vertices[tr.verts_idxs[i]][1],
|
||||
m_vertices[tr.verts_idxs[i]][2]);
|
||||
}
|
||||
::glEnd();
|
||||
|
||||
::glPolygonMode( GL_FRONT_AND_BACK, GL_FILL );
|
||||
|
||||
::glBegin( GL_TRIANGLES);
|
||||
for (const Triangle& tr : m_triangles) {
|
||||
|
@ -1202,7 +1200,57 @@ void TriangleSelector::render() const
|
|||
m_vertices[tr.verts_idxs[i]][2]);
|
||||
}
|
||||
::glEnd();
|
||||
|
||||
#ifdef PRUSASLICER_TRIANGLE_SELECTOR_DEBUG
|
||||
if (imgui)
|
||||
render_debug(imgui);
|
||||
else
|
||||
assert(false); // If you want debug output, pass ptr to ImGuiWrapper.
|
||||
#endif
|
||||
}
|
||||
|
||||
#ifdef PRUSASLICER_TRIANGLE_SELECTOR_DEBUG
|
||||
void TriangleSelector::render_debug(ImGuiWrapper* imgui)
|
||||
{
|
||||
imgui->begin(std::string("TriangleSelector dialog (DEV ONLY)"),
|
||||
ImGuiWindowFlags_AlwaysAutoResize | ImGuiWindowFlags_NoCollapse);
|
||||
static float edge_limit = 1.f;
|
||||
imgui->text("Edge limit (mm): ");
|
||||
imgui->slider_float("", &edge_limit, 0.1f, 8.f);
|
||||
set_edge_limit(edge_limit);
|
||||
imgui->checkbox("Show triangles: ", m_show_triangles);
|
||||
|
||||
int valid_triangles = std::count_if(m_triangles.begin(), m_triangles.end(),
|
||||
[](const Triangle& tr) { return tr.valid; });
|
||||
imgui->text("Valid triangles: " + std::to_string(valid_triangles) +
|
||||
"/" + std::to_string(m_triangles.size()));
|
||||
imgui->text("Number of vertices: " + std::to_string(m_vertices.size()));
|
||||
|
||||
imgui->end();
|
||||
|
||||
if (m_show_triangles) {
|
||||
::glColor3f(0.f, 0.f, 1.f);
|
||||
::glPolygonMode( GL_FRONT_AND_BACK, GL_LINE );
|
||||
|
||||
::glBegin( GL_TRIANGLES);
|
||||
for (int tr_id=0; tr_id<int(m_triangles.size()); ++tr_id) {
|
||||
const Triangle& tr = m_triangles[tr_id];
|
||||
if (! tr.valid)
|
||||
continue;
|
||||
|
||||
if (tr_id == m_orig_size_indices-1)
|
||||
::glColor3f(1.f, 0.f, 0.f);
|
||||
|
||||
for (int i=0; i<3; ++i)
|
||||
::glVertex3f(m_vertices[tr.verts_idxs[i]][0],
|
||||
m_vertices[tr.verts_idxs[i]][1],
|
||||
m_vertices[tr.verts_idxs[i]][2]);
|
||||
}
|
||||
::glEnd();
|
||||
::glPolygonMode( GL_FRONT_AND_BACK, GL_FILL );
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
} // namespace GUI
|
||||
} // namespace Slic3r
|
||||
|
|
|
@ -10,6 +10,9 @@
|
|||
#include <cereal/types/vector.hpp>
|
||||
|
||||
|
||||
#define PRUSASLICER_TRIANGLE_SELECTOR_DEBUG
|
||||
|
||||
|
||||
namespace Slic3r {
|
||||
|
||||
enum class FacetSupportType : int8_t;
|
||||
|
@ -25,7 +28,7 @@ class ClippingPlane;
|
|||
class TriangleSelector {
|
||||
public:
|
||||
void set_edge_limit(float edge_limit) { m_edge_limit_sqr = std::pow(edge_limit, 2.f); }
|
||||
void render() const;
|
||||
|
||||
// Create new object on a TriangleMesh. The referenced mesh must
|
||||
// stay valid, a ptr to it is saved and used.
|
||||
explicit TriangleSelector(const TriangleMesh& mesh);
|
||||
|
@ -40,10 +43,19 @@ public:
|
|||
|
||||
void unselect_all();
|
||||
|
||||
// Render current selection. Transformation matrices are supposed
|
||||
// to be already set.
|
||||
void render(ImGuiWrapper* imgui = nullptr);
|
||||
|
||||
// Remove all unnecessary data (such as vertices that are not needed
|
||||
// because the selection has been made larger.
|
||||
void garbage_collect();
|
||||
|
||||
#ifdef PRUSASLICER_TRIANGLE_SELECTOR_DEBUG
|
||||
void render_debug(ImGuiWrapper* imgui);
|
||||
bool m_show_triangles{true};
|
||||
#endif
|
||||
|
||||
private:
|
||||
// Triangle and info about how it's split.
|
||||
struct Triangle {
|
||||
|
@ -112,13 +124,13 @@ private:
|
|||
|
||||
// Private functions:
|
||||
bool select_triangle(int facet_idx, FacetSupportType type,
|
||||
bool cursor_inside = false);
|
||||
bool recursive_call = false);
|
||||
bool is_point_inside_cursor(const Vec3f& point) const;
|
||||
int vertices_inside(int facet_idx) const;
|
||||
bool faces_camera(int facet) const;
|
||||
void undivide_triangle(int facet_idx);
|
||||
bool split_triangle(int facet_idx);
|
||||
void remove_if_needless(int child_facet);
|
||||
void remove_useless_children(int facet_idx); // No hidden meaning. Triangles are meant.
|
||||
bool is_pointer_in_triangle(int facet_idx) const;
|
||||
bool is_edge_inside_cursor(int facet_idx) const;
|
||||
};
|
||||
|
|
Loading…
Reference in a new issue