惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

爱范儿
爱范儿
Security Latest
Security Latest
NISL@THU
NISL@THU
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
C
Cybersecurity and Infrastructure Security Agency CISA
Cloudbric
Cloudbric
T
Threat Research - Cisco Blogs
大猫的无限游戏
大猫的无限游戏
C
CXSECURITY Database RSS Feed - CXSecurity.com
阮一峰的网络日志
阮一峰的网络日志
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More
雷峰网
雷峰网
C
Cisco Blogs
V
Vulnerabilities – Threatpost
S
Security Archives - TechRepublic
V
Visual Studio Blog
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
cs.AI updates on arXiv.org
cs.AI updates on arXiv.org
J
Java Code Geeks
D
Darknet – Hacking Tools, Hacker News & Cyber Security
Know Your Adversary
Know Your Adversary
博客园 - 叶小钗
腾讯CDC
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
P
Privacy International News Feed
P
Palo Alto Networks Blog
博客园_首页
V
V2EX
WordPress大学
WordPress大学
Schneier on Security
Schneier on Security
月光博客
月光博客
博客园 - 司徒正美
Google DeepMind News
Google DeepMind News
TaoSecurity Blog
TaoSecurity Blog
博客园 - 聂微东
酷 壳 – CoolShell
酷 壳 – CoolShell
人人都是产品经理
人人都是产品经理
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
博客园 - 【当耐特】
The Cloudflare Blog
罗磊的独立博客
美团技术团队
N
News | PayPal Newsroom
K
KPMG report finds enterprise disconnect between AI and its ROI | CIO
Last Week in AI
Last Week in AI
K
Kaspersky official blog
Google Online Security Blog
Google Online Security Blog
S
SegmentFault 最新的问题
Application and Cybersecurity Blog
Application and Cybersecurity Blog
T
Tailwind CSS Blog

又见苍岚

COLMAP PatchMatch Stereo 算法详解 事件驱动的状态机框架:从理论到工程实践 Git 在国内网络环境下无法 Push 的排查与修复 —— 配置 Clash 代理 分段五次多项式插值原理详解 路径插值方法深度对比研究 Claude Code 使用指南 OpenClaw 记忆管理与技能创建指南 CBS(Conflict-Based Search)算法详解 A* 算法及其变种详解 OpenClaw 配置多 Agents Windows Powershell 无法加载文件,因为在此系统上禁止运行脚本问题的解决方案 MaxClaw 安装流程 大模型 AI 名词介绍 AList 网盘聚合工具简介 Protobuf 简介与测试 Claude Code 简介以及 GLM 4.7 模型接入 Github 歌词下载工具 163MusicLyrics Python __getattr__ 懒加载 Python TypedDict 机器人仿真平台 Gazebo 安装记录 机器人仿真平台 Gazebo 简介 多机器人路径规划问题(Multi-Agent Path Finding, MAPF)简介 Python exifread 读取修改过的 jpeg 信息错误问题修复 3D 坐标系变换的理解 3D 旋转矩阵基本概念 MongoDB Compass 介绍 Python 环境管理工具 uv Flutter 开发指南 Snipaste 安装下载与黑屏问题解决方案 全局路径规划算法记录 2025 Python 版本性能测试 Flutter Hello World Flutter 安装环境配置 Ubuntu VMware 硬盘扩容后 SMBus Host controller not enabled 报错问题解决 Python NetworkX 教程 Docker GPU 报错 - Failed to initialize NVML Unknown Error 解决方案 Python matplotlib 图表绘制 cuda-toolkit 安装替代 Cuda 与 Cudnn Jinja2 Python 利用 docxtpl 和 Jinja2 生成基于模板的 Word 文档 Docker 实现 CPU 核心隔离 LoFTR 基于 Transformer 的特征提取匹配算法 OmniGlue 特征匹配 SuperGlue 使用图神经网络学习特征匹配 Ubuntu 下将 xlsx 文件按照 sheet 转换为 图片 Python 使用 SQLAlchemy Python FastAPI 教程 openwrt 软路由配置安装 Nav2 地图文件(PGM/YAML)规范标准 3D OBJ 模型转换为 glb 瓦片格式 Python 源码 Redis 数据库介绍 Ubuntu 22.04 内核自动升级导致 MongoDB 7.0.12 错误记录 ubuntu 20.04 安装 ROS Noetic ubuntu 18.04 安装 ROS Melodic VMware Workstation Pro 个人免费版下载、安装、使用指南 Hybrid A-star 路径规划 Reeds-Shepp 曲线 Dubins 曲线 Linux kvm 虚拟机网络不通的问题解决方法 Ubuntu 自动内存清理 BiliBili 缓存视频转 mp4 Python 求解线性规划 3D Gaussian Splatting 官方源码实践记录 ImageMagick 教程 Ubuntu 22.04 安装 Colmap 对数几率 odds Ubuntu nmcli 网络管理工具使用指南 SuperPoint 自监督深度学习特征点提取 SyncTV Music Tag Web 在线音乐信息整理工具 ncm 格式转 mp3 MusicBrainz 音乐元数据百科数据库 Ubuntu 网络流量监控工具 私人云音乐平台 Navidrome 入门 手眼标定 四元数(Quaternions) OHTTPS 实现免费自动 https 证书申请、更新、部署 ubuntu 22.04 安装 CloudCompare 单机 KVM 虚拟机冷迁移 Ubuntu 22.04 使用 mdadm 实现软 raid 小鱼 一键安装 ROS-humble Fluid -46- 基于 Simpletex API 构建公式识别页面 公式识别 API 简介 -- Simpletex 使用 Python web 部署库 waitress 3D Gaussian Splatting for Real-Time Radiance Field Rendering Ubuntu Swap 简介与空间扩展 Ubuntu 24.04 安装 forticlient Clash Verge 使用 MongoDB 7.0.17 集群 Docker 构建源码 Error code - 2013. Lost connection to MySQL server during query 问题解决 Python 日志记录库 loguru 使用指北 Python 实现 Web 日志查看服务 MySQL LOAD DATA LOCAL INFILE 极速数据加载 Image size exceeds limit of 89478485 pixels 解决方案 Docker 使用 NVIDIA GPU 驱动错误解决 阿里云 docker 镜像仓库 Ubuntu中没有wired connected的解决方案 MinIO 简介 subconverter 代理订阅格式转换 修复 node –openssl-legacy-provider is not allowed in NODE_OPTIONS 错误
鱼眼相机畸变矫正
Yiwei Zhang · 2024-10-10 · via 又见苍岚

