Laval University
Computational
Photography
Homework
1 – Colorizing the Russian Empire
by
Maxime Leclerc
February 1st 2015
Table of
contents
Project description
Approach
Single
scale algorithm results
Multiple
scale algorithm results
Additional
Prokudin-Gorskii results
Personal
photos results
Problems
encountered and solutions
Why
are some images not aligned properly
Bells
and whistles – Automatic contrast
Bells
and whistles – Automatic border cropping
Project
description
Sergei
Mikhailovich Prokudin-Gorskii (1863-1944) was a Russian chemist and a
pioneer in color photography. His photographic method was to divide the
visible
color spectrum into three channels using red, green and blue
filters. This project aims to automatically combine these three
channel colors into a single color image.
Red, green and blue
filters. Source: uic.edu
Approach
The
first thing the application does is to loop through all available JPG
and TIF files. For each source file, we splits its content into three
color channels: red, green and blue. We then automatically enhance the
image's contrast. Please
refer to the bell
and whistles – automatic
contrast section for additional
details about this process.
The next
step is to align the three channels. We first check if the image is
larger than 500 pixels. If that's the case, we recursively re-size
the image half its size until we get an image smaller that is than 500
pixels. This multi-scale representation allows us to speed up
computations on large images and is referred to as a pyramid
representation.
Pyramid
representation. Source: psu.edu
Before
doing additional calculations, we simplify our color channels by
transforming them into binary representations of the
channels' edges. We use the Sobel
method to perform this
transformation. Please note that the
Sobel method was used for this course
by three students in 2014: Diane
Fournier, Jingwei
Cao and Tom
Toulouse. It was also used by many Berkeley students for a Computational Photography course: Bilenko, Hall, Tang, etc. Using binary edge
channels instead of gray scale
channels improves our application's results in terms of quality
and speed.
For each
re-sized or original image, we compute the sum of
square differences
(SSD) for different red-blue and green-blue channel alignments. The
pyramid representation allows us to use a smaller search space for
each image size. As a result, we compute the SSD for translations in
the X and Y axes ranging only from -15 pixels to 15 pixels. In
addition,
we only compute the SSD for a 300 by 300 pixel region located around
the center of each image. These two simplifications allow us to
speed up computation, especially on large images.
The
result of these calculations gives us the best alignment for the
three color channels. We simply keep the alignment that has a minimum
SSD - that is, our channel difference metric - over all the computed
alignments. Note
that when we switch from a small image to a larger image in our
pyramid representation, we use the small image's alignment estimate as
a starting
point to find the best alignment for the larger image.
Once the
original channels have been aligned, we then automatically crop the
image's border. Please refer to the bell and whistles
–
automatic border cropping section for additional details
about this
process.
Single
scale algorithm results
Calculated red offsets: X = 9, Y = -1.
Calculated green offsets: X = 4, Y = 1.
|
data:image/s3,"s3://crabby-images/adb90/adb9054797cf23bec5a4dc4b356c8b3708f5b336" alt="result-00757v-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 5, Y = 5.
Calculated
green offsets: X = 2, Y = 3.
|
data:image/s3,"s3://crabby-images/f260e/f260e7126d86b83e519e368cf2ef088759636a9e" alt="result-00888v-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 12, Y = 0.
Calculated
green offsets: X = 6, Y = 1.
|
data:image/s3,"s3://crabby-images/7712a/7712acf792dc3a42d430b1f53c6f9fabad35da3f" alt="result-00889v-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 4, Y
= 3.
Calculated
green offsets: X = 2, Y = 2.
|
data:image/s3,"s3://crabby-images/0c169/0c16908d1e0e5bdfece6ca83ddaecd97577591b9" alt="result-00907v-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 6, Y
= 0.
Calculated
green offsets: X = 2, Y = 0.
|
data:image/s3,"s3://crabby-images/0092e/0092e60e923409cbb74aec45885151096c6d4a2f" alt="result-00911v-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 13, Y
= -1.
Calculated
green offsets: X = 1, Y = -1.
|
data:image/s3,"s3://crabby-images/d039d/d039d223d44c642239532ed603812d1642becf16" alt="result-01031v-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 4, Y
= 2.
Calculated
green offsets: X = 1, Y = 1.
|
data:image/s3,"s3://crabby-images/4b206/4b20662cc7ec3385a57d3bf0da4913bc5f205be7" alt="result-01657v-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 12, Y
= 1.
Calculated
green offsets: X = 5, Y = 1.
|
Calculated
red offsets: X = 14, Y
= 4.
Calculated
green offsets: X = 6, Y = 2.
|
Multiple
scale algorithm results
data:image/s3,"s3://crabby-images/d0d07/d0d07e9c5383e2d414d6bdfd08db6afd34d3a3bd" alt="result-00029u-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 87, Y
= 38.
Calculated
green offsets: X = 35, Y = 19.
|
data:image/s3,"s3://crabby-images/34352/343523a4a04aa0909c8cf76ff456791d3bff4feb" alt="result-00087u-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 108, Y
= 68.
Calculated
green offsets: X = 49, Y = 51.
|
data:image/s3,"s3://crabby-images/fdc06/fdc06a5fefc55b5f698596aac37ac739eeb74b75" alt="result-00128u-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 52, Y
= 38.
Calculated
green offsets: X = 34, Y = 25.
|
Calculated
red offsets: X = 85, Y
= 32.
Calculated
green offsets: X = 42, Y = 6.
|
Calculated
red offsets: X = 48, Y
= 13.
Calculated
green offsets: X = 14, Y = 5.
|
Calculated
red offsets: X = 123, Y
= 34.
Calculated
green offsets: X = 56, Y = 25.
|
Calculated
red offsets: X = 43, Y
= 6.
Calculated
green offsets: X = 16, Y = 5.
|
Calculated
red offsets: X = 12, Y
= 20.
Calculated
green offsets: X = -15, Y = 10.
|
Calculated
red offsets: X = 71, Y
= 34.
Calculated
green offsets: X = 23, Y = 20.
|
Additional
Prokudin-Gorskii results
Calculated red offsets: X = 14, Y = 5.
Calculated green offsets: X = 6, Y = 3.
|
data:image/s3,"s3://crabby-images/101e0/101e0ee6112c2aa2eae0621cf9bc9a0fbb96d531" alt="result-00033v-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 5, Y = 5.
Calculated
green offsets: X = 2, Y = 3.
|
data:image/s3,"s3://crabby-images/15796/157965ae1f3effbb44e950db95cc55bef6fe9638" alt="result-00540v-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 11, Y = 3.
Calculated
green offsets: X = 5, Y = 2.
|
data:image/s3,"s3://crabby-images/66060/6606021b01cfcfa1c21a1756cf213ffaf352f1e4" alt="result-01192v-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 13, Y
= 4.
Calculated
green offsets: X = 6, Y = 3.
|
data:image/s3,"s3://crabby-images/c484c/c484c143d51a6c374db333289d7ddcb678fe7312" alt="result-01375v-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 10, Y
= -4.
Calculated
green offsets: X = 5, Y = -1.
|
data:image/s3,"s3://crabby-images/5fb4b/5fb4ba57305ee8d767ddc493f17cc5046d029559" alt="result-00023u-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 69, Y
= 8.
Calculated
green offsets: X = 13, Y = -4.
|
data:image/s3,"s3://crabby-images/1c216/1c216f40c1acab57fc596ba36ae6adbe313542b5" alt="result-00147u-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 119, Y
= -15.
Calculated
green offsets: X = 53, Y = -4.
|
data:image/s3,"s3://crabby-images/d550c/d550c90b27f5240c431b2286759d729c510e3871" alt="result-00172u-adapthisteq-histeq.jpg"
Calculated
red offsets: X = 152, Y
= -6.
Calculated
green offsets: X = 39, Y = -1.
|
Calculated
red offsets: X = 129, Y
= 25.
Calculated
green offsets: X = 59, Y = 14.
|
Calculated
red offsets: X = 113, Y
= -24.
Calculated
green offsets: X = 43, Y = -9.
|
Personal
photos results
We tested out algorithms on our own photos. As expected, the best
results we get are when there is no movement in the scene and when the
camera does not move. Here are some before and after images:
Before
|
After
Calculated
red offsets: X = 4, Y
= 1.
Calculated
green offsets: X = 11, Y = 1.
data:image/s3,"s3://crabby-images/c4a50/c4a509e0e3d9032d33c9c7e97c275b2c8cda2d9b" alt="result-candles-adapthisteq-histeq.jpg" |
Before
|
After
Calculated
red offsets: X = 24, Y
= -1.
Calculated
green offsets: X = 27, Y = -1.
data:image/s3,"s3://crabby-images/32cd0/32cd0ee315bdc0f259af4b84935852a3c157434b" alt="result-wall(1)-adapthisteq-histeq.jpg" |
Before
|
After
Calculated
red offsets: X = 12, Y
= 2.
Calculated
green offsets: X = 13, Y = 1.
data:image/s3,"s3://crabby-images/77db6/77db6ca7a5d299db380747d8fc1bf310aa5cbb0f" alt="result-toys(1)-adapthisteq-histeq.jpg" |
Problems
encountered and solutions
The first problem we encountered was that a simple color channel
comparison did not give good alignment results. This explains why we
decided to transform the color channels into edge
images with the Sobel
method. Using this simplified representation gives much
better alignment results than the results we had with raw
images.
Sobel
edge detection. Source: umn.edu
Next, we had to do some trial and error
when we developed the automatic
contrast operation. We had to chose among many different
contrast transformations. We had poor results with some transformations
and decided to pick a hybrid solution that combined the best results in
a single image. See the section below for details about the final
solution. Finally, we had to tweak a few threshold values when we
created the automatic
border cropping. See the last section for additional
details on this process.
Why are some
images not aligned properly
We can see that our algorithms are not entirely robust for all images.
For example,
the following image doesn't seem to be aligned properly:
data:image/s3,"s3://crabby-images/34352/343523a4a04aa0909c8cf76ff456791d3bff4feb" alt="result-00087u-adapthisteq-histeq.jpg"
One possible explanation could be that the woman in
the middle as well as the trees and the clouds located on the left side
moved while the photographer was changing the color filters. We also
see some hard to avoid human movement
in the next photo featuring three
young women. Most of the image seems to be correctly aligned except for
the region around the young women:
Bells
and whistles – Automatic contrast
To
automatically improve image contrast, we use a simple histogram
equalization and a contrast-limited adaptive histogram
equalization
(CLAHE). This gives us two new image versions in addition to the
original version. After some testing, we found that each of these
three image versions contributed interesting visual information not necessarily
present in the other two image versions. As a result, we decided to
average the original image with these two histogram equalizations in
order to benefit from all three versions' visual information content.
Here are some before and after images:
Before automatic contrast
data:image/s3,"s3://crabby-images/86197/8619781f70549943b3d40339da2fae5edaab5c5f" alt="result-00106v.jpg"
|
After automatic contrast
data:image/s3,"s3://crabby-images/653e1/653e1916aa36852dbb721c6ffbb036f1cb0e8e9c" alt="result-00106v-adapthisteq-histeq.jpg" |
The left image above had too much yellow in it. The
image to the right corrected this issue.
Before
data:image/s3,"s3://crabby-images/79eb7/79eb72cc8476a9dae24e486413249f647dea46d5" alt="result-00907v.jpg"
|
data:image/s3,"s3://crabby-images/0092e/0092e60e923409cbb74aec45885151096c6d4a2f" alt="result-00911v-adapthisteq-histeq.jpg" |
We could barely see the clouds in the left image above.
However, the clouds are now visible in image to the right.
Before
data:image/s3,"s3://crabby-images/75adf/75adf36bd9943505ff6da23cb199d4cc106c292b" alt="result-01657v.jpg"
|
data:image/s3,"s3://crabby-images/4b206/4b20662cc7ec3385a57d3bf0da4913bc5f205be7" alt="result-01657v-adapthisteq-histeq.jpg" |
The first image had too much red in it. The second image seems more
realistic.
Bells
and whistles – Automatic border cropping
For
the automatic border cropping process, we first compute the difference
between each channel. We then sum these differences across all the
image's rows and columns. This gives us two
vectors: a vector
containing the sum of
channel differences for each horizontal
row
and one
containing the sum of channel differences for each vertical
column.
Here's a short pseudo code example:
borderDiff = channelDifference( Red, Green ) + channelDifference( Red, Blue )
+ channelDifference( Blue, Green );
sumBorderX = sumOverX( borderDiff );
sumBorderY = sumOverY( borderDiff );
Please note that a
similar cropping method was used in 2014 by Peng Xue for a Berkeley Computational Photography course. In adition, Tom
Toulouse also used a similar method for this course in 2014. Here are sample plots for a given image of
these two summed difference vectors - one summed
horizontally, the other summed vertically.
From these plots, we
can clearly see that the summed
difference
is higher in the first and
last few rows as well as in the first
and last few columns.
Channel difference summed over X
data:image/s3,"s3://crabby-images/4b796/4b796560e2c8084ab8d8e48c4a516ad0b8994380" alt="plot1.jpg"
|
Channel difference summed over Y
data:image/s3,"s3://crabby-images/52a17/52a176b50cc8591c3ca1828f069b051407136ae4" alt="plot2.jpg" |
Next, to automate
the
cropping process, we need to chose thresholds
that work well for most
images. We have two goals here: 1) Preserve interesting visual information and 2) Cut
the image where it is inconsistent.
Preserve interesting visual information
One threshold we implemented is used to
limit the maximum
region we allow to be cropped. After
some tweaking, a value that seems to work well is to limit
the
cropping to a portion of 7.5%
of the total image size for each cropping region
(at the top,
bottom, left and right). This threshold ensures that we don't crop the
image
too much and that we don't
lose interesting visual information.
Cut
the image where it is inconsistent
A difference
value above the mean value indicates that
the region
is dissimilar across the three channels. After some tweaking, we
found that using 115% of the mean value was a better threshold than
using the actual mean value when we crop the test
images.
In other words, we want to crop
a border region when this region
has a channel
difference
metric higher than 115% times the metric mean.
Original border difference summed over X
data:image/s3,"s3://crabby-images/4b796/4b796560e2c8084ab8d8e48c4a516ad0b8994380" alt="plot1.jpg"
|
Selected X values greater than 115% x mean value
The arrows and circles at the bottom represent approximations of the search operations and results.
data:image/s3,"s3://crabby-images/6d691/6d69189ead3caa96eb43104b7a5bdf05dd529d35" alt="plot3.jpg" |
Original border difference summed over Y
data:image/s3,"s3://crabby-images/52a17/52a176b50cc8591c3ca1828f069b051407136ae4" alt="plot2.jpg"
|
Selected Y values greater than 115% x the mean value
The arrows and circles at the bottom represent approximations of the search operations and results.
data:image/s3,"s3://crabby-images/5ed76/5ed76e2e3a796d80fdf28a829c758f41f700cc8d" alt="plot4.jpg" |
With
that in mind, we crop the image using the
rightmost above
average value. Remember that we limit ourselves to the region corresponding
to the first 7.5% portion of
each vector. We
also crop the image using the leftmost
above average value.
Again, we limit ourselves to the region corresponding
to the
last 7.5% portion of each summed difference vector.
maxCropRegion = 7.5%;
firstCroppedX = findFirstVectorElement( vectorSearched = selectedXValues,
findFirstVectorElement > 0, searchDirection = 'fromRightToLeft',
startAt = ( lengthVectorX * maxCropRegion ) );
Here are some
before and after images:
Before cropping
data:image/s3,"s3://crabby-images/467ff/467ff2fa843e4575da8c2c508f55d592190776df" alt="result-00888v.jpg"
|
data:image/s3,"s3://crabby-images/f260e/f260e7126d86b83e519e368cf2ef088759636a9e" alt="result-00888v-adapthisteq-histeq.jpg" |
The first image had yellow, blue, black and white bands before
cropping. The second image got rid of these bands.
Before
data:image/s3,"s3://crabby-images/57ccb/57ccb832cf1f6b583a5e7eb4607639ba50f06828" alt="result-00907v.jpg"
|
data:image/s3,"s3://crabby-images/0c169/0c16908d1e0e5bdfece6ca83ddaecd97577591b9" alt="result-00907v-adapthisteq-histeq.jpg" |
The image on the left had pink, blue, black and white regions before
the application cropped it. This has been fixed in the image on the
right.
Before
data:image/s3,"s3://crabby-images/79eb7/79eb72cc8476a9dae24e486413249f647dea46d5" alt="result-00907v.jpg"
|
data:image/s3,"s3://crabby-images/0092e/0092e60e923409cbb74aec45885151096c6d4a2f" alt="result-00911v-adapthisteq-histeq.jpg" |
Once more, the automatic cropping removed the blue/yellow/green,
black and white bands.