**Examples for Assignment 1** Luminance Operations ======================================================================== Brightness ------------------------------------------------------------------------ `brightnessFilter(image, ratio)` changes the brightness of an image by blending the original colors with black/white color in a ratio. (When `ratio > 0`, we blend with white to make it brighter; when `ratio < 0`, we blend with black to make it darker). data:image/s3,"s3://crabby-images/650f2/650f21930e36edf4a32678e599acca450b3fea4a" alt="-1" data:image/s3,"s3://crabby-images/a9eb0/a9eb0081c9f448b69c56a81a96d47ca2384ad9fb" alt="-0.5" data:image/s3,"s3://crabby-images/5e87f/5e87fac41f7b8cce300493b16773f87cad7d3dd8" alt="0" data:image/s3,"s3://crabby-images/8c31c/8c31c9390d8721f8e63f70496aa62aaef10dae77" alt="0.5" data:image/s3,"s3://crabby-images/61140/61140396f19df41d6e4813f4247aad40f5b49e4b" alt="1.0" Contrast ------------------------------------------------------------------------ `contrastFilter(image, ratio)` changes the contrast of an image by interpolating between a constant gray image (`ratio=-1`) with the average luminance and the original image (`ratio=0`). Interpolation reduces contrast, extrapolation boosts contrast, and negative factors generate inverted images. Use the following formula which is mentioned in [Wiki Contrast](http://en.wikipedia.org/wiki/Image_editing#Contrast_change_and_brightening): `value = (value - 0.5) * (tan ((contrast + 1) * PI/4) ) + 0.5;` data:image/s3,"s3://crabby-images/533c8/533c80be71f8e40b559f83889cb5b2d91da442d3" alt="-1" data:image/s3,"s3://crabby-images/30809/308099a7398223f1e70cbd59bf2c5fc6fcd09ba7" alt="-0.5" data:image/s3,"s3://crabby-images/d3194/d31940d3b633482536973f8c073ecb7218dfea28" alt="0" data:image/s3,"s3://crabby-images/cdeb8/cdeb86d117326cc945f29b7b5718fd64c49026b9" alt="0.5" data:image/s3,"s3://crabby-images/f2a89/f2a89741e0327dc3a415b2209918c5495ff736dc" alt="1.0" Gamma Correction ------------------------------------------------------------------------ `gammaFilter(image, logOfGamma)` changes the image by applying gamma correction, $V_{out} = V_{in} ^ \gamma$, where $\gamma = e^{\log{\gamma}}$ data:image/s3,"s3://crabby-images/e150b/e150b0d56cfeee7d840ca5e3d0cb45d80b12f2e4" alt="-1" data:image/s3,"s3://crabby-images/6b29d/6b29d1d0f5d7d276e264dbc706b43eed35bfef3a" alt="-0.4" data:image/s3,"s3://crabby-images/6ed0f/6ed0f95c144959c8032e49c9ac9787960b0797e1" alt="0" data:image/s3,"s3://crabby-images/2d4d4/2d4d40b684fdcbf8094c4de0aeefebd1643e60de" alt="0.4" data:image/s3,"s3://crabby-images/66226/662261c5db3af49dbdf65ef9ec3e6e725c018fc9" alt="1.0" Vignette ------------------------------------------------------------------------ `vignetteFilter(image, value)` darkens the corners of the image, as observed when using lenses with very wide apertures ([ref](http://en.wikipedia.org/wiki/Vignetting)). The function takes the innerRadius and outerRadius as inputs. The image should be perfectly clear up to `innerRadius`, perfectly dark (black) at `outerRadius` and beyond, and smoothly increase darkness in the circular ring in between. Both are specified as multiples of half the length of the image diagonal (so 1.0 is the distance from the image center to the corner). Note: the vignetting ring should be a perfect circle, not an ellipse. Camera lenses typically have circular apertures, even if the sensor/film is rectangular. data:image/s3,"s3://crabby-images/73ef4/73ef4c31ad108ba538bf47cb593ed7afd3c3b4f5" alt="innerR:0.25, outerR:1" data:image/s3,"s3://crabby-images/2a1a0/2a1a008babe877e84a40590663b63ed01fd5f773" alt="innerR:0.5, outerR:1" data:image/s3,"s3://crabby-images/495da/495dafd54c1397ed6b76f573d2a0d9769d11e7df" alt="innerR:0.25, outerR:0.75" data:image/s3,"s3://crabby-images/5d40e/5d40eed800be5d13777607ea71dcc1387b723cba" alt="innerR:0, outerR:0.75" Histogram Equalization ------------------------------------------------------------------------ `histeqFilter(image)` increase the contrast of the image by [histogram equalization](http://en.wikipedia.org/wiki/Histogram_equalization) in HSL’s L channel, that is, remapping the pixel intensities so that the final histogram is flat. A low contrast image usually clumps most pixels into a few tight clusters of intensities. Histogram equalization redistributes the pixel intensities uniformly over the full range of intensities [0, 1], while maintaining the relationship between light and dark areas of the image. data:image/s3,"s3://crabby-images/1f6e9/1f6e9e9f60459e7778ae697b773453beeb4d6449" alt="Before" data:image/s3,"s3://crabby-images/8dc4f/8dc4f4d36627052fda8fc18c3acbcf0f047fe54a" alt="After" Color Operations ======================================================================== Saturation ------------------------------------------------------------------------ `saturationFilter(image, ratio)` changes the saturation of an image by interpolating between a gray level version of the image (`ratio=-1`) and the original image (`ratio=0`). Interpolation decreases saturation, extrapolation increases it, and negative factors preserve luminance but invert the hue of the input image. See [Graphica Obscura](http://graficaobscura.com/interp/index.html), its parameter `alpha=1+ratio` in our slider. data:image/s3,"s3://crabby-images/4d952/4d952a1956635e198afc53c7c636de91913b7f3b" alt="-1" data:image/s3,"s3://crabby-images/5c957/5c957f2e9ea4fd0d66612916b1222fa7eaaf226e" alt="-0.5" data:image/s3,"s3://crabby-images/72367/723678236cdc4a7407e94776aa0c7a1dc231fb04" alt="0" data:image/s3,"s3://crabby-images/95827/95827208a930ebcbe0881f987179baea9def0271" alt="0.5" data:image/s3,"s3://crabby-images/ab12b/ab12b0c05e50e2ad6709dd386490751751c05e1d" alt="1.0" White Balance ------------------------------------------------------------------------ `whiteBalanceFilter(image, hex)` Adjust the [white balance](http://en.wikipedia.org/wiki/Color_balance) of the scene to compensate for lighting that is too warm, too cool, or tinted, to produce a neutral image. Use [Von Kries method](http://en.wikipedia.org/wiki/Color_balance#Von_Kries.27s_method): convert the image from RGB to the [LMS color space](http://en.wikipedia.org/wiki/LMS_color_space>) (there are several slightly different versions of this space, use any reasonable one, e.g. RLAB), divide by the LMS coordinates of the white point color (the estimated tint of the illumination), and convert back to RGB. data:image/s3,"s3://crabby-images/d1cdc/d1cdce0f50004267d3a659bbcdc58ff73296468e" alt="Before correction: too warm" data:image/s3,"s3://crabby-images/0965d/0965db950d7e75652bfe15a403c3979a7b38391b" alt="After correction: neutral" data:image/s3,"s3://crabby-images/2a5f2/2a5f268a8eaeb48b2cd753754e77d96c10fec9de" alt="Given white hex: #cee2f5" data:image/s3,"s3://crabby-images/b0c47/b0c47a7641d7919d6505e4caad12877d018bf1a8" alt="Given white hex: #f5cece" Historgram Matching ------------------------------------------------------------------------ `histMatchFilter = function(image, refImg)` Adjusts the color/contrast of the input `image` by matching the histgram to `refImg` images in the luminance channel. The results in the row below use the histogram matching in the luminance channel. data:image/s3,"s3://crabby-images/559eb/559ebc11e296a1595ff1b2e7d8b61071585dc5aa" alt="reference image: town" data:image/s3,"s3://crabby-images/d281b/d281b96a62ed9848f0bf731a077c6b94876f6c7c" alt="reference image: flower" Filter Operations ======================================================================== Gaussian Filter ------------------------------------------------------------------------ `gaussianFilter(image, sigma)` Blurs an image by convolving it with a Gaussian filter. In the examples below, the Gaussian function used was $G(x) = e^{\frac{-x^2}{2\sigma^2}}$ and the number below each image indicates the sigma of the filter. You set the filter window size to `Math.round(3*sigma)*2+1`. data:image/s3,"s3://crabby-images/93dc7/93dc7c4551967b8a2259713c300c5b4dddd523df" alt="1" data:image/s3,"s3://crabby-images/7e656/7e65665fce2046480288b0908e5fb9e285baded1" alt="2" data:image/s3,"s3://crabby-images/51718/51718b396d14c739205adf28aafb66cf4b16b174" alt="3" data:image/s3,"s3://crabby-images/c382f/c382f6998887a522e52ef84718fc4dbd2355addc" alt="4" data:image/s3,"s3://crabby-images/dfb0c/dfb0c3d566437e745409286a74a71b511bb89b33" alt="5" Sharpen Filter ------------------------------------------------------------------------ `sharpenFilter(image)` sharpens edges in an image by convolving it with the edge kernel as belows and add it to the original image: \begin{bmatrix} -1 & -1 & -1 \\ -1 & 8 & -1 \\ -1 & -1 & -1 \end{bmatrix} data:image/s3,"s3://crabby-images/ef424/ef4248b559e0944dce2c5427c6200b563704a226" alt="" Edge ------------------------------------------------------------------------ `edgeFilter = function(image)` convolves the image with the edge kernel. We invert the image (`pixel = 1 - pixel`) in the example below for better visualization. data:image/s3,"s3://crabby-images/79ade/79ade1d96c58912ec922ade97d9bbe9f6d7c8b92" alt="" data:image/s3,"s3://crabby-images/426a0/426a05462061af252040dbf674b7d755b47267a1" alt="" Median Filter ------------------------------------------------------------------------ `medianFilter(image, winR)` blurs an image by replacing each pixel by the median of its neighboring pixel(`(2*winR+1) * (2*winR+1)`). The results below are generated by doing median filter in RGB channel separately. You can also sort the pixels using the luminance only. data:image/s3,"s3://crabby-images/40219/402193c78050ba5a6c0f139f74ff7f5058ff070a" alt="1" data:image/s3,"s3://crabby-images/f6698/f66983983d16703b713316f0da6289904d2f2fbe" alt="2" data:image/s3,"s3://crabby-images/7db8e/7db8e1f0808b93d99e6e657fd900476f350f0683" alt="3" data:image/s3,"s3://crabby-images/bc3b7/bc3b7a98366bbdcf6c4ad7bfb4a657aedb98a909" alt="4" data:image/s3,"s3://crabby-images/56442/56442a689f699554620ff2be93af1a8b7fc0f625" alt="5" Bilateral Filter ------------------------------------------------------------------------ `bilateralFilter(image, sigmaR, sigmaS)` blurs an image by replacing each pixel by a weighted average of nearby pixels. The weights depend not only on the euclidean distance of pixels but also on the pixel difference, for the pixel difference it could either be luminance difference or L2 distance in color space. Consider the pixel I(i,j) located in (i,j), the weight of pixel I(k,l) follows the following equation: data:image/s3,"s3://crabby-images/1111f/1111f907d72407d6524b796561a59028634b7264" alt="" We set the filter window size to `2* Math.round( max(sigmaR,sigmaS)*2 ) + 1`. In the examples below, to make the weight for pixel and spatial more fair, we multiply `sigmaR` by `sqrt(2)*winR`. If we don't take this factor into consideration, the filter does not do any blurring and the result looks unchanged. data:image/s3,"s3://crabby-images/248ca/248ca8406976a5500eeb42fd163368ed89302282" alt="sigmaR=1, sigmaS=1" data:image/s3,"s3://crabby-images/d3091/d3091d1fceefb104b5e8857773c7f9971b08ef92" alt="sigmaR=2, sigmaS=1" data:image/s3,"s3://crabby-images/04f10/04f10f6079f940874651112e5265d1faa4f0733e" alt="sigmaR=3, sigmaS=0.5" data:image/s3,"s3://crabby-images/b441c/b441c6c8b0d283000790c4d48a6d2eb8fbae1722" alt="sigmaR=4, sigmaS=2" data:image/s3,"s3://crabby-images/2d6f6/2d6f6b15db8332434e4061c34e79e233fd32a0d0" alt="sigmaR=5, sigmaS=3" Dithering Operations ======================================================================== Quantization ------------------------------------------------------------------------ `quantizeFilter(image)` converts an image to a binary image (0 or 1). data:image/s3,"s3://crabby-images/85565/855652020e17af8c0b5e43e85fd8e73f8a96bf7f" alt="" Random Dither ------------------------------------------------------------------------ `randomFilter(image)` converts an image to a binary image using random dithering. It is similar to uniform quantization, but random noise range in each unit is added to each component during quantization, so that the arithmetic mean of many output pixels with the same input level will be equal to this input level. data:image/s3,"s3://crabby-images/fd86e/fd86e99132dc83ed1139d13cd7badd0bcb10694f" alt="" Floyd-Steinberg Dither ------------------------------------------------------------------------ `floydFilter(image)` converts an image to a binary image using Floyd-Steinberg dither with error diffusion. Each pixel (_x_,_y_) is quantized, and the quantization error is computed. Then the error is diffused to the neighboring pixels (_x_ + 1,_y_), (_x_ - 1,_y_ + 1), (_x_,_y_ + 1), and (_x_ + 1,_y_ + 1) , with weights 7/16, 3/16, 5/16, and 1/16, respectively. data:image/s3,"s3://crabby-images/eb75b/eb75b8ce42872f9413baeffd5e49cacb0c18dc4d" alt="" Ordered Dither ------------------------------------------------------------------------ `orderedFilter(image)` converts an image to a binary image using ordered dithering. The following examples used the pattern: \begin{bmatrix} 15 & 7 & 13 & 5 \\ 3 & 11 & 1 & 9 \\ 12 & 4 & 14 & 6 \\ 0 & 8 & 2 & 10 \end{bmatrix} The values can be used as thresholds for rounding up or down as described in the lecture slides. data:image/s3,"s3://crabby-images/dc23a/dc23a0092e72df7c44f077594acaa81272a8e576" alt="" Resampling Operations ======================================================================== Scale ------------------------------------------------------------------------ `scaleFilter(image, ratio)` scales an image in width and height by ratio. The result depends on the current sampling method (point, bilinear, or Gaussian). In the example below, gamma=1, the window radius of the Gaussian filter is 3, ratio = 0.7. data:image/s3,"s3://crabby-images/64e5a/64e5ab793ab5771efb4956c1734d241f75344b0a" alt="Point" data:image/s3,"s3://crabby-images/04bb5/04bb56525e8431bace9884a82ef5c3818fe06ac9" alt="Bilinear" data:image/s3,"s3://crabby-images/eaa68/eaa6848b96a5fcf037f18f9239e241020a16e6e5" alt="Gaussian" Rotate ------------------------------------------------------------------------ `rotateFilter(image, radians, sampleMode)` rotates an image by the given angle, in radians (a positive angle implies clockwise rotation). The result depends on the current sampling method (point, bilinear, or Gaussian). We set sigma of the gaussian filter to 1.0, and the window radius of the Gaussian filter to 3.0. \ In the example below, radians = 0.2 * pi. data:image/s3,"s3://crabby-images/9d588/9d588c46aadf194f5b320aaad684a969f90b2be6" alt="Point" data:image/s3,"s3://crabby-images/1606f/1606fe0dbd173bd20ebdb262454735d8d8243234" alt="Bilinear" data:image/s3,"s3://crabby-images/9d350/9d350721b9c0ee72a38a1882abc64790c0ffc436" alt="Gaussian" Swirl ------------------------------------------------------------------------ `swirlFilter(image, radians, sampleMode)` warps an image using a creative filter of your choice. In the following example, each pixel is mapped to its corresponding scaled polar coordinates, here radians = 0.4 * pi. data:image/s3,"s3://crabby-images/0c78a/0c78ab6f4e28a4bc7f58d3824744e996e59a2e7a" alt="Point" data:image/s3,"s3://crabby-images/fdc60/fdc609798e9d8aa84a8ca7a46ae74d367a9dfd96" alt="Bilinear" data:image/s3,"s3://crabby-images/0343d/0343dea63c0b5a62e786de17d3a27c267b0be05b" alt="Gaussian" Composite Operations ======================================================================== Composite ------------------------------------------------------------------------ `compositeFilter(backgroundImg, foregroundImg)` composites the foreground image over the background image, using the alpha channel of the foreground image to blend two images. The alpha channel can be obtained by pushing a third image or painting a third image using "Brush". Gaussian smoothing the painted alpha channel usually gives better result. data:image/s3,"s3://crabby-images/368d7/368d776a0ea5f6f9d98024cc8776e02bee4f4129" alt="Background Img" data:image/s3,"s3://crabby-images/c4722/c4722a99d8d7dde0e4b84d48d84b088844f45157" alt="Foreground Img" data:image/s3,"s3://crabby-images/ed5f0/ed5f04015b9106c60cbc35bdd342fce82e829dfc" alt="Foreground Img(alpha channel)" data:image/s3,"s3://crabby-images/5c39f/5c39fe93da05360a1b6c1a5a3926543bf7859f03" alt="Result" data:image/s3,"s3://crabby-images/17e55/17e551cef672777426ee88b9e1c2348006ddcbd8" alt="Background Img" data:image/s3,"s3://crabby-images/d820f/d820f34e74bb0c987b13a96b5cb63f8d7c4d1a55" alt="Foreground Img" data:image/s3,"s3://crabby-images/b33d0/b33d02420f3237ebbf37ab35b69c5c118a87d300" alt="Foreground Img(alpha channel)" data:image/s3,"s3://crabby-images/5343b/5343b43327ffa014ade9b2a15970d9ad85160d4f" alt="Result" Morph ------------------------------------------------------------------------ `morphFilter(initialImg, finalImg, lines, alpha) ` Morph two images using [[Beier92]](http://www.hammerhead.com/thad/morph.html). `initialImg` and `finalImg` are the before and after images, respectively. `lines` are corresponding line segments to be aligned. `alpha` is the morph time: it can be a number between 0 and 1 indicating which point in the morph sequence should be returned, or can be (start:step:end) to define a morph sequence. In terms of parameter choosing, we set p = 0.5, a = 0.01, and b = 2. data:image/s3,"s3://crabby-images/c5289/c5289b090f51470e009e0299fd212be42fb89689" alt="0" data:image/s3,"s3://crabby-images/f7a24/f7a241222397e6deb8d74306e35744b52e751264" alt="0.11" data:image/s3,"s3://crabby-images/09cb2/09cb2a4acbd908419c742ef2031e483abc22533d" alt="0.22" data:image/s3,"s3://crabby-images/96a61/96a61a10468ab5c2382dc4f7393d288438b70da9" alt="0.33" data:image/s3,"s3://crabby-images/5e44e/5e44eb2f8be437748ac3c1482db92c2e8b2ab0e1" alt="0.44" data:image/s3,"s3://crabby-images/4ecee/4eceef28a49b7065cc9c14496259cb3cc5e8f8b3" alt="0.56" data:image/s3,"s3://crabby-images/f96f0/f96f0b6ab2b229617c6e399a0188ce7d3f07590a" alt="0.67" data:image/s3,"s3://crabby-images/8460b/8460bce27fd5c01fe504bdc1de164c167b045313" alt="0.78" data:image/s3,"s3://crabby-images/021fd/021fd9d6b46293b6e175992097a482af2cfbfcaf" alt="0.89" data:image/s3,"s3://crabby-images/ad4bd/ad4bdb173b9f2d701d08504d3f5beb443cbe5978" alt="1.0" Here is an animation of the sequence: data:image/s3,"s3://crabby-images/ed34f/ed34f469fa3b42728f03c5071a93c93d923cd76f" alt="" And here is the morph for parameters `alpha=(0:0.1:1)` for the example images and morph lines provided with the assignment zip, together with a still frame at `alpha=0.5`: data:image/s3,"s3://crabby-images/77840/77840140896c39055d4e5670cd877b68751585c8" alt="" data:image/s3,"s3://crabby-images/c2bc6/c2bc6a4d280b442869bc11b9361727ad9834e7d9" alt="" Miscellaneous ======================================================================== Palette ------------------------------------------------------------------------ `paletteFilter(image, colorNum)` extracts colorNum colors as a palette to represent colors in the image. Here we use k-means method to extract color palette in the image with grid acceleration. data:image/s3,"s3://crabby-images/e3b36/e3b36cd693f715f70b62410b13ca224fb740db16" alt="colorNum = 2" data:image/s3,"s3://crabby-images/dd79e/dd79edc0d9c4ba119739bfca00e0ac7b992885a6" alt="colorNum = 3" data:image/s3,"s3://crabby-images/e3b36/e3b36cd693f715f70b62410b13ca224fb740db16" alt="colorNum = 5" XDoG ------------------------------------------------------------------------ `xDoGFilter(image, value)` Stylizes images into a pencil-drawing fashion using the eXtended Difference-of-Gaussians compendium described by Winnemoeller. You can use the approach of Kang to make the flow field (instead of the one in the XDoG paper) because it is based on bilateral filter that you already implemented. data:image/s3,"s3://crabby-images/c255d/c255dd41c2dd81adbefcde65e270faddbc65bf30" alt="0.75" data:image/s3,"s3://crabby-images/b260a/b260a5c39c91784c68de6d163ce76362e4f5ed87" alt="1.0"