$$
\theta_d=\theta(1+k_1\theta2+k_2\theta4+k_3\theta6+k_4\theta8)
$$
也就是说畸变矫正的目的就是求解这 4 个参数。

Python OpenCV 实现鱼眼相机畸变矫正,数据包 在此,解压出 cali-data 包,运行程序进行畸变矫正,输出 camera_intrinsic.yaml

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
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
import argparse
import cv2
import numpy as np
import os
import vvdutils as vv

W,H=None,None
aa = "python intrinsicCalib.py -input image -path /home/pengmy/Downloads/fishimg/20240815 -image "
parser = argparse.ArgumentParser(description="Camera Intrinsic Calibration")
parser.add_argument('-input', '--INPUT_TYPE', default='image', type=str, help='Input Source: camera/video/image')
parser.add_argument('-output', '--OUTPUT', default='camera_intrinsic.yaml', type=str, help='output file path (eg.: ../config/camera_intrinsic.yaml')
parser.add_argument('-type', '--CAMERA_TYPE', default='fisheye', type=str, help='Camera Type: fisheye/normal')
parser.add_argument('-id', '--CAMERA_ID', default=1, type=int, help='Camera ID')
parser.add_argument('-path', '--INPUT_PATH', default='cali-data', type=str, help='Input Video/Image Path')
parser.add_argument('-video', '--VIDEO_FILE', default='video.mp4', type=str, help='Input Video File Name (eg.: video.mp4)')
parser.add_argument('-image', '--IMAGE_FILE', default='img_raw', type=str, help='Input Image File Name Prefix (eg.: img_raw)')
parser.add_argument('-mode', '--SELECT_MODE', default='auto', type=str, help='Image Select Mode: auto/manual')
parser.add_argument('-fw','--FRAME_WIDTH', default=1920, type=int, help='Camera Frame Width')
parser.add_argument('-fh','--FRAME_HEIGHT', default=1920, type=int, help='Camera Frame Height')
parser.add_argument('-bw','--BORAD_WIDTH', default=11, type=int, help='Chess Board Width (corners number)')
parser.add_argument('-bh','--BORAD_HEIGHT', default=8, type=int, help='Chess Board Height (corners number)')
parser.add_argument('-size','--SQUARE_SIZE', default=25 , type=int, help='Chess Board Square Size (mm)')
parser.add_argument('-num','--CALIB_NUMBER', default=2, type=int, help='Least Required Calibration Frame Number')
parser.add_argument('-delay','--FRAME_DELAY', default=12, type=int, help='Capture Image Time Interval (frame number)')
parser.add_argument('-subpix','--SUBPIX_REGION', default=5, type=int, help='Corners Subpix Optimization Region')
parser.add_argument('-fps','--CAMERA_FPS', default=20, type=int, help='Camera Frame per Second(FPS)')
parser.add_argument('-fs', '--FOCAL_SCALE', default=1, type=float, help='Camera Undistort Focal Scale')
parser.add_argument('-ss', '--SIZE_SCALE', default=1, type=float, help='Camera Undistort Size Scale')
parser.add_argument('-store','--STORE_FLAG', default=False, type=bool, help='Store Captured Images (Ture/False)')
parser.add_argument('-store_path', '--STORE_PATH', default='./data/', type=str, help='Path to Store Captured Images')
parser.add_argument('-crop','--CROP_FLAG', default=False, type=bool, help='Crop Input Video/Image to (fw,fh) (Ture/False)')
parser.add_argument('-resize','--RESIZE_FLAG', default=False, type=bool, help='Resize Input Video/Image to (fw,fh) (Ture/False)')
args = parser.parse_args()

