forked from chanind/hanzi-writer
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgeometry.ts
223 lines (195 loc) · 7.47 KB
/
geometry.ts
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
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
import { Point } from './typings/types';
import { average, arrLast } from './utils';
export const subtract = (p1: Point, p2: Point) => ({ x: p1.x - p2.x, y: p1.y - p2.y });
export const magnitude = (point: Point) =>
Math.sqrt(Math.pow(point.x, 2) + Math.pow(point.y, 2));
export const distance = (point1: Point, point2: Point) =>
magnitude(subtract(point1, point2));
export const equals = (point1: Point, point2: Point) =>
point1.x === point2.x && point1.y === point2.y;
export const round = (point: Point, precision = 1) => {
const multiplier = precision * 10;
return {
x: Math.round(multiplier * point.x) / multiplier,
y: Math.round(multiplier * point.y) / multiplier,
};
};
export const length = (points: Point[]) => {
let lastPoint = points[0];
const pointsSansFirst = points.slice(1);
return pointsSansFirst.reduce((acc, point) => {
const dist = distance(point, lastPoint);
lastPoint = point;
return acc + dist;
}, 0);
};
export const cosineSimilarity = (point1: Point, point2: Point) => {
const rawDotProduct = point1.x * point2.x + point1.y * point2.y;
return rawDotProduct / magnitude(point1) / magnitude(point2);
};
/**
* return a new point, p3, which is on the same line as p1 and p2, but distance away
* from p2. p1, p2, p3 will always lie on the line in that order
*/
export const _extendPointOnLine = (p1: Point, p2: Point, dist: number) => {
const vect = subtract(p2, p1);
const norm = dist / magnitude(vect);
return { x: p2.x + norm * vect.x, y: p2.y + norm * vect.y };
};
/** based on http://www.kr.tuwien.ac.at/staff/eiter/et-archive/cdtr9464.pdf */
export const frechetDist = (curve1: Point[], curve2: Point[]) => {
const longCurve = curve1.length >= curve2.length ? curve1 : curve2;
const shortCurve = curve1.length >= curve2.length ? curve2 : curve1;
const calcVal = (
i: number,
j: number,
prevResultsCol: number[],
curResultsCol: number[],
): number => {
if (i === 0 && j === 0) {
return distance(longCurve[0], shortCurve[0]);
}
if (i > 0 && j === 0) {
return Math.max(prevResultsCol[0], distance(longCurve[i], shortCurve[0]));
}
const lastResult = curResultsCol[curResultsCol.length - 1];
if (i === 0 && j > 0) {
return Math.max(lastResult, distance(longCurve[0], shortCurve[j]));
}
return Math.max(
Math.min(prevResultsCol[j], prevResultsCol[j - 1], lastResult),
distance(longCurve[i], shortCurve[j]),
);
};
let prevResultsCol: number[] = [];
for (let i = 0; i < longCurve.length; i++) {
const curResultsCol: number[] = [];
for (let j = 0; j < shortCurve.length; j++) {
// we only need the results from i - 1 and j - 1 to continue the calculation
// so we only need to hold onto the last column of calculated results
// prevResultsCol is results[i-1][:] in the original algorithm
// curResultsCol is results[i][:j-1] in the original algorithm
curResultsCol.push(calcVal(i, j, prevResultsCol, curResultsCol));
}
prevResultsCol = curResultsCol;
}
return prevResultsCol[shortCurve.length - 1];
};
/** break up long segments in the curve into smaller segments of len maxLen or smaller */
export const subdivideCurve = (curve: Point[], maxLen = 0.05) => {
const newCurve = curve.slice(0, 1);
for (const point of curve.slice(1)) {
const prevPoint = newCurve[newCurve.length - 1];
const segLen = distance(point, prevPoint);
if (segLen > maxLen) {
const numNewPoints = Math.ceil(segLen / maxLen);
const newSegLen = segLen / numNewPoints;
for (let i = 0; i < numNewPoints; i++) {
newCurve.push(_extendPointOnLine(point, prevPoint, -1 * newSegLen * (i + 1)));
}
} else {
newCurve.push(point);
}
}
return newCurve;
};
/** redraw the curve using numPoints equally spaced out along the length of the curve */
export const outlineCurve = (curve: Point[], numPoints = 30) => {
const curveLen = length(curve);
const segmentLen = curveLen / (numPoints - 1);
const outlinePoints = [curve[0]];
const endPoint = arrLast(curve);
const remainingCurvePoints = curve.slice(1);
for (let i = 0; i < numPoints - 2; i++) {
let lastPoint: Point = arrLast(outlinePoints);
let remainingDist = segmentLen;
let outlinePointFound = false;
while (!outlinePointFound) {
const nextPointDist = distance(lastPoint, remainingCurvePoints[0]);
if (nextPointDist < remainingDist) {
remainingDist -= nextPointDist;
lastPoint = remainingCurvePoints.shift()!;
} else {
const nextPoint = _extendPointOnLine(
lastPoint,
remainingCurvePoints[0],
remainingDist - nextPointDist,
);
outlinePoints.push(nextPoint);
outlinePointFound = true;
}
}
}
outlinePoints.push(endPoint);
return outlinePoints;
};
/** translate and scale from https://en.wikipedia.org/wiki/Procrustes_analysis */
export const normalizeCurve = (curve: Point[]) => {
const outlinedCurve = outlineCurve(curve);
const meanX = average(outlinedCurve.map((point) => point.x));
const meanY = average(outlinedCurve.map((point) => point.y));
const mean = { x: meanX, y: meanY };
const translatedCurve = outlinedCurve.map((point) => subtract(point, mean));
const scale = Math.sqrt(
average([
Math.pow(translatedCurve[0].x, 2) + Math.pow(translatedCurve[0].y, 2),
Math.pow(arrLast(translatedCurve).x, 2) + Math.pow(arrLast(translatedCurve).y, 2),
]),
);
const scaledCurve = translatedCurve.map((point) => ({
x: point.x / scale,
y: point.y / scale,
}));
return subdivideCurve(scaledCurve);
};
// rotate around the origin
export const rotate = (curve: Point[], theta: number) => {
return curve.map((point) => ({
x: Math.cos(theta) * point.x - Math.sin(theta) * point.y,
y: Math.sin(theta) * point.x + Math.cos(theta) * point.y,
}));
};
// remove intermediate points that are on the same line as the points to either side
export const _filterParallelPoints = (points: Point[]) => {
if (points.length < 3) return points;
const filteredPoints = [points[0], points[1]];
points.slice(2).forEach((point) => {
const numFilteredPoints = filteredPoints.length;
const curVect = subtract(point, filteredPoints[numFilteredPoints - 1]);
const prevVect = subtract(
filteredPoints[numFilteredPoints - 1],
filteredPoints[numFilteredPoints - 2],
);
// this is the z coord of the cross-product. If this is 0 then they're parallel
const isParallel = curVect.y * prevVect.x - curVect.x * prevVect.y === 0;
if (isParallel) {
filteredPoints.pop();
}
filteredPoints.push(point);
});
return filteredPoints;
};
export function getPathString(points: Point[], close = false) {
const start = round(points[0]);
const remainingPoints = points.slice(1);
let pathString = `M ${start.x} ${start.y}`;
remainingPoints.forEach((point) => {
const roundedPoint = round(point);
pathString += ` L ${roundedPoint.x} ${roundedPoint.y}`;
});
if (close) {
pathString += 'Z';
}
return pathString;
}
/** take points on a path and move their start point backwards by distance */
export const extendStart = (points: Point[], dist: number) => {
const filteredPoints = _filterParallelPoints(points);
if (filteredPoints.length < 2) return filteredPoints;
const p1 = filteredPoints[1];
const p2 = filteredPoints[0];
const newStart = _extendPointOnLine(p1, p2, dist);
const extendedPoints = filteredPoints.slice(1);
extendedPoints.unshift(newStart);
return extendedPoints;
};