#ifndef STIM_ENVI_H #define STIM_ENVI_H #include "../envi/envi_header.h" #include "../envi/bsq.h" #include "../envi/bip.h" #include "../envi/bil.h" #include //#include "../image/image.h" namespace stim{ /** This class implements reading of ENVI hyperspectral files. These files can be stored in multiple orientations (including BSQ, BIP, and BIL) in order to optimize streaming speed depending on applications. Basic ENVI files are stored on disk as a large binary file with a corresponding header. Code for reading and processing ENVI header files is in the envi_header class. */ class envi{ void* file; //void pointer to the relevant file reader (bip, bsq, or bil - with appropriate data type) std::string fname; //file name used for repeated opening and closing public: envi_header header; /// Returns the size of the data type in bytes unsigned int type_size(){ if(header.data_type == envi_header::float32) return 4; if(header.data_type == envi_header::float64) return 8; exit(1); } /// Returns the progress of the current processing operation as a percentage void reset_progress(){ if(header.interleave == envi_header::BSQ){ //if the infile is bsq file if(header.data_type ==envi_header::float32) ((bsq*)file)->reset_progress(); else if(header.data_type == envi_header::float64) ((bsq*)file)->reset_progress(); else std::cout<<"ERROR: unidentified data type"<*)file)->reset_progress(); else if(header.data_type == envi_header::float64) ((bil*)file)->reset_progress(); else std::cout<<"ERROR: unidentified data type"<*)file)->reset_progress(); else if(header.data_type == envi_header::float64) ((bip*)file)->reset_progress(); else std::cout<<"ERROR: unidentified data type"<*)file)->get_progress(); else if(header.data_type == envi_header::float64) return ((bsq*)file)->get_progress(); else std::cout<<"ERROR: unidentified data type"<*)file)->get_progress(); else if(header.data_type == envi_header::float64) return ((bil*)file)->get_progress(); else std::cout<<"ERROR: unidentified data type"<*)file)->get_progress(); else if(header.data_type == envi_header::float64) return ((bip*)file)->get_progress(); else std::cout<<"ERROR: unidentified data type"<()); else if(header.data_type == envi_header::float64) return(file = new bsq()); } else if(header.interleave == envi_header::BIP){ if(header.data_type ==envi_header::float32) return(file = new bip()); else if(header.data_type == envi_header::float64) return(file = new bip()); } else if(header.interleave == envi_header::BIL){ if(header.data_type ==envi_header::float32) return(file = new bil()); else if(header.data_type == envi_header::float64) return(file = new bil()); } exit(1); //if the function hasn't already returned, we don't handle this state } /// Open a previously opened ENVI file bool open(){ //load the file if(header.interleave == envi_header::BSQ) { //if the infile is bsq file if(header.data_type == envi_header::float32) { return ((bsq*)file)->open(fname, header.samples, header.lines, header.bands, header.header_offset, header.wavelength); } else if(header.data_type == envi_header::float64) { return ((bsq*)file)->open(fname, header.samples, header.lines, header.bands, header.header_offset, header.wavelength); } else return false; } else if(header.interleave == envi_header::BIL) { //if the infile is bil file if(header.data_type == envi_header::float32) { return ((bil*)file)->open(fname, header.samples, header.lines, header.bands, header.header_offset, header.wavelength); } else if(header.data_type == envi_header::float64) { return ((bil*)file)->open(fname, header.samples, header.lines, header.bands, header.header_offset, header.wavelength); } else return false; } else if(header.interleave == envi_header::BIP) { //if the infile is bip file if(header.data_type == envi_header::float32) { return ((bip*)file)->open(fname, header.samples, header.lines, header.bands, header.header_offset, header.wavelength); } else if(header.data_type == envi_header::float64) { return ((bip*)file)->open(fname, header.samples, header.lines, header.bands, header.header_offset, header.wavelength); } else return false; } return true; } /// Open an existing ENVI file given the filename and a header structure /// @param filename is the name of the ENVI binary file /// @param header is an ENVI header structure bool open(std::string filename, stim::envi_header h){ allocate(); header = h; //store the header fname = filename; //save the filename return open(); //open the ENVI file; } /// Open an existing ENVI file given the file and header names. /// @param filename is the name of the ENVI binary file /// @param headername is the name of the ENVI header file bool open(std::string filename, std::string headername){ //allocate memory //allocate(); stim::envi_header h; h.load(headername); //load the header //header.load(headername); return open(filename, h); } /// Normalize a hyperspectral ENVI file given a band number and threshold. /// @param outfile is the name of the normalized file to be output /// @param band is the band label to be output /// @param threshold is a threshold value specified such that normalization will only be done to values in the band > threshold (preventing division by small numbers) bool normalize(std::string outfile, double band, unsigned char* mask = NULL, bool PROGRESS = false){ if(header.interleave == envi_header::BSQ){ //if the infile is bsq file if(header.data_type ==envi_header::float32) return ((bsq*)file)->normalize(outfile, band, mask, PROGRESS); else if(header.data_type == envi_header::float64) return ((bsq*)file)->normalize(outfile,band, mask, PROGRESS); else std::cout<<"ERROR: unidentified data type"<*)file)->normalize(outfile, band, mask, PROGRESS); else if(header.data_type == envi_header::float64) return ((bil*)file)->normalize(outfile,band, mask, PROGRESS); else std::cout<<"ERROR: unidentified data type"<*)file)->normalize(outfile, band, mask, PROGRESS); else if(header.data_type == envi_header::float64) return ((bip*)file)->normalize(outfile,band, mask, PROGRESS); else std::cout<<"ERROR: unidentified data type"< w, unsigned char* mask = NULL, bool PROGRESS = false){ header.save(outfile + ".hdr"); if(header.interleave == envi_header::BSQ){ //if the infile is bsq file if(header.data_type ==envi_header::float32) return ((bsq*)file)->baseline(outfile, w, mask, PROGRESS); else if(header.data_type == envi_header::float64) return ((bsq*)file)->baseline(outfile,w, mask, PROGRESS); else{ std::cout<<"ERROR: unidentified data type"<*)file)->baseline(outfile, w, mask, PROGRESS); else if(header.data_type == envi_header::float64) return ((bil*)file)->baseline(outfile, w, mask, PROGRESS); else{ std::cout<<"ERROR: unidentified data type"<*)file)->baseline(outfile, w, mask, PROGRESS); else if(header.data_type == envi_header::float64) return ((bip*)file)->baseline(outfile, w, mask, PROGRESS); else{ std::cout<<"ERROR: unidentified data type"<*)file)->project(outfile, center, basis, M, PROGRESS); else if(header.data_type == envi_header::float64) ((bip*)file)->project(outfile, center, basis, M, PROGRESS); else{ std::cout<<"ERROR: unidentified data type"<*)file)->inverse(outfile, center, basis, B, C, PROGRESS); else if(header.data_type == envi_header::float64) ((bip*)file)->inverse(outfile, center, basis, B, C, PROGRESS); else{ std::cout<<"ERROR: unidentified data type"<*)file)->bil(outfile, PROGRESS); else if(interleave == envi_header::BIP){ //if the target file is bip file std::cout<<"ERROR: conversion from BSQ to BIP isn't practical; use BSQ->BIL->BIP instead"<*)file)->bip(outfile, PROGRESS); exit(1); } } else if(header.data_type == envi_header::float64){ //if the data type is float if(interleave == envi_header::BSQ){ std::cout<<"ERROR: is already BSQ file"<*)file)->bil(outfile, PROGRESS); else if(interleave == envi_header::BIP){ //if the target file is bip file std::cout<<"ERROR: conversion from BSQ to BIP isn't practical; use BSQ->BIL->BIP instead"<*)file)->bip(outfile, PROGRESS); exit(1); } } else{ std::cout<<"ERROR: unidentified data type"<*)file)->bsq(outfile, PROGRESS); else if(interleave == envi_header::BIP) //if the target file is bip file return ((bil*)file)->bip(outfile, PROGRESS); } else if(header.data_type == envi_header::float64){ //if the data type is float if(interleave == envi_header::BIL){ std::cout<<"ERROR: is already BIL file"<*)file)->bsq(outfile, PROGRESS); else if(interleave == envi_header::BIP) return ((bil*)file)->bip(outfile, PROGRESS); } else{ std::cout<<"ERROR: unidentified data type"<*)file)->bil(outfile, PROGRESS); else if(interleave == envi_header::BSQ){ //if the target file is bip file std::cout<<"ERROR: conversion from BIP to BSQ isn't practical; use BIP->BIL->BSQ instead"<*)file)->bip(outfile, PROGRESS); exit(1); } } else if(header.data_type == envi_header::float64){ //if the data type is float if(interleave == envi_header::BIP){ std::cout<<"ERROR: is already BIP file"<*)file)->bil(outfile, PROGRESS); else if(interleave == envi_header::BSQ){ //if the target file is bip file std::cout<<"ERROR: conversion from BIP to BSQ isn't practical; use BIP->BIL->BSQ instead"<*)file)->bip(outfile, PROGRESS); exit(1); } } else{ std::cout<<"ERROR: unidentified data type"<*)file)->build_mask(mask_band, threshold, p, PROGRESS); else if(header.data_type == envi_header::float64) return ((bsq*)file)->build_mask(mask_band, threshold, p, PROGRESS); else std::cout<<"ERROR: unidentified data type"<*)file)->build_mask(mask_band, threshold, p, PROGRESS); else if(header.data_type == envi_header::float64) return ((bil*)file)->build_mask(mask_band, threshold, p, PROGRESS); else std::cout<<"ERROR: unidentified data type"<*)file)->build_mask(mask_band, threshold, p, PROGRESS); else if(header.data_type == envi_header::float64) return ((bip*)file)->build_mask(mask_band, threshold, p, PROGRESS); else std::cout<<"ERROR: unidentified data type"<*)file)->apply_mask(outfile, p); else if (header.data_type == envi_header::float64) return ((bsq*)file)->apply_mask(outfile, p); else std::cout << "ERROR: unidentified data type" << std::endl; } else if (header.interleave == envi_header::BIL){ //if the infile is bil file if (header.data_type == envi_header::float32) return ((bil*)file)->apply_mask(outfile, p); else if (header.data_type == envi_header::float64) return ((bil*)file)->apply_mask(outfile, p); else std::cout << "ERROR: unidentified data type" << std::endl; } else if (header.interleave == envi_header::BIP){ //if the infile is bip file if (header.data_type == envi_header::float32) return ((bip*)file)->apply_mask(outfile, p); else if (header.data_type == envi_header::float64) return ((bip*)file)->apply_mask(outfile, p); else std::cout << "ERROR: unidentified data type" << std::endl; } else{ std::cout << "ERROR: unidentified file type" << std::endl; exit(1); } return false; } /// Copies all spectra corresponding to nonzero values of a mask into a pre-allocated matrix of size (P x B) /// where P is the number of masked pixels and B is the number of bands. The allocated memory can be accessed /// using the following indexing: i = b*P + p /// @param matrix is the destination for the pixel data /// @param p is the mask bool sift(void* matrix, unsigned char* p){ if (header.interleave == envi_header::BSQ){ //if the infile is bsq file if (header.data_type == envi_header::float32) return ((bsq*)file)->sift((float*)matrix, p); else if (header.data_type == envi_header::float64) return ((bsq*)file)->sift((double*)matrix, p); else std::cout << "ERROR: unidentified data type" << std::endl; } if (header.interleave == envi_header::BIP){ std::cout<<"Memory sifting not supported for BIP files"< 0 ) nnz++; //create a new header envi_header new_header = header; //if a BIL file is sifted, it's saved as a BIP if(header.interleave == envi_header::BIL) new_header.interleave = envi_header::BIP; //set the number of lines to 1 (this is a matrix with 1 line and N samples) new_header.lines = 1; new_header.samples = nnz; new_header.save(outfile + ".hdr"); if (header.interleave == envi_header::BSQ){ //if the infile is bsq file if (header.data_type == envi_header::float32) return ((bsq*)file)->sift(outfile, p, PROGRESS); else if (header.data_type == envi_header::float64) return ((bsq*)file)->sift(outfile, p, PROGRESS); else std::cout << "ERROR: unidentified data type" << std::endl; } else if (header.interleave == envi_header::BIL){ //if the infile is bil file if (header.data_type == envi_header::float32) return ((bil*)file)->sift(outfile, p, PROGRESS); else if (header.data_type == envi_header::float64) return ((bil*)file)->sift(outfile, p, PROGRESS); else std::cout << "ERROR: unidentified data type" << std::endl; } else if (header.interleave == envi_header::BIP){ //if the infile is bip file if (header.data_type == envi_header::float32) return ((bip*)file)->sift(outfile, p, PROGRESS); else if (header.data_type == envi_header::float64) return ((bip*)file)->sift(outfile, p, PROGRESS); else std::cout << "ERROR: unidentified data type" << std::endl; } else{ std::cout << "ERROR: unidentified file type" << std::endl; exit(1); } return false; } bool unsift(std::string outfile, unsigned char* mask, unsigned int samples, unsigned int lines){ //create a new header envi_header new_header = header; //set the number of lines and samples in the output file (that's all that changes) new_header.lines = lines; new_header.samples = samples; new_header.save(outfile + ".hdr"); if (header.interleave == envi_header::BSQ){ //if the infile is bsq file if (header.data_type == envi_header::float32) return ((bsq*)file)->unsift(outfile, mask, samples, lines); else if (header.data_type == envi_header::float64) return ((bsq*)file)->unsift(outfile, mask, samples, lines); else std::cout << "ERROR: unidentified data type" << std::endl; } else if (header.interleave == envi_header::BIL){ //if the infile is bil file std::cout << "ERROR in stim::envi::unsift - BIL files aren't supported yet" << std::endl; } else if (header.interleave == envi_header::BIP){ //if the infile is bip file if (header.data_type == envi_header::float32) return ((bip*)file)->unsift(outfile, mask, samples, lines); else if (header.data_type == envi_header::float64) return ((bip*)file)->unsift(outfile, mask, samples, lines); else std::cout << "ERROR: unidentified data type" << std::endl; } else{ std::cout << "ERROR: unidentified file type" << std::endl; } return 0; } /// Compute the ratio of two baseline-corrected peaks. The result is stored in a pre-allocated array. /// @param lb1 is the label value for the left baseline point for the first peak (numerator) /// @param rb1 is the label value for the right baseline point for the first peak (numerator) /// @param pos1 is the label value for the first peak (numerator) position /// @param lb2 is the label value for the left baseline point for the second peak (denominator) /// @param rb2 is the label value for the right baseline point for the second peak (denominator) /// @param pos2 is the label value for the second peak (denominator) position /// @param result is a pointer to a pre-allocated array at least X * Y * sizeof(T) in size bool ph_to_ph(double lb1, double rb1, double pos1, double lb2, double rb2, double pos2, void * result){ if(header.interleave == envi_header::BSQ){ //if the infile is bsq file if(header.data_type ==envi_header::float32) return ((bsq*)file)->ph_to_ph(lb1, rb1, pos1, lb2, rb2, pos2, (float*)result); else if(header.data_type == envi_header::float64) return ((bsq*)file)->ph_to_ph(lb1, rb1, pos1, lb2, rb2, pos2, (double*)result); else std::cout<<"ERROR: unidentified data type"<*)file)->ph_to_ph(lb1, rb1, pos1, lb2, rb2, pos2, (float*)result); else if(header.data_type == envi_header::float64) return ((bil*)file)->ph_to_ph(lb1, rb1, pos1, lb2, rb2, pos2, (double*)result); else std::cout<<"ERROR: unidentified data type"<*)file)->ph_to_ph(lb1, rb1, pos1, lb2, rb2, pos2, (float*)result); else if(header.data_type == envi_header::float64) return ((bip*)file)->ph_to_ph(lb1, rb1, pos1, lb2, rb2, pos2, (double*)result); else std::cout<<"ERROR: unidentified data type"<*)file)->pa_to_ph(lb1, rb1, lab1, rab1, lb2, rb2, pos, (float*)result); else if(header.data_type == envi_header::float64) return ((bsq*)file)->pa_to_ph(lb1, rb1, lab1, rab1, lb2, rb2, pos, (double*)result); else std::cout<<"ERROR: unidentified data type"<*)file)->pa_to_ph(lb1, rb1, lab1, rab1, lb2, rb2, pos, (float*)result); else if(header.data_type == envi_header::float64) return ((bil*)file)->pa_to_ph(lb1, rb1, lab1, rab1, lb2, rb2, pos, (double*)result); else std::cout<<"ERROR: unidentified data type"<*)file)->pa_to_ph(lb1, rb1, lab1, rab1, lb2, rb2, pos, (float*)result); else if(header.data_type == envi_header::float64) return ((bip*)file)->pa_to_ph(lb1, rb1, lab1, rab1, lb2, rb2, pos, (double*)result); else std::cout<<"ERROR: unidentified data type"<*)file)->pa_to_pa(lb1, rb1, lab1, rab1, lb2, rb2, lab2, rab2, (float*)result); else if(header.data_type == envi_header::float64) return ((bsq*)file)->pa_to_pa(lb1, rb1, lab1, rab1, lb2, rb2, lab2, rab2, (double*)result); else std::cout<<"ERROR: unidentified data type"<*)file)->pa_to_pa(lb1, rb1, lab1, rab1, lb2, rb2, lab2, rab2, (float*)result); else if(header.data_type == envi_header::float64) return ((bil*)file)->pa_to_pa(lb1, rb1, lab1, rab1, lb2, rb2, lab2, rab2, (double*)result); else std::cout<<"ERROR: unidentified data type"<*)file)->pa_to_pa(lb1, rb1, lab1, rab1, lb2, rb2, lab2, rab2, (float*)result); else if(header.data_type == envi_header::float64) return ((bip*)file)->pa_to_pa(lb1, rb1, lab1, rab1, lb2, rb2, lab2, rab2, (double*)result); else std::cout<<"ERROR: unidentified data type"<*)file)->cpoint(lb1, rb1, lab1, rab1, (float*)result); else if(header.data_type == envi_header::float64) return ((bsq*)file)->cpoint(lb1, rb1, lab1, rab1, (double*)result); else std::cout<<"ERROR: unidentified data type"<*)file)->cpoint(lb1, rb1, lab1, rab1, (float*)result); else if(header.data_type == envi_header::float64) return ((bil*)file)->cpoint(lb1, rb1, lab1, rab1, (double*)result); else std::cout<<"ERROR: unidentified data type"<*)file)->cpoint(lb1, rb1, lab1, rab1,(float*)result); else if(header.data_type == envi_header::float64) return ((bip*)file)->cpoint(lb1, rb1, lab1, rab1, (double*)result); else std::cout<<"ERROR: unidentified data type"<*)file)->close(); else if(header.data_type == envi_header::float64) return ((bsq*)file)->close(); else{ std::cout<<"ERROR: unidentified data type"<*)file)->close(); else if(header.data_type == envi_header::float64) return ((bil*)file)->close(); else{ std::cout<<"ERROR: unidentified data type"<*)file)->close(); else if(header.data_type == envi_header::float64) return ((bip*)file)->close(); else{ std::cout<<"ERROR: unidentified data type"<*)file)->pixel((float*)p, n); else if(header.data_type == envi_header::float64) return ((bsq*)file)->pixel((double*)p, n); else{ std::cout<<"ERROR: unidentified data type"<*)file)->pixel((float*)p, n); else if(header.data_type == envi_header::float64) return ((bil*)file)->pixel((double*)p, n); else{ std::cout<<"ERROR: unidentified data type"<*)file)->pixel((float*)p, n); else if(header.data_type == envi_header::float64) return ((bip*)file)->pixel((double*)p, n); else{ std::cout<<"ERROR: unidentified data type"<*)file)->band((float*)ptr, wavelength, PROGRESS); else if (header.data_type == envi_header::float64) return ((bsq*)file)->band((double*)ptr, wavelength, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } else if (header.interleave == envi_header::BIL){ if (header.data_type == envi_header::float32) return ((bil*)file)->band((float*)ptr, wavelength, PROGRESS); else if (header.data_type == envi_header::float64) return ((bil*)file)->band((double*)ptr, wavelength, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } else if (header.interleave == envi_header::BIP){ if (header.data_type == envi_header::float32) return ((bip*)file)->band((float*)ptr, wavelength, PROGRESS); else if (header.data_type == envi_header::float64) return ((bip*)file)->band((double*)ptr, wavelength, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } return false; } /// Retrieve a spectrum from the specified location /// @param ptr is a pointer to pre-allocated memory of size B*sizeof(T) /// @param x is the x-coordinate of the spectrum /// @param y is the y-coordinate of the spectrum bool spectrum(void* ptr, unsigned int x, unsigned int y, bool PROGRESS = false){ if(header.interleave == envi_header::BSQ){ //if the infile is bsq file if(header.data_type ==envi_header::float32) return ((bsq*)file)->spectrum((float*)ptr, x, y, PROGRESS); else if (header.data_type == envi_header::float64) return ((bsq*)file)->spectrum((double*)ptr, x, y, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } else if (header.interleave == envi_header::BIL){ if (header.data_type == envi_header::float32) return ((bil*)file)->spectrum((float*)ptr, x, y, PROGRESS); else if (header.data_type == envi_header::float64) return ((bil*)file)->spectrum((double*)ptr, x, y, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } else if (header.interleave == envi_header::BIP){ if (header.data_type == envi_header::float32) return ((bip*)file)->spectrum((float*)ptr, x, y, PROGRESS); else if (header.data_type == envi_header::float64) return ((bip*)file)->spectrum((double*)ptr, x, y, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } return false; } /// Retrieve a single band (based on index) and stores it in pre-allocated memory. /// @param p is a pointer to an allocated region of memory at least X * Y * sizeof(T) in size. /// @param page <= B is the integer number of the band to be copied. bool band_index(void* ptr, unsigned int b){ if (header.interleave == envi_header::BSQ){ //if the infile is bsq file if (header.data_type == envi_header::float32) return ((bsq*)file)->band_index((float*)ptr, b); else if (header.data_type == envi_header::float64) return ((bsq*)file)->band_index((double*)ptr, b); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } else if (header.interleave == envi_header::BIL){ if (header.data_type == envi_header::float32) return ((bil*)file)->band_index((float*)ptr, b); else if (header.data_type == envi_header::float64) return ((bil*)file)->band_index((double*)ptr, b); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } else if (header.interleave == envi_header::BIP){ if (header.data_type == envi_header::float32) return ((bip*)file)->band_index((float*)ptr, b); else if (header.data_type == envi_header::float64) return ((bip*)file)->band_index((double*)ptr, b); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } return false; } /// Calculate the mean value for all masked (or valid) pixels in a band and returns the average spectrum /// @param p is a pointer to pre-allocated memory of size [B * sizeof(T)] that stores the mean spectrum /// @param mask is a pointer to memory of size [X * Y] that stores the mask value at each pixel location bool avg_band(double * p, unsigned char* mask, bool PROGRESS = false){ if (header.interleave == envi_header::BSQ){ if (header.data_type == envi_header::float32) return ((bsq*)file)->avg_band(p, mask, PROGRESS); else if (header.data_type == envi_header::float64) return ((bsq*)file)->avg_band(p, mask, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } else if (header.interleave == envi_header::BIL){ if (header.data_type == envi_header::float32) return ((bil*)file)->avg_band(p, mask, PROGRESS); else if (header.data_type == envi_header::float64) return ((bil*)file)->avg_band(p, mask, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } else if (header.interleave == envi_header::BIP){ if (header.data_type == envi_header::float32) return ((bip*)file)->avg_band(p, mask, PROGRESS); else if (header.data_type == envi_header::float64) return ((bip*)file)->avg_band(p, mask, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } return false; } /// Calculate the covariance matrix for all masked pixels in the image. /// @param co is a pointer to pre-allocated memory of size [B * B] that stores the resulting covariance matrix /// @param avg is a pointer to memory of size B that stores the average spectrum /// @param mask is a pointer to memory of size [X * Y] that stores the mask value at each pixel location bool co_matrix(double* co, double* avg, unsigned char* mask, bool PROGRESS = false){ if (header.interleave == envi_header::BSQ){ std::cout<<"ERROR: calculating the covariance matrix for a BSQ file is impractical; convert to BIL or BIP first"<*)file)->co_matrix(co, avg, mask, PROGRESS); else if (header.data_type == envi_header::float64) return ((bsq*)file)->co_matrix(co, avg, mask, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); }*/ } else if (header.interleave == envi_header::BIL){ if (header.data_type == envi_header::float32) return ((bil*)file)->co_matrix(co, avg, mask, PROGRESS); else if (header.data_type == envi_header::float64) return ((bil*)file)->co_matrix(co, avg, mask, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } else if (header.interleave == envi_header::BIP){ if (header.data_type == envi_header::float32) return ((bip*)file)->co_matrix(co, avg, mask, PROGRESS); else if (header.data_type == envi_header::float64) return ((bip*)file)->co_matrix(co, avg, mask, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } return false; } /// Crop a region of the image and save it to a new file. /// @param outfile is the file name for the new cropped image /// @param x0 is the lower-left x pixel coordinate to be included in the cropped image /// @param y0 is the lower-left y pixel coordinate to be included in the cropped image /// @param x1 is the upper-right x pixel coordinate to be included in the cropped image /// @param y1 is the upper-right y pixel coordinate to be included in the cropped image bool crop(std::string outfile,unsigned x0, unsigned y0, unsigned x1, unsigned y1, unsigned b0, unsigned b1, bool PROGRESS = false){ //save the header for the cropped file stim::envi_header new_header = header; new_header.samples = x1 - x0; new_header.lines = y1 - y0; new_header.bands = b1 - b0; std::vector::const_iterator first = new_header.wavelength.begin() + b0; std::vector::const_iterator last = new_header.wavelength.begin() + b1; new_header.wavelength = std::vector(first, last); new_header.save(outfile + ".hdr"); if (header.interleave == envi_header::BSQ){ if (header.data_type == envi_header::float32) return ((bsq*)file)->crop(outfile, x0, y0, x1, y1, b0, b1, PROGRESS); else if (header.data_type == envi_header::float64) return ((bsq*)file)->crop(outfile, x0, y0, x1, y1, b0, b1, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } else if (header.interleave == envi_header::BIL){ if (header.data_type == envi_header::float32) return ((bil*)file)->crop(outfile, x0, y0, x1, y1, b0, b1, PROGRESS); else if (header.data_type == envi_header::float64) return ((bil*)file)->crop(outfile, x0, y0, x1, y1, b0, b1, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } else if (header.interleave == envi_header::BIP){ if (header.data_type == envi_header::float32) return ((bip*)file)->crop(outfile, x0, y0, x1, y1, b0, b1, PROGRESS); else if (header.data_type == envi_header::float64) return ((bip*)file)->crop(outfile, x0, y0, x1, y1, b0, b1, PROGRESS); else{ std::cout << "ERROR: unidentified data type" << std::endl; exit(1); } } return false; } }; } //end namespace rts #endif