class CalibData:
def __init__(self):
self.type = None
self.camera_mat = None
self.dist_coeff = None
self.iamge_size = None
self.rvecs = None
self.tvecs = None
self.map1 = None
self.map2 = None
self.reproj_err = None
self.ok = False

class Fisheye:
def __init__(self):
self.data = CalibData()
self.inited = False
self.BOARD = np.array([ [(j * args.SQUARE_SIZE, i * args.SQUARE_SIZE, 0.)]
for i in range(args.BORAD_HEIGHT)
for j in range(args.BORAD_WIDTH) ],dtype=np.float32)

def update(self, corners, frame_size):
board = [self.BOARD] * len(corners)
if not self.inited:
self._update_init(board, corners, frame_size)
self.inited = True
else:
self._update_refine(board, corners, frame_size)
self._calc_reproj_err(corners)
self._get_undistort_maps()

def _update_init(self, board, corners, frame_size):
data = self.data
data.type = "FISHEYE"
data.camera_mat = np.eye(3, 3)
data.dist_coeff = np.zeros((4, 1))
data.ok, data.camera_mat, data.dist_coeff, data.rvecs, data.tvecs = cv2.fisheye.calibrate(
board, corners, frame_size, data.camera_mat, data.dist_coeff,
flags=cv2.fisheye.CALIB_FIX_SKEW|cv2.fisheye.CALIB_RECOMPUTE_EXTRINSIC,
criteria=(cv2.TERM_CRITERIA_EPS + cv2.TERM_CRITERIA_COUNT, 30, 1e-6))
data.ok = data.ok and cv2.checkRange(data.camera_mat) and cv2.checkRange(data.dist_coeff)

def _update_refine(self, board, corners, frame_size):
data = self.data
data.ok, data.camera_mat, data.dist_coeff, data.rvecs, data.tvecs = cv2.fisheye.calibrate(
board, corners, frame_size, data.camera_mat, data.dist_coeff,
flags=cv2.fisheye.CALIB_FIX_SKEW|cv2.fisheye.CALIB_RECOMPUTE_EXTRINSIC|cv2.CALIB_USE_INTRINSIC_GUESS,
criteria=(cv2.TERM_CRITERIA_EPS + cv2.TERM_CRITERIA_COUNT, 10, 1e-6))
data.ok = data.ok and cv2.checkRange(data.camera_mat) and cv2.checkRange(data.dist_coeff)

def _calc_reproj_err(self, corners):
if not self.inited: return
data = self.data
data.reproj_err = []
for i in range(len(corners)):
print(i, data.rvecs[i])
corners_reproj, _ = cv2.fisheye.projectPoints(self.BOARD, data.rvecs[i], data.tvecs[i], data.camera_mat, data.dist_coeff)
err = cv2.norm(corners_reproj, corners[i], cv2.NORM_L2) / len(corners_reproj)
data.reproj_err.append(err)

def _get_camera_mat_dst(self, camera_mat):
camera_mat_dst = camera_mat.copy()
camera_mat_dst[0][0] *= args.FOCAL_SCALE
camera_mat_dst[1][1] *= args.FOCAL_SCALE
camera_mat_dst[0][2] = args.FRAME_WIDTH / 2 * args.SIZE_SCALE
camera_mat_dst[1][2] = args.FRAME_HEIGHT / 2 * args.SIZE_SCALE
return camera_mat_dst

