To start, correspondences are labeled among the pictures taken. In order to warp an image to align with its
            reference image, homography H (defining transformation p'=Hp) is recovered. This step is implemented in
            the function H = computeH(im1_pts, im2_pts). To ensure the stability of H, more than 4
            correspondences are used to produce an overdetermined system, which is solved with least squares.
        
|   |   | 
|   |   | 
|   |   | 
            Next warpImage(im, H) leverages the recovered homography H to perform inverse warping. The
            size, or bounding box, of the warped image is predicted by piping the 4 corners of the original image
            through H. Then, the bounding box is inverse warped onto the original image, where an alpha mask drops
            "invalid" (x,y)'s outside of the original image's dimensions. Finally, this warped image's values are
            resampled with scipy's RegularGridInterpolator.
        
To test the above functions, "rectification" is performed on two example images. In both cases, the post-it note is "made rectangular" using a homography.
|   |   |   | 
|   |   |   | 
The images are warped to create an image mosaic. Specifically, the image on the left is warped in reference to the image on the right, which is positioned accordingly.
|   |   | 
|   |   | 
|   |   | 
The images need to be blended together into a single image. A naive approach involves setting alpha to 0.5 in the overlapping region:
|   |   |   | 
A second approach involves the distance transform, where alpha is 1 at the center of the original images, then decreases to 0 at their edges. In the overlapping region, the left image's alpha is set to 1 if its distance transform is greater than the right image's distance transform, and vice versa.
|   |   |   |   |   |   | 
            Moreover, the 2-band blending technique (with a 2-level image pyramid) can remedy wedge-like
            artifacts:
            Porch Results...
        
 
         
                 
                Hallway Results...
 
         
                 
                Shelf Results...
 
         
                 
                The first step of autostitching is finding corners in the images. Harris corners are computed and then thresholded. The images below are overlaid with their respective Harris corners.
|   |   | 
|   |   | 
|   |   | 
Adaptive non-maximal suppression is used to select 200 interest points from the Harris corners. This method sorts the corners by non-maximal suppression radius and retrieves 200 corresponding to the largest radii, ensuring that the interest points are well-distributed across the images.
|   |   | 
|   |   | 
|   |   | 
For each feature point, a feature descriptor is extracted. More specifically, a 40x40 patch is created around each feature point, and an 8x8 patch is sampled with a spacing of s=5 pixels. Finally, the feature descriptors are bias/gain-normalized. For each image below, the 200 feature descriptors are shown in a 10x20 grid.
|   |   | 
|   |   | 
|   |   | 
            Feature descriptors are matched across images in a mosaic. Using scipy.spatial's
            KDTree, a match is recorded if:
        
|   | 
|   | 
|   | 
Given feature matches, 4-point RANSAC is used to compute a robust homography estimate. The algorithm is described in the following steps:
dist(p', Hp)
                < errork iterations|   | 
|   | 
|   | 
Finally, Part A (image warping and blending) is repeated to produce mosaics. The manually and automatically stitched results from both parts are displayed side by side for comparison. For the most part, the automatically stitched results look just as if not better aligned than the manually stitched results.
|   |   | 
|   |   | 
|   |   | 
The coolest thing that I learned from this project was RANSAC. By testing a handful of random samples, a comparable homography can be computed automatically and in a time-efficient manner.