All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.scijava.ops.flim.util.RAHelper Maven / Gradle / Ivy

The newest version!
/*-
 * #%L
 * Fluorescence lifetime analysis in SciJava Ops.
 * %%
 * Copyright (C) 2024 SciJava developers.
 * %%
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 * 
 * 1. Redistributions of source code must retain the above copyright notice,
 *    this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright notice,
 *    this list of conditions and the following disclaimer in the documentation
 *    and/or other materials provided with the distribution.
 * 
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 * #L%
 */

package org.scijava.ops.flim.util;

import net.imglib2.RandomAccess;
import net.imglib2.type.numeric.RealType;
import net.imglib2.type.numeric.integer.IntType;
import net.imglib2.type.numeric.real.FloatType;
import org.scijava.ops.flim.FitResults;
import org.scijava.ops.flim.FitParams;

public class RAHelper> {

	final RandomAccess transRA;
	final RandomAccess intensityRA, initialParamRA, fittedParamRA,
			fittedRA, residualsRA, chisqRA;
	final RandomAccess retcodeRA;
	final int lifetimeAxis, raOffset, bufDataStart, bufDataEnd;

	public RAHelper(FitParams params, FitResults rslts) {
		this.lifetimeAxis = params.ltAxis;
		transRA = params.transMap.randomAccess();
		intensityRA = rslts.intensityMap.randomAccess();
		initialParamRA = params.paramMap != null && params.paramMap.dimension(
			lifetimeAxis) >= params.nComp ? params.paramMap.randomAccess() : null;
		fittedParamRA = params.getParamMap ? rslts.paramMap.randomAccess() : null;
		fittedRA = params.getFittedMap ? rslts.fittedMap.randomAccess() : null;
		residualsRA = params.getResidualsMap ? rslts.residualsMap.randomAccess()
			: null;
		chisqRA = params.getChisqMap ? rslts.chisqMap.randomAccess() : null;
		retcodeRA = params.getReturnCodeMap ? rslts.retCodeMap.randomAccess()
			: null;

		int prefixLen = Math.min(params.instr == null ? 0 : params.instr.length,
			params.fitStart);

		// where does copy start in transRA
		raOffset = params.fitStart - prefixLen;
		// the start and end data (excluding instr prefix/suffix) index in buffer
		bufDataStart = prefixLen;
		bufDataEnd = bufDataStart + params.fitEnd - params.fitStart;
	}

	/**
	 * Fill buffers with data from {@code params}. Starting coordinate specified
	 * by {@code xytPos}. Skip and return {@code false} if
	 * {@code params.intensityMap} is less than intensity threshold.
	 *
	 * @param transBuffer the transient buffer
	 * @param paramBuffer the parameter buffer
	 * @param params the fitting parameters
	 * @param xytPos the starting position of data
	 * @return {@code false} if the transient is skipped, {@code true} otherwise
	 */
	public boolean loadData(float[] transBuffer, float[] paramBuffer,
		FitParams params, int[] xytPos)
	{
		// intensity thresholding
		intensityRA.setPosition(xytPos);
		intensityRA.setPosition(0, lifetimeAxis);
		if (intensityRA.get().getRealFloat() < params.iThresh) {
			return false;
		}

		// load transient
		transRA.setPosition(xytPos);
		// to take the trans before start when convolving
		transRA.setPosition(raOffset, lifetimeAxis);
		for (int t = 0; t < transBuffer.length; t++, transRA.fwd(lifetimeAxis)) {
			transBuffer[t] = transRA.get().getRealFloat();
		}

		// fill initial values from params.paramMap
		if (initialParamRA != null) {
			initialParamRA.setPosition(xytPos);
			for (int p = 0; p < paramBuffer.length; p++, initialParamRA.fwd(
				lifetimeAxis))
			{
				paramBuffer[p] = initialParamRA.get().getRealFloat();
			}
		}
		// try to fill initial values from map if per-pixel parameter not present
		else if (params.param != null) {
			for (int p = 0; p < paramBuffer.length; p++) {
				paramBuffer[p] = params.param[p];
			}
		}
		// or if both local and global settings are not present, set to +Inf "no
		// estimation"
		else {
			for (int p = 0; p < paramBuffer.length; p++) {
				paramBuffer[p] = Float.POSITIVE_INFINITY;
			}
		}
		return true;
	}

	/**
	 * Put results back into the proper position in {@code rslts}. An RA in
	 * {@code rslts} will be skipped if {@code params.getXxMap} is {@code true}.
	 *
	 * @param params the fitting parameters
	 * @param rslts the result to fill in
	 * @param xytPos the coordinate of the single-pixel result in maps.
	 */
	public void commitRslts(FitParams params, FitResults rslts, int[] xytPos) {
		if (params.getReturnCodeMap) {
			retcodeRA.setPosition(xytPos);
			retcodeRA.get().set(rslts.retCode);
		}
		if (params.dropBad && rslts.retCode != FitResults.RET_OK) return;
		if (params.getChisqMap) {
			chisqRA.setPosition(xytPos);
			chisqRA.get().set(rslts.chisq);
		}
		// fill in maps on demand
		if (params.getParamMap) {
			fillRA(fittedParamRA, xytPos, rslts.param, 0, rslts.param.length);
		}
		if (params.getFittedMap) {
			fillRA(fittedRA, xytPos, rslts.fitted, bufDataStart, bufDataEnd);
		}
		if (params.getResidualsMap) {
			fillRA(residualsRA, xytPos, rslts.residuals, bufDataStart, bufDataEnd);
		}
	}

	private void fillRA(RandomAccess ra, int[] xytPos, float[] arr,
		int start, int end)
	{
		xytPos[lifetimeAxis] = 0;
		ra.setPosition(xytPos);
		for (int i = start; i < end; i++) {
			ra.get().set(arr[i]);
			ra.fwd(lifetimeAxis);
		}
	}
}