def _get_undistort_maps(self):
data = self.data
camera_mat_dst = self._get_camera_mat_dst(data.camera_mat)
data.map1, data.map2 = cv2.fisheye.initUndistortRectifyMap(
data.camera_mat, data.dist_coeff, np.eye(3, 3), camera_mat_dst,
(int(args.FRAME_WIDTH * args.SIZE_SCALE), int(args.FRAME_HEIGHT * args.SIZE_SCALE)), cv2.CV_16SC2)

class Normal:
def __init__(self):
self.data = CalibData()
self.inited = False
self.BOARD = np.array([ [(j * args.SQUARE_SIZE, i * args.SQUARE_SIZE, 0.)]
for i in range(args.BORAD_HEIGHT)
for j in range(args.BORAD_WIDTH) ],dtype=np.float32)

def update(self, corners, frame_size):
board = [self.BOARD] * len(corners)
if not self.inited:
self._update_init(board, corners, frame_size)
self.inited = True
else:
self._update_refine(board, corners, frame_size)
self._calc_reproj_err(corners)
self._get_undistort_maps()

def _update_init(self, board, corners, frame_size):
data = self.data
data.type = "NORMAL"
data.camera_mat = np.eye(3, 3)
data.dist_coeff = np.zeros((5, 1))
data.ok, data.camera_mat, data.dist_coeff, data.rvecs, data.tvecs = cv2.calibrateCamera(
board, corners, frame_size, data.camera_mat, data.dist_coeff,
criteria=(cv2.TERM_CRITERIA_EPS + cv2.TERM_CRITERIA_COUNT, 30, 1e-6))
data.ok = data.ok and cv2.checkRange(data.camera_mat) and cv2.checkRange(data.dist_coeff)

def _update_refine(self, board, corners, frame_size):
data = self.data
data.ok, data.camera_mat, data.dist_coeff, data.rvecs, data.tvecs = cv2.calibrateCamera(
board, corners, frame_size, data.camera_mat, data.dist_coeff,
flags = cv2.CALIB_USE_INTRINSIC_GUESS,
criteria=(cv2.TERM_CRITERIA_EPS + cv2.TERM_CRITERIA_COUNT, 10, 1e-6))
data.ok = data.ok and cv2.checkRange(data.camera_mat) and cv2.checkRange(data.dist_coeff)

def _calc_reproj_err(self, corners):
if not self.inited: return
data = self.data
data.reproj_err = []
for i in range(len(corners)):
corners_reproj, _ = cv2.projectPoints(self.BOARD, data.rvecs[i], data.tvecs[i], data.camera_mat, data.dist_coeff)
err = cv2.norm(corners_reproj, corners[i], cv2.NORM_L2) / len(corners_reproj)
data.reproj_err.append(err)

def _get_camera_mat_dst(self, camera_mat):
camera_mat_dst = camera_mat.copy()
camera_mat_dst[0][0] *= args.FOCAL_SCALE
camera_mat_dst[1][1] *= args.FOCAL_SCALE
camera_mat_dst[0][2] = args.FRAME_WIDTH / 2 * args.SIZE_SCALE
camera_mat_dst[1][2] = args.FRAME_HEIGHT / 2 * args.SIZE_SCALE
return camera_mat_dst

def _get_undistort_maps(self):
data = self.data
camera_mat_dst = self._get_camera_mat_dst(data.camera_mat)
data.map1, data.map2 = cv2.initUndistortRectifyMap(
data.camera_mat, data.dist_coeff, np.eye(3, 3), camera_mat_dst,
(int(args.FRAME_WIDTH * args.SIZE_SCALE), int(args.FRAME_HEIGHT * args.SIZE_SCALE)), cv2.CV_16SC2)

class InCalibrator:
def __init__(self, camera):
if camera == 'fisheye':
self.camera = Fisheye()
elif camera == 'normal':
self.camera = Normal()
else:
raise Exception("camera should be fisheye/normal")
self.corners = []

@staticmethod
def get_args():
return args

def get_corners(self, img):
ok, corners = cv2.findChessboardCorners(img, (args.BORAD_WIDTH, args.BORAD_HEIGHT),
flags = cv2.CALIB_CB_ADAPTIVE_THRESH|cv2.CALIB_CB_NORMALIZE_IMAGE|cv2.CALIB_CB_FAST_CHECK)
if ok:
gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
corners = cv2.cornerSubPix(gray, corners, (args.SUBPIX_REGION, args.SUBPIX_REGION), (-1, -1),
(cv2.TERM_CRITERIA_EPS + cv2.TERM_CRITERIA_MAX_ITER, 30, 0.01))
return ok, corners

