Introduction

Problem

This project involves aligning three grayscale images, each for the Red, Green, and Blue channels. The goal is to line them up correctly so they can be combined into a single color image. Proper alignment is essential to avoid color distortions and ensure the final image looks right. This page will explore the different methods used to achieve this alignment. We will assume that a simple (x,y) translation model is sufficient for proper alignment. We will use images from the Prokudin-Gorskii collection for this project.

emir input
Example input
emir output
Example output

Preliminary Steps

We will first cut the input image evenly into 3 equal parts to obtain a single grayscale image for each of the 3 channels. These images contain borders that interfers with the alignment algorithms. To remove these borders, we crop the images by cropping a fixed fraction of the image from the 4 boundaries.

emir blue
Blue channel, before crop
emir green
Green channel, before crop
emir red
Red channel, before crop
emir blue cropped
Blue channel, after crop
emir green cropped
Green channel, after crop
emir red cropped
Red channel, after crop

Alignment Metric

Normalized Cross-Correlation

The NCC can be used to compute how similar two images are. The NCC value can range from -1 to 1. By normalizing, the metric can give a good measure of the difference in intensities (eg exposure) between the 2 images. Normalization ensures that the comparison focuses on the relative patterns and structures within the images, rather than their absolute intensity values.

The NCC can be expressed as:

\[ \text{NCC}(A, B) = \frac{\sum_{x,y} (A(x, y) - \bar{A})(B(x, y) - \bar{B})}{\sqrt{\sum_{x,y} (A(x, y) - \bar{A})^2 \cdot \sum_{x,y} (B(x, y) - \bar{B})^2}} \]

Alignment Algorithm

Single-scale alignment

We will take the blue channel image as the reference image and attempt to align the red and green channel image to it. As mentioned in the Problem Statement, we assume that a simple (x ,y) translation is sufficient for proper alignment. We first attempt to align the red channel with the blue channel. We iterate through a reasonably large window (10% of image size) of (x, y) translations on the red channel image and compute the NCC for each of these translations. The translation that produced the largest NCC is chosen. For simplicity, we used a circular translation (np.roll). Then, we do the same for the green channel (ie attempting to align it with the blue channel). After the alignments, we stack the 3 channels together to produce a color image.

cathedral_no_border_removal
Cathedral, without border removal
cathedral_simple_border_removal
Cathedral, with border removal
monastery_no_border_removal
Monastery, without border removal
monastery_simple_border_removal
Monastery, with border removal

For some cases, the existence of the border does not impact the alignment of the final image much. But for other cases like in Monastery, the impact is quite pronounced.

Multi-scale alignment using Gaussian Pyramid

When images get large, it becomes increasingly infeasible to use the single-scale algorithm for alignment. Specifically, each computation of the NCC takes longer, and the number of computations to run will also be increased (ie the window-size is increased to cover 10% of the image). To speed things up, we will use the technique of multi-scale alignment using Gaussian Pyramids.

The Gaussian Pyramid is a list of images, starting from the original full quality image at the base of the pyramid and becoming smaller (less pixels) and coarser as we go up the pyramid. This rescaling is done by convolving the image with a Gaussian Kernel and subsampling alternate pixels of the image.

Algorithm: Image Alignment using Gaussian Pyramid and NCC Input: Original image with 3 color channels 1. For each color channel: a. Generate Gaussian Pyramid: - Start with the original image - While the current image size is greater than 200x200: - Apply Gaussian filter to smooth the image - Down-sample the image by a factor of 0.5 2. Perform the alignment process: - First, align the red channel against the blue channel - Second, align the blue channel against the green channel For each alignment: a. Set initial window-size to 32x32 b. Iterate through translations within window-size on the smallest image: - For each translation, compute NCC - Find the translation that maximizes NCC c. Apply the optimal translation on the next-finer image: - Double the translation values (since the next-finer image is twice the size) d. Reduce window-size by half - To reduce computation time for larger images. e. Repeat steps 2b to 2d until we reach the original, full-quality image. 3. Stack the 3 images into a single color image

Here are some results:

sculpture_no_border_removal
Sculpture, without border removal
sculpture_simple_border_removal
Sculpture, with border removal
church_no_border_removal
Church, without border removal
church_simple_border_removal
Church, with border removal

Improvements

Canny edge detection

Instead of trying to align the images by their raw grayscale images, it may be better to align them by their edges. The raw images may be subject to noise such as different lighting variations that hinder the alignment algorithm. A popular algorithm used to find edges of an image is the Canny edge detection algorithm.

Canny edge detection is a multi-step algorithm used to detect edges in images. It involves smoothing the image with a Gaussian filter to reduce noise, calculating intensity gradients, and applying non-maximum suppression to thin the edges. Double thresholding is then used to classify strong, weak, and non-edges, followed by edge tracking by hysteresis to finalize the edges. To do this, we used OpenCV's cv2.Canny().

