mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-26 10:11:10 -06:00 
			
		
		
		
	Merge branch 'master' of https://github.com/prusa3d/PrusaSlicer into et_gcode_viewer
This commit is contained in:
		
						commit
						6a603eed25
					
				
					 13 changed files with 212 additions and 144 deletions
				
			
		|  | @ -126,9 +126,9 @@ uqptr<sla::RasterBase> SL1Archive::create_raster() const | ||||||
|     return sla::create_raster_grayscale_aa(res, pxdim, gamma, tr); |     return sla::create_raster_grayscale_aa(res, pxdim, gamma, tr); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| sla::EncodedRaster SL1Archive::encode_raster(const sla::RasterBase &rst) const | sla::RasterEncoder SL1Archive::get_encoder() const | ||||||
| { | { | ||||||
|     return rst.encode(sla::PNGRasterEncoder());     |     return sla::PNGRasterEncoder{}; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void SL1Archive::export_print(Zipper& zipper, | void SL1Archive::export_print(Zipper& zipper, | ||||||
|  |  | ||||||
|  | @ -13,7 +13,7 @@ class SL1Archive: public SLAPrinter { | ||||||
|      |      | ||||||
| protected: | protected: | ||||||
|     uqptr<sla::RasterBase> create_raster() const override; |     uqptr<sla::RasterBase> create_raster() const override; | ||||||
|     sla::EncodedRaster encode_raster(const sla::RasterBase &rst) const override; |     sla::RasterEncoder get_encoder() const override; | ||||||
|      |      | ||||||
| public: | public: | ||||||
|      |      | ||||||
|  |  | ||||||
|  | @ -114,15 +114,6 @@ template<class T> struct remove_cvref | ||||||
| 
 | 
 | ||||||
| template<class T> using remove_cvref_t = typename remove_cvref<T>::type; | template<class T> using remove_cvref_t = typename remove_cvref<T>::type; | ||||||
| 
 | 
 | ||||||
| template<class T, class I, class... Args> // Arbitrary allocator can be used
 |  | ||||||
| inline IntegerOnly<I, std::vector<T, Args...>> reserve_vector(I capacity) |  | ||||||
| { |  | ||||||
|     std::vector<T, Args...> ret; |  | ||||||
|     if (capacity > I(0)) ret.reserve(size_t(capacity)); |  | ||||||
|      |  | ||||||
|     return ret; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /// Exactly like Matlab https://www.mathworks.com/help/matlab/ref/linspace.html
 | /// Exactly like Matlab https://www.mathworks.com/help/matlab/ref/linspace.html
 | ||||||
| template<class T, class I, class = IntegerOnly<I>> | template<class T, class I, class = IntegerOnly<I>> | ||||||
| inline std::vector<T> linspace_vector(const ArithmeticOnly<T> &start,  | inline std::vector<T> linspace_vector(const ArithmeticOnly<T> &start,  | ||||||
|  |  | ||||||
|  | @ -4,6 +4,8 @@ | ||||||
| #include <tbb/spin_mutex.h> | #include <tbb/spin_mutex.h> | ||||||
| #include <tbb/mutex.h> | #include <tbb/mutex.h> | ||||||
| #include <tbb/parallel_for.h> | #include <tbb/parallel_for.h> | ||||||
|  | #include <algorithm> | ||||||
|  | #include <libslic3r/libslic3r.h> | ||||||
| 
 | 
 | ||||||
| namespace Slic3r { | namespace Slic3r { | ||||||
| namespace sla { | namespace sla { | ||||||
|  | @ -20,13 +22,26 @@ template<> struct _ccr<true> | ||||||
|     using BlockingMutex = tbb::mutex; |     using BlockingMutex = tbb::mutex; | ||||||
| 
 | 
 | ||||||
|     template<class It, class Fn> |     template<class It, class Fn> | ||||||
|     static inline void enumerate(It from, It to, Fn fn) |     static IteratorOnly<It, void> for_each(It     from, | ||||||
|  |                                            It     to, | ||||||
|  |                                            Fn &&  fn, | ||||||
|  |                                            size_t granularity = 1) | ||||||
|     { |     { | ||||||
|         auto   iN = to - from; |         tbb::parallel_for(tbb::blocked_range{from, to, granularity}, | ||||||
|         size_t N  = iN < 0 ? 0 : size_t(iN); |                           [&fn, from](const auto &range) { | ||||||
|  |             for (auto &el : range) fn(el); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|         tbb::parallel_for(size_t(0), N, [from, fn](size_t n) { |     template<class I, class Fn> | ||||||
|             fn(*(from + decltype(iN)(n)), n); |     static IntegerOnly<I, void> for_each(I      from, | ||||||
|  |                                          I      to, | ||||||
|  |                                          Fn &&  fn, | ||||||
|  |                                          size_t granularity = 1) | ||||||
|  |     { | ||||||
|  |         tbb::parallel_for(tbb::blocked_range{from, to, granularity}, | ||||||
|  |                           [&fn](const auto &range) { | ||||||
|  |             for (I i = range.begin(); i < range.end(); ++i) fn(i); | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
| }; | }; | ||||||
|  | @ -41,9 +56,21 @@ public: | ||||||
|     using BlockingMutex = _Mtx; |     using BlockingMutex = _Mtx; | ||||||
| 
 | 
 | ||||||
|     template<class It, class Fn> |     template<class It, class Fn> | ||||||
|     static inline void enumerate(It from, It to, Fn fn) |     static IteratorOnly<It, void> for_each(It   from, | ||||||
|  |                                            It   to, | ||||||
|  |                                            Fn &&fn, | ||||||
|  |                                            size_t /* ignore granularity */ = 1) | ||||||
|     { |     { | ||||||
|         for (auto it = from; it != to; ++it) fn(*it, size_t(it - from)); |         for (auto it = from; it != to; ++it) fn(*it); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     template<class I, class Fn> | ||||||
|  |     static IntegerOnly<I, void> for_each(I    from, | ||||||
|  |                                          I    to, | ||||||
|  |                                          Fn &&fn, | ||||||
|  |                                          size_t /* ignore granularity */ = 1) | ||||||
|  |     { | ||||||
|  |         for (I i = from; i < to; ++i) fn(i); | ||||||
|     } |     } | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -320,10 +320,10 @@ PointSet normals(const PointSet& points, | ||||||
|     PointSet ret(range.size(), 3); |     PointSet ret(range.size(), 3); | ||||||
| 
 | 
 | ||||||
|     //    for (size_t ridx = 0; ridx < range.size(); ++ridx)
 |     //    for (size_t ridx = 0; ridx < range.size(); ++ridx)
 | ||||||
|     ccr::enumerate( |     ccr::for_each(size_t(0), range.size(), | ||||||
|         range.begin(), range.end(), |         [&ret, &mesh, &points, thr, eps, &range](size_t ridx) { | ||||||
|         [&ret, &mesh, &points, thr, eps](unsigned el, size_t ridx) { |  | ||||||
|             thr(); |             thr(); | ||||||
|  |             unsigned el = range[ridx]; | ||||||
|             auto  eidx   = Eigen::Index(el); |             auto  eidx   = Eigen::Index(el); | ||||||
|             int   faceid = 0; |             int   faceid = 0; | ||||||
|             Vec3d p; |             Vec3d p; | ||||||
|  |  | ||||||
|  | @ -4,6 +4,7 @@ | ||||||
| #include <tbb/parallel_for.h> | #include <tbb/parallel_for.h> | ||||||
| 
 | 
 | ||||||
| #include "SupportPointGenerator.hpp" | #include "SupportPointGenerator.hpp" | ||||||
|  | #include "Concurrency.hpp" | ||||||
| #include "Model.hpp" | #include "Model.hpp" | ||||||
| #include "ExPolygon.hpp" | #include "ExPolygon.hpp" | ||||||
| #include "SVG.hpp" | #include "SVG.hpp" | ||||||
|  | @ -87,13 +88,15 @@ void SupportPointGenerator::project_onto_mesh(std::vector<sla::SupportPoint>& po | ||||||
|     // The function  makes sure that all the points are really exactly placed on the mesh.
 |     // The function  makes sure that all the points are really exactly placed on the mesh.
 | ||||||
| 
 | 
 | ||||||
|     // Use a reasonable granularity to account for the worker thread synchronization cost.
 |     // Use a reasonable granularity to account for the worker thread synchronization cost.
 | ||||||
|     tbb::parallel_for(tbb::blocked_range<size_t>(0, points.size(), 64), |     static constexpr size_t gransize = 64; | ||||||
|         [this, &points](const tbb::blocked_range<size_t>& range) { | 
 | ||||||
|             for (size_t point_id = range.begin(); point_id < range.end(); ++ point_id) { |     ccr_par::for_each(size_t(0), points.size(), [this, &points](size_t idx) | ||||||
|                 if ((point_id % 16) == 0) |     { | ||||||
|  |         if ((idx % 16) == 0) | ||||||
|             // Don't call the following function too often as it flushes CPU write caches due to synchronization primitves.
 |             // Don't call the following function too often as it flushes CPU write caches due to synchronization primitves.
 | ||||||
|             m_throw_on_cancel(); |             m_throw_on_cancel(); | ||||||
|                 Vec3f& p = points[point_id].pos; | 
 | ||||||
|  |         Vec3f& p = points[idx].pos; | ||||||
|         // Project the point upward and downward and choose the closer intersection with the mesh.
 |         // Project the point upward and downward and choose the closer intersection with the mesh.
 | ||||||
|         sla::IndexedMesh::hit_result hit_up   = m_emesh.query_ray_hit(p.cast<double>(), Vec3d(0., 0., 1.)); |         sla::IndexedMesh::hit_result hit_up   = m_emesh.query_ray_hit(p.cast<double>(), Vec3d(0., 0., 1.)); | ||||||
|         sla::IndexedMesh::hit_result hit_down = m_emesh.query_ray_hit(p.cast<double>(), Vec3d(0., 0., -1.)); |         sla::IndexedMesh::hit_result hit_down = m_emesh.query_ray_hit(p.cast<double>(), Vec3d(0., 0., -1.)); | ||||||
|  | @ -102,12 +105,11 @@ void SupportPointGenerator::project_onto_mesh(std::vector<sla::SupportPoint>& po | ||||||
|         bool down = hit_down.is_hit(); |         bool down = hit_down.is_hit(); | ||||||
| 
 | 
 | ||||||
|         if (!up && !down) |         if (!up && !down) | ||||||
|                     continue; |             return; | ||||||
| 
 | 
 | ||||||
|         sla::IndexedMesh::hit_result& hit = (!down || (hit_up.distance() < hit_down.distance())) ? hit_up : hit_down; |         sla::IndexedMesh::hit_result& hit = (!down || (hit_up.distance() < hit_down.distance())) ? hit_up : hit_down; | ||||||
|         p = p + (hit.distance() * hit.direction()).cast<float>(); |         p = p + (hit.distance() * hit.direction()).cast<float>(); | ||||||
|             } |     }, gransize); | ||||||
|         }); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static std::vector<SupportPointGenerator::MyLayer> make_layers( | static std::vector<SupportPointGenerator::MyLayer> make_layers( | ||||||
|  | @ -126,31 +128,34 @@ static std::vector<SupportPointGenerator::MyLayer> make_layers( | ||||||
|     //const float pixel_area = pow(wxGetApp().preset_bundle->project_config.option<ConfigOptionFloat>("display_width") / wxGetApp().preset_bundle->project_config.option<ConfigOptionInt>("display_pixels_x"), 2.f); //
 |     //const float pixel_area = pow(wxGetApp().preset_bundle->project_config.option<ConfigOptionFloat>("display_width") / wxGetApp().preset_bundle->project_config.option<ConfigOptionInt>("display_pixels_x"), 2.f); //
 | ||||||
|     const float pixel_area = pow(0.047f, 2.f); |     const float pixel_area = pow(0.047f, 2.f); | ||||||
| 
 | 
 | ||||||
|     // Use a reasonable granularity to account for the worker thread synchronization cost.
 |     ccr_par::for_each(size_t(0), layers.size(), | ||||||
|     tbb::parallel_for(tbb::blocked_range<size_t>(0, layers.size(), 32), |         [&layers, &slices, &heights, pixel_area, throw_on_cancel](size_t layer_id) | ||||||
|         [&layers, &slices, &heights, pixel_area, throw_on_cancel](const tbb::blocked_range<size_t>& range) { |     { | ||||||
|             for (size_t layer_id = range.begin(); layer_id < range.end(); ++ layer_id) { |  | ||||||
|         if ((layer_id % 8) == 0) |         if ((layer_id % 8) == 0) | ||||||
|                     // Don't call the following function too often as it flushes CPU write caches due to synchronization primitves.
 |             // Don't call the following function too often as it flushes
 | ||||||
|  |             // CPU write caches due to synchronization primitves.
 | ||||||
|             throw_on_cancel(); |             throw_on_cancel(); | ||||||
|  | 
 | ||||||
|         SupportPointGenerator::MyLayer &layer   = layers[layer_id]; |         SupportPointGenerator::MyLayer &layer   = layers[layer_id]; | ||||||
|                 const ExPolygons		 &islands = slices[layer_id]; |         const ExPolygons &              islands = slices[layer_id]; | ||||||
|                 //FIXME WTF?
 |         // FIXME WTF?
 | ||||||
|                 const float height = (layer_id>2 ? heights[layer_id-3] : heights[0]-(heights[1]-heights[0])); |         const float height = (layer_id > 2 ? | ||||||
|  |                                   heights[layer_id - 3] : | ||||||
|  |                                   heights[0] - (heights[1] - heights[0])); | ||||||
|         layer.islands.reserve(islands.size()); |         layer.islands.reserve(islands.size()); | ||||||
|         for (const ExPolygon &island : islands) { |         for (const ExPolygon &island : islands) { | ||||||
|             float area = float(island.area() * SCALING_FACTOR * SCALING_FACTOR); |             float area = float(island.area() * SCALING_FACTOR * SCALING_FACTOR); | ||||||
|             if (area >= pixel_area) |             if (area >= pixel_area) | ||||||
|                         //FIXME this is not a correct centroid of a polygon with holes.
 |                 // FIXME this is not a correct centroid of a polygon with holes.
 | ||||||
|                         layer.islands.emplace_back(layer, island, get_extents(island.contour), Slic3r::unscale(island.contour.centroid()).cast<float>(), area, height); |                 layer.islands.emplace_back(layer, island, get_extents(island.contour), | ||||||
|  |                                            unscaled<float>(island.contour.centroid()), area, height); | ||||||
|         } |         } | ||||||
|             } |     }, 32 /*gransize*/); | ||||||
|         }); |  | ||||||
| 
 | 
 | ||||||
|     // Calculate overlap of successive layers. Link overlapping islands.
 |     // Calculate overlap of successive layers. Link overlapping islands.
 | ||||||
|     tbb::parallel_for(tbb::blocked_range<size_t>(1, layers.size(), 8), |     ccr_par::for_each(size_t(1), layers.size(), | ||||||
|         [&layers, &heights, throw_on_cancel](const tbb::blocked_range<size_t>& range) { |                       [&layers, &heights, throw_on_cancel] (size_t layer_id) | ||||||
|         for (size_t layer_id = range.begin(); layer_id < range.end(); ++layer_id) { |     { | ||||||
|       if ((layer_id % 2) == 0) |       if ((layer_id % 2) == 0) | ||||||
|           // Don't call the following function too often as it flushes CPU write caches due to synchronization primitves.
 |           // Don't call the following function too often as it flushes CPU write caches due to synchronization primitves.
 | ||||||
|           throw_on_cancel(); |           throw_on_cancel(); | ||||||
|  | @ -196,8 +201,7 @@ static std::vector<SupportPointGenerator::MyLayer> make_layers( | ||||||
|               } |               } | ||||||
|           } |           } | ||||||
|       } |       } | ||||||
|         } |     }, 8 /* gransize */); | ||||||
|     }); |  | ||||||
| 
 | 
 | ||||||
|     return layers; |     return layers; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -209,14 +209,16 @@ IndexedMesh::hit_result SupportTreeBuildsteps::pinhead_mesh_intersect( | ||||||
|     // of the pinhead robe (side) surface. The result will be the smallest
 |     // of the pinhead robe (side) surface. The result will be the smallest
 | ||||||
|     // hit distance.
 |     // hit distance.
 | ||||||
| 
 | 
 | ||||||
|     ccr::enumerate(hits.begin(), hits.end(), |     ccr::for_each(size_t(0), hits.size(), | ||||||
|                    [&m, &rings, sd](HitResult &hit, size_t i) { |                   [&m, &rings, sd, &hits](size_t i) { | ||||||
| 
 | 
 | ||||||
|        // Point on the circle on the pin sphere
 |        // Point on the circle on the pin sphere
 | ||||||
|        Vec3d ps = rings.pinring(i); |        Vec3d ps = rings.pinring(i); | ||||||
|        // This is the point on the circle on the back sphere
 |        // This is the point on the circle on the back sphere
 | ||||||
|        Vec3d p = rings.backring(i); |        Vec3d p = rings.backring(i); | ||||||
| 
 | 
 | ||||||
|  |        auto &hit = hits[i]; | ||||||
|  | 
 | ||||||
|        // Point ps is not on mesh but can be inside or
 |        // Point ps is not on mesh but can be inside or
 | ||||||
|        // outside as well. This would cause many problems
 |        // outside as well. This would cause many problems
 | ||||||
|        // with ray-casting. To detect the position we will
 |        // with ray-casting. To detect the position we will
 | ||||||
|  | @ -265,8 +267,10 @@ IndexedMesh::hit_result SupportTreeBuildsteps::bridge_mesh_intersect( | ||||||
|     // Hit results
 |     // Hit results
 | ||||||
|     std::array<Hit, SAMPLES> hits; |     std::array<Hit, SAMPLES> hits; | ||||||
| 
 | 
 | ||||||
|     ccr::enumerate(hits.begin(), hits.end(), |     ccr::for_each(size_t(0), hits.size(), | ||||||
|                 [this, r, src, /*ins_check,*/ &ring, dir, sd] (Hit &hit, size_t i) { |                  [this, r, src, /*ins_check,*/ &ring, dir, sd, &hits] (size_t i) | ||||||
|  |     { | ||||||
|  |         Hit &hit = hits[i]; | ||||||
| 
 | 
 | ||||||
|         // Point on the circle on the pin sphere
 |         // Point on the circle on the pin sphere
 | ||||||
|         Vec3d p = ring.get(i, src, r + sd); |         Vec3d p = ring.get(i, src, r + sd); | ||||||
|  | @ -744,9 +748,9 @@ void SupportTreeBuildsteps::filter() | ||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     ccr::enumerate(filtered_indices.begin(), filtered_indices.end(), |     ccr::for_each(size_t(0), filtered_indices.size(), | ||||||
|                    [this, &filterfn](unsigned fidx, size_t i) { |                   [this, &filterfn, &filtered_indices] (size_t i) { | ||||||
|                        filterfn(fidx, i, m_cfg.head_back_radius_mm); |                       filterfn(filtered_indices[i], i, m_cfg.head_back_radius_mm); | ||||||
|                   }); |                   }); | ||||||
| 
 | 
 | ||||||
|     for (size_t i = 0; i < heads.size(); ++i) |     for (size_t i = 0; i < heads.size(); ++i) | ||||||
|  | @ -1033,8 +1037,8 @@ void SupportTreeBuildsteps::routing_to_model() | ||||||
|     // If it can be routed there with a bridge shorter than
 |     // If it can be routed there with a bridge shorter than
 | ||||||
|     // min_bridge_distance.
 |     // min_bridge_distance.
 | ||||||
| 
 | 
 | ||||||
|     ccr::enumerate(m_iheads_onmodel.begin(), m_iheads_onmodel.end(), |     ccr::for_each(m_iheads_onmodel.begin(), m_iheads_onmodel.end(), | ||||||
|                    [this] (const unsigned idx, size_t) { |                   [this] (const unsigned idx) { | ||||||
|         m_thr(); |         m_thr(); | ||||||
| 
 | 
 | ||||||
|         auto& head = m_builder.head(idx); |         auto& head = m_builder.head(idx); | ||||||
|  |  | ||||||
|  | @ -374,7 +374,7 @@ protected: | ||||||
|     std::vector<sla::EncodedRaster> m_layers; |     std::vector<sla::EncodedRaster> m_layers; | ||||||
|      |      | ||||||
|     virtual uqptr<sla::RasterBase> create_raster() const = 0; |     virtual uqptr<sla::RasterBase> create_raster() const = 0; | ||||||
|     virtual sla::EncodedRaster encode_raster(const sla::RasterBase &rst) const = 0; |     virtual sla::RasterEncoder get_encoder() const = 0; | ||||||
|      |      | ||||||
| public: | public: | ||||||
|     virtual ~SLAPrinter() = default; |     virtual ~SLAPrinter() = default; | ||||||
|  | @ -385,11 +385,12 @@ public: | ||||||
|     template<class Fn> void draw_layers(size_t layer_num, Fn &&drawfn) |     template<class Fn> void draw_layers(size_t layer_num, Fn &&drawfn) | ||||||
|     { |     { | ||||||
|         m_layers.resize(layer_num); |         m_layers.resize(layer_num); | ||||||
|         sla::ccr::enumerate(m_layers.begin(), m_layers.end(), |         sla::ccr::for_each(size_t(0), m_layers.size(), | ||||||
|                             [this, &drawfn](sla::EncodedRaster& enc, size_t idx) { |                            [this, &drawfn] (size_t idx) { | ||||||
|  |                                sla::EncodedRaster& enc = m_layers[idx]; | ||||||
|                                auto rst = create_raster(); |                                auto rst = create_raster(); | ||||||
|                                drawfn(*rst, idx); |                                drawfn(*rst, idx); | ||||||
|                                 enc = encode_raster(*rst); |                                enc = rst->encode(get_encoder()); | ||||||
|                            }); |                            }); | ||||||
|     } |     } | ||||||
| }; | }; | ||||||
|  |  | ||||||
|  | @ -264,8 +264,9 @@ void SLAPrint::Steps::slice_model(SLAPrintObject &po) | ||||||
|         std::vector<ExPolygons> interior_slices; |         std::vector<ExPolygons> interior_slices; | ||||||
|         interior_slicer.slice(slice_grid, SlicingMode::Regular, closing_r, &interior_slices, thr); |         interior_slicer.slice(slice_grid, SlicingMode::Regular, closing_r, &interior_slices, thr); | ||||||
| 
 | 
 | ||||||
|         sla::ccr::enumerate(interior_slices.begin(), interior_slices.end(), |         sla::ccr::for_each(size_t(0), interior_slices.size(), | ||||||
|                             [&po](const ExPolygons &slice, size_t i) { |                            [&po, &interior_slices] (size_t i) { | ||||||
|  |                               const ExPolygons &slice = interior_slices[i]; | ||||||
|                               po.m_model_slices[i] = |                               po.m_model_slices[i] = | ||||||
|                                   diff_ex(po.m_model_slices[i], slice); |                                   diff_ex(po.m_model_slices[i], slice); | ||||||
|                            }); |                            }); | ||||||
|  | @ -679,14 +680,16 @@ void SLAPrint::Steps::merge_slices_and_eval_stats() { | ||||||
|     using Lock = std::lock_guard<sla::ccr::SpinningMutex>; |     using Lock = std::lock_guard<sla::ccr::SpinningMutex>; | ||||||
|      |      | ||||||
|     // Going to parallel:
 |     // Going to parallel:
 | ||||||
|     auto printlayerfn = [ |     auto printlayerfn = [this, | ||||||
|             // functions and read only vars
 |             // functions and read only vars
 | ||||||
|             areafn, area_fill, display_area, exp_time, init_exp_time, fast_tilt, slow_tilt, delta_fade_time, |             areafn, area_fill, display_area, exp_time, init_exp_time, fast_tilt, slow_tilt, delta_fade_time, | ||||||
|              |              | ||||||
|             // write vars
 |             // write vars
 | ||||||
|             &mutex, &models_volume, &supports_volume, &estim_time, &slow_layers, |             &mutex, &models_volume, &supports_volume, &estim_time, &slow_layers, | ||||||
|             &fast_layers, &fade_layer_time](PrintLayer& layer, size_t sliced_layer_cnt) |             &fast_layers, &fade_layer_time](size_t sliced_layer_cnt) | ||||||
|     { |     { | ||||||
|  |         PrintLayer &layer = m_print->m_printer_input[sliced_layer_cnt]; | ||||||
|  | 
 | ||||||
|         // vector of slice record references
 |         // vector of slice record references
 | ||||||
|         auto& slicerecord_references = layer.slices(); |         auto& slicerecord_references = layer.slices(); | ||||||
|          |          | ||||||
|  | @ -789,7 +792,7 @@ void SLAPrint::Steps::merge_slices_and_eval_stats() { | ||||||
|      |      | ||||||
|     // sequential version for debugging:
 |     // sequential version for debugging:
 | ||||||
|     // for(size_t i = 0; i < m_printer_input.size(); ++i) printlayerfn(i);
 |     // for(size_t i = 0; i < m_printer_input.size(); ++i) printlayerfn(i);
 | ||||||
|     sla::ccr::enumerate(printer_input.begin(), printer_input.end(), printlayerfn); |     sla::ccr::for_each(size_t(0), printer_input.size(), printlayerfn); | ||||||
|      |      | ||||||
|     auto SCALING2 = SCALING_FACTOR * SCALING_FACTOR; |     auto SCALING2 = SCALING_FACTOR * SCALING_FACTOR; | ||||||
|     print_statistics.support_used_material = supports_volume * SCALING2; |     print_statistics.support_used_material = supports_volume * SCALING2; | ||||||
|  |  | ||||||
|  | @ -261,6 +261,20 @@ using IntegerOnly = std::enable_if_t<std::is_integral<T>::value, O>; | ||||||
| template<class T, class O = T> | template<class T, class O = T> | ||||||
| using ArithmeticOnly = std::enable_if_t<std::is_arithmetic<T>::value, O>; | using ArithmeticOnly = std::enable_if_t<std::is_arithmetic<T>::value, O>; | ||||||
| 
 | 
 | ||||||
|  | template<class T, class O = T> | ||||||
|  | using IteratorOnly = std::enable_if_t< | ||||||
|  |     !std::is_same_v<typename std::iterator_traits<T>::value_type, void>, O | ||||||
|  | >; | ||||||
|  | 
 | ||||||
|  | template<class T, class I, class... Args> // Arbitrary allocator can be used
 | ||||||
|  | IntegerOnly<I, std::vector<T, Args...>> reserve_vector(I capacity) | ||||||
|  | { | ||||||
|  |     std::vector<T, Args...> ret; | ||||||
|  |     if (capacity > I(0)) ret.reserve(size_t(capacity)); | ||||||
|  | 
 | ||||||
|  |     return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| } // namespace Slic3r
 | } // namespace Slic3r
 | ||||||
| 
 | 
 | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | @ -715,7 +715,7 @@ void PlaterPresetComboBox::update() | ||||||
| 
 | 
 | ||||||
|     std::map<wxString, wxBitmap*> nonsys_presets; |     std::map<wxString, wxBitmap*> nonsys_presets; | ||||||
| 
 | 
 | ||||||
|     wxString selected = ""; |     wxString selected_user_preset = ""; | ||||||
|     wxString tooltip = ""; |     wxString tooltip = ""; | ||||||
|     const std::deque<Preset>& presets = m_collection->get_presets(); |     const std::deque<Preset>& presets = m_collection->get_presets(); | ||||||
| 
 | 
 | ||||||
|  | @ -742,7 +742,7 @@ void PlaterPresetComboBox::update() | ||||||
|         { |         { | ||||||
|             // Assign an extruder color to the selected item if the extruder color is defined.
 |             // Assign an extruder color to the selected item if the extruder color is defined.
 | ||||||
|             filament_rgb = preset.config.opt_string("filament_colour", 0); |             filament_rgb = preset.config.opt_string("filament_colour", 0); | ||||||
|             extruder_rgb = (selected && !extruder_color.empty()) ? extruder_color : filament_rgb; |             extruder_rgb = (is_selected && !extruder_color.empty()) ? extruder_color : filament_rgb; | ||||||
|             single_bar = filament_rgb == extruder_rgb; |             single_bar = filament_rgb == extruder_rgb; | ||||||
| 
 | 
 | ||||||
|             bitmap_key += single_bar ? filament_rgb : filament_rgb + extruder_rgb; |             bitmap_key += single_bar ? filament_rgb : filament_rgb + extruder_rgb; | ||||||
|  | @ -764,7 +764,7 @@ void PlaterPresetComboBox::update() | ||||||
|         { |         { | ||||||
|             nonsys_presets.emplace(wxString::FromUTF8((name + (preset.is_dirty ? Preset::suffix_modified() : "")).c_str()), bmp); |             nonsys_presets.emplace(wxString::FromUTF8((name + (preset.is_dirty ? Preset::suffix_modified() : "")).c_str()), bmp); | ||||||
|             if (is_selected) { |             if (is_selected) { | ||||||
|                 selected = wxString::FromUTF8((name + (preset.is_dirty ? Preset::suffix_modified() : "")).c_str()); |                 selected_user_preset = wxString::FromUTF8((name + (preset.is_dirty ? Preset::suffix_modified() : "")).c_str()); | ||||||
|                 tooltip = wxString::FromUTF8(preset.name.c_str()); |                 tooltip = wxString::FromUTF8(preset.name.c_str()); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  | @ -776,7 +776,7 @@ void PlaterPresetComboBox::update() | ||||||
|         set_label_marker(Append(separator(L("User presets")), wxNullBitmap)); |         set_label_marker(Append(separator(L("User presets")), wxNullBitmap)); | ||||||
|         for (std::map<wxString, wxBitmap*>::iterator it = nonsys_presets.begin(); it != nonsys_presets.end(); ++it) { |         for (std::map<wxString, wxBitmap*>::iterator it = nonsys_presets.begin(); it != nonsys_presets.end(); ++it) { | ||||||
|             Append(it->first, *it->second); |             Append(it->first, *it->second); | ||||||
|             validate_selection(it->first == selected); |             validate_selection(it->first == selected_user_preset); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -1167,6 +1167,12 @@ SavePresetDialog::SavePresetDialog(Preset::Type type, const std::string& suffix) | ||||||
|     : DPIDialog(nullptr, wxID_ANY, _L("Save preset"), wxDefaultPosition, wxSize(45 * wxGetApp().em_unit(), 5 * wxGetApp().em_unit()), wxDEFAULT_DIALOG_STYLE | wxICON_WARNING | wxRESIZE_BORDER) |     : DPIDialog(nullptr, wxID_ANY, _L("Save preset"), wxDefaultPosition, wxSize(45 * wxGetApp().em_unit(), 5 * wxGetApp().em_unit()), wxDEFAULT_DIALOG_STYLE | wxICON_WARNING | wxRESIZE_BORDER) | ||||||
| { | { | ||||||
|     SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW)); |     SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW)); | ||||||
|  | #if ENABLE_WX_3_1_3_DPI_CHANGED_EVENT | ||||||
|  |     // ys_FIXME! temporary workaround for correct font scaling
 | ||||||
|  |     // Because of from wxWidgets 3.1.3 auto rescaling is implemented for the Fonts,
 | ||||||
|  |     // From the very beginning set dialog font to the wxSYS_DEFAULT_GUI_FONT
 | ||||||
|  |     this->SetFont(wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT)); | ||||||
|  | #endif // ENABLE_WX_3_1_3_DPI_CHANGED_EVENT   
 | ||||||
| 
 | 
 | ||||||
|     wxBoxSizer* topSizer = new wxBoxSizer(wxVERTICAL); |     wxBoxSizer* topSizer = new wxBoxSizer(wxVERTICAL); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -3971,9 +3971,16 @@ void TabSLAPrint::build() | ||||||
|     optgroup->append_single_option_line("support_base_safety_distance"); |     optgroup->append_single_option_line("support_base_safety_distance"); | ||||||
|      |      | ||||||
|     // Mirrored parameter from Pad page for toggling elevation on the same page
 |     // Mirrored parameter from Pad page for toggling elevation on the same page
 | ||||||
|     optgroup->append_single_option_line("pad_around_object"); | //    optgroup->append_single_option_line("pad_around_object");
 | ||||||
|     optgroup->append_single_option_line("support_object_elevation"); |     optgroup->append_single_option_line("support_object_elevation"); | ||||||
| 
 | 
 | ||||||
|  |     Line line{ "", "" }; | ||||||
|  |     line.full_width = 1; | ||||||
|  |     line.widget = [this](wxWindow* parent) { | ||||||
|  |         return description_line_widget(parent, &m_support_object_elevation_description_line); | ||||||
|  |     }; | ||||||
|  |     optgroup->append_line(line); | ||||||
|  | 
 | ||||||
|     optgroup = page->new_optgroup(L("Connection of the support sticks and junctions")); |     optgroup = page->new_optgroup(L("Connection of the support sticks and junctions")); | ||||||
|     optgroup->append_single_option_line("support_critical_angle"); |     optgroup->append_single_option_line("support_critical_angle"); | ||||||
|     optgroup->append_single_option_line("support_max_bridge_length"); |     optgroup->append_single_option_line("support_max_bridge_length"); | ||||||
|  | @ -4047,6 +4054,14 @@ void TabSLAPrint::update() | ||||||
|     m_update_cnt++; |     m_update_cnt++; | ||||||
| 
 | 
 | ||||||
|     m_config_manipulation.update_print_sla_config(m_config, true); |     m_config_manipulation.update_print_sla_config(m_config, true); | ||||||
|  | 
 | ||||||
|  |     bool elev = !m_config->opt_bool("pad_enable") || !m_config->opt_bool("pad_around_object"); | ||||||
|  |     m_support_object_elevation_description_line->SetText(elev ? "" : | ||||||
|  |         from_u8((boost::format(_u8L("\"%1%\" is disabled because \"%2%\" is on in \"%3%\" category.\n" | ||||||
|  |                                     "To enable \"%1%\", please switch off \"%2%\""))  | ||||||
|  |                  % _L("Object elevation") % _L("Pad around object") % _L("Pad")).str())); | ||||||
|  |     Layout(); | ||||||
|  | 
 | ||||||
|     m_update_cnt--; |     m_update_cnt--; | ||||||
| 
 | 
 | ||||||
|     if (m_update_cnt == 0) { |     if (m_update_cnt == 0) { | ||||||
|  |  | ||||||
|  | @ -460,6 +460,9 @@ public: | ||||||
| //         Tab(parent, _(L("Print Settings")), L("sla_print")) {}
 | //         Tab(parent, _(L("Print Settings")), L("sla_print")) {}
 | ||||||
|         Tab(parent, _(L("Print Settings")), Slic3r::Preset::TYPE_SLA_PRINT) {} |         Tab(parent, _(L("Print Settings")), Slic3r::Preset::TYPE_SLA_PRINT) {} | ||||||
|     ~TabSLAPrint() {} |     ~TabSLAPrint() {} | ||||||
|  | 
 | ||||||
|  | 	ogStaticText* m_support_object_elevation_description_line = nullptr; | ||||||
|  | 
 | ||||||
|     void		build() override; |     void		build() override; | ||||||
| 	void		reload_config() override; | 	void		reload_config() override; | ||||||
|     void		update() override; |     void		update() override; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 enricoturri1966
						enricoturri1966