def draw_corners(self, img):
ok, corners = self.get_corners(img)
cv2.drawChessboardCorners(img, (args.BORAD_WIDTH, args.BORAD_HEIGHT), corners, ok)
return img

def undistort(self, img):
data = self.camera.data
return cv2.remap(img, data.map1, data.map2, cv2.INTER_LINEAR)

def calibrate(self, img):
if len(self.corners) >= args.CALIB_NUMBER:
self.camera.update(self.corners, img.shape[1::-1])
return self.camera.data

def __call__(self, raw_frame):
ok, corners = self.get_corners(raw_frame)
result = self.camera.data
if ok:
self.corners.append(corners)
result = self.calibrate(raw_frame)
return result

def centerCrop(img,width,height):
if img.shape[1] < width or img.shape[0] < height:
raise Exception("CROP size should be smaller than original size")
img = img[round((img.shape[0]-height)/2) : round((img.shape[0]-height)/2)+height,
round((img.shape[1]-width)/2) : round((img.shape[1]-width)/2)+width ]
return img

class CalibMode():
def __init__(self, calibrator, input_type, mode):
self.calibrator = calibrator
self.input_type = input_type
self.mode = mode

def imgPreprocess(self, img):
if args.CROP_FLAG:
img = centerCrop(img, args.FRAME_WIDTH, args.FRAME_HEIGHT)
elif args.RESIZE_FLAG:
img = cv2.resize(img, (args.FRAME_WIDTH, args.FRAME_HEIGHT))
return img

def setCamera(self, cap):
cap.set(cv2.CAP_PROP_FOURCC, cv2.VideoWriter.fourcc('M','J','P','G'))
cap.set(cv2.CAP_PROP_FRAME_WIDTH, args.FRAME_WIDTH)
cap.set(cv2.CAP_PROP_FRAME_HEIGHT, args.FRAME_HEIGHT)
cap.set(cv2.CAP_PROP_FPS, args.CAMERA_FPS)
return cap

def runCalib(self, raw_frame, display_raw=True, display_undist=False):
calibrator = self.calibrator
raw_frame = self.imgPreprocess(raw_frame)
result = calibrator(raw_frame)
raw_frame = calibrator.draw_corners(raw_frame)
if display_raw:
cv2.namedWindow("raw_frame", flags = cv2.WINDOW_NORMAL | cv2.WINDOW_KEEPRATIO)
cv2.imshow("raw_frame", raw_frame)
if len(calibrator.corners) > args.CALIB_NUMBER and display_undist:
undist_frame = calibrator.undistort(raw_frame)
cv2.namedWindow("undist_frame", flags = cv2.WINDOW_NORMAL | cv2.WINDOW_KEEPRATIO)
cv2.imshow("undist_frame", undist_frame)
cv2.waitKey(500)
return result

def imageAutoMode(self):
global W,H
filenames = vv.glob_images(args.INPUT_PATH)
for filename in filenames:
print(filename)
raw_frame = cv2.imread(filename)
H,W,_ =raw_frame.shape
result = self.runCalib(raw_frame)
key = cv2.waitKey(1)
if key == 27: break
cv2.destroyAllWindows()
return result

def imageManualMode(self):
filenames = vv.glob_images(args.INPUT_PATH)
for filename in filenames:
print(filename)
raw_frame = cv2.imread(filename)
raw_frame = self.imgPreprocess(raw_frame)
img = raw_frame.copy()
img = self.calibrator.draw_corners(img)
display = "raw_frame: press SPACE to SELECT, other key to SKIP, press ESC to QUIT"
cv2.namedWindow(display, flags = cv2.WINDOW_NORMAL | cv2.WINDOW_KEEPRATIO)
cv2.imshow(display, img)
key = cv2.waitKey(0)
if key == 32:
result = self.runCalib(raw_frame, display_raw = False)
if key == 27: break
cv2.destroyAllWindows()
return result