emir blue cropped
Blue channel
emir green cropped
Green channel
emir red cropped
Red channel
emir blue canny
Blue channel, edges
emir green canny
Green channel, edges
emir red canny
Red channel, edges
emir_no_canny
Alim Khan, aligned with raw images
emir_output
Alim Khan, aligned with edges

Directly aligning the images based on the raw grayscale images may not give us desirable results. For the example above, the blue coat of Alim Khan shows up as pixels with intensity close to 1 in the blue channel but much less information is captured in the red and green channels. Maximizing the NCC on the raw images thus produces a distorted image. Instead, aligning with edges (eg the outline of Alim Khan's bod and key structural patterns on his coat) can better capture important structural information and produce a better image.

Automatic border cropping

Instead of cropping a fixed fraction of the image, this section aims to provide different approaches to automatically crop the image.

Approach 1: Computing mean intensity row-wise/ column-wise. The images each contain 2 borders - a white outer border and a black inner border. To remove the top border, we will scan the image row-wise and compute the mean intensity of the row. If the mean is greater than a threshold of 0.95, we classify this border as a white border and remove it. If the mean is less than a threshold of 0.05, we classify this broder as a black border and remove it. Since we know that the image starts with a white border and is immediately followed by a black border, we can stop scanning once we have reached a row where the mean is within the 2 thresholds, which will mean that we have reached the end of the black border and have reached the image and we can stop scanning. The same approach can be used to also remove the other 3 borders. However, the performance of this approach is hindered by several factors. Firstly, there is a considerable amount of noise on the borders (the borders are often not cleanly separated from the image and the borders have varying intensities). There are also writings/ labellings on the borders which hinder the performance of this approach.

Approach 2: First process the images with cv2.Canny() to obtain the edges image. To remove the top border, start the scan from the 10% mark (this is chosen as a "safe" point where we know we will be within the image and not the border) of the image and scan outwards. Compute the mean intensity of the edges image of the row. Once this mean is above a certain threshold, it will mean that we have reached a border row and we can crop the current row and all rows above the current row out. We can do something similar for the other 3 borders. This approach works better than Approach 1 because the edges image can better capture the structure of the border and is less sensitive to noise such as difference in intensities between different parts of the border. After cropping the 3 channels separately, the 3 cropped images may have different dimensions. The 3 images are thus additionally cropped to the min (x,y) dimensions between them so that the final images have the same dimensions.

b
Blue channel
g
Green channel
r
Red channel
train_cropped_b
Blue channel, fixed border removal
train_cropped_g
Green channel, fixed border removal
train_cropped_r
Red channel, fixed border removal
b_autocrop
Blue channel auto border removal
g_autocrop
Green channel auto border removal
r_autocrop
Red channel auto border removal
train_no_border_removal
No border removal
train_simple_border_removal
Fixed border removal
train_auto_border_removal
Auto border removal

As we can see with this example, if we have no border removal, the borders mess with the alignment algorithm (NCC maximization) and produces a distorted image. If we have a fixed border removal, it is possible that by removing a fixed fraction, we are also removing some key parts of the image that we want to keep (in this case we are cropping out a bit of the top part of the train). Having an automatic border removal will thus remove the borders and we do not have to worry about tuning the exact fraction to crop away (in the case of the fixed border removal)

All Results

The following shows the results for all the images tested, as well as the displacements. The displacements are in (y, x) and are listed for green against blue and red against blue channels respectively.

cathedral
Cathedral
cathedral_output
Displacement: (5,2) / (12,3)
church
Church
church_output
Displacement: (25,4) / (58,-4)
emir
Emir
emir_output
Displacement: (49,24) / (107,40)
harvesters
Harvesters
harvesters_output
Displacement: (59,16) / (124,13)
icon
Icon
icon_output
Displacement: (41,17) / (89,23)
lady
Lady
lady_output
Displacement: (51,9) / (112,11)
melons
Melons
melons_output
Displacement: (81,10) / (178,13)
 monastery
Monastery
monastery_output
Displacement: (-3,2) / (3,2)
 monastery
Onion Church
onion_church_output
Displacement: (51,26) / (108,36)
sculpture
Sculpture
sculpture_output
Displacement: (33,-11) / (140,-27)
self_portrait
Self Portrait
self_portrait_output
Displacement: (78,29) / (176,37)
three_generations
Three Generations
three_generations_output
Displacement: (53,14) / (112,11)
tobolsk
Tobolsk
tobolsk_output
Displacement: (3,3) / (6,3)
train
Train
train_output
Displacement: (-169,19) / (-116,23)