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

org.openimaj.image.analysis.algorithm.SummedSqTiltAreaTable Maven / Gradle / Ivy

/**
 * Copyright (c) 2011, The University of Southampton and the individual contributors.
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without modification,
 * are permitted provided that the following conditions are met:
 *
 *   * 	Redistributions of source code must retain the above copyright notice,
 * 	this list of conditions and the following disclaimer.
 *
 *   *	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.
 *
 *   *	Neither the name of the University of Southampton nor the names of its
 * 	contributors may be used to endorse or promote products derived from this
 * 	software without specific prior written permission.
 *
 * 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 OWNER 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.
 */
package org.openimaj.image.analysis.algorithm;

import org.openimaj.image.FImage;
import org.openimaj.math.geometry.shape.Rectangle;

/**
 * Implementation of an Integral Image or Summed Area Table. This Implementation
 * calculates both the sum, squared sum values, and (optionally) 45-degree
 * tilted sum values.
 * 

* See http://en.wikipedia.org/wiki/Summed_area_table and * http://research.microsoft * .com/en-us/um/people/viola/Pubs/Detect/violaJones_IJCV.pdf *

* Basically, this provides an efficient way to find the sum of all pixels in a * rectangular area of an image. * * @author Jonathon Hare ([email protected]) */ public class SummedSqTiltAreaTable extends SummedSqAreaTable { /** * The tilted sum data */ public FImage tiltSum; /** * Construct an empty SAT. */ public SummedSqTiltAreaTable() { } /** * Construct a SAT for normal sum, squared sum and tilted sum from the * provided image. * * @param image * the image. */ public SummedSqTiltAreaTable(FImage image) { this(image, true); } /** * Construct a SAT for normal sum, squared sum and (optionally) tilted sum * from the provided image. * * @param image * the image. * @param computeTilted * if true compute the tilted features. */ public SummedSqTiltAreaTable(FImage image, boolean computeTilted) { computeTable(image, computeTilted); } private void computeTable(FImage image, boolean computeTilted) { if (computeTilted) { computeRotSqSumIntegralImages(image); } else { computeSqSumIntegralImages(image); } } protected void computeSqSumIntegralImages(FImage img) { final int width = img.width; final int height = img.height; sum = new FImage(width + 1, height + 1); sqSum = new FImage(width + 1, height + 1); final float[][] sumData = sum.pixels; final float[][] sqSumData = sqSum.pixels; for (int y = 1; y <= height; y++) { float rowSum = 0; float rowSumSQ = 0; final float[] row = img.pixels[y - 1]; for (int x = 1; x <= width; x++) { final float pix = row[x - 1]; rowSum += pix; rowSumSQ += pix * pix; sumData[y][x] = sumData[y - 1][x] + rowSum; sqSumData[y][x] = sqSumData[y - 1][x] + rowSumSQ; } } } protected final void computeRotSqSumIntegralImages(FImage image) { final int width = image.width; final int height = image.height; sum = new FImage(width + 1, height + 1); sqSum = new FImage(width + 1, height + 1); tiltSum = new FImage(width + 2, height + 2); final float[] buffer = new float[width]; // first two rows are special // y == 1 if (height > 0) { final float[] row = image.pixels[0]; float rowSum = 0; float sqRowSum = 0; for (int x = 1; x <= width; x++) { final float gray = (row[x - 1]); rowSum += gray; sqRowSum += gray * gray; sum.pixels[1][x] = rowSum; buffer[x - 1] = tiltSum.pixels[1][x] = gray; sqSum.pixels[1][x] = sqRowSum; } } // y == 2 if (height > 1) { final float[] row = image.pixels[1]; float rowSum = 0; float sqRowSum = 0; for (int x = 1; x < width; x++) { final float gray = (row[x - 1]); rowSum += gray; sqRowSum += gray * gray; sum.pixels[2][x] = sum.pixels[1][x] + rowSum; sqSum.pixels[2][x] = sqSum.pixels[1][x] + sqRowSum; tiltSum.pixels[2][x] = tiltSum.pixels[1][x - 1] + buffer[x - 1] + tiltSum.pixels[1][x + 1] + gray; buffer[x - 1] = gray; } // last column is special if (width > 0) { final float gray = (row[width - 1]); rowSum += gray; sqRowSum += gray * gray; sum.pixels[2][width] = sum.pixels[1][width] + rowSum; sqSum.pixels[2][width] = sqSum.pixels[1][width] + sqRowSum; tiltSum.pixels[2][width] = tiltSum.pixels[1][width - 1] + buffer[width - 1] + gray; buffer[width - 1] = gray; } } for (int y = 3; y <= height; y++) { final float[] row = image.pixels[y - 1]; float rowSum = 0; float sqRowSum = 0; if (width > 0) { final float gray = row[0]; rowSum += gray; sqRowSum += gray * gray; sum.pixels[y][1] = sum.pixels[y - 1][1] + rowSum; sqSum.pixels[y][1] = sqSum.pixels[y - 1][1] + sqRowSum; tiltSum.pixels[y][1] = tiltSum.pixels[y - 1][2] + buffer[0] + gray; buffer[0] = gray; } for (int x = 2; x < width; x++) { final float gray = row[x - 1]; rowSum += gray; sqRowSum += gray * gray; sum.pixels[y][x] = sum.pixels[y - 1][x] + rowSum; sqSum.pixels[y][x] = sqSum.pixels[y - 1][x] + sqRowSum; tiltSum.pixels[y][x] = tiltSum.pixels[y - 1][x - 1] + buffer[x - 1] + tiltSum.pixels[y - 1][x + 1] - tiltSum.pixels[y - 2][x] + gray; buffer[x - 1] = gray; } if (width > 0) { final float gray = row[width - 1]; rowSum += gray; sqRowSum += gray * gray; sum.pixels[y][width] = sum.pixels[y - 1][width] + rowSum; sqSum.pixels[y][width] = sqSum.pixels[y - 1][width] + sqRowSum; tiltSum.pixels[y][width] = tiltSum.pixels[y - 1][width - 1] + buffer[width - 1] + gray; buffer[width - 1] = gray; } } } /** * Calculate the sum of pixels in the image used for constructing this SAT * within the 45 degree tilted rectangle. * * @param x * @param y * @param width * @param height * * @return sum of pixels in given rectangle */ public float calculateTiltedSumArea(int x, int y, int width, int height) { final float p0 = tiltSum.pixels[y][x]; final float p1 = tiltSum.pixels[y + height][x - height]; final float p2 = tiltSum.pixels[y + width][x + width]; final float p3 = tiltSum.pixels[y + width + height][x + width - height]; return p0 - p1 - p2 + p3; } /** * Calculate the sum pixels in the image used for constructing this SAT * within the given 45-degree tilted rectangle. * * @param r * rectangle * @return sum of pixels in given rectangle */ public float calculateTiltedSumArea(Rectangle r) { return calculateTiltedSumArea(Math.round(r.x), Math.round(r.y), Math.round(r.width), Math.round(r.height)); } /* * (non-Javadoc) * * @see * org.openimaj.image.analyser.ImageAnalyser#analyseImage(org.openimaj.image * .Image) */ @Override public void analyseImage(FImage image) { computeTable(image, true); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy