10 load(header_location,image_location);
13 HSImage::HSImage(std::string header_location, std::string image_location, std::vector<std::string> &spec_location)
16 load(header_location,image_location,spec_location);
43 image_data = other.image_data;
47 pixel_data = other.pixel_data;
75 std::swap(image_data,tmp.image_data);
76 std::swap(pixel_data, tmp.pixel_data);
77 std::swap(image_map,tmp.image_map);
82 void HSImage::load(std::string header_location, std::string image_location)
84 loadHeader(header_location);
86 loadRawImage(image_location);
90 std::vector<std::string> filenames;
91 filenames.push_back(createAbsoluteSpecFilepath(
vis_spec_file ));
92 filenames.push_back(createAbsoluteSpecFilepath(
nir_spec_file ));
98 void HSImage::load(std::string header_location, std::string image_location, std::vector<std::string> spec_locations)
100 loadHeader(header_location);
101 loadRawImage(image_location);
109 std::vector<std::string> filenames;
110 filenames.push_back(createAbsoluteSpecFilepath(
vis_spec_file ));
111 filenames.push_back(createAbsoluteSpecFilepath(
nir_spec_file ));
117 void HSImage::loadHeader(std::string header_location)
121 std::ifstream in_stream;
122 in_stream.open(header_location);
123 if(in_stream.is_open())
128 if(!line.compare(
"ENVI"))
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))
136 std::vector<std::string> words = split(line,
' ');
140 if(words[0].compare(
"}\r"))
143 in_wavelengths =
false;
148 if(words[0].compare(
"}\r"))
149 fwhm.push_back(stof(words.front()));
153 if(!words[0].compare(
"NIR_spec_file"))
159 if(!words[0].compare(
"VIS_spec_file"))
165 if(!words[0].compare(
"acquisition"))
168 if(!words[0].compare(
"samples"))
171 if(!words[0].compare(
"lines"))
172 lines = stoi(words.back());
174 if(!words[0].compare(
"bands"))
175 bands = stoi(words.back());
177 if(!words[0].compare(
"fps"))
178 fps = stoi(words.back());
180 if(!words[0].compare(
"tint"))
181 tint = stoi(words.back());
183 if(!words[0].compare(
"binning"))
189 if(!words[0].compare(
"Wavelength"))
191 in_wavelengths =
true;
194 if(!words[0].compare(
"fwhm"))
199 if(vis_spec && nir_spec)
206 std::cout <<
"Incorrect Header File! Failed at Line " << line_number << std::endl;
211 std::cout <<
"Could not open " << header_location <<
" for reading." << std::endl;
215 void HSImage::loadRawImage(std::string image_location)
218 std::ifstream raw_file;
220 raw_file.open(image_location,std::ios::binary | std::ios::ate);
222 if(raw_file.is_open())
224 unsigned int size = raw_file.tellg();
233 raw_file.seekg(0,std::ios::beg);
234 int storage_spot = 0;
235 for(
int b=0;b<
bands;b++)
237 for(
int l=0;l<
lines;l++)
239 raw_file.seekg((l*bands*
samples+b*
samples)*
sizeof(u_int16_t),std::ios::beg);
241 raw_file.read((
char*)image_data.data()+storage_spot,samples*
sizeof(u_int16_t));
242 storage_spot+=samples*
sizeof(u_int16_t);
248 for(
int l=0;l<
lines;l++)
252 for(
int b=0;b<
bands;b++)
254 pixel_data[storage_spot] = image_data[b*lines*samples+l*samples+s];
263 image_map.emplace(
wavelengths[i],(uchar*)image_data.data()+lines*samples*i*
sizeof(u_int16_t));
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;
277 std::cout <<
"Cannot Open .raw Image File!" << std::endl;
283 return std::make_tuple(lines,samples,bands);
288 std::vector<float> w,val;
292 for(
auto filename : filenames)
301 for(
int i=0;i<18-1;i++)
302 file.ignore(std::numeric_limits<std::streamsize>::max(),
'\n');
305 float NIR_correction = 0;
307 while(std::getline(file,line))
309 line.erase(line.end()-1,line.end());
310 std::vector<std::string> words = split(line,
'\t');
312 if(words.size() < 3 && line.compare(
">>>>>End Processed Spectral Data<<<<<"))
314 if(w.size() < 1 || stof(words.front()) > w.back())
316 if(first_val == 0 && last_val != -1)
318 NIR_correction = last_val / stof(words.back());
324 if(NIR_correction != 0)
325 current_val = NIR_correction * stof(words.back());
327 current_val = stof(words.back());
329 w.push_back(stof(words.front()));
330 val.push_back(current_val);
337 std::cout << strerror(errno) << std::endl;
338 std::cout <<
"Cannot Open Spectrometer Data File!" << std::endl;
341 last_val = val.back();
347 std::vector<float>::iterator w_it = std::lower_bound(w.begin(),w.end(),
wavelengths[i]);
356 std::ofstream out_stream;
357 out_stream.open(
hdr_file,std::ios_base::app | std::ios_base::out);
359 if(out_stream.is_open())
361 out_stream <<
"VIS_spec_file " << createRelativeSpecFilepath( filenames[0] ) << std::endl;
362 out_stream <<
"NIR_spec_file " << createRelativeSpecFilepath( filenames[1] ) << std::endl;
371 std::ifstream in_stream;
372 in_stream.open(header_location);
373 if(in_stream.is_open())
376 if(!line.compare(
"ENVI"))
378 bool nir_spec =
false,vis_spec =
false;
379 while(std::getline(in_stream,line))
381 std::vector<std::string> words = split(line,
' ');
383 if(!words[0].compare(
"NIR_spec_file"))
386 if(!words[0].compare(
"VIS_spec_file"))
389 if(vis_spec && nir_spec)
401 std::cout <<
"Could not open " << header_location <<
" for reading." << std::endl;
408 int loc = row*samples*bands + col*
bands;
409 std::vector<u_int16_t> output(&pixel_data[loc],&pixel_data[loc + bands]);
415 int loc = row*samples*bands + col*
bands;
416 std::vector<u_int16_t> output(&pixel_data[loc],&pixel_data[loc + bands]);
418 for(
unsigned int i=0;i<output.size();i++)
437 std::vector<double> tf(output.size());
439 for(
unsigned int i = 0;i<output.size();i++)
447 std::vector<cv::Mat>
HSImage::getRange(
const float lower_wavelength,
const float upper_wavelength)
452 std::vector<float>::iterator it;
454 std::vector<cv::Mat> output;
456 for(it = key_1;it != key_2;it++)
458 cv::Mat test_mat(lines,samples,CV_16UC1, image_map[*it]);
460 test_mat.copyTo(ret_mat);
461 output.push_back(ret_mat);
469 std::vector<cv::Mat> output;
470 for(
auto wavelength : wavelength_set)
472 float key = closest<float>(
wavelengths,wavelength);
475 key = wavelengths.back();
477 cv::Mat test_mat(lines,samples,CV_16UC1, image_map[key]);
480 test_mat.copyTo(ret_mat);
484 output.push_back(ret_mat);
492 float key = closest<float>(
wavelengths,wavelength);
495 key = wavelengths.back();
497 cv::Mat test_mat(lines,samples,CV_16UC1, image_map[key]);
499 test_mat.copyTo(ret_mat);
513 std::string HSImage::createRelativeSpecFilepath(std::string abs_spec_filepath)
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());
519 std::string rel_spec_filepath;
520 rel_spec_filepath.resize(count);
522 std::copy(diff_pair.second,abs_spec_filepath.end(),rel_spec_filepath.begin());
524 return rel_spec_filepath;
527 std::string HSImage::createAbsoluteSpecFilepath(std::string rel_spec_filepath)
529 std::string::reverse_iterator it;
531 int count = std::distance(
img_file.begin(),it.base());
533 std::string abs_spec_filepath;
534 abs_spec_filepath.resize(count);
536 std::copy(
img_file.begin(),it.base(),abs_spec_filepath.begin());
538 abs_spec_filepath += rel_spec_filepath;
540 return abs_spec_filepath;
543 void export_hsimage(pybind11::module m)
545 namespace py = pybind11;
549 py::class_<HSImage> hsimage (m,
"hsimage");
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&>())
556 .def(
"load", (
void (
HSImage::*)(std::string, std::string)) &
HSImage::load,
"Load data into HSImage instance", py::arg(
"header_filename"), py::arg(
"image_filename"))
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"))
568 .def(
"getBand", &HSImage::operator [])
std::string acquisition_date
std::vector< u_int16_t > getNormalizedPixelSpectra(int row, int col)
Depracated. Do Not Use.
std::vector< cv::Mat > getSet(const std::vector< float > wavelength_set)
Return a disparate set of wavelength images as OpenCV cv::Mat objects.
std::vector< float > getWavelengths()
Return vector of imaged wavlengths.
HSImage & operator=(const HSImage &other)
std::vector< float > getAmbientIntensities()
Return vector of ambient wavelength intensites.
void addSpecDataToHeader(std::vector< std::string > filenames)
Adds spectrometer file data to the corresponding .hdr file.
std::vector< double > getPixelTransferFunction(int row, int col)
Returns pixel "transfer function". Requires spectrometer data.
cv::Mat operator[](const float wavelength)
Get single deisred wavelength image.
std::string nir_spec_file
static bool hasSpecFiles(std::string header_location)
Checks if spectrometer file locations are stored as part of the ENVI .hdr file.
void load(std::string header_location, std::string image_location)
Load hyperspectral image into HSImage.
std::vector< u_int16_t > getRawPixelData()
Get all data formatted as a row major pixel array (row*col,bands)
std::vector< float > wavelengths
void loadSpectrometerData(std::vector< std::string > filenames)
Load only spectrometer data for use with hyperspectral image.
std::tuple< int, int, int > getShape()
Return tuple containing the shape of the HSImage data array (row,col,bands)
std::string vis_spec_file
std::vector< u_int16_t > getPixelSpectra(int row, int col)
returns vector of pixel data
std::vector< float > ambient_intensities
The HSImage class is the base class for interacting with ENVI type hyperspectral images.
std::vector< cv::Mat > getRange(const float lower_wavelength, const float upper_wavelength)
Return a range of wavelength images as OpenCV cv::Mat objects.
std::vector< float > fwhm