def videoAutoMode(self):
cap = cv2.VideoCapture(args.INPUT_PATH + args.VIDEO_FILE)
if not cap.isOpened():
raise Exception("from {} read video failed".format(args.INPUT_PATH + args.VIDEO_FILE))
frame_id = 0
while True:
ok, raw_frame = cap.read()
raw_frame = self.imgPreprocess(raw_frame)
if frame_id % args.FRAME_DELAY == 0:
if args.STORE_FLAG:
cv2.imwrite(args.STORE_PATH + 'img_raw{}.jpg'.format(len(self.calibrator.corners)), raw_frame)
result = self.runCalib(raw_frame)
print(len(self.calibrator.corners))
frame_id += 1
key = cv2.waitKey(1)
if key == 27: break
cap.release()
cv2.destroyAllWindows()
return result

def videoManualMode(self):
cap = cv2.VideoCapture(args.INPUT_PATH + args.VIDEO_FILE)
if not cap.isOpened():
raise Exception("from {} read video failed".format(args.INPUT_PATH + args.VIDEO_FILE))
while True:
key = cv2.waitKey(1)
ok, raw_frame = cap.read()
raw_frame = self.imgPreprocess(raw_frame)
display = "raw_frame: press SPACE to capture image"
cv2.namedWindow(display, flags = cv2.WINDOW_NORMAL | cv2.WINDOW_KEEPRATIO)
cv2.imshow(display, raw_frame)
if key == 32:
if args.STORE_FLAG:
cv2.imwrite(args.STORE_PATH + 'img_raw{}.jpg'.format(len(self.calibrator.corners)), raw_frame)
result = self.runCalib(raw_frame)
print(len(self.calibrator.corners))
if key == 27: break
cap.release()
cv2.destroyAllWindows()
return result

