mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-22 00:01:09 -06:00 
			
		
		
		
	Assigning of wiping extrusions improved
This commit is contained in:
		
							parent
							
								
									8a47852be2
								
							
						
					
					
						commit
						bc5bd1b42b
					
				
					 6 changed files with 101 additions and 78 deletions
				
			
		|  | @ -1239,7 +1239,7 @@ void GCode::process_layer( | ||||||
|                             continue; |                             continue; | ||||||
| 
 | 
 | ||||||
|                         // This extrusion is part of certain Region, which tells us which extruder should be used for it:
 |                         // This extrusion is part of certain Region, which tells us which extruder should be used for it:
 | ||||||
|                         int correct_extruder_id = entity_type=="infills" ? std::max<int>(0, (is_solid_infill(fill->entities.front()->role()) ? region.config.solid_infill_extruder : region.config.infill_extruder) - 1) : |                         int correct_extruder_id = get_extruder(fill, region); entity_type=="infills" ? std::max<int>(0, (is_solid_infill(fill->entities.front()->role()) ? region.config.solid_infill_extruder : region.config.infill_extruder) - 1) : | ||||||
|                                                                            std::max<int>(region.config.perimeter_extruder.value - 1, 0); |                                                                            std::max<int>(region.config.perimeter_extruder.value - 1, 0); | ||||||
| 
 | 
 | ||||||
|                         // Let's recover vector of extruder overrides:
 |                         // Let's recover vector of extruder overrides:
 | ||||||
|  |  | ||||||
|  | @ -330,8 +330,10 @@ void ToolOrdering::collect_extruder_statistics(bool prime_multi_material) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|     // This function is called from Print::mark_wiping_extrusions and sets extruder that it should be printed with (-1 .. as usual)
 | 
 | ||||||
|     void WipingExtrusions::set_extruder_override(const ExtrusionEntity* entity, unsigned int copy_id, int extruder, unsigned int num_of_copies) { | 
 | ||||||
|  | // This function is called from Print::mark_wiping_extrusions and sets extruder this entity should be printed with (-1 .. as usual)
 | ||||||
|  | void WipingExtrusions::set_extruder_override(const ExtrusionEntity* entity, unsigned int copy_id, int extruder, unsigned int num_of_copies) { | ||||||
|     something_overridden = true; |     something_overridden = true; | ||||||
| 
 | 
 | ||||||
|     auto entity_map_it = (entity_map.insert(std::make_pair(entity, std::vector<int>()))).first; // (add and) return iterator
 |     auto entity_map_it = (entity_map.insert(std::make_pair(entity, std::vector<int>()))).first; // (add and) return iterator
 | ||||||
|  | @ -343,17 +345,17 @@ void ToolOrdering::collect_extruder_statistics(bool prime_multi_material) | ||||||
|         std::cout << "ERROR: Entity extruder overriden multiple times!!!\n";    // A debugging message - this must never happen.
 |         std::cout << "ERROR: Entity extruder overriden multiple times!!!\n";    // A debugging message - this must never happen.
 | ||||||
| 
 | 
 | ||||||
|     copies_vector[copy_id] = extruder; |     copies_vector[copy_id] = extruder; | ||||||
|     } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|     // Following function is called from process_layer and returns pointer to vector with information about which extruders should be used for given copy of this entity.
 | // Following function is called from process_layer and returns pointer to vector with information about which extruders should be used for given copy of this entity.
 | ||||||
|     // It first makes sure the pointer is valid (creates the vector if it does not exist) and contains a record for each copy
 | // It first makes sure the pointer is valid (creates the vector if it does not exist) and contains a record for each copy
 | ||||||
|     // It also modifies the vector in place and changes all -1 to correct_extruder_id (at the time the overrides were created, correct extruders were not known,
 | // It also modifies the vector in place and changes all -1 to correct_extruder_id (at the time the overrides were created, correct extruders were not known,
 | ||||||
|     // so -1 was used as "print as usual".
 | // so -1 was used as "print as usual".
 | ||||||
|     // The resulting vector has to keep track of which extrusions are the ones that were overridden and which were not. In the extruder is used as overridden,
 | // The resulting vector has to keep track of which extrusions are the ones that were overridden and which were not. In the extruder is used as overridden,
 | ||||||
|     // its number is saved as it is (zero-based index). Usual extrusions are saved as -number-1 (unfortunately there is no negative zero).
 | // its number is saved as it is (zero-based index). Usual extrusions are saved as -number-1 (unfortunately there is no negative zero).
 | ||||||
|     const std::vector<int>* WipingExtrusions::get_extruder_overrides(const ExtrusionEntity* entity, int correct_extruder_id, int num_of_copies) { | const std::vector<int>* WipingExtrusions::get_extruder_overrides(const ExtrusionEntity* entity, int correct_extruder_id, int num_of_copies) { | ||||||
|     auto entity_map_it = entity_map.find(entity); |     auto entity_map_it = entity_map.find(entity); | ||||||
|     if (entity_map_it == entity_map.end()) |     if (entity_map_it == entity_map.end()) | ||||||
|         entity_map_it = (entity_map.insert(std::make_pair(entity, std::vector<int>()))).first; |         entity_map_it = (entity_map.insert(std::make_pair(entity, std::vector<int>()))).first; | ||||||
|  | @ -365,7 +367,7 @@ void ToolOrdering::collect_extruder_statistics(bool prime_multi_material) | ||||||
|     std::replace(entity_map_it->second.begin(), entity_map_it->second.end(), -1, -correct_extruder_id-1); |     std::replace(entity_map_it->second.begin(), entity_map_it->second.end(), -1, -correct_extruder_id-1); | ||||||
| 
 | 
 | ||||||
|     return &(entity_map_it->second); |     return &(entity_map_it->second); | ||||||
|     } | } | ||||||
|      |      | ||||||
| 
 | 
 | ||||||
| } // namespace Slic3r
 | } // namespace Slic3r
 | ||||||
