mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-25 01:31:14 -06:00 
			
		
		
		
	Merge remote-tracking branch 'origin/ys_cp_improvements'
This commit is contained in:
		
						commit
						52c590f57a
					
				
					 14 changed files with 446 additions and 257 deletions
				
			
		|  | @ -1088,7 +1088,7 @@ namespace Slic3r { | ||||||
|                 return; |                 return; | ||||||
|             pt::ptree code_tree = main_tree.front().second; |             pt::ptree code_tree = main_tree.front().second; | ||||||
| 
 | 
 | ||||||
|             m_model->custom_gcode_per_print_z.clear(); |             m_model->custom_gcode_per_print_z.gcodes.clear(); | ||||||
| 
 | 
 | ||||||
|             for (const auto& code : code_tree) |             for (const auto& code : code_tree) | ||||||
|             { |             { | ||||||
|  | @ -1100,7 +1100,7 @@ namespace Slic3r { | ||||||
|                 int extruder        = tree.get<int>         ("<xmlattr>.extruder"   ); |                 int extruder        = tree.get<int>         ("<xmlattr>.extruder"   ); | ||||||
|                 std::string color   = tree.get<std::string> ("<xmlattr>.color"      ); |                 std::string color   = tree.get<std::string> ("<xmlattr>.color"      ); | ||||||
| 
 | 
 | ||||||
|                 m_model->custom_gcode_per_print_z.push_back(Model::CustomGCode{print_z, gcode, extruder, color}) ; |                 m_model->custom_gcode_per_print_z.gcodes.push_back(Model::CustomGCode{print_z, gcode, extruder, color}) ; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -2631,12 +2631,12 @@ bool _3MF_Exporter::_add_custom_gcode_per_print_z_file_to_archive( mz_zip_archiv | ||||||
| { | { | ||||||
|     std::string out = ""; |     std::string out = ""; | ||||||
| 
 | 
 | ||||||
|     if (!model.custom_gcode_per_print_z.empty()) |     if (!model.custom_gcode_per_print_z.gcodes.empty()) | ||||||
|     { |     { | ||||||
|         pt::ptree tree; |         pt::ptree tree; | ||||||
|         pt::ptree& main_tree = tree.add("custom_gcodes_per_print_z", ""); |         pt::ptree& main_tree = tree.add("custom_gcodes_per_print_z", ""); | ||||||
| 
 | 
 | ||||||
|         for (const Model::CustomGCode& code : model.custom_gcode_per_print_z) |         for (const Model::CustomGCode& code : model.custom_gcode_per_print_z.gcodes) | ||||||
|         { |         { | ||||||
|             pt::ptree& code_tree = main_tree.add("code", ""); |             pt::ptree& code_tree = main_tree.add("code", ""); | ||||||
|             // store minX and maxZ
 |             // store minX and maxZ
 | ||||||
|  |  | ||||||
|  | @ -653,7 +653,7 @@ void AMFParserContext::endElement(const char * /* name */) | ||||||
|         int extruder = atoi(m_value[2].c_str()); |         int extruder = atoi(m_value[2].c_str()); | ||||||
|         const std::string& color = m_value[3]; |         const std::string& color = m_value[3]; | ||||||
| 
 | 
 | ||||||
|         m_model.custom_gcode_per_print_z.push_back(Model::CustomGCode{height, gcode, extruder, color}); |         m_model.custom_gcode_per_print_z.gcodes.push_back(Model::CustomGCode{height, gcode, extruder, color}); | ||||||
| 
 | 
 | ||||||
|         for (std::string& val: m_value) |         for (std::string& val: m_value) | ||||||
|             val.clear(); |             val.clear(); | ||||||
|  | @ -1250,14 +1250,14 @@ bool store_amf(const char *path, Model *model, const DynamicPrintConfig *config) | ||||||
|         stream << "  </constellation>\n"; |         stream << "  </constellation>\n"; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (!model->custom_gcode_per_print_z.empty()) |     if (!model->custom_gcode_per_print_z.gcodes.empty()) | ||||||
|     { |     { | ||||||
|         std::string out = ""; |         std::string out = ""; | ||||||
|         pt::ptree tree; |         pt::ptree tree; | ||||||
| 
 | 
 | ||||||
|         pt::ptree& main_tree = tree.add("custom_gcodes_per_height", ""); |         pt::ptree& main_tree = tree.add("custom_gcodes_per_height", ""); | ||||||
| 
 | 
 | ||||||
|         for (const Model::CustomGCode& code : model->custom_gcode_per_print_z) |         for (const Model::CustomGCode& code : model->custom_gcode_per_print_z.gcodes) | ||||||
|         { |         { | ||||||
|             pt::ptree& code_tree = main_tree.add("code", ""); |             pt::ptree& code_tree = main_tree.add("code", ""); | ||||||
|             // store minX and maxZ
 |             // store minX and maxZ
 | ||||||
|  |  | ||||||
|  | @ -462,13 +462,13 @@ void ToolOrdering::assign_custom_gcodes(const Print &print) | ||||||
| 	// Only valid for non-sequential print.
 | 	// Only valid for non-sequential print.
 | ||||||
| 	assert(! print.config().complete_objects.value); | 	assert(! print.config().complete_objects.value); | ||||||
| 
 | 
 | ||||||
| 	const std::vector<Model::CustomGCode>	&custom_gcode_per_print_z = print.model().custom_gcode_per_print_z; | 	const Model::CustomGCodeInfo	&custom_gcode_per_print_z = print.model().custom_gcode_per_print_z; | ||||||
| 	if (custom_gcode_per_print_z.empty()) | 	if (custom_gcode_per_print_z.gcodes.empty()) | ||||||
| 		return; | 		return; | ||||||
| 
 | 
 | ||||||
| 	unsigned int 							 num_extruders = *std::max_element(m_all_printing_extruders.begin(), m_all_printing_extruders.end()) + 1; | 	unsigned int 							 num_extruders = *std::max_element(m_all_printing_extruders.begin(), m_all_printing_extruders.end()) + 1; | ||||||
| 	std::vector<unsigned char> 				 extruder_printing_above(num_extruders, false); | 	std::vector<unsigned char> 				 extruder_printing_above(num_extruders, false); | ||||||
| 	auto 									 custom_gcode_it = custom_gcode_per_print_z.rbegin(); | 	auto 									 custom_gcode_it = custom_gcode_per_print_z.gcodes.rbegin(); | ||||||
| 	// If printing on a single extruder machine, make the tool changes trigger color change (M600) events.
 | 	// If printing on a single extruder machine, make the tool changes trigger color change (M600) events.
 | ||||||
| 	bool 									 tool_changes_as_color_changes = num_extruders == 1; | 	bool 									 tool_changes_as_color_changes = num_extruders == 1; | ||||||
| 	// From the last layer to the first one:
 | 	// From the last layer to the first one:
 | ||||||
|  | @ -478,8 +478,8 @@ void ToolOrdering::assign_custom_gcodes(const Print &print) | ||||||
| 		for (unsigned int i : lt.extruders) | 		for (unsigned int i : lt.extruders) | ||||||
| 			extruder_printing_above[i] = true; | 			extruder_printing_above[i] = true; | ||||||
| 		// Skip all custom G-codes above this layer and skip all extruder switches.
 | 		// Skip all custom G-codes above this layer and skip all extruder switches.
 | ||||||
| 		for (; custom_gcode_it != custom_gcode_per_print_z.rend() && (custom_gcode_it->print_z > lt.print_z + EPSILON || custom_gcode_it->gcode == ExtruderChangeCode); ++ custom_gcode_it); | 		for (; custom_gcode_it != custom_gcode_per_print_z.gcodes.rend() && (custom_gcode_it->print_z > lt.print_z + EPSILON || custom_gcode_it->gcode == ExtruderChangeCode); ++ custom_gcode_it); | ||||||
| 		if (custom_gcode_it == custom_gcode_per_print_z.rend()) | 		if (custom_gcode_it == custom_gcode_per_print_z.gcodes.rend()) | ||||||
| 			// Custom G-codes were processed.
 | 			// Custom G-codes were processed.
 | ||||||
| 			break; | 			break; | ||||||
| 		// Some custom G-code is configured for this layer or a layer below.
 | 		// Some custom G-code is configured for this layer or a layer below.
 | ||||||
|  | @ -490,7 +490,8 @@ void ToolOrdering::assign_custom_gcodes(const Print &print) | ||||||
| 			print_z_below = it_lt_below->print_z; | 			print_z_below = it_lt_below->print_z; | ||||||
| 		if (custom_gcode.print_z > print_z_below + 0.5 * EPSILON) { | 		if (custom_gcode.print_z > print_z_below + 0.5 * EPSILON) { | ||||||
| 			// The custom G-code applies to the current layer.
 | 			// The custom G-code applies to the current layer.
 | ||||||
| 			if (tool_changes_as_color_changes || custom_gcode.gcode != ColorChangeCode || extruder_printing_above[unsigned(custom_gcode.extruder - 1)]) | 			if ( tool_changes_as_color_changes || custom_gcode.gcode != ColorChangeCode ||  | ||||||
|  |                 (custom_gcode.extruder <= num_extruders && extruder_printing_above[unsigned(custom_gcode.extruder - 1)])) | ||||||
| 				// If it is color change, it will actually be useful as the exturder above will print.
 | 				// If it is color change, it will actually be useful as the exturder above will print.
 | ||||||
|         		lt.custom_gcode = &custom_gcode; |         		lt.custom_gcode = &custom_gcode; | ||||||
| 			// Consume that custom G-code event.
 | 			// Consume that custom G-code event.
 | ||||||
|  |  | ||||||
|  | @ -126,7 +126,7 @@ Model Model::read_from_file(const std::string& input_file, DynamicPrintConfig* c | ||||||
|     if (add_default_instances) |     if (add_default_instances) | ||||||
|         model.add_default_instances(); |         model.add_default_instances(); | ||||||
| 
 | 
 | ||||||
|     update_custom_gcode_per_print_z_from_config(model.custom_gcode_per_print_z, config); |     update_custom_gcode_per_print_z_from_config(model.custom_gcode_per_print_z.gcodes, config); | ||||||
| 
 | 
 | ||||||
|     return model; |     return model; | ||||||
| } | } | ||||||
|  | @ -163,7 +163,7 @@ Model Model::read_from_archive(const std::string& input_file, DynamicPrintConfig | ||||||
|     if (add_default_instances) |     if (add_default_instances) | ||||||
|         model.add_default_instances(); |         model.add_default_instances(); | ||||||
| 
 | 
 | ||||||
|     update_custom_gcode_per_print_z_from_config(model.custom_gcode_per_print_z, config); |     update_custom_gcode_per_print_z_from_config(model.custom_gcode_per_print_z.gcodes, config); | ||||||
| 
 | 
 | ||||||
|     return model; |     return model; | ||||||
| } | } | ||||||
|  | @ -1846,7 +1846,7 @@ arrangement::ArrangePolygon ModelInstance::get_arrange_polygon() const | ||||||
| std::vector<std::pair<double, unsigned int>> custom_tool_changes(const Model &model, size_t num_extruders) | std::vector<std::pair<double, unsigned int>> custom_tool_changes(const Model &model, size_t num_extruders) | ||||||
| { | { | ||||||
|     std::vector<std::pair<double, unsigned int>> custom_tool_changes; |     std::vector<std::pair<double, unsigned int>> custom_tool_changes; | ||||||
|     for (const Model::CustomGCode &custom_gcode : model.custom_gcode_per_print_z) |     for (const Model::CustomGCode &custom_gcode : model.custom_gcode_per_print_z.gcodes) | ||||||
|         if (custom_gcode.gcode == ExtruderChangeCode) { |         if (custom_gcode.gcode == ExtruderChangeCode) { | ||||||
|             // If extruder count in PrinterSettings was changed, use default (0) extruder for extruders, more than num_extruders
 |             // If extruder count in PrinterSettings was changed, use default (0) extruder for extruders, more than num_extruders
 | ||||||
|             custom_tool_changes.emplace_back(custom_gcode.print_z, static_cast<unsigned int>(custom_gcode.extruder > num_extruders ? 1 : custom_gcode.extruder)); |             custom_tool_changes.emplace_back(custom_gcode.print_z, static_cast<unsigned int>(custom_gcode.extruder > num_extruders ? 1 : custom_gcode.extruder)); | ||||||
|  |  | ||||||
|  | @ -772,7 +772,28 @@ public: | ||||||
|         std::string color;      // if gcode is equal to PausePrintCode, 
 |         std::string color;      // if gcode is equal to PausePrintCode, 
 | ||||||
|                                 // this field is used for save a short message shown on Printer display 
 |                                 // this field is used for save a short message shown on Printer display 
 | ||||||
|     }; |     }; | ||||||
|     std::vector<CustomGCode> custom_gcode_per_print_z; |      | ||||||
|  |     struct CustomGCodeInfo | ||||||
|  |     { | ||||||
|  |         enum MODE | ||||||
|  |         { | ||||||
|  |             SingleExtruder,   // single extruder printer preset is selected
 | ||||||
|  |             MultiAsSingle,    // multiple extruder printer preset is selected, but 
 | ||||||
|  |                               // this mode works just for Single extruder print 
 | ||||||
|  |                               // (For all print from objects settings is used just one extruder) 
 | ||||||
|  |             MultiExtruder     // multiple extruder printer preset is selected
 | ||||||
|  |         } mode; | ||||||
|  |          | ||||||
|  |         std::vector<CustomGCode> gcodes; | ||||||
|  | 
 | ||||||
|  |         bool operator==(const CustomGCodeInfo& rhs) const | ||||||
|  |         { | ||||||
|  |             return  (rhs.mode   == this->mode   ) && | ||||||
|  |                     (rhs.gcodes == this->gcodes ); | ||||||
|  |         } | ||||||
|  |         bool operator!=(const CustomGCodeInfo& rhs) const { return !(*this == rhs); } | ||||||
|  |     }  | ||||||
|  |     custom_gcode_per_print_z; | ||||||
|      |      | ||||||
|     // Default constructor assigns a new ID to the model.
 |     // Default constructor assigns a new ID to the model.
 | ||||||
|     Model() { assert(this->id().valid()); } |     Model() { assert(this->id().valid()); } | ||||||
|  |  | ||||||
|  | @ -723,7 +723,7 @@ Print::ApplyStatus Print::apply(const Model &model, DynamicPrintConfig new_full_ | ||||||
| 			model_object_status.emplace(model_object->id(), ModelObjectStatus::New); | 			model_object_status.emplace(model_object->id(), ModelObjectStatus::New); | ||||||
|     } else { |     } else { | ||||||
|         if (m_model.custom_gcode_per_print_z != model.custom_gcode_per_print_z) { |         if (m_model.custom_gcode_per_print_z != model.custom_gcode_per_print_z) { | ||||||
|             update_apply_status(custom_per_printz_gcodes_tool_changes_differ(m_model.custom_gcode_per_print_z, model.custom_gcode_per_print_z) ? |             update_apply_status(custom_per_printz_gcodes_tool_changes_differ(m_model.custom_gcode_per_print_z.gcodes, model.custom_gcode_per_print_z.gcodes) ? | ||||||
|             	// The Tool Ordering and the Wipe Tower are no more valid.
 |             	// The Tool Ordering and the Wipe Tower are no more valid.
 | ||||||
|             	this->invalidate_steps({ psWipeTower, psGCodeExport }) : |             	this->invalidate_steps({ psWipeTower, psGCodeExport }) : | ||||||
|             	// There is no change in Tool Changes stored in custom_gcode_per_print_z, therefore there is no need to update Tool Ordering.
 |             	// There is no change in Tool Changes stored in custom_gcode_per_print_z, therefore there is no need to update Tool Ordering.
 | ||||||
|  |  | ||||||
|  | @ -401,6 +401,7 @@ public: | ||||||
| 
 | 
 | ||||||
|     // Accessed by SupportMaterial
 |     // Accessed by SupportMaterial
 | ||||||
|     const PrintRegion*  get_region(size_t idx) const  { return m_regions[idx]; } |     const PrintRegion*  get_region(size_t idx) const  { return m_regions[idx]; } | ||||||
|  |     const ToolOrdering& get_tool_ordering() const { return m_wipe_tower_data.tool_ordering; }   // #ys_FIXME just for testing
 | ||||||
| 
 | 
 | ||||||
| protected: | protected: | ||||||
|     // methods for handling regions
 |     // methods for handling regions
 | ||||||
|  |  | ||||||
|  | @ -1011,7 +1011,7 @@ void GLCanvas3D::LegendTexture::fill_color_print_legend_items(  const GLCanvas3D | ||||||
|                                                                 std::vector<float>& colors, |                                                                 std::vector<float>& colors, | ||||||
|                                                                 std::vector<std::string>& cp_legend_items) |                                                                 std::vector<std::string>& cp_legend_items) | ||||||
| { | { | ||||||
|     std::vector<Model::CustomGCode> custom_gcode_per_print_z = wxGetApp().plater()->model().custom_gcode_per_print_z; |     std::vector<Model::CustomGCode> custom_gcode_per_print_z = wxGetApp().plater()->model().custom_gcode_per_print_z.gcodes; | ||||||
| 
 | 
 | ||||||
|     const int extruders_cnt = wxGetApp().extruders_edited_cnt(); |     const int extruders_cnt = wxGetApp().extruders_edited_cnt(); | ||||||
|     if (extruders_cnt == 1)  |     if (extruders_cnt == 1)  | ||||||
|  |  | ||||||
|  | @ -566,7 +566,7 @@ void Preview::update_view_type(bool slice_completed) | ||||||
| { | { | ||||||
|     const DynamicPrintConfig& config = wxGetApp().preset_bundle->project_config; |     const DynamicPrintConfig& config = wxGetApp().preset_bundle->project_config; | ||||||
| 
 | 
 | ||||||
|     const wxString& choice = !wxGetApp().plater()->model().custom_gcode_per_print_z.empty() /*&&
 |     const wxString& choice = !wxGetApp().plater()->model().custom_gcode_per_print_z.gcodes.empty() /*&&
 | ||||||
|                              (wxGetApp().extruders_edited_cnt()==1 || !slice_completed) */?  |                              (wxGetApp().extruders_edited_cnt()==1 || !slice_completed) */?  | ||||||
|                                 _(L("Color Print")) : |                                 _(L("Color Print")) : | ||||||
|                                 config.option<ConfigOptionFloats>("wiping_volumes_matrix")->values.size() > 1 ? |                                 config.option<ConfigOptionFloats>("wiping_volumes_matrix")->values.size() > 1 ? | ||||||
|  | @ -597,7 +597,7 @@ void Preview::create_double_slider() | ||||||
| 
 | 
 | ||||||
|     Bind(wxCUSTOMEVT_TICKSCHANGED, [this](wxEvent&) { |     Bind(wxCUSTOMEVT_TICKSCHANGED, [this](wxEvent&) { | ||||||
|         Model& model = wxGetApp().plater()->model(); |         Model& model = wxGetApp().plater()->model(); | ||||||
|         model.custom_gcode_per_print_z = m_slider->GetTicksValues(); |         model.custom_gcode_per_print_z.gcodes = m_slider->GetTicksValues(); | ||||||
|         m_schedule_background_process(); |         m_schedule_background_process(); | ||||||
| 
 | 
 | ||||||
|         update_view_type(false); |         update_view_type(false); | ||||||
|  | @ -666,7 +666,7 @@ void Preview::update_double_slider(const std::vector<double>& layers_z, bool kee | ||||||
|     bool   snap_to_min = force_sliders_full_range || m_slider->is_lower_at_min(); |     bool   snap_to_min = force_sliders_full_range || m_slider->is_lower_at_min(); | ||||||
| 	bool   snap_to_max  = force_sliders_full_range || m_slider->is_higher_at_max(); | 	bool   snap_to_max  = force_sliders_full_range || m_slider->is_higher_at_max(); | ||||||
| 
 | 
 | ||||||
|     std::vector<Model::CustomGCode> &ticks_from_model = wxGetApp().plater()->model().custom_gcode_per_print_z; |     std::vector<Model::CustomGCode> &ticks_from_model = wxGetApp().plater()->model().custom_gcode_per_print_z.gcodes; | ||||||
|     check_slider_values(ticks_from_model, layers_z); |     check_slider_values(ticks_from_model, layers_z); | ||||||
| 
 | 
 | ||||||
|     m_slider->SetSliderValues(layers_z); |     m_slider->SetSliderValues(layers_z); | ||||||
|  | @ -694,7 +694,60 @@ void Preview::update_double_slider(const std::vector<double>& layers_z, bool kee | ||||||
|     bool color_print_enable = (wxGetApp().plater()->printer_technology() == ptFFF); |     bool color_print_enable = (wxGetApp().plater()->printer_technology() == ptFFF); | ||||||
| 
 | 
 | ||||||
|     m_slider->EnableTickManipulation(color_print_enable); |     m_slider->EnableTickManipulation(color_print_enable); | ||||||
|     m_slider->SetManipulationState(wxGetApp().extruders_edited_cnt()); |     // Detect and set manipulation mode for double slider
 | ||||||
|  |     update_double_slider_mode(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void Preview::update_double_slider_mode() | ||||||
|  | { | ||||||
|  |     //    true  -> single-extruder printer profile OR 
 | ||||||
|  |     //             multi-extruder printer profile , but whole model is printed by only one extruder
 | ||||||
|  |     //    false -> multi-extruder printer profile , and model is printed by several extruders
 | ||||||
|  |     bool    one_extruder_printed_model = true; | ||||||
|  | 
 | ||||||
|  |     // extruder used for whole model for multi-extruder printer profile
 | ||||||
|  |     int     only_extruder = -1;  | ||||||
|  | 
 | ||||||
|  |     if (wxGetApp().extruders_edited_cnt() > 1) | ||||||
|  |     { | ||||||
|  |         const ModelObjectPtrs& objects = wxGetApp().plater()->model().objects; | ||||||
|  | 
 | ||||||
|  |         // check if whole model uses just only one extruder
 | ||||||
|  |         if (!objects.empty()) | ||||||
|  |         { | ||||||
|  |             const int extruder = objects[0]->config.has("extruder") ? | ||||||
|  |                                  objects[0]->config.option("extruder")->getInt() : 0; | ||||||
|  | 
 | ||||||
|  |             auto is_one_extruder_printed_model = [objects, extruder]() | ||||||
|  |             { | ||||||
|  |                 for (ModelObject* object : objects) | ||||||
|  |                 { | ||||||
|  |                     if (object->config.has("extruder") && | ||||||
|  |                         object->config.option("extruder")->getInt() != extruder) | ||||||
|  |                         return false; | ||||||
|  | 
 | ||||||
|  |                     if (object->volumes.size() > 1) | ||||||
|  |                         for (ModelVolume* volume : object->volumes) | ||||||
|  |                             if (volume->config.has("extruder") && | ||||||
|  |                                 volume->config.option("extruder")->getInt() != extruder) | ||||||
|  |                                 return false; | ||||||
|  | 
 | ||||||
|  |                     for (const auto& range : object->layer_config_ranges) | ||||||
|  |                         if (range.second.has("extruder") && | ||||||
|  |                             range.second.option("extruder")->getInt() != extruder) | ||||||
|  |                             return false; | ||||||
|  |                 } | ||||||
|  |                 return true; | ||||||
|  |             }; | ||||||
|  | 
 | ||||||
|  |             if (is_one_extruder_printed_model()) | ||||||
|  |                 only_extruder = extruder; | ||||||
|  |             else | ||||||
|  |                 one_extruder_printed_model = false; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     m_slider->SetModeAndOnlyExtruder(one_extruder_printed_model, only_extruder); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void Preview::reset_double_slider() | void Preview::reset_double_slider() | ||||||
|  | @ -786,7 +839,7 @@ void Preview::load_print_as_fff(bool keep_z_range) | ||||||
|         colors.push_back("#808080"); // gray color for pause print or custom G-code 
 |         colors.push_back("#808080"); // gray color for pause print or custom G-code 
 | ||||||
| 
 | 
 | ||||||
|         if (!gcode_preview_data_valid) |         if (!gcode_preview_data_valid) | ||||||
|             color_print_values = wxGetApp().plater()->model().custom_gcode_per_print_z; |             color_print_values = wxGetApp().plater()->model().custom_gcode_per_print_z.gcodes; | ||||||
|     } |     } | ||||||
|     else if (gcode_preview_data_valid || (m_gcode_preview_data->extrusion.view_type == GCodePreviewData::Extrusion::Tool) ) |     else if (gcode_preview_data_valid || (m_gcode_preview_data->extrusion.view_type == GCodePreviewData::Extrusion::Tool) ) | ||||||
|     { |     { | ||||||
|  |  | ||||||
|  | @ -157,8 +157,9 @@ private: | ||||||
|     void create_double_slider(); |     void create_double_slider(); | ||||||
|     void check_slider_values(std::vector<Model::CustomGCode> &ticks_from_model, |     void check_slider_values(std::vector<Model::CustomGCode> &ticks_from_model, | ||||||
|                              const std::vector<double> &layers_z); |                              const std::vector<double> &layers_z); | ||||||
|     void update_double_slider(const std::vector<double>& layers_z, bool keep_z_range = false); |  | ||||||
|     void reset_double_slider(); |     void reset_double_slider(); | ||||||
|  |     void update_double_slider(const std::vector<double>& layers_z, bool keep_z_range = false); | ||||||
|  |     void update_double_slider_mode(); | ||||||
|     // update DoubleSlider after keyDown in canvas
 |     // update DoubleSlider after keyDown in canvas
 | ||||||
|     void update_double_slider_from_canvas(wxKeyEvent& event); |     void update_double_slider_from_canvas(wxKeyEvent& event); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -2791,7 +2791,7 @@ void Plater::priv::reset() | ||||||
|     // The hiding of the slicing results, if shown, is not taken care by the background process, so we do it here
 |     // The hiding of the slicing results, if shown, is not taken care by the background process, so we do it here
 | ||||||
|     this->sidebar->show_sliced_info_sizer(false); |     this->sidebar->show_sliced_info_sizer(false); | ||||||
| 
 | 
 | ||||||
|     model.custom_gcode_per_print_z.clear(); |     model.custom_gcode_per_print_z.gcodes.clear(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void Plater::priv::mirror(Axis axis) | void Plater::priv::mirror(Axis axis) | ||||||
|  | @ -5349,9 +5349,9 @@ std::vector<std::string> Plater::get_extruder_colors_from_plater_config() const | ||||||
| std::vector<std::string> Plater::get_colors_for_color_print() const | std::vector<std::string> Plater::get_colors_for_color_print() const | ||||||
| { | { | ||||||
|     std::vector<std::string> colors = get_extruder_colors_from_plater_config(); |     std::vector<std::string> colors = get_extruder_colors_from_plater_config(); | ||||||
|     colors.reserve(colors.size() + p->model.custom_gcode_per_print_z.size()); |     colors.reserve(colors.size() + p->model.custom_gcode_per_print_z.gcodes.size()); | ||||||
| 
 | 
 | ||||||
|     for (const Model::CustomGCode& code : p->model.custom_gcode_per_print_z) |     for (const Model::CustomGCode& code : p->model.custom_gcode_per_print_z.gcodes) | ||||||
|         if (code.gcode == ColorChangeCode) |         if (code.gcode == ColorChangeCode) | ||||||
|             colors.emplace_back(code.color); |             colors.emplace_back(code.color); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -877,7 +877,7 @@ void PresetBundle::load_config_file_config(const std::string &name_or_path, bool | ||||||
|         // 4) Load the project config values (the per extruder wipe matrix etc).
 |         // 4) Load the project config values (the per extruder wipe matrix etc).
 | ||||||
|         this->project_config.apply_only(config, s_project_options); |         this->project_config.apply_only(config, s_project_options); | ||||||
| 
 | 
 | ||||||
|         update_custom_gcode_per_print_z_from_config(GUI::wxGetApp().plater()->model().custom_gcode_per_print_z, &this->project_config); |         update_custom_gcode_per_print_z_from_config(GUI::wxGetApp().plater()->model().custom_gcode_per_print_z.gcodes, &this->project_config); | ||||||
| 
 | 
 | ||||||
|         break; |         break; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -5,6 +5,7 @@ | ||||||
| 
 | 
 | ||||||
| #include "libslic3r/Utils.hpp" | #include "libslic3r/Utils.hpp" | ||||||
| #include "libslic3r/Model.hpp" | #include "libslic3r/Model.hpp" | ||||||
|  | #include "libslic3r/Print.hpp" | ||||||
| 
 | 
 | ||||||
| #include <wx/sizer.h> | #include <wx/sizer.h> | ||||||
| #include <wx/bmpcbox.h> | #include <wx/bmpcbox.h> | ||||||
|  | @ -2537,7 +2538,7 @@ std::vector<t_custom_code> DoubleSlider::GetTicksValues() const | ||||||
| 
 | 
 | ||||||
|     const int val_size = m_values.size(); |     const int val_size = m_values.size(); | ||||||
|     if (!m_values.empty()) |     if (!m_values.empty()) | ||||||
|         for (const TICK_CODE& tick : m_ticks_) { |         for (const TICK_CODE& tick : m_ticks) { | ||||||
|             if (tick.tick > val_size) |             if (tick.tick > val_size) | ||||||
|                 break; |                 break; | ||||||
|             values.emplace_back(t_custom_code{m_values[tick.tick], tick.gcode, tick.extruder, tick.color}); |             values.emplace_back(t_custom_code{m_values[tick.tick], tick.gcode, tick.extruder, tick.color}); | ||||||
|  | @ -2551,19 +2552,19 @@ void DoubleSlider::SetTicksValues(const std::vector<t_custom_code>& heights) | ||||||
|     if (m_values.empty()) |     if (m_values.empty()) | ||||||
|         return; |         return; | ||||||
| 
 | 
 | ||||||
|     const bool was_empty = m_ticks_.empty(); |     const bool was_empty = m_ticks.empty(); | ||||||
| 
 | 
 | ||||||
|     m_ticks_.clear(); |     m_ticks.clear(); | ||||||
|     for (auto h : heights) { |     for (auto h : heights) { | ||||||
|         auto it = std::lower_bound(m_values.begin(), m_values.end(), h.print_z - epsilon()); |         auto it = std::lower_bound(m_values.begin(), m_values.end(), h.print_z - epsilon()); | ||||||
| 
 | 
 | ||||||
|         if (it == m_values.end()) |         if (it == m_values.end()) | ||||||
|             continue; |             continue; | ||||||
| 
 | 
 | ||||||
|         m_ticks_.emplace(TICK_CODE{int(it-m_values.begin()), h.gcode, h.extruder, h.color}); |         m_ticks.emplace(TICK_CODE{int(it-m_values.begin()), h.gcode, h.extruder, h.color}); | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     if (!was_empty && m_ticks_.empty()) |     if (!was_empty && m_ticks.empty()) | ||||||
|         // Switch to the "Feature type"/"Tool" from the very beginning of a new object slicing after deleting of the old one
 |         // Switch to the "Feature type"/"Tool" from the very beginning of a new object slicing after deleting of the old one
 | ||||||
|         wxPostEvent(this->GetParent(), wxCommandEvent(wxCUSTOMEVT_TICKSCHANGED)); |         wxPostEvent(this->GetParent(), wxCommandEvent(wxCUSTOMEVT_TICKSCHANGED)); | ||||||
| 
 | 
 | ||||||
|  | @ -2614,11 +2615,6 @@ void DoubleSlider::render() | ||||||
|     // draw line
 |     // draw line
 | ||||||
|     draw_scroll_line(dc, lower_pos, higher_pos); |     draw_scroll_line(dc, lower_pos, higher_pos); | ||||||
| 
 | 
 | ||||||
| //     //lower slider:
 |  | ||||||
| //     draw_thumb(dc, lower_pos, ssLower);
 |  | ||||||
| //     //higher slider:
 |  | ||||||
| //     draw_thumb(dc, higher_pos, ssHigher);
 |  | ||||||
| 
 |  | ||||||
|     //draw color print ticks
 |     //draw color print ticks
 | ||||||
|     draw_ticks(dc); |     draw_ticks(dc); | ||||||
| 
 | 
 | ||||||
|  | @ -2644,7 +2640,7 @@ void DoubleSlider::draw_action_icon(wxDC& dc, const wxPoint pt_beg, const wxPoin | ||||||
|         return; |         return; | ||||||
| 
 | 
 | ||||||
|     wxBitmap* icon = m_is_action_icon_focesed ? &m_bmp_add_tick_off.bmp() : &m_bmp_add_tick_on.bmp(); |     wxBitmap* icon = m_is_action_icon_focesed ? &m_bmp_add_tick_off.bmp() : &m_bmp_add_tick_on.bmp(); | ||||||
|     if (m_ticks_.find(TICK_CODE{tick}) != m_ticks_.end()) |     if (m_ticks.find(TICK_CODE{tick}) != m_ticks.end()) | ||||||
|         icon = m_is_action_icon_focesed ? &m_bmp_del_tick_off.bmp() : &m_bmp_del_tick_on.bmp(); |         icon = m_is_action_icon_focesed ? &m_bmp_del_tick_off.bmp() : &m_bmp_del_tick_on.bmp(); | ||||||
| 
 | 
 | ||||||
|     wxCoord x_draw, y_draw; |     wxCoord x_draw, y_draw; | ||||||
|  | @ -2787,7 +2783,7 @@ void DoubleSlider::draw_ticks(wxDC& dc) | ||||||
|     int height, width; |     int height, width; | ||||||
|     get_size(&width, &height); |     get_size(&width, &height); | ||||||
|     const wxCoord mid = is_horizontal() ? 0.5*height : 0.5*width; |     const wxCoord mid = is_horizontal() ? 0.5*height : 0.5*width; | ||||||
|     for (auto tick : m_ticks_) |     for (auto tick : m_ticks) | ||||||
|     { |     { | ||||||
|         const wxCoord pos = get_position_from_value(tick.tick); |         const wxCoord pos = get_position_from_value(tick.tick); | ||||||
| 
 | 
 | ||||||
|  | @ -2810,6 +2806,40 @@ void DoubleSlider::draw_ticks(wxDC& dc) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | std::string DoubleSlider::get_color_for_tool_change_tick(std::set<TICK_CODE>::const_iterator it) const | ||||||
|  | { | ||||||
|  |     const int current_extruder = it->extruder == 0 ? std::max<int>(m_only_extruder, 1) : it->extruder; | ||||||
|  | 
 | ||||||
|  |     auto it_n = it; | ||||||
|  |     while (it_n != m_ticks.begin()) { | ||||||
|  |         --it_n; | ||||||
|  |         if (it_n->gcode == Slic3r::ColorChangeCode && it_n->extruder == current_extruder) | ||||||
|  |             return it_n->color; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return it->color; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | std::string DoubleSlider::get_color_for_color_change_tick(std::set<TICK_CODE>::const_iterator it) const | ||||||
|  | { | ||||||
|  |     const int def_extruder = std::max<int>(1, m_only_extruder); | ||||||
|  |     auto it_n = it; | ||||||
|  |     bool is_tool_change = false; | ||||||
|  |     while (it_n != m_ticks.begin()) { | ||||||
|  |         --it_n; | ||||||
|  |         if (it_n->gcode == Slic3r::ExtruderChangeCode) { | ||||||
|  |             is_tool_change = true; | ||||||
|  |             if (it_n->extruder == it->extruder) | ||||||
|  |                 return it->color; | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     if (!is_tool_change && it->extruder == def_extruder) | ||||||
|  |         return it->color; | ||||||
|  | 
 | ||||||
|  |     return ""; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void DoubleSlider::draw_colored_band(wxDC& dc) | void DoubleSlider::draw_colored_band(wxDC& dc) | ||||||
| { | { | ||||||
|     if (!m_is_enabled_tick_manipulation) |     if (!m_is_enabled_tick_manipulation) | ||||||
|  | @ -2832,29 +2862,36 @@ void DoubleSlider::draw_colored_band(wxDC& dc) | ||||||
|         dc.DrawRectangle(band_rc); |         dc.DrawRectangle(band_rc); | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     const std::vector<std::string>& colors = Slic3r::GUI::wxGetApp().plater()->get_extruder_colors_from_plater_config(); |     // don't color a band for MultiExtruder mode
 | ||||||
|     int colors_cnt = colors.size(); |     if (m_ticks.empty() || m_mode == mmMultiExtruder) | ||||||
| 
 |  | ||||||
|     const wxColour bg_clr = GetParent()->GetBackgroundColour(); |  | ||||||
| 
 |  | ||||||
|     wxColour clr = m_state == msSingleExtruder ? wxColour(colors[0]) : bg_clr; |  | ||||||
|     draw_band(dc, clr, main_band); |  | ||||||
| 
 |  | ||||||
|     size_t i = 1; |  | ||||||
|     for (auto tick : m_ticks_) |  | ||||||
|     { |     { | ||||||
|         if ( (m_state == msSingleExtruder && tick.gcode != Slic3r::ColorChangeCode) || |         draw_band(dc, GetParent()->GetBackgroundColour(), main_band); | ||||||
|              (m_state == msMultiExtruder && tick.gcode != Slic3r::ExtruderChangeCode) ) |         return; | ||||||
|             continue; |     } | ||||||
| 
 | 
 | ||||||
|         const wxCoord pos = get_position_from_value(tick.tick); |     const int default_color_idx = m_mode==mmMultiAsSingle ? std::max<int>(m_only_extruder - 1, 0) : 0; | ||||||
|         is_horizontal() ? main_band.SetLeft(SLIDER_MARGIN + pos) : |     draw_band(dc, wxColour(Slic3r::GUI::wxGetApp().plater()->get_extruder_colors_from_plater_config()[default_color_idx]), main_band); | ||||||
|             main_band.SetBottom(pos - 1); |  | ||||||
| 
 | 
 | ||||||
|         clr = (m_state == msMultiExtruder && tick.color.empty()) ? bg_clr : |     std::set<TICK_CODE>::const_iterator tick_it = m_ticks.begin(); | ||||||
|                m_state == msMultiExtruder ? wxColour(colors[std::min<int>(colors_cnt - 1, tick.extruder-1)]) : wxColour(tick.color); | 
 | ||||||
|         draw_band(dc, clr, main_band); |     while (tick_it != m_ticks.end()) | ||||||
|         i++; |     { | ||||||
|  |         if ( (m_mode == mmSingleExtruder &&  tick_it->gcode == Slic3r::ColorChangeCode  ) || | ||||||
|  |              (m_mode == mmMultiAsSingle  && (tick_it->gcode == Slic3r::ExtruderChangeCode || tick_it->gcode == Slic3r::ColorChangeCode)) )  | ||||||
|  |         {         | ||||||
|  |             const wxCoord pos = get_position_from_value(tick_it->tick); | ||||||
|  |             is_horizontal() ? main_band.SetLeft(SLIDER_MARGIN + pos) : | ||||||
|  |                               main_band.SetBottom(pos - 1); | ||||||
|  | 
 | ||||||
|  |             const std::string clr_str = m_mode == mmSingleExtruder ? tick_it->color : | ||||||
|  |                                         tick_it->gcode == Slic3r::ExtruderChangeCode ? | ||||||
|  |                                         get_color_for_tool_change_tick(tick_it) : | ||||||
|  |                                         get_color_for_color_change_tick(tick_it); | ||||||
|  | 
 | ||||||
|  |             if (!clr_str.empty()) | ||||||
|  |                 draw_band(dc, wxColour(clr_str), main_band); | ||||||
|  |         } | ||||||
|  |         ++tick_it; | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -2879,7 +2916,7 @@ void DoubleSlider::draw_one_layer_icon(wxDC& dc) | ||||||
| 
 | 
 | ||||||
| void DoubleSlider::draw_revert_icon(wxDC& dc) | void DoubleSlider::draw_revert_icon(wxDC& dc) | ||||||
| { | { | ||||||
|     if (m_ticks_.empty() || !m_is_enabled_tick_manipulation) |     if (m_ticks.empty() || !m_is_enabled_tick_manipulation) | ||||||
|         return; |         return; | ||||||
| 
 | 
 | ||||||
|     int width, height; |     int width, height; | ||||||
|  | @ -2897,7 +2934,7 @@ void DoubleSlider::draw_revert_icon(wxDC& dc) | ||||||
| 
 | 
 | ||||||
| void DoubleSlider::draw_cog_icon(wxDC& dc) | void DoubleSlider::draw_cog_icon(wxDC& dc) | ||||||
| { | { | ||||||
|     if (m_state != msMultiExtruder) |     if (m_mode != mmMultiExtruder) | ||||||
|         return; |         return; | ||||||
| 
 | 
 | ||||||
|     int width, height; |     int width, height; | ||||||
|  | @ -2941,15 +2978,13 @@ void DoubleSlider::detect_selected_slider(const wxPoint& pt) | ||||||
| 
 | 
 | ||||||
| bool DoubleSlider::is_point_in_rect(const wxPoint& pt, const wxRect& rect) | bool DoubleSlider::is_point_in_rect(const wxPoint& pt, const wxRect& rect) | ||||||
| { | { | ||||||
|     if (rect.GetLeft() <= pt.x && pt.x <= rect.GetRight() &&  |     return  rect.GetLeft() <= pt.x && pt.x <= rect.GetRight() &&  | ||||||
|         rect.GetTop()  <= pt.y && pt.y <= rect.GetBottom()) |             rect.GetTop()  <= pt.y && pt.y <= rect.GetBottom(); | ||||||
|         return true; |  | ||||||
|     return false; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int DoubleSlider::is_point_near_tick(const wxPoint& pt) | int DoubleSlider::is_point_near_tick(const wxPoint& pt) | ||||||
| { | { | ||||||
|     for (auto tick : m_ticks_) { |     for (auto tick : m_ticks) { | ||||||
|         const wxCoord pos = get_position_from_value(tick.tick); |         const wxCoord pos = get_position_from_value(tick.tick); | ||||||
| 
 | 
 | ||||||
|         if (is_horizontal()) { |         if (is_horizontal()) { | ||||||
|  | @ -3010,10 +3045,10 @@ void DoubleSlider::OnLeftDown(wxMouseEvent& event) | ||||||
|         m_selection == ssLower ? correct_lower_value() : correct_higher_value(); |         m_selection == ssLower ? correct_lower_value() : correct_higher_value(); | ||||||
|         if (!m_selection) m_selection = ssHigher; |         if (!m_selection) m_selection = ssHigher; | ||||||
| 
 | 
 | ||||||
|         m_ticks_.clear(); |         m_ticks.clear(); | ||||||
|         wxPostEvent(this->GetParent(), wxCommandEvent(wxCUSTOMEVT_TICKSCHANGED)); |         wxPostEvent(this->GetParent(), wxCommandEvent(wxCUSTOMEVT_TICKSCHANGED)); | ||||||
|     } |     } | ||||||
|     else if (is_point_in_rect(pos, m_rect_cog_icon) && m_state == msMultiExtruder) { |     else if (is_point_in_rect(pos, m_rect_cog_icon) && m_mode == mmMultiExtruder) { | ||||||
|         // show dialog for set extruder sequence
 |         // show dialog for set extruder sequence
 | ||||||
|         m_edit_extruder_sequence = true; |         m_edit_extruder_sequence = true; | ||||||
|     } |     } | ||||||
|  | @ -3083,17 +3118,18 @@ wxString DoubleSlider::get_tooltip(IconFocus icon_focus) | ||||||
|     else if (m_is_action_icon_focesed) |     else if (m_is_action_icon_focesed) | ||||||
|     { |     { | ||||||
|         const int tick = m_selection == ssLower ? m_lower_value : m_higher_value; |         const int tick = m_selection == ssLower ? m_lower_value : m_higher_value; | ||||||
|         const auto tick_code_it = m_ticks_.find(TICK_CODE{tick}); |         const auto tick_code_it = m_ticks.find(TICK_CODE{tick}); | ||||||
|         tooltip = tick_code_it == m_ticks_.end()            ? (m_state == msSingleExtruder ? |         tooltip = tick_code_it == m_ticks.end()                    ? (m_mode == mmMultiAsSingle ? | ||||||
|                         _(L("For add color change use left mouse button click")) : |                       _(L("For add change extruder use left mouse button click")) : | ||||||
|                         _(L("For add change extruder use left mouse button click"))) + "\n" + |                       _(L("For add color change use left mouse button click"))  ) + "\n" + | ||||||
|                         _(L("For add another code use right mouse button click")) : |                       _(L("For add another code use right mouse button click"))   : | ||||||
|                   tick_code_it->gcode == Slic3r::ColorChangeCode             ? ( m_state == msSingleExtruder ? |                   tick_code_it->gcode == Slic3r::ColorChangeCode    ? ( m_mode == mmSingleExtruder ? | ||||||
|                       _(L("For Delete color change use left mouse button click\n" |                       _(L("For Delete color change use left mouse button click\n" | ||||||
|                           "For Edit color use right mouse button click")) : |                           "For Edit color use right mouse button click")) : | ||||||
|                       from_u8((boost::format(_utf8(L("Delete color change for Extruder %1%"))) % tick_code_it->extruder).str()) ): |                       from_u8((boost::format(_utf8(L("Delete color change for Extruder %1%"))) % tick_code_it->extruder).str()) ): | ||||||
| //                  tick_code_it->gcode == Slic3r::PausePrintCode             ? _(L("Delete pause")) :
 |                   tick_code_it->gcode == Slic3r::PausePrintCode     ?  | ||||||
|                   tick_code_it->gcode == Slic3r::ExtruderChangeCode         ? |                       _(L("Delete pause")) : | ||||||
|  |                   tick_code_it->gcode == Slic3r::ExtruderChangeCode ? | ||||||
|                       from_u8((boost::format(_utf8(L("Delete extruder change to \"%1%\""))) % tick_code_it->extruder).str()) : |                       from_u8((boost::format(_utf8(L("Delete extruder change to \"%1%\""))) % tick_code_it->extruder).str()) : | ||||||
|                       from_u8((boost::format(_utf8(L("For Delete \"%1%\" code use left mouse button click\n" |                       from_u8((boost::format(_utf8(L("For Delete \"%1%\" code use left mouse button click\n" | ||||||
|                                                        "For Edit \"%1%\" code use right mouse button click"))) % tick_code_it->gcode ).str()); |                                                        "For Edit \"%1%\" code use right mouse button click"))) % tick_code_it->gcode ).str()); | ||||||
|  | @ -3114,7 +3150,7 @@ void DoubleSlider::OnMotion(wxMouseEvent& event) | ||||||
| 
 | 
 | ||||||
|     if (!m_is_left_down && !m_is_one_layer) { |     if (!m_is_left_down && !m_is_one_layer) { | ||||||
|         m_is_action_icon_focesed = is_point_in_rect(pos, m_rect_tick_action); |         m_is_action_icon_focesed = is_point_in_rect(pos, m_rect_tick_action); | ||||||
|         if (!m_ticks_.empty() && is_point_in_rect(pos, m_rect_revert_icon)) |         if (!m_ticks.empty() && is_point_in_rect(pos, m_rect_revert_icon)) | ||||||
|             icon_focus = ifRevert; |             icon_focus = ifRevert; | ||||||
|         else if (is_point_in_rect(pos, m_rect_cog_icon)) |         else if (is_point_in_rect(pos, m_rect_cog_icon)) | ||||||
|             icon_focus = ifCog; |             icon_focus = ifCog; | ||||||
|  | @ -3149,6 +3185,67 @@ void DoubleSlider::OnMotion(wxMouseEvent& event) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void DoubleSlider::append_change_extruder_menu_item(wxMenu* menu) | ||||||
|  | { | ||||||
|  |     const int extruders_cnt = Slic3r::GUI::wxGetApp().extruders_edited_cnt(); | ||||||
|  |     if (extruders_cnt > 1) | ||||||
|  |     { | ||||||
|  |         const int initial_extruder = std::max<int>(1 , get_extruder_for_tick(m_selection == ssLower ? m_lower_value : m_higher_value)); | ||||||
|  | 
 | ||||||
|  |         wxMenu* change_extruder_menu = new wxMenu(); | ||||||
|  | 
 | ||||||
|  |         for (int i = 1; i <= extruders_cnt; i++) | ||||||
|  |         { | ||||||
|  |             const bool is_active_extruder = i == initial_extruder; | ||||||
|  |             const wxString item_name = wxString::Format(_(L("Extruder %d")), i) + | ||||||
|  |                                        (is_active_extruder ? " (" + _(L("active")) + ")" : ""); | ||||||
|  | 
 | ||||||
|  |             if (m_mode == mmMultiAsSingle) | ||||||
|  |                 append_menu_item(change_extruder_menu, wxID_ANY, item_name, "", | ||||||
|  |                     [this, i](wxCommandEvent&) { change_extruder(i); }, "", menu, | ||||||
|  |                     [is_active_extruder]() { return !is_active_extruder; }, Slic3r::GUI::wxGetApp().plater()); | ||||||
|  | //                append_menu_radio_item(change_extruder_menu, wxID_ANY, item_name, "",
 | ||||||
|  | //                    [this, i](wxCommandEvent&) { change_extruder(i); }, menu)->Check(i == initial_extruder);
 | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         const wxString change_extruder_menu_name = m_mode == mmMultiAsSingle ? _(L("Change extruder")) : _(L("Change extruder (N/A)")); | ||||||
|  | 
 | ||||||
|  |         wxMenuItem* change_extruder_menu_item = menu->AppendSubMenu(change_extruder_menu, change_extruder_menu_name, _(L("Use another extruder"))); | ||||||
|  |         change_extruder_menu_item->SetBitmap(create_scaled_bitmap(this, "change_extruder")); | ||||||
|  | 
 | ||||||
|  |         Slic3r::GUI::wxGetApp().plater()->Bind(wxEVT_UPDATE_UI, [this, change_extruder_menu_item](wxUpdateUIEvent& evt) { | ||||||
|  |             enable_menu_item(evt, [this]() {return m_mode == mmMultiAsSingle; }, change_extruder_menu_item, this); }, | ||||||
|  |             change_extruder_menu_item->GetId()); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void DoubleSlider::append_add_color_change_menu_item(wxMenu* menu) | ||||||
|  | { | ||||||
|  |     const int extruders_cnt = Slic3r::GUI::wxGetApp().extruders_edited_cnt(); | ||||||
|  |     if (extruders_cnt > 1) | ||||||
|  |     { | ||||||
|  |         std::set<int> used_extruders_for_tick = get_used_extruders_for_tick(m_selection == ssLower ? m_lower_value : m_higher_value); | ||||||
|  | 
 | ||||||
|  |         wxMenu* add_color_change_menu = new wxMenu(); | ||||||
|  | 
 | ||||||
|  |         for (int i = 1; i <= extruders_cnt; i++) | ||||||
|  |         { | ||||||
|  |             const bool is_used_extruder = used_extruders_for_tick.empty() ? true : // #ys_FIXME till used_extruders_for_tick doesn't filled correct for mmMultiExtruder
 | ||||||
|  |                                           used_extruders_for_tick.find(i) != used_extruders_for_tick.end(); | ||||||
|  |             const wxString item_name = wxString::Format(_(L("Extruder %d")), i) + | ||||||
|  |                                        (is_used_extruder ? " (" + _(L("used")) + ")" : ""); | ||||||
|  | 
 | ||||||
|  |             append_menu_item(add_color_change_menu, wxID_ANY, item_name, "", | ||||||
|  |                 [this, i](wxCommandEvent&) { add_code(Slic3r::ColorChangeCode, i); }, "", menu, | ||||||
|  |                 [is_used_extruder]() { return is_used_extruder; }, Slic3r::GUI::wxGetApp().plater()); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         const wxString menu_name = from_u8((boost::format(_utf8(L("Add color change (%1%) for:"))) % Slic3r::ColorChangeCode).str()); | ||||||
|  |         wxMenuItem* add_color_change_menu_item = menu->AppendSubMenu(add_color_change_menu, menu_name, ""); | ||||||
|  |         add_color_change_menu_item->SetBitmap(create_scaled_bitmap(nullptr, "colorchange_add_m")); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void DoubleSlider::OnLeftUp(wxMouseEvent& event) | void DoubleSlider::OnLeftUp(wxMouseEvent& event) | ||||||
| { | { | ||||||
|     if (!HasCapture()) |     if (!HasCapture()) | ||||||
|  | @ -3158,31 +3255,19 @@ void DoubleSlider::OnLeftUp(wxMouseEvent& event) | ||||||
| 
 | 
 | ||||||
|     if (m_show_context_menu) |     if (m_show_context_menu) | ||||||
|     { |     { | ||||||
|         if (m_state == msMultiExtruder) |         if (m_mode == mmSingleExtruder) | ||||||
|  |             add_code(Slic3r::ColorChangeCode); | ||||||
|  |         else | ||||||
|         { |         { | ||||||
|             wxMenu menu; |             wxMenu menu; | ||||||
|             const int extruders_cnt = Slic3r::GUI::wxGetApp().extruders_edited_cnt(); |  | ||||||
|             if (extruders_cnt > 1) |  | ||||||
|             { |  | ||||||
|                 const int initial_extruder = get_extruder_for_tick(m_selection == ssLower ? m_lower_value : m_higher_value); |  | ||||||
| 
 | 
 | ||||||
|                 wxMenu* change_extruder_menu = new wxMenu(); |             if (m_mode == mmMultiAsSingle) | ||||||
| 
 |                 append_change_extruder_menu_item(&menu); | ||||||
|                 for (int i = 0; i <= extruders_cnt; i++) { |             else | ||||||
|                     const wxString item_name = i == 0 ? _(L("Default")) : wxString::Format(_(L("Extruder %d")), i); |                 append_add_color_change_menu_item(&menu); | ||||||
| 
 |  | ||||||
|                     append_menu_radio_item(change_extruder_menu, wxID_ANY, item_name, "", |  | ||||||
|                         [this, i](wxCommandEvent&) { change_extruder(i); }, &menu)->Check(i == initial_extruder); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 wxMenuItem* change_extruder_menu_item = menu.AppendSubMenu(change_extruder_menu, _(L("Change extruder")), _(L("Use another extruder"))); |  | ||||||
|                 change_extruder_menu_item->SetBitmap(create_scaled_bitmap(nullptr, "change_extruder")); |  | ||||||
|             } |  | ||||||
| 
 | 
 | ||||||
|             Slic3r::GUI::wxGetApp().plater()->PopupMenu(&menu); |             Slic3r::GUI::wxGetApp().plater()->PopupMenu(&menu); | ||||||
|         } |         } | ||||||
|         else |  | ||||||
|             add_code(Slic3r::ColorChangeCode); |  | ||||||
|          |          | ||||||
|         m_show_context_menu = false; |         m_show_context_menu = false; | ||||||
|     } |     } | ||||||
|  | @ -3242,13 +3327,13 @@ void DoubleSlider::action_tick(const TicksAction action) | ||||||
| 
 | 
 | ||||||
|     const int tick = m_selection == ssLower ? m_lower_value : m_higher_value; |     const int tick = m_selection == ssLower ? m_lower_value : m_higher_value; | ||||||
| 
 | 
 | ||||||
|     const auto it = m_ticks_.find(TICK_CODE{tick}); |     const auto it = m_ticks.find(TICK_CODE{tick}); | ||||||
| 
 | 
 | ||||||
|     if (it != m_ticks_.end()) // erase this tick
 |     if (it != m_ticks.end()) // erase this tick
 | ||||||
|     { |     { | ||||||
|         if (action == taAdd) |         if (action == taAdd) | ||||||
|             return; |             return; | ||||||
|         m_ticks_.erase(TICK_CODE{tick}); |         m_ticks.erase(TICK_CODE{tick}); | ||||||
| 
 | 
 | ||||||
|         wxPostEvent(this->GetParent(), wxCommandEvent(wxCUSTOMEVT_TICKSCHANGED)); |         wxPostEvent(this->GetParent(), wxCommandEvent(wxCUSTOMEVT_TICKSCHANGED)); | ||||||
|         Refresh(); |         Refresh(); | ||||||
|  | @ -3265,7 +3350,7 @@ void DoubleSlider::action_tick(const TicksAction action) | ||||||
|         if (m_suppress_add_code) |         if (m_suppress_add_code) | ||||||
|             return; |             return; | ||||||
|         m_suppress_add_code = true; |         m_suppress_add_code = true; | ||||||
|         if (m_state != msMultiExtruder) |         if (m_mode == mmSingleExtruder) // if (m_mode != mmMultiExtruder)
 | ||||||
|             add_code(Slic3r::ColorChangeCode); |             add_code(Slic3r::ColorChangeCode); | ||||||
|         m_suppress_add_code = false; |         m_suppress_add_code = false; | ||||||
|         return; |         return; | ||||||
|  | @ -3352,8 +3437,8 @@ void DoubleSlider::OnRightDown(wxMouseEvent& event) | ||||||
|     { |     { | ||||||
|         const int tick = m_selection == ssLower ? m_lower_value : m_higher_value; |         const int tick = m_selection == ssLower ? m_lower_value : m_higher_value; | ||||||
|         // if on this Z doesn't exist tick
 |         // if on this Z doesn't exist tick
 | ||||||
|         auto it = m_ticks_.find(TICK_CODE{ tick }); |         auto it = m_ticks.find(TICK_CODE{ tick }); | ||||||
|         if (it == m_ticks_.end()) |         if (it == m_ticks.end()) | ||||||
|         { |         { | ||||||
|             // show context menu on OnRightUp()
 |             // show context menu on OnRightUp()
 | ||||||
|             m_show_context_menu = true; |             m_show_context_menu = true; | ||||||
|  | @ -3386,17 +3471,77 @@ void DoubleSlider::OnRightDown(wxMouseEvent& event) | ||||||
| 
 | 
 | ||||||
| int DoubleSlider::get_extruder_for_tick(int tick) | int DoubleSlider::get_extruder_for_tick(int tick) | ||||||
| { | { | ||||||
|     if (m_ticks_.empty()) |     int default_initial_extruder = m_mode == mmMultiAsSingle ? m_only_extruder : 0; | ||||||
|         return 0; |     if (m_ticks.empty()) | ||||||
|  |         return default_initial_extruder; | ||||||
|      |      | ||||||
|     auto it = m_ticks_.lower_bound(TICK_CODE{tick}); |     auto it = m_ticks.lower_bound(TICK_CODE{tick}); | ||||||
|     while (it != m_ticks_.begin()) { |     while (it != m_ticks.begin()) { | ||||||
|         --it; |         --it; | ||||||
|         if(it->gcode == Slic3r::ExtruderChangeCode) |         if(it->gcode == Slic3r::ExtruderChangeCode) | ||||||
|             return it->extruder; |             return it->extruder; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     return 0; |     return default_initial_extruder; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | std::set<int> DoubleSlider::get_used_extruders_for_tick(int tick) | ||||||
|  | { | ||||||
|  |     if (m_mode == mmMultiExtruder) | ||||||
|  |     { | ||||||
|  |         // #ys_FIXME: get tool ordering from _correct_ place
 | ||||||
|  |         const Slic3r::ToolOrdering& tool_ordering = Slic3r::GUI::wxGetApp().plater()->fff_print().get_tool_ordering(); | ||||||
|  | 
 | ||||||
|  |         if (tool_ordering.empty()) | ||||||
|  |             return {}; | ||||||
|  | 
 | ||||||
|  |         std::set<int> used_extruders; | ||||||
|  | 
 | ||||||
|  |         auto it_layer_tools = std::lower_bound(tool_ordering.begin(), tool_ordering.end(), Slic3r::LayerTools(m_values[tick])); | ||||||
|  |         for (; it_layer_tools != tool_ordering.end(); it_layer_tools++) | ||||||
|  |         { | ||||||
|  |             const std::vector<unsigned>& extruders = it_layer_tools->extruders; | ||||||
|  |             for (const auto& extruder : extruders) | ||||||
|  |                 used_extruders.emplace(extruder+1); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         return used_extruders; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const int default_initial_extruder = m_mode == mmMultiAsSingle ? std::max(m_only_extruder, 1) : 1; | ||||||
|  |     if (m_ticks.empty()) | ||||||
|  |         return {default_initial_extruder}; | ||||||
|  | 
 | ||||||
|  |     std::set<int> used_extruders; | ||||||
|  |     auto it_start = m_ticks.lower_bound(TICK_CODE{tick}); | ||||||
|  | 
 | ||||||
|  |     auto it = it_start; | ||||||
|  |     if (it == m_ticks.begin() && it->gcode == Slic3r::ExtruderChangeCode) { | ||||||
|  |         used_extruders.emplace(it->extruder); | ||||||
|  |         if (tick < it->tick) | ||||||
|  |             used_extruders.emplace(default_initial_extruder); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     while (it != m_ticks.begin()) { | ||||||
|  |         --it; | ||||||
|  |         if(it->gcode == Slic3r::ExtruderChangeCode) | ||||||
|  |         { | ||||||
|  |             used_extruders.emplace(it->extruder); | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (it == m_ticks.begin() && used_extruders.empty()) | ||||||
|  |         used_extruders.emplace(default_initial_extruder); | ||||||
|  | 
 | ||||||
|  |     it = it_start; | ||||||
|  |     while (it != m_ticks.end()) { | ||||||
|  |         if(it->gcode == Slic3r::ExtruderChangeCode) | ||||||
|  |             used_extruders.emplace(it->extruder); | ||||||
|  |         ++it; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return used_extruders; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void DoubleSlider::OnRightUp(wxMouseEvent& event) | void DoubleSlider::OnRightUp(wxMouseEvent& event) | ||||||
|  | @ -3409,45 +3554,23 @@ void DoubleSlider::OnRightUp(wxMouseEvent& event) | ||||||
|     if (m_show_context_menu) { |     if (m_show_context_menu) { | ||||||
|         wxMenu menu; |         wxMenu menu; | ||||||
| 
 | 
 | ||||||
|         if (m_state == msMultiExtruder) |         if (m_mode == mmSingleExtruder) | ||||||
|         { |             append_menu_item(&menu, wxID_ANY, _(L("Add color change")) + " (M600)", "", | ||||||
|             const int extruders_cnt = Slic3r::GUI::wxGetApp().extruders_edited_cnt(); |                 [this](wxCommandEvent&) { add_code(Slic3r::ColorChangeCode); }, "colorchange_add_m", &menu, | ||||||
|             if (extruders_cnt > 1) |                 [](){return true;}, this); | ||||||
|             { |  | ||||||
|                 const int initial_extruder = get_extruder_for_tick(m_selection == ssLower ? m_lower_value : m_higher_value); |  | ||||||
| 
 |  | ||||||
|                 wxMenu* change_extruder_menu = new wxMenu(); |  | ||||||
|                 wxMenu* add_color_change_menu = new wxMenu(); |  | ||||||
| 
 |  | ||||||
|                 for (int i = 0; i <= extruders_cnt; i++) { |  | ||||||
|                     const wxString item_name = i == 0 ? _(L("Default")) : wxString::Format(_(L("Extruder %d")), i); |  | ||||||
| 
 |  | ||||||
|                     append_menu_radio_item(change_extruder_menu, wxID_ANY, item_name, "", |  | ||||||
|                         [this, i](wxCommandEvent&) { change_extruder(i); }, &menu)->Check(i == initial_extruder); |  | ||||||
| 
 |  | ||||||
|                     if (i==0)       // don't use M600 for default extruder, if multimaterial print is selected 
 |  | ||||||
|                         continue; |  | ||||||
|                     append_menu_item(add_color_change_menu, wxID_ANY, item_name, "", |  | ||||||
|                         [this, i](wxCommandEvent&) { add_code(Slic3r::ColorChangeCode, i); }, "", &menu); |  | ||||||
|                 } |  | ||||||
| 
 |  | ||||||
|                 wxMenuItem* change_extruder_menu_item = menu.AppendSubMenu(change_extruder_menu, _(L("Change extruder")), _(L("Use another extruder"))); |  | ||||||
|                 change_extruder_menu_item->SetBitmap(create_scaled_bitmap(nullptr, "change_extruder")); |  | ||||||
| 
 |  | ||||||
|                 const wxString menu_name = from_u8((boost::format(_utf8(L("Add color change (%1%) for:"))) % Slic3r::ColorChangeCode).str()); |  | ||||||
|                 wxMenuItem* add_color_change_menu_item = menu.AppendSubMenu(add_color_change_menu, menu_name, ""); |  | ||||||
|                 add_color_change_menu_item->SetBitmap(create_scaled_bitmap(nullptr, "colorchange_add_m")); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         else |         else | ||||||
|         append_menu_item(&menu, wxID_ANY, _(L("Add color change")) + " (M600)", "", |         { | ||||||
|             [this](wxCommandEvent&) { add_code(Slic3r::ColorChangeCode); }, "colorchange_add_m", &menu); |             append_change_extruder_menu_item(&menu); | ||||||
|  |             append_add_color_change_menu_item(&menu); | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         append_menu_item(&menu, wxID_ANY, _(L("Add pause print")) + " (M601)", "", |         append_menu_item(&menu, wxID_ANY, _(L("Add pause print")) + " (M601)", "", | ||||||
|             [this](wxCommandEvent&) { add_code(Slic3r::PausePrintCode); }, "pause_print", &menu); |             [this](wxCommandEvent&) { add_code(Slic3r::PausePrintCode); }, "pause_print", &menu, | ||||||
|  |             []() {return true; }, this); | ||||||
|      |      | ||||||
|         append_menu_item(&menu, wxID_ANY, _(L("Add custom G-code")), "", |         append_menu_item(&menu, wxID_ANY, _(L("Add custom G-code")), "", | ||||||
|             [this](wxCommandEvent&) { add_code(""); }, "edit_gcode", &menu); |             [this](wxCommandEvent&) { add_code(""); }, "edit_gcode", &menu, | ||||||
|  |             []() {return true; }, this); | ||||||
|      |      | ||||||
|         Slic3r::GUI::wxGetApp().plater()->PopupMenu(&menu); |         Slic3r::GUI::wxGetApp().plater()->PopupMenu(&menu); | ||||||
| 
 | 
 | ||||||
|  | @ -3456,7 +3579,7 @@ void DoubleSlider::OnRightUp(wxMouseEvent& event) | ||||||
|     else if (m_show_edit_menu) { |     else if (m_show_edit_menu) { | ||||||
|         wxMenu menu; |         wxMenu menu; | ||||||
| 
 | 
 | ||||||
|         std::set<TICK_CODE>::iterator it = m_ticks_.find(TICK_CODE{ m_selection == ssLower ? m_lower_value : m_higher_value }); |         std::set<TICK_CODE>::iterator it = m_ticks.find(TICK_CODE{ m_selection == ssLower ? m_lower_value : m_higher_value }); | ||||||
|         const bool is_color_change = it->gcode == Slic3r::ColorChangeCode; |         const bool is_color_change = it->gcode == Slic3r::ColorChangeCode; | ||||||
| 
 | 
 | ||||||
|         append_menu_item(&menu, wxID_ANY, it->gcode == Slic3r::ColorChangeCode ? _(L("Edit color")) : |         append_menu_item(&menu, wxID_ANY, it->gcode == Slic3r::ColorChangeCode ? _(L("Edit color")) : | ||||||
|  | @ -3528,74 +3651,69 @@ void DoubleSlider::add_code(std::string code, int selected_extruder/* = -1*/) | ||||||
| { | { | ||||||
|     const int tick = m_selection == ssLower ? m_lower_value : m_higher_value; |     const int tick = m_selection == ssLower ? m_lower_value : m_higher_value; | ||||||
|     // if on this Z doesn't exist tick
 |     // if on this Z doesn't exist tick
 | ||||||
|     auto it = m_ticks_.find(TICK_CODE{ tick }); |     auto it = m_ticks.find(TICK_CODE{ tick }); | ||||||
|     if (it == m_ticks_.end()) |     if (it != m_ticks.end()) | ||||||
|  |         return; | ||||||
|  | 
 | ||||||
|  |     std::string color; | ||||||
|  |     const int extruder = selected_extruder > 0 ? selected_extruder : std::max<int>(1, m_only_extruder); | ||||||
|  | 
 | ||||||
|  |     if (code == Slic3r::ColorChangeCode) | ||||||
|     { |     { | ||||||
|         std::string color = ""; |         std::vector<std::string> colors = Slic3r::GUI::wxGetApp().plater()->get_extruder_colors_from_plater_config(); | ||||||
|         if (code == Slic3r::ColorChangeCode) |  | ||||||
|         { |  | ||||||
|             std::vector<std::string> colors = Slic3r::GUI::wxGetApp().plater()->get_extruder_colors_from_plater_config(); |  | ||||||
| 
 | 
 | ||||||
|             if (m_state == msSingleExtruder && !m_ticks_.empty()) { |         if (m_ticks.empty()) | ||||||
|                 auto before_tick_it = std::lower_bound(m_ticks_.begin(), m_ticks_.end(), TICK_CODE{ tick }); |             color = colors[extruder-1]; | ||||||
|                 while (before_tick_it != m_ticks_.begin()) { |         else | ||||||
|                     --before_tick_it; |         { | ||||||
|                     if (before_tick_it->gcode == Slic3r::ColorChangeCode) { |             auto before_tick_it = std::lower_bound(m_ticks.begin(), m_ticks.end(), TICK_CODE{ tick }); | ||||||
|                         color = before_tick_it->color; |             while (before_tick_it != m_ticks.begin()) { | ||||||
|                         break; |                 --before_tick_it; | ||||||
|                     } |                 if (before_tick_it->gcode == Slic3r::ColorChangeCode && before_tick_it->extruder == extruder) { | ||||||
|  |                     color = before_tick_it->color; | ||||||
|  |                     break; | ||||||
|                 } |                 } | ||||||
| 
 |  | ||||||
|                 if (color.empty()) |  | ||||||
|                     color = colors[0]; |  | ||||||
|             } |             } | ||||||
|             else |  | ||||||
|                 color = colors[selected_extruder > 0 ? selected_extruder-1 : 0]; |  | ||||||
| 
 | 
 | ||||||
|             color = get_new_color(color); |  | ||||||
|             if (color.empty()) |             if (color.empty()) | ||||||
|                 return; |                 color = colors[extruder-1]; | ||||||
|         } |  | ||||||
|         else if (code == Slic3r::PausePrintCode) |  | ||||||
|         { |  | ||||||
|             /* PausePrintCode doesn't need a color, so
 |  | ||||||
|              * this field is used for save a short message shown on Printer display  |  | ||||||
|              * */ |  | ||||||
|             color = get_pause_print_msg(m_pause_print_msg, m_values[tick]); |  | ||||||
|             if (color.empty()) |  | ||||||
|                 return; |  | ||||||
|             m_pause_print_msg = color; |  | ||||||
|         } |  | ||||||
|         else if (code.empty()) |  | ||||||
|         { |  | ||||||
|             code = get_custom_code(m_custom_gcode, m_values[tick]); |  | ||||||
|             if (code.empty()) |  | ||||||
|                 return; |  | ||||||
|             m_custom_gcode = code; |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         int extruder = 1; |         color = get_new_color(color); | ||||||
|         if (m_state == msMultiExtruder) {  |         if (color.empty()) | ||||||
|             if (code == Slic3r::ColorChangeCode && selected_extruder >= 0) |             return; | ||||||
|                 extruder = selected_extruder; |  | ||||||
|             else |  | ||||||
|                 extruder = get_extruder_for_tick(m_selection == ssLower ? m_lower_value : m_higher_value); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         m_ticks_.emplace(TICK_CODE{tick, code, extruder, color}); |  | ||||||
| 
 |  | ||||||
|         wxPostEvent(this->GetParent(), wxCommandEvent(wxCUSTOMEVT_TICKSCHANGED)); |  | ||||||
|         Refresh(); |  | ||||||
|         Update(); |  | ||||||
|     } |     } | ||||||
|  |     else if (code == Slic3r::PausePrintCode) | ||||||
|  |     { | ||||||
|  |         /* PausePrintCode doesn't need a color, so
 | ||||||
|  |          * this field is used for save a short message shown on Printer display  | ||||||
|  |          * */ | ||||||
|  |         color = get_pause_print_msg(m_pause_print_msg, m_values[tick]); | ||||||
|  |         if (color.empty()) | ||||||
|  |             return; | ||||||
|  |         m_pause_print_msg = color; | ||||||
|  |     } | ||||||
|  |     else if (code.empty()) | ||||||
|  |     { | ||||||
|  |         code = get_custom_code(m_custom_gcode, m_values[tick]); | ||||||
|  |         if (code.empty()) | ||||||
|  |             return; | ||||||
|  |         m_custom_gcode = code; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     m_ticks.emplace(TICK_CODE{tick, code, extruder, color}); | ||||||
|  | 
 | ||||||
|  |     wxPostEvent(this->GetParent(), wxCommandEvent(wxCUSTOMEVT_TICKSCHANGED)); | ||||||
|  |     Refresh(); | ||||||
|  |     Update(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void DoubleSlider::edit_tick() | void DoubleSlider::edit_tick() | ||||||
| { | { | ||||||
|     const int tick = m_selection == ssLower ? m_lower_value : m_higher_value; |     const int tick = m_selection == ssLower ? m_lower_value : m_higher_value; | ||||||
|     // if on this Z exists tick
 |     // if on this Z exists tick
 | ||||||
|     std::set<TICK_CODE>::iterator it = m_ticks_.find(TICK_CODE{ tick }); |     std::set<TICK_CODE>::iterator it = m_ticks.find(TICK_CODE{ tick }); | ||||||
|     if (it != m_ticks_.end()) |     if (it != m_ticks.end()) | ||||||
|     { |     { | ||||||
|         std::string edited_value; |         std::string edited_value; | ||||||
|         if (it->gcode == Slic3r::ColorChangeCode) |         if (it->gcode == Slic3r::ColorChangeCode) | ||||||
|  | @ -3620,8 +3738,8 @@ void DoubleSlider::edit_tick() | ||||||
|             changed_tick.gcode = edited_value; |             changed_tick.gcode = edited_value; | ||||||
|         } |         } | ||||||
|          |          | ||||||
|         m_ticks_.erase(it); |         m_ticks.erase(it); | ||||||
|         m_ticks_.emplace(changed_tick); |         m_ticks.emplace(changed_tick); | ||||||
| 
 | 
 | ||||||
|         wxPostEvent(this->GetParent(), wxCommandEvent(wxCUSTOMEVT_TICKSCHANGED)); |         wxPostEvent(this->GetParent(), wxCommandEvent(wxCUSTOMEVT_TICKSCHANGED)); | ||||||
|     } |     } | ||||||
|  | @ -3634,9 +3752,9 @@ void DoubleSlider::change_extruder(int extruder) | ||||||
|     std::vector<std::string> colors = Slic3r::GUI::wxGetApp().plater()->get_extruder_colors_from_plater_config(); |     std::vector<std::string> colors = Slic3r::GUI::wxGetApp().plater()->get_extruder_colors_from_plater_config(); | ||||||
| 
 | 
 | ||||||
|     // if on this Y doesn't exist tick
 |     // if on this Y doesn't exist tick
 | ||||||
|     if (m_ticks_.find(TICK_CODE{tick}) == m_ticks_.end()) |     if (m_ticks.find(TICK_CODE{tick}) == m_ticks.end()) | ||||||
|     {         |     {         | ||||||
|         m_ticks_.emplace(TICK_CODE{tick, Slic3r::ExtruderChangeCode, extruder, extruder == 0 ? "" : colors[extruder-1]}); |         m_ticks.emplace(TICK_CODE{tick, Slic3r::ExtruderChangeCode, extruder, extruder == 0 ? "" : colors[extruder-1]}); | ||||||
| 
 | 
 | ||||||
|         wxPostEvent(this->GetParent(), wxCommandEvent(wxCUSTOMEVT_TICKSCHANGED)); |         wxPostEvent(this->GetParent(), wxCommandEvent(wxCUSTOMEVT_TICKSCHANGED)); | ||||||
|         Refresh(); |         Refresh(); | ||||||
|  | @ -3656,10 +3774,10 @@ void DoubleSlider::edit_extruder_sequence() | ||||||
| 
 | 
 | ||||||
|     m_extruders_sequence = from_dlg_val; |     m_extruders_sequence = from_dlg_val; | ||||||
| 
 | 
 | ||||||
|     auto it = m_ticks_.begin(); |     auto it = m_ticks.begin(); | ||||||
|     while (it != m_ticks_.end()) { |     while (it != m_ticks.end()) { | ||||||
|         if (it->gcode == Slic3r::ExtruderChangeCode) |         if (it->gcode == Slic3r::ExtruderChangeCode) | ||||||
|             it = m_ticks_.erase(it); |             it = m_ticks.erase(it); | ||||||
|         else |         else | ||||||
|             ++it; |             ++it; | ||||||
|     } |     } | ||||||
|  | @ -3674,7 +3792,7 @@ void DoubleSlider::edit_extruder_sequence() | ||||||
|     while (tick <= m_max_value) |     while (tick <= m_max_value) | ||||||
|     { |     { | ||||||
|         int cur_extruder = m_extruders_sequence.extruders[extruder]; |         int cur_extruder = m_extruders_sequence.extruders[extruder]; | ||||||
|         m_ticks_.emplace(TICK_CODE{tick, Slic3r::ExtruderChangeCode, cur_extruder + 1, colors[cur_extruder]}); |         m_ticks.emplace(TICK_CODE{tick, Slic3r::ExtruderChangeCode, cur_extruder + 1, colors[cur_extruder]}); | ||||||
| 
 | 
 | ||||||
|         extruder++; |         extruder++; | ||||||
|         if (extruder == extr_cnt) |         if (extruder == extr_cnt) | ||||||
|  |  | ||||||
|  | @ -821,17 +821,23 @@ public: | ||||||
|         EnableTickManipulation(false); |         EnableTickManipulation(false); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     enum ManipulationState { |     enum ManipulationMode { | ||||||
|         msSingleExtruder,   // single extruder printer preset is selected
 |         mmSingleExtruder,   // single extruder printer preset is selected
 | ||||||
|         msMultiExtruder     // multiple extruder printer preset is selected, and "Whole print" is selected 
 |         mmMultiAsSingle,    // multiple extruder printer preset is selected, but 
 | ||||||
|  |                             // this mode works just for Single extruder print 
 | ||||||
|  |                             // (For all print from objects settings is used just one extruder) 
 | ||||||
|  |         mmMultiExtruder     // multiple extruder printer preset is selected
 | ||||||
|     }; |     }; | ||||||
|     void SetManipulationState(ManipulationState state) { |     void SetManipulationMode(ManipulationMode mode) { m_mode = mode; } | ||||||
|         m_state = state; |     ManipulationMode GetManipulationMode() const    { return m_mode; } | ||||||
|  | 
 | ||||||
|  |     void SetModeAndOnlyExtruder(const bool is_one_extruder_printed_model, const int only_extruder) | ||||||
|  |     { | ||||||
|  |         m_mode = !is_one_extruder_printed_model ? mmMultiExtruder : | ||||||
|  |                  only_extruder < 0              ? mmSingleExtruder : | ||||||
|  |                                                   mmMultiAsSingle; | ||||||
|  |         m_only_extruder = only_extruder; | ||||||
|     } |     } | ||||||
|     void SetManipulationState(int extruders_cnt) { |  | ||||||
|         m_state = extruders_cnt ==1 ? msSingleExtruder : msMultiExtruder; |  | ||||||
|     } |  | ||||||
|     ManipulationState GetManipulationState() const { return m_state; } |  | ||||||
| 
 | 
 | ||||||
|     bool is_horizontal() const { return m_style == wxSL_HORIZONTAL; } |     bool is_horizontal() const { return m_style == wxSL_HORIZONTAL; } | ||||||
|     bool is_one_layer() const { return m_is_one_layer; } |     bool is_one_layer() const { return m_is_one_layer; } | ||||||
|  | @ -850,13 +856,23 @@ public: | ||||||
|     void OnKeyUp(wxKeyEvent &event); |     void OnKeyUp(wxKeyEvent &event); | ||||||
|     void OnChar(wxKeyEvent &event); |     void OnChar(wxKeyEvent &event); | ||||||
|     void OnRightDown(wxMouseEvent& event); |     void OnRightDown(wxMouseEvent& event); | ||||||
|     int  get_extruder_for_tick(int tick); |  | ||||||
|     void OnRightUp(wxMouseEvent& event); |     void OnRightUp(wxMouseEvent& event); | ||||||
|     void add_code(std::string code, int selected_extruder = -1); |     void add_code(std::string code, int selected_extruder = -1); | ||||||
|     void edit_tick(); |     void edit_tick(); | ||||||
|     void change_extruder(int extruder); |     void change_extruder(int extruder); | ||||||
|     void edit_extruder_sequence(); |     void edit_extruder_sequence(); | ||||||
| 
 | 
 | ||||||
|  |     struct TICK_CODE | ||||||
|  |     { | ||||||
|  |         bool operator<(const TICK_CODE& other) const { return other.tick > this->tick; } | ||||||
|  |         bool operator>(const TICK_CODE& other) const { return other.tick < this->tick; } | ||||||
|  | 
 | ||||||
|  |         int         tick = 0; | ||||||
|  |         std::string gcode = Slic3r::ColorChangeCode; | ||||||
|  |         int         extruder = 0; | ||||||
|  |         std::string color; | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
| protected: | protected: | ||||||
| 
 | 
 | ||||||
|     void    render(); |     void    render(); | ||||||
|  | @ -878,11 +894,12 @@ protected: | ||||||
|     void    detect_selected_slider(const wxPoint& pt); |     void    detect_selected_slider(const wxPoint& pt); | ||||||
|     void    correct_lower_value(); |     void    correct_lower_value(); | ||||||
|     void    correct_higher_value(); |     void    correct_higher_value(); | ||||||
|     wxString get_tooltip(IconFocus icon_focus); |  | ||||||
|     void    move_current_thumb(const bool condition); |     void    move_current_thumb(const bool condition); | ||||||
|     void    action_tick(const TicksAction action); |     void    action_tick(const TicksAction action); | ||||||
|     void    enter_window(wxMouseEvent& event, const bool enter); |     void    enter_window(wxMouseEvent& event, const bool enter); | ||||||
| 
 | 
 | ||||||
|  | private: | ||||||
|  | 
 | ||||||
|     bool    is_point_in_rect(const wxPoint& pt, const wxRect& rect); |     bool    is_point_in_rect(const wxPoint& pt, const wxRect& rect); | ||||||
|     int     is_point_near_tick(const wxPoint& pt); |     int     is_point_near_tick(const wxPoint& pt); | ||||||
| 
 | 
 | ||||||
|  | @ -894,8 +911,17 @@ protected: | ||||||
|     wxSize      get_size(); |     wxSize      get_size(); | ||||||
|     void        get_size(int *w, int *h); |     void        get_size(int *w, int *h); | ||||||
|     double      get_double_value(const SelectedSlider& selection); |     double      get_double_value(const SelectedSlider& selection); | ||||||
|  |     wxString    get_tooltip(IconFocus icon_focus); | ||||||
|  | 
 | ||||||
|  |     std::string get_color_for_tool_change_tick(std::set<TICK_CODE>::const_iterator it) const; | ||||||
|  |     std::string get_color_for_color_change_tick(std::set<TICK_CODE>::const_iterator it) const; | ||||||
|  |     int         get_extruder_for_tick(int tick); | ||||||
|  |     std::set<int>   get_used_extruders_for_tick(int tick); | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |     void        append_change_extruder_menu_item(wxMenu*); | ||||||
|  |     void        append_add_color_change_menu_item(wxMenu*); | ||||||
| 
 | 
 | ||||||
| private: |  | ||||||
|     bool        is_osx { false }; |     bool        is_osx { false }; | ||||||
|     wxFont      m_font; |     wxFont      m_font; | ||||||
|     int         m_min_value; |     int         m_min_value; | ||||||
|  | @ -914,7 +940,7 @@ private: | ||||||
|     ScalableBitmap    m_bmp_one_layer_unlock_off; |     ScalableBitmap    m_bmp_one_layer_unlock_off; | ||||||
|     ScalableBitmap    m_bmp_revert; |     ScalableBitmap    m_bmp_revert; | ||||||
|     ScalableBitmap    m_bmp_cog; |     ScalableBitmap    m_bmp_cog; | ||||||
|     SelectedSlider  m_selection; |     SelectedSlider    m_selection; | ||||||
|     bool        m_is_left_down = false; |     bool        m_is_left_down = false; | ||||||
|     bool        m_is_right_down = false; |     bool        m_is_right_down = false; | ||||||
|     bool        m_is_one_layer = false; |     bool        m_is_one_layer = false; | ||||||
|  | @ -926,9 +952,10 @@ private: | ||||||
|     bool        m_show_edit_menu = false; |     bool        m_show_edit_menu = false; | ||||||
|     bool        m_edit_extruder_sequence = false; |     bool        m_edit_extruder_sequence = false; | ||||||
|     bool        m_suppress_add_code = false; |     bool        m_suppress_add_code = false; | ||||||
|     ManipulationState m_state = msSingleExtruder; |     ManipulationMode m_mode = mmSingleExtruder; | ||||||
|     std::string m_custom_gcode = ""; |     std::string m_custom_gcode = ""; | ||||||
|     std::string m_pause_print_msg; |     std::string m_pause_print_msg; | ||||||
|  |     int         m_only_extruder = -1; | ||||||
| 
 | 
 | ||||||
|     wxRect      m_rect_lower_thumb; |     wxRect      m_rect_lower_thumb; | ||||||
|     wxRect      m_rect_higher_thumb; |     wxRect      m_rect_higher_thumb; | ||||||
|  | @ -957,50 +984,17 @@ private: | ||||||
| 
 | 
 | ||||||
|     std::vector<wxPen*> m_line_pens; |     std::vector<wxPen*> m_line_pens; | ||||||
|     std::vector<wxPen*> m_segm_pens; |     std::vector<wxPen*> m_segm_pens; | ||||||
|     std::set<int>       m_ticks; |  | ||||||
|     std::vector<double> m_values; |     std::vector<double> m_values; | ||||||
| 
 |     std::set<TICK_CODE> m_ticks; | ||||||
|     struct TICK_CODE |  | ||||||
|     { |  | ||||||
|         bool operator<(const TICK_CODE& other) const { return other.tick > this->tick; } |  | ||||||
|         bool operator>(const TICK_CODE& other) const { return other.tick < this->tick; } |  | ||||||
| 
 |  | ||||||
|         int         tick = 0; |  | ||||||
|         std::string gcode = Slic3r::ColorChangeCode; |  | ||||||
|         int         extruder = 0; |  | ||||||
|         std::string color; |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     std::set<TICK_CODE> m_ticks_; |  | ||||||
| 
 | 
 | ||||||
| public: | public: | ||||||
|     struct ExtrudersSequence |     struct ExtrudersSequence | ||||||
|     { |     { | ||||||
|         bool            is_mm_intervals; |         bool            is_mm_intervals     = true; | ||||||
|         double          interval_by_mm; |         double          interval_by_mm      = 3.0; | ||||||
|         int             interval_by_layers; |         int             interval_by_layers  = 10; | ||||||
|         std::vector<size_t>  extruders; |         std::vector<size_t>  extruders      = { 0 }; | ||||||
| 
 | 
 | ||||||
|         ExtrudersSequence() : |  | ||||||
|             is_mm_intervals(true), |  | ||||||
|             interval_by_mm(3.0), |  | ||||||
|             interval_by_layers(10), |  | ||||||
|             extruders({ 0 }) {} |  | ||||||
| 
 |  | ||||||
|         ExtrudersSequence(const ExtrudersSequence& other) : |  | ||||||
|             is_mm_intervals(other.is_mm_intervals), |  | ||||||
|             interval_by_mm(other.interval_by_mm), |  | ||||||
|             interval_by_layers(other.interval_by_layers), |  | ||||||
|             extruders(other.extruders) {} |  | ||||||
| 
 |  | ||||||
|         ExtrudersSequence& operator=(const ExtrudersSequence& other) { |  | ||||||
|             this->is_mm_intervals   = other.is_mm_intervals; |  | ||||||
|             this->interval_by_mm    = other.interval_by_mm; |  | ||||||
|             this->interval_by_layers= other.interval_by_layers; |  | ||||||
|             this->extruders         = other.extruders; |  | ||||||
| 
 |  | ||||||
|             return *this; |  | ||||||
|         } |  | ||||||
|         bool operator==(const ExtrudersSequence& other) const |         bool operator==(const ExtrudersSequence& other) const | ||||||
|         { |         { | ||||||
|             return  (other.is_mm_intervals      == this->is_mm_intervals    ) && |             return  (other.is_mm_intervals      == this->is_mm_intervals    ) && | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 YuSanka
						YuSanka