-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathLabSegment.m
205 lines (164 loc) · 7.95 KB
/
LabSegment.m
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
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
% Author: Hersh Godse
% Using: http://www.mathworks.com/help/images/examples/correcting-nonuniform-illumination.html
%% Pre-configuration:
% Set current folder to file where image is
% Image should be called "fabric.png"
%% Get the image:
fabric = imread('695.jpg');
% figure(1), imshow(fabric), title('fabric');
%% Calculate sample colors in L*a*b color space for each image color
% L*a*b* space consists of luminosity/brightness L* layer,
% chromaticity a* layer indicating where color falls along red-green axis,
% and chromaticity b* layer indicating where color falls along blue-yellow axis.
% we want to find small sample region for each color, and calculate region's
% avg color in a*b* space; will use these color markers to classify each image pixel
load regioncoordinates;
% n colors to segment image by (in this image case, 6)
nColors = 2;
% create empty matrix of dimentions image x, image y, n number of colors
sample_regions = false([size(fabric,1) size(fabric,2) nColors]);
% cycle through the n colors, in each loop change specific layer of sample_regions
% roipoly specifies polygonal region of interest (ROI) in image. returns
% binary image, which we will use as mask for filtering. ROI is object.
% the 2 vectors passed to roipoly in 2nd and 3rd arg create ROI defined
% by those 2 vectors. The 2 vectors specify column and row position,
% respectively, of each point in the ROI. Some points identical to create
% closed shape.
for count = 1:nColors
sample_regions(:,:,count) = roipoly(fabric);
end
% imshow(sample_regions(:,:,2)),title('sample region for red');
% Convert RGB image into an L*a*b* image
lab = applycform(fabric, makecform('srgb2lab'));
% Calculate avg a*,b* value for each ROI from roipoly. These values function
% as color markers in a*b* space.
% L = lab(:,:,1);
a = lab(:,:,2);
b = lab(:,:,3);
% color_markers = zeros([nColors, 2]);
% for each of n colors, for every pixel in image cooresponding to those in
% color's ROI, put average a*/b* color in 1st/2nd column respectively
for count = 1:nColors
color_markers(count,1) = mean2(a(sample_regions(:,:,count)));
color_markers(count,2) = mean2(b(sample_regions(:,:,count)));
end
color_markers
% color_markers =[168.9801,146.0199; 132.0216,140.7208];
% color_markers =[169.8017,146.8760; 134.9135,141.4784];
% color_markers =[170.5556,146.6667; 131.5000,140.3276];
% Avg color of red sample region in a*b* space is
% fprintf('[%0.3f,%0.3f] \n',color_markers(2,1),color_markers(2,2));
%% Classify each pixel using nearest neighbor rule
% now we know a*,b* values for each color, use these to classify pixels of image
% for each pixel of image, get (a*,b*) coordinate. Calculate dist from this
% point to the 6 color marker points, shortest distance tells which color it is
% array containing color labels (0-black,1-red,2-green,3-purple,4-pink,5-yellow)
color_labels = 0:nColors-1;
% Initialize matrices to be used in nearest neighbor classification
a = double(a);
b = double(b);
distance = zeros([size(a), nColors]);
% Classify: for each of n colors, go through each pixel in image, calculate
% distance from pixel's a*,b* values and color marker values
% all distances stored in distance matrix
for count = 1:nColors
distance(:,:,count) = ( (a - color_markers(count,1)).^2 + ...
(b - color_markers(count,2)).^2 ).^0.5;
end
% min inputs: find minimum of distance matrix in 3rd(z) axis. As all n
% distances are stacked in z axis, this finds smallest distance. Outputs:
% first parameter is for actual elements with minimum values, while second
% is for the position in the matrix of the elements. We don't need actual
% element values, so we remove this from output using ~ character
[~, label] = min(distance,[],3);
% now we have position of min elements, but these are from 1:6, while in
% color_labels we defined colors matching to number from 0:5. To make
% values of label matrix match to color they represent in color_labels, do
% the following (in effect, subtracts 1 from each element):
label = color_labels(label);
clear distance;
%% Display results of classification
% create matrix rgb_label (stacks 3 'label' matrixes in z axis)
rgb_label = repmat(label,[1 1 3]);
% create empty 4D matrix, giving each pixel an nColors assignment spot
segmented_images = zeros([size(fabric), nColors],'uint8');
% for each of the n colors:
for count = 1:nColors
% create temporary duplicate of image
color = fabric;
% wherever the label matrix of the image (rgb_label) doesn't match with
% the color label we are looking for in this loop iteration (ex: 2 for
% green, so wherever the label matrix is not 2, make 0). This removes all
% of the image except for the specific color we are currently looking for
color(rgb_label ~= color_labels(count)) = 0;
% store the remaining part of image (only parts with color looking for
% are left) in the 4th dimension of the matrix (this layer will
% correspond to the current color we are scanning for)
segmented_images(:,:,:,count) = color;
% at end, the image will be split up into n layers, each with one of n colors
end
% Display all of the red portion of the image
subplot(1,2,1), imshow(segmented_images(:,:,:,1))
subplot(1,2,2), imshow(segmented_images(:,:,:,2))
cells = segmented_images(:,:,:,1);
%% Convert image from RGB format to grayscale
% Do this because later processing requires grayscale images
% As can see in workspace variables, RGB image has three layers (R,G,B),
% whereas grayscale image has one layer (compare I2,I3 variable dimensions)
cells2 = rgb2gray(cells);
% imshow(I3)
%% Increase contrast of image
cells3 = imadjust(cells2);
% imshow(I4)
%% Threshold image, convert to binary file (only black, white)
% graythresh computes global threshold, which we use to convert image with
% grayscale intensity to a binary image with im2bw
% Then, bwareopen removes background noise (removes objects with fewer than 50 pixels)
level = graythresh(cells3);
bw = im2bw(cells3,level);
bw = bwareaopen(bw, 400);
% imshow(bw)
% fill gaps
se = strel('disk',2);
bw = imclose(bw,se);
% fill any holes
bw = imfill(bw,'holes');
% imshow(bw)
%% Identify objects in image
% bwconncomp finds all Connected Components (objects) in the binary image
% The accuracy of bwconncomp depend on size of objects, connectivity parameter
% (4/8 for 2D image), and if objects are touching (could be labeled as one object)
% Outputs: 1.connectivity of connected components, 2.size of scanned image,
% 3.number of detected objects, 4.array of length(number of objects), kth
% element is vector with pixels in kth object
cc = bwconncomp(bw, 4);
% Print connected components structure
% cc
% Print array containing pixels in objects
% cc.PixelIdxList
% Print pixels in 5th object
% cc.PixelIdxList{5}
%% Examine one object
grain = false(size(bw)); % Create 2D matrix of size of bw, false to make all black
grain(cc.PixelIdxList{12}) = true; % In matrix, make true wherever pixel of object detected
% imshow(grain); % Show image, false/0 -> background; true/1 -> object
%% View all objects
% A label matrix is matrix of same size as input matrix. Input matrix has
% 1s to represent pixels where an object is, 0s to represent bkgnd. In
% label matrix, all pixels of different cc/object are given the same,
% unique label (all pixels in cc 1 are stored as 1, in cc 2 as 2, ...)
% We will now create a label matrix from output of bwconncomp using labelmatrix
labeled = labelmatrix(cc);
% whos labeled
% Convert label matrix into RGB image. @summer references set of colors
% Matlab will use to color each object in the label matrix. 'shuffle' will
% make matlab randomly assign colors to objects. The RGB triplet sets
% color for bkgnd
RGB_label = label2rgb(labeled, @summer, 'k', 'shuffle');
imshow(RGB_label)
%% Calculate area of each object
% regionprops gives area of connected components in cc structure. Output is
% 1.Area of each cc in pixels, 2. coordinates of the smallest rectangle
% that can be fitted within the cc, 3. coordinates of the center of mass of the cc
graindata = regionprops(cc,'basic');
% graindata(50).Area