QuickHull is a well known algorithm for generating a convex hull given a set of points. The expected complexity is O(n⋅log(n)), with a worst case complexity of O(n2). The algorithm generalizes nicely into 3D, but for now let's just focus on the simpler 2D case.
Algorithm
The algorithm is fairly straightforward:- Create an edge by finding the min and max points along some axis (in my case I used the x-axis.)
- For each edge in the hull (initially only one) we assign a set of external points that can "see" the edge.
- For each set of points, find the point farthest away from its corresponding edge.
- The point farthest away (out of all the sets) is added to the convex hull by splitting its associated edge.
- The remaining points in the edge's set are now re-evaluated. They may either be inside the hull or belong to one of the two newly created edges
- The algorithm terminates when there are no more external points
Implementation
This algorithm has an expected complexity of O(n⋅log(n)). To actually get this there are some implementation details worth mentioning. First thing to note is that the convex hull is stored as a linked list. This allows for cheap insertion as well as easily traversing the hull.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
QuickHull.Node = function(v) | |
{ | |
this.v = v; | |
this.conflicts = []; | |
this.next = undefined; | |
}; | |
QuickHull.Node.prototype.Insert = function(v) | |
{ | |
var n = this.next; | |
this.next = new QuickHull.Node(v); | |
this.next.next = n; | |
return this.next; | |
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
//find farthest point from all segments | |
var startNode = undefined; | |
var farthestIndex = -1; | |
var maxDistance = 0; | |
convexNode = convexHull; | |
do | |
{ | |
if(convexNode.cachedInfo == undefined) //we only want to compute this if something in the set has changed | |
{ | |
convexNode.cachedInfo = this.FarthestPointToSegment(convexNode.v, convexNode.next.v, convexNode.conflicts); | |
} | |
var info = convexNode.cachedInfo; | |
var d2 = info[1]; | |
if(d2 >= maxDistance) | |
{ | |
maxDistance = d2; | |
farthestIndex = info[0]; | |
startNode = convexNode; | |
} | |
convexNode = convexNode.next; | |
}while(convexNode != convexHull); | |
var newHullPt = startNode.conflicts[farthestIndex]; | |
startNode.cachedInfo = undefined; //we're splitting the edge so the points info is no longer valid | |
startNode.Insert(newHullPt); | |
var conflicts = startNode.conflicts; | |
conflicts.splice(farthestIndex, 1); | |
pointsLeft-= 1; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
//now fixup conflicts in startNode based on the added triangle | |
for(var i=0; i<conflicts.length; i++) | |
{ | |
var pt = conflicts[i]; | |
var uvw = Barycentric(pt, startNode.v, startNode.next.v, startNode.next.next.v); | |
var u = uvw[0]; | |
var v = uvw[1]; | |
var w = uvw[2]; | |
//use barycentric coordinates to determine point location | |
// startNode (u =1) | |
// /\ | |
// / \ | |
// / \ | |
// /______\ startNode.next (v=1) | |
// next.next (w=1) | |
if(u >= 0 && v >= 0 && w >= 0) //pt is in new triangle | |
{ | |
//pt is finally covered by hull so remove it from pts | |
conflicts.splice(i, 1); | |
i--; | |
pointsLeft -= 1; | |
}else if(v < 0) //edge between next next and start | |
{ | |
//this can only happen in first triangle | |
startNode.next.next.conflicts.push(pt); | |
conflicts.splice(i,1); | |
startNode.next.next.cachedInfo = undefined; //modifying conflicts list so cache is invalid | |
i--; | |
}else if(u < 0) //edge between next and next next | |
{ | |
startNode.next.conflicts.push(pt); | |
conflicts.splice(i,1); | |
startNode.next.cachedInfo = undefined; //modifying conflicts list so cache is invalid | |
i--; | |
} | |
} |
That's it! Hope this has been useful for someone.