mirror of
https://github.com/SoftFever/OrcaSlicer.git
synced 2025-10-24 01:01:15 -06:00
Merge branch 'master' of https://github.com/prusa3d/Slic3r
This commit is contained in:
commit
0f96213dce
9 changed files with 183 additions and 86 deletions
|
@ -163,7 +163,7 @@ public:
|
||||||
{
|
{
|
||||||
dir_ = OptDir::MIN;
|
dir_ = OptDir::MIN;
|
||||||
return static_cast<Subclass*>(this)->template optimize<Func, Args...>(
|
return static_cast<Subclass*>(this)->template optimize<Func, Args...>(
|
||||||
objectfunction, initvals, Bound<Args>()... );
|
forward<Func>(objectfunction), initvals, Bound<Args>()... );
|
||||||
}
|
}
|
||||||
|
|
||||||
template<class...Args, class Func>
|
template<class...Args, class Func>
|
||||||
|
@ -171,7 +171,7 @@ public:
|
||||||
{
|
{
|
||||||
dir_ = OptDir::MIN;
|
dir_ = OptDir::MIN;
|
||||||
return static_cast<Subclass*>(this)->template optimize<Func, Args...>(
|
return static_cast<Subclass*>(this)->template optimize<Func, Args...>(
|
||||||
objectfunction,
|
forward<Func>(objectfunction),
|
||||||
Input<Args...>(),
|
Input<Args...>(),
|
||||||
Bound<Args>()... );
|
Bound<Args>()... );
|
||||||
}
|
}
|
||||||
|
@ -184,7 +184,7 @@ public:
|
||||||
{
|
{
|
||||||
dir_ = OptDir::MAX;
|
dir_ = OptDir::MAX;
|
||||||
return static_cast<Subclass*>(this)->template optimize<Func, Args...>(
|
return static_cast<Subclass*>(this)->template optimize<Func, Args...>(
|
||||||
objectfunction, initvals, bounds... );
|
forward<Func>(objectfunction), initvals, bounds... );
|
||||||
}
|
}
|
||||||
|
|
||||||
template<class Func, class...Args>
|
template<class Func, class...Args>
|
||||||
|
@ -193,7 +193,7 @@ public:
|
||||||
{
|
{
|
||||||
dir_ = OptDir::MAX;
|
dir_ = OptDir::MAX;
|
||||||
return static_cast<Subclass*>(this)->template optimize<Func, Args...>(
|
return static_cast<Subclass*>(this)->template optimize<Func, Args...>(
|
||||||
objectfunction, initvals, Bound<Args>()... );
|
forward<Func>(objectfunction), initvals, Bound<Args>()... );
|
||||||
}
|
}
|
||||||
|
|
||||||
template<class...Args, class Func>
|
template<class...Args, class Func>
|
||||||
|
@ -201,7 +201,7 @@ public:
|
||||||
{
|
{
|
||||||
dir_ = OptDir::MAX;
|
dir_ = OptDir::MAX;
|
||||||
return static_cast<Subclass*>(this)->template optimize<Func, Args...>(
|
return static_cast<Subclass*>(this)->template optimize<Func, Args...>(
|
||||||
objectfunction,
|
forward<Func>(objectfunction),
|
||||||
Input<Args...>(),
|
Input<Args...>(),
|
||||||
Bound<Args>()... );
|
Bound<Args>()... );
|
||||||
}
|
}
|
||||||
|
|
|
@ -2731,14 +2731,14 @@ void PrintConfigDef::init_sla_params()
|
||||||
def->min = 0;
|
def->min = 0;
|
||||||
def->default_value = new ConfigOptionFloat(1.0);
|
def->default_value = new ConfigOptionFloat(1.0);
|
||||||
|
|
||||||
def = this->add("pad_wall_tilt", coFloat);
|
def = this->add("pad_wall_slope", coFloat);
|
||||||
def->label = L("Pad wall tilt");
|
def->label = L("Pad wall slope");
|
||||||
def->category = L("Pad");
|
def->category = L("Pad");
|
||||||
def->tooltip = L("The tilt of the pad wall relative to the bed plane. "
|
def->tooltip = L("The slope of the pad wall relative to the bed plane. "
|
||||||
"90 degrees means straight walls.");
|
"90 degrees means straight walls.");
|
||||||
def->sidetext = L("degrees");
|
def->sidetext = L("degrees");
|
||||||
def->cli = "";
|
def->cli = "";
|
||||||
def->min = 0.1; // What should be the minimum?
|
def->min = 45;
|
||||||
def->max = 90;
|
def->max = 90;
|
||||||
def->default_value = new ConfigOptionFloat(45.0);
|
def->default_value = new ConfigOptionFloat(45.0);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1028,8 +1028,8 @@ public:
|
||||||
// The smoothing radius of the pad edges
|
// The smoothing radius of the pad edges
|
||||||
ConfigOptionFloat pad_edge_radius /*= 1*/;
|
ConfigOptionFloat pad_edge_radius /*= 1*/;
|
||||||
|
|
||||||
// The tilt of the pad wall...
|
// The slope of the pad wall...
|
||||||
ConfigOptionFloat pad_wall_tilt;
|
ConfigOptionFloat pad_wall_slope;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void initialize(StaticCacheBase &cache, const char *base_ptr)
|
void initialize(StaticCacheBase &cache, const char *base_ptr)
|
||||||
|
@ -1056,7 +1056,7 @@ protected:
|
||||||
OPT_PTR(pad_wall_height);
|
OPT_PTR(pad_wall_height);
|
||||||
OPT_PTR(pad_max_merge_distance);
|
OPT_PTR(pad_max_merge_distance);
|
||||||
OPT_PTR(pad_edge_radius);
|
OPT_PTR(pad_edge_radius);
|
||||||
OPT_PTR(pad_wall_tilt);
|
OPT_PTR(pad_wall_slope);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -601,8 +601,8 @@ void create_base_pool(const ExPolygons &ground_layer, TriangleMesh& out,
|
||||||
const double thickness = cfg.min_wall_thickness_mm;
|
const double thickness = cfg.min_wall_thickness_mm;
|
||||||
const double wingheight = cfg.min_wall_height_mm;
|
const double wingheight = cfg.min_wall_height_mm;
|
||||||
const double fullheight = wingheight + thickness;
|
const double fullheight = wingheight + thickness;
|
||||||
const double tilt = cfg.wall_tilt;
|
const double slope = cfg.wall_slope;
|
||||||
const double wingdist = wingheight / std::tan(tilt);
|
const double wingdist = wingheight / std::tan(slope);
|
||||||
|
|
||||||
// scaled values
|
// scaled values
|
||||||
const coord_t s_thickness = mm(thickness);
|
const coord_t s_thickness = mm(thickness);
|
||||||
|
|
|
@ -28,17 +28,17 @@ struct PoolConfig {
|
||||||
double min_wall_height_mm = 5;
|
double min_wall_height_mm = 5;
|
||||||
double max_merge_distance_mm = 50;
|
double max_merge_distance_mm = 50;
|
||||||
double edge_radius_mm = 1;
|
double edge_radius_mm = 1;
|
||||||
double wall_tilt = std::atan(1.0); // Universal constant for Pi/4
|
double wall_slope = std::atan(1.0); // Universal constant for Pi/4
|
||||||
|
|
||||||
ThrowOnCancel throw_on_cancel = [](){};
|
ThrowOnCancel throw_on_cancel = [](){};
|
||||||
|
|
||||||
inline PoolConfig() {}
|
inline PoolConfig() {}
|
||||||
inline PoolConfig(double wt, double wh, double md, double er, double tilt):
|
inline PoolConfig(double wt, double wh, double md, double er, double slope):
|
||||||
min_wall_thickness_mm(wt),
|
min_wall_thickness_mm(wt),
|
||||||
min_wall_height_mm(wh),
|
min_wall_height_mm(wh),
|
||||||
max_merge_distance_mm(md),
|
max_merge_distance_mm(md),
|
||||||
edge_radius_mm(er),
|
edge_radius_mm(er),
|
||||||
wall_tilt(tilt) {}
|
wall_slope(slope) {}
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Calculate the pool for the mesh for SLA printing
|
/// Calculate the pool for the mesh for SLA printing
|
||||||
|
|
|
@ -12,6 +12,7 @@
|
||||||
#include <libslic3r/ClipperUtils.hpp>
|
#include <libslic3r/ClipperUtils.hpp>
|
||||||
#include <libslic3r/Model.hpp>
|
#include <libslic3r/Model.hpp>
|
||||||
|
|
||||||
|
#include <libnest2d/optimizers/nlopt/simplex.hpp>
|
||||||
#include <boost/log/trivial.hpp>
|
#include <boost/log/trivial.hpp>
|
||||||
#include <tbb/parallel_for.h>
|
#include <tbb/parallel_for.h>
|
||||||
|
|
||||||
|
@ -594,7 +595,7 @@ double pinhead_mesh_intersect(const Vec3d& s,
|
||||||
double r_back,
|
double r_back,
|
||||||
double width,
|
double width,
|
||||||
const EigenMesh3D& m,
|
const EigenMesh3D& m,
|
||||||
unsigned samples = 8,
|
unsigned samples = 16,
|
||||||
double safety_distance = 0.001)
|
double safety_distance = 0.001)
|
||||||
{
|
{
|
||||||
// method based on:
|
// method based on:
|
||||||
|
@ -620,9 +621,20 @@ double pinhead_mesh_intersect(const Vec3d& s,
|
||||||
std::vector<double> phis(samples);
|
std::vector<double> phis(samples);
|
||||||
for(size_t i = 0; i < phis.size(); ++i) phis[i] = i*2*PI/phis.size();
|
for(size_t i = 0; i < phis.size(); ++i) phis[i] = i*2*PI/phis.size();
|
||||||
|
|
||||||
a(Z) = -(v(X)*a(X) + v(Y)*a(Y)) / v(Z);
|
// We have to address the case when the direction vector v (same as dir)
|
||||||
|
// is coincident with one of the world axes. In this case two of its
|
||||||
b = a.cross(v);
|
// components will be completely zero and one is 1.0. Our method becomes
|
||||||
|
// dangerous here due to division with zero. Instead, vector 'a' can be an
|
||||||
|
// element-wise rotated version of 'v'
|
||||||
|
auto chk1 = [] (double val) { return std::abs(std::abs(val) - 1) < 1e-20; };
|
||||||
|
if(chk1(v(X)) || chk1(v(Y)) || chk1(v(Z))) {
|
||||||
|
a = {v(Z), v(X), v(Y)};
|
||||||
|
b = {v(Y), v(Z), v(X)};
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
a(Z) = -(v(Y)*a(Y)) / v(Z); a.normalize();
|
||||||
|
b = a.cross(v);
|
||||||
|
}
|
||||||
|
|
||||||
// Now a and b vectors are perpendicular to v and to each other. Together
|
// Now a and b vectors are perpendicular to v and to each other. Together
|
||||||
// they define the plane where we have to iterate with the given angles
|
// they define the plane where we have to iterate with the given angles
|
||||||
|
@ -647,18 +659,14 @@ double pinhead_mesh_intersect(const Vec3d& s,
|
||||||
s(Z) + rpscos * a(Z) + rpssin * b(Z));
|
s(Z) + rpscos * a(Z) + rpssin * b(Z));
|
||||||
|
|
||||||
// Point ps is not on mesh but can be inside or outside as well. This
|
// Point ps is not on mesh but can be inside or outside as well. This
|
||||||
// would cause many problems with ray-casting. So we query the closest
|
// would cause many problems with ray-casting. To detect the position we
|
||||||
// point on the mesh to this.
|
// will use the ray-casting result (which has an is_inside predicate).
|
||||||
// auto psq = m.signed_distance(ps);
|
|
||||||
|
|
||||||
// This is the point on the circle on the back sphere
|
// This is the point on the circle on the back sphere
|
||||||
Vec3d p(c(X) + rpbcos * a(X) + rpbsin * b(X),
|
Vec3d p(c(X) + rpbcos * a(X) + rpbsin * b(X),
|
||||||
c(Y) + rpbcos * a(Y) + rpbsin * b(Y),
|
c(Y) + rpbcos * a(Y) + rpbsin * b(Y),
|
||||||
c(Z) + rpbcos * a(Z) + rpbsin * b(Z));
|
c(Z) + rpbcos * a(Z) + rpbsin * b(Z));
|
||||||
|
|
||||||
// Vec3d n = (p - psq.point_on_mesh()).normalized();
|
|
||||||
// phi = m.query_ray_hit(psq.point_on_mesh() + sd*n, n);
|
|
||||||
|
|
||||||
Vec3d n = (p - ps).normalized();
|
Vec3d n = (p - ps).normalized();
|
||||||
auto hr = m.query_ray_hit(ps + sd*n, n);
|
auto hr = m.query_ray_hit(ps + sd*n, n);
|
||||||
|
|
||||||
|
@ -693,8 +701,18 @@ double bridge_mesh_intersect(const Vec3d& s,
|
||||||
Vec3d a(0, 1, 0), b;
|
Vec3d a(0, 1, 0), b;
|
||||||
const double& sd = safety_distance;
|
const double& sd = safety_distance;
|
||||||
|
|
||||||
a(Z) = -(dir(X)*a(X) + dir(Y)*a(Y)) / dir(Z);
|
// INFO: for explanation of the method used here, see the previous method's
|
||||||
b = a.cross(dir);
|
// comments.
|
||||||
|
|
||||||
|
auto chk1 = [] (double val) { return std::abs(std::abs(val) - 1) < 1e-20; };
|
||||||
|
if(chk1(dir(X)) || chk1(dir(Y)) || chk1(dir(Z))) {
|
||||||
|
a = {dir(Z), dir(X), dir(Y)};
|
||||||
|
b = {dir(Y), dir(Z), dir(X)};
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
a(Z) = -(dir(Y)*a(Y)) / dir(Z); a.normalize();
|
||||||
|
b = a.cross(dir);
|
||||||
|
}
|
||||||
|
|
||||||
// circle portions
|
// circle portions
|
||||||
std::vector<double> phis(samples);
|
std::vector<double> phis(samples);
|
||||||
|
@ -1156,16 +1174,16 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
//...
|
//...
|
||||||
};
|
};
|
||||||
|
|
||||||
// t-hrow i-f c-ance-l-ed: It will be called many times so a shorthand will
|
// throw if canceled: It will be called many times so a shorthand will
|
||||||
// come in handy.
|
// come in handy.
|
||||||
auto& tifcl = ctl.cancelfn;
|
auto& thr = ctl.cancelfn;
|
||||||
|
|
||||||
// Filtering step: here we will discard inappropriate support points and
|
// Filtering step: here we will discard inappropriate support points and
|
||||||
// decide the future of the appropriate ones. We will check if a pinhead
|
// decide the future of the appropriate ones. We will check if a pinhead
|
||||||
// is applicable and adjust its angle at each support point.
|
// is applicable and adjust its angle at each support point.
|
||||||
// We will also merge the support points that are just too close and can be
|
// We will also merge the support points that are just too close and can be
|
||||||
// considered as one.
|
// considered as one.
|
||||||
auto filterfn = [tifcl] (
|
auto filterfn = [thr] (
|
||||||
const SupportConfig& cfg,
|
const SupportConfig& cfg,
|
||||||
const PointSet& points,
|
const PointSet& points,
|
||||||
const EigenMesh3D& mesh,
|
const EigenMesh3D& mesh,
|
||||||
|
@ -1179,9 +1197,9 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
// first one
|
// first one
|
||||||
auto aliases =
|
auto aliases =
|
||||||
cluster(points,
|
cluster(points,
|
||||||
[tifcl](const SpatElement& p, const SpatElement& se)
|
[thr](const SpatElement& p, const SpatElement& se)
|
||||||
{
|
{
|
||||||
tifcl();
|
thr();
|
||||||
return distance(p.first, se.first) < D_SP;
|
return distance(p.first, se.first) < D_SP;
|
||||||
}, 2);
|
}, 2);
|
||||||
|
|
||||||
|
@ -1192,10 +1210,10 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
filt_pts.row(count++) = points.row(a.front());
|
filt_pts.row(count++) = points.row(a.front());
|
||||||
}
|
}
|
||||||
|
|
||||||
tifcl();
|
thr();
|
||||||
|
|
||||||
// calculate the normals to the triangles belonging to filtered points
|
// calculate the normals to the triangles belonging to filtered points
|
||||||
auto nmls = sla::normals(filt_pts, mesh, cfg.head_front_radius_mm, tifcl);
|
auto nmls = sla::normals(filt_pts, mesh, cfg.head_front_radius_mm, thr);
|
||||||
|
|
||||||
head_norm.resize(count, 3);
|
head_norm.resize(count, 3);
|
||||||
head_pos.resize(count, 3);
|
head_pos.resize(count, 3);
|
||||||
|
@ -1207,9 +1225,15 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
// not be enough space for the pinhead. Filtering is applied for
|
// not be enough space for the pinhead. Filtering is applied for
|
||||||
// these reasons.
|
// these reasons.
|
||||||
|
|
||||||
|
using libnest2d::opt::bound;
|
||||||
|
using libnest2d::opt::initvals;
|
||||||
|
using libnest2d::opt::SimplexOptimizer;
|
||||||
|
using libnest2d::opt::StopCriteria;
|
||||||
|
static const unsigned MAX_TRIES = 100;
|
||||||
|
|
||||||
int pcount = 0, hlcount = 0;
|
int pcount = 0, hlcount = 0;
|
||||||
for(int i = 0; i < count; i++) {
|
for(int i = 0; i < count; i++) {
|
||||||
tifcl();
|
thr();
|
||||||
auto n = nmls.row(i);
|
auto n = nmls.row(i);
|
||||||
|
|
||||||
// for all normals we generate the spherical coordinates and
|
// for all normals we generate the spherical coordinates and
|
||||||
|
@ -1230,32 +1254,67 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
// We saturate the polar angle to 3pi/4
|
// We saturate the polar angle to 3pi/4
|
||||||
polar = std::max(polar, 3*PI / 4);
|
polar = std::max(polar, 3*PI / 4);
|
||||||
|
|
||||||
// Reassemble the now corrected normal
|
|
||||||
Vec3d nn(std::cos(azimuth) * std::sin(polar),
|
|
||||||
std::sin(azimuth) * std::sin(polar),
|
|
||||||
std::cos(polar));
|
|
||||||
|
|
||||||
nn.normalize();
|
|
||||||
|
|
||||||
// save the head (pinpoint) position
|
// save the head (pinpoint) position
|
||||||
Vec3d hp = filt_pts.row(i);
|
Vec3d hp = filt_pts.row(i);
|
||||||
|
|
||||||
// the full width of the head
|
|
||||||
double w = cfg.head_width_mm +
|
double w = cfg.head_width_mm +
|
||||||
cfg.head_back_radius_mm +
|
cfg.head_back_radius_mm +
|
||||||
2*cfg.head_front_radius_mm;
|
2*cfg.head_front_radius_mm;
|
||||||
|
|
||||||
// We should shoot a ray in the direction of the pinhead and
|
// Reassemble the now corrected normal
|
||||||
// see if there is enough space for it
|
auto nn = Vec3d(std::cos(azimuth) * std::sin(polar),
|
||||||
double t = pinhead_mesh_intersect(
|
std::sin(azimuth) * std::sin(polar),
|
||||||
hp, // touching point
|
std::cos(polar)).normalized();
|
||||||
nn,
|
|
||||||
cfg.head_front_radius_mm, // approx the radius
|
|
||||||
cfg.head_back_radius_mm,
|
|
||||||
w,
|
|
||||||
mesh);
|
|
||||||
|
|
||||||
if(t > w || std::isinf(t)) {
|
// check available distance
|
||||||
|
double t = pinhead_mesh_intersect(
|
||||||
|
hp, // touching point
|
||||||
|
nn, // normal
|
||||||
|
cfg.head_front_radius_mm,
|
||||||
|
cfg.head_back_radius_mm,
|
||||||
|
w,
|
||||||
|
mesh);
|
||||||
|
|
||||||
|
if(t <= w) {
|
||||||
|
// Let's try to optimize this angle, there might be a viable
|
||||||
|
// normal that doesn't collide with the model geometry and
|
||||||
|
// its very close to the default.
|
||||||
|
|
||||||
|
StopCriteria stc;
|
||||||
|
stc.max_iterations = MAX_TRIES;
|
||||||
|
stc.relative_score_difference = 1e-3;
|
||||||
|
stc.stop_score = w; // space greater than w is enough
|
||||||
|
SimplexOptimizer solver(stc);
|
||||||
|
|
||||||
|
auto oresult = solver.optimize_max(
|
||||||
|
[&mesh, &cfg, w, hp](double plr, double azm)
|
||||||
|
{
|
||||||
|
auto n = Vec3d(std::cos(azm) * std::sin(plr),
|
||||||
|
std::sin(azm) * std::sin(plr),
|
||||||
|
std::cos(plr)).normalized();
|
||||||
|
|
||||||
|
double score = pinhead_mesh_intersect(
|
||||||
|
hp, n,
|
||||||
|
cfg.head_front_radius_mm,
|
||||||
|
cfg.head_back_radius_mm,
|
||||||
|
w,
|
||||||
|
mesh);
|
||||||
|
return score;
|
||||||
|
},
|
||||||
|
initvals(polar, azimuth), // let's start with what we have
|
||||||
|
bound(3*PI/4, PI), // Must not exceed the tilt limit
|
||||||
|
bound(-PI, PI) // azimuth can be a full range search
|
||||||
|
);
|
||||||
|
|
||||||
|
t = oresult.score;
|
||||||
|
polar = std::get<0>(oresult.optimum);
|
||||||
|
azimuth = std::get<1>(oresult.optimum);
|
||||||
|
nn = Vec3d(std::cos(azimuth) * std::sin(polar),
|
||||||
|
std::sin(azimuth) * std::sin(polar),
|
||||||
|
std::cos(polar)).normalized();
|
||||||
|
}
|
||||||
|
|
||||||
|
if(t > w) {
|
||||||
head_pos.row(pcount) = hp;
|
head_pos.row(pcount) = hp;
|
||||||
|
|
||||||
// save the verified and corrected normal
|
// save the verified and corrected normal
|
||||||
|
@ -1263,6 +1322,7 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
|
|
||||||
++pcount;
|
++pcount;
|
||||||
} else if( polar >= 3*PI/4 ) {
|
} else if( polar >= 3*PI/4 ) {
|
||||||
|
|
||||||
// Headless supports do not tilt like the headed ones so
|
// Headless supports do not tilt like the headed ones so
|
||||||
// the normal should point almost to the ground.
|
// the normal should point almost to the ground.
|
||||||
headless_norm.row(hlcount) = nn;
|
headless_norm.row(hlcount) = nn;
|
||||||
|
@ -1279,7 +1339,7 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
|
|
||||||
// Pinhead creation: based on the filtering results, the Head objects will
|
// Pinhead creation: based on the filtering results, the Head objects will
|
||||||
// be constructed (together with their triangle meshes).
|
// be constructed (together with their triangle meshes).
|
||||||
auto pinheadfn = [tifcl] (
|
auto pinheadfn = [thr] (
|
||||||
const SupportConfig& cfg,
|
const SupportConfig& cfg,
|
||||||
PointSet& head_pos,
|
PointSet& head_pos,
|
||||||
PointSet& nmls,
|
PointSet& nmls,
|
||||||
|
@ -1292,7 +1352,7 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
/* ******************************************************** */
|
/* ******************************************************** */
|
||||||
|
|
||||||
for (int i = 0; i < head_pos.rows(); ++i) {
|
for (int i = 0; i < head_pos.rows(); ++i) {
|
||||||
tifcl();
|
thr();
|
||||||
result.add_head(
|
result.add_head(
|
||||||
cfg.head_back_radius_mm,
|
cfg.head_back_radius_mm,
|
||||||
cfg.head_front_radius_mm,
|
cfg.head_front_radius_mm,
|
||||||
|
@ -1311,7 +1371,7 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
// will process it. Also, the pillars will be grouped into clusters that can
|
// will process it. Also, the pillars will be grouped into clusters that can
|
||||||
// be interconnected with bridges. Elements of these groups may or may not
|
// be interconnected with bridges. Elements of these groups may or may not
|
||||||
// be interconnected. Here we only run the clustering algorithm.
|
// be interconnected. Here we only run the clustering algorithm.
|
||||||
auto classifyfn = [tifcl] (
|
auto classifyfn = [thr] (
|
||||||
const SupportConfig& cfg,
|
const SupportConfig& cfg,
|
||||||
const EigenMesh3D& mesh,
|
const EigenMesh3D& mesh,
|
||||||
PointSet& head_pos,
|
PointSet& head_pos,
|
||||||
|
@ -1320,7 +1380,8 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
std::vector<double>& gndheight,
|
std::vector<double>& gndheight,
|
||||||
ClusteredPoints& ground_clusters,
|
ClusteredPoints& ground_clusters,
|
||||||
Result& result
|
Result& result
|
||||||
) {
|
)
|
||||||
|
{
|
||||||
|
|
||||||
/* ******************************************************** */
|
/* ******************************************************** */
|
||||||
/* Classification */
|
/* Classification */
|
||||||
|
@ -1331,11 +1392,11 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
gndidx.reserve(size_t(head_pos.rows()));
|
gndidx.reserve(size_t(head_pos.rows()));
|
||||||
nogndidx.reserve(size_t(head_pos.rows()));
|
nogndidx.reserve(size_t(head_pos.rows()));
|
||||||
|
|
||||||
// First we search decide which heads reach the ground and can be full
|
// First we decide which heads reach the ground and can be full
|
||||||
// pillars and which shall be connected to the model surface (or search
|
// pillars and which shall be connected to the model surface (or search
|
||||||
// a suitable path around the surface that leads to the ground -- TODO)
|
// a suitable path around the surface that leads to the ground -- TODO)
|
||||||
for(unsigned i = 0; i < head_pos.rows(); i++) {
|
for(unsigned i = 0; i < head_pos.rows(); i++) {
|
||||||
tifcl();
|
thr();
|
||||||
auto& head = result.head(i);
|
auto& head = result.head(i);
|
||||||
|
|
||||||
Vec3d dir(0, 0, -1);
|
Vec3d dir(0, 0, -1);
|
||||||
|
@ -1344,9 +1405,38 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
double t = std::numeric_limits<double>::infinity();
|
double t = std::numeric_limits<double>::infinity();
|
||||||
double hw = head.width_mm;
|
double hw = head.width_mm;
|
||||||
|
|
||||||
|
|
||||||
|
{
|
||||||
|
// using libnest2d::opt::Method;
|
||||||
|
// using libnest2d::opt::bound;
|
||||||
|
// using libnest2d::opt::Optimizer;
|
||||||
|
// using libnest2d::opt::TOptimizer;
|
||||||
|
// using libnest2d::opt::StopCriteria;
|
||||||
|
|
||||||
|
// auto stopcond = [] () { return false; };
|
||||||
|
// static const unsigned max_tries = 100;
|
||||||
|
|
||||||
|
// auto objfunc =
|
||||||
|
// [&head](double polar, double azimuth, double width)
|
||||||
|
// {
|
||||||
|
// Vec3d nn(std::cos(azimuth) * std::sin(polar),
|
||||||
|
// std::sin(azimuth) * std::sin(polar),
|
||||||
|
// std::cos(polar));
|
||||||
|
|
||||||
|
|
||||||
|
// };
|
||||||
|
|
||||||
|
// StopCriteria stc;
|
||||||
|
// stc.max_iterations = max_tries;
|
||||||
|
// stc.relative_score_difference = 1e-3;
|
||||||
|
// stc.stop_condition = stopcond;
|
||||||
|
// TOptimizer<Method::L_SIMPLEX> solver(stc);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// We will try to assign a pillar to all the pinheads. If a pillar
|
// We will try to assign a pillar to all the pinheads. If a pillar
|
||||||
// would pierce the model surface, we will try to adjust slightly
|
// would pierce the model surface, we will try to adjust slightly
|
||||||
// the head with so that the pillar can be deployed.
|
// the head width so that the pillar can be deployed.
|
||||||
while(!accept && head.width_mm > 0) {
|
while(!accept && head.width_mm > 0) {
|
||||||
|
|
||||||
Vec3d startpoint = head.junction_point();
|
Vec3d startpoint = head.junction_point();
|
||||||
|
@ -1358,11 +1448,18 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
double tprec = ray_mesh_intersect(startpoint, dir, mesh);
|
double tprec = ray_mesh_intersect(startpoint, dir, mesh);
|
||||||
|
|
||||||
if(std::isinf(tprec) && !std::isinf(t)) {
|
if(std::isinf(tprec) && !std::isinf(t)) {
|
||||||
// This is a damned case where the pillar melds into the
|
// This is a damned case where the pillar melts into the
|
||||||
// model but its center ray can reach the ground. We can
|
// model but its center ray can reach the ground. We can
|
||||||
// not route this to the ground nor to the model surface.
|
// not route this to the ground nor to the model surface.
|
||||||
head.width_mm = hw + (ri % 2? -1 : 1) * ri * head.r_back_mm;
|
head.width_mm = hw + (ri % 2? -1 : 1) * ri * head.r_back_mm;
|
||||||
} else {
|
} else {
|
||||||
|
if(!std::isinf(t) && !std::isinf(tprec) &&
|
||||||
|
std::abs(tprec - t) > hw)
|
||||||
|
{
|
||||||
|
// In this case the head would scratch the model body
|
||||||
|
BOOST_LOG_TRIVIAL(warning) << "Head scratch detected.";
|
||||||
|
}
|
||||||
|
|
||||||
accept = true; t = tprec;
|
accept = true; t = tprec;
|
||||||
|
|
||||||
auto id = head.id;
|
auto id = head.id;
|
||||||
|
@ -1417,9 +1514,9 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
ground_clusters =
|
ground_clusters =
|
||||||
cluster(
|
cluster(
|
||||||
gnd,
|
gnd,
|
||||||
[d_base, tifcl](const SpatElement& p, const SpatElement& s)
|
[d_base, thr](const SpatElement& p, const SpatElement& s)
|
||||||
{
|
{
|
||||||
tifcl();
|
thr();
|
||||||
return distance(Vec2d(p.first(X), p.first(Y)),
|
return distance(Vec2d(p.first(X), p.first(Y)),
|
||||||
Vec2d(s.first(X), s.first(Y))) < d_base;
|
Vec2d(s.first(X), s.first(Y))) < d_base;
|
||||||
}, 3); // max 3 heads to connect to one centroid
|
}, 3); // max 3 heads to connect to one centroid
|
||||||
|
@ -1492,7 +1589,7 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
// a full pillar (ground connected). Some will connect to a nearby pillar
|
// a full pillar (ground connected). Some will connect to a nearby pillar
|
||||||
// using a bridge. The max number of such side-heads for a central pillar
|
// using a bridge. The max number of such side-heads for a central pillar
|
||||||
// is limited to avoid bad weight distribution.
|
// is limited to avoid bad weight distribution.
|
||||||
auto routing_ground_fn = [gnd_head_pt, interconnect, tifcl](
|
auto routing_ground_fn = [gnd_head_pt, interconnect, thr](
|
||||||
const SupportConfig& cfg,
|
const SupportConfig& cfg,
|
||||||
const ClusteredPoints& gnd_clusters,
|
const ClusteredPoints& gnd_clusters,
|
||||||
const IndexSet& gndidx,
|
const IndexSet& gndidx,
|
||||||
|
@ -1508,7 +1605,7 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
cl_centroids.reserve(gnd_clusters.size());
|
cl_centroids.reserve(gnd_clusters.size());
|
||||||
|
|
||||||
SpatIndex pheadindex; // spatial index for the junctions
|
SpatIndex pheadindex; // spatial index for the junctions
|
||||||
for(auto& cl : gnd_clusters) { tifcl();
|
for(auto& cl : gnd_clusters) { thr();
|
||||||
// place all the centroid head positions into the index. We will
|
// place all the centroid head positions into the index. We will
|
||||||
// query for alternative pillar positions. If a sidehead cannot
|
// query for alternative pillar positions. If a sidehead cannot
|
||||||
// connect to the cluster centroid, we have to search for another
|
// connect to the cluster centroid, we have to search for another
|
||||||
|
@ -1519,9 +1616,9 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
|
|
||||||
// get the current cluster centroid
|
// get the current cluster centroid
|
||||||
long lcid = cluster_centroid(cl, gnd_head_pt,
|
long lcid = cluster_centroid(cl, gnd_head_pt,
|
||||||
[tifcl](const Vec3d& p1, const Vec3d& p2)
|
[thr](const Vec3d& p1, const Vec3d& p2)
|
||||||
{
|
{
|
||||||
tifcl();
|
thr();
|
||||||
return distance(Vec2d(p1(X), p1(Y)), Vec2d(p2(X), p2(Y)));
|
return distance(Vec2d(p1(X), p1(Y)), Vec2d(p2(X), p2(Y)));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -1542,7 +1639,7 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
// sidepoints with the cluster centroid (which is a ground pillar)
|
// sidepoints with the cluster centroid (which is a ground pillar)
|
||||||
// or a nearby pillar if the centroid is unreachable.
|
// or a nearby pillar if the centroid is unreachable.
|
||||||
size_t ci = 0;
|
size_t ci = 0;
|
||||||
for(auto cl : gnd_clusters) { tifcl();
|
for(auto cl : gnd_clusters) { thr();
|
||||||
|
|
||||||
auto cidx = cl_centroids[ci];
|
auto cidx = cl_centroids[ci];
|
||||||
cl_centroids[ci++] = cl[cidx];
|
cl_centroids[ci++] = cl[cidx];
|
||||||
|
@ -1566,12 +1663,12 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
// is distributed more effectively on the pillar.
|
// is distributed more effectively on the pillar.
|
||||||
|
|
||||||
auto search_nearest =
|
auto search_nearest =
|
||||||
[&tifcl, &cfg, &result, &emesh, maxbridgelen, gndlvl, pradius]
|
[&thr, &cfg, &result, &emesh, maxbridgelen, gndlvl, pradius]
|
||||||
(SpatIndex& spindex, const Vec3d& jsh)
|
(SpatIndex& spindex, const Vec3d& jsh)
|
||||||
{
|
{
|
||||||
long nearest_id = -1;
|
long nearest_id = -1;
|
||||||
const double max_len = maxbridgelen / 2;
|
const double max_len = maxbridgelen / 2;
|
||||||
while(nearest_id < 0 && !spindex.empty()) { tifcl();
|
while(nearest_id < 0 && !spindex.empty()) { thr();
|
||||||
// loop until a suitable head is not found
|
// loop until a suitable head is not found
|
||||||
// if there is a pillar closer than the cluster center
|
// if there is a pillar closer than the cluster center
|
||||||
// (this may happen as the clustering is not perfect)
|
// (this may happen as the clustering is not perfect)
|
||||||
|
@ -1610,7 +1707,7 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
return nearest_id;
|
return nearest_id;
|
||||||
};
|
};
|
||||||
|
|
||||||
for(auto c : cl) { tifcl();
|
for(auto c : cl) { thr();
|
||||||
auto& sidehead = result.head(gndidx[c]);
|
auto& sidehead = result.head(gndidx[c]);
|
||||||
sidehead.transform();
|
sidehead.transform();
|
||||||
|
|
||||||
|
@ -1676,7 +1773,7 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
ClusterEl ring;
|
ClusterEl ring;
|
||||||
|
|
||||||
while(!rem.empty()) { // loop until all the points belong to some ring
|
while(!rem.empty()) { // loop until all the points belong to some ring
|
||||||
tifcl();
|
thr();
|
||||||
std::sort(rem.begin(), rem.end());
|
std::sort(rem.begin(), rem.end());
|
||||||
|
|
||||||
auto newring = pts_convex_hull(rem,
|
auto newring = pts_convex_hull(rem,
|
||||||
|
@ -1688,7 +1785,7 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
if(!ring.empty()) {
|
if(!ring.empty()) {
|
||||||
// inner ring is now in 'newring' and outer ring is in 'ring'
|
// inner ring is now in 'newring' and outer ring is in 'ring'
|
||||||
SpatIndex innerring;
|
SpatIndex innerring;
|
||||||
for(unsigned i : newring) { tifcl();
|
for(unsigned i : newring) { thr();
|
||||||
const Pillar& pill = result.head_pillar(gndidx[i]);
|
const Pillar& pill = result.head_pillar(gndidx[i]);
|
||||||
assert(pill.id >= 0);
|
assert(pill.id >= 0);
|
||||||
innerring.insert(pill.endpoint, unsigned(pill.id));
|
innerring.insert(pill.endpoint, unsigned(pill.id));
|
||||||
|
@ -1697,7 +1794,7 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
// For all pillars in the outer ring find the closest in the
|
// For all pillars in the outer ring find the closest in the
|
||||||
// inner ring and connect them. This will create the spider web
|
// inner ring and connect them. This will create the spider web
|
||||||
// fashioned connections between pillars
|
// fashioned connections between pillars
|
||||||
for(unsigned i : ring) { tifcl();
|
for(unsigned i : ring) { thr();
|
||||||
const Pillar& outerpill = result.head_pillar(gndidx[i]);
|
const Pillar& outerpill = result.head_pillar(gndidx[i]);
|
||||||
auto res = innerring.nearest(outerpill.endpoint, 1);
|
auto res = innerring.nearest(outerpill.endpoint, 1);
|
||||||
if(res.empty()) continue;
|
if(res.empty()) continue;
|
||||||
|
@ -1723,7 +1820,7 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
next != ring.end();
|
next != ring.end();
|
||||||
++it, ++next)
|
++it, ++next)
|
||||||
{
|
{
|
||||||
tifcl();
|
thr();
|
||||||
const Pillar& pillar = result.head_pillar(gndidx[*it]);
|
const Pillar& pillar = result.head_pillar(gndidx[*it]);
|
||||||
const Pillar& nextpillar = result.head_pillar(gndidx[*next]);
|
const Pillar& nextpillar = result.head_pillar(gndidx[*next]);
|
||||||
interconnect(pillar, nextpillar, emesh, result);
|
interconnect(pillar, nextpillar, emesh, result);
|
||||||
|
@ -1738,19 +1835,19 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// Step: routing the pinheads that are would connect to the model surface
|
// Step: routing the pinheads that would connect to the model surface
|
||||||
// along the Z axis downwards. For now these will actually be connected with
|
// along the Z axis downwards. For now these will actually be connected with
|
||||||
// the model surface with a flipped pinhead. In the future here we could use
|
// the model surface with a flipped pinhead. In the future here we could use
|
||||||
// some smart algorithms to search for a safe path to the ground or to a
|
// some smart algorithms to search for a safe path to the ground or to a
|
||||||
// nearby pillar that can hold the supported weight.
|
// nearby pillar that can hold the supported weight.
|
||||||
auto routing_nongnd_fn = [tifcl](
|
auto routing_nongnd_fn = [thr](
|
||||||
const SupportConfig& cfg,
|
const SupportConfig& cfg,
|
||||||
const std::vector<double>& gndheight,
|
const std::vector<double>& gndheight,
|
||||||
const IndexSet& nogndidx,
|
const IndexSet& nogndidx,
|
||||||
Result& result)
|
Result& result)
|
||||||
{
|
{
|
||||||
// TODO: connect these to the ground pillars if possible
|
// TODO: connect these to the ground pillars if possible
|
||||||
for(auto idx : nogndidx) { tifcl();
|
for(auto idx : nogndidx) { thr();
|
||||||
double gh = gndheight[idx];
|
double gh = gndheight[idx];
|
||||||
double base_width = cfg.head_width_mm;
|
double base_width = cfg.head_width_mm;
|
||||||
|
|
||||||
|
@ -1807,7 +1904,7 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
// Step: process the support points where there is not enough space for a
|
// Step: process the support points where there is not enough space for a
|
||||||
// full pinhead. In this case we will use a rounded sphere as a touching
|
// full pinhead. In this case we will use a rounded sphere as a touching
|
||||||
// point and use a thinner bridge (let's call it a stick).
|
// point and use a thinner bridge (let's call it a stick).
|
||||||
auto process_headless = [tifcl](
|
auto process_headless = [thr](
|
||||||
const SupportConfig& cfg,
|
const SupportConfig& cfg,
|
||||||
const PointSet& headless_pts,
|
const PointSet& headless_pts,
|
||||||
const PointSet& headless_norm,
|
const PointSet& headless_norm,
|
||||||
|
@ -1822,7 +1919,7 @@ bool SLASupportTree::generate(const PointSet &points,
|
||||||
|
|
||||||
// We will sink the pins into the model surface for a distance of 1/3 of
|
// We will sink the pins into the model surface for a distance of 1/3 of
|
||||||
// the pin radius
|
// the pin radius
|
||||||
for(int i = 0; i < headless_pts.rows(); i++) { tifcl();
|
for(int i = 0; i < headless_pts.rows(); i++) { thr();
|
||||||
Vec3d sph = headless_pts.row(i); // Exact support position
|
Vec3d sph = headless_pts.row(i); // Exact support position
|
||||||
Vec3d n = headless_norm.row(i); // mesh outward normal
|
Vec3d n = headless_norm.row(i); // mesh outward normal
|
||||||
Vec3d sp = sph - n * HWIDTH_MM; // stick head start point
|
Vec3d sp = sph - n * HWIDTH_MM; // stick head start point
|
||||||
|
|
|
@ -753,7 +753,7 @@ void SLAPrint::process()
|
||||||
double md = po.m_config.pad_max_merge_distance.getFloat();
|
double md = po.m_config.pad_max_merge_distance.getFloat();
|
||||||
// Radius is disabled for now...
|
// Radius is disabled for now...
|
||||||
double er = 0; // po.m_config.pad_edge_radius.getFloat();
|
double er = 0; // po.m_config.pad_edge_radius.getFloat();
|
||||||
double tilt = po.m_config.pad_wall_tilt.getFloat() * PI / 180.0;
|
double tilt = po.m_config.pad_wall_slope.getFloat() * PI / 180.0;
|
||||||
double lh = po.m_config.layer_height.getFloat();
|
double lh = po.m_config.layer_height.getFloat();
|
||||||
double elevation = po.m_config.support_object_elevation.getFloat();
|
double elevation = po.m_config.support_object_elevation.getFloat();
|
||||||
if(!po.m_config.supports_enable.getBool()) elevation = 0;
|
if(!po.m_config.supports_enable.getBool()) elevation = 0;
|
||||||
|
@ -1369,7 +1369,7 @@ bool SLAPrintObject::invalidate_state_by_config_options(const std::vector<t_conf
|
||||||
|| opt_key == "pad_wall_thickness"
|
|| opt_key == "pad_wall_thickness"
|
||||||
|| opt_key == "pad_wall_height"
|
|| opt_key == "pad_wall_height"
|
||||||
|| opt_key == "pad_max_merge_distance"
|
|| opt_key == "pad_max_merge_distance"
|
||||||
|| opt_key == "pad_wall_tilt"
|
|| opt_key == "pad_wall_slope"
|
||||||
|| opt_key == "pad_edge_radius") {
|
|| opt_key == "pad_edge_radius") {
|
||||||
steps.emplace_back(slaposBasePool);
|
steps.emplace_back(slaposBasePool);
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -465,7 +465,7 @@ const std::vector<std::string>& Preset::sla_print_options()
|
||||||
"pad_wall_height",
|
"pad_wall_height",
|
||||||
"pad_max_merge_distance",
|
"pad_max_merge_distance",
|
||||||
"pad_edge_radius",
|
"pad_edge_radius",
|
||||||
"pad_wall_tilt",
|
"pad_wall_slope",
|
||||||
"output_filename_format",
|
"output_filename_format",
|
||||||
"default_sla_print_profile",
|
"default_sla_print_profile",
|
||||||
"compatible_printers",
|
"compatible_printers",
|
||||||
|
|
|
@ -3290,7 +3290,7 @@ void TabSLAPrint::build()
|
||||||
optgroup->append_single_option_line("pad_max_merge_distance");
|
optgroup->append_single_option_line("pad_max_merge_distance");
|
||||||
// TODO: Disabling this parameter for the beta release
|
// TODO: Disabling this parameter for the beta release
|
||||||
// optgroup->append_single_option_line("pad_edge_radius");
|
// optgroup->append_single_option_line("pad_edge_radius");
|
||||||
optgroup->append_single_option_line("pad_wall_tilt");
|
optgroup->append_single_option_line("pad_wall_slope");
|
||||||
|
|
||||||
page = add_options_page(_(L("Output options")), "page_white_go.png");
|
page = add_options_page(_(L("Output options")), "page_white_go.png");
|
||||||
optgroup = page->new_optgroup(_(L("Output file")));
|
optgroup = page->new_optgroup(_(L("Output file")));
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue