org.csa.rstb.polarimetric.gpf.decompositions.Touzi.java Source code

Java tutorial

Introduction

Here is the source code for org.csa.rstb.polarimetric.gpf.decompositions.Touzi.java

Source

/*
 * Copyright (C) 2015 by Array Systems Computing Inc. http://www.array.ca
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU 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 General Public License for
 * more details.
 *
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, see http://www.gnu.org/licenses/
 */
package org.csa.rstb.polarimetric.gpf.decompositions;

import org.apache.commons.math3.util.FastMath;
import org.csa.rstb.polarimetric.gpf.PolOpUtils;
import org.esa.s1tbx.io.PolBandUtils;
import org.esa.s1tbx.io.PolBandUtils.MATRIX;
import org.esa.snap.core.datamodel.Band;
import org.esa.snap.core.datamodel.ProductData;
import org.esa.snap.core.gpf.Operator;
import org.esa.snap.core.gpf.OperatorException;
import org.esa.snap.core.gpf.Tile;
import org.esa.snap.engine_utilities.eo.Constants;
import org.esa.snap.engine_utilities.gpf.TileIndex;

import java.awt.*;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;

/**
 * Perform Touzi decomposition for given tile.
 */
public class Touzi extends DecompositionBase implements Decomposition {

    private final boolean outputTouziParamSet0;
    private final boolean outputTouziParamSet1;
    private final boolean outputTouziParamSet2;
    private final boolean outputTouziParamSet3;

    public Touzi(final PolBandUtils.PolSourceBand[] srcBandList, final MATRIX sourceProductType,
            final int windowSize, final int srcImageWidth, final int srcImageHeight,
            final boolean outputTouziParamSet0, final boolean outputTouziParamSet1,
            final boolean outputTouziParamSet2, final boolean outputTouziParamSet3) {
        super(srcBandList, sourceProductType, windowSize, windowSize, srcImageWidth, srcImageHeight);

        this.outputTouziParamSet0 = outputTouziParamSet0;
        this.outputTouziParamSet1 = outputTouziParamSet1;
        this.outputTouziParamSet2 = outputTouziParamSet2;
        this.outputTouziParamSet3 = outputTouziParamSet3;
    }

    /**
     * Return the list of band names for the target product
     *
     * @return list of band names
     */
    public String[] getTargetBandNames() {
        final List<String> targetBandNameList = new ArrayList<>(4);

        if (!outputTouziParamSet0 && !outputTouziParamSet1 && !outputTouziParamSet2 && !outputTouziParamSet3) {
            throw new OperatorException("Please select decomposition parameters to output");
        }

        if (outputTouziParamSet0) {
            targetBandNameList.add("Psi");
            targetBandNameList.add("Tau");
            targetBandNameList.add("Alpha");
            targetBandNameList.add("Phi");
        }
        if (outputTouziParamSet1) {
            targetBandNameList.add("Psi1");
            targetBandNameList.add("Tau1");
            targetBandNameList.add("Alpha1");
            targetBandNameList.add("Phi1");
        }
        if (outputTouziParamSet2) {
            targetBandNameList.add("Psi2");
            targetBandNameList.add("Tau2");
            targetBandNameList.add("Alpha2");
            targetBandNameList.add("Phi2");
        }
        if (outputTouziParamSet3) {
            targetBandNameList.add("Psi3");
            targetBandNameList.add("Tau3");
            targetBandNameList.add("Alpha3");
            targetBandNameList.add("Phi3");
        }

        return targetBandNameList.toArray(new String[targetBandNameList.size()]);
    }

    /**
     * Sets the unit for the new target band
     *
     * @param targetBandName the band name
     * @param targetBand     the new target band
     */
    public void setBandUnit(final String targetBandName, final Band targetBand) {
        targetBand.setUnit("rad");
    }

