mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-25 01:31:14 -06:00 
			
		
		
		
	Wipe tower postprocessing, wipe tower block on 3D plate improved.
- it renders red with one egde as indeterminate, the front edge is where the wipe tower will start - changing width changes depth of the block (as requested) - the block shows the brim of the wipe tower - after slicing, the block is rendered in usual dark green and takes the exact shape of the tower (also with brim) - moving or rotationg the block after slicing does not invalidate the wipe tower (and hence the exact block dimensions are preserved) - changing anything that invalidates the wipe tower reverts the block back to the "indeterminate" shape - the block is not shown after slicing, if the wipe tower is not actually generated (printing single color object with the wipe tower enabled) This required changes in the wipe tower generator, which now generates the tower at origin with no rotation. Resulting gcode is postprocessed and transformed during gcode export. This means the wipe tower needs not be invalidated when it is moved or rotated.
This commit is contained in:
		
							parent
							
								
									dd724e9dab
								
							
						
					
					
						commit
						d5f042b4b8
					
				
					 14 changed files with 264 additions and 77 deletions
				
			
		|  | @ -643,20 +643,62 @@ std::vector<int> GLVolumeCollection::load_object( | |||
|     return volumes_idx;  | ||||
| } | ||||
| 
 | ||||
| int GLVolumeCollection::load_wipe_tower_preview( | ||||
|     int obj_idx, float pos_x, float pos_y, float width, float depth, float height, float rotation_angle, bool use_VBOs) | ||||
| { | ||||
|     float color[4] = { 0.5f, 0.5f, 0.0f, 0.5f }; | ||||
|     this->volumes.emplace_back(new GLVolume(color)); | ||||
|     GLVolume &v = *this->volumes.back(); | ||||
| 
 | ||||
| int GLVolumeCollection::load_wipe_tower_preview( | ||||
|     int obj_idx, float pos_x, float pos_y, float width, float depth, float height, float rotation_angle, bool use_VBOs, bool size_unknown, float brim_width) | ||||
| { | ||||
|     if (depth < 0.01f) | ||||
|         return int(this->volumes.size() - 1); | ||||
|     if (height == 0.0f) | ||||
|         height = 0.1f; | ||||
| 
 | ||||
|     auto mesh = make_cube(width, depth, height); | ||||
|     mesh.translate(-width / 2.f, -depth / 2.f, 0.f); | ||||
|     Point origin_of_rotation(0.f, 0.f); | ||||
|     mesh.rotate(rotation_angle,&origin_of_rotation); | ||||
|     TriangleMesh mesh; | ||||
|     float color[4] = { 0.5f, 0.5f, 0.0f, 1.f }; | ||||
| 
 | ||||
|     // In case we don't know precise dimensions of the wipe tower yet, we'll draw the box with different color with one side jagged:
 | ||||
|     if (size_unknown) { | ||||
|         color[0] = 1.f; | ||||
|         color[1] = 0.f; | ||||
| 
 | ||||
|         depth = std::max(depth, 10.f); // Too narrow tower would interfere with the teeth. The estimate is not precise anyway.
 | ||||
|         float min_width = 30.f; | ||||
|         // We'll now create the box with jagged edge. y-coordinates of the pre-generated model are shifted so that the front
 | ||||
|         // edge has y=0 and centerline of the back edge has y=depth:
 | ||||
|         Pointf3s points; | ||||
|         std::vector<Point3> facets; | ||||
|         float out_points_idx[][3] = {{0, -depth, 0}, {0, 0, 0}, {38.453, 0, 0}, {61.547, 0, 0}, {100, 0, 0}, {100, -depth, 0}, {55.7735, -10, 0}, {44.2265, 10, 0}, | ||||
|                                      {38.453, 0, 1}, {0, 0, 1}, {0, -depth, 1}, {100, -depth, 1}, {100, 0, 1}, {61.547, 0, 1}, {55.7735, -10, 1}, {44.2265, 10, 1}}; | ||||
|         int out_facets_idx[][3] = {{0, 1, 2}, {3, 4, 5}, {6, 5, 0}, {3, 5, 6}, {6, 2, 7}, {6, 0, 2}, {8, 9, 10}, {11, 12, 13}, {10, 11, 14}, {14, 11, 13}, {15, 8, 14}, | ||||
|                                    {8, 10, 14}, {3, 12, 4}, {3, 13, 12}, {6, 13, 3}, {6, 14, 13}, {7, 14, 6}, {7, 15, 14}, {2, 15, 7}, {2, 8, 15}, {1, 8, 2}, {1, 9, 8}, | ||||
|                                    {0, 9, 1}, {0, 10, 9}, {5, 10, 0}, {5, 11, 10}, {4, 11, 5}, {4, 12, 11}}; | ||||
|         for (int i=0;i<16;++i) | ||||
|             points.push_back(Pointf3(out_points_idx[i][0] / (100.f/min_width), out_points_idx[i][1] + depth, out_points_idx[i][2])); | ||||
|         for (int i=0;i<28;++i) | ||||
|             facets.push_back(Point3(out_facets_idx[i][0], out_facets_idx[i][1], out_facets_idx[i][2])); | ||||
|         TriangleMesh tooth_mesh(points, facets); | ||||
| 
 | ||||
|         // We have the mesh ready. It has one tooth and width of min_width. We will now append several of these together until we are close to
 | ||||
|         // the required width of the block. Than we can scale it precisely.
 | ||||
|         size_t n = std::max(1, int(width/min_width)); // How many shall be merged?
 | ||||
|         for (size_t i=0;i<n;++i) { | ||||
|             mesh.merge(tooth_mesh); | ||||
|             tooth_mesh.translate(min_width, 0.f, 0.f); | ||||
|         } | ||||
| 
 | ||||
|         mesh.scale(Pointf3(width/(n*min_width), 1.f, height)); // Scaling to proper width
 | ||||
|     } | ||||
|     else | ||||
|         mesh = make_cube(width, depth, height); | ||||
| 
 | ||||
|     // We'll make another mesh to show the brim (fixed layer height):
 | ||||
|     TriangleMesh brim_mesh = make_cube(width+2.f*brim_width, depth+2.f*brim_width, 0.2f); | ||||
|     brim_mesh.translate(-brim_width, -brim_width, 0.f); | ||||
|     mesh.merge(brim_mesh); | ||||
| 
 | ||||
|     mesh.rotate(rotation_angle, &origin_of_rotation); // rotates the box according to the config rotation setting
 | ||||
| 
 | ||||
|     this->volumes.emplace_back(new GLVolume(color)); | ||||
|     GLVolume &v = *this->volumes.back(); | ||||
| 
 | ||||
|     if (use_VBOs) | ||||
|         v.indexed_vertex_array.load_mesh_full_shading(mesh); | ||||
|  |  | |||
|  | @ -399,7 +399,7 @@ public: | |||
|         bool                     use_VBOs); | ||||
| 
 | ||||
|     int load_wipe_tower_preview( | ||||
|         int obj_idx, float pos_x, float pos_y, float width, float depth, float height, float rotation_angle, bool use_VBOs); | ||||
|         int obj_idx, float pos_x, float pos_y, float width, float depth, float height, float rotation_angle, bool use_VBOs, bool size_unknown, float brim_width); | ||||
| 
 | ||||
