dft.py 2.63 KB
Newer Older
1
#!/usr/bin/env python
2

3 4 5
# Python 2/3 compatibility
from __future__ import print_function

6 7 8 9 10 11 12 13
import cv2
import numpy as np
import sys


def shift_dft(src, dst=None):
    '''
        Rearrange the quadrants of Fourier image so that the origin is at
Andrey Kamaev's avatar
Andrey Kamaev committed
14 15
        the image center. Swaps quadrant 1 with 3, and 2 with 4.

16 17
        src and dst arrays must be equal size & type
    '''
Andrey Kamaev's avatar
Andrey Kamaev committed
18

19 20 21 22 23 24
    if dst is None:
        dst = np.empty(src.shape, src.dtype)
    elif src.shape != dst.shape:
        raise ValueError("src and dst must have equal sizes")
    elif src.dtype != dst.dtype:
        raise TypeError("src and dst must have equal types")
Andrey Kamaev's avatar
Andrey Kamaev committed
25

26 27 28 29
    if src is dst:
        ret = np.empty(src.shape, src.dtype)
    else:
        ret = dst
Andrey Kamaev's avatar
Andrey Kamaev committed
30

31
    h, w = src.shape[:2]
Andrey Kamaev's avatar
Andrey Kamaev committed
32

33 34
    cx1 = cx2 = w/2
    cy1 = cy2 = h/2
Andrey Kamaev's avatar
Andrey Kamaev committed
35

36 37 38 39
    # if the size is odd, then adjust the bottom/right quadrants
    if w % 2 != 0:
        cx2 += 1
    if h % 2 != 0:
Andrey Kamaev's avatar
Andrey Kamaev committed
40 41
        cy2 += 1

42
    # swap quadrants
Andrey Kamaev's avatar
Andrey Kamaev committed
43

44 45 46
    # swap q1 and q3
    ret[h-cy1:, w-cx1:] = src[0:cy1 , 0:cx1 ]   # q1 -> q3
    ret[0:cy2 , 0:cx2 ] = src[h-cy2:, w-cx2:]   # q3 -> q1
Andrey Kamaev's avatar
Andrey Kamaev committed
47

48 49 50
    # swap q2 and q4
    ret[0:cy2 , w-cx2:] = src[h-cy2:, 0:cx2 ]   # q2 -> q4
    ret[h-cy1:, 0:cx1 ] = src[0:cy1 , w-cx1:]   # q4 -> q2
Andrey Kamaev's avatar
Andrey Kamaev committed
51

52 53
    if src is dst:
        dst[:,:] = ret
Andrey Kamaev's avatar
Andrey Kamaev committed
54

55 56 57
    return dst

if __name__ == "__main__":
Andrey Kamaev's avatar
Andrey Kamaev committed
58

59 60 61
    if len(sys.argv)>1:
        im = cv2.imread(sys.argv[1])
    else :
Dmitriy Anisimov's avatar
Dmitriy Anisimov committed
62
        im = cv2.imread('../data/baboon.jpg')
63
        print("usage : python dft.py <image_file>")
64 65 66 67

    # convert to grayscale
    im = cv2.cvtColor(im, cv2.COLOR_BGR2GRAY)
    h, w = im.shape[:2]
Andrey Kamaev's avatar
Andrey Kamaev committed
68

69
    realInput = im.astype(np.float64)
Andrey Kamaev's avatar
Andrey Kamaev committed
70

71 72 73 74 75 76 77
    # perform an optimally sized dft
    dft_M = cv2.getOptimalDFTSize(w)
    dft_N = cv2.getOptimalDFTSize(h)

    # copy A to dft_A and pad dft_A with zeros
    dft_A = np.zeros((dft_N, dft_M, 2), dtype=np.float64)
    dft_A[:h, :w, 0] = realInput
Andrey Kamaev's avatar
Andrey Kamaev committed
78

79 80 81
    # no need to pad bottom part of dft_A with zeros because of
    # use of nonzeroRows parameter in cv2.dft()
    cv2.dft(dft_A, dst=dft_A, nonzeroRows=h)
Andrey Kamaev's avatar
Andrey Kamaev committed
82

83
    cv2.imshow("win", im)
Andrey Kamaev's avatar
Andrey Kamaev committed
84

85 86
    # Split fourier into real and imaginary parts
    image_Re, image_Im = cv2.split(dft_A)
Andrey Kamaev's avatar
Andrey Kamaev committed
87

88 89
    # Compute the magnitude of the spectrum Mag = sqrt(Re^2 + Im^2)
    magnitude = cv2.sqrt(image_Re**2.0 + image_Im**2.0)
Andrey Kamaev's avatar
Andrey Kamaev committed
90

91 92
    # Compute log(1 + Mag)
    log_spectrum = cv2.log(1.0 + magnitude)
Andrey Kamaev's avatar
Andrey Kamaev committed
93

94 95 96 97 98
    # Rearrange the quadrants of Fourier image so that the origin is at
    # the image center
    shift_dft(log_spectrum, log_spectrum)

    # normalize and display the results as rgb
99
    cv2.normalize(log_spectrum, log_spectrum, 0.0, 1.0, cv2.NORM_MINMAX)
100 101 102 103
    cv2.imshow("magnitude", log_spectrum)

    cv2.waitKey(0)
    cv2.destroyAllWindows()