Java tutorial
package name.mjw.cytospade; /** * fcsFile.java * --- * <p>Contains the code to read fcs files.</p> * * <p>Based on the code by Jonathan Irish.</p> * * * Cytobank (TM) is server and client software for web-based management, analysis, * and sharing of flow cytometry data. * * Copyright (C) 2009 Cytobank, Inc. All rights reserved. * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. * * Cytobank, Inc. * 659 Oak Grove Avenue #205 * Menlo Park, CA 94025 * * http://www.cytobank.org */ import cytoscape.logger.CyLogger; import java.io.*; import java.util.*; // Use the new I/O for speed import java.nio.*; import java.nio.channels.*; import java.nio.charset.*; import org.apache.commons.math.linear.Array2DRowRealMatrix; import org.apache.commons.math.linear.BlockRealMatrix; import org.apache.commons.math.linear.LUDecompositionImpl; import org.apache.commons.math.linear.RealMatrix; /** * fcsFile --- * <p> * A class to read FCS files based on fastFacsClasses.cs. * </p> * * <p> * The class is final and does not implement the * <code>java.io.Serializable</code> interface. This is because it should not be * subclassed and is highly file-dependent, respectively. * </p> */ public final class fcsFile { // The ENCODING to use for decoding the text data in the file // ISO-8859-1 is the standard extension of ASCII private static final String ENCODING = "ISO-8859-1"; private static final Charset charset = Charset.forName(ENCODING); // Size of the version string in bytes private static final int VERSION_SIZE = 6; // Default prefix for FCS files to check whether the file is a FCS file. private static final String FCS_PREFIX = "FCS"; // Default behavior for whether to extract events private static final boolean EXTRACTP = false; /** * Decoder for parsing the text portions */ private CharsetDecoder decoder; /** * The underlying file */ private File file; /** * Boolean flag of whether the file is an FCS file. */ private boolean isFCSP; /** * File Information */ public String version = null; // Version string public int textStart = 0; public int textEnd = 0; public int dataStart = 0; public int dataEnd = 0; public int analysisStart = 0; public int analysisEnd = 0; public int supplementalStart = 0; public int supplementalEnd = 0; public char delimiter = '\\'; // TEXT segment delimiter character public String text = null; // The entire TEXT segment /** * settings --- * <p> * <code>java.util.Properties</code> object settings contains all the * key/value pairs in the TEXT segment. * </p> * * <p> * This is a good way to handle all the pairs for hardcore Java people. * </p> */ private Properties settings = null; /** * All the public fields --- * <p> * These should be self-explanatory. * </p> * * <p> * They are initialized since they may never be set if the file is not an * FCS file. * </p> * * <p> * At some point, we probably want to wean people away from these and use * the Properties settings object instead. * </p> */ public boolean littleEndianP; public int parameters = 0; public String sampleName = null; public String dataType = null; public String cytometer = null; public String mode = null; public String instrument = null; public String expTime = null; public String expFile = null; public String operatorName = null; public String operatingSystem = null; public String creatorSoftware = null; public String cytometerNumber = null; public String experimentDate = null; public String experimentName = null; public String exportTime = null; public String exportUser = null; public String GUID = null; public String windowExtension = null; public String threshold = null; public String tubeName = null; public String wellId = null; public String plateId = null; public String plateName = null; public String comment = null; public String spillString = null; public boolean applyCompensation = false; public String source = null; public String nextData = null; public String endsText = null; public String bTime = null; public double timeStep = 0; public boolean[] isLog = null; public int lasers = 0; public String[] laserASF = null; public String[] laserName = null; public String[] laserDelay = null; public String[] channelName = null; public String[] channelShortname = null; public String[] channelGain = null; public int[] channelBits = null; public String[] channelAmp = null; public double[] channelRange = null; public String[] channelVoltage = null; public boolean[] displayLog = null; public double[] ampValue = null; public int totalEvents = 0; protected double[][] eventList = null; /** * Constructor --- * <p> * Given the path to a file, the class grabs all the information about the * file. Whether events are extracted is determined by the default value in * EXTRACTP. * </p> * * <p> * Throws <code>FileNotFoundException</code> and <code>IOException</code>. * This way, whatever code is calling the class can handle the exception * properly. * </p> * * @param path * path to the underlying file. * @throws <code>java.io.FileNotFoundException</code> if the file is not * found. * @throws <code>java.io.IOException</code> if an IO exception occurred. */ public fcsFile(String path) throws FileNotFoundException, IOException { this(path, EXTRACTP); } /** * Constructor --- * <p> * Given the path to a file, the class grabs all the information about the * file. The flag extractEventsP controls whether to extract the data from * the file. * </p> * * <p> * Set extractEventsP to false to improve speed. * </p> * * <p> * Throws <code>FileNotFoundException</code> and <code>IOException</code>. * This way, whatever code is calling the class can handle the exception * properly. * </p> * * @param path * path to the underlying file. * @param extractEventsP * boolean flag for whether to extract events in the underlying * file. * @throws <code>java.io.FileNotFoundException</code> if the file is not * found. * @throws <code>java.io.IOException</code> if an IO exception occurred. */ public fcsFile(String path, boolean extractEventsP) throws FileNotFoundException, IOException { // Create a file using the path and use the other constructor that takes // a File. this(new File(path), extractEventsP); } /** * Constructor --- * <p> * Given a File f, the class grabs all the information about the file. * Whether events are extracted is determined by the default value in * EXTRACTP. * </p> * * <p> * Throws <code>FileNotFoundException</code> and <code>IOException</code>. * This way, whatever code is calling the class can handle the exception * properly. * </p> * * @param file * <code>File</code> object pointing to the underlying file. * @throws <code>java.io.FileNotFoundException</code> if the file is not * found. * @throws <code>java.io.IOException</code> if an IO exception occurred. */ public fcsFile(File file) throws FileNotFoundException, IOException { this(file, EXTRACTP); } /** * Constructor --- * <p> * Given a File f, the class grabs all the information about the file. The * flag extractEventsP controls whether to extract the data from the file. * </p> * * <p> * Set extractEventsP to false to improve speed. * </p> * * <p> * Throws <code>FileNotFoundException</code> and <code>IOException</code>. * This way, whatever code is calling the class can handle the exception * properly. * </p> * * @param file * <code>File</code> object pointing to the underlying file. * @param extractEventsP * boolean flag for whether to extract events in the underlying * file. * @throws <code>java.io.FileNotFoundException</code> if the file is not * found. * @throws <code>java.io.IOException</code> if an IO exception occurred. */ public fcsFile(File file, boolean extractEventsP) throws FileNotFoundException, IOException { this.file = file; // Set isFCSP to false - start by assuming the file is not an FCS file isFCSP = false; // Read the file and initialize all the fields readFile(extractEventsP); } /** * readFile --- * <p> * A helper function to read all the fields in the TEXT segment of the FCS * file. * </p> * * <p> * This helper function should only be called once by the constructor as it * is quite expensive. * </p> * * @param extractEventsP * boolean flag indicating whether to extract events in the * underlying file. * @throws <code>java.io.FileNotFoundException</code> if the file is not * found. * @throws <code>java.io.IOException</code> if an IO exception occurred. */ private void readFile(boolean extractEventsP) throws FileNotFoundException, IOException { // Open a file input stream to the file FileInputStream fis = new FileInputStream(file); // Create a byte array to hold the version byte[] versionArray = new byte[VERSION_SIZE]; // Read the version into the byte array int numRead = fis.read(versionArray); if (numRead < VERSION_SIZE) { // If the number of bytes read is less than the number of bytes in // the version string, then the file is too small to be an FCS file. isFCSP = false; // Close the file input stream fis.close(); // Quit return; } // Decode the version using the default encoding version = new String(versionArray); // Determine whether the file is an FCS file by whether the version // string starts with the FCS_PREFIX isFCSP = version.startsWith(FCS_PREFIX); if (!isFCSP) { // If the file is not an FCS file, then close the file and quit. // Close the file input stream fis.close(); // Quit return; } /** * At this point, we are pretty sure that the file is an FCS file. So, * we parse it. */ /** * Get the standard HEADER stuff */ // Skip 4 bytes to get to byte 10 fis.skip(4); // Create a byte array to hold the HEADER byte[] headerArray = new byte[48]; // Read the header into the byte array numRead = fis.read(headerArray); if (numRead < 48) { // If the number of bytes read is less than 48, then the file is too // small to be an FCS file. isFCSP = false; // Close the file input stream fis.close(); // Quit return; } try { // Try to parse the TEXT segment start and end and DATA segment // start and end textStart = Integer.parseInt((new String(headerArray, 0, 8)).trim()); textEnd = Integer.parseInt((new String(headerArray, 8, 8)).trim()); dataStart = Integer.parseInt((new String(headerArray, 16, 8)).trim()); dataEnd = Integer.parseInt((new String(headerArray, 24, 8)).trim()); } catch (NumberFormatException nfe) { // If a NumberFormatException occured, then quit because there's // nothing we can do without the TEXT or DATA segment. // Close the file input stream fis.close(); return; } /** * Get the ANALYSIS segment limits */ try { // Try to parse the analysisStart and analysisEnd analysisStart = Integer.parseInt((new String(headerArray, 32, 8)).trim()); analysisEnd = Integer.parseInt((new String(headerArray, 40, 8)).trim()); } catch (NumberFormatException nfe) { // If a NumberFormatException occured, then set the ANALYSIS start // and end to 0 since this segment is optional. analysisStart = 0; analysisEnd = 0; } /** * Use NIO to read the OTHER and TEXT segments */ // Get the channel for the input file FileChannel fc = fis.getChannel(); // Move the channel's position back to 0 fc.position(0); // Map the TEXT segment to memory MappedByteBuffer mbb = fc.map(FileChannel.MapMode.READ_ONLY, 0, textEnd + 1); /** * Create the character decoder for parsing characters */ decoder = charset.newDecoder(); /** * Get the OTHER segment */ mbb.limit(textStart); mbb.position(58); CharBuffer other = decoder.decode(mbb.slice()); /** * Get the TEXT segment */ mbb.limit(textEnd + 1); mbb.position(textStart); text = decoder.decode(mbb.slice()).toString(); /** * Close the file since we have the string version of the TEXT segment */ // Close the file channel fc.close(); // Close the file input stream fis.close(); /** * Decode the TEXT segment */ // The first character of the primary TEXT segment contains the // delimiter character delimiter = text.charAt(0); /** * Key/Value Pairs */ // Generate all the pairs String[] pairs; if (delimiter == '\\') { // If the delimiter character is a backslash, then we have to escape // it in the regular expression. pairs = text.split("[\\\\]"); } else { // Otherwise, we can just split it normally by using the character // in the regular expression. pairs = text.split("[" + Character.toString(delimiter) + "]"); } /** * Calculate the number of pairs --- The number of pairs is the length * of the pairs array minus 1 divided by 2. The one is due to the empty * first element from the Java split above. */ int numPairs = (pairs.length - 1) / 2; // Create a mapping for each key and its value settings = new Properties(); // Loop through the TEXT segment we just split to get the keys and // values // The key is in (i * 2) + 1 to account for the empty first element. // The value is in (i * 2) + 2 to account for the empty first element. for (int i = 0; i < numPairs; i++) { settings.setProperty(pairs[(i * 2) + 1].trim(), pairs[(i * 2) + 2].trim()); } // Go through all the key/value pairs and parse them parseSettings(); /** * Extract Events */ if (extractEventsP) { // If we are extracting data, then do so. extractEvents(); } } /** * parseSettings --- * <p> * Uses all the properties in Properties settings to initialize the fields * of fcsFile. * </p> * * <p> * I pulled it out since it involves a lot of constants and I don't want to * mix it with the file reading and parsing code. * </p> */ private void parseSettings() { if (settings == null) { // If settings is null, then quit since there is nothing that can be // done here. return; } /** * At this point, we know settings is not null. */ if (settings.isEmpty()) { // If settings is empty, then quit. return; } /** * At this point, we know settings has some mappings, so try to load * them. */ if ((settings.getProperty("$BEGINSTEXT") != null) && (settings.getProperty("$ENDSTEXT") != null)) { // If the byte offset keywords for the supplemental TEXT segment are // not null, then parse the byte offsets. try { // Try to parse the supplemental start supplementalStart = Integer.parseInt(settings.getProperty("$BEGINSTEXT")); } catch (NumberFormatException nfe) { // If a NumberFormatException occurs, then set the supplemental // start to 0. supplementalStart = 0; } try { // Try to parse the supplemental end supplementalEnd = Integer.parseInt(settings.getProperty("$ENDSTEXT")); } catch (NumberFormatException nfe) { // If a NumberFormatException occurs, then set the supplemental // end to 0. supplementalEnd = 0; } } if ((dataStart == 0) && (dataEnd == 0)) { // If the begin and the end byte offsets for the DATA segment is 0, // then parse the byte offsets from the TEXT segment. try { // Try to parse the data start dataStart = Integer.parseInt(settings.getProperty("$BEGINDATA")); } catch (NumberFormatException nfe) { // If a NumberFormatException occurs, then set the data start to // 0. dataStart = 0; } try { // Try to parse the data end dataEnd = Integer.parseInt(settings.getProperty("$ENDDATA")); } catch (NumberFormatException nfe) { // If a NumberFormatException occurs, then set the data end to // 0. dataEnd = 0; } } if ((analysisStart == 0) && (analysisEnd == 0)) { // If the begin and the end byte offsets for the ANALYSIS segment is // 0, then parse the byte offsets from the TEXT segment. try { // Try to parse the analysis start analysisStart = Integer.parseInt(settings.getProperty("$BEGINANALYSIS")); } catch (NumberFormatException nfe) { // If a NumberFormatException occurs, then set the analysis // start to 0. analysisStart = 0; } try { // Try to parse the analysis end analysisEnd = Integer.parseInt(settings.getProperty("$ENDANALYSIS")); } catch (NumberFormatException nfe) { // If a NumberFormatException occurs, then set the analysis end // to 0. analysisEnd = 0; } } if (settings.getProperty("$PAR") != null) { // If "$PAR" has a mapped value, then try to parse it. try { // Try to parse the number of parameters parameters = Integer.parseInt(settings.getProperty("$PAR")); } catch (NumberFormatException nfe) { // If a NumberFormatException occurs, then set the number of // parameters to 0. parameters = 0; } } if (settings.getProperty("$TOT") != null) { // If "$TOT" has a mapped value, then try to parse it. try { // Try to parse the number of events totalEvents = Integer.parseInt(settings.getProperty("$TOT")); } catch (NumberFormatException nfe2) { // If a NumberFormatException occurs, then set the number of // events to 0. totalEvents = 0; } } // Get the data type dataType = settings.getProperty("$DATATYPE"); // Get the sample name sampleName = settings.getProperty("SAMPLE ID"); if (sampleName == null) { // If the sample name is null, then use the filename as the sample // name. sampleName = file.getName(); } // Initialize whether the byte order is little endian to false littleEndianP = false; // Get the byte order String byteOrder = settings.getProperty("$BYTEORD"); if ((byteOrder != null) && (byteOrder.length() > 0)) { // If the byte order is not null and not empty, then set whether the // byte order is little endian. littleEndianP = (byteOrder.equals("1,2,3,4") || byteOrder.equals("1,2")); } // Initialize all the parameter-based arrays channelName = new String[parameters]; channelShortname = new String[parameters]; channelGain = new String[parameters]; channelBits = new int[parameters]; channelAmp = new String[parameters]; channelRange = new double[parameters]; channelVoltage = new String[parameters]; isLog = new boolean[parameters]; ampValue = new double[parameters]; displayLog = new boolean[parameters]; cytometer = settings.getProperty("$CYT"); mode = settings.getProperty("$MODE"); instrument = settings.getProperty("$INST"); expTime = settings.getProperty("$ETIM"); expFile = settings.getProperty("$FIL"); operatorName = settings.getProperty("$OP"); operatingSystem = settings.getProperty("$SYS"); experimentDate = settings.getProperty("$DATE"); comment = settings.getProperty("$COM"); creatorSoftware = settings.getProperty("CREATOR"); cytometerNumber = settings.getProperty("CYTNUM"); experimentName = settings.getProperty("EXPERIMENT NAME"); exportTime = settings.getProperty("EXPORT TIME"); exportUser = settings.getProperty("EXPORT USER NAME"); GUID = settings.getProperty("GUID"); windowExtension = settings.getProperty("WINDOW EXTENSION"); threshold = settings.getProperty("THRESHOLD"); spillString = settings.getProperty("SPILL"); tubeName = settings.getProperty("TUBE NAME"); wellId = settings.getProperty("WELL ID"); plateId = settings.getProperty("PLATE ID"); plateName = settings.getProperty("PLATE NAME"); // Try to read the $TIMESTEP key, knowing it may be null, so default to // 0 timeStep = 0; if (settings.getProperty("$TIMESTEP") != null) { // If "$TIMESTEP" has a mapped value, then try to parse it. try { // Try to parse the time step timeStep = Double.parseDouble(settings.getProperty("$TIMESTEP")); } catch (NumberFormatException nfe3) { // If a NumberFormatException occurs, then set the time step to // 0. timeStep = 0; } } source = settings.getProperty("$SRC"); endsText = settings.getProperty("$ENDSTEXT"); nextData = settings.getProperty("$NEXTDATA"); bTime = settings.getProperty("$BTIM"); // Try to read the APPLY COMPENSATION key, knowing it may be null, so // default to false applyCompensation = false; String applyString = settings.getProperty("APPLY COMPENSATION"); if ((applyString != null) && applyString.equalsIgnoreCase("true")) { // If the value of "APPLY COMPENSATION" is "true", then set // applyCompensation to true. applyCompensation = true; } lasers = 0; // Count the number of lasers for (int i = 1; i <= settings.size(); i++) { if (settings.getProperty("LASER" + i + "NAME") != null) { // If "LASER#NAME" exists, then increment the number of lasers. lasers++; } } laserASF = new String[lasers]; laserDelay = new String[lasers]; laserName = new String[lasers]; for (int i = 1; i <= lasers; i++) { laserASF[i - 1] = settings.getProperty("LASER" + i + "ASF"); laserDelay[i - 1] = settings.getProperty("LASER" + i + "DELAY"); laserName[i - 1] = settings.getProperty("LASER" + i + "NAME"); } for (int i = 1; i <= parameters; i++) { channelShortname[i - 1] = settings.getProperty("$P" + i + "N"); channelName[i - 1] = settings.getProperty("$P" + i + "S"); if (channelName[i - 1] == null) { channelName[i - 1] = channelShortname[i - 1]; } channelGain[i - 1] = settings.getProperty("$P" + i + "G"); try { channelBits[i - 1] = Integer.parseInt(settings.getProperty("$P" + i + "B")); } catch (NumberFormatException nfe) { if (dataType != null) { if (dataType.equalsIgnoreCase("I")) { // If the data type is "I", then it is binary integer. channelBits[i - 1] = 16; } else if (dataType.equalsIgnoreCase("F")) { // If the data type is "F", then it is floating point. channelBits[i - 1] = 32; } else if (dataType.equalsIgnoreCase("D")) { // If the data type is "D", then it is double precision // floating point channelBits[i - 1] = 64; } else if (dataType.equalsIgnoreCase("A")) { // If the data type is "A", then it is ASCII. channelBits[i - 1] = 8; } } else { // Otherwise, set the number of channel bits to 0. channelBits[i - 1] = 0; } } channelAmp[i - 1] = settings.getProperty("$P" + i + "E"); displayLog[i - 1] = false; String displayString = settings.getProperty("P" + i + "DISPLAY"); if ((displayString != null) && displayString.equalsIgnoreCase("log")) { displayLog[i - 1] = true; } if (channelAmp[i - 1] != null) { String[] ampArray = channelAmp[i - 1].split(","); try { // Try to parse the amp value ampValue[i - 1] = Double.parseDouble(ampArray[0]); } catch (NumberFormatException nfe4) { // If a NumberFormatException occurs, then set the amp value // to 0. ampValue[i - 1] = 0; } } else { // Otherwise, set the amp value to 0 in the event that the // channel amp is missing. ampValue[i - 1] = 0d; } if (ampValue[i - 1] > 0) { isLog[i - 1] = true; } else { // Otherwise, set is log to false. isLog[i - 1] = false; } try { channelRange[i - 1] = Double.parseDouble(settings.getProperty("$P" + i + "R")); } catch (NumberFormatException nfe5) { channelRange[i - 1] = 0; } channelVoltage[i - 1] = settings.getProperty("$P" + i + "V"); if ((channelVoltage[i - 1] == null) && (cytometer != null) && cytometer.equals("FACSCalibur")) { switch (i - 1) { case 2: channelVoltage[i - 1] = settings.getProperty("BD$WORD3"); break; case 3: channelVoltage[i - 1] = settings.getProperty("BD$WORD5"); break; case 4: channelVoltage[i - 1] = settings.getProperty("BD$WORD7"); break; case 5: channelVoltage[i - 1] = settings.getProperty("BD$WORD9"); break; case 6: channelVoltage[i - 1] = settings.getProperty("BD$WORD11"); break; case 7: channelVoltage[i - 1] = settings.getProperty("BD$WORD55"); break; case 8: channelVoltage[i - 1] = settings.getProperty("BD$WORD24"); break; default: channelVoltage[i - 1] = ""; break; } } } /** * Calculate the number of events in the flow file based on the number * of bits in each event */ // Initialize the number of bits in each event to 0 int numBitsPerEvent = 0; // Loop through all the parameters adding the number of bits in each // parameter for (int i = 0; i < parameters; i++) { numBitsPerEvent += channelBits[i]; } if (numBitsPerEvent > 0) { // If the number of bits in each event is greater than 0, then // calculate the number of events based on the size of the DATA // segment. int calculatedNumEvents = (dataEnd - dataStart + 1) * 8 / numBitsPerEvent; if (totalEvents > calculatedNumEvents) { // If the number of events is greater than the calculated number // of events, then update the number of events. totalEvents = calculatedNumEvents; } } } /** * extractEvents --- * <p> * Extracts the events from the FCS file using NIO. * </p> * * @throws <code>java.io.FileNotFoundException</code> if the file is not * found. * @throws <code>java.io.IOException</code> if an IO exception occurred. */ private void extractEvents() throws FileNotFoundException, IOException { if ((dataStart >= dataEnd) || (totalEvents <= 0)) { // If the byte offset of the start of the DATA segment is greater // than or equal to the end of the DATA segment or the number of // events is equal to 0, then create an empty array of events. eventList = new double[0][parameters]; return; } // Open a file input stream to the file FileInputStream fis = new FileInputStream(file); // Get the channel for the file FileChannel fc = fis.getChannel(); // Map the DATA segment to memory MappedByteBuffer data; try { data = fc.map(FileChannel.MapMode.READ_ONLY, dataStart, dataEnd - dataStart + 1); } catch (Throwable t) { // Try again with a workaround to see if we can compensate for off-by-one errors that // some FCS files have been known to incorporate in the ENDDATA property. data = fc.map(FileChannel.MapMode.READ_ONLY, dataStart, dataEnd - dataStart); } /** * We don't need to worry about endian-ness here since ASCII is one * byte, and float and double are IEEE standards. */ if (dataType != null) { if (dataType.equalsIgnoreCase("I")) { // If the data type is "I", then it is binary integer. readBinIntData(data); } else if (dataType.equalsIgnoreCase("F")) { // If the data type is "F", then it is floating point. readFloatData(data); } else if (dataType.equalsIgnoreCase("D")) { // If the data type is "D", then it is double precision floating // point readDoubleData(data); } else if (dataType.equalsIgnoreCase("A")) { // If the data type is "A", then it is ASCII. readASCIIData(data); } } // Close the file channel fc.close(); // Close the file input stream fis.close(); } /** * readBinIntData --- * <p> * Reads binary integers in list mode in the DATA segment and updates * eventList. * </p> * * <p> * Assumes that the bits for the values are byte-aligned. It needs to be * fixed if that is not the case. * </p> * * @param data * <code>ByteBuffer</code> containing the DATA segment of the * underlying file. */ private void readBinIntData(ByteBuffer data) { // Allocate the eventList eventList = new double[parameters][totalEvents]; int numBytes, value, range, currByte; final int totalEvents = this.totalEvents; final int parameters = this.parameters; for (int i = 0; i < totalEvents; i++) { for (int j = 0; j < parameters; j++) { // Calculate the number of bytes used from the number of bits // used // Round up to the next full byte numBytes = (int) Math.ceil(((double) channelBits[j]) / Byte.SIZE); // Get the range of the current parameter - will let us build // the mask range = (int) channelRange[j]; // Initialize the current value to 0 value = 0; if (littleEndianP) { // If the byte order is little endian, then build the value // to the left. for (int k = 0; k < numBytes; k++) { // Get the next 8 bits masking to make sure Java doesn't // prepend 1's currByte = (data.get() & 0xFF); // Shift the 8 bits into position currByte <<= (8 * k); // Or the 8 bits with the current value value |= currByte; } } else { // Otherwise, the byte order is big endian, so build the // value to the right. for (int k = 0; k < numBytes; k++) { // Left shift the previous bits in value to make room value <<= 8; // Grab the next 8 bits masking to make sure Java // doesn't prepend 1's value |= (data.get() & 0xFF); } } /** * From the FCS specification: --- The remaining bits are * usually unused and set to "0"; however, some file writers * store non-data information in that bit-space. Implementers * must use a bit mask when reading these list mode parameter * values to insure that erroneous values are not read from the * unused bits. */ // Mask the value based on the range value &= (range - 1); // Store the value into the array eventList[j][i] = value; } } } /** * readFloatData --- * <p> * Reads floating point values in list mode in the DATA segment and updates * eventList with the integer values of the values. * </p> * * @param data * <code>ByteBuffer</code> containing the DATA segment of the * underlying file. */ private void readFloatData(ByteBuffer data) { // Allocate the eventList eventList = new double[parameters][totalEvents]; if (littleEndianP) { data.order(ByteOrder.LITTLE_ENDIAN); } // Convert the byte buffer into a float buffer - doesn't get any easier FloatBuffer fb = data.asFloatBuffer(); final int totalEvents = this.totalEvents; final int parameters = this.parameters; for (int i = 0; i < totalEvents; i++) { for (int j = 0; j < parameters; j++) { // Store the value into the array eventList[j][i] = fb.get(); } } } /** * readDoubleData --- * <p> * Reads double precision floating point values in list mode in the DATA * segment and updates eventList with the integer values of the values. * </p> * * @param data * <code>ByteBuffer</code> containing the DATA segment of the * underlying file. */ private void readDoubleData(ByteBuffer data) { // Allocate the eventList eventList = new double[parameters][totalEvents]; if (littleEndianP) { data.order(ByteOrder.LITTLE_ENDIAN); } // Convert the byte buffer into a double buffer - doesn't get any easier DoubleBuffer db = data.asDoubleBuffer(); final int totalEvents = this.totalEvents; final int parameters = this.parameters; for (int i = 0; i < totalEvents; i++) { for (int j = 0; j < parameters; j++) { // Store the value into the array eventList[j][i] = db.get(); } } } /** * readASCIIData --- * <p> * Reads ASCII values in list mode in the DATA segment and updates eventList * with the integer values of the values. * </p> * * @param data * <code>ByteBuffer</code> containing the DATA segment of the * underlying file. */ private void readASCIIData(ByteBuffer data) { /** * Calculate the number of characters in each event of the flow file */ // Initialize the number of characters in each event to 0 int numCharsPerEvent = 0; // Loop through all the parameters adding the number of characters in // each parameter for (int i = 0; i < parameters; i++) { numCharsPerEvent += channelBits[i]; } // Allocate the eventList eventList = new double[parameters][totalEvents]; // Convert the byte buffer into a char buffer - doesn't get any easier CharBuffer cb = data.asCharBuffer(); // Initialize the current character to 0 int currChar = 0; for (int i = 0; i < totalEvents; i++) { for (int j = 0; j < parameters; j++) { try { // Store the value into the array eventList[j][i] = Integer .parseInt(cb.subSequence(currChar, currChar + channelBits[i]).toString()); } catch (NumberFormatException nfe) { eventList[j][i] = 0; } // Increment the current character currChar += channelBits[i]; } } } /** * getEventList --- * <p> * Returns the event list. * <p> * * @return array of double arrays containing the events. */ public double[][] getEventList() { if (eventList == null) { // If the array of events is null, then try to extract the events // from the FCS file. try { // Try to extract the events extractEvents(); } catch (FileNotFoundException fnfe) { // If a FileNotFoundException occurred, then return an empty // array of events. return new double[0][0]; } catch (IOException ioe) { // If a IOException occurred, then return an empty array of // events. return new double[0][0]; } } // Return the array of events return eventList; } /** * getCompensatedEventList --- * <p> * Returns the event list compensated by the SPILL matrix. * <p> * * @return array of double arrays containing the events. */ public double[][] getCompensatedEventList() { double[][] events = this.getEventList(); if (events.length != this.getNumChannels()) return events; // Unable to extract the underlying events // Convert the SPILL string to a compensation matrix String compString = this.getSpillString(); if (compString == null) return events; // No compensation, just return the events // Split the compensation string into its values // // The basic structure for SPILL* is: // $SPILLOVER/n,string1,string2,...,f1,f2,f3,f4,.../ String[] compValues = compString.split(","); String[] compNames = null; String[] compData = null; int compDataStart = 0; int n = 0; try { // Try to parse the number of acquisition parameters n = Integer.parseInt(compValues[0]); if (n <= 0 || n > this.parameters) throw new NumberFormatException(); } catch (NumberFormatException nfe) { CyLogger.getLogger().error("Failed to parse parameter count in spill string", nfe); return events; } compNames = Arrays.copyOfRange(compValues, 1, n + 1); // Match names in spill string to columns in parameter lists compDataStart = Arrays.asList(this.channelShortname).indexOf(compNames[0]); if (compDataStart < 0) { CyLogger.getLogger().error("Failed to match channel " + compNames[0] + " to parameter in file"); return events; // Failure match spill string names to channels } for (int i = 0; i < n; i++) { if (!compNames[i].equals(this.channelShortname[compDataStart + i])) { CyLogger.getLogger().error("Spill channel are not continguous parameters in file"); return events; // Spill string columns not in order } } // Extract actual compensation data compData = Arrays.copyOfRange(compValues, n + 1, compValues.length); if (compData.length != (n * n)) return events; /** * Populate the compensation matrix --- The values are stored in * row-major order, i.e., the elements in the first row appear * first. */ double[][] matrix = new double[n][n]; // Loop through the array of compensation values for (int i = 0; i < n; i++) { for (int j = 0; j < n; j++) { try { matrix[i][j] = Double.parseDouble(compData[i * n + j]); } catch (NumberFormatException nfe) { // Set default value If a NumberFormatException occurred matrix[i][j] = 0.0d; } } } // Compute the inverse of the compensation data and then apply // to data matrix (which is column major). Specifically compute // transpose(inverse(<SPILL MATRIX>)) * data RealMatrix comp = (new LUDecompositionImpl(new Array2DRowRealMatrix(matrix))).getSolver().getInverse(); RealMatrix data = new BlockRealMatrix(events); data.setSubMatrix( // Update compensated portion of data matrix comp.transpose() .multiply(data.getSubMatrix(compDataStart, compDataStart + n - 1, 0, this.getEventCount() - 1)) .getData(), compDataStart, 0); return data.getData(); } /** * Accessors */ /** * isFCS --- * <p> * Returns whether the file is an FCS file. * </p> * * <p> * Note: All the public fields are valid if and only if the file is an FCS * file. * </p> * * @return boolean flag indicating whether the file is an FCS file. */ public boolean isFCS() { return isFCSP; } /** * getVersion --- * <p> * Returns the FCS version string of the underlying FCS file. * </p> * * @return <code>String</code> FCS version string of the underlying FCS * file. */ public String getVersion() { return version; } public int getTextStart() { return textStart; } public int getTextEnd() { return textEnd; } public int getDataStart() { return dataStart; } public int getDataEnd() { return dataEnd; } public int getAnalysisStart() { return analysisStart; } public int getAnalysisEnd() { return analysisEnd; } public char getDelimiter() { return delimiter; } public String getText() { return text; } public String getCytometer() { return cytometer; } /** * getChannelHelper --- * <p> * Returns the channel number of the forward scatter channel if sideScatterP * is false or the side scatter channel if sideScatterP is true or -1 if it * is not found. * </p> * * <p> * For forward scatter, it looks in all the channel names for the channel * whose name is "Forward Scatter" or whose channel name or shortname starts * with "FS". * </p> * * <p> * For side scatter, it looks in all the channel names for the channel whose * name is "Side Scatter" or whose channel name or shortname starts with * "SS". * </p> * * @return int channel number of the forward scatter channel if sideScatterP * is false or the side scatter channel if sideScatterP is true or * -1 if it is not found. */ private int getChannelHelper(boolean sideScatterP) { if ((channelName == null) || (channelShortname == null)) { // If the channel name array or the channel shortname array is null, // then quit, since they have not been initialized suggesting // something went wrong in the TEXT parsing. return -1; } // Loop through all the channels for (int i = 0; i < parameters; i++) { if (channelName[i] == null) { // If either the channel name is null, then there is nothing we // can do with the channel, so skip to the next channel. continue; } else if (sideScatterP) { // If sideScatterP, then try to find the side scatter channel. if (channelName[i].indexOf("Side Scatter") >= 0) { // If the channel name contains "Side Scatter", then we are // pretty sure the channel is the side scatter channel, so // return the channel number. return i; } else if (channelName[i].startsWith("SS") || ((channelShortname[i] != null) && channelName[i].startsWith("SS"))) { // If the channel name starts with "SS" or the channel // shortname starts with "SS", then we less certain the // channel is side scatter, but it probably is, so return // the channel number. // This part can be changed. return i; } } else { // Otherwise, try to find the forward scatter channel. if (channelName[i].indexOf("Forward Scatter") >= 0) { // If the channel name contains "Forward Scatter", then we // are pretty sure the channel is the forward scatter // channel, so return the channel number. return i; } else if (channelName[i].startsWith("FS") || ((channelShortname[i] != null) && channelName[i].startsWith("FS"))) { // If the channel name starts with "FS" or the channel // shortname starts with "FS", then we less certain the // channel is forward scatter, but it probably is, so return // the channel number. // This part can be changed. return i; } } } // At this point, we couldn't determine which channel was Side Scatter, // so return -1. return -1; } /** * getForwardScatterChannel --- * <p> * Returns the channel number of the forward scatter channel or -1 if it is * not found. * </p> * * @return int channel number of the forward scatter channel or -1 if it is * not found. */ public int getForwardScatterChannel() { return getChannelHelper(false); } /** * getSideScatterChannel --- * <p> * Returns the channel number of the side scatter channel or -1 if it is not * found. * </p> * * @return int channel number of the side scatter channel or -1 if it is not * found. */ public int getSideScatterChannel() { return getChannelHelper(true); } /** * getChannelShortName --- * <p> * Returns the $PN of the channel with number channelNumber. * </p> * * @param channelNumber * int number of the channel whose name to return. * @return <code>String</code> short name of the channel with number * channelNumber. */ public String getChannelShortName(int channelNumber) { if ((channelShortname == null) || (channelNumber < 0) || (channelNumber >= getNumChannels())) { // If the channel name array or the channel shortname array is null // or the channel number is invalid, then quit, since they have not // been initialized suggesting something went wrong in the TEXT // parsing. return "Error reading this channel!"; } if (channelShortname[channelNumber] == null) { // If the channel short name for the channel is also null, then // return "Channel #" as the channel name. return ("Channel " + channelNumber); } else { // Otherwise, return the channel short name as the channel name. return channelShortname[channelNumber]; } } /** * getChannelIdFromShortName * */ public int getChannelIdFromShortName(String name) { for (int i = 0; i < channelShortname.length; i++) if (channelShortname[i].equals(name)) return i; return -1; } /** * getChannelName --- * <p> * Returns the name of the channel with number channelNumber. * </p> * * <p> * I do this way too often in the code. I don't know why I didn't realize * that a method is needed sooner. * </p> * * @param channelNumber * int number of the channel whose name to return. * @return <code>String</code> name of the channel with number * channelNumber. */ public String getChannelName(int channelNumber) { if ((channelName == null) || (channelShortname == null) || (channelNumber < 0) || (channelNumber >= getNumChannels())) { // If the channel name array or the channel shortname array is null // or the channel number is invalid, then quit, since they have not // been initialized suggesting something went wrong in the TEXT // parsing. return "Error reading this channel!"; } if (channelName[channelNumber] == null) { // If the channel name for the channel is null, then check the // channel short name. if (channelShortname[channelNumber] == null) { // If the channel short name for the channel is also null, then // return "Channel #" as the channel name. return ("Channel " + channelNumber); } else { // Otherwise, return the channel short name as the channel name. return channelShortname[channelNumber]; } } else { // Otherwise, return the channel name as the channel name. return channelName[channelNumber]; } } /** * getChannelCount --- * <p> * Returns the number of channels in the flow file. * </p> * * @return int number of channels in the flow file. */ public int getChannelCount() { return parameters; } /** * getNumChannels --- * <p> * Returns the number of channels in the flow file. * </p> * * @return int number of channels in the flow file. */ public int getNumChannels() { return getChannelCount(); } /** * isLog --- * <p> * Returns a boolean flag indicating whether the channel with channel number * channelNumber is stored in log format. * </p> * * @param channelNumber * int number of the channel. * @return boolean flag indicating whether the channel with channel number * channelNumber is stored in log format. */ public boolean isLog(int channelNumber) { if ((isLog != null) && (channelNumber >= 0) && (channelNumber < isLog.length)) { // If the isLog array is not null and the channel number is in the // range of the array, // then return the value at channelNumber in the isLog array. return isLog[channelNumber]; } // Otherwise, return false. return false; } /** * getIsLog --- * <p> * Returns the <code>boolean</code> status of whether the given channel is * stored in log format. * </p> * * @param channelNumber * int number of the channel. * @return boolean flag indicating whether the channel with channel number * channelNumber is stored in log format. */ public boolean getIsLog(int channelNumber) { return isLog(channelNumber); } /** * isDisplayLog --- * <p> * Returns a boolean flag indicating whether the channel with channel number * channelNumber should be displayed in log format. * </p> * * @param channelNumber * int number of the channel. * @return boolean flag indicating whether the channel with channel number * channelNumber should be displayed in log format. */ public boolean isDisplayLog(int channelNumber) { if ((displayLog != null) && (channelNumber >= 0) && (channelNumber < displayLog.length)) { // If the displayLog array is not null and the channel number is in // the range of the array, // then return the value at channelNumber in the displayLog array. return displayLog[channelNumber]; } // Otherwise, return false. return false; } /** * getDisplayLog --- * <p> * Returns the <code>boolean</code> status of whether the given channel * should be displayed in log format. * </p> * * @param channelNumber * int number of the channel. * @return boolean flag indicating whether the channel with channel number * channelNumber should be displayed in log format. */ public boolean getDisplayLog(int channelNumber) { return isDisplayLog(channelNumber); } /** * getAmpValue --- * <p> * Returns the <code>int</code> log amplifier value. * </p> * * @return <code>double</code> log amplifier value. * @param channelNumber * <code>int</code> the number index of the channel in the array */ public double getAmpValue(int channelNumber) { if (ampValue != null && channelNumber >= 0 && channelNumber < ampValue.length) { return ampValue[channelNumber]; } return 0; } /** * getChannelRange --- * <p> * Returns the <code>int</code> range of the given channel (before any log * amp, as stored in the fcs file). * </p> * * @return <code>int</code> range of the given channel. * @param channelNumber * <code>int</code> the number index of the channel in the array */ public double getChannelRange(int channelNumber) { if (channelRange != null && channelNumber >= 0 && channelNumber < channelRange.length) { return channelRange[channelNumber]; } return 0; } /** * getScaleRange --- * <p> * Returns the <code>int</code> scale range of the given channel (after any * log amp). * </p> * * @return <code>int</code> range of the given channel. * @param channelNumber * <code>int</code> the number index of the channel in the array */ public double getScaleRange(int channelNumber) { if (getIsLog(channelNumber)) { return Math.pow(10, getAmpValue(channelNumber)); } else { return getChannelRange(channelNumber); } } /** * getAmpDisplay --- * <p> * Returns <code>boolean</code> need to log amplify the channel display or * not. * </p> * * @return <code>boolean</code> of whether to log amplify or not. * @param channelNumber * <code>int</code> the number index of the channel in the array */ public boolean getAmpDisplay(int channelNumber) { // If the channel is not log but is being displayed as log (i.e. LSR II // data) if (!getIsLog(channelNumber) && getDisplayLog(channelNumber)) { return true; } else { return false; } } /** * getSettings --- * <p> * Returns the <code>Properties</code> object containing all the key/value * pairs containing all the settings of the FCS file. * </p> * * @return <code>Properties</code> object containing all the key/value pairs * containing all the settings of the FCS file. */ public Properties getSettings() { return settings; } /** * getEventCount --- * <p> * Returns the number of events in the flow file. * </p> * * @return int number of events in the flow file. */ public int getEventCount() { return totalEvents; } /** * getSpillString --- * <p> * Returns the spill string of the flow file. * </p> * * @return <code>String</code> spill string of the flow file. */ public String getSpillString() { return spillString; } /** * getFile --- * <p> * Returns a <code>File</code> object corresponding to the underlying file. * </p> * * @return <code>File</code> object corresponding to the underlying file. */ public File getFile() { return file; } /** * getPath --- * <p> * Returns the <code>String</code> path to the underlying file. * </p> * * @return <code>String</code> path to the underlying file. */ public String getPath() { return file.getPath(); } /** * getName --- * <p> * Returns the <code>String</code> name of the underlying file. * </p> * * @return <code>String</code> name of the underlying file. */ public String getName() { return file.getName(); } /** * length --- * <p> * Returns the length of the underlying file. * </p> * * @return long length of the underlying file. */ public long length() { return file.length(); } /** * isFCSFile --- * <p> * A static helper method to determine whether the file in the * <code>java.io.File</code> object file is a FCS file. * </p> * * @param file * <code>java.io.File</code> object to the file to test. * @return boolean flag indicating whether the file in the * <code>java.io.File</code> object file is a FCS file. */ public static boolean isFCSFile(File file) throws FileNotFoundException, IOException { if ((file == null) || (!file.exists()) || (!file.isFile())) { // If the file is null, does not exist, or is not a file, then // return false. return false; } // Open a file input stream to the file FileInputStream fis = new FileInputStream(file); // Create a byte array to hold the version byte[] versionArray = new byte[VERSION_SIZE]; // Read the version into the byte array int numRead = fis.read(versionArray); // Close the file input stream fis.close(); if (numRead < VERSION_SIZE) { // If the number of bytes read is less than the number of bytes in // the version string, then the file is too small to be an FCS file. return false; } else { // Otherwise, at least 6 bytes were read, so decode it and determine // whether the file is an FCS file. // Decode the version using the default encoding String version = new String(versionArray); // Determine whether the file is an FCS file by whether the version // string starts with the FCS_PREFIX return version.startsWith(FCS_PREFIX); } } /** * Testing Code --- * <p> * The functions below are static functions used to test the fcsFile class. * </p> * * <p> * Different testing functions to exercise different parts of the class. * </p> */ /** * printFCSFile --- * <p> * Prints all the fields of <code>fcsFile</code> f to standard output - for * testing. * <p> * * @param f * <code>fcsFile</code> object to print to standard output. */ public static void printFCSFile(fcsFile f) { System.out.println("Version ID -=" + f.version + "=-"); System.out.println("Text starts at -=" + f.textStart + "=-"); System.out.println("Text ends at -=" + f.textEnd + "=-"); System.out.println("Data starts at -=" + f.dataStart + "=-"); System.out.println("Data ends at -=" + f.dataEnd + "=-"); System.out.println("Analysis starts at -=" + f.analysisStart + "=-"); System.out.println("Analysis ends at -=" + f.analysisEnd + "=-"); System.out.println("Text delimiter -=" + Character.toString(f.delimiter) + "=-"); System.out.println("Data type -=" + f.dataType + "=-"); System.out.println("Cytometer -=" + f.cytometer + "=-"); System.out.println("Lasers -=" + f.lasers + "=-"); /* * System.out.println("LaserName (Count: " + f.laserASF.length + "):"); * for(int i = 0; i < f.laserASF.length; i++) { * System.out.println("\t::" + f.laserASF[i]); } * * System.out.println("LaserASF (Count: " + f.laserName.length + "):"); * for(int i = 0; i < f.laserName.length; i++) { * System.out.println("\t::" + f.laserName[i]); } * * System.out.println("LaserDelay (Count: " + f.laserDelay.length + * "):"); for(int i = 0; i < f.laserDelay.length; i++) { * System.out.println("\t::" + f.laserDelay[i]); } * * System.out.println("MODE -=" + f.mode + "=-"); * System.out.println("Instrument -=" + f.instrument + "=-"); * System.out.println("Experiment Time -=" + f.expTime + "=-"); * System.out.println("Experiment File Name -=" + f.expFile + "=-"); * System.out.println("Operator -=" + f.operatorName + "=-"); * System.out.println("OS -=" + f.operatingSystem + "=-"); * System.out.println("Date -=" + f.experimentDate + "=-"); * System.out.println("Creator Software -=" + f.creatorSoftware + "=-"); * System.out.println("Cytometer Number -=" + f.cytometerNumber + "=-"); * System.out.println("Experiment Name -=" + f.experimentName + "=-"); * System.out.println("Export Time -=" + f.exportTime + "=-"); * System.out.println("Export User -=" + f.exportUser + "=-"); * System.out.println("GUID -=" + f.GUID + "=-"); * System.out.println("Window Extension -=" + f.windowExtension + "=-"); * System.out.println("Threshold -=" + f.threshold + "=-"); * System.out.println("Spill String -=" + f.spillString + "=-"); * System.out.println("Tube Name -=" + f.tubeName + "=-"); * System.out.println("Time Step -=" + f.timeStep + "=-"); * System.out.println("Source -=" + f.source + "=-"); * System.out.println("Text Ends -=" + f.endsText + "=-"); * System.out.println("Next Data -=" + f.nextData + "=-"); * System.out.println("B Time -=" + f.bTime + "=-"); * System.out.println("Apply Compensation -=" + f.applyCompensation + * "=-"); System.out.println("Sample name -=" + f.sampleName + "=-"); * System.out.println("Parameters -=" + f.parameters + "=-"); * System.out.println("Total events -=" + f.totalEvents + "=-"); * * System.out.println("Channel names (Count: " + f.channelName.length + * "):"); for(int i = 0; i < f.channelName.length; i++) { * System.out.println("\t::" + f.channelName[i]); } * * System.out.println("Channel short names (Count: " + * f.channelShortname.length + "):"); for(int i = 0; i < * f.channelShortname.length; i++) { System.out.println("\t::" + * f.channelShortname[i]); } * * System.out.println("Channel gains (Count: " + f.channelGain.length + * "):"); for(int i = 0; i < f.channelGain.length; i++) { * System.out.println("\t::" + f.channelGain[i]); } * * System.out.println( * "Channel amps (4=log, 0=linear -- aka the number of decades on the plot) (Count: " * + f.channelAmp.length + "):"); for(int i = 0; i < * f.channelAmp.length; i++) { System.out.println("\t::" + * f.channelAmp[i]); } * * System.out.println("Therefore, are channels log? (Count: " + * f.isLog.length + "):"); for(int i = 0; i < f.isLog.length; i++) { * System.out.println("\t::" + f.isLog[i]); } * * System.out.println("Display as log? (Count: " + f.displayLog.length + * "):"); for(int i = 0; i < f.displayLog.length; i++) { * System.out.println("\t::" + f.displayLog[i]); } * * System.out.println("Amp values (Count: " + f.ampValue.length + "):"); * for(int i = 0; i < f.ampValue.length; i++) { * System.out.println("\t::" + f.ampValue[i]); } * * System.out.println("Channel bits (Count: " + f.channelBits.length + * "):"); for(int i = 0; i < f.channelBits.length; i++) { * System.out.println("\t::" + f.channelBits[i]); } * * System.out.println( * "Channel ranges (maximum number of bins; 1024 on Calibur; 262144 on LSRII) (Count: " * + f.channelRange.length + "):"); for(int i = 0; i < * f.channelRange.length; i++) { System.out.println("\t::" + * f.channelRange[i]); } * * System.out.println( * "Channel voltages (not working for Calibur yet) (Count: " + * f.channelVoltage.length + "):"); for(int i = 0; i < * f.channelVoltage.length; i++) { System.out.println("\t::" + * f.channelVoltage[i]); } * * System.out.println("All settings key / value pairs:"); * //f.getSettings().list(System.out); * * // Get the settings Properties settings = f.getSettings(); * * // Get all the keys Enumeration keys = settings.propertyNames(); * * // Create a list to hold all the keys ArrayList keysList = new * ArrayList(); * * String key; * * // Loop through all the keys getting their values * while(keys.hasMoreElements()) { // Get the current key key = * (String)keys.nextElement(); * * // Add the current key to the list of keys keysList.add(key); } * * if(keysList.size() > 1) { // If there are more than one key in the * list of keys, then sort it. Collections.sort(keysList); } * * // Loop through the list of keys for(int i = 0; i < keysList.size(); * i++) { // Get the current key key = (String)keysList.get(i); * * // Print the key/value pair System.out.print(key); * System.out.print("="); System.out.println(settings.getProperty(key)); * } */ // System.out.println("The entire text section (unformatted) -=" + // f.text + "=-"); // Print out the first 50 events, if there are any f.printEvents(50); } /** * printEvents --- * <p> * Prints the first numEvents events to standard output for testing. * </p> * * @param numEvents * int number of events to print. */ private void printEvents(int numEvents) { if (eventList == null) { // If the eventList was not populated, then simply quit. return; } System.out.println("The events (Count: " + numEvents + "):"); // Print out all the channel short names for (int i = 0; i < parameters; i++) { System.out.print("\t" + channelShortname[i]); } System.out.println(); // Loop through the first numEvents for (int i = 0; i < numEvents; i++) { System.out.print("Event " + (i + 1) + ":"); // Loop through all the parameters for that event for (int j = 0; j < parameters; j++) { // Print out all the parameters for that event System.out.print("\t" + eventList[i][j]); } System.out.println(); } } /** * main --- * <p> * A main method to test the class. * </p> * * @param args * <code>String</code> array of arguments at the command prompt. */ public static void main(String[] args) { if (args.length <= 0) { // If there are no arguments, then exit. System.out.println("Usage:"); System.out.println("---"); System.out.println(">java fcsFile <path to FCS file>"); System.exit(0); } try { // Echo the file we are checking System.out.println("Checking \"" + args[0] + "\" ..."); fcsFile f; // Read in the file if (args.length > 1) { // If there are more than one argument, then extract the events. f = new fcsFile(args[0], true); } else { // Otherwise, don't extract the events. f = new fcsFile(args[0], false); } // Test whether the file is an FCS file if (f.isFCS()) { // If the file is an FCS file, then print out the file is an FCS // file. System.out.println("\"" + args[0] + "\" is an FCS file."); // Print out the information if the file is an FCS file printFCSFile(f); } else { // Otherwise, print out the file is not an FCS file. System.out.println("\"" + args[0] + "\" is not an FCS file."); } /* * // Print out whether the file is an FCS file using the static * method if(isFCSFile(f.getFile())) { // If the file is an FCS * file, then print out the file is an FCS file. * System.out.println("\"" + args[0] + "\" is an FCS file."); } else * { // Otherwise, print out the file is not an FCS file. * System.out.println("\"" + args[0] + "\" is not an FCS file."); } */ } catch (Exception e) { // Print out any exceptions System.err.println(e.toString()); } } }