mirror of
https://github.com/SoftFever/OrcaSlicer.git
synced 2025-10-24 09:11:23 -06:00
Bugfixes of the new adaptive elephant foot compensation.
This commit is contained in:
parent
bb8d59391f
commit
4c735192ef
3 changed files with 34 additions and 23 deletions
|
@ -951,6 +951,7 @@ ClipperLib::Paths fix_after_inner_offset(const ClipperLib::Path &input, ClipperL
|
||||||
ClipperLib::Path mittered_offset_path_scaled(const Points &contour, const std::vector<float> &deltas, double miter_limit)
|
ClipperLib::Path mittered_offset_path_scaled(const Points &contour, const std::vector<float> &deltas, double miter_limit)
|
||||||
{
|
{
|
||||||
assert(contour.size() == deltas.size());
|
assert(contour.size() == deltas.size());
|
||||||
|
|
||||||
#ifndef NDEBUG
|
#ifndef NDEBUG
|
||||||
// Verify that the deltas are either all positive, or all negative.
|
// Verify that the deltas are either all positive, or all negative.
|
||||||
bool positive = false;
|
bool positive = false;
|
||||||
|
@ -986,7 +987,10 @@ ClipperLib::Path mittered_offset_path_scaled(const Points &contour, const std::v
|
||||||
double lmin = *std::max_element(deltas.begin(), deltas.end()) * CLIPPER_OFFSET_SHORTEST_EDGE_FACTOR;
|
double lmin = *std::max_element(deltas.begin(), deltas.end()) * CLIPPER_OFFSET_SHORTEST_EDGE_FACTOR;
|
||||||
double l2min = lmin * lmin;
|
double l2min = lmin * lmin;
|
||||||
// Minimum angle to consider two edges to be parallel.
|
// Minimum angle to consider two edges to be parallel.
|
||||||
double sin_min_parallel = EPSILON + 1. / double(CLIPPER_OFFSET_SCALE);
|
// Vojtech's estimate.
|
||||||
|
// const double sin_min_parallel = EPSILON + 1. / double(CLIPPER_OFFSET_SCALE);
|
||||||
|
// Implementation equal to Clipper.
|
||||||
|
const double sin_min_parallel = 1.;
|
||||||
|
|
||||||
// Find the last point further from pt by l2min.
|
// Find the last point further from pt by l2min.
|
||||||
Vec2d pt = contour.front().cast<double>();
|
Vec2d pt = contour.front().cast<double>();
|
||||||
|
@ -1012,8 +1016,12 @@ ClipperLib::Path mittered_offset_path_scaled(const Points &contour, const std::v
|
||||||
if (l2 > l2min)
|
if (l2 > l2min)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (j > ilast)
|
if (j > ilast) {
|
||||||
|
assert(i <= ilast);
|
||||||
|
// If the last edge is too short, merge it with the previous edge.
|
||||||
|
i = ilast;
|
||||||
ptnext = contour.front().cast<double>();
|
ptnext = contour.front().cast<double>();
|
||||||
|
}
|
||||||
|
|
||||||
// Normal to the (ptnext - pt) segment.
|
// Normal to the (ptnext - pt) segment.
|
||||||
Vec2d nnext = perp(ptnext - pt).normalized();
|
Vec2d nnext = perp(ptnext - pt).normalized();
|
||||||
|
@ -1026,12 +1034,13 @@ ClipperLib::Path mittered_offset_path_scaled(const Points &contour, const std::v
|
||||||
add_offset_point(pt + nprev * delta);
|
add_offset_point(pt + nprev * delta);
|
||||||
add_offset_point(pt);
|
add_offset_point(pt);
|
||||||
add_offset_point(pt + nnext * delta);
|
add_offset_point(pt + nnext * delta);
|
||||||
} else if (convex < sin_min_parallel) {
|
} else {
|
||||||
|
double dot = nprev.dot(nnext);
|
||||||
|
if (convex < sin_min_parallel && dot > 0.) {
|
||||||
// Nearly parallel.
|
// Nearly parallel.
|
||||||
add_offset_point((nprev.dot(nnext) > 0.) ? (pt + nprev * delta) : pt);
|
add_offset_point((nprev.dot(nnext) > 0.) ? (pt + nprev * delta) : pt);
|
||||||
} else {
|
} else {
|
||||||
// Convex corner
|
// Convex corner, possibly extremely sharp if convex < sin_min_parallel.
|
||||||
double dot = nprev.dot(nnext);
|
|
||||||
double r = 1. + dot;
|
double r = 1. + dot;
|
||||||
if (r >= miter_limit)
|
if (r >= miter_limit)
|
||||||
add_offset_point(pt + (nprev + nnext) * (delta / r));
|
add_offset_point(pt + (nprev + nnext) * (delta / r));
|
||||||
|
@ -1042,12 +1051,13 @@ ClipperLib::Path mittered_offset_path_scaled(const Points &contour, const std::v
|
||||||
#ifndef NDEBUG
|
#ifndef NDEBUG
|
||||||
Vec2d vedge = 0.5 * (newpt1 + newpt2) - pt;
|
Vec2d vedge = 0.5 * (newpt1 + newpt2) - pt;
|
||||||
double dist_norm = vedge.norm();
|
double dist_norm = vedge.norm();
|
||||||
assert(std::abs(dist_norm - delta) < EPSILON);
|
assert(std::abs(dist_norm - std::abs(delta)) < SCALED_EPSILON);
|
||||||
#endif /* NDEBUG */
|
#endif /* NDEBUG */
|
||||||
add_offset_point(newpt1);
|
add_offset_point(newpt1);
|
||||||
add_offset_point(newpt2);
|
add_offset_point(newpt2);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (i == ilast)
|
if (i == ilast)
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -88,7 +88,7 @@ std::vector<float> contour_distance(const EdgeGrid::Grid &grid, const size_t idx
|
||||||
|
|
||||||
if (std::abs(denom) >= EPSILON) {
|
if (std::abs(denom) >= EPSILON) {
|
||||||
double t = cross2(dir2, vptpt2) / denom;
|
double t = cross2(dir2, vptpt2) / denom;
|
||||||
assert(t > 0. && t <= 1.);
|
assert(t > - EPSILON && t < 1. + EPSILON);
|
||||||
bool this_valid = true;
|
bool this_valid = true;
|
||||||
if (it_contour_and_segment->first == idx_contour) {
|
if (it_contour_and_segment->first == idx_contour) {
|
||||||
// The intersected segment originates from the same contour as the starting point.
|
// The intersected segment originates from the same contour as the starting point.
|
||||||
|
@ -105,7 +105,7 @@ std::vector<float> contour_distance(const EdgeGrid::Grid &grid, const size_t idx
|
||||||
auto it = std::lower_bound(resampled_point_parameters.begin(), resampled_point_parameters.end(), key, lower);
|
auto it = std::lower_bound(resampled_point_parameters.begin(), resampled_point_parameters.end(), key, lower);
|
||||||
assert(it != resampled_point_parameters.end() && it->idx_src == ipt && ! it->interpolated);
|
assert(it != resampled_point_parameters.end() && it->idx_src == ipt && ! it->interpolated);
|
||||||
double t2 = cross2(dir, vptpt2) / denom;
|
double t2 = cross2(dir, vptpt2) / denom;
|
||||||
assert(t2 >= 0. && t2 <= 1.);
|
assert(t2 > - EPSILON && t2 < 1. + EPSILON);
|
||||||
if (++ ipt == ipts.size())
|
if (++ ipt == ipts.size())
|
||||||
param_hi = t2 * dir2.norm();
|
param_hi = t2 * dir2.norm();
|
||||||
else
|
else
|
||||||
|
|
|
@ -1819,11 +1819,12 @@ end:
|
||||||
if (delta < 0.f || elephant_foot_compensation > 0.f) {
|
if (delta < 0.f || elephant_foot_compensation > 0.f) {
|
||||||
// Apply the negative XY compensation.
|
// Apply the negative XY compensation.
|
||||||
Polygons trimming;
|
Polygons trimming;
|
||||||
|
static const float eps = float(scale_(m_config.slice_closing_radius.value) * 1.5);
|
||||||
if (elephant_foot_compensation > 0.f) {
|
if (elephant_foot_compensation > 0.f) {
|
||||||
trimming = to_polygons(Slic3r::elephant_foot_compensation(offset_ex(layer->merged(float(EPSILON)), std::min(delta, 0.f) - float(EPSILON)),
|
trimming = to_polygons(Slic3r::elephant_foot_compensation(offset_ex(layer->merged(eps), std::min(delta, 0.f) - eps),
|
||||||
layer->m_regions.front()->flow(frExternalPerimeter), unscale<double>(elephant_foot_compensation)));
|
layer->m_regions.front()->flow(frExternalPerimeter), unscale<double>(elephant_foot_compensation)));
|
||||||
} else
|
} else
|
||||||
trimming = offset(layer->merged(float(EPSILON)), delta - float(EPSILON));
|
trimming = offset(layer->merged(float(SCALED_EPSILON)), delta - float(SCALED_EPSILON));
|
||||||
for (size_t region_id = 0; region_id < layer->m_regions.size(); ++ region_id)
|
for (size_t region_id = 0; region_id < layer->m_regions.size(); ++ region_id)
|
||||||
layer->m_regions[region_id]->trim_surfaces(trimming);
|
layer->m_regions[region_id]->trim_surfaces(trimming);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue