/* * 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 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 performs a box blur on an image. The horizontal and vertical blurs can be specified separately * and a number of iterations can be given which allows an approximation to Gaussian blur. */ public class BoxBlurFilter extends AbstractBufferedImageOp implements DynFiltering { private float hRadius; private float vRadius; private int iterations = 1; private boolean premultiplyAlpha = true; /** * Construct a default BoxBlurFilter. */ public BoxBlurFilter() { } /** * Construct a BoxBlurFilter. * @param hRadius the horizontal radius of blur * @param vRadius the vertical radius of blur * @param iterations the number of time to iterate the blur */ public BoxBlurFilter( float hRadius, float vRadius, int iterations ) { this.hRadius = hRadius; this.vRadius = vRadius; this.iterations = iterations; } /** * Set whether to premultiply the alpha channel. * @param premultiplyAlpha true to premultiply the alpha * @see #getPremultiplyAlpha */ public void setPremultiplyAlpha( boolean premultiplyAlpha ) { this.premultiplyAlpha = premultiplyAlpha; } /** * Get whether to premultiply the alpha channel. * @return true to premultiply the alpha * @see #setPremultiplyAlpha */ public boolean getPremultiplyAlpha() { return premultiplyAlpha; } public BufferedImage filter( BufferedImage src, BufferedImage dst ) { int width = src.getWidth(); int height = src.getHeight(); if ( dst == null ) dst = createCompatibleDestImage( src, null ); int[] inPixels = new int[width*height]; int[] outPixels = new int[width*height]; getRGB( src, 0, 0, width, height, inPixels ); if ( premultiplyAlpha ) ImageMath.premultiply( inPixels, 0, inPixels.length ); for (int i = 0; i < iterations; i++ ) { blur( inPixels, outPixels, width, height, hRadius ); blur( outPixels, inPixels, height, width, vRadius ); } blurFractional( inPixels, outPixels, width, height, hRadius ); blurFractional( outPixels, inPixels, height, width, vRadius ); if ( premultiplyAlpha ) ImageMath.unpremultiply( inPixels, 0, inPixels.length ); setRGB( dst, 0, 0, width, height, inPixels ); return dst; } /** * Blur and transpose a block of ARGB pixels. * @param in the input pixels * @param out the output pixels * @param width the width of the pixel array * @param height the height of the pixel array * @param radius the radius of blur */ public static void blur( int[] in, int[] out, int width, int height, float radius ) { int widthMinus1 = width-1; int r = (int)radius; int tableSize = 2*r+1; int divide[] = new int[256*tableSize]; for ( int i = 0; i < 256*tableSize; i++ ) divide[i] = i/tableSize; int inIndex = 0; for ( int y = 0; y < height; y++ ) { int outIndex = y; int ta = 0, tr = 0, tg = 0, tb = 0; for ( int i = -r; i <= r; i++ ) { int rgb = in[inIndex + ImageMath.clamp(i, 0, width-1)]; ta += (rgb >> 24) & 0xff; tr += (rgb >> 16) & 0xff; tg += (rgb >> 8) & 0xff; tb += rgb & 0xff; } for ( int x = 0; x < width; x++ ) { out[ outIndex ] = (divide[ta] << 24) | (divide[tr] << 16) | (divide[tg] << 8) | divide[tb]; int i1 = x+r+1; if ( i1 > widthMinus1 ) i1 = widthMinus1; int i2 = x-r; if ( i2 < 0 ) i2 = 0; int rgb1 = in[inIndex+i1]; int rgb2 = in[inIndex+i2]; ta += ((rgb1 >> 24) & 0xff)-((rgb2 >> 24) & 0xff); tr += ((rgb1 & 0xff0000)-(rgb2 & 0xff0000)) >> 16; tg += ((rgb1 & 0xff00)-(rgb2 & 0xff00)) >> 8; tb += (rgb1 & 0xff)-(rgb2 & 0xff); outIndex += height; } inIndex += width; } } public static void blurFractional( int[] in, int[] out, int width, int height, float radius ) { radius -= (int)radius; float f = 1.0f/(1+2*radius); int inIndex = 0; for ( int y = 0; y < height; y++ ) { int outIndex = y; out[ outIndex ] = in[0]; outIndex += height; for ( int x = 1; x < width-1; x++ ) { int i = inIndex+x; int rgb1 = in[i-1]; int rgb2 = in[i]; int rgb3 = in[i+1]; int a1 = (rgb1 >> 24) & 0xff; int r1 = (rgb1 >> 16) & 0xff; int g1 = (rgb1 >> 8) & 0xff; int b1 = rgb1 & 0xff; int a2 = (rgb2 >> 24) & 0xff; int r2 = (rgb2 >> 16) & 0xff; int g2 = (rgb2 >> 8) & 0xff; int b2 = rgb2 & 0xff; int a3 = (rgb3 >> 24) & 0xff; int r3 = (rgb3 >> 16) & 0xff; int g3 = (rgb3 >> 8) & 0xff; int b3 = rgb3 & 0xff; a1 = a2 + (int)((a1 + a3) * radius); r1 = r2 + (int)((r1 + r3) * radius); g1 = g2 + (int)((g1 + g3) * radius); b1 = b2 + (int)((b1 + b3) * radius); a1 *= f; r1 *= f; g1 *= f; b1 *= f; out[ outIndex ] = (a1 << 24) | (r1 << 16) | (g1 << 8) | b1; outIndex += height; } out[ outIndex ] = in[width-1]; inIndex += width; } } /** * Set the horizontal size of the blur. * @param hRadius the radius of the blur in the horizontal direction * @min-value 0 * @see #getHRadius */ public void setHRadius(float hRadius) { this.hRadius = hRadius; } /** * Get the horizontal size of the blur. * @return the radius of the blur in the horizontal direction * @see #setHRadius */ public float getHRadius() { return hRadius; } /** * Set the vertical size of the blur. * @param vRadius the radius of the blur in the vertical direction * @min-value 0 * @see #getVRadius */ public void setVRadius(float vRadius) { this.vRadius = vRadius; } /** * Get the vertical size of the blur. * @return the radius of the blur in the vertical direction * @see #setVRadius */ public float getVRadius() { return vRadius; } /** * Set both the horizontal and vertical sizes of the blur. * @param radius the radius of the blur in both directions * @min-value 0 * @see #getRadius */ public void setRadius(float radius) { this.hRadius = this.vRadius = radius; } /** * Get the size of the blur. * @return the radius of the blur in the horizontal direction * @see #setRadius */ public float getRadius() { return hRadius; } /** * Set the number of iterations the blur is performed. * @param iterations the number of iterations * @min-value 0 * @see #getIterations */ public void setIterations(int iterations) { this.iterations = iterations; } /** * Get the number of iterations the blur is performed. * @return the number of iterations * @see #setIterations */ public int getIterations() { return iterations; } public String toString() { return "Blur/Box Blur..."; } public BufferedImage filter(BufferedImage src, Struct parameters) throws PageException {BufferedImage dst=ImageUtil.createBufferedImage(src); Object o; if((o=parameters.removeEL(KeyImpl.init("PremultiplyAlpha")))!=null)setPremultiplyAlpha(ImageFilterUtil.toBooleanValue(o,"PremultiplyAlpha")); if((o=parameters.removeEL(KeyImpl.init("Iterations")))!=null)setIterations(ImageFilterUtil.toIntValue(o,"Iterations")); if((o=parameters.removeEL(KeyImpl.init("HRadius")))!=null)setHRadius(ImageFilterUtil.toFloatValue(o,"HRadius")); if((o=parameters.removeEL(KeyImpl.init("VRadius")))!=null)setVRadius(ImageFilterUtil.toFloatValue(o,"VRadius")); if((o=parameters.removeEL(KeyImpl.init("Radius")))!=null)setRadius(ImageFilterUtil.toFloatValue(o,"Radius")); // 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 [PremultiplyAlpha, Iterations, HRadius, VRadius, Radius]"); } return filter(src, dst); } }