Path: blob/master/samples/python/camera_calibration_show_extrinsics.py
16337 views
#!/usr/bin/env python1# -*- coding: utf-8 -*-23from mpl_toolkits.mplot3d import Axes3D4import matplotlib.pyplot as plt5import numpy as np6from matplotlib import cm7from numpy import linspace8import argparse9import cv2 as cv1011def inverse_homogeneoux_matrix(M):12R = M[0:3, 0:3]13T = M[0:3, 3]14M_inv = np.identity(4)15M_inv[0:3, 0:3] = R.T16M_inv[0:3, 3] = -(R.T).dot(T)1718return M_inv1920def transform_to_matplotlib_frame(cMo, X, inverse=False):21M = np.identity(4)22M[1,1] = 023M[1,2] = 124M[2,1] = -125M[2,2] = 02627if inverse:28return M.dot(inverse_homogeneoux_matrix(cMo).dot(X))29else:30return M.dot(cMo.dot(X))3132def create_camera_model(camera_matrix, width, height, scale_focal, draw_frame_axis=False):33fx = camera_matrix[0,0]34fy = camera_matrix[1,1]35focal = 2 / (fx + fy)36f_scale = scale_focal * focal3738# draw image plane39X_img_plane = np.ones((4,5))40X_img_plane[0:3,0] = [-width, height, f_scale]41X_img_plane[0:3,1] = [width, height, f_scale]42X_img_plane[0:3,2] = [width, -height, f_scale]43X_img_plane[0:3,3] = [-width, -height, f_scale]44X_img_plane[0:3,4] = [-width, height, f_scale]4546# draw triangle above the image plane47X_triangle = np.ones((4,3))48X_triangle[0:3,0] = [-width, -height, f_scale]49X_triangle[0:3,1] = [0, -2*height, f_scale]50X_triangle[0:3,2] = [width, -height, f_scale]5152# draw camera53X_center1 = np.ones((4,2))54X_center1[0:3,0] = [0, 0, 0]55X_center1[0:3,1] = [-width, height, f_scale]5657X_center2 = np.ones((4,2))58X_center2[0:3,0] = [0, 0, 0]59X_center2[0:3,1] = [width, height, f_scale]6061X_center3 = np.ones((4,2))62X_center3[0:3,0] = [0, 0, 0]63X_center3[0:3,1] = [width, -height, f_scale]6465X_center4 = np.ones((4,2))66X_center4[0:3,0] = [0, 0, 0]67X_center4[0:3,1] = [-width, -height, f_scale]6869# draw camera frame axis70X_frame1 = np.ones((4,2))71X_frame1[0:3,0] = [0, 0, 0]72X_frame1[0:3,1] = [f_scale/2, 0, 0]7374X_frame2 = np.ones((4,2))75X_frame2[0:3,0] = [0, 0, 0]76X_frame2[0:3,1] = [0, f_scale/2, 0]7778X_frame3 = np.ones((4,2))79X_frame3[0:3,0] = [0, 0, 0]80X_frame3[0:3,1] = [0, 0, f_scale/2]8182if draw_frame_axis:83return [X_img_plane, X_triangle, X_center1, X_center2, X_center3, X_center4, X_frame1, X_frame2, X_frame3]84else:85return [X_img_plane, X_triangle, X_center1, X_center2, X_center3, X_center4]8687def create_board_model(extrinsics, board_width, board_height, square_size, draw_frame_axis=False):88width = board_width*square_size89height = board_height*square_size9091# draw calibration board92X_board = np.ones((4,5))93#X_board_cam = np.ones((extrinsics.shape[0],4,5))94X_board[0:3,0] = [0,0,0]95X_board[0:3,1] = [width,0,0]96X_board[0:3,2] = [width,height,0]97X_board[0:3,3] = [0,height,0]98X_board[0:3,4] = [0,0,0]99100# draw board frame axis101X_frame1 = np.ones((4,2))102X_frame1[0:3,0] = [0, 0, 0]103X_frame1[0:3,1] = [height/2, 0, 0]104105X_frame2 = np.ones((4,2))106X_frame2[0:3,0] = [0, 0, 0]107X_frame2[0:3,1] = [0, height/2, 0]108109X_frame3 = np.ones((4,2))110X_frame3[0:3,0] = [0, 0, 0]111X_frame3[0:3,1] = [0, 0, height/2]112113if draw_frame_axis:114return [X_board, X_frame1, X_frame2, X_frame3]115else:116return [X_board]117118def draw_camera_boards(ax, camera_matrix, cam_width, cam_height, scale_focal,119extrinsics, board_width, board_height, square_size,120patternCentric):121min_values = np.zeros((3,1))122min_values = np.inf123max_values = np.zeros((3,1))124max_values = -np.inf125126if patternCentric:127X_moving = create_camera_model(camera_matrix, cam_width, cam_height, scale_focal)128X_static = create_board_model(extrinsics, board_width, board_height, square_size)129else:130X_static = create_camera_model(camera_matrix, cam_width, cam_height, scale_focal, True)131X_moving = create_board_model(extrinsics, board_width, board_height, square_size)132133cm_subsection = linspace(0.0, 1.0, extrinsics.shape[0])134colors = [ cm.jet(x) for x in cm_subsection ]135136for i in range(len(X_static)):137X = np.zeros(X_static[i].shape)138for j in range(X_static[i].shape[1]):139X[:,j] = transform_to_matplotlib_frame(np.eye(4), X_static[i][:,j])140ax.plot3D(X[0,:], X[1,:], X[2,:], color='r')141min_values = np.minimum(min_values, X[0:3,:].min(1))142max_values = np.maximum(max_values, X[0:3,:].max(1))143144for idx in range(extrinsics.shape[0]):145R, _ = cv.Rodrigues(extrinsics[idx,0:3])146cMo = np.eye(4,4)147cMo[0:3,0:3] = R148cMo[0:3,3] = extrinsics[idx,3:6]149for i in range(len(X_moving)):150X = np.zeros(X_moving[i].shape)151for j in range(X_moving[i].shape[1]):152X[0:4,j] = transform_to_matplotlib_frame(cMo, X_moving[i][0:4,j], patternCentric)153ax.plot3D(X[0,:], X[1,:], X[2,:], color=colors[idx])154min_values = np.minimum(min_values, X[0:3,:].min(1))155max_values = np.maximum(max_values, X[0:3,:].max(1))156157return min_values, max_values158159def main():160parser = argparse.ArgumentParser(description='Plot camera calibration extrinsics.',161formatter_class=argparse.ArgumentDefaultsHelpFormatter)162parser.add_argument('--calibration', type=str, default="../data/left_intrinsics.yml",163help='YAML camera calibration file.')164parser.add_argument('--cam_width', type=float, default=0.064/2,165help='Width/2 of the displayed camera.')166parser.add_argument('--cam_height', type=float, default=0.048/2,167help='Height/2 of the displayed camera.')168parser.add_argument('--scale_focal', type=float, default=40,169help='Value to scale the focal length.')170parser.add_argument('--patternCentric', action='store_true',171help='The calibration board is static and the camera is moving.')172args = parser.parse_args()173174fs = cv.FileStorage(args.calibration, cv.FILE_STORAGE_READ)175board_width = int(fs.getNode('board_width').real())176board_height = int(fs.getNode('board_height').real())177square_size = fs.getNode('square_size').real()178camera_matrix = fs.getNode('camera_matrix').mat()179extrinsics = fs.getNode('extrinsic_parameters').mat()180181fig = plt.figure()182ax = fig.gca(projection='3d')183ax.set_aspect("equal")184185cam_width = args.cam_width186cam_height = args.cam_height187scale_focal = args.scale_focal188min_values, max_values = draw_camera_boards(ax, camera_matrix, cam_width, cam_height,189scale_focal, extrinsics, board_width,190board_height, square_size, args.patternCentric)191192X_min = min_values[0]193X_max = max_values[0]194Y_min = min_values[1]195Y_max = max_values[1]196Z_min = min_values[2]197Z_max = max_values[2]198max_range = np.array([X_max-X_min, Y_max-Y_min, Z_max-Z_min]).max() / 2.0199200mid_x = (X_max+X_min) * 0.5201mid_y = (Y_max+Y_min) * 0.5202mid_z = (Z_max+Z_min) * 0.5203ax.set_xlim(mid_x - max_range, mid_x + max_range)204ax.set_ylim(mid_y - max_range, mid_y + max_range)205ax.set_zlim(mid_z - max_range, mid_z + max_range)206207ax.set_xlabel('x')208ax.set_ylabel('z')209ax.set_zlabel('-y')210ax.set_title('Extrinsic Parameters Visualization')211212plt.show()213214if __name__ == "__main__":215main()216217218