mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-24 17:21:11 -06:00 
			
		
		
		
	Started work on extending EigenMesh3D to account for possible drain holes when raycasting
This commit is contained in:
		
							parent
							
								
									673549d608
								
							
						
					
					
						commit
						9dd18a8d6d
					
				
					 6 changed files with 149 additions and 32 deletions
				
			
		|  | @ -126,4 +126,100 @@ bool DrainHole::operator==(const DrainHole &sp) const | |||
|             is_approx(height, sp.height); | ||||
| } | ||||
| 
 | ||||
| bool DrainHole::is_inside(const Vec3f& pt) const | ||||
| { | ||||
|     Eigen::Hyperplane<float, 3> plane(normal, pos); | ||||
|     float dist = plane.signedDistance(pt); | ||||
|     if (dist < EPSILON || dist > height) | ||||
|         return false; | ||||
| 
 | ||||
|     Eigen::ParametrizedLine<float, 3> axis(pos, normal); | ||||
|     if ( axis.squaredDistance(pt) < pow(radius, 2.f)) | ||||
|         return true; | ||||
| 
 | ||||
|     return false; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| // Given a line s+dir*t, return parameter t of intersections  with the hole.
 | ||||
| // If there is no intersection, returns nan.
 | ||||
| std::pair<float, float> DrainHole::get_intersections(const Vec3f& s, | ||||
|                                                      const Vec3f& dir) const | ||||
| { | ||||
|     assert(is_approx(normal.norm(), 1.f)); | ||||
|     const Eigen::ParametrizedLine<float, 3> ray(s, dir.normalized()); | ||||
| 
 | ||||
|     std::pair<float, float> out(std::nan(""), std::nan("")); | ||||
|     const float sqr_radius = pow(radius, 2.f); | ||||
| 
 | ||||
|     // first check a bounding sphere of the hole:
 | ||||
|     Vec3f center = pos+normal*height/2.f; | ||||
|     float sqr_dist_limit = pow(height/2.f, 2.f) + sqr_radius ; | ||||
|     if (ray.squaredDistance(center) > sqr_dist_limit) | ||||
|         return out; | ||||
| 
 | ||||
|     // The line intersects the bounding sphere, look for intersections with
 | ||||
|     // bases of the cylinder.
 | ||||
| 
 | ||||
|     size_t found = 0; // counts how many intersections were found
 | ||||
|     Eigen::Hyperplane<float, 3> base; | ||||
|     if (! is_approx(ray.direction().dot(normal), 0.f)) { | ||||
|         for (size_t i=1; i<=1; --i) { | ||||
|             Vec3f cylinder_center = pos+i*height*normal; | ||||
|             base = Eigen::Hyperplane<float, 3>(normal, cylinder_center); | ||||
|             Vec3f intersection = ray.intersectionPoint(base); | ||||
|             // Only accept the point if it is inside the cylinder base.
 | ||||
|             if ((cylinder_center-intersection).squaredNorm() < sqr_radius) { | ||||
|                 (found ? out.second : out.first) = ray.intersectionParameter(base); | ||||
|                 ++found; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         // In case the line was perpendicular to the cylinder axis, previous
 | ||||
|         // block was skipped, but base will later be assumed to be valid.
 | ||||
|         base = Eigen::Hyperplane<float, 3>(normal, pos); | ||||
|     } | ||||
| 
 | ||||
|     // In case there is still an intersection to be found, check the wall
 | ||||
|     if (found != 2 && ! is_approx(std::abs(ray.direction().dot(normal)), 1.f)) { | ||||
|         // Project the ray onto the base plane
 | ||||
|         Vec3f proj_origin = base.projection(ray.origin()); | ||||
|         Vec3f proj_dir = base.projection(ray.origin()+ray.direction())-proj_origin; | ||||
|         // save how the parameter scales and normalize the projected direction
 | ||||
|         float par_scale = proj_dir.norm(); | ||||
|         proj_dir = proj_dir/par_scale; | ||||
|         Eigen::ParametrizedLine<float, 3> projected_ray(proj_origin, proj_dir); | ||||
|         // Calculate point on the secant that's closest to the center
 | ||||
|         // and its distance to the circle along the projected line
 | ||||
|         Vec3f closest = projected_ray.projection(pos); | ||||
|         float dist = sqrt((sqr_radius - (closest-pos).squaredNorm())); | ||||
|         // Unproject both intersections on the original line and check
 | ||||
|         // they are on the cylinder and not past it:
 | ||||
|         for (int i=-1; i<=1 && found !=2; i+=2) { | ||||
|             Vec3f isect = closest + i*dist * projected_ray.direction(); | ||||
|             float par = (isect-proj_origin).norm() / par_scale; | ||||
|             isect = ray.pointAt(par); | ||||
|             // check that the intersection is between the base planes:
 | ||||
|             float vert_dist = base.signedDistance(isect); | ||||
|             if (vert_dist > 0.f && vert_dist < height) { | ||||
|                 (found ? out.second : out.first) = par; | ||||
|                 ++found; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // If only one intersection was found, it is some corner case,
 | ||||
|     // no intersection will be returned:
 | ||||
|     if (found != 0) | ||||
|         return std::pair<float, float>(std::nan(""), std::nan("")); | ||||
| 
 | ||||
|     // Sort the intersections:
 | ||||
|     if (out.first > out.second) | ||||
|         std::swap(out.first, out.second); | ||||
| 
 | ||||
|     return out; | ||||
| } | ||||
| 
 | ||||
| }} // namespace Slic3r::sla
 | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Lukas Matena
						Lukas Matena