All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.myfaces.trinidadinternal.image.encode.OctreeQuantizer Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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 org.apache.myfaces.trinidadinternal.image.encode;

import java.awt.Image;
import java.awt.image.ImageObserver;
import java.awt.image.PixelGrabber;
import org.apache.myfaces.trinidadinternal.image.painter.ImageLoader;
import org.apache.myfaces.trinidad.logging.TrinidadLogger;

/**
 * Reduces an image's color palette by means of an octree quantization method.
 * 

* @version $Name: $ ($Revision: adfrt/faces/adf-faces-impl/src/main/java/oracle/adfinternal/view/faces/image/encode/OctreeQuantizer.java#0 $) $Date: 10-nov-2005.19:05:19 $ * @since 0.1.4 * @see OctreeNode */ class OctreeQuantizer { // The octree quantization method reduces a set of colors in an // image to a desired color palette size while maintaining as much // image quality as possible. The principle structure employed here // is an octree, or tree with a branching factor of 8. The tree is // also 8 nodes deep, as each color may be represented by an 8 bit // number. // // The image is scanned and pixel colors are read into the tree and // stored one at a time. Branch selection is as follows: at depth n, // the nth most significant r, g, and b values form rgb, a binary // number from 0 to 7. The red, green, and blue values of the entire // color are added to red, green, and blue fields at every node // along the way, so every node has the sum of color value for all // of its children. The number of unique colors in the tree is // noted, though red, green, and blue value totals per node as // described above are done for *each* pixel. // // Once the number of unique colors exceeds the desired threshhold, // from 2 to 8 colors are combined into a single color. A search is // done from the bottom of the tree, since colors with a common // node share all of that nodes's parental hierarchy. Nodes in // deeper levels are combined first, however which node is combined // first within a row results in slightly different outcomes, which // are not discussed here. In this implementation, the first viable // node is reduced. // // Once a node has been selected, its children are removed and it // takes a color equal to a weighted average of its children. Since // color sums are kept in each node, all that needs be done is // divide the total red, green, and blue values by the number of // pixels passing through the node. // // Note that further colors that would have been added to pruned // nodes are not added further than the prune. The image color is // not changed either, though red, green, and blue totals are still // updated as before. // // Once the image has been scanned in, it is trivial to map a color // to its equivalent from the new color table. Simply traverse the // tree as before, stopping when a node with no children is reached // and return that node's color value. // // This algorithm is made more efficient by the use of color lists // to more quickly find nodes to reduce. As nodes are created they // are added to the appropriate list for their level. When a reduce // is performed, the lists are scanned, one at a time, for a viable // node. If the end of a list is reached, the next highest list is // scanned, and no further node will ever be created in the lower // list. This thus makes the adding nodes operation progressively // faster. /** * Create a new octree and insert image data. This constructor is mostly * used by OctreeFilter. * @param im The image that provides pixel data to construct the tree */ public OctreeQuantizer(Image im) { this(); // first retrieve the pixels ImageLoader il = new ImageLoader(im); il.start(); if(!il.waitFor()){ throw new IllegalArgumentException(_LOG.getMessage( "PROBLEM_LOADING")); } int width = im.getWidth(il); int height = im.getHeight(il); int[] pixels = new int[width*height]; // going to hold all // the image's pixels PixelGrabber grabber = new PixelGrabber(im.getSource(), 0, 0, width, height, pixels, 0, width); try // get the pixels { grabber.grabPixels(); } catch (InterruptedException e) { throw new IllegalArgumentException(_LOG.getMessage( "GRABBING_PIXELS")); } if ((grabber.getStatus() & ImageObserver.ABORT) != 0) { throw new IllegalArgumentException(_LOG.getMessage( "ERROR_FETCHING_IMAGE", new Object[]{pixels.length,width,height})); } // add all pixels to the tree. for (int i = 0; i < pixels.length; i++) addColor(pixels[i]); } /** * Map an array of color values into an octree to reduce the colors. * Modify the array to reflect the color changes. This method is used * to do all the creation and mapping at once. * @param pixels The image data. */ public static void mapColors(int[] pixels) { OctreeQuantizer o = new OctreeQuantizer(); // add all the colors to the tree for (int i = 0; i < pixels.length; i++) o.addColor(pixels[i]); // modify the array for (int i = 0; i < pixels.length; i++) pixels[i] = o.mapColor(pixels[i]); } /** * Add a color to the tree. This is a wrapper for a call to the root * OctreeNode. * @param rgb the color to be added */ public void addColor(int rgb) { _root.addColor(rgb); } /** * Map a color to the tree. This is a wrapper for a call to the root * OctreeNode. * @param rgb the color to be mapped * @return the mapped color */ public int mapColor(int rgb) { return _root.mapColor(rgb); } OctreeNode _getListHead(int i) { return _listHead[i]; } int _getMaxDepth() { return _maxDepth; } // increment the number of colors, and if this goes over the mark, reduce void _incColors() { if(++_colors > _MAX_COLORS) _reduce(); } // define the beginning of a list void _setListHead(int i, OctreeNode n) { _listHead[i] = n; _listEnd[i] = n; } // add to a list void _setListEnd(int i, OctreeNode n) { _listEnd[i]._setNext(n); // continue the chain _listEnd[i] = n; } // create a root private OctreeQuantizer() { _root = new OctreeNode(this); _listHead = new OctreeNode[8]; _listEnd = new OctreeNode[8]; _maxDepth = 8; } // pick the first viable node (lowest level that will remove at least // one color private OctreeNode _pickNode() { // apparently, once you've risen a level, you don't go back. I // question apple's claim, but it works okay nonetheless. // why _maxDepth-2? -1 since _maxDepth starts at 8 and the array is 0-based. // -2 because we're only concerned with the seventh level and above - leaves // can't be pruned. for (int i = _maxDepth-2; i >=0; i--) { // traverse list i for a good node OctreeNode n = _getListHead(i); OctreeNode p = null; // previously selected node // picking specific node OctreeNode saveNode = null; OctreeNode savePrev = null; // traverse the entire level looking for the best node, which is placed in saveNode while (n != null) { // usable nodes have siblings if (n._getChildren() > 1) { // picking specific node if ((saveNode == null) || (n._getChildren() < saveNode._getChildren())) { saveNode = n; savePrev = p; } } // still looking p = n; n = n._getNext(); } if (saveNode != null) { // if we picked the first node, then we have to change the value // in the list of heads. if (savePrev == null) { _listHead[i] = saveNode._getNext(); } else { OctreeNode nextNode = saveNode._getNext(); // if we picked the last node, then we have to change the value // in the list of ends if (nextNode == null) { _listEnd[i] = savePrev; } savePrev._setNext(nextNode); // otherwise preserve the chain // if nextNode is null, we still want to erase n from p.next } return saveNode; } // we've exhausted this level _maxDepth--; } return null; } // get rid of some nodes private void _reduce() { // which node to reduce? OctreeNode n = _pickNode(); // now no colors will be created under this node n._setMaxLevel(n._getLevel()); // determine color based on weighted average n._computeColor(); _colors -= (n._getChildren()-1); // children are pruned // now children will never be used by _pickNode n._setChildren(0); } // the size of the desired color table private static final int _MAX_COLORS = 255; private int _colors; // how many colors currently in the tree // heads of lists of available nodes. Where all _pickNode calls look from private OctreeNode[] _listHead; // tails of lists of available nodes. Where new nodes are added private OctreeNode[] _listEnd; private int _maxDepth; // max. relevant depth of the tree private OctreeNode _root; // root node of the tree private static final TrinidadLogger _LOG = TrinidadLogger.createTrinidadLogger( OctreeQuantizer.class); }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy