mirror of
https://git.mirrors.martin98.com/https://github.com/prusa3d/PrusaSlicer.git
synced 2025-08-04 15:50:37 +08:00
83 lines
3.8 KiB
C++
83 lines
3.8 KiB
C++
#include "../ClipperUtils.hpp"
|
|
#include "../ShortestPath.hpp"
|
|
#include "../Arachne/WallToolPaths.hpp"
|
|
|
|
#include "FillEnsuring.hpp"
|
|
|
|
#include <boost/log/trivial.hpp>
|
|
|
|
namespace Slic3r {
|
|
|
|
ThickPolylines FillEnsuring::fill_surface_arachne(const Surface *surface, const FillParams ¶ms)
|
|
{
|
|
assert(params.use_arachne);
|
|
assert(this->print_config != nullptr && this->print_object_config != nullptr && this->print_region_config != nullptr);
|
|
|
|
const coord_t scaled_spacing = scaled<coord_t>(this->spacing);
|
|
|
|
// Perform offset.
|
|
Slic3r::ExPolygons expp = this->overlap != 0. ? offset_ex(surface->expolygon, scaled<float>(this->overlap)) : ExPolygons{surface->expolygon};
|
|
// Create the infills for each of the regions.
|
|
ThickPolylines thick_polylines_out;
|
|
for (ExPolygon &ex_poly : expp) {
|
|
Point bbox_size = ex_poly.contour.bounding_box().size();
|
|
coord_t loops_count = std::max(bbox_size.x(), bbox_size.y()) / scaled_spacing + 1;
|
|
Polygons polygons = to_polygons(ex_poly);
|
|
Arachne::WallToolPaths wall_tool_paths(polygons, scaled_spacing, scaled_spacing, loops_count, 0, params.layer_height, *this->print_object_config, *this->print_config);
|
|
if (std::vector<Arachne::VariableWidthLines> loops = wall_tool_paths.getToolPaths(); !loops.empty()) {
|
|
std::vector<const Arachne::ExtrusionLine *> all_extrusions;
|
|
for (Arachne::VariableWidthLines &loop : loops) {
|
|
if (loop.empty())
|
|
continue;
|
|
for (const Arachne::ExtrusionLine &wall : loop)
|
|
all_extrusions.emplace_back(&wall);
|
|
}
|
|
|
|
// Split paths using a nearest neighbor search.
|
|
size_t firts_poly_idx = thick_polylines_out.size();
|
|
Point last_pos(0, 0);
|
|
for (const Arachne::ExtrusionLine *extrusion : all_extrusions) {
|
|
if (extrusion->empty())
|
|
continue;
|
|
|
|
ThickPolyline thick_polyline = Arachne::to_thick_polyline(*extrusion);
|
|
if (thick_polyline.length() == 0.)
|
|
//FIXME this should not happen.
|
|
continue;
|
|
assert(thick_polyline.size() > 1);
|
|
assert(thick_polyline.length() > 0.);
|
|
//assert(thick_polyline.points.size() == thick_polyline.width.size());
|
|
if (extrusion->is_closed)
|
|
thick_polyline.start_at_index(nearest_point_index(thick_polyline.points, last_pos));
|
|
|
|
assert(thick_polyline.size() > 1);
|
|
//assert(thick_polyline.points.size() == thick_polyline.width.size());
|
|
thick_polylines_out.emplace_back(std::move(thick_polyline));
|
|
last_pos = thick_polylines_out.back().last_point();
|
|
}
|
|
|
|
// clip the paths to prevent the extruder from getting exactly on the first point of the loop
|
|
// Keep valid paths only.
|
|
size_t j = firts_poly_idx;
|
|
for (size_t i = firts_poly_idx; i < thick_polylines_out.size(); ++i) {
|
|
assert(thick_polylines_out[i].size() > 1);
|
|
assert(thick_polylines_out[i].length() > 0.);
|
|
//assert(thick_polylines_out[i].points.size() == thick_polylines_out[i].width.size());
|
|
thick_polylines_out[i].clip_end(this->loop_clipping);
|
|
assert(thick_polylines_out[i].size() > 1);
|
|
if (thick_polylines_out[i].is_valid()) {
|
|
if (j < i)
|
|
thick_polylines_out[j] = std::move(thick_polylines_out[i]);
|
|
++j;
|
|
}
|
|
}
|
|
if (j < thick_polylines_out.size())
|
|
thick_polylines_out.erase(thick_polylines_out.begin() + int(j), thick_polylines_out.end());
|
|
}
|
|
}
|
|
|
|
return thick_polylines_out;
|
|
}
|
|
|
|
} // namespace Slic3r
|