mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-22 00:01:09 -06:00 
			
		
		
		
	Overriddable infills that were not overridden are now printed according to infill_first
This commit is contained in:
		
							parent
							
								
									bb288f2a1b
								
							
						
					
					
						commit
						5bf795ec6f
					
				
					 4 changed files with 90 additions and 30 deletions
				
			
		|  | @ -1334,7 +1334,7 @@ void GCode::process_layer( | ||||||
|         if (objects_by_extruder_it == by_extruder.end()) |         if (objects_by_extruder_it == by_extruder.end()) | ||||||
|             continue; |             continue; | ||||||
| 
 | 
 | ||||||
|         // We are almost ready to print. However, we must go through all the object twice and only print the overridden extrusions first (infill/primeter wiping feature):
 |         // We are almost ready to print. However, we must go through all the object twice and only print the overridden extrusions first (infill/perimeter wiping feature):
 | ||||||
|         for (int print_wipe_extrusions=layer_tools.wiping_extrusions.is_anything_overridden(); print_wipe_extrusions>=0; --print_wipe_extrusions) { |         for (int print_wipe_extrusions=layer_tools.wiping_extrusions.is_anything_overridden(); print_wipe_extrusions>=0; --print_wipe_extrusions) { | ||||||
|             for (ObjectByExtruder &object_by_extruder : objects_by_extruder_it->second) { |             for (ObjectByExtruder &object_by_extruder : objects_by_extruder_it->second) { | ||||||
|                 const size_t       layer_id     = &object_by_extruder - objects_by_extruder_it->second.data(); |                 const size_t       layer_id     = &object_by_extruder - objects_by_extruder_it->second.data(); | ||||||
|  |  | ||||||
|  | @ -381,13 +381,24 @@ void WipingExtrusions::set_extruder_override(const ExtrusionEntity* entity, unsi | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | // Finds first non-soluble extruder on the layer
 | ||||||
|  | int WipingExtrusions::first_nonsoluble_extruder_on_layer(const PrintConfig& print_config, const LayerTools& lt) const | ||||||
|  | { | ||||||
|  |     for (auto extruders_it = lt.extruders.begin(); extruders_it != lt.extruders.end(); ++extruders_it) | ||||||
|  |         if (!print_config.filament_soluble.get_at(*extruders_it)) | ||||||
|  |             return (*extruders_it); | ||||||
|  | 
 | ||||||
|  |     return (-1); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // Finds last non-soluble extruder on the layer
 | // Finds last non-soluble extruder on the layer
 | ||||||
| bool WipingExtrusions::is_last_nonsoluble_on_layer(const PrintConfig& print_config, const LayerTools& lt, unsigned int extruder) const | int WipingExtrusions::last_nonsoluble_extruder_on_layer(const PrintConfig& print_config, const LayerTools& lt) const | ||||||
| { | { | ||||||
|     for (auto extruders_it = lt.extruders.rbegin(); extruders_it != lt.extruders.rend(); ++extruders_it) |     for (auto extruders_it = lt.extruders.rbegin(); extruders_it != lt.extruders.rend(); ++extruders_it) | ||||||
|         if (!print_config.filament_soluble.get_at(*extruders_it)) |         if (!print_config.filament_soluble.get_at(*extruders_it)) | ||||||
|             return (*extruders_it == extruder); |             return (*extruders_it); | ||||||
|     return false; | 
 | ||||||
|  |     return (-1); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -416,7 +427,7 @@ float WipingExtrusions::mark_wiping_extrusions(const Print& print, const LayerTo | ||||||
|     if (print.config.filament_soluble.get_at(new_extruder)) |     if (print.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
 | ||||||
| 
 | 
 | ||||||
|     bool last_nonsoluble = is_last_nonsoluble_on_layer(print.config, layer_tools, new_extruder); |     bool is_last_nonsoluble = ((int)new_extruder == last_nonsoluble_extruder_on_layer(print.config, layer_tools)); | ||||||
| 
 | 
 | ||||||
|     // we will sort objects so that dedicated for wiping are at the beginning:
 |     // we will sort objects so that dedicated for wiping are at the beginning:
 | ||||||
|     PrintObjectPtrs object_list = print.objects; |     PrintObjectPtrs object_list = print.objects; | ||||||
|  | @ -430,15 +441,15 @@ float WipingExtrusions::mark_wiping_extrusions(const Print& print, const LayerTo | ||||||
|     // this is controlled by the following variable:
 |     // this is controlled by the following variable:
 | ||||||
|     bool perimeters_done = false; |     bool perimeters_done = false; | ||||||
| 
 | 
 | ||||||
|     for (int i=0 ; i<(int)object_list.size() ; ++i) { |     for (int i=0 ; i<(int)object_list.size() + (perimeters_done ? 0 : 1); ++i) { | ||||||
|         const auto& object = object_list[i]; |         if (!perimeters_done && (i==(int)object_list.size() || !object_list[i]->config.wipe_into_objects)) { // we passed the last dedicated object in list
 | ||||||
| 
 |  | ||||||
|         if (!perimeters_done && (i+1==(int)object_list.size() || !object_list[i]->config.wipe_into_objects)) { // we passed the last dedicated object in list
 |  | ||||||
|             perimeters_done = true; |             perimeters_done = true; | ||||||
|             i=-1;   // let's go from the start again
 |             i=-1;   // let's go from the start again
 | ||||||
|             continue; |             continue; | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         const auto& object = object_list[i]; | ||||||
|  | 
 | ||||||
|         // Finds this layer:
 |         // 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; }); |         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()) |         if (this_layer_it == object->layers.end()) | ||||||
|  | @ -455,9 +466,8 @@ float WipingExtrusions::mark_wiping_extrusions(const Print& print, const LayerTo | ||||||
|                     continue; |                     continue; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|                 if (((!print.config.infill_first ? perimeters_done : !perimeters_done) || !object->config.wipe_into_objects) && region.config.wipe_into_infill) { |                 if ((!print.config.infill_first ? perimeters_done : !perimeters_done) || (!object->config.wipe_into_objects && region.config.wipe_into_infill)) { | ||||||
|                     const ExtrusionEntityCollection& eec = this_layer->regions[region_id]->fills; |                     for (const ExtrusionEntity* ee : this_layer->regions[region_id]->fills.entities) {                      // iterate through all infill Collections
 | ||||||
|                     for (const ExtrusionEntity* ee : eec.entities) {                      // iterate through all infill Collections
 |  | ||||||
|                         auto* fill = dynamic_cast<const ExtrusionEntityCollection*>(ee); |                         auto* fill = dynamic_cast<const ExtrusionEntityCollection*>(ee); | ||||||
| 
 | 
 | ||||||
|                         if (!is_overriddable(*fill, print.config, *object, region)) |                         if (!is_overriddable(*fill, print.config, *object, region)) | ||||||
|  | @ -466,15 +476,10 @@ float WipingExtrusions::mark_wiping_extrusions(const Print& print, const LayerTo | ||||||
|                         // What extruder would this normally be printed with?
 |                         // What extruder would this normally be printed with?
 | ||||||
|                         unsigned int correct_extruder = get_extruder(*fill, region); |                         unsigned int correct_extruder = get_extruder(*fill, region); | ||||||
| 
 | 
 | ||||||
|                         bool force_override = false; |                         if (volume_to_wipe<=0) | ||||||
|                         // If the extruder is not in layer tools - we MUST override it. This happens whenever all extrusions, that would normally
 |  | ||||||
|                         // be printed with this extruder on this layer are "dont care" (part of infill/perimeter wiping):
 |  | ||||||
|                         if (last_nonsoluble && std::find(layer_tools.extruders.begin(), layer_tools.extruders.end(), correct_extruder) == layer_tools.extruders.end()) |  | ||||||
|                             force_override = true; |  | ||||||
|                         if (!force_override && volume_to_wipe<=0) |  | ||||||
|                             continue; |                             continue; | ||||||
| 
 | 
 | ||||||
|                         if (!object->config.wipe_into_objects && !print.config.infill_first && !force_override) { |                         if (!object->config.wipe_into_objects && !print.config.infill_first) { | ||||||
|                             // In this case we must check that the original extruder is used on this layer before the one we are overridding
 |                             // 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):
 |                             // (and the perimeters will be finished before the infill is printed):
 | ||||||
|                             if ((!print.config.infill_first && region.config.wipe_into_infill)) { |                             if ((!print.config.infill_first && region.config.wipe_into_infill)) { | ||||||
|  | @ -490,7 +495,7 @@ float WipingExtrusions::mark_wiping_extrusions(const Print& print, const LayerTo | ||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
| 
 | 
 | ||||||
|                         if (force_override || (!is_entity_overridden(fill, copy) && fill->total_volume() > min_infill_volume)) {     // this infill will be used to wipe this extruder
 |                         if ((!is_entity_overridden(fill, copy) && fill->total_volume() > min_infill_volume)) {     // this infill will be used to wipe this extruder
 | ||||||
|                             set_extruder_override(fill, copy, new_extruder, num_of_copies); |                             set_extruder_override(fill, copy, new_extruder, num_of_copies); | ||||||
|                             volume_to_wipe -= fill->total_volume(); |                             volume_to_wipe -= fill->total_volume(); | ||||||
|                         } |                         } | ||||||
|  | @ -500,21 +505,15 @@ float WipingExtrusions::mark_wiping_extrusions(const Print& print, const LayerTo | ||||||
|                 // Now the same for perimeters - see comments above for explanation:
 |                 // Now the same for perimeters - see comments above for explanation:
 | ||||||
|                 if (object->config.wipe_into_objects && (print.config.infill_first ? perimeters_done : !perimeters_done)) |                 if (object->config.wipe_into_objects && (print.config.infill_first ? perimeters_done : !perimeters_done)) | ||||||
|                 { |                 { | ||||||
|                     const ExtrusionEntityCollection& eec = this_layer->regions[region_id]->perimeters; |                     for (const ExtrusionEntity* ee : this_layer->regions[region_id]->perimeters.entities) { | ||||||
|                     for (const ExtrusionEntity* ee : eec.entities) {                      // iterate through all perimeter Collections
 |  | ||||||
|                         auto* fill = dynamic_cast<const ExtrusionEntityCollection*>(ee); |                         auto* fill = dynamic_cast<const ExtrusionEntityCollection*>(ee); | ||||||
|                         if (!is_overriddable(*fill, print.config, *object, region)) |                         if (!is_overriddable(*fill, print.config, *object, region)) | ||||||
|                             continue; |                             continue; | ||||||
| 
 | 
 | ||||||
|                         // What extruder would this normally be printed with?
 |                         if (volume_to_wipe<=0) | ||||||
|                         unsigned int correct_extruder = get_extruder(*fill, region); |  | ||||||
|                         bool force_override = false; |  | ||||||
|                         if (last_nonsoluble && std::find(layer_tools.extruders.begin(), layer_tools.extruders.end(), correct_extruder) == layer_tools.extruders.end()) |  | ||||||
|                             force_override = true; |  | ||||||
|                         if (!force_override && volume_to_wipe<=0) |  | ||||||
|                             continue; |                             continue; | ||||||
| 
 | 
 | ||||||
|                         if (force_override || (!is_entity_overridden(fill, copy) && fill->total_volume() > min_infill_volume)) { |                         if ((!is_entity_overridden(fill, copy) && fill->total_volume() > min_infill_volume)) { | ||||||
|                             set_extruder_override(fill, copy, new_extruder, num_of_copies); |                             set_extruder_override(fill, copy, new_extruder, num_of_copies); | ||||||
|                             volume_to_wipe -= fill->total_volume(); |                             volume_to_wipe -= fill->total_volume(); | ||||||
|                         } |                         } | ||||||
|  | @ -528,6 +527,63 @@ float WipingExtrusions::mark_wiping_extrusions(const Print& print, const LayerTo | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | // Called after all toolchanges on a layer were mark_infill_overridden. There might still be overridable entities,
 | ||||||
|  | // that were not actually overridden. If they are part of a dedicated object, printing them with the extruder
 | ||||||
|  | // they were initially assigned to might mean violating the perimeter-infill order. We will therefore go through
 | ||||||
|  | // them again and make sure we override it.
 | ||||||
|  | void WipingExtrusions::ensure_perimeters_infills_order(const Print& print, const LayerTools& layer_tools) | ||||||
|  | { | ||||||
|  |     unsigned int first_nonsoluble_extruder = first_nonsoluble_extruder_on_layer(print.config, layer_tools); | ||||||
|  |     unsigned int last_nonsoluble_extruder = last_nonsoluble_extruder_on_layer(print.config, layer_tools); | ||||||
|  | 
 | ||||||
|  |     for (const PrintObject* object : print.objects) { | ||||||
|  |         // 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 (size_t region_id = 0; region_id < object->print()->regions.size(); ++ region_id) { | ||||||
|  |                 const auto& region = *object->print()->regions[region_id]; | ||||||
|  | 
 | ||||||
|  |                 if (!region.config.wipe_into_infill && !object->config.wipe_into_objects) | ||||||
|  |                     continue; | ||||||
|  | 
 | ||||||
|  |                 for (const ExtrusionEntity* ee : this_layer->regions[region_id]->fills.entities) {                      // iterate through all infill Collections
 | ||||||
|  |                     auto* fill = dynamic_cast<const ExtrusionEntityCollection*>(ee); | ||||||
|  | 
 | ||||||
|  |                     if (!is_overriddable(*fill, print.config, *object, region) | ||||||
|  |                      || is_entity_overridden(fill, copy) ) | ||||||
|  |                         continue; | ||||||
|  | 
 | ||||||
|  |                     // This infill could have been overridden but was not - unless we do somthing, it could be
 | ||||||
|  |                     // printed before its perimeter, or not be printed at all (in case its original extruder has
 | ||||||
|  |                     // not been added to LayerTools
 | ||||||
|  |                     // Either way, we will now force-override it with something suitable:
 | ||||||
|  |                     set_extruder_override(fill, copy, (print.config.infill_first ? first_nonsoluble_extruder : last_nonsoluble_extruder), num_of_copies); | ||||||
|  |                 } | ||||||
|  | 
 | ||||||
|  |                 // Now the same for perimeters - see comments above for explanation:
 | ||||||
|  |                 for (const ExtrusionEntity* ee : this_layer->regions[region_id]->perimeters.entities) {                      // iterate through all perimeter Collections
 | ||||||
|  |                     auto* fill = dynamic_cast<const ExtrusionEntityCollection*>(ee); | ||||||
|  |                     if (!is_overriddable(*fill, print.config, *object, region) | ||||||
|  |                      || is_entity_overridden(fill, copy) ) | ||||||
|  |                         continue; | ||||||
|  | 
 | ||||||
|  |                     set_extruder_override(fill, copy, (print.config.infill_first ? last_nonsoluble_extruder : first_nonsoluble_extruder), num_of_copies); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| // 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
 | ||||||
|  |  | ||||||
|  | @ -30,10 +30,13 @@ public: | ||||||
|     // marks them by the extruder id. Returns volume that remains to be wiped on the wipe tower:
 |     // marks them by the extruder id. Returns volume that remains to be wiped on the wipe tower:
 | ||||||
|     float mark_wiping_extrusions(const Print& print, const LayerTools& layer_tools, unsigned int new_extruder, float volume_to_wipe); |     float mark_wiping_extrusions(const Print& print, const LayerTools& layer_tools, unsigned int new_extruder, float volume_to_wipe); | ||||||
| 
 | 
 | ||||||
|  |     void ensure_perimeters_infills_order(const Print& print, const LayerTools& layer_tools); | ||||||
|  | 
 | ||||||
|     bool is_overriddable(const ExtrusionEntityCollection& ee, const PrintConfig& print_config, const PrintObject& object, const PrintRegion& region) const; |     bool is_overriddable(const ExtrusionEntityCollection& ee, const PrintConfig& print_config, const PrintObject& object, const PrintRegion& region) const; | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|     bool is_last_nonsoluble_on_layer(const PrintConfig& print_config, const LayerTools& lt, unsigned int extruder) const; |     int first_nonsoluble_extruder_on_layer(const PrintConfig& print_config, const LayerTools& lt) const; | ||||||
|  |     int last_nonsoluble_extruder_on_layer(const PrintConfig& print_config, const LayerTools& lt) const; | ||||||
| 
 | 
 | ||||||
|     // This function is called from mark_wiping_extrusions and sets extruder that it should be printed with (-1 .. as usual)
 |     // This function is called from mark_wiping_extrusions and sets extruder that it should be printed with (-1 .. as usual)
 | ||||||
|     void set_extruder_override(const ExtrusionEntity* entity, unsigned int copy_id, int extruder, unsigned int num_of_copies); |     void set_extruder_override(const ExtrusionEntity* entity, unsigned int copy_id, int extruder, unsigned int num_of_copies); | ||||||
|  |  | ||||||
|  | @ -1143,6 +1143,7 @@ void Print::_make_wipe_tower() | ||||||
|                     current_extruder_id = extruder_id; |                     current_extruder_id = extruder_id; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|  |             layer_tools.wiping_extrusions.ensure_perimeters_infills_order(*this, layer_tools); | ||||||
|             if (&layer_tools == &m_tool_ordering.back() || (&layer_tools + 1)->wipe_tower_partitions == 0) |             if (&layer_tools == &m_tool_ordering.back() || (&layer_tools + 1)->wipe_tower_partitions == 0) | ||||||
|                 break; |                 break; | ||||||
|         } |         } | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Lukas Matena
						Lukas Matena