Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Hidden Point Removal (qhull) (feature upgrade) #486

Open
dagata-mining opened this issue Dec 17, 2021 · 4 comments
Open

Hidden Point Removal (qhull) (feature upgrade) #486

dagata-mining opened this issue Dec 17, 2021 · 4 comments

Comments

@dagata-mining
Copy link
Contributor

dagata-mining commented Dec 17, 2021

Hi François,

From working with lidar scans in various environments (Mostly underground mines), I've come to realize that water holes, grid meshes or glass often tend to create unprecise points behind the surface. This affects the Icp and induce an unwanted error. So I implemented on my side a filter based on a hidden point removal approach:
https://www.researchgate.net/publication/200018717_Direct_visibility_of_point_sets
I implemented an octree based approach for fast hidden point removal in my SLAM. It worked pretty well on my side. I would see a that approach a new type of filter for Libpointmatcher. As you can see in my bit of code I'm using PCL, I'm sure there is way to implement it without it, and I'd like to investigate that, but I wanted to seek out the interest of this approach on your guys side, or if it has already been implemented.

      void cleanCloudHull(pcl::PointCloud<pcl::PointXYZI>::Ptr inputCloud, Transform origin, double maxDistance = 0, double octreeSize = 0.025, double fParam = 2)
{
    // Generate the voxel for improving search speed
    pcl::octree::OctreePointCloudSearch<pcl::PointXYZI> octree(octreeSize);
    octree.setInputCloud(inputCloud);
    // update bounding box automatically
    octree.defineBoundingBox();
    // add points in the tree
    octree.addPointsFromInputCloud();

    std::vector<pcl::Indices> octreeIn;
    pcl::PointCloud<pcl::PointXYZ>::Ptr sphericalCloud(new pcl::PointCloud<pcl::PointXYZ>);
    // Get Highest radius
    double maxRadius = 0;
    origin = origin.inverse();
    for (auto it = octree.leaf_begin(); it != octree.leaf_end(); ++it)
    {
        Eigen::Vector3f voxel_min, voxel_max;
        octree.getVoxelBounds(it, voxel_min, voxel_max);
        pcl::PointXYZ voxelLowBound(voxel_min[0], voxel_min[1],voxel_min[2]);
        voxelLowBound = pcl::transformPoint(voxelLowBound, origin.toEigen3f());

        double r2 = sqrt(pow(voxelLowBound.x,2)+pow(voxelLowBound.y,2)+pow(voxelLowBound.z,2));
        if (r2 < maxDistance || maxDistance <= 0.0 )
        {
            pcl::Indices indices;
            it.getLeafContainer().getPointIndices(indices);
            octreeIn.push_back(indices);
            sphericalCloud->push_back(voxelLowBound);
            if (maxRadius < r2)
            {
                maxRadius = r2;
            }
        }
    }
    maxRadius *= pow(10.0,fParam) * 2;
    // Inverting the sphericals
    for (size_t i = 0; i < sphericalCloud->size();i++)
    {
        pcl::PointXYZ point = sphericalCloud->at(i);
        double r = sqrt(pow(point.x,2)+pow(point.y,2)+pow(point.z,2));
        r = (maxRadius/r) - 1.0;
        point.x *= r;
        point.y *= r;
        point.z *= r;
        sphericalCloud->at(i)= point;
    }

    // Create ConvexHull
    pcl::ConvexHull< pcl::PointXYZ > chull;
    pcl::PointCloud<pcl::PointXYZ>::Ptr cloud_hull (new pcl::PointCloud<pcl::PointXYZ>);
    pcl::PointIndices hull_leaves_indices;
    chull.setInputCloud(sphericalCloud);
    chull.reconstruct(*cloud_hull);
    chull.getHullPointIndices (hull_leaves_indices);

    // Clean
    cloud_hull ->clear();
    sphericalCloud ->clear();
    pcl::PointIndices::Ptr inliers (new pcl::PointIndices ());
    for(size_t l=0; l<hull_leaves_indices.indices.size(); ++l) {
        int leafIndex = hull_leaves_indices.indices[l];
        pcl::Indices octreePts = octreeIn[leafIndex];
        for (int ii = 0; ii < octreePts.size(); ++ii) {
            // Get 3D from laser scan
            inliers->indices.push_back(octreePts[ii]);
        }
    }
    pcl::ExtractIndices<pcl::PointXYZI> extract;
    extract.setInputCloud (inputCloud);
    extract.setIndices (inliers);
    extract.setNegative (false);
    extract.filter (*inputCloud);
}
@pomerlef
Copy link
Collaborator

Always happy to get new filters!

I can see advantages for extracting local visible maps in a large point cloud when doing real-time localization. Do you have screenshots of before and after? I'm curious about the water effect. But we ofter observe reflection caused by glasses, but there is typically a hole on the glass so it's hard to confirm that we are behind a surface :
image
image
image

@dagata-mining
Copy link
Contributor Author

dagata-mining commented Dec 18, 2021

Yes, here's an example of a scan where the points where going through a mesh grid with small holes. The points returned from the mesh where blurry : Here's the texture:
stitched_1622159358_832134485

Here's the before (yellow points are the unwanted points):
beforehull2

Here's the after with a 2.5cm octree size and a fparam of 2
afterhull

You can send me some samples with the origin view point origin at (0,0,0) and I'll run it to see the results.

@pomerlef
Copy link
Collaborator

The data can be retrieved from here: https://projects.asl.ethz.ch/datasets/doku.php?id=laserregistration:apartment:home

I think pose id 6-7 see the windows.

@zgxsin
Copy link

zgxsin commented Jun 1, 2022

Hey, I also found the feature very useful. I have tried the API from Open3D, which looks quite good. Is there any plan to have this feature in libpointmacher?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

3 participants