1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62
|
import numpy as num
import convolve
import convolve._correlate as _correlate
MLab=num
def _translate(a, dx, dy, output=None, mode="nearest", cval=0.0):
"""_translate does positive sub-pixel shifts using bilinear interpolation."""
assert 0 <= dx < 1.0
assert 0 <= dy < 1.0
w = (1-dy) * (1-dx)
x = (1-dy) * dx
y = (1-dx) * dy
z = dx * dy
kernel = num.array([
[ z, y ],
[ x, w ],
])
return convolve.correlate2d(a, kernel, output, mode, cval)
def translate(a, sdx, sdy, output=None, mode="nearest", cval=0.0):
"""translate performs a translation of 'a' by (sdx, sdy)
storing the result in 'output'.
sdx, sdy are float values.
supported 'mode's include:
'nearest' elements beyond boundary come from nearest edge pixel.
'wrap' elements beyond boundary come from the opposite array edge.
'reflect' elements beyond boundary come from reflection on same array edge.
'constant' elements beyond boundary are set to 'cval'
"""
a = num.asarray(a)
sdx, sdy = -sdx, -sdy # Flip sign to match IRAF sign convention
# _translate works "backwords" due to implementation of 2x2 correlation.
if sdx >= 0 and sdy >= 0:
rotation = 2
dx, dy = abs(sdx), abs(sdy)
elif sdy < 0 and sdx >= 0:
rotation = 1
dx, dy = abs(sdy), abs(sdx)
elif sdx < 0 and sdy >= 0:
rotation = 3
dx, dy = abs(sdy), abs(sdx)
elif sdx < 0 and sdy < 0:
rotation = 0
dx, dy = abs(sdx), abs(sdy)
b = MLab.rot90(a, rotation)
c = _correlate.Shift2d(b, int(dx), int(dy),
mode=convolve.pix_modes[mode])
d = _translate(c, dx % 1, dy % 1, output, mode, cval)
if output is not None:
output._copyFrom(MLab.rot90(output, -rotation%4))
else:
return MLab.rot90(d, -rotation % 4).astype(a.type())
|