/* * RapidMiner * * Copyright (C) 2001-2008 by Rapid-I and the contributors * * Complete list of developers available at our web site: * * http://rapid-i.com * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see http://www.gnu.org/licenses/. */ package com.rapidminer.gui.look.painters; import java.awt.Component; import java.awt.Graphics; import java.awt.GraphicsConfiguration; import java.awt.Image; import java.awt.Transparency; import java.awt.image.BufferedImage; import java.awt.image.VolatileImage; import java.util.HashMap; import java.util.Map; /** * A cached painter. * * @author Ingo Mierswa * @version $Id: AbstractCachedPainter.java,v 1.2 2008/05/09 19:23:16 ingomierswa Exp $ */ public abstract class AbstractCachedPainter { protected static final Map<Object, Cache> cacheMap = new HashMap<Object, Cache>(); public AbstractCachedPainter(int cacheCount) { getCache(getClass()).setMaxCount(cacheCount); } /** Paints the representation to cache to the supplied Graphics. */ protected abstract void paintToImage(Component c, Graphics g, int w, int h, Object[] args); public void paint(Component c, Graphics g, int x, int y, int w, int h, Object[] args) { if ((w <= 0) || (h <= 0)) { return; } Object key = getClass(); GraphicsConfiguration config = c.getGraphicsConfiguration(); Cache cache = getCache(key); Image image = cache.getImage(key, config, w, h, args); int attempts = 0; do { boolean draw = false; if (image instanceof VolatileImage) { switch (((VolatileImage) image).validate(config)) { case VolatileImage.IMAGE_INCOMPATIBLE: ((VolatileImage) image).flush(); image = null; break; case VolatileImage.IMAGE_RESTORED: draw = true; break; } } if (image == null) { image = createImage(c, w, h, config); cache.setImage(key, config, w, h, args, image); draw = true; } if (draw) { Graphics g2 = image.getGraphics(); paintToImage(c, g2, w, h, args); g2.dispose(); } paintImage(c, g, x, y, w, h, image, args); } while ((image instanceof VolatileImage) && ((VolatileImage) image).contentsLost() && (++attempts < 3)); } protected void paintImage(Component c, Graphics g, int x, int y, int w, int h, Image image, Object[] args) { g.drawImage(image, x, y, null); } protected Image createImage(Component c, int w, int h, GraphicsConfiguration config) { if (config == null) { return new BufferedImage(w, h, BufferedImage.TYPE_4BYTE_ABGR); } return config.createCompatibleImage(w, h, Transparency.BITMASK); } public static void clearCache() { synchronized (cacheMap) { cacheMap.clear(); } } private static Cache getCache(Object key) { synchronized (cacheMap) { Cache cache = cacheMap.get(key); if (cache == null) { cache = new Cache(1); cacheMap.put(key, cache); } return cache; } } }