|     // Render the volumes by OpenGL.
 | ||||
|     void render_VBOs() const; | ||||
|  |  | |||
|  | @ -2292,7 +2292,12 @@ void GLCanvas3D::reload_scene(bool force) | |||
|             float w = dynamic_cast<const ConfigOptionFloat*>(m_config->option("wipe_tower_width"))->value; | ||||
|             float a = dynamic_cast<const ConfigOptionFloat*>(m_config->option("wipe_tower_rotation_angle"))->value; | ||||
| 
 | ||||
|             m_volumes.load_wipe_tower_preview(1000, x, y, w, 15.0f * (float)(extruders_count - 1), (float)height, a, m_use_VBOs && m_initialized); | ||||
|             float depth = m_print->get_wipe_tower_depth(); | ||||
|             if (!m_print->state.is_done(psWipeTower)) | ||||
|                 depth = (900.f/w) * (float)(extruders_count - 1) ; | ||||
| 
 | ||||
|             m_volumes.load_wipe_tower_preview(1000, x, y, w, depth, (float)height, a, m_use_VBOs && m_initialized, !m_print->state.is_done(psWipeTower), | ||||
|                                               m_print->config.nozzle_diameter.values[0] * 1.25f * 4.5f); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|  | @ -2554,6 +2559,8 @@ void GLCanvas3D::load_wipe_tower_toolpaths(const std::vector<std::string>& str_t | |||
|     { | ||||
|         const Print                 *print; | ||||
|         const std::vector<float>    *tool_colors; | ||||
|         WipeTower::xy                wipe_tower_pos; | ||||
|         float                        wipe_tower_angle; | ||||
| 
 | ||||
|         // Number of vertices (each vertex is 6x4=24 bytes long)
 | ||||
|         static const size_t          alloc_size_max() { return 131072; } // 3.15MB
 | ||||
|  | @ -2586,6 +2593,9 @@ void GLCanvas3D::load_wipe_tower_toolpaths(const std::vector<std::string>& str_t | |||
|     if (m_print->m_wipe_tower_final_purge) | ||||
|         ctxt.final.emplace_back(*m_print->m_wipe_tower_final_purge.get()); | ||||
| 
 | ||||
|     ctxt.wipe_tower_angle = ctxt.print->config.wipe_tower_rotation_angle.value/180.f * M_PI; | ||||
|     ctxt.wipe_tower_pos = WipeTower::xy(ctxt.print->config.wipe_tower_x.value, ctxt.print->config.wipe_tower_y.value); | ||||
| 
 | ||||
|     BOOST_LOG_TRIVIAL(debug) << "Loading wipe tower toolpaths in parallel - start"; | ||||
| 
 | ||||
|     //FIXME Improve the heuristics for a grain size.
 | ||||
|  | @ -2644,12 +2654,25 @@ void GLCanvas3D::load_wipe_tower_toolpaths(const std::vector<std::string>& str_t | |||
|                     lines.reserve(n_lines); | ||||
|                     widths.reserve(n_lines); | ||||
|                     heights.assign(n_lines, extrusions.layer_height); | ||||
|                     WipeTower::Extrusion e_prev = extrusions.extrusions[i-1]; | ||||
| 
 | ||||
|                     if (!extrusions.priming) { // wipe tower extrusions describe the wipe tower at the origin with no rotation
 | ||||
|                         e_prev.pos.rotate(ctxt.wipe_tower_angle); | ||||
|                         e_prev.pos.translate(ctxt.wipe_tower_pos); | ||||
|                     } | ||||
| 
 | ||||
|                     for (; i < j; ++i) { | ||||
|                         const WipeTower::Extrusion &e = extrusions.extrusions[i]; | ||||
|                         WipeTower::Extrusion e = extrusions.extrusions[i]; | ||||
|                         assert(e.width > 0.f); | ||||
|                         const WipeTower::Extrusion &e_prev = *(&e - 1); | ||||
|                         if (!extrusions.priming) { | ||||
|                             e.pos.rotate(ctxt.wipe_tower_angle); | ||||
|                             e.pos.translate(ctxt.wipe_tower_pos); | ||||
|                         } | ||||
| 
 | ||||
|                         lines.emplace_back(Point::new_scale(e_prev.pos.x, e_prev.pos.y), Point::new_scale(e.pos.x, e.pos.y)); | ||||
|                         widths.emplace_back(e.width); | ||||
| 
 | ||||
|                         e_prev = e; | ||||
|                     } | ||||
|                     _3DScene::thick_lines_to_verts(lines, widths, heights, lines.front().a == lines.back().b, extrusions.print_z, | ||||
|                         *vols[ctxt.volume_idx(e.tool, 0)]); | ||||
|  | @ -3652,7 +3675,7 @@ void GLCanvas3D::_camera_tranform() const | |||
|     ::glMatrixMode(GL_MODELVIEW); | ||||
|     ::glLoadIdentity(); | ||||
| 
 | ||||
|     ::glRotatef(-m_camera.get_theta(), 1.0f, 0.0f, 0.0f); // pitch
 | ||||
|     ::glRotatef(-m_camera.get_theta(), 1.0f, 0.0f, 0.0f); // pitch
 | ||||
|     ::glRotatef(m_camera.phi, 0.0f, 0.0f, 1.0f);          // yaw
 | ||||
| 
 | ||||
|     Pointf3 neg_target = m_camera.target.negative(); | ||||
|  | @ -4627,8 +4650,11 @@ void GLCanvas3D::_load_shells() | |||
|     const PrintConfig& config = m_print->config; | ||||
|     unsigned int extruders_count = config.nozzle_diameter.size(); | ||||
|     if ((extruders_count > 1) && config.single_extruder_multi_material && config.wipe_tower && !config.complete_objects) { | ||||
|         const float width_per_extruder = 15.0f; // a simple workaround after wipe_tower_per_color_wipe got obsolete
 | ||||
|         m_volumes.load_wipe_tower_preview(1000, config.wipe_tower_x, config.wipe_tower_y, config.wipe_tower_width, width_per_extruder * (extruders_count - 1), max_z, config.wipe_tower_rotation_angle, m_use_VBOs && m_initialized); | ||||
|         float depth = m_print->get_wipe_tower_depth(); | ||||
|         if (!m_print->state.is_done(psWipeTower)) | ||||
|             depth = (900.f/config.wipe_tower_width) * (float)(extruders_count - 1) ; | ||||
|         m_volumes.load_wipe_tower_preview(1000, config.wipe_tower_x, config.wipe_tower_y, config.wipe_tower_width, depth, max_z, config.wipe_tower_rotation_angle, | ||||
|                                           m_use_VBOs && m_initialized, !m_print->state.is_done(psWipeTower), m_print->config.nozzle_diameter.values[0] * 1.25f * 4.5f); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
|  | @ -4696,7 +4722,7 @@ void GLCanvas3D::_on_move(const std::vector<int>& volume_idxs) | |||
|     if (m_model == nullptr) | ||||
|         return; | ||||
| 
 | ||||
|     std::set<std::string> done;  // prevent moving instances twice
 | ||||
|     std::set<std::string> done;  // prevent moving instances twice
 | ||||
|     bool object_moved = false; | ||||
|     Pointf3 wipe_tower_origin(0.0, 0.0, 0.0); | ||||
|     for (int volume_idx : volume_idxs) | ||||
|  | @ -4705,7 +4731,7 @@ void GLCanvas3D::_on_move(const std::vector<int>& volume_idxs) | |||
|         int obj_idx = volume->object_idx(); | ||||
|         int instance_idx = volume->instance_idx(); | ||||
| 
 | ||||
|         // prevent moving instances twice
 | ||||
|         // prevent moving instances twice
 | ||||
|         char done_id[64]; | ||||
|         ::sprintf(done_id, "%d_%d", obj_idx, instance_idx); | ||||
|         if (done.find(done_id) != done.end()) | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Lukas Matena
						Lukas Matena