/* * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ package railo.runtime.img.filter;import java.awt.image.BufferedImage; import java.util.Random; import railo.runtime.engine.ThreadLocalPageContext; import railo.runtime.exp.FunctionException; import railo.runtime.exp.PageException; import railo.runtime.img.ImageUtil; import railo.runtime.type.KeyImpl; import railo.runtime.type.Struct; import railo.runtime.type.util.CollectionUtil; /** * A filter which adds random noise into an image. */ public class NoiseFilter extends PointFilter implements DynFiltering { /** * Gaussian distribution for the noise. */ public final static int GAUSSIAN = 0; /** * Uniform distribution for the noise. */ public final static int UNIFORM = 1; private int amount = 25; private int distribution = UNIFORM; private boolean monochrome = false; private float density = 1; private Random randomNumbers = new Random(); public NoiseFilter() { } /** * Set the amount of effect. * @param amount the amount * @min-value 0 * @max-value 1 * @see #getAmount */ public void setAmount(int amount) { this.amount = amount; } /** * Get the amount of noise. * @return the amount * @see #setAmount */ public int getAmount() { return amount; } /** * Set the distribution of the noise. * @param distribution the distribution * @see #getDistribution */ public void setDistribution( int distribution ) { this.distribution = distribution; } /** * Get the distribution of the noise. * @return the distribution * @see #setDistribution */ public int getDistribution() { return distribution; } /** * Set whether to use monochrome noise. * @param monochrome true for monochrome noise * @see #getMonochrome */ public void setMonochrome(boolean monochrome) { this.monochrome = monochrome; } /** * Get whether to use monochrome noise. * @return true for monochrome noise * @see #setMonochrome */ public boolean getMonochrome() { return monochrome; } /** * Set the density of the noise. * @param density the density * @see #getDensity */ public void setDensity( float density ) { this.density = density; } /** * Get the density of the noise. * @return the density * @see #setDensity */ public float getDensity() { return density; } private int random(int x) { x += (int)(((distribution == GAUSSIAN ? randomNumbers.nextGaussian() : 2*randomNumbers.nextFloat() - 1)) * amount); if (x < 0) x = 0; else if (x > 0xff) x = 0xff; return x; } public int filterRGB(int x, int y, int rgb) { if ( randomNumbers.nextFloat() <= density ) { int a = rgb & 0xff000000; int r = (rgb >> 16) & 0xff; int g = (rgb >> 8) & 0xff; int b = rgb & 0xff; if (monochrome) { int n = (int)(((distribution == GAUSSIAN ? randomNumbers.nextGaussian() : 2*randomNumbers.nextFloat() - 1)) * amount); r = PixelUtils.clamp(r+n); g = PixelUtils.clamp(g+n); b = PixelUtils.clamp(b+n); } else { r = random(r); g = random(g); b = random(b); } return a | (r << 16) | (g << 8) | b; } return rgb; } public String toString() { return "Stylize/Add Noise..."; } public BufferedImage filter(BufferedImage src, Struct parameters) throws PageException {BufferedImage dst=ImageUtil.createBufferedImage(src); Object o; if((o=parameters.removeEL(KeyImpl.init("Amount")))!=null)setAmount(ImageFilterUtil.toIntValue(o,"Amount")); if((o=parameters.removeEL(KeyImpl.init("Monochrome")))!=null)setMonochrome(ImageFilterUtil.toBooleanValue(o,"Monochrome")); if((o=parameters.removeEL(KeyImpl.init("Density")))!=null)setDensity(ImageFilterUtil.toFloatValue(o,"Density")); if((o=parameters.removeEL(KeyImpl.init("Distribution")))!=null)setDistribution(ImageFilterUtil.toIntValue(o,"Distribution")); if((o=parameters.removeEL(KeyImpl.init("Dimensions")))!=null){ int[] dim=ImageFilterUtil.toDimensions(o,"Dimensions"); setDimensions(dim[0],dim[1]); } // check for arguments not supported if(parameters.size()>0) { throw new FunctionException(ThreadLocalPageContext.get(), "ImageFilter", 3, "parameters", "the parameter"+(parameters.size()>1?"s":"")+" ["+CollectionUtil.getKeyList(parameters,", ")+"] "+(parameters.size()>1?"are":"is")+" not allowed, only the following parameters are supported [Amount, Monochrome, Density, Distribution, Dimensions]"); } return filter(src, dst); } }