HSImage
Hyperspectral Image Interface Library for ENVI-BIL image files
 All Classes Functions Variables Typedefs Groups Pages
classifiedhsimage.cpp
1 #include "classifiedhsimage.h"
2 
3 
4 template <typename T>
5 std::vector<T> operator+(const std::vector<T>& a, const std::vector<T>& b)
6 {
7  assert(a.size() == b.size());
8 
9  std::vector<T> result;
10  result.reserve(a.size());
11 
12  std::transform(a.begin(), a.end(), b.begin(),
13  std::back_inserter(result), std::plus<T>());
14  return result;
15 }
16 
17 
19 
20 ClassifiedHSImage::ClassifiedHSImage(HSImage hsimage, cv::Mat labels, std::vector<classColor> c_names)
21 {
22  load(hsimage,labels,c_names);
23 }
24 
25 ClassifiedHSImage::ClassifiedHSImage(std::string raw_file, std::string hdr_file, std::string label_file, std::string class_hdr_file)
26 {
27  HSImage im(hdr_file,raw_file);
28  cv::Mat labels = cv::imread(label_file);
29 
30  colorMap c;
31  std::fstream in(class_hdr_file,std::ios_base::in);
32  c.fromFile(in);
33 
34  std::vector<classColor> class_list;
35 
36  //get class colors and names, then load into vector
37  const std::vector<target> *target_list = c.getTargetList(targetType::targetClass);
38  for(auto t : *target_list)
39  {
40  std::string name = t.getTitle();
41  cv::Vec3b color;
42  color[0] = t.getB();
43  color[1] = t.getG();
44  color[2] = t.getR();
45 
46  classColor c;
47  c.first = name;
48  c.second = color;
49 
50  if(std::find(class_list.begin(),class_list.end(),c)==class_list.end())
51  class_list.push_back(c);
52  }
53 
54  load(im,labels,class_list);
55 }
56 
57 ClassifiedHSImage::ClassifiedHSImage(std::string raw_file, std::string hdr_file, std::string lif_file)
58 {
59  if (PyObject *err = PyErr_Occurred()) PyErr_Clear();
60  HSImage im(hdr_file,raw_file);
61  LabelFile lf(lif_file);
62  load(im,lf.getLabelImage(),lf.getClassInfo());
63 }
64 
65 void ClassifiedHSImage::load(HSImage hsimage, cv::Mat labels, std::vector<classColor> c_names)
66 {
67  image = hsimage;
68  labels.copyTo(label);
69 
70  for(auto c : c_names)
71  {
72  class_names.emplace(c.first,c.second);
73  class_keys.emplace(c.second,c.first);
74  }
75 }
76 
77 std::vector<std::vector<u_int16_t> > ClassifiedHSImage::getClassSpectra(std::string class_label, unsigned int num_spectra)
78 {
79  std::vector<std::vector<u_int16_t>> output;
80 
81  if(class_names.count(class_label) > 0)
82  {
83  cv::Vec3b class_color = class_names[class_label];
84  cv::Mat mask;
85  std::vector<cv::Point> idx;
86  cv::inRange(label,class_color,class_color,mask);
87  cv::findNonZero(mask,idx);
88 
89  if(idx.size() > 0)
90  {
91  cv::randShuffle(idx);
92 
93  if(num_spectra > 0 && num_spectra < idx.size())
94  {
95  for(unsigned int i=0; i< num_spectra; i++)
96  {
97  cv::Point p = idx[i];
98  std::vector<u_int16_t> tmp = image.getPixelSpectra(p.y, p.x);
99  output.push_back(tmp);
100  }
101  }
102  else
103  {
104  for(unsigned int i=0; i< idx.size(); i++)
105  {
106  cv::Point p = idx[i];
107  std::vector<u_int16_t> tmp = image.getPixelSpectra(p.y, p.x);
108 
109  output.push_back(tmp);
110  }
111  }
112  }
113  }
114  return output;
115 }
116 
117 std::vector<std::vector<double> > ClassifiedHSImage::getClassTF(std::string class_label, unsigned int num_spectra)
118 {
119  std::vector<std::vector<double>> output;
120 
121  if(class_names.count(class_label) > 0)
122  {
123  cv::Vec3b class_color = class_names[class_label];
124  cv::Mat mask,idx;
125  cv::inRange(label,class_color,class_color,mask);
126  cv::findNonZero(mask,idx);
127  if(cv::countNonZero(mask) > 0)
128  {
129  cv::randShuffle(idx);
130 
131  if(num_spectra > 0 && num_spectra < idx.total())
132  {
133  for(unsigned int i=0; i< num_spectra; i++)
134  {
135  cv::Point p = idx.at<cv::Point>(i);
136  std::vector<double> tmp = image.getPixelTransferFunction(p.y, p.x);
137  output.push_back(tmp);
138  }
139  }
140  else
141  {
142  for(unsigned int i=0; i< idx.total(); i++)
143  {
144  cv::Point p = idx.at<cv::Point>(i);
145  std::vector<double> tmp = image.getPixelTransferFunction(p.y, p.x);
146 
147  output.push_back(tmp);
148  }
149  }
150 
151  }
152  }
153  return output;
154 }
155 
156 std::vector<double> ClassifiedHSImage::getAvgClassTF(std::string class_label)
157 {
158  cv::Vec3b class_color = class_names[class_label];
159  cv::Mat mask,idx;
160  cv::inRange(label,class_color,class_color,mask);
161  cv::findNonZero(mask,idx);
162  std::vector<double> transfer_function,var;
163  transfer_function.resize(image.wavelengths.size());
164  transfer_function.assign(transfer_function.size(),0);
165 
166  for(unsigned int i=0;i<idx.total();i++)
167  {
168  cv::Point p = idx.at<cv::Point>(i);
169  std::vector<double> tmp = image.getPixelTransferFunction(p.y, p.x);
170  var = tmp + transfer_function;
171  transfer_function = var;
172  }
173 
174  transform(transfer_function.begin(), transfer_function.end(), transfer_function.begin(),
175  std::bind2nd( std::multiplies<double>(), 1.0/(double)idx.total()));
176 
177  return transfer_function;
178 }
179 
180 std::string ClassifiedHSImage::getPixelClass(int row, int col)
181 {
182  cv::Vec3b val = label.at<cv::Vec3b>(row,col);
183 
184  std::string name = class_keys[val];
185 
186  if(name.empty())
187  name = "Void";
188 
189  return name;
190 }
191 
192 void ClassifiedHSImage::setSpectraClass(int row, int col, std::string class_label)
193 {
194  cv::Vec3b val;
195  val = class_names[class_label];
196 
197  label.at<cv::Vec3b>(row, col) = val;
198 }
199 
200 void ClassifiedHSImage::setSpectraClass(std::vector<std::pair< int, int > > pixel_locs, std::string class_label)
201 {
202  for(auto loc : pixel_locs)
203  {
204  cv::Vec3b val;
205  val = class_names[class_label];
206 
207  label.at<cv::Vec3b>(loc.first, loc.second) = val;
208  }
209 }
210 
211 void ClassifiedHSImage::setImageClass(cv::Mat class_labels, std::vector<classColor> class_list)
212 {
213  class_labels.copyTo(label);
214 
215  class_names.clear();
216  for(auto c : class_list)
217  class_names.emplace(c.first,c.second);
218 }
219 
221 {
222  return label;
223 }
224 
225 void export_classifiedhsimage(pybind11::module m)
226 {
227  namespace py = pybind11;
228 
229 // py::module m2 = m.def_submodule("classified_hsimage","Labeled ENVI-BIL Hyperspectral Image Interface Module using CSAIL/LabelMe label-style.");
230 
231  py::class_<ClassifiedHSImage> classified_hsimage(m, "classified_hsimage");
232  classified_hsimage
233  .def(py::init<>())
234  .def(py::init<HSImage, cv::Mat, std::vector<classColor> >()) //Constructors
235  .def(py::init<std::string, std::string, std::string>())
236  .def_readwrite("hsimage",&ClassifiedHSImage::image)
237  .def_readonly("classes",&ClassifiedHSImage::class_names)
238 
239  .def("getClassSpectra", &ClassifiedHSImage::getClassSpectra)
240  .def("getClassTF",&ClassifiedHSImage::getClassTF)
241  .def("getAvgClassTF", &ClassifiedHSImage::getAvgClassTF)
242  .def("getPixelClass", &ClassifiedHSImage::getPixelClass)
243  .def("setSpectraClass", (void (ClassifiedHSImage::*)(int,int,std::string)) &ClassifiedHSImage::setSpectraClass, "Set Spectra Class", py::arg("row"), py::arg("column"), py::arg("class_name"))
244  .def("setSpectraClass", (void (ClassifiedHSImage::*)(std::vector<std::pair<int,int>>,std::string)) &ClassifiedHSImage::setSpectraClass, "Set Spectra Class", py::arg("point_vector"), py::arg("class_name"))
245  .def("setImageClass", &ClassifiedHSImage::setImageClass)
246  .def("getImageClass", &ClassifiedHSImage::getImageClass);
247 
248 
249 
250 }
The LabelFile class allows an interface to read and process CSAIL/LabelMe style label (...
Definition: labelfile.h:108
void setImageClass(cv::Mat class_labels, std::vector< classColor > class_list)
Set class for whole image with cv::Mat.
std::vector< std::vector< u_int16_t > > getClassSpectra(std::string class_label, unsigned int num_spectra=100)
Returns up to num_spectra pixel spectra of the specified class.
std::vector< std::vector< double > > getClassTF(std::string class_label, unsigned int num_spectra=100)
Returns up to num_spectra pixel transfer function intensities of the specified class.
std::vector< classColor > getClassInfo()
Get vector of class name and color associations.
Definition: labelfile.cpp:178
void setSpectraClass(int row, int col, std::string class_label)
Set class for a pixel by row,column.
std::vector< double > getPixelTransferFunction(int row, int col)
Returns pixel "transfer function". Requires spectrometer data.
Definition: hsimage.cpp:434
The ClassifiedHSImage class is the interface for interacting with classified hyperspectral images...
std::map< cv::Vec3b, std::string, Vec3bCompare > class_keys
std::map< std::string, cv::Vec3b > class_names
std::vector< float > wavelengths
Definition: hsimage.h:231
std::vector< double > getAvgClassTF(std::string class_label)
Returns average transfer function for a specfied class.
cv::Mat getImageClass()
Get cv::Mat showing image classes.
void load(HSImage hsimage, cv::Mat labels, std::vector< classColor > c_names)
Load ClassifiedHSImage from data in memory.
std::pair< std::string, cv::Vec3b > classColor
The classColor typedef creates a simple interface for pairing a class name with a specific OpenCV col...
cv::Mat getLabelImage()
Get image showing class colors and contours for whole label file.
Definition: labelfile.cpp:165
std::vector< u_int16_t > getPixelSpectra(int row, int col)
returns vector of pixel data
Definition: hsimage.cpp:406
The HSImage class is the base class for interacting with ENVI type hyperspectral images.
Definition: hsimage.h:113
std::string getPixelClass(int row, int col)
Returns class of given pixel.