From 4cca6f8724321e1247fd7c5ba92bada7a561cc51 Mon Sep 17 00:00:00 2001 From: Lukas Matena Date: Wed, 14 Aug 2019 15:44:32 +0200 Subject: [PATCH] Allowed the skirt to intersect brim (https://github.com/prusa3d/PrusaSlicer/issues/724) Brim lines are split at the intersection so there is no overextrusion. --- src/libslic3r/GCode.cpp | 10 +++---- src/libslic3r/Print.cpp | 66 ++++++++++++++++++++++++++++++++++++++--- 2 files changed, 67 insertions(+), 9 deletions(-) diff --git a/src/libslic3r/GCode.cpp b/src/libslic3r/GCode.cpp index b691203c9..96aa60c25 100644 --- a/src/libslic3r/GCode.cpp +++ b/src/libslic3r/GCode.cpp @@ -1752,8 +1752,9 @@ void GCode::process_layer( if (! m_brim_done) { this->set_origin(0., 0.); m_avoid_crossing_perimeters.use_external_mp = true; - for (const ExtrusionEntity *ee : print.brim().entities) - gcode += this->extrude_loop(*dynamic_cast(ee), "brim", m_config.support_material_speed.value); + for (const ExtrusionEntity *ee : print.brim().entities) { + gcode += this->extrude_entity(*ee, "brim", m_config.support_material_speed.value); + } m_brim_done = true; m_avoid_crossing_perimeters.use_external_mp = false; // Allow a straight travel move to the first object point. @@ -2419,10 +2420,9 @@ std::string GCode::extrude_entity(const ExtrusionEntity &entity, std::string des return this->extrude_multi_path(*multipath, description, speed); else if (const ExtrusionLoop* loop = dynamic_cast(&entity)) return this->extrude_loop(*loop, description, speed, lower_layer_edge_grid); - else { + else throw std::invalid_argument("Invalid argument supplied to extrude()"); - return ""; - } + return ""; } std::string GCode::extrude_path(ExtrusionPath path, std::string description, double speed) diff --git a/src/libslic3r/Print.cpp b/src/libslic3r/Print.cpp index e376f54c3..ddd99a3ee 100644 --- a/src/libslic3r/Print.cpp +++ b/src/libslic3r/Print.cpp @@ -1603,9 +1603,7 @@ void Print::_make_skirt() // Initial offset of the brim inner edge from the object (possible with a support & raft). // The skirt will touch the brim if the brim is extruded. - Flow brim_flow = this->brim_flow(); - double actual_brim_width = brim_flow.spacing() * floor(m_config.brim_width.value / brim_flow.spacing()); - auto distance = float(scale_(std::max(m_config.skirt_distance.value, actual_brim_width) - spacing/2.)); + auto distance = float(scale_(m_config.skirt_distance.value) - spacing/2.); // Draw outlines from outside to inside. // Loop while we have less skirts than required or any extruder hasn't reached the min length if any. std::vector extruded_length(extruders.size(), 0.); @@ -1687,11 +1685,71 @@ void Print::_make_brim() } polygons_append(loops, offset(islands, -0.5f * float(flow.scaled_spacing()))); } - loops = union_pt_chained(loops, false); // The function above produces ordering well suited for concentric infill (from outside to inside). // For Brim, the ordering should be reversed (from inside to outside). std::reverse(loops.begin(), loops.end()); + + // If there is a possibility that brim intersects skirt, go through loops and split those extrusions + // The result is either the original Polygon or a list of Polylines + if (! m_skirt.empty() && m_config.skirt_distance.value < m_config.brim_width) + { + // Find the bounding polygons of the skirt + const Polygons skirt_inners = offset(dynamic_cast(m_skirt.entities.back())->polygon(), + -float(scale_(this->skirt_flow().spacing()))/2.f, + ClipperLib::jtRound, + float(scale_(0.1))); + const Polygons skirt_outers = offset(dynamic_cast(m_skirt.entities.front())->polygon(), + float(scale_(this->skirt_flow().spacing()))/2.f, + ClipperLib::jtRound, + float(scale_(0.1))); + + const Polygon& skirt_inner = skirt_inners.front(); + const Polygon& skirt_outer = skirt_outers.front(); + + for (size_t i=0; i 0) { + Line line(*last_pt, *pt); + bool inner = skirt_inner.first_intersection(line, &intersections[0]); + bool outer = skirt_outer.first_intersection(line, &intersections[1]); + del = del || inner || outer; + if (inner != outer) {// there is exactly one intersection + lines.back().append(inner ? intersections[0] : intersections[1]); + } + if (inner && outer) { + int nearest_idx = pt->nearest_point_index(intersections); + lines.back().append(intersections[! nearest_idx]); + lines.push_back(Polyline()); + lines.back().append(intersections[nearest_idx]); + } + } + if (valid_point) + lines.back().append(*pt); + else { + del = true; + lines.push_back(Polyline()); + } + } + // If we found a single intersection, we should erase the respective ExtrusionLoop + // and append the polylines that we created. + if (del) { + loops.erase(loops.begin() + (i--)); + extrusion_entities_append_paths(m_brim.entities, std::move(lines), erSkirt, float(flow.mm3_per_mm()), float(flow.width), float(this->skirt_first_layer_height())); + } + } + } + extrusion_entities_append_loops(m_brim.entities, std::move(loops), erSkirt, float(flow.mm3_per_mm()), float(flow.width), float(this->skirt_first_layer_height())); }