/* * Processor.cpp * * Created on: Jun 13, 2010 * Author: ethan */ #include "Calibration.h" #include using namespace cv; Calibration::Calibration() : patternsize(6, 8) { } Calibration::~Calibration() { } namespace { double computeReprojectionErrors(const vector >& objectPoints, const vector >& imagePoints, const vector& rvecs, const vector< Mat>& tvecs, const Mat& cameraMatrix, const Mat& distCoeffs, vector& perViewErrors) { vector imagePoints2; int i, totalPoints = 0; double totalErr = 0, err; perViewErrors.resize(objectPoints.size()); for (i = 0; i < (int)objectPoints.size(); i++) { projectPoints(Mat(objectPoints[i]), rvecs[i], tvecs[i], cameraMatrix, distCoeffs, imagePoints2); err = norm(Mat(imagePoints[i]), Mat(imagePoints2), CV_L1); int n = (int)objectPoints[i].size(); perViewErrors[i] = err / n; totalErr += err; totalPoints += n; } return totalErr / totalPoints; } void calcChessboardCorners(Size boardSize, float squareSize, vector& corners) { corners.resize(0); for (int i = 0; i < boardSize.height; i++) for (int j = 0; j < boardSize.width; j++) corners.push_back(Point3f(float(j * squareSize), float(i * squareSize), 0)); } /**from opencv/samples/cpp/calibration.cpp * */ bool runCalibration(vector > imagePoints, Size imageSize, Size boardSize, float squareSize, float aspectRatio, int flags, Mat& cameraMatrix, Mat& distCoeffs, vector& rvecs, vector& tvecs, vector& reprojErrs, double& totalAvgErr) { cameraMatrix = Mat::eye(3, 3, CV_64F); if (flags & CV_CALIB_FIX_ASPECT_RATIO) cameraMatrix.at (0, 0) = aspectRatio; distCoeffs = Mat::zeros(4, 1, CV_64F); vector > objectPoints(1); calcChessboardCorners(boardSize, squareSize, objectPoints[0]); for (size_t i = 1; i < imagePoints.size(); i++) objectPoints.push_back(objectPoints[0]); calibrateCamera(objectPoints, imagePoints, imageSize, cameraMatrix, distCoeffs, rvecs, tvecs, flags); bool ok = checkRange(cameraMatrix, CV_CHECK_QUIET) && checkRange(distCoeffs, CV_CHECK_QUIET); totalAvgErr = computeReprojectionErrors(objectPoints, imagePoints, rvecs, tvecs, cameraMatrix, distCoeffs, reprojErrs); return ok; } void saveCameraParams(const string& filename, Size imageSize, Size boardSize, float squareSize, float aspectRatio, int flags, const Mat& cameraMatrix, const Mat& distCoeffs, const vector& rvecs, const vector& tvecs, const vector& reprojErrs, const vector >& imagePoints, double totalAvgErr) { FileStorage fs(filename, FileStorage::WRITE); time_t t; time(&t); struct tm *t2 = localtime(&t); char buf[1024]; strftime(buf, sizeof(buf) - 1, "%c", t2); fs << "calibration_time" << buf; if (!rvecs.empty() || !reprojErrs.empty()) fs << "nframes" << (int)std::max(rvecs.size(), reprojErrs.size()); fs << "image_width" << imageSize.width; fs << "image_height" << imageSize.height; fs << "board_width" << boardSize.width; fs << "board_height" << boardSize.height; fs << "squareSize" << squareSize; if (flags & CV_CALIB_FIX_ASPECT_RATIO) fs << "aspectRatio" << aspectRatio; if (flags != 0) { sprintf(buf, "flags: %s%s%s%s", flags & CV_CALIB_USE_INTRINSIC_GUESS ? "+use_intrinsic_guess" : "", flags & CV_CALIB_FIX_ASPECT_RATIO ? "+fix_aspectRatio" : "", flags & CV_CALIB_FIX_PRINCIPAL_POINT ? "+fix_principal_point" : "", flags & CV_CALIB_ZERO_TANGENT_DIST ? "+zero_tangent_dist" : ""); cvWriteComment(*fs, buf, 0); } fs << "flags" << flags; fs << "camera_matrix" << cameraMatrix; fs << "distortion_coefficients" << distCoeffs; fs << "avg_reprojection_error" << totalAvgErr; if (!reprojErrs.empty()) fs << "per_view_reprojection_errors" << Mat(reprojErrs); if (!rvecs.empty() && !tvecs.empty()) { Mat bigmat(rvecs.size(), 6, CV_32F); for (size_t i = 0; i < rvecs.size(); i++) { Mat r = bigmat(Range(i, i + 1), Range(0, 3)); Mat t = bigmat(Range(i, i + 1), Range(3, 6)); rvecs[i].copyTo(r); tvecs[i].copyTo(t); } cvWriteComment(*fs, "a set of 6-tuples (rotation vector + translation vector) for each view", 0); fs << "extrinsic_parameters" << bigmat; } if (!imagePoints.empty()) { Mat imagePtMat(imagePoints.size(), imagePoints[0].size(), CV_32FC2); for (size_t i = 0; i < imagePoints.size(); i++) { Mat r = imagePtMat.row(i).reshape(2, imagePtMat.cols); Mat(imagePoints[i]).copyTo(r); } fs << "image_points" << imagePtMat; } } }//anon namespace bool Calibration::detectAndDrawChessboard(int idx, image_pool* pool) { bool patternfound = false; Mat grey = pool->getGrey(idx); if (grey.empty()) return false; vector corners; IplImage iplgrey = grey; patternfound = cvCheckChessboard(&iplgrey, patternsize); if (!patternfound) return false; float factor = grey.cols / 320.0f; if (factor < 1) factor = 1; cv::Size r_size = cv::Size(grey.cols / factor, grey.rows / factor); Mat grey_sub; while (patternfound && (r_size.width < grey.size().width)) { cv::resize(grey, grey_sub, r_size); patternfound = findChessboardCorners(grey_sub, patternsize, corners); if (patternfound) { r_size.width *= 2; r_size.height *= 2; } else { return false; } } // hopefully if we're going to fail we fail on a smaller size image in the while-loop patternfound = findChessboardCorners(grey, patternsize, corners); Mat img = pool->getImage(idx); if (corners.size() < 1) return false; if (patternfound) { cornerSubPix(grey, corners, Size(11, 11), Size(-1, -1), TermCriteria(CV_TERMCRIT_EPS + CV_TERMCRIT_ITER, 30, 0.1)); imagepoints.push_back(corners); } drawChessboardCorners(img, patternsize, Mat(corners), patternfound); imgsize = grey.size(); return patternfound; } void Calibration::drawText(int i, image_pool* pool, const char* ctext) { // Use "y" to show that the baseLine is about string text = ctext; int fontFace = FONT_HERSHEY_COMPLEX_SMALL; double fontScale = .8; int thickness = .5; Mat img = pool->getImage(i); int baseline = 0; Size textSize = getTextSize(text, fontFace, fontScale, thickness, &baseline); baseline += thickness; // center the text Point textOrg((img.cols - textSize.width) / 2, (img.rows - textSize.height * 2)); // draw the box rectangle(img, textOrg + Point(0, baseline), textOrg + Point(textSize.width, -textSize.height), Scalar(0, 0, 255), CV_FILLED); // ... and the baseline first line(img, textOrg + Point(0, thickness), textOrg + Point(textSize.width, thickness), Scalar(0, 0, 255)); // then put the text itself putText(img, text, textOrg, fontFace, fontScale, Scalar::all(255), thickness, 8); } void Calibration::resetChess() { imagepoints.clear(); } void Calibration::calibrate(const char* filename) { vector rvecs, tvecs; vector reprojErrs; double totalAvgErr = 0; int flags = 0; flags |= CV_CALIB_FIX_PRINCIPAL_POINT | CV_CALIB_FIX_ASPECT_RATIO; bool writeExtrinsics = true; bool writePoints = true; bool ok = runCalibration(imagepoints, imgsize, patternsize, 1.f, 1.f, flags, K, distortion, rvecs, tvecs, reprojErrs, totalAvgErr); if (ok) { saveCameraParams(filename, imgsize, patternsize, 1.f, 1.f, flags, K, distortion, writeExtrinsics ? rvecs : vector< Mat> (), writeExtrinsics ? tvecs : vector (), writeExtrinsics ? reprojErrs : vector (), writePoints ? imagepoints : vector > (), totalAvgErr); } } int Calibration::getNumberDetectedChessboards() { return imagepoints.size(); }