-
Notifications
You must be signed in to change notification settings - Fork 1
/
connected_components.py
142 lines (125 loc) · 4.73 KB
/
connected_components.py
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
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
#!/usr/bin/python
# vim: set ts=2 expandtab:
"""
Module: connected_components.py
Desc:
Author: John O'Neil
Email: [email protected]
DATE: Saturday, August 10th 2013
Revision: Thursday, August 15th 2013
Connected component generation and manipulation
utility functions.
"""
import numpy as np
import scipy.ndimage
from pylab import zeros,amax,median
import cv2
def area_bb(a):
return np.prod([max(x.stop-x.start,0) for x in a[:2]])
def width_bb(a):
return a[1].stop-a[1].start
def height_bb(a):
return a[0].stop-a[0].start
def area_nz(slice, image):
return np.count_nonzero(image[slice])
def get_connected_components(image):
s = scipy.ndimage.morphology.generate_binary_structure(2,2)
labels,n = scipy.ndimage.measurements.label(image)#,structure=s)
objects = scipy.ndimage.measurements.find_objects(labels)
return objects
def bounding_boxes(image,connected_components,max_size,min_size):
mask = np.zeros(image.shape,'B')#np.uint8)#'B')
for component in connected_components:
if area_bb(component)**.5<min_size: continue
if area_bb(component)**.5>max_size: continue
#a = area_nz(component,image)
#if a<min_size: continue
#if a>max_size: continue
mask[component] = 1#255
return mask
def masks(image,connected_components,max_size,min_size):
mask = zeros(image.shape,np.uint8)#,'B')
for component in connected_components:
if area_bb(component)**.5<min_size: continue
if area_bb(component)**.5>max_size: continue
#a = area_nz(component,image)
#if a<min_size: continue
#if a>max_size: continue
#print str(image[component])
mask[component] = image[component]>0
#print str(mask[component])
return mask
def draw_bounding_boxes(img,connected_components,max_size=0,min_size=0,color=(0,0,255),line_size=2):
for component in connected_components:
if min_size > 0 and area_bb(component)**0.5<min_size: continue
if max_size > 0 and area_bb(component)**0.5>max_size: continue
#a = area_nz(component,img)
#if a<min_size: continue
#if a>max_size: continue
(ys,xs)=component[:2]
cv2.rectangle(img,(xs.start,ys.start),(xs.stop,ys.stop),color,line_size)
def filter_by_size(image,connected_components,max_size,min_size):
filtered = []
for cc in connected_components:
if area_bb(cc)**0.5<min_size: continue
if area_bb(cc)**0.5>max_size: continue
filtered.append(cc)
return filtered
def filter_by_black_white_ratio(img,connected_components,maximum=1.0,minimum=0.0):
filtered = []
for component in connected_components:
black = area_nz(component,img)
a = area_bb(component)
percent_black = float(black)/float(a)
if percent_black < minimum or percent_black > maximum:
#print 'component removed for percent ' + str(percent_black)
continue
filtered.append(component)
return filtered
def average_size(img, minimum_area=3, maximum_area=100):
components = get_connected_components(img)
sorted_components = sorted(components,key=area_bb)
#sorted_components = sorted(components,key=lambda x:area_nz(x,binary))
areas = zeros(img.shape)
for component in sorted_components:
#As the input components are sorted, we don't overwrite
#a given area again (it will already have our max value)
if amax(areas[component])>0: continue
#take the sqrt of the area of the bounding box
areas[component] = area_bb(component)**0.5
#alternate implementation where we just use area of black pixels in cc
#areas[component]=area_nz(component,binary)
#we lastly take the median (middle value of sorted array) within the region of interest
#region of interest is defaulted to those ccs between 3 and 100 pixels on a side (text sized)
aoi = areas[(areas>minimum_area)&(areas<maximum_area)]
if len(aoi)==0:
return 0
return np.median(aoi)
def mean_width(img, minimum=3, maximum=100):
components = get_connected_components(img)
sorted_components = sorted(components,key=area_bb)
widths = zeros(img.shape)
for c in sorted_components:
if amax(widths[c])>0: continue
widths[c]=width_bb(c)
aoi = widths[(widths>minimum)&(widths<maximum)]
if len(aoi)>0:
return np.mean(aoi)
return 0
def mean_height(img, minimum=3, maximum=100):
components = get_connected_components(img)
sorted_components = sorted(components,key=area_bb)
heights = zeros(img.shape)
for c in sorted_components:
if amax(heights[c])>0: continue
heights[c]=height_bb(c)
aoi = heights[(heights>minimum)&(heights<maximum)]
if len(aoi)>0:
return np.mean(aoi)
return 0
def form_mask(img, max_size, min_size):
components = get_connected_components(img)
sorted_components = sorted(components,key=area_bb)
#mask = bounding_boxes(img,sorted_components,max_size,min_size)
mask = masks(img,sorted_components,max_size,min_size)
return mask