-
-
Notifications
You must be signed in to change notification settings - Fork 35.5k
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
mergeVertices truncates coordinates which misses some neighbors #24621
Comments
It sounds like we should be explicitly rounding to N digits, rather than truncating? That sounds like a bug to me as well, and a PR to fix it would be welcome I think. Thank you! FWIW I recently implemented a version of this that explicitly compares against the tolerance value rather than rounding or truncating. It isn't as slow as N^2 if you sort the vertices first and are selective about which comparisons you make using sort order, but it's still quite a bit slower than what three.js currently does. |
This will have the same problem - both truncating and rounding bin the space and merge vertices in common bins. There will always be cases where two numbers are very close to each other but otherwise fall on opposite sides of the binning operation (rounding or truncation in this case).
Performance is the reason I chose to use the truncation method. If there are other methods that are comparably performant I can support a change (or possibly an option if the performance is noticeably worse). I've been wondering if there's a way to rely on two binning methods (ie rounding and truncation) to enable a fast search for neighbor vertices without falling into this issue. |
Hi @gkjohnson and thank you for writing this method!
rounding will present the same problem as the bin will just shift lower a half-threshold.
That sounds interesting, as far as I know there's at least 3 dimensions to sort the points by, and could be much more. can you elaborate on your sorting metric?
The other methods i can think of are:
|
I'm just sorting on one axis, you can see the implementation here: That could be improved, on average, by choosing the axis with the widest range. Even better (as you suggest) would be to build a BVH or similar, so you can do search outward from any vertex. @gkjohnson would know better than me on this, but I believe the cost would be There is one other, related, issue I've noticed in BufferGeometryUtils.mergeVertices — using the same tolerance for all vertex attributes isn't optimal. The choice of |
I'm not sure I follow this. Rounding vs truncation just shifts where the bin boundaries are - it will just change where failures occur. It may make some models work more correctly given the way (ie your box would merge correctly with rounding instead of truncation) but numerically it's the same issue.
Agreed - I'd support changing the tolerance option to take a map of attribute to tolerance values if it's posing a problem somewhere. -- Building a spatial data structure seems overkill and expensive for this especially for a three.js example file. Three-mesh-bvh can be built relatively fast and perform fast spatial queries - I wouldn't be against adding a utility to that project for merging vertices, but I'd hope there'd be a simpler solution. I'd prefer to start by taking a look at how other engines, tools, or papers address this. Do we have any reference implementations we can take a look at? Meshlab and Blender are both open source and there are certainly more. |
I would guess that tools like Blender and Meshlab are building a half-edge or similar data structure, without GPU-related constraints around per-face normals on the same vertex. Perhaps there's some way to build and maintain such a data structure on top of BufferGeometry, which could be an interesting problem in its own right, but may not be any cheaper or simpler than spatial indexing. |
@donmccurdy Actually just implemented halfedge structure above threejs's
I meant that it would cause the problem to appear less often, especially in parametric models. However, you are correct, this is not a viable solution to the problem. |
An important distinction here is that you only need to merge based on vertex position to create the halfedge. You can (optionally) store per-face normals and other attributes for the same vertex in the halfedge. But yeah, if the geometry is non-indexed you still need to merge positions without the assistance of a halfedge, and in any case this a halfedge probably more complex than we want to add in BufferGeometryUtils.. I have no issue with rounding instead of truncating if it helps in real use cases. |
I agree with this. Instead of rounding we can also shift the bin boundaries away from the round locations where vertices are more likely to land naturally in models. Something like changing this line so it shifts the boundaries by the threshold epsilon or comparable value: ~ ~ ( attribute[ getters[ k ] ]( index ) * shiftMultiplier ) + 1
// or
~ ~ ( ( attribute[ getters[ k ] ]( index ) + tolerance ) * shiftMultiplier ) I'm not sure if there are other thoughts on a better way to pick an offset for these bin boundaries. |
Encountered the issue and my intuition is that we could improve this by running the process twice: once with the hash table boundaries on round numbers, and once with all numbers shifted by This does not exaclty ensure that all vertices closer than However, I believe it would ensure that all vertices closer than The cost of this would be only doubled compared to the cost of the current algorithm. If one is interested by merging vertices only by position, this process can be implemented by translating the geometry by const mergedGeometry1 = Utils.mergeVertices(geometry, tolerance);
mergedGeometry1.translate(tolerance / 2, tolerance / 2, tolerance / 2);
const mergedGeometry2 = Utils.mergeVertices(mergedGeometry1, tolerance);
mergedGeometry2.translate(-tolerance / 2, -tolerance / 2, -tolerance / 2); // Reset vertices to their original positions Any thouhgts ? Am I missing something ? |
Describe the bug
BufferGeometryUtils.mergeVertices()
truncates all coordinates causing it to miss some neighboring vertices which are closer than the threshold sensitivity given tomergeVertices()
.Two vertices should merge when all their coordinates (x, y, z, u, v, nx, etc...) have values closer than the threshold. But if one vertex has at least one coordinate with value that is lower than an integer multiplication of the threshold, they would not be merged.
For example if the threshold is 1e-4 and vertices a and b are the same except x:
This is caused when creating the hash for each vertex using ~~ to truncate/floor each of the vertices coordinates.
(wanted to know what are anyone's thoughts about the matter before i might just solve this in a PR)
To Reproduce
Steps to reproduce the behavior:
Code
Expected behavior
There are almost three identical cubes, applying
mergeVertices()
on their positions (after removing normals and uvs).The difference between them is the x-coordinate of the first vertex: the top is 0.500001, the middle is 0.5 and the bottom has 0.499999.
Since the difference is well below the 1e-4 default threshold of mergeVertices() we would expect the vertices to be merged the same way.
But the top and middle vertex is merged to its neighbors (since they are truncated to 0.5) and the bottom is not merged (since it is truncated to 0.4999)
Screenshots
Platform:
(the problem exists on all platforms)
The text was updated successfully, but these errors were encountered: