mirror of
https://github.com/penpot/penpot.git
synced 2025-01-24 15:39:50 -05:00
Simplify and fix kdtree impl.
This commit is contained in:
parent
d64253dd1b
commit
808c964d3d
1 changed files with 23 additions and 179 deletions
202
vendor/kdtree.js
vendored
202
vendor/kdtree.js
vendored
|
@ -15,6 +15,7 @@ goog.provide("kdtree");
|
||||||
goog.provide("kdtree.KDTree");
|
goog.provide("kdtree.KDTree");
|
||||||
|
|
||||||
goog.require('goog.array');
|
goog.require('goog.array');
|
||||||
|
goog.require('goog.structs.Heap');
|
||||||
goog.require('goog.asserts');
|
goog.require('goog.asserts');
|
||||||
|
|
||||||
goog.scope(function() {
|
goog.scope(function() {
|
||||||
|
@ -186,43 +187,38 @@ goog.scope(function() {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
nearest(point, maxNodes, maxDistance) {
|
nearest(point, maxNodes) {
|
||||||
let i, result, bestNodes;
|
|
||||||
|
|
||||||
if (maxNodes === undefined) {
|
if (maxNodes === undefined) {
|
||||||
maxNodes = 1;
|
maxNodes = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
bestNodes = new BinaryHeap(function (e) { return -e[1]; });
|
let best = new goog.structs.Heap();
|
||||||
|
|
||||||
const nearestSearch = (node) => {
|
const nearestSearch = (node) => {
|
||||||
const ownDistance = this.metric(point, node.obj);
|
let bestChild;
|
||||||
|
const distance = this.metric(point, node.obj);
|
||||||
const dimension = node.dimension;
|
const dimension = node.dimension;
|
||||||
const linearPoint = [null, null];
|
|
||||||
|
|
||||||
let otherChild, linearDistance, bestChild, i;
|
if (best.getCount() < maxNodes || distance < best.peek()[1]) {
|
||||||
|
best.insert(-distance, [node.obj, distance]);
|
||||||
function saveNode(node, distance) {
|
if (best.getCount() > maxNodes) {
|
||||||
bestNodes.push([node, distance]);
|
best.remove();
|
||||||
if (bestNodes.size() > maxNodes) {
|
|
||||||
bestNodes.pop();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0; i < this.dimensions; i += 1) {
|
if (best.isEmpty()) {
|
||||||
if (i === node.dimension) {
|
best.insert(-distance, [node.obj, distance]);
|
||||||
linearPoint[i] = point[i];
|
} else {
|
||||||
} else {
|
if (distance < best.peek()[1]) {
|
||||||
linearPoint[i] = node.obj[i]
|
best.insert(-distance, [node.obj, distance]);
|
||||||
|
|
||||||
|
if (best.getCount() > maxNodes) {
|
||||||
|
best.remove();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
linearDistance = this.metric(linearPoint, node.obj);
|
|
||||||
|
|
||||||
if (node.right === null && node.left === null) {
|
if (node.right === null && node.left === null) {
|
||||||
if (bestNodes.size() < maxNodes || ownDistance < bestNodes.peek()[1]) {
|
|
||||||
saveNode(node, ownDistance);
|
|
||||||
}
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -232,47 +228,21 @@ goog.scope(function() {
|
||||||
bestChild = node.right;
|
bestChild = node.right;
|
||||||
} else {
|
} else {
|
||||||
if (point[dimension] < node.obj[dimension]) {
|
if (point[dimension] < node.obj[dimension]) {
|
||||||
bestChild = node.left;
|
bestChild = node.left;
|
||||||
} else {
|
} else {
|
||||||
bestChild = node.right;
|
bestChild = node.right;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
nearestSearch(bestChild);
|
nearestSearch(bestChild);
|
||||||
|
|
||||||
if (bestNodes.size() < maxNodes || ownDistance < bestNodes.peek()[1]) {
|
|
||||||
saveNode(node, ownDistance);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (bestNodes.size() < maxNodes || Math.abs(linearDistance) < bestNodes.peek()[1]) {
|
|
||||||
if (bestChild === node.left) {
|
|
||||||
otherChild = node.right;
|
|
||||||
} else {
|
|
||||||
otherChild = node.left;
|
|
||||||
}
|
|
||||||
if (otherChild !== null) {
|
|
||||||
nearestSearch(otherChild);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (maxDistance) {
|
|
||||||
for (i = 0; i < maxNodes; i += 1) {
|
|
||||||
bestNodes.push([null, maxDistance]);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if(this.root) {
|
if(this.root) {
|
||||||
nearestSearch(this.root);
|
nearestSearch(this.root);
|
||||||
}
|
}
|
||||||
|
|
||||||
result = [];
|
result = best.getValues();
|
||||||
|
result.sort((x) => x[1]);
|
||||||
for (i = 0; i < Math.min(maxNodes, bestNodes.content.length); i += 1) {
|
|
||||||
if (bestNodes.content[i][0]) {
|
|
||||||
result.push([bestNodes.content[i][0].obj, bestNodes.content[i][1]]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -295,134 +265,8 @@ goog.scope(function() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Binary heap implementation from:
|
|
||||||
// http://eloquentjavascript.net/appendix2.html
|
|
||||||
|
|
||||||
function BinaryHeap(scoreFunction){
|
|
||||||
this.content = [];
|
|
||||||
this.scoreFunction = scoreFunction;
|
|
||||||
}
|
|
||||||
|
|
||||||
BinaryHeap.prototype = {
|
|
||||||
push: function(element) {
|
|
||||||
// Add the new element to the end of the array.
|
|
||||||
this.content.push(element);
|
|
||||||
// Allow it to bubble up.
|
|
||||||
this.bubbleUp(this.content.length - 1);
|
|
||||||
},
|
|
||||||
|
|
||||||
pop: function() {
|
|
||||||
// Store the first element so we can return it later.
|
|
||||||
var result = this.content[0];
|
|
||||||
// Get the element at the end of the array.
|
|
||||||
var end = this.content.pop();
|
|
||||||
// If there are any elements left, put the end element at the
|
|
||||||
// start, and let it sink down.
|
|
||||||
if (this.content.length > 0) {
|
|
||||||
this.content[0] = end;
|
|
||||||
this.sinkDown(0);
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
},
|
|
||||||
|
|
||||||
peek: function() {
|
|
||||||
return this.content[0];
|
|
||||||
},
|
|
||||||
|
|
||||||
remove: function(node) {
|
|
||||||
var len = this.content.length;
|
|
||||||
// To remove a value, we must search through the array to find
|
|
||||||
// it.
|
|
||||||
for (var i = 0; i < len; i++) {
|
|
||||||
if (this.content[i] == node) {
|
|
||||||
// When it is found, the process seen in 'pop' is repeated
|
|
||||||
// to fill up the hole.
|
|
||||||
var end = this.content.pop();
|
|
||||||
if (i != len - 1) {
|
|
||||||
this.content[i] = end;
|
|
||||||
if (this.scoreFunction(end) < this.scoreFunction(node))
|
|
||||||
this.bubbleUp(i);
|
|
||||||
else
|
|
||||||
this.sinkDown(i);
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
throw new Error("Node not found.");
|
|
||||||
},
|
|
||||||
|
|
||||||
size: function() {
|
|
||||||
return this.content.length;
|
|
||||||
},
|
|
||||||
|
|
||||||
bubbleUp: function(n) {
|
|
||||||
// Fetch the element that has to be moved.
|
|
||||||
var element = this.content[n];
|
|
||||||
// When at 0, an element can not go up any further.
|
|
||||||
while (n > 0) {
|
|
||||||
// Compute the parent element's index, and fetch it.
|
|
||||||
var parentN = Math.floor((n + 1) / 2) - 1,
|
|
||||||
parent = this.content[parentN];
|
|
||||||
// Swap the elements if the parent is greater.
|
|
||||||
if (this.scoreFunction(element) < this.scoreFunction(parent)) {
|
|
||||||
this.content[parentN] = element;
|
|
||||||
this.content[n] = parent;
|
|
||||||
// Update 'n' to continue at the new position.
|
|
||||||
n = parentN;
|
|
||||||
}
|
|
||||||
// Found a parent that is less, no need to move it further.
|
|
||||||
else {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
sinkDown: function(n) {
|
|
||||||
// Look up the target element and its score.
|
|
||||||
var length = this.content.length,
|
|
||||||
element = this.content[n],
|
|
||||||
elemScore = this.scoreFunction(element);
|
|
||||||
|
|
||||||
while(true) {
|
|
||||||
// Compute the indices of the child elements.
|
|
||||||
var child2N = (n + 1) * 2, child1N = child2N - 1;
|
|
||||||
// This is used to store the new position of the element,
|
|
||||||
// if any.
|
|
||||||
var swap = null;
|
|
||||||
// If the first child exists (is inside the array)...
|
|
||||||
if (child1N < length) {
|
|
||||||
// Look it up and compute its score.
|
|
||||||
var child1 = this.content[child1N],
|
|
||||||
child1Score = this.scoreFunction(child1);
|
|
||||||
// If the score is less than our element's, we need to swap.
|
|
||||||
if (child1Score < elemScore)
|
|
||||||
swap = child1N;
|
|
||||||
}
|
|
||||||
// Do the same checks for the other child.
|
|
||||||
if (child2N < length) {
|
|
||||||
var child2 = this.content[child2N],
|
|
||||||
child2Score = this.scoreFunction(child2);
|
|
||||||
if (child2Score < (swap == null ? elemScore : child1Score)){
|
|
||||||
swap = child2N;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the element needs to be moved, swap it, and continue.
|
|
||||||
if (swap != null) {
|
|
||||||
this.content[n] = this.content[swap];
|
|
||||||
this.content[swap] = element;
|
|
||||||
n = swap;
|
|
||||||
}
|
|
||||||
// Otherwise, we are done.
|
|
||||||
else {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
function distance2d(a, b){
|
function distance2d(a, b){
|
||||||
return Math.pow(a[0] - b[0], 2) + Math.pow(a[1] - b[1], 2);
|
return Math.sqrt(Math.pow(a[0] - b[0], 2) + Math.pow(a[1] - b[1], 2));
|
||||||
}
|
}
|
||||||
|
|
||||||
function create2d(points) {
|
function create2d(points) {
|
||||||
|
|
Loading…
Add table
Reference in a new issue