This is a collection of algorithms related to multiple view camera calibration in computer vision. Please note that the goal of this package is to provide minimal examples to demonstrate the concept for beginners (i.e., students). For large-scale, realtime, accurate, robust, production-quality implementations, or for implementations for your specific situation, please consult your advisor.
This is research software and may contain bugs or other issues -- please use it at your own risk. If you experience major problems with it, you may contact us, but please note that we do not have the resources to deal with all issues.
You can simply install the package by pip as follows.
python3 -m pip install -U pycalib-simple
Or to use the latest version, specifiy the github repository as follows. (visit https://pypi.org/project/pycalib-simple/ to check the latest version in PyPI)
python3 -m pip install -U git+https://github.com/nbhr/pycalib.git
Notice that the pip installation does not include examples in ./ipynb/ or tools in ./tools/. To run examples and tools, download the repository explicitly. For example,
- Local: You can clone/download this repository to your local PC, and open
./ipynb/*.ipynbfiles by your local Jupyter (e.g., VSCode + Jupyter plugin). - Colaboratory: You can open each Jupyter notebook directly in Google Colaboratory by clicking the
buttons below.
- Most of them do not run properly as-is, since colab does not clone images used in the Jupyter notebooks. Please upload required files manually. (or run
!pip installand!git cloneat the beginning of each notebook.) - The scripts in
./tools/are not supposed to run in Colab/Jupyter.
- Most of them do not run properly as-is, since colab does not clone images used in the Jupyter notebooks. Please upload required files manually. (or run
- Intrinsic calibration
- Intrinsic camera calibration from a video of ChAruCo pattern.
- GoPro fisheye lens distortion is handled by the rational model in OpenCV
- 2D keypoint detection
- ChAruCo corner detection to find 2D-2D corresponding points between cameras.
- Extrinsic calibration
- Extrinsic camera calibartion from 2D-2D correspondences.
- Scale / orientation alignment
- Scale / orientation alignment of the world coordinate system by capturing an AruCo marker on the floor.
- Intrinsic calibration with charuco images
- Intrinsic calibration with chessboard images
- Zhang's method
- Extrinsic calibration w.r.t. a charuco board
- PnP with ChAruco
- Extrinsic calibration w.r.t. a chessboard
- PnP with chessboard
- Intrinsic / Extrinsic calibration with 2D-3D correspondences
- for non-planar reference objects
-
Multi-view triangulation
- N-view DLT
-
Robust brute-force multi-view triangulation
- Robust but brute-force n-view DLT with occlusion handling and outlier rejection
-
ChAruco diamond marker detection for 2D-2D correspondences
- For extrinsic calibration using a ChAruco diamond marker.
- Also can be used for PnP, i.e., extrinsic calibration w.r.t. the diamond marker
-
2-view extrinsic calibration from 2D-2D correspondences
- Decomposition of the essential matrix to
$R$ and$t$ .
- Decomposition of the essential matrix to
-
N-view registration
- A linear registration of pairwise poses into a single coordinate system
-
N-view bundle adjustment
- A non-linear minization of reprojection errors
- Each camera can specify the parameters to optimize and to share with each other
-
N-view time sync
- GoPro compatible QR time sync pattern generator, detector, and offset estimator
-
Homography
- Homography from one camera plane to another
-
Stereo rectification
- Horizontal / vertical stereo rectification for multi-view system and triangulation in WCS
- Mirror-based extrinsic camera calibration
- Extrinsic calibration of a camera w.r.t. a reference object not directly visible from the camera.
- This is an implementation of
Takahashi, Nobuhara and Matsuyama "A New Mirror-based Camera Pose Estimation Using an Orthogonality Constraint," CVPR 2012.
- Display-camera calibration with mirrors
- How to calibrate a webcam w.r.t. a display using a ChAruCo pattern.
- 3D circle estimation
- 3D circle center / normal estimation from its 2D projection, i.e., 2D ellipse.
- Sphere center estimation
- 3D sphere center estimation from its 2D projection, i.e., 2D ellipse.
- For extrinsic calibration using a ball.
- A color-based ball detection is provided as
tools/detect_by_color_gui.pyandtools/detect_by_color.py. The former GUI version can be used to sample foreground and background pixel colors, and the latter can be used to process each frame.
In general, prepare some synthetic dataset, i.e., a toy example, first so that your code can return the exact solution up to the machine epsillon. Then you can try with real data or synthetic data with noise to mimic it.
Also you may want to read Section A6.3 "A sparse Levenberg-Marquardt algorithm" of the textbook "Multiple View Geometry in Computer Vision" by Hartley and Zisserman.
- Linear calibration: Use
numpy. - Non-linear (including bundule adjustment): Try
scipy.optimize.least_squaresfirst.- Implement your objective function as simply as possible. You do not need to consider the computational efficiency at all. "Done is better than perfect."
- Test with the toy example and make sure that your objective function returns zero for the ground-truth parameter.
- If your simple objective function above is unacceptably slow, try the followings in this order.
- Ask yourself again before trying to make it faster. Is it really unacceptable? If your calibration can finish in an hour and you do not do it so often, it might be OK for example. "Premature optimization is the root of all evil." (D. Knuth).
- Make sure that the calibration runs successfully anyway. In what follows, double-check that the calibration results do not change before and after the code optimization.
- Vectorize the computation with
numpy, i.e., no for-loops in the objective function.- or use
numba(e.g.@numba.jit)
- or use
- If the system is sparse, use
jac_sparsityoption. It makesscipy.optimize.least_squaresmuch faster even without analitical Jacobian. - Implement the analytical Jacobian. You may want to use maxima to automate the calculation, or you may use JAX or other autodiff solutions for this.
- Reimplement in C++ with ceres-solver, g2o, or sba if the computation speed is really important. You can also consider using PyTorch/Tensorflow for GPU-acceleration and autodiff by Theseus or similar libraries.
- Implement your objective function as simply as possible. You do not need to consider the computational efficiency at all. "Done is better than perfect."