Files
scikit-image/skimage/filter/ctmf.py
T
2012-08-20 07:55:17 -07:00

95 lines
3.1 KiB
Python

'''ctmf.py - constant time per pixel median filtering with an octagonal shape
Reference: S. Perreault and P. Hebert, "Median Filtering in Constant Time",
IEEE Transactions on Image Processing, September 2007.
Originally part of CellProfiler, code licensed under both GPL and BSD licenses.
Website: http://www.cellprofiler.org
Copyright (c) 2003-2009 Massachusetts Institute of Technology
Copyright (c) 2009-2011 Broad Institute
All rights reserved.
Original author: Lee Kamentsky
'''
import numpy as np
from . import _ctmf
from ._rank_order import rank_order
def median_filter(image, radius=2, mask=None, percent=50):
'''Masked median filter with octagon shape.
Parameters
----------
image : (M,N) ndarray, dtype uint8
Input image.
radius : {int, 2}, optional
The radius of a circle inscribed into the filtering
octagon. Must be at least 2. Default radius is 2.
mask : (M,N) ndarray, dtype uint8, optional
A value of 1 indicates a significant pixel, 0
that a pixel is masked. By default, all pixels
are considered.
percent : {int, 50}, optional
The unmasked pixels within the octagon are sorted, and the
value at the `percent`-th index chosen. For example, the
default value of 50 chooses the median pixel.
Returns
-------
out : (M,N) ndarray, dtype uint8
Filtered array. In areas where the median filter does
not overlap the mask, the filtered result is underfined, but
in practice, it will be the lowest value in the valid area.
'''
if image.ndim != 2:
raise TypeError("The input 'image' must be a two dimensional array.")
if radius < 2:
raise ValueError("The input 'radius' must be >= 2.")
if mask is None:
mask = np.ones(image.shape, dtype=np.bool)
mask = np.ascontiguousarray(mask, dtype=np.bool)
if np.all(~ mask):
return image.copy()
#
# Normalize the ranked image to 0-255
#
if (not np.issubdtype(image.dtype, np.int) or
np.min(image) < 0 or np.max(image) > 255):
ranked_image, translation = rank_order(image[mask])
max_ranked_image = np.max(ranked_image)
if max_ranked_image == 0:
return image
if max_ranked_image > 255:
ranked_image = ranked_image * 255 // max_ranked_image
was_ranked = True
else:
ranked_image = image[mask]
was_ranked = False
input = np.zeros(image.shape, np.uint8)
input[mask] = ranked_image
mask.dtype = np.uint8
output = np.zeros(image.shape, np.uint8)
_ctmf.median_filter(input, mask, output, radius, percent)
if was_ranked:
#
# The translation gives the original value at each ranking.
# We rescale the output to the original ranking and then
# use the translation to look up the original value in the image.
#
if max_ranked_image > 255:
result = translation[output.astype(np.uint32) *
max_ranked_image // 255]
else:
result = translation[output]
else:
result = output
return result