|  |  | ||||||
|  | @ -11,7 +11,6 @@ class Print; | ||||||
| class PrintObject; | class PrintObject; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| // Object of this class holds information about whether an extrusion is printed immediately
 | // Object of this class holds information about whether an extrusion is printed immediately
 | ||||||
| // after a toolchange (as part of infill/perimeter wiping) or not. One extrusion can be a part
 | // after a toolchange (as part of infill/perimeter wiping) or not. One extrusion can be a part
 | ||||||
| // of several copies - this has to be taken into account.
 | // of several copies - this has to be taken into account.
 | ||||||
|  |  | ||||||
|  | @ -1177,66 +1177,50 @@ void Print::_make_wipe_tower() | ||||||
| // and returns volume that is left to be wiped on the wipe tower.
 | // and returns volume that is left to be wiped on the wipe tower.
 | ||||||
| float Print::mark_wiping_extrusions(ToolOrdering::LayerTools& layer_tools, unsigned int new_extruder, float volume_to_wipe) | float Print::mark_wiping_extrusions(ToolOrdering::LayerTools& layer_tools, unsigned int new_extruder, float volume_to_wipe) | ||||||
| { | { | ||||||
|     // Strategy for  wiping (TODO):
 |  | ||||||
|     // if !infill_first
 |  | ||||||
|     //      start with dedicated objects
 |  | ||||||
|     //            print a perimeter and its corresponding infill immediately after
 |  | ||||||
|     //            repeat until there are no dedicated objects left
 |  | ||||||
|     //            if there are some left and this is the last toolchange on the layer, mark all remaining extrusions of the object (so we don't have to travel back to it later)
 |  | ||||||
|     //      move to normal objects
 |  | ||||||
|     //            start with one object and start assigning its infill, if their perimeters ARE ALREADY EXTRUDED
 |  | ||||||
|     //            never touch perimeters
 |  | ||||||
|     //
 |  | ||||||
|     // if infill first
 |  | ||||||
|     //        start with dedicated objects
 |  | ||||||
|     //            print an infill and its corresponding perimeter immediately after
 |  | ||||||
|     //            repeat until you run out of infills
 |  | ||||||
|     //        move to normal objects
 |  | ||||||
|     //            start assigning infills (one copy after another)
 |  | ||||||
|     //            repeat until you run out of infills, leave perimeters be
 |  | ||||||
| 
 |  | ||||||
|     const float min_infill_volume = 0.f; // ignore infill with smaller volume than this
 |     const float min_infill_volume = 0.f; // ignore infill with smaller volume than this
 | ||||||
| 
 | 
 | ||||||
|     if (config.filament_soluble.get_at(new_extruder)) |     if (config.filament_soluble.get_at(new_extruder)) | ||||||
|         return volume_to_wipe;      // Soluble filament cannot be wiped in a random infill
 |         return volume_to_wipe;      // Soluble filament cannot be wiped in a random infill
 | ||||||
| 
 | 
 | ||||||
|  |     PrintObjectPtrs object_list = objects; | ||||||
|  | 
 | ||||||
|  |     // sort objects so that dedicated for wiping are at the beginning:
 | ||||||
|  |     std::sort(object_list.begin(), object_list.end(), [](const PrintObject* a, const PrintObject* b) { return a->config.wipe_into_objects; });  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|     for (size_t i = 0; i < objects.size(); ++ i) {                              // Let's iterate through all objects...
 |     // We will now iterate through objects
 | ||||||
|         if (!objects[i]->config.wipe_into_infill && !objects[i]->config.wipe_into_objects) |     //  - first through the dedicated ones to mark perimeters or infills (depending on infill_first)
 | ||||||
|  |     //  - second through the dedicated ones again to mark infills or perimeters (depending on infill_first)
 | ||||||
|  |     //  - then for the others to mark infills
 | ||||||
|  |     // this is controlled by the following variable:
 | ||||||
|  |     bool perimeters_done = false; | ||||||
|  | 
 | ||||||
|  |     for (int i=0 ; i<(int)object_list.size() ; ++i) {                              // Let's iterate through all objects...
 | ||||||
|  |         const auto& object = object_list[i]; | ||||||
|  | 
 | ||||||
|  |         if (!perimeters_done && (i+1==objects.size() || !objects[i+1]->config.wipe_into_objects)) { // last dedicated object in list
 | ||||||
|  |             perimeters_done = true; | ||||||
|  |             i=-1;   // let's go from the start again
 | ||||||
|             continue; |             continue; | ||||||
| 
 |  | ||||||
|         Layer* this_layer = nullptr; |  | ||||||
|         for (unsigned int a = 0; a < objects[i]->layers.size(); ++a) // Finds this layer
 |  | ||||||
|             if (std::abs(layer_tools.print_z - objects[i]->layers[a]->print_z) < EPSILON) { |  | ||||||
|                 this_layer = objects[i]->layers[a]; |  | ||||||
|                 break; |  | ||||||
|         } |         } | ||||||
|         if (this_layer == nullptr) |  | ||||||
|             continue; |  | ||||||
| 
 | 
 | ||||||
|         unsigned int num_of_copies = objects[i]->_shifted_copies.size(); |         // Finds this layer:
 | ||||||
|  |         auto this_layer_it = std::find_if(object->layers.begin(), object->layers.end(), [&layer_tools](const Layer* lay) { return std::abs(layer_tools.print_z - lay->print_z)<EPSILON; }); | ||||||
|  |         if (this_layer_it == object->layers.end()) | ||||||
|  |             continue; | ||||||
|  |         const Layer* this_layer = *this_layer_it; | ||||||
|  |         unsigned int num_of_copies = object->_shifted_copies.size(); | ||||||
| 
 | 
 | ||||||
|         for (unsigned int copy = 0; copy < num_of_copies; ++copy) {    // iterate through copies first, so that we mark neighbouring infills to minimize travel moves
 |         for (unsigned int copy = 0; copy < num_of_copies; ++copy) {    // iterate through copies first, so that we mark neighbouring infills to minimize travel moves
 | ||||||
| 
 | 
 | ||||||
|             for (size_t region_id = 0; region_id < objects[i]->print()->regions.size(); ++ region_id) { |             for (size_t region_id = 0; region_id < object->print()->regions.size(); ++ region_id) { | ||||||
|                 unsigned int region_extruder = objects[i]->print()->regions[region_id]->config.infill_extruder - 1; // config value is 1-based
 |                 const auto& region = *object->print()->regions[region_id]; | ||||||
|                 if (config.filament_soluble.get_at(region_extruder)) // if this entity is meant to be soluble, keep it that way
 | 
 | ||||||
|  |                 if (!region.config.wipe_into_infill && !object->config.wipe_into_objects) | ||||||
|                     continue; |                     continue; | ||||||
| 
 | 
 | ||||||
|                 if (!config.infill_first) { // in this case we must verify that region_extruder was already used at this layer (and perimeters of the infill are therefore extruded)
 |  | ||||||
|                     bool unused_yet = false; |  | ||||||
|                     for (unsigned i = 0; i < layer_tools.extruders.size(); ++i) { |  | ||||||
|                         if (layer_tools.extruders[i] == new_extruder) |  | ||||||
|                             unused_yet = true; |  | ||||||
|                         if (layer_tools.extruders[i] == region_extruder) |  | ||||||
|                             break; |  | ||||||
|                     } |  | ||||||
|                     if (unused_yet) |  | ||||||
|                         continue; |  | ||||||
|                 } |  | ||||||
| 
 | 
 | ||||||
|                 if (objects[i]->config.wipe_into_infill) { |                 if (((!config.infill_first ? perimeters_done : !perimeters_done) || !object->config.wipe_into_objects) && region.config.wipe_into_infill) { | ||||||
|                     ExtrusionEntityCollection& eec = this_layer->regions[region_id]->fills; |                     ExtrusionEntityCollection& eec = this_layer->regions[region_id]->fills; | ||||||
|                     for (ExtrusionEntity* ee : eec.entities) {                      // iterate through all infill Collections
 |                     for (ExtrusionEntity* ee : eec.entities) {                      // iterate through all infill Collections
 | ||||||
|                         if (volume_to_wipe <= 0.f) |                         if (volume_to_wipe <= 0.f) | ||||||
|  | @ -1244,21 +1228,49 @@ float Print::mark_wiping_extrusions(ToolOrdering::LayerTools& layer_tools, unsig | ||||||
|                         auto* fill = dynamic_cast<ExtrusionEntityCollection*>(ee); |                         auto* fill = dynamic_cast<ExtrusionEntityCollection*>(ee); | ||||||
|                         if (fill->role() == erTopSolidInfill || fill->role() == erGapFill)  // these cannot be changed - such infill is / may be visible
 |                         if (fill->role() == erTopSolidInfill || fill->role() == erGapFill)  // these cannot be changed - such infill is / may be visible
 | ||||||
|                             continue; |                             continue; | ||||||
|                         if (/*!fill->is_extruder_overridden(copy)*/ !layer_tools.wiping_extrusions.is_entity_overridden(fill, copy) && fill->total_volume() > min_infill_volume) {     // this infill will be used to wipe this extruder
 | 
 | ||||||
|  |                         // What extruder would this normally be printed with?
 | ||||||
|  |                         unsigned int correct_extruder = get_extruder(fill, region); | ||||||
|  |                         if (config.filament_soluble.get_at(correct_extruder)) // if this entity is meant to be soluble, keep it that way
 | ||||||
|  |                             continue; | ||||||
|  | 
 | ||||||
|  |                         if (!object->config.wipe_into_objects && !config.infill_first)  { | ||||||
|  |                             // In this case we must check that the original extruder is used on this layer before the one we are overridding
 | ||||||
|  |                             // (and the perimeters will be finished before the infill is printed):
 | ||||||
|  |                             if (!config.infill_first && region.config.wipe_into_infill) { | ||||||
|  |                                 bool unused_yet = false; | ||||||
|  |                                 for (unsigned i = 0; i < layer_tools.extruders.size(); ++i) { | ||||||
|  |                                     if (layer_tools.extruders[i] == new_extruder) | ||||||
|  |                                         unused_yet = true; | ||||||
|  |                                     if (layer_tools.extruders[i] == correct_extruder) | ||||||
|  |                                         break; | ||||||
|  |                                 } | ||||||
|  |                                 if (unused_yet) | ||||||
|  |                                     continue; | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  | 
 | ||||||
|  |                         if (!layer_tools.wiping_extrusions.is_entity_overridden(fill, copy) && fill->total_volume() > min_infill_volume) {     // this infill will be used to wipe this extruder
 | ||||||
|                             layer_tools.wiping_extrusions.set_extruder_override(fill, copy, new_extruder, num_of_copies); |                             layer_tools.wiping_extrusions.set_extruder_override(fill, copy, new_extruder, num_of_copies); | ||||||
|                             volume_to_wipe -= fill->total_volume(); |                             volume_to_wipe -= fill->total_volume(); | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|                 if (objects[i]->config.wipe_into_objects) | 
 | ||||||
|  |                 if ((config.infill_first ? perimeters_done : !perimeters_done) && object->config.wipe_into_objects) | ||||||
|                 { |                 { | ||||||
|                     ExtrusionEntityCollection& eec = this_layer->regions[region_id]->perimeters; |                     ExtrusionEntityCollection& eec = this_layer->regions[region_id]->perimeters; | ||||||
|                     for (ExtrusionEntity* ee : eec.entities) {                      // iterate through all perimeter Collections
 |                     for (ExtrusionEntity* ee : eec.entities) {                      // iterate through all perimeter Collections
 | ||||||
|                         if (volume_to_wipe <= 0.f) |                         if (volume_to_wipe <= 0.f) | ||||||
|                             break; |                             break; | ||||||
|                         auto* fill = dynamic_cast<ExtrusionEntityCollection*>(ee); |                         auto* fill = dynamic_cast<ExtrusionEntityCollection*>(ee); | ||||||
|                         if (/*!fill->is_extruder_overridden(copy)*/ !layer_tools.wiping_extrusions.is_entity_overridden(fill, copy) && fill->total_volume() > min_infill_volume) { |                         // What extruder would this normally be printed with?
 | ||||||
|  |                         unsigned int correct_extruder = get_extruder(fill, region); | ||||||
|  |                         if (config.filament_soluble.get_at(correct_extruder)) // if this entity is meant to be soluble, keep it that way
 | ||||||
|  |                             continue; | ||||||
|  | 
 | ||||||
|  |                         if (!layer_tools.wiping_extrusions.is_entity_overridden(fill, copy) && fill->total_volume() > min_infill_volume) { | ||||||
|                             layer_tools.wiping_extrusions.set_extruder_override(fill, copy, new_extruder, num_of_copies); |                             layer_tools.wiping_extrusions.set_extruder_override(fill, copy, new_extruder, num_of_copies); | ||||||
|                             volume_to_wipe -= fill->total_volume(); |                             volume_to_wipe -= fill->total_volume(); | ||||||
|                         } |                         } | ||||||
|  |  | ||||||
|  | @ -24,6 +24,7 @@ class Print; | ||||||
| class PrintObject; | class PrintObject; | ||||||
| class ModelObject; | class ModelObject; | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
| // Print step IDs for keeping track of the print state.
 | // Print step IDs for keeping track of the print state.
 | ||||||
| enum PrintStep { | enum PrintStep { | ||||||
|     psSkirt, psBrim, psWipeTower, psCount, |     psSkirt, psBrim, psWipeTower, psCount, | ||||||
|  | @ -323,6 +324,15 @@ private: | ||||||
|     tbb::atomic<bool>   m_canceled; |     tbb::atomic<bool>   m_canceled; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | // Returns extruder this eec should be printed with, according to PrintRegion config
 | ||||||
|  | static int get_extruder(const ExtrusionEntityCollection* fill, const PrintRegion ®ion) { | ||||||
|  |     return is_infill(fill->role()) ? std::max<int>(0, (is_solid_infill(fill->entities.front()->role()) ? region.config.solid_infill_extruder : region.config.infill_extruder) - 1) : | ||||||
|  |                                      std::max<int>(region.config.perimeter_extruder.value - 1, 0); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| #define FOREACH_BASE(type, container, iterator) for (type::const_iterator iterator = (container).begin(); iterator != (container).end(); ++iterator) | #define FOREACH_BASE(type, container, iterator) for (type::const_iterator iterator = (container).begin(); iterator != (container).end(); ++iterator) | ||||||
| #define FOREACH_REGION(print, region)       FOREACH_BASE(PrintRegionPtrs, (print)->regions, region) | #define FOREACH_REGION(print, region)       FOREACH_BASE(PrintRegionPtrs, (print)->regions, region) | ||||||
| #define FOREACH_OBJECT(print, object)       FOREACH_BASE(PrintObjectPtrs, (print)->objects, object) | #define FOREACH_OBJECT(print, object)       FOREACH_BASE(PrintObjectPtrs, (print)->objects, object) | ||||||
|  |  | ||||||
|  | @ -337,7 +337,6 @@ public: | ||||||
|     ConfigOptionFloatOrPercent      support_material_xy_spacing; |     ConfigOptionFloatOrPercent      support_material_xy_spacing; | ||||||
|     ConfigOptionFloat               xy_size_compensation; |     ConfigOptionFloat               xy_size_compensation; | ||||||
|     ConfigOptionBool                wipe_into_objects; |     ConfigOptionBool                wipe_into_objects; | ||||||
|     ConfigOptionBool                wipe_into_infill; |  | ||||||
| 
 | 
 | ||||||
| protected: | protected: | ||||||
|     void initialize(StaticCacheBase &cache, const char *base_ptr) |     void initialize(StaticCacheBase &cache, const char *base_ptr) | ||||||
|  | @ -375,7 +374,6 @@ protected: | ||||||
|         OPT_PTR(support_material_with_sheath); |         OPT_PTR(support_material_with_sheath); | ||||||
|         OPT_PTR(xy_size_compensation); |         OPT_PTR(xy_size_compensation); | ||||||
|         OPT_PTR(wipe_into_objects); |         OPT_PTR(wipe_into_objects); | ||||||
|         OPT_PTR(wipe_into_infill); |  | ||||||
|     } |     } | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | @ -418,6 +416,7 @@ public: | ||||||
|     ConfigOptionFloatOrPercent      top_infill_extrusion_width; |     ConfigOptionFloatOrPercent      top_infill_extrusion_width; | ||||||
|     ConfigOptionInt                 top_solid_layers; |     ConfigOptionInt                 top_solid_layers; | ||||||
|     ConfigOptionFloatOrPercent      top_solid_infill_speed; |     ConfigOptionFloatOrPercent      top_solid_infill_speed; | ||||||
|  |     ConfigOptionBool                wipe_into_infill; | ||||||
|      |      | ||||||
| protected: | protected: | ||||||
|     void initialize(StaticCacheBase &cache, const char *base_ptr) |     void initialize(StaticCacheBase &cache, const char *base_ptr) | ||||||
|  | @ -456,6 +455,7 @@ protected: | ||||||
|         OPT_PTR(top_infill_extrusion_width); |         OPT_PTR(top_infill_extrusion_width); | ||||||
|         OPT_PTR(top_solid_infill_speed); |         OPT_PTR(top_solid_infill_speed); | ||||||
|         OPT_PTR(top_solid_layers); |         OPT_PTR(top_solid_layers); | ||||||
|  |         OPT_PTR(wipe_into_infill); | ||||||
|     } |     } | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Lukas Matena
						Lukas Matena