mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-24 09:11:23 -06:00 
			
		
		
		
	Merge branch 'master' into fs_realtime_simplify
This commit is contained in:
		
						commit
						d101d031dc
					
				
					 18 changed files with 209 additions and 89 deletions
				
			
		|  | @ -50,7 +50,7 @@ static ExPolygons get_print_object_bottom_layer_expolygons(const PrintObject &pr | ||||||
| { | { | ||||||
|     ExPolygons ex_polygons; |     ExPolygons ex_polygons; | ||||||
|     for (LayerRegion *region : print_object.layers().front()->regions()) |     for (LayerRegion *region : print_object.layers().front()->regions()) | ||||||
|         Slic3r::append(ex_polygons, offset_ex(offset_ex(region->slices.surfaces, float(SCALED_EPSILON)), -float(SCALED_EPSILON))); |         Slic3r::append(ex_polygons, closing_ex(region->slices.surfaces, float(SCALED_EPSILON))); | ||||||
|     return ex_polygons; |     return ex_polygons; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -452,6 +452,11 @@ ExPolygons offset2_ex(const ExPolygons &expolygons, const float delta1, const fl | ||||||
| { | { | ||||||
|     return PolyTreeToExPolygons(offset_paths<ClipperLib::PolyTree>(expolygons_offset(expolygons, delta1, joinType, miterLimit), delta2, joinType, miterLimit)); |     return PolyTreeToExPolygons(offset_paths<ClipperLib::PolyTree>(expolygons_offset(expolygons, delta1, joinType, miterLimit), delta2, joinType, miterLimit)); | ||||||
| } | } | ||||||
|  | ExPolygons offset2_ex(const Surfaces &surfaces, const float delta1, const float delta2, ClipperLib::JoinType joinType, double miterLimit) | ||||||
|  | { | ||||||
|  |     //FIXME it may be more efficient to offset to_expolygons(surfaces) instead of to_polygons(surfaces).
 | ||||||
|  |     return PolyTreeToExPolygons(offset_paths<ClipperLib::PolyTree>(expolygons_offset(surfaces, delta1, joinType, miterLimit), delta2, joinType, miterLimit)); | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| // Offset outside, then inside produces morphological closing. All deltas should be positive.
 | // Offset outside, then inside produces morphological closing. All deltas should be positive.
 | ||||||
| Slic3r::Polygons closing(const Slic3r::Polygons &polygons, const float delta1, const float delta2, ClipperLib::JoinType joinType, double miterLimit) | Slic3r::Polygons closing(const Slic3r::Polygons &polygons, const float delta1, const float delta2, ClipperLib::JoinType joinType, double miterLimit) | ||||||
|  | @ -466,6 +471,13 @@ Slic3r::ExPolygons closing_ex(const Slic3r::Polygons &polygons, const float delt | ||||||
|     assert(delta2 > 0); |     assert(delta2 > 0); | ||||||
|     return PolyTreeToExPolygons(shrink_paths<ClipperLib::PolyTree>(expand_paths<ClipperLib::Paths>(ClipperUtils::PolygonsProvider(polygons), delta1, joinType, miterLimit), delta2, joinType, miterLimit)); |     return PolyTreeToExPolygons(shrink_paths<ClipperLib::PolyTree>(expand_paths<ClipperLib::Paths>(ClipperUtils::PolygonsProvider(polygons), delta1, joinType, miterLimit), delta2, joinType, miterLimit)); | ||||||
| } | } | ||||||
|  | Slic3r::ExPolygons closing_ex(const Slic3r::Surfaces &surfaces, const float delta1, const float delta2, ClipperLib::JoinType joinType, double miterLimit) | ||||||
|  | { | ||||||
|  |     assert(delta1 > 0); | ||||||
|  |     assert(delta2 > 0); | ||||||
|  |     //FIXME it may be more efficient to offset to_expolygons(surfaces) instead of to_polygons(surfaces).
 | ||||||
|  |     return PolyTreeToExPolygons(shrink_paths<ClipperLib::PolyTree>(expand_paths<ClipperLib::Paths>(ClipperUtils::SurfacesProvider(surfaces), delta1, joinType, miterLimit), delta2, joinType, miterLimit)); | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| // Offset inside, then outside produces morphological opening. All deltas should be positive.
 | // Offset inside, then outside produces morphological opening. All deltas should be positive.
 | ||||||
| Slic3r::Polygons opening(const Slic3r::Polygons &polygons, const float delta1, const float delta2, ClipperLib::JoinType joinType, double miterLimit) | Slic3r::Polygons opening(const Slic3r::Polygons &polygons, const float delta1, const float delta2, ClipperLib::JoinType joinType, double miterLimit) | ||||||
|  | @ -474,6 +486,19 @@ Slic3r::Polygons opening(const Slic3r::Polygons &polygons, const float delta1, c | ||||||
|     assert(delta2 > 0); |     assert(delta2 > 0); | ||||||
|     return to_polygons(expand_paths<ClipperLib::Paths>(shrink_paths<ClipperLib::Paths>(ClipperUtils::PolygonsProvider(polygons), delta1, joinType, miterLimit), delta2, joinType, miterLimit)); |     return to_polygons(expand_paths<ClipperLib::Paths>(shrink_paths<ClipperLib::Paths>(ClipperUtils::PolygonsProvider(polygons), delta1, joinType, miterLimit), delta2, joinType, miterLimit)); | ||||||
| } | } | ||||||
|  | Slic3r::Polygons opening(const Slic3r::ExPolygons &expolygons, const float delta1, const float delta2, ClipperLib::JoinType joinType, double miterLimit) | ||||||
|  | { | ||||||
|  |     assert(delta1 > 0); | ||||||
|  |     assert(delta2 > 0); | ||||||
|  |     return to_polygons(expand_paths<ClipperLib::Paths>(shrink_paths<ClipperLib::Paths>(ClipperUtils::ExPolygonsProvider(expolygons), delta1, joinType, miterLimit), delta2, joinType, miterLimit)); | ||||||
|  | } | ||||||
|  | Slic3r::Polygons opening(const Slic3r::Surfaces &surfaces, const float delta1, const float delta2, ClipperLib::JoinType joinType, double miterLimit) | ||||||
|  | { | ||||||
|  |     assert(delta1 > 0); | ||||||
|  |     assert(delta2 > 0); | ||||||
|  |     //FIXME it may be more efficient to offset to_expolygons(surfaces) instead of to_polygons(surfaces).
 | ||||||
|  |     return to_polygons(expand_paths<ClipperLib::Paths>(shrink_paths<ClipperLib::Paths>(ClipperUtils::SurfacesProvider(surfaces), delta1, joinType, miterLimit), delta2, joinType, miterLimit)); | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| // Fix of #117: A large fractal pyramid takes ages to slice
 | // Fix of #117: A large fractal pyramid takes ages to slice
 | ||||||
| // The Clipper library has difficulties processing overlapping polygons.
 | // The Clipper library has difficulties processing overlapping polygons.
 | ||||||
|  | @ -525,6 +550,8 @@ Slic3r::Polygons diff(const Slic3r::ExPolygons &subject, const Slic3r::Polygons | ||||||
|     { return _clipper(ClipperLib::ctDifference, ClipperUtils::ExPolygonsProvider(subject), ClipperUtils::PolygonsProvider(clip), do_safety_offset); } |     { return _clipper(ClipperLib::ctDifference, ClipperUtils::ExPolygonsProvider(subject), ClipperUtils::PolygonsProvider(clip), do_safety_offset); } | ||||||
| Slic3r::Polygons diff(const Slic3r::ExPolygons &subject, const Slic3r::ExPolygons &clip, ApplySafetyOffset do_safety_offset) | Slic3r::Polygons diff(const Slic3r::ExPolygons &subject, const Slic3r::ExPolygons &clip, ApplySafetyOffset do_safety_offset) | ||||||
|     { return _clipper(ClipperLib::ctDifference, ClipperUtils::ExPolygonsProvider(subject), ClipperUtils::ExPolygonsProvider(clip), do_safety_offset); } |     { return _clipper(ClipperLib::ctDifference, ClipperUtils::ExPolygonsProvider(subject), ClipperUtils::ExPolygonsProvider(clip), do_safety_offset); } | ||||||
|  | Slic3r::Polygons diff(const Slic3r::Surfaces &subject, const Slic3r::Polygons &clip, ApplySafetyOffset do_safety_offset) | ||||||
|  |     { return _clipper(ClipperLib::ctDifference, ClipperUtils::SurfacesProvider(subject), ClipperUtils::PolygonsProvider(clip), do_safety_offset); } | ||||||
| Slic3r::Polygons intersection(const Slic3r::Polygon &subject, const Slic3r::Polygon &clip, ApplySafetyOffset do_safety_offset) | Slic3r::Polygons intersection(const Slic3r::Polygon &subject, const Slic3r::Polygon &clip, ApplySafetyOffset do_safety_offset) | ||||||
|     { return _clipper(ClipperLib::ctIntersection, ClipperUtils::SinglePathProvider(subject.points), ClipperUtils::SinglePathProvider(clip.points), do_safety_offset); } |     { return _clipper(ClipperLib::ctIntersection, ClipperUtils::SinglePathProvider(subject.points), ClipperUtils::SinglePathProvider(clip.points), do_safety_offset); } | ||||||
| Slic3r::Polygons intersection(const Slic3r::Polygons &subject, const Slic3r::Polygons &clip, ApplySafetyOffset do_safety_offset) | Slic3r::Polygons intersection(const Slic3r::Polygons &subject, const Slic3r::Polygons &clip, ApplySafetyOffset do_safety_offset) | ||||||
|  |  | ||||||
|  | @ -345,19 +345,35 @@ inline Slic3r::ExPolygons shrink_ex(const Slic3r::Polygons &polygons, const floa | ||||||
| // Input polygons for negative offset shall be "normalized": There must be no overlap / intersections between the input polygons.
 | // Input polygons for negative offset shall be "normalized": There must be no overlap / intersections between the input polygons.
 | ||||||
| Slic3r::Polygons   offset2(const Slic3r::ExPolygons &expolygons, const float delta1, const float delta2, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | Slic3r::Polygons   offset2(const Slic3r::ExPolygons &expolygons, const float delta1, const float delta2, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | ||||||
| Slic3r::ExPolygons offset2_ex(const Slic3r::ExPolygons &expolygons, const float delta1, const float delta2, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | Slic3r::ExPolygons offset2_ex(const Slic3r::ExPolygons &expolygons, const float delta1, const float delta2, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | ||||||
|  | Slic3r::ExPolygons offset2_ex(const Slic3r::Surfaces &surfaces, const float delta1, const float delta2, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | ||||||
| 
 | 
 | ||||||
| // Offset outside, then inside produces morphological closing. All deltas should be positive.
 | // Offset outside, then inside produces morphological closing. All deltas should be positive.
 | ||||||
| Slic3r::Polygons          closing(const Slic3r::Polygons &polygons, const float delta1, const float delta2, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | Slic3r::Polygons          closing(const Slic3r::Polygons &polygons, const float delta1, const float delta2, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | ||||||
| inline Slic3r::Polygons   closing(const Slic3r::Polygons &polygons, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit) { return closing(polygons, delta, delta, joinType, miterLimit); } | inline Slic3r::Polygons   closing(const Slic3r::Polygons &polygons, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit)  | ||||||
|  |     { return closing(polygons, delta, delta, joinType, miterLimit); } | ||||||
| Slic3r::ExPolygons        closing_ex(const Slic3r::Polygons &polygons, const float delta1, const float delta2, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | Slic3r::ExPolygons        closing_ex(const Slic3r::Polygons &polygons, const float delta1, const float delta2, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | ||||||
| inline Slic3r::ExPolygons closing_ex(const Slic3r::Polygons &polygons, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit) { return closing_ex(polygons, delta, delta, joinType, miterLimit); } | inline Slic3r::ExPolygons closing_ex(const Slic3r::Polygons &polygons, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit)  | ||||||
| inline Slic3r::ExPolygons closing_ex(const Slic3r::ExPolygons &polygons, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit) { return offset2_ex(polygons, delta, - delta, joinType, miterLimit); } |     { return closing_ex(polygons, delta, delta, joinType, miterLimit); } | ||||||
|  | inline Slic3r::ExPolygons closing_ex(const Slic3r::ExPolygons &polygons, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit)  | ||||||
|  |     { assert(delta > 0); return offset2_ex(polygons, delta, - delta, joinType, miterLimit); } | ||||||
|  | inline Slic3r::ExPolygons closing_ex(const Slic3r::Surfaces &surfaces, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit)  | ||||||
|  |     { assert(delta > 0); return offset2_ex(surfaces, delta, - delta, joinType, miterLimit); } | ||||||
| 
 | 
 | ||||||
| // Offset inside, then outside produces morphological opening. All deltas should be positive.
 | // Offset inside, then outside produces morphological opening. All deltas should be positive.
 | ||||||
| // Input polygons for opening shall be "normalized": There must be no overlap / intersections between the input polygons.
 | // Input polygons for opening shall be "normalized": There must be no overlap / intersections between the input polygons.
 | ||||||
| Slic3r::Polygons          opening(const Slic3r::Polygons &polygons, const float delta1, const float delta2, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | Slic3r::Polygons          opening(const Slic3r::Polygons &polygons, const float delta1, const float delta2, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | ||||||
| inline Slic3r::Polygons   opening(const Slic3r::Polygons &polygons, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit) { return opening(polygons, delta, delta, joinType, miterLimit); } | Slic3r::Polygons          opening(const Slic3r::ExPolygons &expolygons, const float delta1, const float delta2, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | ||||||
| inline Slic3r::ExPolygons opening_ex(const Slic3r::ExPolygons &polygons, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit) { return offset2_ex(polygons, - delta, delta, joinType, miterLimit); } | Slic3r::Polygons          opening(const Slic3r::Surfaces &surfaces, const float delta1, const float delta2, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit); | ||||||
|  | inline Slic3r::Polygons   opening(const Slic3r::Polygons &polygons, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit)  | ||||||
|  |     { return opening(polygons, delta, delta, joinType, miterLimit); } | ||||||
|  | inline Slic3r::Polygons   opening(const Slic3r::ExPolygons &expolygons, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit)  | ||||||
|  |     { return opening(expolygons, delta, delta, joinType, miterLimit); } | ||||||
|  | inline Slic3r::Polygons   opening(const Slic3r::Surfaces &surfaces, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit)  | ||||||
|  |     { return opening(surfaces, delta, delta, joinType, miterLimit); } | ||||||
|  | inline Slic3r::ExPolygons opening_ex(const Slic3r::ExPolygons &polygons, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit)  | ||||||
|  |     { assert(delta > 0); return offset2_ex(polygons, - delta, delta, joinType, miterLimit); } | ||||||
|  | inline Slic3r::ExPolygons opening_ex(const Slic3r::Surfaces &surfaces, const float delta, ClipperLib::JoinType joinType = DefaultJoinType, double miterLimit = DefaultMiterLimit)  | ||||||
|  |     { assert(delta > 0); return offset2_ex(surfaces, - delta, delta, joinType, miterLimit); } | ||||||
| 
 | 
 | ||||||
| Slic3r::Lines _clipper_ln(ClipperLib::ClipType clipType, const Slic3r::Lines &subject, const Slic3r::Polygons &clip); | Slic3r::Lines _clipper_ln(ClipperLib::ClipType clipType, const Slic3r::Lines &subject, const Slic3r::Polygons &clip); | ||||||
| 
 | 
 | ||||||
|  | @ -366,6 +382,7 @@ Slic3r::Polygons   diff(const Slic3r::Polygons &subject, const Slic3r::Polygons | ||||||
| Slic3r::Polygons   diff(const Slic3r::Polygons &subject, const Slic3r::ExPolygons &clip, ApplySafetyOffset do_safety_offset = ApplySafetyOffset::No); | Slic3r::Polygons   diff(const Slic3r::Polygons &subject, const Slic3r::ExPolygons &clip, ApplySafetyOffset do_safety_offset = ApplySafetyOffset::No); | ||||||
| Slic3r::Polygons   diff(const Slic3r::ExPolygons &subject, const Slic3r::Polygons &clip, ApplySafetyOffset do_safety_offset = ApplySafetyOffset::No); | Slic3r::Polygons   diff(const Slic3r::ExPolygons &subject, const Slic3r::Polygons &clip, ApplySafetyOffset do_safety_offset = ApplySafetyOffset::No); | ||||||
| Slic3r::Polygons   diff(const Slic3r::ExPolygons &subject, const Slic3r::ExPolygons &clip, ApplySafetyOffset do_safety_offset = ApplySafetyOffset::No); | Slic3r::Polygons   diff(const Slic3r::ExPolygons &subject, const Slic3r::ExPolygons &clip, ApplySafetyOffset do_safety_offset = ApplySafetyOffset::No); | ||||||
|  | Slic3r::Polygons   diff(const Slic3r::Surfaces &subject, const Slic3r::Polygons &clip, ApplySafetyOffset do_safety_offset = ApplySafetyOffset::No); | ||||||
| Slic3r::ExPolygons diff_ex(const Slic3r::Polygons &subject, const Slic3r::Polygons &clip, ApplySafetyOffset do_safety_offset = ApplySafetyOffset::No); | Slic3r::ExPolygons diff_ex(const Slic3r::Polygons &subject, const Slic3r::Polygons &clip, ApplySafetyOffset do_safety_offset = ApplySafetyOffset::No); | ||||||
| Slic3r::ExPolygons diff_ex(const Slic3r::Polygons &subject, const Slic3r::ExPolygons &clip, ApplySafetyOffset do_safety_offset = ApplySafetyOffset::No); | Slic3r::ExPolygons diff_ex(const Slic3r::Polygons &subject, const Slic3r::ExPolygons &clip, ApplySafetyOffset do_safety_offset = ApplySafetyOffset::No); | ||||||
| Slic3r::ExPolygons diff_ex(const Slic3r::Polygons &subject, const Slic3r::Surfaces &clip, ApplySafetyOffset do_safety_offset = ApplySafetyOffset::No); | Slic3r::ExPolygons diff_ex(const Slic3r::Polygons &subject, const Slic3r::Surfaces &clip, ApplySafetyOffset do_safety_offset = ApplySafetyOffset::No); | ||||||
|  |  | ||||||
|  | @ -35,6 +35,7 @@ void CoolingBuffer::reset(const Vec3d &position) | ||||||
|     m_current_pos[1] = float(position.y()); |     m_current_pos[1] = float(position.y()); | ||||||
|     m_current_pos[2] = float(position.z()); |     m_current_pos[2] = float(position.z()); | ||||||
|     m_current_pos[4] = float(m_config.travel_speed.value); |     m_current_pos[4] = float(m_config.travel_speed.value); | ||||||
|  |     m_fan_speed = -1; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| struct CoolingLine | struct CoolingLine | ||||||
|  | @ -689,10 +690,9 @@ std::string CoolingBuffer::apply_layer_cooldown( | ||||||
|     // Second generate the adjusted G-code.
 |     // Second generate the adjusted G-code.
 | ||||||
|     std::string new_gcode; |     std::string new_gcode; | ||||||
|     new_gcode.reserve(gcode.size() * 2); |     new_gcode.reserve(gcode.size() * 2); | ||||||
|     int  fan_speed          = -1; |  | ||||||
|     bool bridge_fan_control = false; |     bool bridge_fan_control = false; | ||||||
|     int  bridge_fan_speed   = 0; |     int  bridge_fan_speed   = 0; | ||||||
|     auto change_extruder_set_fan = [ this, layer_id, layer_time, &new_gcode, &fan_speed, &bridge_fan_control, &bridge_fan_speed ]() { |     auto change_extruder_set_fan = [ this, layer_id, layer_time, &new_gcode, &bridge_fan_control, &bridge_fan_speed ]() { | ||||||
| #define EXTRUDER_CONFIG(OPT) m_config.OPT.get_at(m_current_extruder) | #define EXTRUDER_CONFIG(OPT) m_config.OPT.get_at(m_current_extruder) | ||||||
|         int min_fan_speed = EXTRUDER_CONFIG(min_fan_speed); |         int min_fan_speed = EXTRUDER_CONFIG(min_fan_speed); | ||||||
|         int fan_speed_new = EXTRUDER_CONFIG(fan_always_on) ? min_fan_speed : 0; |         int fan_speed_new = EXTRUDER_CONFIG(fan_always_on) ? min_fan_speed : 0; | ||||||
|  | @ -733,9 +733,9 @@ std::string CoolingBuffer::apply_layer_cooldown( | ||||||
|             bridge_fan_speed   = 0; |             bridge_fan_speed   = 0; | ||||||
|             fan_speed_new      = 0; |             fan_speed_new      = 0; | ||||||
|         } |         } | ||||||
|         if (fan_speed_new != fan_speed) { |         if (fan_speed_new != m_fan_speed) { | ||||||
|             fan_speed = fan_speed_new; |             m_fan_speed = fan_speed_new; | ||||||
|             new_gcode += GCodeWriter::set_fan(m_config.gcode_flavor, m_config.gcode_comments, fan_speed); |             new_gcode  += GCodeWriter::set_fan(m_config.gcode_flavor, m_config.gcode_comments, m_fan_speed); | ||||||
|         } |         } | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|  | @ -759,7 +759,7 @@ std::string CoolingBuffer::apply_layer_cooldown( | ||||||
|                 new_gcode += GCodeWriter::set_fan(m_config.gcode_flavor, m_config.gcode_comments, bridge_fan_speed); |                 new_gcode += GCodeWriter::set_fan(m_config.gcode_flavor, m_config.gcode_comments, bridge_fan_speed); | ||||||
|         } else if (line->type & CoolingLine::TYPE_BRIDGE_FAN_END) { |         } else if (line->type & CoolingLine::TYPE_BRIDGE_FAN_END) { | ||||||
|             if (bridge_fan_control) |             if (bridge_fan_control) | ||||||
|                 new_gcode += GCodeWriter::set_fan(m_config.gcode_flavor, m_config.gcode_comments, fan_speed); |                 new_gcode += GCodeWriter::set_fan(m_config.gcode_flavor, m_config.gcode_comments, m_fan_speed); | ||||||
|         } else if (line->type & CoolingLine::TYPE_EXTRUDE_END) { |         } else if (line->type & CoolingLine::TYPE_EXTRUDE_END) { | ||||||
|             // Just remove this comment.
 |             // Just remove this comment.
 | ||||||
|         } else if (line->type & (CoolingLine::TYPE_ADJUSTABLE | CoolingLine::TYPE_EXTERNAL_PERIMETER | CoolingLine::TYPE_WIPE | CoolingLine::TYPE_HAS_F)) { |         } else if (line->type & (CoolingLine::TYPE_ADJUSTABLE | CoolingLine::TYPE_EXTERNAL_PERIMETER | CoolingLine::TYPE_WIPE | CoolingLine::TYPE_HAS_F)) { | ||||||
|  |  | ||||||
|  | @ -41,6 +41,8 @@ private: | ||||||
|     // X,Y,Z,E,F
 |     // X,Y,Z,E,F
 | ||||||
|     std::vector<char>           m_axis; |     std::vector<char>           m_axis; | ||||||
|     std::vector<float>          m_current_pos; |     std::vector<float>          m_current_pos; | ||||||
|  |     // Current known fan speed or -1 if not known yet.
 | ||||||
|  |     int                         m_fan_speed; | ||||||
|     // Cached from GCodeWriter.
 |     // Cached from GCodeWriter.
 | ||||||
|     // Printing extruder IDs, zero based.
 |     // Printing extruder IDs, zero based.
 | ||||||
|     std::vector<unsigned int>   m_extruder_ids; |     std::vector<unsigned int>   m_extruder_ids; | ||||||
|  |  | ||||||
|  | @ -152,7 +152,7 @@ bool GCodeReader::parse_file_raw_internal(const std::string &filename, ParseLine | ||||||
|             auto it_end = it; |             auto it_end = it; | ||||||
|             for (; it_end != it_bufend && ! (eol = *it_end == '\r' || *it_end == '\n'); ++ it_end) |             for (; it_end != it_bufend && ! (eol = *it_end == '\r' || *it_end == '\n'); ++ it_end) | ||||||
|                 if (*it_end == '\n') |                 if (*it_end == '\n') | ||||||
|                     line_end_callback((it_end - buffer.begin()) + 1); |                     line_end_callback(file_pos + (it_end - buffer.begin()) + 1); | ||||||
|             // End of line is indicated also if end of file was reached.
 |             // End of line is indicated also if end of file was reached.
 | ||||||
|             eol |= eof && it_end == it_bufend; |             eol |= eof && it_end == it_bufend; | ||||||
|             if (eol) { |             if (eol) { | ||||||
|  | @ -173,7 +173,7 @@ bool GCodeReader::parse_file_raw_internal(const std::string &filename, ParseLine | ||||||
|             if (it != it_bufend && *it == '\r') |             if (it != it_bufend && *it == '\r') | ||||||
|                 ++ it; |                 ++ it; | ||||||
|             if (it != it_bufend && *it == '\n') { |             if (it != it_bufend && *it == '\n') { | ||||||
|                 line_end_callback((it - buffer.begin()) + 1); |                 line_end_callback(file_pos + (it - buffer.begin()) + 1); | ||||||
|                 ++ it; |                 ++ it; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  | @ -431,9 +431,8 @@ void LayerRegion::elephant_foot_compensation_step(const float elephant_foot_comp | ||||||
|     for (const Surface &surface : this->slices.surfaces) |     for (const Surface &surface : this->slices.surfaces) | ||||||
|         assert(surface.surface_type == stInternal); |         assert(surface.surface_type == stInternal); | ||||||
| #endif /* NDEBUG */ | #endif /* NDEBUG */ | ||||||
|     ExPolygons surfaces = to_expolygons(std::move(this->slices.surfaces)); |     Polygons tmp = intersection(this->slices.surfaces, trimming_polygons); | ||||||
|     Polygons tmp = intersection(surfaces, trimming_polygons); |     append(tmp, diff(this->slices.surfaces, opening(this->slices.surfaces, elephant_foot_compensation_perimeter_step))); | ||||||
|     append(tmp, diff(surfaces, offset(offset_ex(surfaces, -elephant_foot_compensation_perimeter_step), elephant_foot_compensation_perimeter_step))); |  | ||||||
|     this->slices.set(union_ex(tmp), stInternal); |     this->slices.set(union_ex(tmp), stInternal); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -159,8 +159,9 @@ template<class _Mesh> TriangleMesh cgal_to_triangle_mesh(const _Mesh &cgalmesh) | ||||||
|         int i = 0; |         int i = 0; | ||||||
|         Vec3i facet; |         Vec3i facet; | ||||||
|         for (auto v : vtc) { |         for (auto v : vtc) { | ||||||
|             if (i > 2 || v < 0 || v >= cgalmesh.vertices().size()) { i = 0; break; } |             int iv = v; | ||||||
|             facet(i++) = v; |             if (i > 2 || iv < 0 || iv >= int(cgalmesh.vertices().size())) { i = 0; break; } | ||||||
|  |             facet(i++) = iv; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if (i == 3) |         if (i == 3) | ||||||
|  |  | ||||||
|  | @ -1088,7 +1088,7 @@ void PrintObject::discover_vertical_shells() | ||||||
|                     // For a multi-material print, simulate perimeter / infill split as if only a single extruder has been used for the whole print.
 |                     // For a multi-material print, simulate perimeter / infill split as if only a single extruder has been used for the whole print.
 | ||||||
|                     if (perimeter_offset > 0.) { |                     if (perimeter_offset > 0.) { | ||||||
|                         // The layer.lslices are forced to merge by expanding them first.
 |                         // The layer.lslices are forced to merge by expanding them first.
 | ||||||
|                         polygons_append(cache.holes, offset(offset_ex(layer.lslices, 0.3f * perimeter_min_spacing), - perimeter_offset - 0.3f * perimeter_min_spacing)); |                         polygons_append(cache.holes, offset2(layer.lslices, 0.3f * perimeter_min_spacing, - perimeter_offset - 0.3f * perimeter_min_spacing)); | ||||||
| #ifdef SLIC3R_DEBUG_SLICE_PROCESSING | #ifdef SLIC3R_DEBUG_SLICE_PROCESSING | ||||||
|                         { |                         { | ||||||
|                             Slic3r::SVG svg(debug_out_path("discover_vertical_shells-extra-holes-%d.svg", debug_idx), get_extents(layer.lslices)); |                             Slic3r::SVG svg(debug_out_path("discover_vertical_shells-extra-holes-%d.svg", debug_idx), get_extents(layer.lslices)); | ||||||
|  | @ -1325,7 +1325,7 @@ void PrintObject::discover_vertical_shells() | ||||||
| #if 1 | #if 1 | ||||||
|                     // Intentionally inflate a bit more than how much the region has been shrunk, 
 |                     // Intentionally inflate a bit more than how much the region has been shrunk, 
 | ||||||
|                     // so there will be some overlap between this solid infill and the other infill regions (mainly the sparse infill).
 |                     // so there will be some overlap between this solid infill and the other infill regions (mainly the sparse infill).
 | ||||||
|                     shell = offset(offset_ex(union_ex(shell), - 0.5f * min_perimeter_infill_spacing), 0.8f * min_perimeter_infill_spacing, ClipperLib::jtSquare); |                     shell = opening(union_(shell), 0.5f * min_perimeter_infill_spacing, 0.8f * min_perimeter_infill_spacing, ClipperLib::jtSquare); | ||||||
|                     if (shell.empty()) |                     if (shell.empty()) | ||||||
|                         continue; |                         continue; | ||||||
| #else | #else | ||||||
|  |  | ||||||
|  | @ -208,7 +208,7 @@ void name_tbb_thread_pool_threads_set_locale() | ||||||
| 	nthreads = 1; | 	nthreads = 1; | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| 	std::atomic<size_t>		nthreads_running(0); | 	size_t                  nthreads_running(0); | ||||||
| 	std::condition_variable cv; | 	std::condition_variable cv; | ||||||
| 	std::mutex				cv_m; | 	std::mutex				cv_m; | ||||||
| 	auto					master_thread_id = std::this_thread::get_id(); | 	auto					master_thread_id = std::this_thread::get_id(); | ||||||
|  | @ -216,13 +216,13 @@ void name_tbb_thread_pool_threads_set_locale() | ||||||
|         tbb::blocked_range<size_t>(0, nthreads, 1), |         tbb::blocked_range<size_t>(0, nthreads, 1), | ||||||
|         [&nthreads_running, nthreads, &master_thread_id, &cv, &cv_m](const tbb::blocked_range<size_t> &range) { |         [&nthreads_running, nthreads, &master_thread_id, &cv, &cv_m](const tbb::blocked_range<size_t> &range) { | ||||||
|         	assert(range.begin() + 1 == range.end()); |         	assert(range.begin() + 1 == range.end()); | ||||||
|         	if (nthreads_running.fetch_add(1) + 1 == nthreads) { | 			if (std::unique_lock<std::mutex> lk(cv_m);  ++nthreads_running == nthreads) { | ||||||
|  | 				lk.unlock(); | ||||||
|         		// All threads are spinning.
 |         		// All threads are spinning.
 | ||||||
|         		// Wake them up.
 |         		// Wake them up.
 | ||||||
|     			cv.notify_all(); |     			cv.notify_all(); | ||||||
|         	} else { |         	} else { | ||||||
|         		// Wait for the last thread to wake the others.
 |         		// Wait for the last thread to wake the others.
 | ||||||
| 				std::unique_lock<std::mutex> lk(cv_m); |  | ||||||
| 			    cv.wait(lk, [&nthreads_running, nthreads]{return nthreads_running == nthreads;}); | 			    cv.wait(lk, [&nthreads_running, nthreads]{return nthreads_running == nthreads;}); | ||||||
|         	} |         	} | ||||||
|         	auto thread_id = std::this_thread::get_id(); |         	auto thread_id = std::this_thread::get_id(); | ||||||
|  |  | ||||||
|  | @ -676,16 +676,18 @@ void GUI_App::post_init() | ||||||
|     if (this->preset_updater) { |     if (this->preset_updater) { | ||||||
|         this->check_updates(false); |         this->check_updates(false); | ||||||
|         CallAfter([this] { |         CallAfter([this] { | ||||||
|             this->config_wizard_startup(); |             bool cw_showed = this->config_wizard_startup(); | ||||||
|             this->preset_updater->slic3r_update_notify(); |             this->preset_updater->slic3r_update_notify(); | ||||||
|             this->preset_updater->sync(preset_bundle); |             this->preset_updater->sync(preset_bundle); | ||||||
|  |             if (! cw_showed) { | ||||||
|  |                 // The CallAfter is needed as well, without it, GL extensions did not show.
 | ||||||
|  |                 // Also, we only want to show this when the wizard does not, so the new user
 | ||||||
|  |                 // sees something else than "we want something" on the first start.
 | ||||||
|  |                 show_send_system_info_dialog_if_needed(); | ||||||
|  |             } | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // 'Send system info' dialog. Again, a CallAfter is needed on mac.
 |  | ||||||
|     // Without it, GL extensions did not show.
 |  | ||||||
|     CallAfter([] { show_send_system_info_dialog_if_needed(); }); |  | ||||||
| 
 |  | ||||||
| #ifdef _WIN32 | #ifdef _WIN32 | ||||||
|     // Sets window property to mainframe so other instances can indentify it.
 |     // Sets window property to mainframe so other instances can indentify it.
 | ||||||
|     OtherInstanceMessageHandler::init_windows_properties(mainframe, m_instance_hash_int); |     OtherInstanceMessageHandler::init_windows_properties(mainframe, m_instance_hash_int); | ||||||
|  |  | ||||||
|  | @ -215,7 +215,8 @@ void NotificationManager::PopNotification::render(GLCanvas3D& canvas, float init | ||||||
| 	} | 	} | ||||||
| 	 | 	 | ||||||
| 	if (mouse_pos.x < win_pos.x && mouse_pos.x > win_pos.x - m_window_width && mouse_pos.y > win_pos.y && mouse_pos.y < win_pos.y + m_window_height) { | 	if (mouse_pos.x < win_pos.x && mouse_pos.x > win_pos.x - m_window_width && mouse_pos.y > win_pos.y && mouse_pos.y < win_pos.y + m_window_height) { | ||||||
| 		ImGui::SetNextWindowFocus(); | 		// Uncomment if imgui window focus is needed on hover. I cant find any case.
 | ||||||
|  | 		//ImGui::SetNextWindowFocus();
 | ||||||
| 		set_hovered(); | 		set_hovered(); | ||||||
| 	} | 	} | ||||||
| 	 | 	 | ||||||
|  |  | ||||||
|  | @ -452,8 +452,10 @@ void PreferencesDialog::build(size_t selected_tab) | ||||||
| 
 | 
 | ||||||
| 	activate_options_tab(m_optgroup_gui); | 	activate_options_tab(m_optgroup_gui); | ||||||
| 	// set Field for notify_release to its value to activate the object
 | 	// set Field for notify_release to its value to activate the object
 | ||||||
| 	boost::any val = s_keys_map_NotifyReleaseMode.at(app_config->get("notify_release")); | 	if (is_editor) { | ||||||
| 	m_optgroup_gui->get_field("notify_release")->set_value(val, false); | 		boost::any val = s_keys_map_NotifyReleaseMode.at(app_config->get("notify_release")); | ||||||
|  | 		m_optgroup_gui->get_field("notify_release")->set_value(val, false); | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	if (is_editor) { | 	if (is_editor) { | ||||||
| 		create_icon_size_slider(); | 		create_icon_size_slider(); | ||||||
|  |  | ||||||
|  | @ -6,6 +6,7 @@ | ||||||
| 
 | 
 | ||||||
| #include "slic3r/GUI/format.hpp" | #include "slic3r/GUI/format.hpp" | ||||||
| #include "slic3r/Utils/Http.hpp" | #include "slic3r/Utils/Http.hpp" | ||||||
|  | #include "slic3r/Utils/PresetUpdater.hpp" | ||||||
| 
 | 
 | ||||||
| #include "GUI_App.hpp" | #include "GUI_App.hpp" | ||||||
| #include "GUI_Utils.hpp" | #include "GUI_Utils.hpp" | ||||||
|  | @ -17,6 +18,7 @@ | ||||||
| #include <boost/algorithm/hex.hpp> | #include <boost/algorithm/hex.hpp> | ||||||
| #include <boost/algorithm/string/split.hpp> | #include <boost/algorithm/string/split.hpp> | ||||||
| #include <boost/algorithm/string/trim_all.hpp> | #include <boost/algorithm/string/trim_all.hpp> | ||||||
|  | #include <boost/log/trivial.hpp> | ||||||
| #include <boost/property_tree/json_parser.hpp> | #include <boost/property_tree/json_parser.hpp> | ||||||
| #include <boost/uuid/detail/md5.hpp> | #include <boost/uuid/detail/md5.hpp> | ||||||
| 
 | 
 | ||||||
|  | @ -36,12 +38,17 @@ | ||||||
|     #include <Iphlpapi.h> |     #include <Iphlpapi.h> | ||||||
|     #pragma comment(lib, "iphlpapi.lib") |     #pragma comment(lib, "iphlpapi.lib") | ||||||
| #elif __APPLE__ | #elif __APPLE__ | ||||||
| #import <IOKit/IOKitLib.h> |     #import <IOKit/IOKitLib.h> | ||||||
|  |     #include <CoreFoundation/CoreFoundation.h> | ||||||
|  | #else // Linux/BSD
 | ||||||
|  |     #include <charconv> | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| namespace Slic3r { | namespace Slic3r { | ||||||
| namespace GUI { | namespace GUI { | ||||||
| 
 | 
 | ||||||
|  | static const std::string SEND_SYSTEM_INFO_DOMAIN = "prusa3d.com"; | ||||||
|  | static const std::string SEND_SYSTEM_INFO_URL = "https://files." + SEND_SYSTEM_INFO_DOMAIN + "/wp-json/v1/ps"; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| // Declaration of a free function defined in OpenGLManager.cpp:
 | // Declaration of a free function defined in OpenGLManager.cpp:
 | ||||||
|  | @ -52,8 +59,8 @@ std::string gl_get_string_safe(GLenum param, const std::string& default_value); | ||||||
| class SendSystemInfoDialog : public DPIDialog | class SendSystemInfoDialog : public DPIDialog | ||||||
| { | { | ||||||
|     enum { |     enum { | ||||||
|         MIN_WIDTH = 80, |         MIN_WIDTH = 70, | ||||||
|         MIN_HEIGHT = 50 |         MIN_HEIGHT = 34 | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
| public: | public: | ||||||
|  | @ -129,20 +136,36 @@ public: | ||||||
| // current version is newer. Only major and minor versions are compared.
 | // current version is newer. Only major and minor versions are compared.
 | ||||||
| static bool should_dialog_be_shown() | static bool should_dialog_be_shown() | ||||||
| { | { | ||||||
|     return false; |  | ||||||
| 
 |  | ||||||
|     std::string last_sent_version = wxGetApp().app_config->get("version_system_info_sent"); |     std::string last_sent_version = wxGetApp().app_config->get("version_system_info_sent"); | ||||||
|     Semver semver_current(SLIC3R_VERSION); |     Semver semver_current(SLIC3R_VERSION); | ||||||
|     Semver semver_last_sent; |     Semver semver_last_sent; | ||||||
|     if (! last_sent_version.empty()) |     if (! last_sent_version.empty()) | ||||||
|         semver_last_sent = Semver(last_sent_version); |         semver_last_sent = Semver(last_sent_version); | ||||||
| 
 | 
 | ||||||
|     if (semver_current.prerelease() && std::string(semver_current.prerelease()) == "alpha") |     // set whether to show in alpha builds, or only betas/rcs/finals:
 | ||||||
|         return false; // Don't show in alphas.
 |     const bool show_in_alphas = true; | ||||||
| 
 | 
 | ||||||
|     // Show the dialog if current > last, but they differ in more than just patch.
 |     if (! show_in_alphas && semver_current.prerelease() | ||||||
|     return ((semver_current.maj() > semver_last_sent.maj()) |        && std::string(semver_current.prerelease()).find("alpha") != std::string::npos) | ||||||
|  |             return false; | ||||||
|  | 
 | ||||||
|  |     // New version means current > last, but they must differ in more than just patch.
 | ||||||
|  |     bool new_version = ((semver_current.maj() > semver_last_sent.maj()) | ||||||
|         || (semver_current.maj() == semver_last_sent.maj() && semver_current.min() > semver_last_sent.min() )); |         || (semver_current.maj() == semver_last_sent.maj() && semver_current.min() > semver_last_sent.min() )); | ||||||
|  | 
 | ||||||
|  |     if (! new_version) | ||||||
|  |         return false; | ||||||
|  | 
 | ||||||
|  |     // We'll misuse the version check to check internet connection here.
 | ||||||
|  |     bool is_internet = false; | ||||||
|  |     Http::get(wxGetApp().app_config->version_check_url()) | ||||||
|  |         .size_limit(SLIC3R_VERSION_BODY_MAX) | ||||||
|  |         .timeout_max(2) | ||||||
|  |         .on_complete([&](std::string, unsigned) { | ||||||
|  |             is_internet = true; | ||||||
|  |         }) | ||||||
|  |         .perform_sync(); | ||||||
|  |     return is_internet; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -162,9 +185,10 @@ static std::map<std::string, std::string> get_cpu_info_from_registry() | ||||||
|     std::map<std::string, std::string> out; |     std::map<std::string, std::string> out; | ||||||
| 
 | 
 | ||||||
|     int idx = -1; |     int idx = -1; | ||||||
|     constexpr DWORD bufsize_ = 200; |     constexpr DWORD bufsize_ = 500; | ||||||
|     DWORD bufsize = bufsize_; |     DWORD bufsize = bufsize_-1; // Ensure a terminating zero.
 | ||||||
|     char buf[bufsize_] = ""; |     char buf[bufsize_] = ""; | ||||||
|  |     memset(buf, 0, bufsize_); | ||||||
|     const std::string reg_dir = "HARDWARE\\DESCRIPTION\\System\\CentralProcessor\\"; |     const std::string reg_dir = "HARDWARE\\DESCRIPTION\\System\\CentralProcessor\\"; | ||||||
|     std::string reg_path = reg_dir; |     std::string reg_path = reg_dir; | ||||||
| 
 | 
 | ||||||
|  | @ -186,7 +210,7 @@ static std::map<std::string, std::string> get_cpu_info_from_registry() | ||||||
|         } |         } | ||||||
|         ++idx; |         ++idx; | ||||||
|         reg_path = reg_dir + std::to_string(idx) + "\\"; |         reg_path = reg_dir + std::to_string(idx) + "\\"; | ||||||
|         bufsize = bufsize_; |         bufsize = bufsize_-1; | ||||||
|     } |     } | ||||||
|     return out; |     return out; | ||||||
| } | } | ||||||
|  | @ -194,7 +218,7 @@ static std::map<std::string, std::string> get_cpu_info_from_registry() | ||||||
| static std::map<std::string, std::string> parse_lscpu_etc(const std::string& name, char delimiter) | static std::map<std::string, std::string> parse_lscpu_etc(const std::string& name, char delimiter) | ||||||
| { | { | ||||||
|     std::map<std::string, std::string> out; |     std::map<std::string, std::string> out; | ||||||
|     constexpr size_t max_len = 100; |     constexpr size_t max_len = 1000; | ||||||
|     char cline[max_len] = ""; |     char cline[max_len] = ""; | ||||||
|     FILE* fp = popen(name.data(), "r"); |     FILE* fp = popen(name.data(), "r"); | ||||||
|     if (fp != NULL) { |     if (fp != NULL) { | ||||||
|  | @ -260,10 +284,12 @@ static std::string get_unique_id() | ||||||
|     char buf[buf_size] = ""; |     char buf[buf_size] = ""; | ||||||
|     memset(&buf, 0, sizeof(buf)); |     memset(&buf, 0, sizeof(buf)); | ||||||
|     io_registry_entry_t ioRegistryRoot = IORegistryEntryFromPath(kIOMasterPortDefault, "IOService:/"); |     io_registry_entry_t ioRegistryRoot = IORegistryEntryFromPath(kIOMasterPortDefault, "IOService:/"); | ||||||
|     CFStringRef uuidCf = (CFStringRef)IORegistryEntryCreateCFProperty(ioRegistryRoot, CFSTR(kIOPlatformUUIDKey), kCFAllocatorDefault, 0); |     if (ioRegistryRoot != MACH_PORT_NULL) { | ||||||
|     IOObjectRelease(ioRegistryRoot); |         CFStringRef uuidCf = (CFStringRef)IORegistryEntryCreateCFProperty(ioRegistryRoot, CFSTR(kIOPlatformUUIDKey), kCFAllocatorDefault, 0); | ||||||
|     CFStringGetCString(uuidCf, buf, buf_size, kCFStringEncodingMacRoman); |         IOObjectRelease(ioRegistryRoot); | ||||||
|     CFRelease(uuidCf); |         CFStringGetCString(uuidCf, buf, buf_size, kCFStringEncodingMacRoman); | ||||||
|  |         CFRelease(uuidCf); | ||||||
|  |     } | ||||||
|     // Now convert the string to std::vector<unsigned char>.
 |     // Now convert the string to std::vector<unsigned char>.
 | ||||||
|     for (char* c = buf; *c != 0; ++c) |     for (char* c = buf; *c != 0; ++c) | ||||||
|         unique.emplace_back((unsigned char)(*c)); |         unique.emplace_back((unsigned char)(*c)); | ||||||
|  | @ -318,11 +344,20 @@ static std::string generate_system_info_json() | ||||||
|     std::string unique_id = get_unique_id(); |     std::string unique_id = get_unique_id(); | ||||||
| 
 | 
 | ||||||
|     // Get system language.
 |     // Get system language.
 | ||||||
|     std::string sys_language = "Unknown"; |     std::string sys_language = "Unknown"; // important to init, see the __APPLE__ block.
 | ||||||
|     const wxLanguage lang_system = wxLanguage(wxLocale::GetSystemLanguage()); |     #ifndef __APPLE__ | ||||||
|     if (lang_system != wxLANGUAGE_UNKNOWN) |         // Following apparently does not work on macOS.
 | ||||||
|         sys_language = wxLocale::GetLanguageInfo(lang_system)->CanonicalName.ToUTF8().data(); |         const wxLanguage lang_system = wxLanguage(wxLocale::GetSystemLanguage()); | ||||||
| 
 |         if (lang_system != wxLANGUAGE_UNKNOWN) | ||||||
|  |             sys_language = wxLocale::GetLanguageInfo(lang_system)->CanonicalName.ToUTF8().data(); | ||||||
|  |     #else // __APPLE__
 | ||||||
|  |         CFLocaleRef cflocale = CFLocaleCopyCurrent(); | ||||||
|  |         CFStringRef value = (CFStringRef)CFLocaleGetValue(cflocale, kCFLocaleLanguageCode); | ||||||
|  |         char temp[10] = ""; | ||||||
|  |         CFStringGetCString(value, temp, 10, kCFStringEncodingUTF8); | ||||||
|  |         sys_language = temp; | ||||||
|  |         CFRelease(cflocale); | ||||||
|  |     #endif | ||||||
|     // Build a property tree with all the information.
 |     // Build a property tree with all the information.
 | ||||||
|     namespace pt = boost::property_tree; |     namespace pt = boost::property_tree; | ||||||
| 
 | 
 | ||||||
|  | @ -364,9 +399,13 @@ static std::string generate_system_info_json() | ||||||
|     data_node.put("SystemLanguage", sys_language); |     data_node.put("SystemLanguage", sys_language); | ||||||
|     data_node.put("TranslationLanguage: ", wxGetApp().app_config->get("translation_language")); |     data_node.put("TranslationLanguage: ", wxGetApp().app_config->get("translation_language")); | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|     pt::ptree hw_node; |     pt::ptree hw_node; | ||||||
|     hw_node.put("ArchName", wxPlatformInfo::Get().GetArchName()); |     { | ||||||
|     hw_node.put("RAM_MB", size_t(Slic3r::total_physical_memory()/1000000)); |         hw_node.put("ArchName", wxPlatformInfo::Get().GetArchName()); | ||||||
|  |         size_t num = std::round(Slic3r::total_physical_memory()/107374100.); | ||||||
|  |         hw_node.put("RAM_GiB", std::to_string(num / 10) + "." + std::to_string(num % 10)); | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     // Now get some CPU info:
 |     // Now get some CPU info:
 | ||||||
|     pt::ptree cpu_node; |     pt::ptree cpu_node; | ||||||
|  | @ -381,31 +420,32 @@ static std::string generate_system_info_json() | ||||||
|      cpu_node.put("Model",  sysctl["machdep.cpu.brand_string"]); |      cpu_node.put("Model",  sysctl["machdep.cpu.brand_string"]); | ||||||
|      cpu_node.put("Vendor", sysctl["machdep.cpu.vendor"]); |      cpu_node.put("Vendor", sysctl["machdep.cpu.vendor"]); | ||||||
| #else // linux/BSD
 | #else // linux/BSD
 | ||||||
|     std::map<std::string, std::string> lscpu = parse_lscpu_etc("lscpu", ':'); |     std::map<std::string, std::string> lscpu = parse_lscpu_etc("cat /proc/cpuinfo", ':'); | ||||||
|     cpu_node.put("Arch",   lscpu["Architecture"]); |     if (auto ncpu_it = lscpu.find("processor"); ncpu_it != lscpu.end()) { | ||||||
|     cpu_node.put("Cores",  lscpu["CPU(s)"]); |         std::string& ncpu = ncpu_it->second; | ||||||
|     cpu_node.put("Model",  lscpu["Model name"]); |         if (int num=0; std::from_chars(ncpu.data(), ncpu.data() + ncpu.size(), num).ec != std::errc::invalid_argument) | ||||||
|     cpu_node.put("Vendor", lscpu["Vendor ID"]); |             ncpu = std::to_string(num + 1); | ||||||
|  |     } | ||||||
|  |     cpu_node.put("Cores",  lscpu["processor"]); | ||||||
|  |     cpu_node.put("Model",  lscpu["model name"]); | ||||||
|  |     cpu_node.put("Vendor", lscpu["vendor_id"]); | ||||||
| #endif | #endif | ||||||
|     hw_node.add_child("CPU", cpu_node); |     hw_node.add_child("CPU", cpu_node); | ||||||
| 
 | 
 | ||||||
|     pt::ptree monitors_node; |     pt::ptree monitors_node; | ||||||
|     for (int i=0; i<int(wxDisplay::GetCount()); ++i) { |     for (int i=0; i<int(wxDisplay::GetCount()); ++i) { | ||||||
|         wxDisplay display(i); |         wxDisplay display(i); | ||||||
|         double scaling = -1.; |  | ||||||
|         #if wxCHECK_VERSION(3, 1, 2) // we have wxDisplag::GetPPI
 |  | ||||||
|             int std_ppi = 96; |  | ||||||
|             #ifdef __WXOSX__ // see impl of wxDisplay::GetStdPPIValue from 3.1.5
 |  | ||||||
|                 std_ppi = 72; |  | ||||||
|             #endif |  | ||||||
|             scaling = double(display.GetPPI().GetWidth()) / std_ppi; |  | ||||||
|         #endif |  | ||||||
|         pt::ptree monitor_node; // Create an unnamed node containing the value
 |         pt::ptree monitor_node; // Create an unnamed node containing the value
 | ||||||
|         monitor_node.put("width", display.GetGeometry().GetWidth()); |         monitor_node.put("width", display.GetGeometry().GetWidth()); | ||||||
|         monitor_node.put("height", display.GetGeometry().GetHeight()); |         monitor_node.put("height", display.GetGeometry().GetHeight()); | ||||||
|         std::stringstream ss; | 
 | ||||||
|         ss << std::setprecision(3) << scaling; |         // Only get the scaling on Win, it is not reliable on other platforms.
 | ||||||
|         monitor_node.put("scaling", ss.str() ); |         #if defined(_WIN32) && wxCHECK_VERSION(3, 1, 2) | ||||||
|  |             double scaling = display.GetPPI().GetWidth() / 96.; | ||||||
|  |             std::stringstream ss; | ||||||
|  |             ss << std::setprecision(3) << scaling; | ||||||
|  |             monitor_node.put("scaling", ss.str() ); | ||||||
|  |         #endif | ||||||
|         monitors_node.push_back(std::make_pair("", monitor_node)); |         monitors_node.push_back(std::make_pair("", monitor_node)); | ||||||
|     } |     } | ||||||
|     hw_node.add_child("Monitors", monitors_node); |     hw_node.add_child("Monitors", monitors_node); | ||||||
|  | @ -435,15 +475,23 @@ static std::string generate_system_info_json() | ||||||
|     pt::ptree root; |     pt::ptree root; | ||||||
|     root.add_child("data", data_node); |     root.add_child("data", data_node); | ||||||
| 
 | 
 | ||||||
|  |     // Now go through all the values and trim leading/trailing whitespace.
 | ||||||
|  |     // Some CPU names etc apparently have trailing spaces...
 | ||||||
|  |     std::function<void(pt::ptree&)> remove_whitespace; | ||||||
|  |     remove_whitespace = [&remove_whitespace](pt::ptree& t) -> void | ||||||
|  |     { | ||||||
|  |         if (t.empty()) // Trim whitespace
 | ||||||
|  |             boost::algorithm::trim(t.data()); | ||||||
|  |         else | ||||||
|  |             for (auto it = t.begin(); it != t.end(); ++it) | ||||||
|  |                 remove_whitespace(it->second); | ||||||
|  |     }; | ||||||
|  |     remove_whitespace(root); | ||||||
|  | 
 | ||||||
|     // Serialize the tree into JSON and return it.
 |     // Serialize the tree into JSON and return it.
 | ||||||
|     std::stringstream ss; |     std::stringstream ss; | ||||||
|     pt::write_json(ss, root); |     pt::write_json(ss, root); | ||||||
|     return ss.str(); |     return ss.str(); | ||||||
| 
 |  | ||||||
|     // FURTHER THINGS TO CONSIDER:
 |  | ||||||
|     //std::cout << wxPlatformInfo::Get().GetOperatingSystemFamilyName() << std::endl;          // Unix
 |  | ||||||
|     // ? CPU, GPU, UNKNOWN ?
 |  | ||||||
|     // printers? will they be installed already?
 |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -453,6 +501,8 @@ SendSystemInfoDialog::SendSystemInfoDialog(wxWindow* parent) | ||||||
|     GUI::DPIDialog(parent, wxID_ANY, _L("Send system info"), wxDefaultPosition, wxDefaultSize, |     GUI::DPIDialog(parent, wxID_ANY, _L("Send system info"), wxDefaultPosition, wxDefaultSize, | ||||||
|            wxDEFAULT_DIALOG_STYLE) |            wxDEFAULT_DIALOG_STYLE) | ||||||
| { | { | ||||||
|  |     const int em = GUI::wxGetApp().em_unit(); | ||||||
|  | 
 | ||||||
|     // Get current PrusaSliver version info.
 |     // Get current PrusaSliver version info.
 | ||||||
|     std::string app_name; |     std::string app_name; | ||||||
|     { |     { | ||||||
|  | @ -500,7 +550,7 @@ SendSystemInfoDialog::SendSystemInfoDialog(wxWindow* parent) | ||||||
|            std::string("<i>") + filename + "</i>"); |            std::string("<i>") + filename + "</i>"); | ||||||
|     wxString label3 = _L("Show verbatim data that will be sent"); |     wxString label3 = _L("Show verbatim data that will be sent"); | ||||||
| 
 | 
 | ||||||
|     auto* html_window = new wxHtmlWindow(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHW_SCROLLBAR_NEVER); |     auto* html_window = new wxHtmlWindow(this, wxID_ANY, wxDefaultPosition, wxSize(70*em, 34*em), wxHW_SCROLLBAR_NEVER); | ||||||
|     wxString html = GUI::format_wxstr( |     wxString html = GUI::format_wxstr( | ||||||
|             "<html><body bgcolor=%1%><font color=%2%>" |             "<html><body bgcolor=%1%><font color=%2%>" | ||||||
|             "<table><tr><td>" |             "<table><tr><td>" | ||||||
|  | @ -514,7 +564,7 @@ SendSystemInfoDialog::SendSystemInfoDialog(wxWindow* parent) | ||||||
|             + "<b><a href=\"show\">" + label3 + "</a></b><br />" |             + "<b><a href=\"show\">" + label3 + "</a></b><br />" | ||||||
|             + "</font></body></html>", bgr_clr_str, text_clr_str); |             + "</font></body></html>", bgr_clr_str, text_clr_str); | ||||||
|     html_window->SetPage(html); |     html_window->SetPage(html); | ||||||
|     html_window->Bind(wxEVT_HTML_LINK_CLICKED, [this](wxHtmlLinkEvent &evt) { |     html_window->Bind(wxEVT_HTML_LINK_CLICKED, [this](wxHtmlLinkEvent&) { | ||||||
|                                                    ShowJsonDialog dlg(this, m_system_info_json, GetSize().Scale(0.9, 0.7)); |                                                    ShowJsonDialog dlg(this, m_system_info_json, GetSize().Scale(0.9, 0.7)); | ||||||
|                                                    dlg.ShowModal(); |                                                    dlg.ShowModal(); | ||||||
|     }); |     }); | ||||||
|  | @ -526,7 +576,6 @@ SendSystemInfoDialog::SendSystemInfoDialog(wxWindow* parent) | ||||||
|     m_btn_send = new wxButton(this, wxID_ANY, _L("Send system info")); |     m_btn_send = new wxButton(this, wxID_ANY, _L("Send system info")); | ||||||
| 
 | 
 | ||||||
|     auto* hsizer = new wxBoxSizer(wxHORIZONTAL); |     auto* hsizer = new wxBoxSizer(wxHORIZONTAL); | ||||||
|     const int em = GUI::wxGetApp().em_unit(); |  | ||||||
|     hsizer->Add(m_btn_ask_later); |     hsizer->Add(m_btn_ask_later); | ||||||
|     hsizer->AddSpacer(em); |     hsizer->AddSpacer(em); | ||||||
|     hsizer->Add(m_btn_dont_send); |     hsizer->Add(m_btn_dont_send); | ||||||
|  | @ -548,6 +597,8 @@ SendSystemInfoDialog::SendSystemInfoDialog(wxWindow* parent) | ||||||
|     SetSize(std::max(size.GetWidth(), MIN_WIDTH * em), |     SetSize(std::max(size.GetWidth(), MIN_WIDTH * em), | ||||||
|             std::max(size.GetHeight(), MIN_HEIGHT * em)); |             std::max(size.GetHeight(), MIN_HEIGHT * em)); | ||||||
| 
 | 
 | ||||||
|  |     CenterOnParent(); | ||||||
|  | 
 | ||||||
|     m_btn_send->Bind(wxEVT_BUTTON, [this](const wxEvent&) |     m_btn_send->Bind(wxEVT_BUTTON, [this](const wxEvent&) | ||||||
|                                     { |                                     { | ||||||
|                                         if (send_info()) { |                                         if (send_info()) { | ||||||
|  | @ -592,15 +643,16 @@ bool SendSystemInfoDialog::send_info() | ||||||
|     } result; // No synchronization needed, UI thread reads only after worker is joined.
 |     } result; // No synchronization needed, UI thread reads only after worker is joined.
 | ||||||
| 
 | 
 | ||||||
|     auto send = [&job_done, &result](const std::string& data) { |     auto send = [&job_done, &result](const std::string& data) { | ||||||
|         const std::string url = "https://files.prusa3d.com/wp-json/v1/ps"; |         Http http = Http::post(SEND_SYSTEM_INFO_URL); | ||||||
|         Http http = Http::post(url); |  | ||||||
|         http.header("Content-Type", "application/json") |         http.header("Content-Type", "application/json") | ||||||
|  |             .timeout_max(6) // seconds
 | ||||||
|             .set_post_body(data) |             .set_post_body(data) | ||||||
|             .on_complete([&result](std::string body, unsigned status) { |             .on_complete([&result](std::string body, unsigned status) { | ||||||
|                 result = { Result::Success, _L("System info sent successfully. Thank you.") }; |                 result = { Result::Success, _L("System info sent successfully. Thank you.") }; | ||||||
|             }) |             }) | ||||||
|             .on_error([&result](std::string body, std::string error, unsigned status) { |             .on_error([&result](std::string body, std::string error, unsigned status) { | ||||||
|                 result = { Result::Error, GUI::format_wxstr(_L("Sending system info failed! Status: %1%"), status) }; |                 result = { Result::Error, _L("Sending system info failed!") }; | ||||||
|  |                 BOOST_LOG_TRIVIAL(error) << "Sending system info failed! STATUS: " << status; | ||||||
|             }) |             }) | ||||||
|             .on_progress([&job_done, &result](Http::Progress, bool &cancel) { |             .on_progress([&job_done, &result](Http::Progress, bool &cancel) { | ||||||
|                 if (job_done) // UI thread wants us to cancel.
 |                 if (job_done) // UI thread wants us to cancel.
 | ||||||
|  | @ -622,8 +674,10 @@ bool SendSystemInfoDialog::send_info() | ||||||
|     job_done = true;       // In case the user closed the dialog, let the other thread know
 |     job_done = true;       // In case the user closed the dialog, let the other thread know
 | ||||||
|     sending_thread.join(); // and wait until it terminates.
 |     sending_thread.join(); // and wait until it terminates.
 | ||||||
| 
 | 
 | ||||||
|     InfoDialog info_dlg(wxGetApp().mainframe, wxEmptyString, result.str); |     if (result.value != Result::Cancelled) { // user knows he cancelled, no need to tell him.
 | ||||||
|     info_dlg.ShowModal(); |         InfoDialog info_dlg(wxGetApp().mainframe, wxEmptyString, result.str); | ||||||
|  |         info_dlg.ShowModal(); | ||||||
|  |     } | ||||||
|     return result.value == Result::Success; |     return result.value == Result::Success; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -104,6 +104,7 @@ struct Http::priv | ||||||
| { | { | ||||||
| 	enum { | 	enum { | ||||||
| 		DEFAULT_TIMEOUT_CONNECT = 10, | 		DEFAULT_TIMEOUT_CONNECT = 10, | ||||||
|  |         DEFAULT_TIMEOUT_MAX = 0, | ||||||
| 		DEFAULT_SIZE_LIMIT = 5 * 1024 * 1024, | 		DEFAULT_SIZE_LIMIT = 5 * 1024 * 1024, | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
|  | @ -137,6 +138,7 @@ struct Http::priv | ||||||
| 	static size_t form_file_read_cb(char *buffer, size_t size, size_t nitems, void *userp); | 	static size_t form_file_read_cb(char *buffer, size_t size, size_t nitems, void *userp); | ||||||
| 
 | 
 | ||||||
| 	void set_timeout_connect(long timeout); | 	void set_timeout_connect(long timeout); | ||||||
|  |     void set_timeout_max(long timeout); | ||||||
| 	void form_add_file(const char *name, const fs::path &path, const char* filename); | 	void form_add_file(const char *name, const fs::path &path, const char* filename); | ||||||
| 	void set_post_body(const fs::path &path); | 	void set_post_body(const fs::path &path); | ||||||
| 	void set_post_body(const std::string &body); | 	void set_post_body(const std::string &body); | ||||||
|  | @ -163,6 +165,7 @@ Http::priv::priv(const std::string &url) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	set_timeout_connect(DEFAULT_TIMEOUT_CONNECT); | 	set_timeout_connect(DEFAULT_TIMEOUT_CONNECT); | ||||||
|  |     set_timeout_max(DEFAULT_TIMEOUT_MAX); | ||||||
| 	::curl_easy_setopt(curl, CURLOPT_URL, url.c_str());   // curl makes a copy internally
 | 	::curl_easy_setopt(curl, CURLOPT_URL, url.c_str());   // curl makes a copy internally
 | ||||||
| 	::curl_easy_setopt(curl, CURLOPT_USERAGENT, SLIC3R_APP_NAME "/" SLIC3R_VERSION); | 	::curl_easy_setopt(curl, CURLOPT_USERAGENT, SLIC3R_APP_NAME "/" SLIC3R_VERSION); | ||||||
| 	::curl_easy_setopt(curl, CURLOPT_ERRORBUFFER, &error_buffer.front()); | 	::curl_easy_setopt(curl, CURLOPT_ERRORBUFFER, &error_buffer.front()); | ||||||
|  | @ -253,6 +256,11 @@ void Http::priv::set_timeout_connect(long timeout) | ||||||
| 	::curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT, timeout); | 	::curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT, timeout); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void Http::priv::set_timeout_max(long timeout) | ||||||
|  | { | ||||||
|  |     ::curl_easy_setopt(curl, CURLOPT_TIMEOUT, timeout); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void Http::priv::form_add_file(const char *name, const fs::path &path, const char* filename) | void Http::priv::form_add_file(const char *name, const fs::path &path, const char* filename) | ||||||
| { | { | ||||||
| 	// We can't use CURLFORM_FILECONTENT, because curl doesn't support Unicode filenames on Windows
 | 	// We can't use CURLFORM_FILECONTENT, because curl doesn't support Unicode filenames on Windows
 | ||||||
|  | @ -409,6 +417,13 @@ Http& Http::timeout_connect(long timeout) | ||||||
| 	return *this; | 	return *this; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | Http& Http::timeout_max(long timeout) | ||||||
|  | { | ||||||
|  |     if (timeout < 1) { timeout = priv::DEFAULT_TIMEOUT_MAX; } | ||||||
|  |     if (p) { p->set_timeout_max(timeout); } | ||||||
|  |     return *this; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| Http& Http::size_limit(size_t sizeLimit) | Http& Http::size_limit(size_t sizeLimit) | ||||||
| { | { | ||||||
| 	if (p) { p->limit = sizeLimit; } | 	if (p) { p->limit = sizeLimit; } | ||||||
|  |  | ||||||
|  | @ -58,6 +58,8 @@ public: | ||||||
| 
 | 
 | ||||||
| 	// Sets a maximum connection timeout in seconds
 | 	// Sets a maximum connection timeout in seconds
 | ||||||
| 	Http& timeout_connect(long timeout); | 	Http& timeout_connect(long timeout); | ||||||
|  |     // Sets a maximum total request timeout in seconds
 | ||||||
|  |     Http& timeout_max(long timeout); | ||||||
| 	// Sets a maximum size of the data that can be received.
 | 	// Sets a maximum size of the data that can be received.
 | ||||||
| 	// A value of zero sets the default limit, which is is 5MB.
 | 	// A value of zero sets the default limit, which is is 5MB.
 | ||||||
| 	Http& size_limit(size_t sizeLimit); | 	Http& size_limit(size_t sizeLimit); | ||||||
|  |  | ||||||
|  | @ -46,10 +46,6 @@ using Slic3r::GUI::Config::SnapshotDB; | ||||||
| namespace Slic3r { | namespace Slic3r { | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| enum { |  | ||||||
| 	SLIC3R_VERSION_BODY_MAX = 256, |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| static const char *INDEX_FILENAME = "index.idx"; | static const char *INDEX_FILENAME = "index.idx"; | ||||||
| static const char *TMP_EXTENSION = ".download"; | static const char *TMP_EXTENSION = ".download"; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -13,6 +13,8 @@ class AppConfig; | ||||||
| class PresetBundle; | class PresetBundle; | ||||||
| class Semver; | class Semver; | ||||||
| 
 | 
 | ||||||
|  | const int SLIC3R_VERSION_BODY_MAX = 256; | ||||||
|  | 
 | ||||||
| class PresetUpdater | class PresetUpdater | ||||||
| { | { | ||||||
| public: | public: | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Filip Sykala
						Filip Sykala