HSImage
Hyperspectral Image Interface Library for ENVI-BIL image files
 All Classes Functions Variables Typedefs Groups Pages
hsimage.cpp
1 #include "hsimage.h"
3 {
4  has_spec_data = false;
5 }
6 
7 HSImage::HSImage(std::string header_location, std::string image_location)
8 {
9  has_spec_data = false;
10  load(header_location,image_location);
11 }
12 
13 HSImage::HSImage(std::string header_location, std::string image_location, std::vector<std::string> &spec_location)
14 {
15  has_spec_data = false;
16  load(header_location,image_location,spec_location);
17 }
18 
20 {
22  tint = other.tint;
23  samples = other.samples;
24  lines = other.lines;
25  bands = other.bands;
26  fps = other.fps;
27  binning[0] = other.binning[0];
28  binning[1] = other.binning[1];
29 
30  hdr_file = other.hdr_file;
31  img_file = other.img_file;
35 
36  wavelengths = other.wavelengths;
37  fwhm = other.fwhm;
38 
40 
41 // image_data.reset(new u_int16_t[lines*samples*bands]);
42 // memcpy((void*)image_data.get(),(void*)other.image_data.get(),lines*bands*samples*sizeof(u_int16_t));
43  image_data = other.image_data;
44 
45 // pixel_data.reset(new u_int16_t[lines*samples*bands]);
46 // memcpy((void*)pixel_data.get(),(void*)other.pixel_data.get(),lines*bands*samples*sizeof(u_int16_t));
47  pixel_data = other.pixel_data;
48 
49  //create map of wavelengths -> OpenCV Mat data pointers
50  for(unsigned int i=0;i<wavelengths.size();i++)
51  image_map.emplace(wavelengths[i],(uchar*)image_data.data()+lines*samples*i*sizeof(u_int16_t));
52 }
53 
55 {
56  HSImage tmp(other);
57  std::swap(acquisition_date,tmp.acquisition_date);
58  std::swap(tint,tmp.tint);
59  std::swap(samples,tmp.samples);
60  std::swap(lines,tmp.lines);
61  std::swap(bands,tmp.bands);
62  std::swap(fps,tmp.fps);
63  std::swap(binning[0],tmp.binning[0]);
64  std::swap(binning[1],tmp.binning[1]);
65 
66  std::swap(hdr_file,tmp.hdr_file);
67  std::swap(img_file,tmp.img_file);
68  std::swap(vis_spec_file,tmp.vis_spec_file);
69  std::swap(nir_spec_file,tmp.nir_spec_file);
70  std::swap(has_spec_data,tmp.has_spec_data);
71 
72  std::swap(wavelengths,tmp.wavelengths);
73  std::swap(fwhm,tmp.fwhm);
75  std::swap(image_data,tmp.image_data);
76  std::swap(pixel_data, tmp.pixel_data);
77  std::swap(image_map,tmp.image_map);
78 
79  return *this;
80 }
81 
82 void HSImage::load(std::string header_location, std::string image_location)
83 {
84  loadHeader(header_location);
85 
86  loadRawImage(image_location);
87 
88  if(has_spec_data)
89  {
90  std::vector<std::string> filenames;
91  filenames.push_back(createAbsoluteSpecFilepath( vis_spec_file ));
92  filenames.push_back(createAbsoluteSpecFilepath( nir_spec_file ));
93 
94  loadSpectrometerData(filenames);
95  }
96 }
97 
98 void HSImage::load(std::string header_location, std::string image_location, std::vector<std::string> spec_locations)
99 {
100  loadHeader(header_location);
101  loadRawImage(image_location);
102  if(vis_spec_file.empty() && vis_spec_file.empty())
103  {
104  addSpecDataToHeader(spec_locations);
105  loadSpectrometerData(spec_locations);
106  }
107  else
108  {
109  std::vector<std::string> filenames;
110  filenames.push_back(createAbsoluteSpecFilepath( vis_spec_file ));
111  filenames.push_back(createAbsoluteSpecFilepath( nir_spec_file ));
112 
113  loadSpectrometerData(filenames);
114  }
115 }
116 
117 void HSImage::loadHeader(std::string header_location)
118 {
119  //load in header file
120  std::string line;
121  std::ifstream in_stream;
122  in_stream.open(header_location);
123  if(in_stream.is_open())
124  {
125  int line_number = 1;
126 
127  in_stream >> line;
128  if(!line.compare("ENVI"))
129  {
130  bool in_wavelengths = false;
131  bool in_fwhm = false;
132  bool nir_spec = false,vis_spec = false;
133  while(std::getline(in_stream,line))
134  {
135  line_number++;
136  std::vector<std::string> words = split(line,' ');
137 
138  if(in_wavelengths)
139  {
140  if(words[0].compare("}\r"))
141  wavelengths.push_back(stof(words.front()));
142  else
143  in_wavelengths = false;
144  }
145 
146  if(in_fwhm)
147  {
148  if(words[0].compare("}\r"))
149  fwhm.push_back(stof(words.front()));
150  else
151  in_fwhm = false;
152  }
153  if(!words[0].compare("NIR_spec_file"))
154  {
155  nir_spec_file = words.back();
156  nir_spec = true;
157  }
158 
159  if(!words[0].compare("VIS_spec_file"))
160  {
161  vis_spec_file = words.back();
162  vis_spec = true;
163  }
164 
165  if(!words[0].compare("acquisition"))
166  acquisition_date = words.back();
167 
168  if(!words[0].compare("samples"))
169  samples = stoi(words.back());
170 
171  if(!words[0].compare("lines"))
172  lines = stoi(words.back());
173 
174  if(!words[0].compare("bands"))
175  bands = stoi(words.back());
176 
177  if(!words[0].compare("fps"))
178  fps = stoi(words.back());
179 
180  if(!words[0].compare("tint"))
181  tint = stoi(words.back());
182 
183  if(!words[0].compare("binning"))
184  {
185  binning[0] = stoi(words[3]);
186  binning[1] = stoi(words[4]);
187  }
188 
189  if(!words[0].compare("Wavelength"))
190  {
191  in_wavelengths = true;
192  }
193 
194  if(!words[0].compare("fwhm"))
195  {
196  in_fwhm = true;
197  }
198  }
199  if(vis_spec && nir_spec)
200  has_spec_data = true;
201 
202  hdr_file = header_location;
203  }
204  else
205  {
206  std::cout << "Incorrect Header File! Failed at Line " << line_number << std::endl;
207  }
208  }
209  else
210  {
211  std::cout << "Could not open " << header_location << " for reading." << std::endl;
212  }
213 }
214 
215 void HSImage::loadRawImage(std::string image_location)
216 {
217  //load in image file
218  std::ifstream raw_file;
219 
220  raw_file.open(image_location,std::ios::binary | std::ios::ate);
221 
222  if(raw_file.is_open())
223  {
224  unsigned int size = raw_file.tellg();
225  if(size == sizeof(u_int16_t)*lines*samples*bands)
226  {
227  //image_data.reset(new u_int16_t[lines*samples*bands]);
228  //pixel_data.reset(new u_int16_t[lines*samples*bands]);
229 
230  image_data.resize(lines*samples*bands);
231  pixel_data.resize(lines*samples*bands);
232 
233  raw_file.seekg(0,std::ios::beg);
234  int storage_spot = 0;
235  for(int b=0;b<bands;b++)
236  {
237  for(int l=0;l<lines;l++)
238  {
239  raw_file.seekg((l*bands*samples+b*samples)*sizeof(u_int16_t),std::ios::beg);
240  //raw_file.read((char*)image_data.get()+storage_spot,samples*sizeof(u_int16_t));
241  raw_file.read((char*)image_data.data()+storage_spot,samples*sizeof(u_int16_t));
242  storage_spot+=samples*sizeof(u_int16_t);
243  }
244  }
245  raw_file.close();
246 
247  storage_spot = 0;
248  for(int l=0;l<lines;l++)
249  {
250  for(int s=0;s<samples;s++)
251  {
252  for(int b=0;b<bands;b++)
253  {
254  pixel_data[storage_spot] = image_data[b*lines*samples+l*samples+s];//
255  storage_spot++;
256  }
257  }
258  }
259 
260  //create map of wavelengths -> OpenCV Mat data pointers
261  for(unsigned int i=0;i<wavelengths.size();i++)
262  //image_map.emplace(wavelengths[i],(uchar*)image_data.get()+lines*samples*i*sizeof(u_int16_t));
263  image_map.emplace(wavelengths[i],(uchar*)image_data.data()+lines*samples*i*sizeof(u_int16_t));
264 
265  img_file = image_location;
266  }
267  else
268  {
269  std::cout << "Header File contains wrong data size for .raw image file." << std::endl
270  << "Header size: " << sizeof(u_int16_t)*lines*samples*bands << std::endl
271  << ".raw file size: " << size << std::endl;
272  }
273 
274  }
275  else
276  {
277  std::cout << "Cannot Open .raw Image File!" << std::endl;
278  }
279 }
280 
281 std::tuple<int,int,int> HSImage::getShape()
282 {
283  return std::make_tuple(lines,samples,bands);
284 }
285 
286 void HSImage::loadSpectrometerData(std::vector<std::string> filenames)
287 {
288  std::vector<float> w,val;
289 
290  float last_val = -1;
291 
292  for(auto filename : filenames)
293  {
294  std::ifstream file;
295  file.open(filename);
296 
297  if(file.is_open())
298  {
299  std::string line;
300 
301  for(int i=0;i<18-1;i++)
302  file.ignore(std::numeric_limits<std::streamsize>::max(),'\n');
303 
304  int first_val = 0;
305  float NIR_correction = 0;
306 
307  while(std::getline(file,line))
308  {
309  line.erase(line.end()-1,line.end());
310  std::vector<std::string> words = split(line,'\t');
311 
312  if(words.size() < 3 && line.compare(">>>>>End Processed Spectral Data<<<<<"))
313  {
314  if(w.size() < 1 || stof(words.front()) > w.back())
315  {
316  if(first_val == 0 && last_val != -1)
317  {
318  NIR_correction = last_val / stof(words.back());
319  first_val = 1;
320  }
321 
322  float current_val;
323 
324  if(NIR_correction != 0)
325  current_val = NIR_correction * stof(words.back());
326  else
327  current_val = stof(words.back());
328 
329  w.push_back(stof(words.front()));
330  val.push_back(current_val);
331  }
332  }
333  }
334  }
335  else
336  {
337  std::cout << strerror(errno) << std::endl;
338  std::cout << "Cannot Open Spectrometer Data File!" << std::endl;
339  return;
340  }
341  last_val = val.back();
342  }
343 
344  ambient_intensities.resize(wavelengths.size());
345  for(unsigned int i=0;i<ambient_intensities.size();i++)
346  {
347  std::vector<float>::iterator w_it = std::lower_bound(w.begin(),w.end(),wavelengths[i]);
348  ambient_intensities[i] = val[std::distance(w.begin(),w_it)];
349  }
350  has_spec_data = true;
351 
352 }
353 
354 void HSImage::addSpecDataToHeader(std::vector<std::string> filenames)
355 {
356  std::ofstream out_stream;
357  out_stream.open(hdr_file,std::ios_base::app | std::ios_base::out);
358 
359  if(out_stream.is_open())
360  {
361  out_stream << "VIS_spec_file " << createRelativeSpecFilepath( filenames[0] ) << std::endl;
362  out_stream << "NIR_spec_file " << createRelativeSpecFilepath( filenames[1] ) << std::endl;
363  out_stream.close();
364  }
365 }
366 
367 bool HSImage::hasSpecFiles(std::string header_location)
368 {
369  //load in header file
370  std::string line;
371  std::ifstream in_stream;
372  in_stream.open(header_location);
373  if(in_stream.is_open())
374  {
375  in_stream >> line;
376  if(!line.compare("ENVI"))
377  {
378  bool nir_spec = false,vis_spec = false;
379  while(std::getline(in_stream,line))
380  {
381  std::vector<std::string> words = split(line,' ');
382 
383  if(!words[0].compare("NIR_spec_file"))
384  nir_spec = true;
385 
386  if(!words[0].compare("VIS_spec_file"))
387  vis_spec = true;
388  }
389  if(vis_spec && nir_spec)
390  return true;
391  else
392  return false;
393  }
394  else
395  {
396  return false;
397  }
398  }
399  else
400  {
401  std::cout << "Could not open " << header_location << " for reading." << std::endl;
402  return false;
403  }
404 }
405 
406 std::vector<u_int16_t> HSImage::getPixelSpectra(int row, int col)
407 {
408  int loc = row*samples*bands + col*bands;
409  std::vector<u_int16_t> output(&pixel_data[loc],&pixel_data[loc + bands]);
410  return output;
411 }
412 
413 std::vector<u_int16_t> HSImage::getNormalizedPixelSpectra(int row, int col)
414 {
415  int loc = row*samples*bands + col*bands;
416  std::vector<u_int16_t> output(&pixel_data[loc],&pixel_data[loc + bands]);
417 
418  for(unsigned int i=0;i<output.size();i++)
419  output[i] = output[i] - ambient_intensities[i];
420 
421  return output;
422 }
423 
424 std::vector<float> HSImage::getWavelengths()
425 {
426  return wavelengths;
427 }
428 
429 std::vector<float> HSImage::getAmbientIntensities()
430 {
431  return ambient_intensities;
432 }
433 
434 std::vector<double> HSImage::getPixelTransferFunction(int row, int col)
435 {
436  std::vector<u_int16_t> output = getPixelSpectra(row,col);
437  std::vector<double> tf(output.size());
438 
439  for(unsigned int i = 0;i<output.size();i++)
440  {
441  tf[i] = (double)output[i]/(double)ambient_intensities[i];
442  }
443 
444  return tf;
445 }
446 
447 std::vector<cv::Mat> HSImage::getRange(const float lower_wavelength, const float upper_wavelength)
448 {
449  auto const key_1 = std::lower_bound(wavelengths.begin(), wavelengths.end(), lower_wavelength);
450  auto const key_2 = std::lower_bound(wavelengths.begin(), wavelengths.end(), upper_wavelength);
451 
452  std::vector<float>::iterator it;
453 
454  std::vector<cv::Mat> output;
455 
456  for(it = key_1;it != key_2;it++)
457  {
458  cv::Mat test_mat(lines,samples,CV_16UC1, image_map[*it]);
459  cv::Mat ret_mat;
460  test_mat.copyTo(ret_mat);
461  output.push_back(ret_mat);
462  }
463 
464  return output;
465 }
466 
467 std::vector<cv::Mat> HSImage::getSet(const std::vector<float> wavelength_set)
468 {
469  std::vector<cv::Mat> output;
470  for(auto wavelength : wavelength_set)
471  {
472  float key = closest<float>(wavelengths,wavelength);
473  if(key < 0)
474  {
475  key = wavelengths.back();
476  }
477  cv::Mat test_mat(lines,samples,CV_16UC1, image_map[key]);
478 
479  cv::Mat ret_mat;
480  test_mat.copyTo(ret_mat);
481 // cv::Mat temp,show_mat;
482 // cv::transpose(ret_mat,temp);
483 // cv::flip(temp,show_mat,0);
484  output.push_back(ret_mat);
485  }
486 
487  return output;
488 }
489 
490 cv::Mat HSImage::operator[] (const float wavelength)
491 {
492  float key = closest<float>(wavelengths,wavelength);
493  if(key < 0)
494  {
495  key = wavelengths.back();
496  }
497  cv::Mat test_mat(lines,samples,CV_16UC1, image_map[key]);
498  cv::Mat ret_mat;
499  test_mat.copyTo(ret_mat);
500 // cv::Mat temp,show_mat;
501 // cv::transpose(ret_mat,temp);
502 // cv::flip(temp,show_mat,0);
503 
504  return ret_mat;
505 
506 }
507 
508 std::vector<u_int16_t> HSImage::getRawPixelData()
509 {
510  return pixel_data;
511 }
512 
513 std::string HSImage::createRelativeSpecFilepath(std::string abs_spec_filepath)
514 {
515  std::pair<std::string::iterator, std::string::iterator> diff_pair;
516  diff_pair = std::mismatch(img_file.begin(),img_file.end(),abs_spec_filepath.begin());
517  int count = std::distance(diff_pair.second,abs_spec_filepath.end());
518 
519  std::string rel_spec_filepath;
520  rel_spec_filepath.resize(count);
521 
522  std::copy(diff_pair.second,abs_spec_filepath.end(),rel_spec_filepath.begin());
523 
524  return rel_spec_filepath;
525 }
526 
527 std::string HSImage::createAbsoluteSpecFilepath(std::string rel_spec_filepath)
528 {
529  std::string::reverse_iterator it;
530  it = std::find(img_file.rbegin(),img_file.rend(),'/');
531  int count = std::distance(img_file.begin(),it.base());
532 
533  std::string abs_spec_filepath;
534  abs_spec_filepath.resize(count);
535 
536  std::copy(img_file.begin(),it.base(),abs_spec_filepath.begin());
537 
538  abs_spec_filepath += rel_spec_filepath;
539 
540  return abs_spec_filepath;
541 }
542 
543 void export_hsimage(pybind11::module m)
544 {
545  namespace py = pybind11;
546 
547 // py::module m2 = m.def_submodule("hsimage","ENVI-BIL Hyperspectral Image Interface Module");
548 
549  py::class_<HSImage> hsimage (m, "hsimage");
550  hsimage
551  .def(py::init<>())
552  .def(py::init<std::string, std::string>())
553  .def(py::init<std::string, std::string, std::vector<std::string>>())
554  .def(py::init<const HSImage&>())
555 
556  .def("load", (void (HSImage::*)(std::string, std::string)) &HSImage::load, "Load data into HSImage instance", py::arg("header_filename"), py::arg("image_filename"))
557 
558  .def("load", (void (HSImage::*)(std::string, std::string, std::vector<std::string>)) &HSImage::load, "Load data into HSImage instance", py::arg("header_filename"), py::arg("image_filename"),py::arg("spec_filename_vector"))
559 
560  .def("loadSpectrometerData", &HSImage::loadSpectrometerData)
561  .def_static("hasSpecFiles", &HSImage::hasSpecFiles)
562  .def("getPixelSpectra", &HSImage::getPixelSpectra)
563  .def("getWavelengths", &HSImage::getWavelengths)
564  .def("getAmbientIntensities",&HSImage::getAmbientIntensities)
565  .def("getPixelTransferFunction", &HSImage::getPixelTransferFunction)
566  .def("getRange", &HSImage::getRange)
567  .def("getSet", &HSImage::getSet)
568  .def("getBand", &HSImage::operator [])
569  .def("getPixelArray",&HSImage::getRawPixelData)
570  .def("getShape", &HSImage::getShape);
571 
572 }
std::string acquisition_date
Definition: hsimage.h:224
int bands
Definition: hsimage.h:228
std::vector< u_int16_t > getNormalizedPixelSpectra(int row, int col)
Depracated. Do Not Use.
Definition: hsimage.cpp:413
std::vector< cv::Mat > getSet(const std::vector< float > wavelength_set)
Return a disparate set of wavelength images as OpenCV cv::Mat objects.
Definition: hsimage.cpp:467
std::vector< float > getWavelengths()
Return vector of imaged wavlengths.
Definition: hsimage.cpp:424
int binning[2]
Definition: hsimage.h:230
HSImage & operator=(const HSImage &other)
Definition: hsimage.cpp:54
std::vector< float > getAmbientIntensities()
Return vector of ambient wavelength intensites.
Definition: hsimage.cpp:429
void addSpecDataToHeader(std::vector< std::string > filenames)
Adds spectrometer file data to the corresponding .hdr file.
Definition: hsimage.cpp:354
std::string hdr_file
Definition: hsimage.h:238
int samples
Definition: hsimage.h:226
bool has_spec_data
Definition: hsimage.h:234
std::vector< double > getPixelTransferFunction(int row, int col)
Returns pixel "transfer function". Requires spectrometer data.
Definition: hsimage.cpp:434
cv::Mat operator[](const float wavelength)
Get single deisred wavelength image.
Definition: hsimage.cpp:490
int fps
Definition: hsimage.h:229
std::string nir_spec_file
Definition: hsimage.h:239
static bool hasSpecFiles(std::string header_location)
Checks if spectrometer file locations are stored as part of the ENVI .hdr file.
Definition: hsimage.cpp:367
void load(std::string header_location, std::string image_location)
Load hyperspectral image into HSImage.
Definition: hsimage.cpp:82
std::vector< u_int16_t > getRawPixelData()
Get all data formatted as a row major pixel array (row*col,bands)
Definition: hsimage.cpp:508
std::vector< float > wavelengths
Definition: hsimage.h:231
int tint
Definition: hsimage.h:225
void loadSpectrometerData(std::vector< std::string > filenames)
Load only spectrometer data for use with hyperspectral image.
Definition: hsimage.cpp:286
std::tuple< int, int, int > getShape()
Return tuple containing the shape of the HSImage data array (row,col,bands)
Definition: hsimage.cpp:281
std::string vis_spec_file
Definition: hsimage.h:240
HSImage()
Definition: hsimage.cpp:2
std::vector< u_int16_t > getPixelSpectra(int row, int col)
returns vector of pixel data
Definition: hsimage.cpp:406
std::string img_file
Definition: hsimage.h:237
int lines
Definition: hsimage.h:227
std::vector< float > ambient_intensities
Definition: hsimage.h:233
The HSImage class is the base class for interacting with ENVI type hyperspectral images.
Definition: hsimage.h:113
std::vector< cv::Mat > getRange(const float lower_wavelength, const float upper_wavelength)
Return a range of wavelength images as OpenCV cv::Mat objects.
Definition: hsimage.cpp:447
std::vector< float > fwhm
Definition: hsimage.h:232