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 package org.openimaj.image.feature.dense.binarypattern;
31
32 import gnu.trove.list.array.TIntArrayList;
33 import gnu.trove.map.hash.TIntObjectHashMap;
34
35 import java.util.ArrayList;
36 import java.util.Arrays;
37 import java.util.List;
38
39 import org.openimaj.citation.annotation.Reference;
40 import org.openimaj.citation.annotation.ReferenceType;
41 import org.openimaj.image.FImage;
42 import org.openimaj.image.pixel.Pixel;
43
44
45
46
47
48
49
50
51
52
53
54
55
56 @Reference(
57 type = ReferenceType.Article,
58 author = { "Ojala, T.", "Pietikainen, M.", "Maenpaa, T." },
59 title = "Multiresolution gray-scale and rotation invariant texture classification with local binary patterns",
60 year = "2002",
61 journal = "Pattern Analysis and Machine Intelligence, IEEE Transactions on",
62 pages = { "971 ", "987" },
63 month = "jul",
64 number = "7",
65 volume = "24",
66 customData = {
67 "doi", "10.1109/TPAMI.2002.1017623",
68 "ISSN", "0162-8828"
69 })
70 public class UniformBinaryPattern {
71 protected static TIntObjectHashMap<TIntArrayList> lut = new TIntObjectHashMap<TIntArrayList>();
72
73 static {
74
75 lut.put(8, calculateUniformPatterns(8));
76
77 }
78
79 protected static TIntArrayList calculateUniformPatterns(int nbits) {
80 final TIntArrayList result = new TIntArrayList();
81
82 final boolean[] bits = new boolean[nbits];
83
84 for (int i = 0; i < Math.pow(2, nbits); i++) {
85 Arrays.fill(bits, false);
86
87 for (int temp = i, j = 1; j <= nbits; j++) {
88 final int pow = (int) Math.pow(2, (nbits - j));
89
90 if (temp / pow > 0) {
91 bits[j - 1] = true;
92 }
93 temp = temp % pow;
94 }
95
96 if (isUniform(bits)) {
97 result.add(i);
98 }
99 }
100
101 return result;
102 }
103
104 protected static boolean isUniform(boolean[] pattern) {
105 int count = 0;
106
107 for (int i = 0; i < pattern.length - 1; i++) {
108 if (pattern[i] != pattern[i + 1]) {
109 count++;
110 }
111 }
112
113 return count <= 2;
114 }
115
116
117
118
119
120
121
122
123
124
125
126
127 public static TIntArrayList getUniformPatterns(int nbits) {
128 if (nbits < 1 || nbits > 32)
129 throw new IllegalArgumentException("Only patterns with lengths between 1 and 32 bits are supported");
130
131 TIntArrayList patterns = lut.get(nbits);
132
133 if (patterns == null) {
134 patterns = calculateUniformPatterns(nbits);
135 lut.put(nbits, patterns);
136 }
137
138 return patterns;
139 }
140
141
142
143
144
145
146
147
148
149
150 public static boolean isPatternUniform(int pattern, int nbits) {
151 return getUniformPatterns(nbits).contains(pattern);
152 }
153
154
155
156
157
158
159
160
161
162
163 public static FImage extractPatternImage(int[][] patternImage, int code) {
164 final FImage image = new FImage(patternImage[0].length, patternImage.length);
165
166 for (int y = 0; y < image.height; y++) {
167 for (int x = 0; x < image.width; x++) {
168 if (patternImage[y][x] == code) {
169 image.pixels[y][x] = 1;
170 }
171 }
172 }
173
174 return image;
175 }
176
177
178
179
180
181
182
183
184
185
186
187 public static FImage[] extractPatternImages(int[][] patternImage, int nbits) {
188 final TIntArrayList uniformPatterns = getUniformPatterns(nbits);
189
190 final FImage[] images = new FImage[uniformPatterns.size() + 1];
191 final int width = patternImage[0].length;
192 final int height = patternImage.length;
193
194 for (int y = 0; y < height; y++) {
195 for (int x = 0; x < width; x++) {
196 final int idx = uniformPatterns.indexOf(patternImage[y][x]);
197
198 if (images[idx + 1] == null) {
199 images[idx + 1] = new FImage(width, height);
200 }
201
202 images[idx + 1].pixels[y][x] = 1;
203 }
204 }
205
206 return images;
207 }
208
209
210
211
212
213
214
215
216
217
218
219 public static boolean[][][] extractPatternMaps(int[][] patternImage, int nbits) {
220 final TIntArrayList uniformPatterns = getUniformPatterns(nbits);
221
222 final int width = patternImage[0].length;
223 final int height = patternImage.length;
224 final boolean[][][] maps = new boolean[uniformPatterns.size() + 1][height][width];
225
226 for (int y = 0; y < height; y++) {
227 for (int x = 0; x < width; x++) {
228 final int idx = uniformPatterns.indexOf(patternImage[y][x]);
229
230 maps[idx + 1][y][x] = true;
231 }
232 }
233
234 return maps;
235 }
236
237
238
239
240
241
242
243
244
245
246
247 public static List<List<Pixel>> extractPatternPixels(int[][] patternImage, int nbits) {
248 final TIntArrayList uniformPatterns = getUniformPatterns(nbits);
249
250 final List<List<Pixel>> images = new ArrayList<List<Pixel>>(uniformPatterns.size() + 1);
251 final int width = patternImage[0].length;
252 final int height = patternImage.length;
253
254 for (int i = 0; i < uniformPatterns.size() + 1; i++)
255 images.add(null);
256
257 for (int y = 0; y < height; y++) {
258 for (int x = 0; x < width; x++) {
259 final int idx = uniformPatterns.indexOf(patternImage[y][x]);
260
261 if (images.get(idx + 1) == null) {
262 images.set(idx + 1, new ArrayList<Pixel>());
263 }
264
265 images.get(idx + 1).add(new Pixel(x, y));
266 }
267 }
268
269 return images;
270 }
271 }