    /**
     * Perform decomposition for given tile.
     *
     * @param targetTiles     The current tiles to be computed for each target band.
     * @param targetRectangle The area in pixel coordinates to be computed.
     * @param op              the polarimetric decomposition operator
     * @throws OperatorException If an error occurs during computation of the filtered value.
     */
    public void computeTile(final Map<Band, Tile> targetTiles, final Rectangle targetRectangle, final Operator op) {

        final int x0 = targetRectangle.x;
        final int y0 = targetRectangle.y;
        final int w = targetRectangle.width;
        final int h = targetRectangle.height;
        final int maxY = y0 + h;
        final int maxX = x0 + w;
        //System.out.println("x0 = " + x0 + ", y0 = " + y0 + ", w = " + w + ", h = " + h);

        final TileIndex trgIndex = new TileIndex(targetTiles.get(op.getTargetProduct().getBandAt(0)));

        final double[][] Tr = new double[3][3];
        final double[][] Ti = new double[3][3];
        final double[][] EigenVectRe = new double[3][3];
        final double[][] EigenVectIm = new double[3][3];
        final double[] EigenVal = new double[3];

        final double[] psi = new double[3];
        final double[] tau = new double[3];
        final double[] alpha = new double[3];
        final double[] phi = new double[3];
        final double[] vr = new double[3];
        final double[] vi = new double[3];
        double p1, p2, p3, psiMean, tauMean, alphaMean, phiMean;
        double phase, c, s, tmp1r, tmp1i, tmp2r, tmp2i;

        for (final PolBandUtils.PolSourceBand bandList : srcBandList) {

            final Tile[] sourceTiles = new Tile[bandList.srcBands.length];
            final ProductData[] dataBuffers = new ProductData[bandList.srcBands.length];
            final Rectangle sourceRectangle = getSourceRectangle(x0, y0, w, h);
            for (int i = 0; i < bandList.srcBands.length; ++i) {
                sourceTiles[i] = op.getSourceTile(bandList.srcBands[i], sourceRectangle);
                dataBuffers[i] = sourceTiles[i].getDataBuffer();
            }
            final TileIndex srcIndex = new TileIndex(sourceTiles[0]);

            for (int y = y0; y < maxY; ++y) {
                trgIndex.calculateStride(y);
                for (int x = x0; x < maxX; ++x) {
                    final int idx = trgIndex.getIndex(x);

                    PolOpUtils.getMeanCoherencyMatrix(x, y, halfWindowSizeX, halfWindowSizeY, sourceImageWidth,
                            sourceImageHeight, sourceProductType, srcIndex, dataBuffers, Tr, Ti);

                    PolOpUtils.eigenDecomposition(3, Tr, Ti, EigenVectRe, EigenVectIm, EigenVal);

                    for (int k = 0; k < 3; ++k) {
                        for (int l = 0; l < 3; ++l) {
                            vr[l] = EigenVectRe[l][k];
                            vi[l] = EigenVectIm[l][k];
                        }

                        phase = Math.atan2(vi[0], vr[0] + PolOpUtils.EPS);
                        c = FastMath.cos(phase);
                        s = FastMath.sin(phase);
                        for (int l = 0; l < 3; ++l) {
                            tmp1r = vr[l];
                            tmp1i = vi[l];
                            vr[l] = tmp1r * c + tmp1i * s;
                            vi[l] = tmp1i * c - tmp1r * s;
                        }

                        psi[k] = 0.5 * Math.atan2(vr[2], vr[1] + PolOpUtils.EPS);

                        tmp1r = vr[1];
                        tmp1i = vi[1];
                        tmp2r = vr[2];
                        tmp2i = vi[2];
                        c = FastMath.cos(2.0 * psi[k]);
                        s = FastMath.sin(2.0 * psi[k]);
                        vr[1] = tmp1r * c + tmp2r * s;
                        vi[1] = tmp1i * c + tmp2i * s;
                        vr[2] = -tmp1r * s + tmp2r * c;
                        vi[2] = -tmp1i * s + tmp2i * c;

                        tau[k] = 0.5 * Math.atan2(-vi[2], vr[0] + PolOpUtils.EPS);

                        phi[k] = Math.atan2(vi[1], vr[1] + PolOpUtils.EPS);

                        alpha[k] = Math
                                .atan(Math.sqrt((vr[1] * vr[1] + vi[1] * vi[1]) / (vr[0] * vr[0] + vi[2] * vi[2])));

                        if ((psi[k] < -Constants.PI / 4.0) || (psi[k] > Constants.PI / 4.0)) {
                            tau[k] = -tau[k];
                            phi[k] = -phi[k];
                        }

                    }

                    final double sum = EigenVal[0] + EigenVal[1] + EigenVal[2];
                    p1 = EigenVal[0] / sum;
                    p2 = EigenVal[1] / sum;
                    p3 = EigenVal[2] / sum;

                    psiMean = p1 * psi[0] + p2 * psi[1] + p3 * psi[2];
                    tauMean = p1 * tau[0] + p2 * tau[1] + p3 * tau[2];
                    alphaMean = p1 * alpha[0] + p2 * alpha[1] + p3 * alpha[2];
                    phiMean = p1 * phi[0] + p2 * phi[1] + p3 * phi[2];

                    for (final Band band : bandList.targetBands) {
                        final String targetBandName = band.getName();
                        final ProductData dataBuffer = targetTiles.get(band).getDataBuffer();
                        if (outputTouziParamSet0) {
                            if (targetBandName.equals("Psi") || targetBandName.contains("Psi_"))
                                dataBuffer.setElemFloatAt(idx, (float) psiMean);
                            else if (targetBandName.equals("Tau") || targetBandName.contains("Tau_"))
                                dataBuffer.setElemFloatAt(idx, (float) tauMean);
                            else if (targetBandName.equals("Alpha") || targetBandName.contains("Alpha_"))
                                dataBuffer.setElemFloatAt(idx, (float) alphaMean);
                            else if (targetBandName.equals("Phi") || targetBandName.contains("Phi_"))
                                dataBuffer.setElemFloatAt(idx, (float) phiMean);
                        }
                        if (outputTouziParamSet1) {
                            if (targetBandName.contains("Psi1"))
                                dataBuffer.setElemFloatAt(idx, (float) psi[0]);
                            else if (targetBandName.contains("Tau1"))
                                dataBuffer.setElemFloatAt(idx, (float) tau[0]);
                            else if (targetBandName.contains("Alpha1"))
                                dataBuffer.setElemFloatAt(idx, (float) alpha[0]);
                            else if (targetBandName.contains("Phi1"))
                                dataBuffer.setElemFloatAt(idx, (float) phi[0]);
                        }
                        if (outputTouziParamSet2) {
                            if (targetBandName.contains("Psi2"))
                                dataBuffer.setElemFloatAt(idx, (float) psi[1]);
                            else if (targetBandName.contains("Tau2"))
                                dataBuffer.setElemFloatAt(idx, (float) tau[1]);
                            else if (targetBandName.contains("Alpha2"))
                                dataBuffer.setElemFloatAt(idx, (float) alpha[1]);
                            else if (targetBandName.contains("Phi2"))
                                dataBuffer.setElemFloatAt(idx, (float) phi[1]);
                        }
                        if (outputTouziParamSet3) {
                            if (targetBandName.contains("Psi3"))
                                dataBuffer.setElemFloatAt(idx, (float) psi[2]);
                            else if (targetBandName.contains("Tau3"))
                                dataBuffer.setElemFloatAt(idx, (float) tau[2]);
                            else if (targetBandName.contains("Alpha3"))
                                dataBuffer.setElemFloatAt(idx, (float) alpha[2]);
                            else if (targetBandName.contains("Phi3"))
                                dataBuffer.setElemFloatAt(idx, (float) phi[2]);
                        }
                    }

                }
            }
        }
    }
}