From e5fb4cc491af516a2e4febd53e9e7a24e3ebe8a3 Mon Sep 17 00:00:00 2001 From: Melissa Linkert Date: Thu, 15 Jun 2023 16:24:41 -0500 Subject: [PATCH 1/5] Add action information to channel names --- components/formats-gpl/src/loci/formats/in/CV7000Reader.java | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/components/formats-gpl/src/loci/formats/in/CV7000Reader.java b/components/formats-gpl/src/loci/formats/in/CV7000Reader.java index 47029608dc3..71b17bf5c28 100644 --- a/components/formats-gpl/src/loci/formats/in/CV7000Reader.java +++ b/components/formats-gpl/src/loci/formats/in/CV7000Reader.java @@ -580,7 +580,8 @@ public int compare(Channel c1, Channel c2) { // the index here is the original bts:Ch index in // the *.mes and *.mrf files - store.setChannelName("Channel #" + (channel.index + 1) + ", Camera #" + channel.cameraNumber, i, c); + store.setChannelName("Action #" + (channel.actionIndex + 1) + + ", Channel #" + (channel.index + 1) + ", Camera #" + channel.cameraNumber, i, c); if (channel.color != null) { store.setChannelColor(channel.color, i, c); From 5cd0ea51430e73c15f16d95f7700a71ab7a36540 Mon Sep 17 00:00:00 2001 From: Melissa Linkert Date: Thu, 15 Jun 2023 16:30:32 -0500 Subject: [PATCH 2/5] Add an option to duplicate the first plane for any missing planes Meant to help with visualizing mixed single channel/Z stack data. --- .../src/loci/formats/in/CV7000Reader.java | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) diff --git a/components/formats-gpl/src/loci/formats/in/CV7000Reader.java b/components/formats-gpl/src/loci/formats/in/CV7000Reader.java index 71b17bf5c28..b044ebbf408 100644 --- a/components/formats-gpl/src/loci/formats/in/CV7000Reader.java +++ b/components/formats-gpl/src/loci/formats/in/CV7000Reader.java @@ -70,6 +70,9 @@ public class CV7000Reader extends FormatReader { // -- Constants -- + public static final String DUPLICATE_PLANES_KEY = "cv7000.duplicate_missing_planes"; + public static final boolean DUPLICATE_PLANES_DEFAULT = true; + private static final Logger LOGGER = LoggerFactory.getLogger(CV7000Reader.class); private static final String MEASUREMENT_FILE = "MeasurementData.mlf"; @@ -104,6 +107,17 @@ public CV7000Reader() { datasetDescription = "Directory with XML files and one .tif/.tiff file per plane"; } + // -- CV7000Reader API methods -- + + public boolean duplicatePlanes() { + MetadataOptions options = getMetadataOptions(); + if (options instanceof DynamicMetadataOptions) { + return ((DynamicMetadataOptions) options).getBoolean( + DUPLICATE_PLANES_KEY, DUPLICATE_PLANES_DEFAULT); + } + return DUPLICATE_PLANES_DEFAULT; + } + // -- IFormatReader API methods -- /* @see loci.formats.IFormatReader#getRequiredDirectories(String[]) */ @@ -228,6 +242,9 @@ public byte[] openBytes(int no, byte[] buf, int x, int y, int w, int h) reader.setId(p.file); return reader.openBytes(0, buf, x, y, w, h); } + else if (duplicatePlanes() && no > 0) { + return openBytes(0, buf, x, y, w, h); + } return buf; } From a292baf06d478308f90d8ecbd566334947d44063 Mon Sep 17 00:00:00 2001 From: Melissa Linkert Date: Thu, 29 Jun 2023 11:39:51 -0500 Subject: [PATCH 3/5] Store bts:Fluorophore (if present) in Channel.Fluor --- .../formats-gpl/src/loci/formats/in/CV7000Reader.java | 7 +++++++ 1 file changed, 7 insertions(+) diff --git a/components/formats-gpl/src/loci/formats/in/CV7000Reader.java b/components/formats-gpl/src/loci/formats/in/CV7000Reader.java index b044ebbf408..2009e10baf2 100644 --- a/components/formats-gpl/src/loci/formats/in/CV7000Reader.java +++ b/components/formats-gpl/src/loci/formats/in/CV7000Reader.java @@ -603,6 +603,9 @@ public int compare(Channel c1, Channel c2) { if (channel.color != null) { store.setChannelColor(channel.color, i, c); } + if (channel.fluor != null && !channel.fluor.isEmpty()) { + store.setChannelFluor(channel.fluor, i, c); + } if (channel.excitation != null && channel.lightSourceRefs != null) { int index = -1; @@ -955,6 +958,8 @@ else if (qName.equals("bts:Channel")) { currentChannel.excitation = DataTools.parseDouble(wave); } } + + currentChannel.fluor = attributes.getValue("bts:Fluorophore"); } } } @@ -1019,6 +1024,8 @@ class Channel { public String binning; public Color color; + public String fluor; + @Override public String toString() { return "timelineIndex=" + timelineIndex + ", actionIndex=" + actionIndex + ", index=" + index; From 6058cec7fb45f4cb3ce3369090d174b5b6a8a2a3 Mon Sep 17 00:00:00 2001 From: Melissa Linkert Date: Thu, 3 Aug 2023 19:03:44 -0500 Subject: [PATCH 4/5] Duplicate the first plane in the requested channel, not first plane overall --- .../src/loci/formats/in/CV7000Reader.java | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/components/formats-gpl/src/loci/formats/in/CV7000Reader.java b/components/formats-gpl/src/loci/formats/in/CV7000Reader.java index 2009e10baf2..4909eedec5d 100644 --- a/components/formats-gpl/src/loci/formats/in/CV7000Reader.java +++ b/components/formats-gpl/src/loci/formats/in/CV7000Reader.java @@ -243,7 +243,17 @@ public byte[] openBytes(int no, byte[] buf, int x, int y, int w, int h) return reader.openBytes(0, buf, x, y, w, h); } else if (duplicatePlanes() && no > 0) { - return openBytes(0, buf, x, y, w, h); + int[] zct = getZCTCoords(no); + // pick the first plane in the same channel + int dupPlane = getIndex(0, zct[1], 0); + + // very unlikely to happen, but catching the case + // where no is the first plane in the channel prevents + // a potential infinite loop + if (dupPlane == no) { + dupPlane = 0; + } + return openBytes(dupPlane, buf, x, y, w, h); } return buf; } From 48c1baf7239b042d6a99262ebcf060196cfee5d9 Mon Sep 17 00:00:00 2001 From: Melissa Linkert Date: Thu, 31 Aug 2023 14:31:12 -0500 Subject: [PATCH 5/5] One more fix for channels that are acquired in more than one action --- .../src/loci/formats/in/CV7000Reader.java | 68 ++++++++++++++----- 1 file changed, 52 insertions(+), 16 deletions(-) diff --git a/components/formats-gpl/src/loci/formats/in/CV7000Reader.java b/components/formats-gpl/src/loci/formats/in/CV7000Reader.java index 4909eedec5d..0d2a65c707a 100644 --- a/components/formats-gpl/src/loci/formats/in/CV7000Reader.java +++ b/components/formats-gpl/src/loci/formats/in/CV7000Reader.java @@ -307,13 +307,6 @@ protected void initFile(String id) throws FormatException, IOException { if (settingsPath != null) { settingsPath = new Location(parent, settingsPath).getAbsolutePath(); } - - channels.sort(new Comparator() { - @Override - public int compare(Channel c1, Channel c2) { - return c1.index - c2.index; - } - }); } if (settingsPath != null && new Location(settingsPath).exists()) { @@ -325,6 +318,16 @@ public int compare(Channel c1, Channel c2) { } } + channels.sort(new Comparator() { + @Override + public int compare(Channel c1, Channel c2) { + if (c1.actionIndex != c2.actionIndex) { + return c1.actionIndex - c2.actionIndex; + } + return c1.index - c2.index; + } + }); + for (Channel ch : channels) { if (ch.correctionFile != null) { ch.correctionFile = new Location(parent, ch.correctionFile).getAbsolutePath(); @@ -582,7 +585,7 @@ public int compare(Channel c1, Channel c2) { // particular plane. Skip it. continue; } - Channel channel = lookupChannel(p.channel); + Channel channel = lookupChannel(p); if (channel == null) { continue; } @@ -607,7 +610,7 @@ public int compare(Channel c1, Channel c2) { // the index here is the original bts:Ch index in // the *.mes and *.mrf files - store.setChannelName("Action #" + (channel.actionIndex + 1) + + store.setChannelName("Action #" + (p.actionIndex + 1) + ", Channel #" + (channel.index + 1) + ", Camera #" + channel.cameraNumber, i, c); if (channel.color != null) { @@ -669,7 +672,7 @@ private int getChannelIndex(Plane p) { ch.actionIndex == action) { index++; - if (ch.index == p.channel) { + if (ch.index == p.channel && ch.actionIndex == p.actionIndex) { return index; } } @@ -678,9 +681,9 @@ private int getChannelIndex(Plane p) { return index; } - private Channel lookupChannel(int index) { + private Channel lookupChannel(Plane p) { for (Channel ch : channels) { - if (ch.index == index) { + if (ch.index == p.channel) { return ch; } } @@ -1000,9 +1003,22 @@ public void endElement(String uri, String localName, String qName) { else if (qName.equals("bts:Ch")) { int channelIndex = Integer.parseInt(value) - 1; if (channelIndex >= 0 && channelIndex < channels.size()) { + // the same channel may be acquired multiple times + // if this is the first time the channel is acquired, set the indexes + // if this is the second (or more) time the channel is acquired, + // duplicate the channel so that each action has its own copy with + // the correct indexes Channel ch = channels.get(channelIndex); - ch.timelineIndex = timelineIndex; - ch.actionIndex = actionIndex; + if (ch.timelineIndex == -1 && ch.actionIndex == -1) { + ch.timelineIndex = timelineIndex; + ch.actionIndex = actionIndex; + } + else { + Channel duplicate = new Channel(ch); + duplicate.timelineIndex = timelineIndex; + duplicate.actionIndex = actionIndex; + channels.add(duplicate); + } } } } @@ -1017,8 +1033,8 @@ class LightSource { } class Channel { - public int timelineIndex; - public int actionIndex; + public int timelineIndex = -1; + public int actionIndex = -1; public int index; public double xSize; public double ySize; @@ -1036,6 +1052,26 @@ class Channel { public String fluor; + public Channel() { + } + + public Channel(Channel ch) { + index = ch.index; + xSize = ch.xSize; + ySize = ch.ySize; + cameraNumber = ch.cameraNumber; + correctionFile = ch.correctionFile; + lightSourceRefs = ch.lightSourceRefs; + excitation = ch.excitation; + objectiveID = ch.objectiveID; + objective = ch.objective; + magnification = ch.magnification; + exposureTime = ch.exposureTime; + binning = ch.binning; + color = ch.color; + fluor = ch.fluor; + } + @Override public String toString() { return "timelineIndex=" + timelineIndex + ", actionIndex=" + actionIndex + ", index=" + index;