001/** 002 * 003 * Copyright (c) 2014, the Railo Company Ltd. All rights reserved. 004 * 005 * This library is free software; you can redistribute it and/or 006 * modify it under the terms of the GNU Lesser General Public 007 * License as published by the Free Software Foundation; either 008 * version 2.1 of the License, or (at your option) any later version. 009 * 010 * This library is distributed in the hope that it will be useful, 011 * but WITHOUT ANY WARRANTY; without even the implied warranty of 012 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 013 * Lesser General Public License for more details. 014 * 015 * You should have received a copy of the GNU Lesser General Public 016 * License along with this library. If not, see <http://www.gnu.org/licenses/>. 017 * 018 **/ 019/* 020* 021 022Licensed under the Apache License, Version 2.0 (the "License"); 023you may not use this file except in compliance with the License. 024You may obtain a copy of the License at 025 026 http://www.apache.org/licenses/LICENSE-2.0 027 028Unless required by applicable law or agreed to in writing, software 029distributed under the License is distributed on an "AS IS" BASIS, 030WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 031See the License for the specific language governing permissions and 032limitations under the License. 033*/ 034 035package lucee.runtime.img.filter;import java.awt.image.BufferedImage; 036 037import lucee.runtime.engine.ThreadLocalPageContext; 038import lucee.runtime.exp.FunctionException; 039import lucee.runtime.exp.PageException; 040import lucee.runtime.img.ImageUtil; 041import lucee.runtime.type.KeyImpl; 042import lucee.runtime.type.Struct; 043import lucee.runtime.type.util.CollectionUtil; 044 045/** 046 * A filter which can be used to produce wipes by transferring the luma of a mask image into the alpha channel of the source. 047 */ 048public class HalftoneFilter extends AbstractBufferedImageOp implements DynFiltering { 049 050 private float density = 0; 051 private float softness = 0; 052 private boolean invert; 053 private BufferedImage mask; 054 055 public HalftoneFilter() { 056 } 057 058 /** 059 * Set the density of the image in the range 0..1. 060 * *arg density The density 061 */ 062 public void setDensity( float density ) { 063 this.density = density; 064 } 065 066 public float getDensity() { 067 return density; 068 } 069 070 /** 071 * Set the softness of the effect in the range 0..1. 072 * @param softness the softness 073 * @min-value 0 074 * @max-value 1 075 * @see #getSoftness 076 */ 077 public void setSoftness( float softness ) { 078 this.softness = softness; 079 } 080 081 /** 082 * Get the softness of the effect. 083 * @return the softness 084 * @see #setSoftness 085 */ 086 public float getSoftness() { 087 return softness; 088 } 089 090 public void setMask( BufferedImage mask ) { 091 this.mask = mask; 092 } 093 094 public BufferedImage getMask() { 095 return mask; 096 } 097 098 public void setInvert( boolean invert ) { 099 this.invert = invert; 100 } 101 102 public boolean getInvert() { 103 return invert; 104 } 105 106 public BufferedImage filter( BufferedImage src, BufferedImage dst ) { 107 int width = src.getWidth(); 108 int height = src.getHeight(); 109 110 if ( dst == null ) 111 dst = createCompatibleDestImage( src, null ); 112 if ( mask == null ) 113 return dst; 114 115 int maskWidth = mask.getWidth(); 116 int maskHeight = mask.getHeight(); 117 118 //float d = density * (1+softness); 119 //float lower = 255 * (d-softness); 120 //float upper = 255 * d; 121 float s = 255*softness; 122 123 int[] inPixels = new int[width]; 124 int[] maskPixels = new int[maskWidth]; 125 126 for ( int y = 0; y < height; y++ ) { 127 getRGB( src, 0, y, width, 1, inPixels ); 128 getRGB( mask, 0, y % maskHeight, maskWidth, 1, maskPixels ); 129 130 for ( int x = 0; x < width; x++ ) { 131 int maskRGB = maskPixels[x % maskWidth]; 132 int inRGB = inPixels[x]; 133 int v = PixelUtils.brightness( maskRGB ); 134 int iv = PixelUtils.brightness( inRGB ); 135 float f = ImageMath.smoothStep( iv-s, iv+s, v ); 136 int a = (int)(255 * f); 137 138 if ( invert ) 139 a = 255-a; 140// inPixels[x] = (a << 24) | (inRGB & 0x00ffffff); 141 inPixels[x] = (inRGB & 0xff000000) | (a << 16) | (a << 8) | a; 142 } 143 144 setRGB( dst, 0, y, width, 1, inPixels ); 145 } 146 147 return dst; 148 } 149 150 public String toString() { 151 return "Stylize/Halftone..."; 152 } 153 public BufferedImage filter(BufferedImage src, Struct parameters) throws PageException {BufferedImage dst=ImageUtil.createBufferedImage(src); 154 Object o; 155 if((o=parameters.removeEL(KeyImpl.init("Density")))!=null)setDensity(ImageFilterUtil.toFloatValue(o,"Density")); 156 if((o=parameters.removeEL(KeyImpl.init("Softness")))!=null)setSoftness(ImageFilterUtil.toFloatValue(o,"Softness")); 157 if((o=parameters.removeEL(KeyImpl.init("Invert")))!=null)setInvert(ImageFilterUtil.toBooleanValue(o,"Invert")); 158 if((o=parameters.removeEL(KeyImpl.init("Mask")))!=null)setMask(ImageFilterUtil.toBufferedImage(o,"Mask")); 159 160 // check for arguments not supported 161 if(parameters.size()>0) { 162 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 [Density, Softness, Invert, Mask]"); 163 } 164 165 return filter(src, dst); 166 } 167}