Files
scikit-image/skimage/transform/_seam_carving.pyx
T
2015-06-06 22:42:18 +05:30

180 lines
5.9 KiB
Cython

# cython: cdivision=True
# cython: boundscheck=False
# cython: nonecheck=False
# cython: wraparound=False
import numpy as np
cimport numpy as cnp
cdef cnp.double_t DBL_MAX = np.finfo(np.double).max
cdef _find_seam_v(cnp.double_t[:, ::1] energy_img, cnp.int8_t[:, ::1] track_img,
cnp.double_t[::1] current_cost, cnp.double_t[::1] prev_cost,
Py_ssize_t cols):
"""Find a single vertical seam in an image that will be removed.
Parameters
----------
energy_img : (M, N) ndarray
The energy image where a higher value signifies a pixel of more
importance. Pixels with a lower value will be cropped first.
track_img : (M, N) ndarray
The image used to store the optimal decision made at each point while
finding a minimum cost path. For each pixel it stores the offset that
produced that least cost.
current_cost : (N,) ndarray
An array to store the current cost of the optimal path for each column
in row currently being processed.
prev_cost : (N,) ndarray
An array to store the current cost of the optimal path for each column
in row prior to the one being processed.
cols : int
The number of cols to process for seam carving. Columns with indices
more than `cols` are ignored.
Returns
-------
seam : (M, ) ndarray of int
An array containing the index of the row of the pixel to be removed
for each column in the image.
Notes
-----
`track_img`, `current_cost` and `prev_cost` are passed as arguments to
avoid memory allocation at each iteration of `_seam_carve_v`.
"""
cdef Py_ssize_t rows, row, col
rows = energy_img.shape[0]
cdef cnp.double_t tmp, min_cost
cdef Py_ssize_t offset, idx, offset_clip
cdef Py_ssize_t[::1] seam = np.zeros(rows, dtype=np.int)
for idx in range(cols):
prev_cost[idx] = energy_img[0, idx]
for row in range(1, rows):
for col in range(0, cols):
min_cost = DBL_MAX
for offset in range(-1, 2):
idx = col + offset
if idx > cols - 1 or idx < 0:
continue
if prev_cost[idx] < min_cost:
min_cost = prev_cost[idx]
track_img[row, col] = offset
current_cost[col] = min_cost + energy_img[row, col]
prev_cost[:] = current_cost
seam[rows-1] = np.argmin(current_cost)
for row in range(rows-2, -1, -1):
col = seam[row + 1]
offset = track_img[row, col]
seam[row] = seam[row + 1] + offset
return seam
cdef remove_seam_v(cnp.double_t[:, :, ::1] img, Py_ssize_t[::1] seam,
Py_ssize_t cols):
""" Removes one horizontal seam from the image.
The method modifies `img` so that all pixels to the right of the vertical
seam are pushed one place left.
image : (M, N, 3) ndarray
Input image whose vertical seam is to be removed.
seam : (M, ) ndarray
An array use to store the index of the column in the seam for each row.
cols : int
Number of columns in the input image to process. Column indices more
than `cols` are ingored.
Notes
-----
`seam` is passed as an argument so that we don't have to reallocate it for
each iteration in `_seam_carve_v`.
"""
cdef Py_ssize_t rows, row, col, idx
rows = img.shape[0]
for row in range(rows):
for idx in range(seam[row], cols - 1):
img[row, idx, :] = img[row, idx + 1, :]
def _seam_carve_v(img, iters, energy_func, extra_args , extra_kwargs, border):
""" Carve vertical seams off an image.
Carves out vertical seams off an image while using the given energy
function to decide the importance of each pixel.[1]
Parameters
----------
image : (M, N) or (M, N, 3) ndarray
Input image whose vertical seams are to be removed.
iters : int
Number of vertical seams are to be removed.
energy_func : callable
The function used to decide the importance of each pixel. The higher
the value corresponding to a pixel, the more the algorithm will try
to keep it in the image. For every iteration `energy_func` is called
as `energy_func(image, *extra_args, **extra_kwargs)`, where `image`
is the cropped image during each iteration and is expected to return a
(M, N) ndarray depicting each pixel's importance.
extra_args : iterable
The extra arguments supplied to `energy_func`.
extra_kwargs : dict
The extra keyword arguments supplied to `energy_func`.
border : int
The number of pixels in the right and left end of the image to be
excluded from being considered for a seam. This is important as certain
filters just ignore image boundaries and set them to `0`.
Returns
-------
image : (M, N - iters) or (M, N - iters, 3) ndarray
The cropped image with the vertical seams removed.
References
----------
.. [1] Shai Avidan and Ariel Shamir
"Seam Carving for Content-Aware Image Resizing"
http://www.cs.jhu.edu/~misha/ReadingSeminar/Papers/Avidan07.pdf
"""
cdef Py_ssize_t[::1] seam
cdef Py_ssize_t ndim = img.ndim
cdef Py_ssize_t cols = img.shape[1]
track_img = np.zeros(img.shape[0:2], dtype=np.int8)
current_cost = np.zeros_like(track_img[0], dtype = img.dtype)
prev_cost = np.zeros_like(track_img[0], dtype = img.dtype)
for i in range(iters):
sliced_img = np.squeeze(img[:, 0:cols])
energy_img = energy_func(sliced_img, *extra_args, **extra_kwargs)
# So that borders are ignored.
energy_img[:, 0:border] = DBL_MAX
energy_img[:, cols-border:cols] = DBL_MAX
seam = _find_seam_v(energy_img, track_img, current_cost, prev_cost,
cols)
remove_seam_v(img, seam, cols)
cols -= 1
return img[:, 0:cols]