/* Copyright 2006 Jerry Huxtable 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 com.jhlabs.image; import java.awt.AlphaComposite; import java.awt.Graphics2D; import java.awt.RenderingHints; import java.awt.image.BufferedImage; import java.awt.image.BufferedImageOp; import java.beans.BeanInfo; import java.beans.IntrospectionException; import java.beans.Introspector; import java.beans.PropertyDescriptor; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; /** * A filter which uses another filter to perform a transition. * e.g. to create a blur transition, you could write: new TransitionFilter( new BoxBlurFilter(), "radius", 0, 100 ); */ public class TransitionFilter extends AbstractBufferedImageOp { private float transition = 0; private BufferedImage destination; private String property; private Method method; /** * The filter used for the transition. */ protected BufferedImageOp filter; /** * The start value for the filter property. */ protected float minValue; /** * The end value for the filter property. */ protected float maxValue; /** * Construct a TransitionFilter. */ private TransitionFilter() { } /** * Construct a TransitionFilter. * @param filter the filter to use * @param property the filter property which is changed over the transition * @param minValue the start value for the filter property * @param maxValue the end value for the filter property */ public TransitionFilter( BufferedImageOp filter, String property, float minValue, float maxValue ) { this.filter = filter; this.property = property; this.minValue = minValue; this.maxValue = maxValue; try { BeanInfo info = Introspector.getBeanInfo( filter.getClass() ); PropertyDescriptor[] pds = info.getPropertyDescriptors(); for (PropertyDescriptor pd : pds) { if ( property.equals( pd.getName() ) ) { method = pd.getWriteMethod(); break; } } if ( method == null ) { throw new IllegalArgumentException( "No such property in object: "+property ); } } catch (IntrospectionException e) { throw new IllegalArgumentException( e.toString() ); } } /** * Set the transition of the image in the range 0..1. * @param transition the transition * @min-value 0 * @max-value 1 * @see #getTransition */ public void setTransition( float transition ) { this.transition = transition; } /** * Get the transition of the image. * @return the transition * @see #setTransition */ public float getTransition() { return transition; } /** * Set the destination image. * @param destination the destination image * @see #getDestination */ public void setDestination( BufferedImage destination ) { this.destination = destination; } /** * Get the destination image. * @return the destination image * @see #setDestination */ public BufferedImage getDestination() { return destination; } /* public void setFilter( BufferedImageOp filter ) { this.filter = filter; } public int getFilter() { return filter; } */ /** * Prepare the filter for the transiton at a given time. * The default implementation sets the given filter property, but you could override this method to make other changes. * @param transition the transition time in the range 0 - 1 */ public void prepareFilter( float transition ) { try { method.invoke( filter, new Object[] { transition} ); } catch ( IllegalAccessException | IllegalArgumentException | InvocationTargetException e ) { throw new IllegalArgumentException("Error setting value for property: "+property); } } @Override public BufferedImage filter( BufferedImage src, BufferedImage dst ) { if ( dst == null ) { dst = createCompatibleDestImage( src, null ); } if ( destination == null ) { return dst; } float itransition = 1-transition; Graphics2D g = dst.createGraphics(); g.setRenderingHint(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_SPEED); g.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_OFF); g.setRenderingHint(RenderingHints.KEY_TEXT_ANTIALIASING, RenderingHints.VALUE_TEXT_ANTIALIAS_OFF); g.setRenderingHint(RenderingHints.KEY_FRACTIONALMETRICS, RenderingHints.VALUE_FRACTIONALMETRICS_OFF); g.setRenderingHint(RenderingHints.KEY_COLOR_RENDERING, RenderingHints.VALUE_COLOR_RENDER_SPEED); g.setRenderingHint(RenderingHints.KEY_DITHERING, RenderingHints.VALUE_DITHER_DISABLE); if ( transition != 1 ) { float t = minValue + transition * ( maxValue-minValue ); prepareFilter( t ); g.drawImage( src, filter, 0, 0 ); } if ( transition != 0 ) { g.setComposite( AlphaComposite.getInstance( AlphaComposite.SRC_OVER, transition ) ); float t = minValue + itransition * ( maxValue-minValue ); prepareFilter( t ); g.drawImage( destination, filter, 0, 0 ); } g.dispose(); return dst; } @Override public String toString() { return "Transitions/Transition..."; } @Override public Object clone() throws CloneNotSupportedException { return super.clone(); //To change body of generated methods, choose Tools | Templates. } }