def cameraAutoMode(self):
cap = cv2.VideoCapture(args.CAMERA_ID)
if not cap.isOpened():
raise Exception("from {} read video failed".format(args.CAMERA_ID))
cap = self.setCamera(cap)
frame_id = 0
start_flag = False
while True:
key = cv2.waitKey(1)
ok, raw_frame = cap.read()
raw_frame = self.imgPreprocess(raw_frame)
if key == 32: start_flag = True
if key == 27: break
if not start_flag:
cv2.putText(raw_frame, 'press SPACE to start!', (args.FRAME_WIDTH//4,args.FRAME_HEIGHT//2),
cv2.FONT_HERSHEY_COMPLEX, 1.5, (0,0,255), 2)
cv2.imshow("raw_frame", raw_frame)
continue
if frame_id % args.FRAME_DELAY == 0:
if args.STORE_FLAG:
cv2.imwrite(args.STORE_PATH + 'img_raw{}.jpg'.format(len(self.calibrator.corners)), raw_frame)
result = self.runCalib(raw_frame)
print(len(self.calibrator.corners))
frame_id += 1
cap.release()
cv2.destroyAllWindows()
return result

def cameraManualMode(self):
cap = cv2.VideoCapture(args.CAMERA_ID)
if not cap.isOpened():
raise Exception("from {} read video failed".format(args.CAMERA_ID))
cap = self.setCamera(cap)
while True:
key = cv2.waitKey(1)
ok, raw_frame = cap.read()
raw_frame = self.imgPreprocess(raw_frame)
display = "raw_frame: press SPACE to capture image"
cv2.namedWindow(display, flags = cv2.WINDOW_NORMAL | cv2.WINDOW_KEEPRATIO)
cv2.imshow(display, raw_frame)
if key == 32:
if args.STORE_FLAG:
cv2.imwrite(args.STORE_PATH + 'img_raw{}.jpg'.format(len(self.calibrator.corners)), raw_frame)
result = self.runCalib(raw_frame)
print(len(self.calibrator.corners))
if key == 27: break
cap.release()
cv2.destroyAllWindows()
return result

def __call__(self):
input_type = self.input_type
mode = self.mode
if input_type == 'image' and mode == 'auto':
result = self.imageAutoMode()
if input_type == 'image' and mode == 'manual':
result = self.imageManualMode()
if input_type == 'video' and mode == 'auto':
result = self.videoAutoMode()
if input_type == 'video' and mode == 'manual':
result = self.videoManualMode()
if input_type == 'camera' and mode == 'auto':
result = self.cameraAutoMode()
if input_type == 'camera' and mode == 'manual':
result = self.cameraManualMode()
return result

def main():
calibrator = InCalibrator(args.CAMERA_TYPE)
calib = CalibMode(calibrator, args.INPUT_TYPE, args.SELECT_MODE)
result = calib()

if len(calibrator.corners) == 0:
raise Exception("Calibration failed. Chessboard not found, check the parameters")
if len(calibrator.corners) < args.CALIB_NUMBER:
raise Exception("Warning: Calibration images are not enough. At least {} valid images are needed.".format(args.CALIB_NUMBER))

print("Calibration Complete")
print("Camera Matrix is : {}".format(result.camera_mat.tolist()))
print("Distortion Coefficient is : {}".format(result.dist_coeff.tolist()))
print("Reprojection Error is : {}".format(np.mean(result.reproj_err)))

fs = cv2.FileStorage(args.OUTPUT, cv2.FILE_STORAGE_WRITE)
fs.write("camera_type", result.type)
fs.write("resolution", np.int32([W, H]))
fs.write("camera_matrix", result.camera_mat)
fs.write("dist_coeffs", result.dist_coeff)

fs.release()
print("successfully saved camera data")

if __name__ == '__main__':
main()

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
import numpy as np
import cv2
import os
import vvdutils as vv

class FisheyeCameraModel(object):
"""
Fisheye camera model, for undistorting, projecting and flipping camera frames.
"""

def __init__(self, camera_param_file, camera_name):
if not os.path.isfile(camera_param_file):
raise ValueError("Cannot find camera param file")

# if camera_name not in settings.camera_names:
# raise ValueError("Unknown camera name: {}".format(camera_name))

self.camera_file = camera_param_file
self.camera_name = camera_name
self.scale_xy = (1.0, 1.0)
self.shift_xy = (0, 0)
self.undistort_maps = None
self.project_matrix = None
self.project_shape = (3200, 3200) # settings.project_shapes[self.camera_name]
self.load_camera_params()

def load_camera_params(self):
fs = cv2.FileStorage(self.camera_file, cv2.FILE_STORAGE_READ)
self.camera_matrix = fs.getNode("camera_matrix").mat()
self.dist_coeffs = fs.getNode("dist_coeffs").mat()
self.resolution = fs.getNode("resolution").mat().flatten()
self.camera_type = fs.getNode("camera_type").string()

if fs.getNode("scale_xy").mat() is not None:
self.scale_xy = fs.getNode("scale_xy").mat().flatten()

if fs.getNode("shift_xy").mat() is not None:
self.shift_xy = fs.getNode("shift_xy").mat().flatten()

if fs.getNode("project_matrix").mat() is not None:
self.project_matrix = fs.getNode("project_matrix").mat()

fs.release()
self.update_undistort_maps()

def update_undistort_maps(self):
new_matrix = self.camera_matrix.copy()
new_matrix[0, 0] *= self.scale_xy[0]
new_matrix[1, 1] *= self.scale_xy[1]
new_matrix[0, 2] += self.shift_xy[0]
new_matrix[1, 2] += self.shift_xy[1]
width, height = self.resolution

self.undistort_maps = cv2.fisheye.initUndistortRectifyMap(
self.camera_matrix,
self.dist_coeffs,
np.eye(3),
new_matrix,
(width, height),
cv2.CV_16SC2
)
return self

def set_scale_and_shift(self, scale_xy=(1.0, 1.0), shift_xy=(0, 0)):
self.scale_xy = scale_xy
self.shift_xy = shift_xy
self.update_undistort_maps()
return self

def undistort(self, image):
result = cv2.remap(image, *self.undistort_maps, interpolation=cv2.INTER_LINEAR,
borderMode=cv2.BORDER_CONSTANT)
return result

def project(self, image):
if self.project_matrix is None:
return image.copy()
result = cv2.warpPerspective(image, self.project_matrix, self.project_shape)
return result

def flip(self, image):
if self.camera_name == "front":
return image.copy()

elif self.camera_name == "back":
return image.copy()[::-1, ::-1, :]

elif self.camera_name == "left":
return cv2.transpose(image)[::-1]

else:
return np.flip(cv2.transpose(image), 1)

def save_data(self):
fs = cv2.FileStorage(self.camera_file, cv2.FILE_STORAGE_WRITE)
fs.write("camera_matrix", self.camera_matrix)
fs.write("dist_coeffs", self.dist_coeffs)
fs.write("resolution", self.resolution)
fs.write("project_matrix", self.project_matrix)
fs.write("scale_xy", np.float32(self.scale_xy))
fs.write("shift_xy", np.float32(self.shift_xy))
fs.release()

class NormalCameraModel(object):
"""
normal camera model, for undistorting, projecting
"""

def __init__(self, camera_param_file, camera_name):
if not os.path.isfile(camera_param_file):
raise ValueError("找不到相机参数文件")

self.camera_file = camera_param_file
self.camera_name = camera_name
self.undistort_map = None
self.scale_xy = (1.0, 1.0)
self.shift_xy = (0, 0)
self.load_camera_params()

def load_camera_params(self):
fs = cv2.FileStorage(self.camera_file, cv2.FILE_STORAGE_READ)
self.camera_matrix = fs.getNode("camera_matrix").mat()
self.dist_coeffs = fs.getNode("dist_coeffs").mat()
self.resolution = fs.getNode("resolution").mat().flatten()
if fs.getNode("scale_xy").mat() is not None:
self.scale_xy = fs.getNode("scale_xy").mat().flatten()

if fs.getNode("shift_xy").mat() is not None:
self.shift_xy = fs.getNode("shift_xy").mat().flatten()

if fs.getNode("project_matrix").mat() is not None:
self.project_matrix = fs.getNode("project_matrix").mat()

fs.release()
self.update_undistort_map()

def set_scale_and_shift(self, scale_xy=(1.0, 1.0), shift_xy=(0, 0)):
self.scale_xy = scale_xy
self.shift_xy = shift_xy
self.update_undistort_map()
return self

def update_undistort_map(self):
new_matrix = self.camera_matrix.copy()
new_matrix[0, 0] *= self.scale_xy[0]
new_matrix[1, 1] *= self.scale_xy[1]
new_matrix[0, 2] += self.shift_xy[0]
new_matrix[1, 2] += self.shift_xy[1]
width, height = self.resolution

width, height = self.resolution
self.undistort_map = cv2.initUndistortRectifyMap(
self.camera_matrix,
self.dist_coeffs,
np.eye(3, 3),
new_matrix,
(width, height),
cv2.CV_16SC2
)

def undistort(self, image):
result = cv2.remap(image, *self.undistort_map, interpolation=cv2.INTER_LINEAR,
borderMode=cv2.BORDER_CONSTANT)
return result

def save_data(self):
fs = cv2.FileStorage(self.camera_file, cv2.FILE_STORAGE_WRITE)
fs.write("camera_matrix", self.camera_matrix)
fs.write("dist_coeffs", self.dist_coeffs)
fs.write("resolution", self.resolution)
fs.release()

# 示例用法
camera_model = FisheyeCameraModel("camera_intrinsic.yaml", "camera_0")
# 设置新的缩放和偏移
camera_model.set_scale_and_shift(scale_xy=(0.7, 0.7), shift_xy=(0, 0))
camera_model.update_undistort_maps()

image_path_list = vv.glob_images('cali-data')
for image_path in image_path_list:
distorted_image= cv2.imread(image_path)
undistorted_image = camera_model.undistort(distorted_image)
save_path = vv.OS_join('result', vv.OS_basename(image_path))
vv.cv_bgr_imwrite(undistorted_image, save_path)
vv.PIS(undistorted_image)

# camera_model.save_data()
pass

经纬度映射法是一种传统的鱼眼校正方法,就是先将鱼眼图像映射在一个球面上。这样做不无道理,因为鱼眼镜头成像的时候,光线就是从一个球面经过折射,投射在成像平面上的(虽然事实并没有那么简单,但大体可以看作这样),如下图所示。而现在就是将成像过程反过来做。

但这样的效果并不理想,畸变仍然十分严重。举个例子,格陵兰岛的面积不大,但它所跨的纬度很大,所以如果直接以经纬度为坐标投射在二维平面的话,它的面积会远远超过澳大利亚的面积。

将球面上的点经过一定的偏折,再映射到图像平面上。下图中(注意,此图是从上向下看的,图中的坐标轴与字母表示跟上图是不对应的,从这里开始是纬度,是经度),是一个常数,它表示线段AC(点A就是要映射的点)与出射光线的夹角,最大为3/4Π,论文以及本文代码中,将其设定为Π/2。实际上,如果光线不发生偏折,其映射结果是最精确的,但这需要一个无穷大的平面。偏折的意义在于将映射结果限定在一定范围内。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
from defisheye import Defisheye

dtype = 'linear'
format = 'fullframe'
fov = 180
pfov = 120

img = "./images/example3.jpg"
img_out = f"./images/out/example3_{dtype}_{format}_{pfov}_{fov}.jpg"

obj = Defisheye(img, dtype=dtype, format=format, fov=fov, pfov=pfov)

# To save image locally
obj.convert(outfile=img_out)

# To use the converted image in memory

new_image = obj.convert()