mirror of
				https://git.mirrors.martin98.com/https://github.com/slic3r/Slic3r.git
				synced 2025-10-21 04:41:06 +08:00 
			
		
		
		
	Merge remote-tracking branch 'origin/master' into ys_color_print_extension
This commit is contained in:
		
						commit
						5d9a136b8a
					
				| @ -194,6 +194,8 @@ add_library(libslic3r STATIC | |||||||
|     SLA/SLARaster.cpp |     SLA/SLARaster.cpp | ||||||
|     SLA/SLARasterWriter.hpp |     SLA/SLARasterWriter.hpp | ||||||
|     SLA/SLARasterWriter.cpp |     SLA/SLARasterWriter.cpp | ||||||
|  |     SLA/ConcaveHull.hpp | ||||||
|  |     SLA/ConcaveHull.cpp | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| encoding_check(libslic3r) | encoding_check(libslic3r) | ||||||
|  | |||||||
| @ -507,7 +507,7 @@ std::string WipeTowerIntegration::prime(GCode &gcodegen) | |||||||
| std::string WipeTowerIntegration::tool_change(GCode &gcodegen, int extruder_id, bool finish_layer) | std::string WipeTowerIntegration::tool_change(GCode &gcodegen, int extruder_id, bool finish_layer) | ||||||
| { | { | ||||||
|     std::string gcode; |     std::string gcode; | ||||||
| 	assert(m_layer_idx >= 0 && size_t(m_layer_idx) <= m_tool_changes.size()); |     assert(m_layer_idx >= 0); | ||||||
|     if (! m_brim_done || gcodegen.writer().need_toolchange(extruder_id) || finish_layer) { |     if (! m_brim_done || gcodegen.writer().need_toolchange(extruder_id) || finish_layer) { | ||||||
| 		if (m_layer_idx < (int)m_tool_changes.size()) { | 		if (m_layer_idx < (int)m_tool_changes.size()) { | ||||||
| 			if (! (size_t(m_tool_change_idx) < m_tool_changes[m_layer_idx].size())) | 			if (! (size_t(m_tool_change_idx) < m_tool_changes[m_layer_idx].size())) | ||||||
|  | |||||||
							
								
								
									
										171
									
								
								src/libslic3r/SLA/ConcaveHull.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										171
									
								
								src/libslic3r/SLA/ConcaveHull.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,171 @@ | |||||||
|  | #include "ConcaveHull.hpp" | ||||||
|  | #include <libslic3r/MTUtils.hpp> | ||||||
|  | #include <libslic3r/ClipperUtils.hpp> | ||||||
|  | #include "SLASpatIndex.hpp" | ||||||
|  | #include <boost/log/trivial.hpp> | ||||||
|  | 
 | ||||||
|  | namespace Slic3r { | ||||||
|  | namespace sla { | ||||||
|  | 
 | ||||||
|  | inline Vec3d to_vec3(const Vec2crd &v2) { return {double(v2(X)), double(v2(Y)), 0.}; } | ||||||
|  | inline Vec3d to_vec3(const Vec2d &v2) { return {v2(X), v2(Y), 0.}; } | ||||||
|  | inline Vec2crd to_vec2(const Vec3d &v3) { return {coord_t(v3(X)), coord_t(v3(Y))}; } | ||||||
|  | 
 | ||||||
|  | Point ConcaveHull::centroid(const Points &pp) | ||||||
|  | { | ||||||
|  |     Point c; | ||||||
|  |     switch(pp.size()) { | ||||||
|  |     case 0: break; | ||||||
|  |     case 1: c = pp.front(); break; | ||||||
|  |     case 2: c = (pp[0] + pp[1]) / 2; break; | ||||||
|  |     default: { | ||||||
|  |         auto MAX = std::numeric_limits<Point::coord_type>::max(); | ||||||
|  |         auto MIN = std::numeric_limits<Point::coord_type>::min(); | ||||||
|  |         Point min = {MAX, MAX}, max = {MIN, MIN}; | ||||||
|  | 
 | ||||||
|  |         for(auto& p : pp) { | ||||||
|  |             if(p(0) < min(0)) min(0) = p(0); | ||||||
|  |             if(p(1) < min(1)) min(1) = p(1); | ||||||
|  |             if(p(0) > max(0)) max(0) = p(0); | ||||||
|  |             if(p(1) > max(1)) max(1) = p(1); | ||||||
|  |         } | ||||||
|  |         c(0) = min(0) + (max(0) - min(0)) / 2; | ||||||
|  |         c(1) = min(1) + (max(1) - min(1)) / 2; | ||||||
|  |         break; | ||||||
|  |     } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return c; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // As it shows, the current offset_ex in ClipperUtils hangs if used in jtRound
 | ||||||
|  | // mode
 | ||||||
|  | ClipperLib::Paths fast_offset(const ClipperLib::Paths &paths, | ||||||
|  |                               coord_t                  delta, | ||||||
|  |                               ClipperLib::JoinType     jointype) | ||||||
|  | { | ||||||
|  |     using ClipperLib::ClipperOffset; | ||||||
|  |     using ClipperLib::etClosedPolygon; | ||||||
|  |     using ClipperLib::Paths; | ||||||
|  |     using ClipperLib::Path; | ||||||
|  | 
 | ||||||
|  |     ClipperOffset offs; | ||||||
|  |     offs.ArcTolerance = scaled<double>(0.01); | ||||||
|  | 
 | ||||||
|  |     for (auto &p : paths) | ||||||
|  |         // If the input is not at least a triangle, we can not do this algorithm
 | ||||||
|  |         if(p.size() < 3) { | ||||||
|  |             BOOST_LOG_TRIVIAL(error) << "Invalid geometry for offsetting!"; | ||||||
|  |             return {}; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |     offs.AddPaths(paths, jointype, etClosedPolygon); | ||||||
|  | 
 | ||||||
|  |     Paths result; | ||||||
|  |     offs.Execute(result, static_cast<double>(delta)); | ||||||
|  | 
 | ||||||
|  |     return result; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Points ConcaveHull::calculate_centroids() const | ||||||
|  | { | ||||||
|  |     // We get the centroids of all the islands in the 2D slice
 | ||||||
|  |     Points centroids = reserve_vector<Point>(m_polys.size()); | ||||||
|  |     std::transform(m_polys.begin(), m_polys.end(), | ||||||
|  |                    std::back_inserter(centroids), | ||||||
|  |                    [this](const Polygon &poly) { return centroid(poly); }); | ||||||
|  | 
 | ||||||
|  |     return centroids; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void ConcaveHull::merge_polygons() { m_polys = get_contours(union_ex(m_polys)); } | ||||||
|  | 
 | ||||||
|  | void ConcaveHull::add_connector_rectangles(const Points ¢roids, | ||||||
|  |                                            coord_t       max_dist, | ||||||
|  |                                            ThrowOnCancel thr) | ||||||
|  | { | ||||||
|  |     // Centroid of the centroids of islands. This is where the additional
 | ||||||
|  |     // connector sticks are routed.
 | ||||||
|  |     Point cc = centroid(centroids); | ||||||
|  | 
 | ||||||
|  |     PointIndex ctrindex; | ||||||
|  |     unsigned  idx = 0; | ||||||
|  |     for(const Point &ct : centroids) ctrindex.insert(to_vec3(ct), idx++); | ||||||
|  | 
 | ||||||
|  |     m_polys.reserve(m_polys.size() + centroids.size()); | ||||||
|  | 
 | ||||||
|  |     idx = 0; | ||||||
|  |     for (const Point &c : centroids) { | ||||||
|  |         thr(); | ||||||
|  | 
 | ||||||
|  |         double dx = c.x() - cc.x(), dy = c.y() - cc.y(); | ||||||
|  |         double l  = std::sqrt(dx * dx + dy * dy); | ||||||
|  |         double nx = dx / l, ny = dy / l; | ||||||
|  | 
 | ||||||
|  |         const Point &ct = centroids[idx]; | ||||||
|  | 
 | ||||||
|  |         std::vector<PointIndexEl> result = ctrindex.nearest(to_vec3(ct), 2); | ||||||
|  | 
 | ||||||
|  |         double dist = max_dist; | ||||||
|  |         for (const PointIndexEl &el : result) | ||||||
|  |             if (el.second != idx) { | ||||||
|  |                 dist = Line(to_vec2(el.first), ct).length(); | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|  |         idx++; | ||||||
|  | 
 | ||||||
|  |         if (dist >= max_dist) return; | ||||||
|  | 
 | ||||||
|  |         Polygon r; | ||||||
|  |         r.points.reserve(3); | ||||||
|  |         r.points.emplace_back(cc); | ||||||
|  | 
 | ||||||
|  |         Point n(scaled(nx), scaled(ny)); | ||||||
|  |         r.points.emplace_back(c + Point(n.y(), -n.x())); | ||||||
|  |         r.points.emplace_back(c + Point(-n.y(), n.x())); | ||||||
|  |         offset(r, scaled<float>(1.)); | ||||||
|  | 
 | ||||||
|  |         m_polys.emplace_back(r); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | ConcaveHull::ConcaveHull(const Polygons &polys, double mergedist, ThrowOnCancel thr) | ||||||
|  | { | ||||||
|  |     if(polys.empty()) return; | ||||||
|  | 
 | ||||||
|  |     m_polys = polys; | ||||||
|  |     merge_polygons(); | ||||||
|  | 
 | ||||||
|  |     if(m_polys.size() == 1) return; | ||||||
|  | 
 | ||||||
|  |     Points centroids = calculate_centroids(); | ||||||
|  | 
 | ||||||
|  |     add_connector_rectangles(centroids, scaled(mergedist), thr); | ||||||
|  | 
 | ||||||
|  |     merge_polygons(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | ExPolygons ConcaveHull::to_expolygons() const | ||||||
|  | { | ||||||
|  |     auto ret = reserve_vector<ExPolygon>(m_polys.size()); | ||||||
|  |     for (const Polygon &p : m_polys) ret.emplace_back(ExPolygon(p)); | ||||||
|  |     return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | ExPolygons offset_waffle_style_ex(const ConcaveHull &hull, coord_t delta) | ||||||
|  | { | ||||||
|  |     ClipperLib::Paths paths = Slic3rMultiPoints_to_ClipperPaths(hull.polygons()); | ||||||
|  |     paths = fast_offset(paths, 2 * delta, ClipperLib::jtRound); | ||||||
|  |     paths = fast_offset(paths, -delta, ClipperLib::jtRound); | ||||||
|  |     ExPolygons ret = ClipperPaths_to_Slic3rExPolygons(paths); | ||||||
|  |     for (ExPolygon &p : ret) p.holes = {}; | ||||||
|  |     return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Polygons offset_waffle_style(const ConcaveHull &hull, coord_t delta) | ||||||
|  | { | ||||||
|  |     return to_polygons(offset_waffle_style_ex(hull, delta)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | }} // namespace Slic3r::sla
 | ||||||
							
								
								
									
										53
									
								
								src/libslic3r/SLA/ConcaveHull.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								src/libslic3r/SLA/ConcaveHull.hpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,53 @@ | |||||||
|  | #ifndef CONCAVEHULL_HPP | ||||||
|  | #define CONCAVEHULL_HPP | ||||||
|  | 
 | ||||||
|  | #include <libslic3r/ExPolygon.hpp> | ||||||
|  | 
 | ||||||
|  | namespace Slic3r { | ||||||
|  | namespace sla { | ||||||
|  | 
 | ||||||
|  | inline Polygons get_contours(const ExPolygons &poly) | ||||||
|  | { | ||||||
|  |     Polygons ret; ret.reserve(poly.size()); | ||||||
|  |     for (const ExPolygon &p : poly) ret.emplace_back(p.contour); | ||||||
|  | 
 | ||||||
|  |     return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | using ThrowOnCancel = std::function<void()>; | ||||||
|  | 
 | ||||||
|  | /// A fake concave hull that is constructed by connecting separate shapes
 | ||||||
|  | /// with explicit bridges. Bridges are generated from each shape's centroid
 | ||||||
|  | /// to the center of the "scene" which is the centroid calculated from the shape
 | ||||||
|  | /// centroids (a star is created...)
 | ||||||
|  | class ConcaveHull { | ||||||
|  |     Polygons m_polys; | ||||||
|  | 
 | ||||||
|  |     static Point centroid(const Points& pp); | ||||||
|  | 
 | ||||||
|  |     static inline Point centroid(const Polygon &poly) { return poly.centroid(); } | ||||||
|  | 
 | ||||||
|  |     Points calculate_centroids() const; | ||||||
|  | 
 | ||||||
|  |     void merge_polygons(); | ||||||
|  | 
 | ||||||
|  |     void add_connector_rectangles(const Points ¢roids, | ||||||
|  |                                   coord_t       max_dist, | ||||||
|  |                                   ThrowOnCancel thr); | ||||||
|  | public: | ||||||
|  | 
 | ||||||
|  |     ConcaveHull(const ExPolygons& polys, double merge_dist, ThrowOnCancel thr) | ||||||
|  |         : ConcaveHull{to_polygons(polys), merge_dist, thr} {} | ||||||
|  | 
 | ||||||
|  |     ConcaveHull(const Polygons& polys, double mergedist, ThrowOnCancel thr); | ||||||
|  | 
 | ||||||
|  |     const Polygons & polygons() const { return m_polys; } | ||||||
|  | 
 | ||||||
|  |     ExPolygons to_expolygons() const; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | ExPolygons offset_waffle_style_ex(const ConcaveHull &ccvhull, coord_t delta); | ||||||
|  | Polygons   offset_waffle_style(const ConcaveHull &polys, coord_t delta); | ||||||
|  | 
 | ||||||
|  | }}     // namespace Slic3r::sla
 | ||||||
|  | #endif // CONCAVEHULL_HPP
 | ||||||
| @ -1,6 +1,7 @@ | |||||||
| #include "SLAPad.hpp" | #include "SLAPad.hpp" | ||||||
| #include "SLABoilerPlate.hpp" | #include "SLABoilerPlate.hpp" | ||||||
| #include "SLASpatIndex.hpp" | #include "SLASpatIndex.hpp" | ||||||
|  | #include "ConcaveHull.hpp" | ||||||
| 
 | 
 | ||||||
| #include "boost/log/trivial.hpp" | #include "boost/log/trivial.hpp" | ||||||
| #include "SLABoostAdapter.hpp" | #include "SLABoostAdapter.hpp" | ||||||
| @ -206,36 +207,6 @@ Contour3D inline straight_walls(const Polygon &plate, | |||||||
|     return walls(plate, plate, lo_z, hi_z, .0 /*offset_diff*/, thr); |     return walls(plate, plate, lo_z, hi_z, .0 /*offset_diff*/, thr); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // As it shows, the current offset_ex in ClipperUtils hangs if used in jtRound
 |  | ||||||
| // mode
 |  | ||||||
| ClipperLib::Paths fast_offset(const ClipperLib::Paths &paths, |  | ||||||
|                               coord_t                  delta, |  | ||||||
|                               ClipperLib::JoinType     jointype) |  | ||||||
| { |  | ||||||
|     using ClipperLib::ClipperOffset; |  | ||||||
|     using ClipperLib::etClosedPolygon; |  | ||||||
|     using ClipperLib::Paths; |  | ||||||
|     using ClipperLib::Path; |  | ||||||
| 
 |  | ||||||
|     ClipperOffset offs; |  | ||||||
|     offs.ArcTolerance = scaled<double>(0.01); |  | ||||||
| 
 |  | ||||||
|     for (auto &p : paths) |  | ||||||
|         // If the input is not at least a triangle, we can not do this algorithm
 |  | ||||||
|         if(p.size() < 3) { |  | ||||||
|             BOOST_LOG_TRIVIAL(error) << "Invalid geometry for offsetting!"; |  | ||||||
|             return {}; |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|     offs.AddPaths(paths, jointype, etClosedPolygon); |  | ||||||
| 
 |  | ||||||
|     Paths result; |  | ||||||
|     offs.Execute(result, static_cast<double>(delta)); |  | ||||||
| 
 |  | ||||||
|     return result; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| // Function to cut tiny connector cavities for a given polygon. The input poly
 | // Function to cut tiny connector cavities for a given polygon. The input poly
 | ||||||
| // will be offsetted by "padding" and small rectangle shaped cavities will be
 | // will be offsetted by "padding" and small rectangle shaped cavities will be
 | ||||||
| // inserted along the perimeter in every "stride" distance. The stick rectangles
 | // inserted along the perimeter in every "stride" distance. The stick rectangles
 | ||||||
| @ -322,158 +293,15 @@ ExPolygons breakstick_holes(const ExPolygons &input, Args...args) | |||||||
|     return ret; |     return ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// A fake concave hull that is constructed by connecting separate shapes
 | static inline coord_t get_waffle_offset(const PadConfig &c) | ||||||
| /// with explicit bridges. Bridges are generated from each shape's centroid
 | { | ||||||
| /// to the center of the "scene" which is the centroid calculated from the shape
 |     return scaled(c.brim_size_mm + c.wing_distance()); | ||||||
| /// centroids (a star is created...)
 | } | ||||||
| class ConcaveHull { |  | ||||||
|     Polygons m_polys; |  | ||||||
| 
 | 
 | ||||||
|     Point centroid(const Points& pp) const | static inline double get_merge_distance(const PadConfig &c) | ||||||
|     { | { | ||||||
|         Point c; |     return 2. * (1.8 * c.wall_thickness_mm) + c.max_merge_dist_mm; | ||||||
|         switch(pp.size()) { | } | ||||||
|         case 0: break; |  | ||||||
|         case 1: c = pp.front(); break; |  | ||||||
|         case 2: c = (pp[0] + pp[1]) / 2; break; |  | ||||||
|         default: { |  | ||||||
|             auto MAX = std::numeric_limits<Point::coord_type>::max(); |  | ||||||
|             auto MIN = std::numeric_limits<Point::coord_type>::min(); |  | ||||||
|             Point min = {MAX, MAX}, max = {MIN, MIN}; |  | ||||||
| 
 |  | ||||||
|             for(auto& p : pp) { |  | ||||||
|                 if(p(0) < min(0)) min(0) = p(0); |  | ||||||
|                 if(p(1) < min(1)) min(1) = p(1); |  | ||||||
|                 if(p(0) > max(0)) max(0) = p(0); |  | ||||||
|                 if(p(1) > max(1)) max(1) = p(1); |  | ||||||
|             } |  | ||||||
|             c(0) = min(0) + (max(0) - min(0)) / 2; |  | ||||||
|             c(1) = min(1) + (max(1) - min(1)) / 2; |  | ||||||
|             break; |  | ||||||
|         } |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         return c; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     inline Point centroid(const Polygon &poly) const { return poly.centroid(); } |  | ||||||
| 
 |  | ||||||
|     Points calculate_centroids() const |  | ||||||
|     { |  | ||||||
|         // We get the centroids of all the islands in the 2D slice
 |  | ||||||
|         Points centroids = reserve_vector<Point>(m_polys.size()); |  | ||||||
|         std::transform(m_polys.begin(), m_polys.end(), |  | ||||||
|                        std::back_inserter(centroids), |  | ||||||
|                        [this](const Polygon &poly) { return centroid(poly); }); |  | ||||||
| 
 |  | ||||||
|         return centroids; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void merge_polygons() { m_polys = union_(m_polys); } |  | ||||||
| 
 |  | ||||||
|     void add_connector_rectangles(const Points ¢roids, |  | ||||||
|                                   coord_t       max_dist, |  | ||||||
|                                   ThrowOnCancel thr) |  | ||||||
|     { |  | ||||||
|         namespace bgi = boost::geometry::index; |  | ||||||
|         using PointIndexElement = std::pair<Point, unsigned>; |  | ||||||
|         using PointIndex = bgi::rtree<PointIndexElement, bgi::rstar<16, 4>>; |  | ||||||
| 
 |  | ||||||
|         // Centroid of the centroids of islands. This is where the additional
 |  | ||||||
|         // connector sticks are routed.
 |  | ||||||
|         Point cc = centroid(centroids); |  | ||||||
| 
 |  | ||||||
|         PointIndex ctrindex; |  | ||||||
|         unsigned  idx = 0; |  | ||||||
|         for(const Point &ct : centroids) |  | ||||||
|             ctrindex.insert(std::make_pair(ct, idx++)); |  | ||||||
| 
 |  | ||||||
|         m_polys.reserve(m_polys.size() + centroids.size()); |  | ||||||
| 
 |  | ||||||
|         idx = 0; |  | ||||||
|         for (const Point &c : centroids) { |  | ||||||
|             thr(); |  | ||||||
| 
 |  | ||||||
|             double dx = c.x() - cc.x(), dy = c.y() - cc.y(); |  | ||||||
|             double l  = std::sqrt(dx * dx + dy * dy); |  | ||||||
|             double nx = dx / l, ny = dy / l; |  | ||||||
| 
 |  | ||||||
|             const Point &ct = centroids[idx]; |  | ||||||
| 
 |  | ||||||
|             std::vector<PointIndexElement> result; |  | ||||||
|             ctrindex.query(bgi::nearest(ct, 2), std::back_inserter(result)); |  | ||||||
| 
 |  | ||||||
|             double dist = max_dist; |  | ||||||
|             for (const PointIndexElement &el : result) |  | ||||||
|                 if (el.second != idx) { |  | ||||||
|                     dist = Line(el.first, ct).length(); |  | ||||||
|                     break; |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|             idx++; |  | ||||||
| 
 |  | ||||||
|             if (dist >= max_dist) return; |  | ||||||
| 
 |  | ||||||
|             Polygon r; |  | ||||||
|             r.points.reserve(3); |  | ||||||
|             r.points.emplace_back(cc); |  | ||||||
| 
 |  | ||||||
|             Point d(scaled(nx), scaled(ny)); |  | ||||||
|             r.points.emplace_back(c + Point(-d.y(), d.x())); |  | ||||||
|             r.points.emplace_back(c + Point(d.y(), -d.x())); |  | ||||||
|             offset(r, scaled<float>(1.)); |  | ||||||
| 
 |  | ||||||
|             m_polys.emplace_back(r); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| public: |  | ||||||
| 
 |  | ||||||
|     ConcaveHull(const ExPolygons& polys, double merge_dist, ThrowOnCancel thr) |  | ||||||
|         : ConcaveHull{to_polygons(polys), merge_dist, thr} {} |  | ||||||
| 
 |  | ||||||
|     ConcaveHull(const Polygons& polys, double mergedist, ThrowOnCancel thr) |  | ||||||
|     { |  | ||||||
|         if(polys.empty()) return; |  | ||||||
| 
 |  | ||||||
|         m_polys = polys; |  | ||||||
|         merge_polygons(); |  | ||||||
| 
 |  | ||||||
|         if(m_polys.size() == 1) return; |  | ||||||
| 
 |  | ||||||
|         Points centroids = calculate_centroids(); |  | ||||||
| 
 |  | ||||||
|         add_connector_rectangles(centroids, scaled(mergedist), thr); |  | ||||||
| 
 |  | ||||||
|         merge_polygons(); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // const Polygons & polygons() const { return m_polys; }
 |  | ||||||
| 
 |  | ||||||
|     ExPolygons to_expolygons() const |  | ||||||
|     { |  | ||||||
|         auto ret = reserve_vector<ExPolygon>(m_polys.size()); |  | ||||||
|         for (const Polygon &p : m_polys) ret.emplace_back(ExPolygon(p)); |  | ||||||
|         return ret; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     void offset_waffle_style(coord_t delta) { |  | ||||||
|         ClipperLib::Paths paths = Slic3rMultiPoints_to_ClipperPaths(m_polys); |  | ||||||
|         paths = fast_offset(paths, 2 * delta, ClipperLib::jtRound); |  | ||||||
|         paths = fast_offset(paths, -delta, ClipperLib::jtRound); |  | ||||||
|         m_polys = ClipperPaths_to_Slic3rPolygons(paths); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     static inline coord_t get_waffle_offset(const PadConfig &c) |  | ||||||
|     { |  | ||||||
|         return scaled(c.brim_size_mm + c.wing_distance()); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     static inline double get_merge_distance(const PadConfig &c) |  | ||||||
|     { |  | ||||||
|         return 2. * (1.8 * c.wall_thickness_mm) + c.max_merge_dist_mm; |  | ||||||
|     } |  | ||||||
| }; |  | ||||||
| 
 | 
 | ||||||
| // Part of the pad configuration that is used for 3D geometry generation
 | // Part of the pad configuration that is used for 3D geometry generation
 | ||||||
| struct PadConfig3D { | struct PadConfig3D { | ||||||
| @ -591,7 +419,7 @@ public: | |||||||
|                       scaled<float>(cfg.embed_object.object_gap_mm), |                       scaled<float>(cfg.embed_object.object_gap_mm), | ||||||
|                       ClipperLib::jtMiter, 1); |                       ClipperLib::jtMiter, 1); | ||||||
| 
 | 
 | ||||||
|         ConcaveHull fullcvh = |         ExPolygons fullcvh = | ||||||
|             wafflized_concave_hull(support_blueprint, model_bp_offs, cfg, thr); |             wafflized_concave_hull(support_blueprint, model_bp_offs, cfg, thr); | ||||||
| 
 | 
 | ||||||
|         auto model_bp_sticks = |         auto model_bp_sticks = | ||||||
| @ -600,7 +428,7 @@ public: | |||||||
|                              cfg.embed_object.stick_width_mm, |                              cfg.embed_object.stick_width_mm, | ||||||
|                              cfg.embed_object.stick_penetration_mm); |                              cfg.embed_object.stick_penetration_mm); | ||||||
| 
 | 
 | ||||||
|         ExPolygons fullpad = diff_ex(fullcvh.to_expolygons(), model_bp_sticks); |         ExPolygons fullpad = diff_ex(fullcvh, model_bp_sticks); | ||||||
| 
 | 
 | ||||||
|         remove_redundant_parts(fullpad); |         remove_redundant_parts(fullpad); | ||||||
| 
 | 
 | ||||||
| @ -619,7 +447,7 @@ private: | |||||||
| 
 | 
 | ||||||
|     // Create the wafflized pad around all object in the scene. This pad doesnt
 |     // Create the wafflized pad around all object in the scene. This pad doesnt
 | ||||||
|     // have any holes yet.
 |     // have any holes yet.
 | ||||||
|     ConcaveHull wafflized_concave_hull(const ExPolygons &supp_bp, |     ExPolygons wafflized_concave_hull(const ExPolygons &supp_bp, | ||||||
|                                        const ExPolygons &model_bp, |                                        const ExPolygons &model_bp, | ||||||
|                                        const PadConfig  &cfg, |                                        const PadConfig  &cfg, | ||||||
|                                        ThrowOnCancel     thr) |                                        ThrowOnCancel     thr) | ||||||
| @ -629,10 +457,8 @@ private: | |||||||
|         for (auto &ep : supp_bp) allin.emplace_back(ep.contour); |         for (auto &ep : supp_bp) allin.emplace_back(ep.contour); | ||||||
|         for (auto &ep : model_bp) allin.emplace_back(ep.contour); |         for (auto &ep : model_bp) allin.emplace_back(ep.contour); | ||||||
| 
 | 
 | ||||||
|         ConcaveHull ret{allin, ConcaveHull::get_merge_distance(cfg), thr}; |         ConcaveHull cchull{allin, get_merge_distance(cfg), thr}; | ||||||
|         ret.offset_waffle_style(ConcaveHull::get_waffle_offset(cfg)); |         return offset_waffle_style_ex(cchull, get_waffle_offset(cfg)); | ||||||
| 
 |  | ||||||
|         return ret; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // To remove parts of the pad skeleton which do not host any supports
 |     // To remove parts of the pad skeleton which do not host any supports
 | ||||||
| @ -663,10 +489,9 @@ public: | |||||||
|         for (auto &ep : support_blueprint) outer.emplace_back(ep.contour); |         for (auto &ep : support_blueprint) outer.emplace_back(ep.contour); | ||||||
|         for (auto &ep : model_blueprint) outer.emplace_back(ep.contour); |         for (auto &ep : model_blueprint) outer.emplace_back(ep.contour); | ||||||
| 
 | 
 | ||||||
|         ConcaveHull ochull{outer, ConcaveHull::get_merge_distance(cfg), thr}; |         ConcaveHull ochull{outer, get_merge_distance(cfg), thr}; | ||||||
| 
 | 
 | ||||||
|         ochull.offset_waffle_style(ConcaveHull::get_waffle_offset(cfg)); |         outer = offset_waffle_style_ex(ochull, get_waffle_offset(cfg)); | ||||||
|         outer = ochull.to_expolygons(); |  | ||||||
|     } |     } | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| @ -861,7 +686,7 @@ std::string PadConfig::validate() const | |||||||
| 
 | 
 | ||||||
|     if (brim_size_mm < MIN_BRIM_SIZE_MM || |     if (brim_size_mm < MIN_BRIM_SIZE_MM || | ||||||
|         bottom_offset() > brim_size_mm + wing_distance() || |         bottom_offset() > brim_size_mm + wing_distance() || | ||||||
|         ConcaveHull::get_waffle_offset(*this) <= MIN_BRIM_SIZE_MM) |         get_waffle_offset(*this) <= MIN_BRIM_SIZE_MM) | ||||||
|         return L("Pad brim size is too small for the current configuration."); |         return L("Pad brim size is too small for the current configuration."); | ||||||
| 
 | 
 | ||||||
|     return ""; |     return ""; | ||||||
|  | |||||||
| @ -108,6 +108,8 @@ void GLCanvas3DManager::GLInfo::detect() const | |||||||
| 
 | 
 | ||||||
|     glsafe(::glGetIntegerv(GL_MAX_TEXTURE_SIZE, &m_max_tex_size)); |     glsafe(::glGetIntegerv(GL_MAX_TEXTURE_SIZE, &m_max_tex_size)); | ||||||
| 
 | 
 | ||||||
|  |     m_max_tex_size /= 2; | ||||||
|  | 
 | ||||||
|     if (GLEW_EXT_texture_filter_anisotropic) |     if (GLEW_EXT_texture_filter_anisotropic) | ||||||
|         glsafe(::glGetFloatv(GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT, &m_max_anisotropy)); |         glsafe(::glGetFloatv(GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT, &m_max_anisotropy)); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -2331,6 +2331,7 @@ void ObjectList::changed_object(const int obj_idx/* = -1*/) const | |||||||
| 
 | 
 | ||||||
| void ObjectList::part_selection_changed() | void ObjectList::part_selection_changed() | ||||||
| { | { | ||||||
|  |     if (m_extruder_editor) m_extruder_editor->Hide(); | ||||||
|     int obj_idx = -1; |     int obj_idx = -1; | ||||||
|     int volume_id = -1; |     int volume_id = -1; | ||||||
|     m_config = nullptr; |     m_config = nullptr; | ||||||
| @ -2954,7 +2955,6 @@ int ObjectList::get_selected_layers_range_idx() const | |||||||
| 
 | 
 | ||||||
| void ObjectList::update_selections() | void ObjectList::update_selections() | ||||||
| { | { | ||||||
|     if (m_extruder_editor) m_extruder_editor->Hide(); |  | ||||||
|     const Selection& selection = wxGetApp().plater()->canvas3D()->get_selection(); |     const Selection& selection = wxGetApp().plater()->canvas3D()->get_selection(); | ||||||
|     wxDataViewItemArray sels; |     wxDataViewItemArray sels; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -417,6 +417,9 @@ bool GLGizmosManager::on_mouse_wheel(wxMouseEvent& evt) | |||||||
| 
 | 
 | ||||||
| bool GLGizmosManager::on_mouse(wxMouseEvent& evt) | bool GLGizmosManager::on_mouse(wxMouseEvent& evt) | ||||||
| { | { | ||||||
|  |     // used to set a right up event as processed when needed
 | ||||||
|  |     static bool pending_right_up = false; | ||||||
|  | 
 | ||||||
|     Point pos(evt.GetX(), evt.GetY()); |     Point pos(evt.GetX(), evt.GetY()); | ||||||
|     Vec2d mouse_pos((double)evt.GetX(), (double)evt.GetY()); |     Vec2d mouse_pos((double)evt.GetX(), (double)evt.GetY()); | ||||||
| 
 | 
 | ||||||
| @ -442,7 +445,14 @@ bool GLGizmosManager::on_mouse(wxMouseEvent& evt) | |||||||
|     else if (evt.MiddleUp()) |     else if (evt.MiddleUp()) | ||||||
|         m_mouse_capture.middle = false; |         m_mouse_capture.middle = false; | ||||||
|     else if (evt.RightUp()) |     else if (evt.RightUp()) | ||||||
|  |     { | ||||||
|         m_mouse_capture.right = false; |         m_mouse_capture.right = false; | ||||||
|  |         if (pending_right_up) | ||||||
|  |         { | ||||||
|  |             pending_right_up = false; | ||||||
|  |             processed = true; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|     else if (evt.Dragging() && m_mouse_capture.any()) |     else if (evt.Dragging() && m_mouse_capture.any()) | ||||||
|         // if the button down was done on this toolbar, prevent from dragging into the scene
 |         // if the button down was done on this toolbar, prevent from dragging into the scene
 | ||||||
|         processed = true; |         processed = true; | ||||||
| @ -473,8 +483,12 @@ bool GLGizmosManager::on_mouse(wxMouseEvent& evt) | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         else if (evt.RightDown() && (selected_object_idx != -1) && (m_current == SlaSupports) && gizmo_event(SLAGizmoEventType::RightDown)) |         else if (evt.RightDown() && (selected_object_idx != -1) && (m_current == SlaSupports) && gizmo_event(SLAGizmoEventType::RightDown)) | ||||||
|  |         { | ||||||
|  |             // we need to set the following right up as processed to avoid showing the context menu if the user release the mouse over the object
 | ||||||
|  |             pending_right_up = true; | ||||||
|             // event was taken care of by the SlaSupports gizmo
 |             // event was taken care of by the SlaSupports gizmo
 | ||||||
|             processed = true; |             processed = true; | ||||||
|  |         } | ||||||
|         else if (evt.Dragging() && (m_parent.get_move_volume_id() != -1) && (m_current == SlaSupports)) |         else if (evt.Dragging() && (m_parent.get_move_volume_id() != -1) && (m_current == SlaSupports)) | ||||||
|                         // don't allow dragging objects with the Sla gizmo on
 |                         // don't allow dragging objects with the Sla gizmo on
 | ||||||
|             processed = true; |             processed = true; | ||||||
|  | |||||||
| @ -3837,9 +3837,9 @@ bool Plater::priv::can_reload_from_disk() const | |||||||
|     for (unsigned int idx : selected_volumes_idxs) |     for (unsigned int idx : selected_volumes_idxs) | ||||||
|     { |     { | ||||||
|         const GLVolume* v = selection.get_volume(idx); |         const GLVolume* v = selection.get_volume(idx); | ||||||
|         int o_idx = v->object_idx(); |  | ||||||
|         int v_idx = v->volume_idx(); |         int v_idx = v->volume_idx(); | ||||||
|         selected_volumes.push_back({ o_idx, v_idx }); |         if (v_idx >= 0) | ||||||
|  |             selected_volumes.push_back({ v->object_idx(), v_idx }); | ||||||
|     } |     } | ||||||
|     std::sort(selected_volumes.begin(), selected_volumes.end()); |     std::sort(selected_volumes.begin(), selected_volumes.end()); | ||||||
|     selected_volumes.erase(std::unique(selected_volumes.begin(), selected_volumes.end()), selected_volumes.end()); |     selected_volumes.erase(std::unique(selected_volumes.begin(), selected_volumes.end()), selected_volumes.end()); | ||||||
|  | |||||||
| @ -2179,7 +2179,12 @@ wxWindow* BitmapChoiceRenderer::CreateEditorCtrl(wxWindow* parent, wxRect labelR | |||||||
|     c_editor->SetSelection(atoi(data.GetText().c_str())); |     c_editor->SetSelection(atoi(data.GetText().c_str())); | ||||||
| 
 | 
 | ||||||
|     // to avoid event propagation to other sidebar items
 |     // to avoid event propagation to other sidebar items
 | ||||||
|     c_editor->Bind(wxEVT_COMBOBOX, [](wxCommandEvent& evt) { evt.StopPropagation(); }); |     c_editor->Bind(wxEVT_COMBOBOX, [this](wxCommandEvent& evt) { | ||||||
|  |             evt.StopPropagation(); | ||||||
|  |             // FinishEditing grabs new selection and triggers config update. We better call
 | ||||||
|  |             // it explicitly, automatic update on KILL_FOCUS didn't work on Linux.
 | ||||||
|  |             this->FinishEditing(); | ||||||
|  |     }); | ||||||
| 
 | 
 | ||||||
|     return c_editor; |     return c_editor; | ||||||
| } | } | ||||||
|  | |||||||
| @ -17,6 +17,7 @@ | |||||||
| #include "libslic3r/SLA/SLASupportTreeBuildsteps.hpp" | #include "libslic3r/SLA/SLASupportTreeBuildsteps.hpp" | ||||||
| #include "libslic3r/SLA/SLAAutoSupports.hpp" | #include "libslic3r/SLA/SLAAutoSupports.hpp" | ||||||
| #include "libslic3r/SLA/SLARaster.hpp" | #include "libslic3r/SLA/SLARaster.hpp" | ||||||
|  | #include "libslic3r/SLA/ConcaveHull.hpp" | ||||||
| #include "libslic3r/MTUtils.hpp" | #include "libslic3r/MTUtils.hpp" | ||||||
| 
 | 
 | ||||||
| #include "libslic3r/SVG.hpp" | #include "libslic3r/SVG.hpp" | ||||||
| @ -79,6 +80,43 @@ struct PadByproducts | |||||||
|     TriangleMesh mesh; |     TriangleMesh mesh; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | void _test_concave_hull(const Polygons &hull, const ExPolygons &polys) | ||||||
|  | { | ||||||
|  |     REQUIRE(polys.size() >=hull.size()); | ||||||
|  | 
 | ||||||
|  |     double polys_area = 0; | ||||||
|  |     for (const ExPolygon &p : polys) polys_area += p.area(); | ||||||
|  | 
 | ||||||
|  |     double cchull_area = 0; | ||||||
|  |     for (const Slic3r::Polygon &p : hull) cchull_area += p.area(); | ||||||
|  | 
 | ||||||
|  |     REQUIRE(cchull_area >= Approx(polys_area)); | ||||||
|  | 
 | ||||||
|  |     size_t cchull_holes = 0; | ||||||
|  |     for (const Slic3r::Polygon &p : hull) | ||||||
|  |         cchull_holes += p.is_clockwise() ? 1 : 0; | ||||||
|  | 
 | ||||||
|  |     REQUIRE(cchull_holes == 0); | ||||||
|  | 
 | ||||||
|  |     Polygons intr = diff(to_polygons(polys), hull); | ||||||
|  |     REQUIRE(intr.empty()); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void test_concave_hull(const ExPolygons &polys) { | ||||||
|  |     sla::PadConfig pcfg; | ||||||
|  | 
 | ||||||
|  |     Slic3r::sla::ConcaveHull cchull{polys, pcfg.max_merge_dist_mm, []{}}; | ||||||
|  | 
 | ||||||
|  |     _test_concave_hull(cchull.polygons(), polys); | ||||||
|  | 
 | ||||||
|  |     coord_t delta = scaled(pcfg.brim_size_mm + pcfg.wing_distance()); | ||||||
|  |     ExPolygons wafflex = sla::offset_waffle_style_ex(cchull, delta); | ||||||
|  |     Polygons waffl = sla::offset_waffle_style(cchull, delta); | ||||||
|  | 
 | ||||||
|  |     _test_concave_hull(to_polygons(wafflex), polys); | ||||||
|  |     _test_concave_hull(waffl, polys); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void test_pad(const std::string &   obj_filename, | void test_pad(const std::string &   obj_filename, | ||||||
|               const sla::PadConfig &padcfg, |               const sla::PadConfig &padcfg, | ||||||
|               PadByproducts &       out) |               PadByproducts &       out) | ||||||
| @ -92,6 +130,8 @@ void test_pad(const std::string &   obj_filename, | |||||||
|     // Create pad skeleton only from the model
 |     // Create pad skeleton only from the model
 | ||||||
|     Slic3r::sla::pad_blueprint(mesh, out.model_contours); |     Slic3r::sla::pad_blueprint(mesh, out.model_contours); | ||||||
| 
 | 
 | ||||||
|  |     test_concave_hull(out.model_contours); | ||||||
|  | 
 | ||||||
|     REQUIRE_FALSE(out.model_contours.empty()); |     REQUIRE_FALSE(out.model_contours.empty()); | ||||||
| 
 | 
 | ||||||
|     // Create the pad geometry for the model contours only
 |     // Create the pad geometry for the model contours only
 | ||||||
| @ -362,7 +402,7 @@ template <class I, class II> void test_pairhash() | |||||||
| 
 | 
 | ||||||
|     int bits = IIbits / 2 < Ibits ? Ibits / 2 : Ibits; |     int bits = IIbits / 2 < Ibits ? Ibits / 2 : Ibits; | ||||||
|     if (std::is_signed<I>::value) bits -= 1; |     if (std::is_signed<I>::value) bits -= 1; | ||||||
|     const I Imin = std::is_signed<I>::value ? -I(std::pow(2., bits)) : 0; |     const I Imin = 0; | ||||||
|     const I Imax = I(std::pow(2., bits) - 1); |     const I Imax = I(std::pow(2., bits) - 1); | ||||||
| 
 | 
 | ||||||
|     std::uniform_int_distribution<I> dis(Imin, Imax); |     std::uniform_int_distribution<I> dis(Imin, Imax); | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 YuSanka
						YuSanka