Skip to content

Commit b8d6f19

Browse files
committed
Added EDColor support to EdgeDrawing class.
Fixed a bug where params.PFmode = true caused incorrect edge detection results. Updated edge_drawing.py to support the new EDColor feature. Created a new edge_drawing_demo.cpp for demonstrating the EDColor feature. Updated ximgproc.bib to include references for EDColor. Added tests for PFmode and EDColor in test_fld.cpp. Added CV_WRAP to Params::read and Params::write functions to expose them to Python bindings. the class NFALUT uses the code from original ED code
1 parent 438e67f commit b8d6f19

File tree

9 files changed

+1075
-317
lines changed

9 files changed

+1075
-317
lines changed

modules/ximgproc/doc/ximgproc.bib

+9
Original file line numberDiff line numberDiff line change
@@ -388,6 +388,15 @@ @article{akinlar2013edcircles
388388
publisher={Elsevier}
389389
}
390390

391+
@article{akinlar201782,
392+
title = {ColorED: Color edge and segment detection by Edge Drawing (ED)},
393+
author = {Cuneyt Akinlar and Cihan Topal},
394+
journal = {Journal of Visual Communication and Image Representation},
395+
volume = {44},
396+
pages = {82-94},
397+
year = {2017},
398+
publisher={Academic Press}
399+
391400
@article{loke2021accelerated,
392401
title={Accelerated superpixel image segmentation with a parallelized DBSCAN algorithm},
393402
author={Loke, Seng Cheong and MacDonald, Bruce A and Parsons, Matthew and W{\"u}nsche, Burkhard Claus},

modules/ximgproc/include/opencv2/ximgproc.hpp

+36-12
Original file line numberDiff line numberDiff line change
@@ -83,18 +83,42 @@
8383
8484
@defgroup ximgproc_fast_line_detector Fast line detector
8585
86-
@defgroup ximgproc_edge_drawing EdgeDrawing
87-
88-
EDGE DRAWING LIBRARY FOR GEOMETRIC FEATURE EXTRACTION AND VALIDATION
89-
90-
Edge Drawing (ED) algorithm is an proactive approach on edge detection problem. In contrast to many other existing edge detection algorithms which follow a subtractive
91-
approach (i.e. after applying gradient filters onto an image eliminating pixels w.r.t. several rules, e.g. non-maximal suppression and hysteresis in Canny), ED algorithm
92-
works via an additive strategy, i.e. it picks edge pixels one by one, hence the name Edge Drawing. Then we process those random shaped edge segments to extract higher level
93-
edge features, i.e. lines, circles, ellipses, etc. The popular method of extraction edge pixels from the thresholded gradient magnitudes is non-maximal supression that tests
94-
every pixel whether it has the maximum gradient response along its gradient direction and eliminates if it does not. However, this method does not check status of the
95-
neighboring pixels, and therefore might result low quality (in terms of edge continuity, smoothness, thinness, localization) edge segments. Instead of non-maximal supression,
96-
ED points a set of edge pixels and join them by maximizing the total gradient response of edge segments. Therefore it can extract high quality edge segments without need for
97-
an additional hysteresis step.
86+
@defgroup ximgproc_edge_drawing Edge Drawing
87+
88+
Edge Drawing (ED) algorithm for geometric feature extraction and validation.
89+
90+
The Edge Drawing (ED) algorithm is a proactive approach to the edge detection problem.
91+
In contrast to many existing edge detection algorithms, which follow a subtractive
92+
approach (i.e., applying gradient filters and eliminating pixels based on several rules,
93+
such as non-maximal suppression and hysteresis in the Canny Edge Detector), the ED algorithm
94+
operates via an additive strategy. It selects edge pixels one by one and connects them,
95+
hence the name Edge Drawing.
96+
97+
ED offers several key advantages:
98+
99+
1. **Additive Strategy**: Instead of eliminating non-edge pixels after gradient filtering,
100+
ED incrementally builds up edge segments by selecting and connecting pixels based on
101+
their gradient response. This differs from traditional methods, which rely on
102+
non-maximal suppression and hysteresis to filter out non-edge pixels.
103+
104+
2. **Edge Pixel Selection**: ED selects edge pixels by analyzing their local gradient
105+
response, while also considering neighboring pixels. This results in smoother and
106+
more continuous edge segments, as ED aims to maximize the overall gradient strength
107+
along the edge segment.
108+
109+
3. **Edge Segment Formation**: Traditional methods, such as non-maximal suppression,
110+
check whether a pixel has the maximum gradient response along its gradient direction,
111+
eliminating it otherwise. However, this approach doesn't consider neighboring pixels,
112+
often resulting in lower-quality edge segments. ED, on the other hand, joins a set of
113+
edge pixels together by maximizing the total gradient response of the segment, leading
114+
to high-quality, well-localized edges.
115+
116+
4. **Higher-Level Feature Extraction**: After forming edge segments, ED enables the
117+
extraction of higher-level geometric features such as lines, circles, ellipses, and
118+
other shapes, making it useful for tasks involving geometric feature extraction and validation.
119+
120+
The ED algorithm produces continuous, smooth, and localized edge segments, making it ideal
121+
for applications requiring precise edge detection and geometric shape analysis.
98122
99123
@defgroup ximgproc_fourier Fourier descriptors
100124

modules/ximgproc/include/opencv2/ximgproc/edge_drawing.hpp

+5-5
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ namespace ximgproc
1515
//! @addtogroup ximgproc_edge_drawing
1616
//! @{
1717

18-
/** @brief Class implementing the ED (EdgeDrawing) @cite topal2012edge, EDLines @cite akinlar2011edlines, EDPF @cite akinlar2012edpf and EDCircles @cite akinlar2013edcircles algorithms
18+
/** @brief Class implementing the ED (EdgeDrawing) @cite topal2012edge, EDLines @cite akinlar2011edlines, EDPF @cite akinlar2012edpf, EDCircles @cite akinlar2013edcircles and ColorED @cite akinlar201782 algorithms.
1919
*/
2020

2121
class CV_EXPORTS_W EdgeDrawing : public Algorithm
@@ -66,13 +66,13 @@ class CV_EXPORTS_W EdgeDrawing : public Algorithm
6666
//! Default value is 1.3
6767
CV_PROP_RW double MaxErrorThreshold;
6868

69-
void read(const FileNode& fn);
70-
void write(FileStorage& fs) const;
69+
CV_WRAP void read(const FileNode& fn);
70+
CV_WRAP void write(FileStorage& fs) const;
7171
};
7272

73-
/** @brief Detects edges in a grayscale image and prepares them to detect lines and ellipses.
73+
/** @brief Detects edges in a grayscale or color image and prepares them to detect lines and ellipses.
7474
75-
@param src 8-bit, single-channel, grayscale input image.
75+
@param src 8-bit, single-channel (CV_8UC1) or color (CV_8UC3, CV_8UC4) input image.
7676
*/
7777
CV_WRAP virtual void detectEdges(InputArray src) = 0;
7878

modules/ximgproc/samples/edge_drawing.py

+106-43
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,24 @@
11
#!/usr/bin/python
22

33
'''
4-
This example illustrates how to use cv.ximgproc.EdgeDrawing class.
4+
This example script illustrates how to use cv.ximgproc.EdgeDrawing class.
5+
6+
It uses the OpenCV library to load an image, and then use the EdgeDrawing class
7+
to detect edges, lines, and ellipses. The detected features are then drawn and displayed.
8+
9+
The main loop allows the user changing parameters of EdgeDrawing by pressing following keys:
10+
11+
to toggle the grayscale conversion press 'space' key
12+
to increase MinPathLength value press '/' key
13+
to decrease MinPathLength value press '*' key
14+
to increase MinLineLength value press '+' key
15+
to decrease MinLineLength value press '-' key
16+
to toggle NFAValidation value press 'n' key
17+
to toggle PFmode value press 'p' key
18+
to save parameters to file press 's' key
19+
to load parameters from file press 'l' key
20+
21+
The program exits when the Esc key is pressed.
522
623
Usage:
724
ed.py [<image_name>]
@@ -16,71 +33,117 @@
1633
import random as rng
1734
import sys
1835

19-
rng.seed(12345)
20-
21-
def main():
22-
try:
23-
fn = sys.argv[1]
24-
except IndexError:
25-
fn = 'board.jpg'
26-
27-
src = cv.imread(cv.samples.findFile(fn))
28-
gray = cv.cvtColor(src, cv.COLOR_BGR2GRAY)
29-
cv.imshow("source", src)
30-
31-
ssrc = src.copy()*0
36+
def EdgeDrawingDemo(src, ed, EDParams, convert_to_gray):
37+
rng.seed(12345)
38+
ssrc = np.zeros_like(src)
3239
lsrc = src.copy()
3340
esrc = src.copy()
3441

35-
ed = cv.ximgproc.createEdgeDrawing()
42+
img_to_detect = cv.cvtColor(src, cv.COLOR_BGR2GRAY) if convert_to_gray else src
3643

37-
# you can change parameters (refer the documentation to see all parameters)
38-
EDParams = cv.ximgproc_EdgeDrawing_Params()
39-
EDParams.MinPathLength = 50 # try changing this value between 5 to 1000
40-
EDParams.PFmode = False # defaut value try to swich it to True
41-
EDParams.MinLineLength = 10 # try changing this value between 5 to 100
42-
EDParams.NFAValidation = True # defaut value try to swich it to False
44+
cv.imshow("source image", img_to_detect)
4345

44-
ed.setParams(EDParams)
46+
print("")
47+
print("convert_to_gray:", convert_to_gray)
48+
print("MinPathLength:", EDParams.MinPathLength)
49+
print("MinLineLength:", EDParams.MinLineLength)
50+
print("PFmode:", EDParams.PFmode)
51+
print("NFAValidation:", EDParams.NFAValidation)
52+
53+
tm = cv.TickMeter()
54+
tm.start()
4555

4656
# Detect edges
4757
# you should call this before detectLines() and detectEllipses()
48-
ed.detectEdges(gray)
58+
ed.detectEdges(img_to_detect)
4959

5060
segments = ed.getSegments()
5161
lines = ed.detectLines()
5262
ellipses = ed.detectEllipses()
5363

54-
#Draw detected edge segments
55-
for i in range(len(segments)):
56-
color = (rng.randint(0,256), rng.randint(0,256), rng.randint(0,256))
57-
cv.polylines(ssrc, [segments[i]], False, color, 1, cv.LINE_8)
64+
tm.stop()
65+
66+
print("Detection time : {:.2f} ms. using the parameters above".format(tm.getTimeMilli()))
67+
68+
# Draw detected edge segments
69+
for segment in segments:
70+
color = (rng.randint(0, 256), rng.randint(0, 256), rng.randint(0, 256))
71+
cv.polylines(ssrc, [segment], False, color, 1, cv.LINE_8)
5872

5973
cv.imshow("detected edge segments", ssrc)
6074

61-
#Draw detected lines
62-
if lines is not None: # Check if the lines have been found and only then iterate over these and add them to the image
75+
# Draw detected lines
76+
if lines is not None: # Check if the lines have been found and only then iterate over these and add them to the image
6377
lines = np.uint16(np.around(lines))
64-
for i in range(len(lines)):
65-
cv.line(lsrc, (lines[i][0][0], lines[i][0][1]), (lines[i][0][2], lines[i][0][3]), (0, 0, 255), 1, cv.LINE_AA)
78+
for line in lines:
79+
cv.line(lsrc, (line[0][0], line[0][1]), (line[0][2], line[0][3]), (0, 0, 255), 1, cv.LINE_AA)
6680

6781
cv.imshow("detected lines", lsrc)
6882

69-
#Draw detected circles and ellipses
70-
if ellipses is not None: # Check if circles and ellipses have been found and only then iterate over these and add them to the image
71-
for i in range(len(ellipses)):
72-
center = (int(ellipses[i][0][0]), int(ellipses[i][0][1]))
73-
axes = (int(ellipses[i][0][2])+int(ellipses[i][0][3]),int(ellipses[i][0][2])+int(ellipses[i][0][4]))
74-
angle = ellipses[i][0][5]
75-
color = (0, 0, 255)
76-
if ellipses[i][0][2] == 0:
77-
color = (0, 255, 0)
78-
cv.ellipse(esrc, center, axes, angle,0, 360, color, 2, cv.LINE_AA)
83+
# Draw detected circles and ellipses
84+
if ellipses is not None: # Check if circles and ellipses have been found and only then iterate over these and add them to the image
85+
for ellipse in ellipses:
86+
center = (int(ellipse[0][0]), int(ellipse[0][1]))
87+
axes = (int(ellipse[0][2] + ellipse[0][3]), int(ellipse[0][2] + ellipse[0][4]))
88+
angle = ellipse[0][5]
89+
90+
color = (0, 255, 0) if ellipse[0][2] == 0 else (0, 0, 255)
91+
92+
cv.ellipse(esrc, center, axes, angle, 0, 360, color, 2, cv.LINE_AA)
7993

8094
cv.imshow("detected circles and ellipses", esrc)
81-
cv.waitKey(0)
82-
print('Done')
8395

96+
def main():
97+
try:
98+
fn = sys.argv[1]
99+
except IndexError:
100+
fn = 'board.jpg'
101+
src = cv.imread(cv.samples.findFile(fn))
102+
if src is None:
103+
print("Error loading image")
104+
return
105+
106+
ed = cv.ximgproc.createEdgeDrawing()
107+
108+
# Set parameters (refer to the documentation for all parameters)
109+
EDParams = cv.ximgproc_EdgeDrawing_Params()
110+
EDParams.MinPathLength = 10 # try changing this value by pressing '/' and '*' keys
111+
EDParams.MinLineLength = 10 # try changing this value by pressing '+' and '-' keys
112+
EDParams.PFmode = False # default value is False, try switching by pressing 'p' key
113+
EDParams.NFAValidation = True # default value is True, try switching by pressing 'n' key
114+
115+
convert_to_gray = True
116+
key = 0
117+
118+
while key != 27:
119+
ed.setParams(EDParams)
120+
EdgeDrawingDemo(src, ed, EDParams, convert_to_gray)
121+
key = cv.waitKey()
122+
if key == 32: # space key
123+
convert_to_gray = not convert_to_gray
124+
if key == 112: # 'p' key
125+
EDParams.PFmode = not EDParams.PFmode
126+
if key == 110: # 'n' key
127+
EDParams.NFAValidation = not EDParams.NFAValidation
128+
if key == 43: # '+' key
129+
EDParams.MinLineLength = EDParams.MinLineLength + 5
130+
if key == 45: # '-' key
131+
EDParams.MinLineLength = max(0, EDParams.MinLineLength - 5)
132+
if key == 47: # '/' key
133+
EDParams.MinPathLength = EDParams.MinPathLength + 20
134+
if key == 42: # '*' key
135+
EDParams.MinPathLength = max(0, EDParams.MinPathLength - 20)
136+
if key == 115: # 's' key
137+
fs = cv.FileStorage("ed-params.xml",cv.FileStorage_WRITE)
138+
EDParams.write(fs)
139+
fs.release()
140+
print("parameters saved to ed-params.xml")
141+
if key == 108: # 'l' key
142+
fs = cv.FileStorage("ed-params.xml",cv.FileStorage_READ)
143+
if fs.isOpened():
144+
EDParams.read(fs.root())
145+
fs.release()
146+
print("parameters loaded from ed-params.xml")
84147

85148
if __name__ == '__main__':
86149
print(__doc__)

0 commit comments

Comments
 (0)