-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfit_ellipse.py
59 lines (53 loc) · 1.94 KB
/
fit_ellipse.py
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
from numpy.linalg import eig, inv, svd
from math import atan2
import numpy as np
def __fit_ellipse(x, y):
x, y = x[:, np.newaxis], y[:, np.newaxis]
D = np.hstack((x * x, x * y, y * y, x, y, np.ones_like(x)))
S, C = np.dot(D.T, D), np.zeros([6, 6])
C[0, 2], C[2, 0], C[1, 1] = 2, 2, -1
U, s, V = svd(np.dot(inv(S), C))
a = U[:, 0]
return a
def ellipse_center(a):
b, c, d, f, g, a = a[1] / 2, a[2], a[3] / 2, a[4] / 2, a[5], a[0]
num = b * b - a * c
x0 = (c * d - b * f) / num
y0 = (a * f - b * d) / num
return np.array([x0, y0])
def ellipse_axis_length(a):
b, c, d, f, g, a = a[1] / 2, a[2], a[3] / 2, a[4] / 2, a[5], a[0]
up = 2 * (a * f * f + c * d * d + g * b * b - 2 * b * d * f - a * c * g)
down1 = (b * b - a * c) * (
(c - a) * np.sqrt(1 + 4 * b * b / ((a - c) * (a - c))) - (c + a)
)
down2 = (b * b - a * c) * (
(a - c) * np.sqrt(1 + 4 * b * b / ((a - c) * (a - c))) - (c + a)
)
res1 = np.sqrt(up / down1)
res2 = np.sqrt(up / down2)
return np.array([res1, res2])
def ellipse_angle_of_rotation(a):
b, c, d, f, g, a = a[1] / 2, a[2], a[3] / 2, a[4] / 2, a[5], a[0]
return atan2(2 * b, (a - c)) / 2
def fit_ellipse(x, y):
"""@brief fit an ellipse to supplied data points: the 5 params
returned are:
M - major axis length
m - minor axis length
cx - ellipse centre (x coord.)
cy - ellipse centre (y coord.)
phi - rotation angle of ellipse bounding box
@param x first coordinate of points to fit (array)
@param y second coord. of points to fit (array)
"""
a = __fit_ellipse(x, y)
centre = ellipse_center(a)
phi = ellipse_angle_of_rotation(a)
M, m = ellipse_axis_length(a)
# assert that the major axix M > minor axis m
if m > M:
M, m = m, M
# ensure the angle is betwen 0 and 2*pi
phi -= 2 * np.pi * int(phi / (2 * np.pi))
return [M, m, centre[0], centre[1], phi]