Created
February 10, 2020 16:36
-
-
Save sciunto/079fd7d466ca50bd8f6e133f882b035d to your computer and use it in GitHub Desktop.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
def canny(image, sigma=1., low_threshold=None, high_threshold=None, mask=None, | |
use_quantiles=False): | |
"""Edge filter an image using the Canny algorithm. | |
Parameters | |
----------- | |
image : 2D array | |
Grayscale input image to detect edges on; can be of any dtype. | |
sigma : float, optional | |
Standard deviation of the Gaussian filter. | |
low_threshold : float, optional | |
Lower bound for hysteresis thresholding (linking edges). | |
If None, low_threshold is set to 10% of dtype's max. | |
high_threshold : float, optional | |
Upper bound for hysteresis thresholding (linking edges). | |
If None, high_threshold is set to 20% of dtype's max. | |
mask : array, dtype=bool, optional | |
Mask to limit the application of Canny to a certain area. | |
use_quantiles : bool, optional | |
If True then treat low_threshold and high_threshold as quantiles of the | |
edge magnitude image, rather than absolute edge magnitude values. If True | |
then the thresholds must be in the range [0, 1]. | |
Returns | |
------- | |
output : 2D array (image) | |
The binary edge map. | |
See also | |
-------- | |
skimage.sobel | |
Notes | |
----- | |
The steps of the algorithm are as follows: | |
* Smooth the image using a Gaussian with ``sigma`` width. | |
* Apply the horizontal and vertical Sobel operators to get the gradients | |
within the image. The edge strength is the norm of the gradient. | |
* Thin potential edges to 1-pixel wide curves. First, find the normal | |
to the edge at each point. This is done by looking at the | |
signs and the relative magnitude of the X-Sobel and Y-Sobel | |
to sort the points into 4 categories: horizontal, vertical, | |
diagonal and antidiagonal. Then look in the normal and reverse | |
directions to see if the values in either of those directions are | |
greater than the point in question. Use interpolation to get a mix of | |
points instead of picking the one that's the closest to the normal. | |
* Perform a hysteresis thresholding: first label all points above the | |
high threshold as edges. Then recursively label any point above the | |
low threshold that is 8-connected to a labeled point as an edge. | |
References | |
----------- | |
.. [1] Canny, J., A Computational Approach To Edge Detection, IEEE Trans. | |
Pattern Analysis and Machine Intelligence, 8:679-714, 1986 | |
:DOI:`10.1109/TPAMI.1986.4767851` | |
.. [2] William Green's Canny tutorial | |
https://en.wikipedia.org/wiki/Canny_edge_detector | |
Examples | |
-------- | |
>>> from skimage import feature | |
>>> # Generate noisy image of a square | |
>>> im = np.zeros((256, 256)) | |
>>> im[64:-64, 64:-64] = 1 | |
>>> im += 0.2 * np.random.rand(*im.shape) | |
>>> # First trial with the Canny filter, with the default smoothing | |
>>> edges1 = feature.canny(im) | |
>>> # Increase the smoothing for better results | |
>>> edges2 = feature.canny(im, sigma=3) | |
""" | |
# | |
# The steps involved: | |
# | |
# * Smooth using the Gaussian with sigma above. | |
# | |
# * Apply the horizontal and vertical Sobel operators to get the gradients | |
# within the image. The edge strength is the sum of the magnitudes | |
# of the gradients in each direction. | |
# | |
# * Find the normal to the edge at each point using the arctangent of the | |
# ratio of the Y sobel over the X sobel - pragmatically, we can | |
# look at the signs of X and Y and the relative magnitude of X vs Y | |
# to sort the points into 4 categories: horizontal, vertical, | |
# diagonal and antidiagonal. | |
# | |
# * Look in the normal and reverse directions to see if the values | |
# in either of those directions are greater than the point in question. | |
# Use interpolation to get a mix of points instead of picking the one | |
# that's the closest to the normal. | |
# | |
# * Label all points above the high threshold as edges. | |
# * Recursively label any point above the low threshold that is 8-connected | |
# to a labeled point as an edge. | |
# | |
# Regarding masks, any point touching a masked point will have a gradient | |
# that is "infected" by the masked point, so it's enough to erode the | |
# mask by one and then mask the output. We also mask out the border points | |
# because who knows what lies beyond the edge of the image? | |
# | |
check_nD(image, 2) | |
dtype_max = dtype_limits(image, clip_negative=False)[1] | |
if low_threshold is None: | |
low_threshold = 0.1 | |
elif use_quantiles: | |
if not(0.0 <= low_threshold <= 1.0): | |
raise ValueError("Quantile thresholds must be between 0 and 1.") | |
else: | |
low_threshold = low_threshold / dtype_max | |
if not(0.0 <= low_threshold <= 1.0): | |
raise ValueError("Quantile thresholds must be between 0 and 1.") | |
if high_threshold is None: | |
high_threshold = 0.2 | |
elif use_quantiles: | |
if not(0.0 <= high_threshold <= 1.0): | |
raise ValueError("Quantile thresholds are normalized by the maximum dtype. " | |
"The resulting thresholds must be between 0 and 1.") | |
else: | |
high_threshold = high_threshold / dtype_max | |
if not(0.0 <= high_threshold <= 1.0): | |
raise ValueError("Quantile thresholds are normalized by the maximum dtype. " | |
"The resulting thresholds must be between 0 and 1.") | |
if mask is None: | |
mask = np.ones(image.shape, dtype=bool) | |
def fsmooth(x): | |
return img_as_float(gaussian(x, sigma, mode='constant')) | |
smoothed = smooth_with_function_and_mask(image, fsmooth, mask) | |
jsobel = ndi.sobel(smoothed, axis=1) | |
isobel = ndi.sobel(smoothed, axis=0) | |
abs_isobel = np.abs(isobel) | |
abs_jsobel = np.abs(jsobel) | |
magnitude = np.hypot(isobel, jsobel) | |
# | |
# Make the eroded mask. Setting the border value to zero will wipe | |
# out the image edges for us. | |
# | |
s = generate_binary_structure(2, 2) | |
eroded_mask = binary_erosion(mask, s, border_value=0) | |
eroded_mask = eroded_mask & (magnitude > 0) | |
# | |
#--------- Find local maxima -------------- | |
# | |
# Assign each point to have a normal of 0-45 degrees, 45-90 degrees, | |
# 90-135 degrees and 135-180 degrees. | |
# | |
local_maxima = np.zeros(image.shape, bool) | |
#----- 0 to 45 degrees ------ | |
pts_plus = (isobel >= 0) & (jsobel >= 0) & (abs_isobel >= abs_jsobel) | |
pts_minus = (isobel <= 0) & (jsobel <= 0) & (abs_isobel >= abs_jsobel) | |
pts = pts_plus | pts_minus | |
pts = eroded_mask & pts | |
# Get the magnitudes shifted left to make a matrix of the points to the | |
# right of pts. Similarly, shift left and down to get the points to the | |
# top right of pts. | |
c1 = magnitude[1:, :][pts[:-1, :]] | |
c2 = magnitude[1:, 1:][pts[:-1, :-1]] | |
m = magnitude[pts] | |
w = abs_jsobel[pts] / abs_isobel[pts] | |
c_plus = c2 * w + c1 * (1 - w) <= m | |
c1 = magnitude[:-1, :][pts[1:, :]] | |
c2 = magnitude[:-1, :-1][pts[1:, 1:]] | |
c_minus = c2 * w + c1 * (1 - w) <= m | |
local_maxima[pts] = c_plus & c_minus | |
#----- 45 to 90 degrees ------ | |
# Mix diagonal and vertical | |
# | |
pts_plus = (isobel >= 0) & (jsobel >= 0) & (abs_isobel <= abs_jsobel) | |
pts_minus = (isobel <= 0) & (jsobel <= 0) & (abs_isobel <= abs_jsobel) | |
pts = pts_plus | pts_minus | |
pts = eroded_mask & pts | |
c1 = magnitude[:, 1:][pts[:, :-1]] | |
c2 = magnitude[1:, 1:][pts[:-1, :-1]] | |
m = magnitude[pts] | |
w = abs_isobel[pts] / abs_jsobel[pts] | |
c_plus = c2 * w + c1 * (1 - w) <= m | |
c1 = magnitude[:, :-1][pts[:, 1:]] | |
c2 = magnitude[:-1, :-1][pts[1:, 1:]] | |
c_minus = c2 * w + c1 * (1 - w) <= m | |
local_maxima[pts] = c_plus & c_minus | |
#----- 90 to 135 degrees ------ | |
# Mix anti-diagonal and vertical | |
# | |
pts_plus = (isobel <= 0) & (jsobel >= 0) & (abs_isobel <= abs_jsobel) | |
pts_minus = (isobel >= 0) & (jsobel <= 0) & (abs_isobel <= abs_jsobel) | |
pts = pts_plus | pts_minus | |
pts = eroded_mask & pts | |
c1a = magnitude[:, 1:][pts[:, :-1]] | |
c2a = magnitude[:-1, 1:][pts[1:, :-1]] | |
m = magnitude[pts] | |
w = abs_isobel[pts] / abs_jsobel[pts] | |
c_plus = c2a * w + c1a * (1.0 - w) <= m | |
c1 = magnitude[:, :-1][pts[:, 1:]] | |
c2 = magnitude[1:, :-1][pts[:-1, 1:]] | |
c_minus = c2 * w + c1 * (1.0 - w) <= m | |
local_maxima[pts] = c_plus & c_minus | |
#----- 135 to 180 degrees ------ | |
# Mix anti-diagonal and anti-horizontal | |
# | |
pts_plus = (isobel <= 0) & (jsobel >= 0) & (abs_isobel >= abs_jsobel) | |
pts_minus = (isobel >= 0) & (jsobel <= 0) & (abs_isobel >= abs_jsobel) | |
pts = pts_plus | pts_minus | |
pts = eroded_mask & pts | |
c1 = magnitude[:-1, :][pts[1:, :]] | |
c2 = magnitude[:-1, 1:][pts[1:, :-1]] | |
m = magnitude[pts] | |
w = abs_jsobel[pts] / abs_isobel[pts] | |
c_plus = c2 * w + c1 * (1 - w) <= m | |
c1 = magnitude[1:, :][pts[:-1, :]] | |
c2 = magnitude[1:, :-1][pts[:-1, 1:]] | |
c_minus = c2 * w + c1 * (1 - w) <= m | |
local_maxima[pts] = c_plus & c_minus | |
# | |
#---- If use_quantiles is set then calculate the thresholds to use | |
# | |
if use_quantiles: | |
high_threshold = np.percentile(magnitude, 100.0 * high_threshold) | |
low_threshold = np.percentile(magnitude, 100.0 * low_threshold) | |
# | |
#---- Create two masks at the two thresholds. | |
# | |
high_mask = local_maxima & (magnitude >= high_threshold) | |
low_mask = local_maxima & (magnitude >= low_threshold) | |
# | |
# Segment the low-mask, then only keep low-segments that have | |
# some high_mask component in them | |
# | |
strel = np.ones((3, 3), bool) | |
labels, count = label(low_mask, strel) | |
if count == 0: | |
return low_mask | |
sums = (np.array(ndi.sum(high_mask, labels, | |
np.arange(count, dtype=np.int32) + 1), | |
copy=False, ndmin=1)) | |
good_label = np.zeros((count + 1,), bool) | |
good_label[1:] = sums > 0 | |
output_mask = good_label[labels] | |
return output_mask |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment