forked from swatbotics/apriltag
-
Notifications
You must be signed in to change notification settings - Fork 0
/
calibrate_camera.py
122 lines (89 loc) · 3.62 KB
/
calibrate_camera.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
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
from __future__ import print_function
import numpy as np
import cv2
import argparse
def main():
parser = argparse.ArgumentParser(
description='calibrate camera intrinsics using OpenCV')
parser.add_argument('filenames', metavar='IMAGE', nargs='+',
help='input image files')
parser.add_argument('-r', '--rows', metavar='N', type=int,
required=True,
help='# of chessboard corners in vertical direction')
parser.add_argument('-c', '--cols', metavar='N', type=int,
required=True,
help='# of chessboard corners in horizontal direction')
parser.add_argument('-s', '--size', metavar='NUM', type=float, default=1.0,
help='chessboard square size in user-chosen units (should not affect results)')
parser.add_argument('-d', '--show-detections', action='store_true',
help='show detections in window')
options = parser.parse_args()
if options.rows < options.cols:
patternsize = (options.cols, options.rows)
else:
patternsize = (options.rows, options.cols)
sz = options.size
x = np.arange(patternsize[0])*sz
y = np.arange(patternsize[1])*sz
xgrid, ygrid = np.meshgrid(x, y)
zgrid = np.zeros_like(xgrid)
opoints = np.dstack((xgrid, ygrid, zgrid)).reshape((-1, 1, 3)).astype(np.float32)
imagesize = None
win = 'Calibrate'
cv2.namedWindow(win)
ipoints = []
for filename in options.filenames:
rgb = cv2.imread(filename)
if rgb is None:
print('warning: error opening {}, skipping'.format(filename))
continue
cursize = (rgb.shape[1], rgb.shape[0])
if imagesize is None:
imagesize = cursize
else:
assert imagesize == cursize
print('loaded ' + filename + ' of size {}x{}'.format(*imagesize))
if len(rgb.shape) == 3:
gray = cv2.cvtColor(rgb, cv2.COLOR_RGB2GRAY)
else:
gray = rgb
retval, corners = cv2.findChessboardCorners(gray, patternsize)
if options.show_detections:
display = cv2.cvtColor(gray, cv2.COLOR_GRAY2RGB)
cv2.drawChessboardCorners(display, patternsize, corners, retval)
cv2.imshow(win, display)
while cv2.waitKey(5) not in range(128): pass
if not retval:
print('warning: no chessboard found in {}, skipping'.format(filename))
else:
ipoints.append( corners )
flags = (cv2.CALIB_ZERO_TANGENT_DIST |
cv2.CALIB_FIX_K1 |
cv2.CALIB_FIX_K2 |
cv2.CALIB_FIX_K3 |
cv2.CALIB_FIX_K4 |
cv2.CALIB_FIX_K5 |
cv2.CALIB_FIX_K6)
opoints = [opoints] * len(ipoints)
retval, K, dcoeffs, rvecs, tvecs = cv2.calibrateCamera(opoints, ipoints, imagesize,
cameraMatrix=None,
distCoeffs=np.zeros(5),
flags=flags)
assert( np.all(dcoeffs == 0) )
fx = K[0,0]
fy = K[1,1]
cx = K[0,2]
cy = K[1,2]
params = (fx, fy, cx, cy)
print()
print('all units below measured in pixels:')
print(' fx = {}'.format(K[0,0]))
print(' fy = {}'.format(K[1,1]))
print(' cx = {}'.format(K[0,2]))
print(' cy = {}'.format(K[1,2]))
print()
print('pastable into Python:')
print(' fx, fy, cx, cy = {}'.format(repr(params)))
print()
if __name__ == '__main__':
main()