-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.js
114 lines (92 loc) · 3.25 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
var _ = require("lodash");
var sylvester = require("sylvester");
// TODO: Provide a way to go back to 3D
// - To go back from local coordinates (Lx, Ly) to 3D the transformation is
// p = locOrigin + Lx*locX + Ly*locY
// Input: [[x,y,z], [...]]
// Output: [[x,y], [x,y]]
// Source: http://stackoverflow.com/a/26370192/997339
var points3dto2d = function(points3d, zUP, parent) {
parent = (!parent) ? {} : parent;
var p0, p1, p2;
// TODO: Find a good value for this threshold
// - 0.01 still resulted in collinear issues
// - 0.05 seems to work well, though unsure if it causes false-positives
// - 0.08 covers most of the issues seen with collinear points
// TODO: Look at exposing this as an option rather than hard-coding it
var collinearThreshold = 0.05;
var nonCollinear;
// TODO: Try and pick points that will give the best non-collinear result
// Otherwise, fall back to looping over all points
//
// TODO: De-dupe this and the other checks after this
if (points3d.length > 5) {
var offset = Math.floor(points3d.length / 3);
p0 = $V(points3d[0]);
p1 = $V(points3d[offset]);
p2 = $V(points3d[offset*2]);
// Colinear or near-colinear?
var cross = p0.subtract(p1).cross(p0.subtract(p2));
// Exit if non-collinear points are found
if (Math.abs(cross.e(1)) > collinearThreshold || Math.abs(cross.e(2)) > collinearThreshold || Math.abs(cross.e(3)) > collinearThreshold) {
nonCollinear = true;
}
}
if (!nonCollinear) {
var firstPoint;
// Find first sequence of points that aren't collinear
_.each(points3d, function(point, pIndex) {
// Exit if no more points are available
if (pIndex === points3d.length - 1) {
return false;
}
if (pIndex === 0 && !firstPoint) {
firstPoint = $V(point);
return;
}
p0 = firstPoint;
p1 = $V(point);
p2 = $V(points3d[pIndex+1]);
// Colinear or near-colinear?
var cross = p0.subtract(p1).cross(p0.subtract(p2));
// Exit if non-collinear points are found
if (Math.abs(cross.e(1)) > collinearThreshold || Math.abs(cross.e(2)) > collinearThreshold || Math.abs(cross.e(3)) > collinearThreshold) {
return false;
}
});
}
var locOrigin = (parent.origin) ? parent.origin : p0.dup();
var locX = (parent.locX) ? parent.locX : p1.subtract(locOrigin);
var normal = (parent.normal) ? parent.normal : locX.cross(p2.subtract(locOrigin));
var locY = (parent.locY) ? parent.locY : normal.cross(locX);
var locXUnit = locX.toUnitVector();
var locYUnit = locY.toUnitVector();
var points2d = [];
_.each(points3d, function(point) {
var pointVec = $V(point);
var pointVecMinusOrigin;
var x, y;
if (!zUP) {
// Original code
pointVecMinusOrigin = pointVec.subtract(locOrigin);
x = pointVecMinusOrigin.dot(locXUnit);
y = pointVecMinusOrigin.dot(locYUnit);
} else {
// Flip axis
pointVecMinusOrigin = pointVec.subtract(locOrigin);
x = pointVecMinusOrigin.dot(locYUnit);
y = pointVecMinusOrigin.dot(locXUnit);
}
points2d.push([x, y]);
});
return {
points: points2d,
state: {
origin: locOrigin,
locX: locX,
locY: locY,
normal: normal
}
};
};
module.exports = points3dto2d;