Package nextapp.echo2.webcontainer.syncpeer

Source Code of nextapp.echo2.webcontainer.syncpeer.GridProcessor

/*
* This file is part of the Echo Web Application Framework (hereinafter "Echo").
* Copyright (C) 2002-2009 NextApp, Inc.
*
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
*
* The contents of this file are subject to the Mozilla Public License Version
* 1.1 (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.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License for
* the specific language governing rights and limitations under the License.
*
* Alternatively, the contents of this file may be used under the terms of
* either the GNU General Public License Version 2 or later (the "GPL"), or the
* GNU Lesser General Public License Version 2.1 or later (the "LGPL"), in which
* case the provisions of the GPL or the LGPL are applicable instead of those
* above. If you wish to allow use of your version of this file only under the
* terms of either the GPL or the LGPL, and not to allow others to use your
* version of this file under the terms of the MPL, indicate your decision by
* deleting the provisions above and replace them with the notice and other
* provisions required by the GPL or the LGPL. If you do not delete the
* provisions above, a recipient may use your version of this file under the
* terms of any one of the MPL, the GPL or the LGPL.
*/

package nextapp.echo2.webcontainer.syncpeer;

import java.util.ArrayList;
import java.util.BitSet;
import java.util.List;

import nextapp.echo2.app.Component;
import nextapp.echo2.app.Extent;
import nextapp.echo2.app.Grid;
import nextapp.echo2.app.LayoutData;
import nextapp.echo2.app.layout.GridLayoutData;

/**
* Provides analysis of a <code>Grid</code> for rendering purposes.
* <p>
* This object defines the <code>Grid</code> in terms of two axes, "x" and
* "y". The axes are transposed based on whether the <code>origin</code>
* property of the <code>Grid</code> is horizontal or vertical. For
* horizontally-oriented <code>Grid</code>s, the x-axis represents columns
* and the y-axis represents rows. For vertically oriented <code>Grid</code>s,
* the x-axis represents rows and the y-axis represents columns.
* <p>
* Once a <code>GridProcessor</code> has been instantiated, the rendering
* <code>GridPeer</code> can make inquiries to it to determine how the
* HTML table representing the Grid should be rendered. 
* <p>
* Upon instantiation, the dimensions of the grid are calculated, and the
* content of each cell within those dimensions is determined.  By specifying
* an "x" and "y" coordinate to various getXXX() methods, the renderer can
* determine what <code>Component</code> exists at a particular coordinate,
* how many rows and columns that <code>Component</code> spans, and the index
* of the <code>Component</code> within its parent <code>Grid</code>'s
* children.
* <p>
* This class should not be extended or used by classes outside of the Echo
* framework.
*/
public class GridProcessor {
   
    private static final Integer DEFAULT_SIZE = new Integer(Grid.DEFAULT_SIZE);
   
    /**
     * Internal representation of a rendered cell at a specific coordinate.
     */
    private class Cell {
       
        /**
         * Creates a new <code>Cell</code>.
         *
         * @param component the represented <code>Component</code>
         * @param index the index of <code>component</code> within the
         *        parent <code>Grid</code>
         * @param xSpan the current calculated x-span of the cell
         * @param ySpan the current calculated y-span of the cell
         */
        private Cell(Component component, int index, int xSpan, int ySpan) {
            super();
            this.component = component;
            this.index = index;
            this.xSpan = xSpan;
            this.ySpan = ySpan;
        }
       
        /** the current calculated x-span of the cell */
        private int xSpan;

        /** the current calculated y-span of the cell */
        private int ySpan;

        /** the represented <code>Component</code> */
        private Component component;
       
        /**
         * the index of <code>component</code> within the parent
         * <code>Grid</code>
         */
        private int index;
    }
   
    private Grid grid;
   
    /**
     * A <code>List</code> containing arrays of <code>Cell</code>s.
     * Each contained array of <code>Cell</code>s represents a single point on
     * the y-axis.  The cell indices represent points on the x-axis.
     * Individual y-axis entries should be obtained via the
     * <code>getCellArray()</code> method.
     *
     * @see #getCellArray(int, boolean)
     */
    private List cellArrays;

    /** The current calculated size of the x-axis. */
    private int gridXSize;
   
    /** The current calculated size of the y-axis. */
    private int gridYSize;

    /**
     * The calculated dimensions of each array of cells on the x-axis.
     * These values may change as a result of x-axis reductions.
     */
    private List xExtents;
   
    /**
     * The calculated dimensions of each array of cells on the y-axis.
     * These values may change as a result of y-axis reductions.
     */
    private List yExtents;
   
    /**
     * Flag indicating whether grid has a horizontal (or vertical) orientation.
     */
    private boolean horizontalOrientation;
   
    /**
     * Creates a new <code>GridProcessor</code> for the specified
     * <code>Grid</code>. Creating a new <code>GridProcessor</code> will
     * cause immediately analyze the <code>Grid</code> which will immediately
     * consume processor and memory resources. Such operations should be done at
     * most once per rendering.
     *
     * @param grid the <code>Grid</code>
     */
    public GridProcessor(Grid grid) {
        super();
        this.grid = grid;
        cellArrays = new ArrayList();
       
        Integer orientationValue = (Integer) grid.getRenderProperty(Grid.PROPERTY_ORIENTATION);
        int orientation = orientationValue == null ? Grid.ORIENTATION_HORIZONTAL : orientationValue.intValue();
        horizontalOrientation = orientation != Grid.ORIENTATION_VERTICAL;
       
        Cell[] cells = createCells();
        if (cells == null) {
            // Special case: empty Grid.
            gridXSize = 0;
            gridYSize = 0;
            return;
        }
        renderCellMatrix(cells);
       
        calculateExtents();
       
        reduceY();
        reduceX();
        trimX();
    }
   
    private void calculateExtents() {
        String xProperty = horizontalOrientation ? Grid.PROPERTY_COLUMN_WIDTH : Grid.PROPERTY_ROW_HEIGHT;
        String yProperty = horizontalOrientation ? Grid.PROPERTY_ROW_HEIGHT : Grid.PROPERTY_COLUMN_WIDTH;
       
        xExtents = new ArrayList();
        for (int i = 0; i < gridXSize; ++i) {
            xExtents.add(grid.getRenderIndexedProperty(xProperty, i));
        }
       
        yExtents = new ArrayList();
        for (int i = 0; i < gridYSize; ++i) {
            yExtents.add(grid.getRenderIndexedProperty(yProperty, i));
        }
    }
   
    /**
     * Creates a one-dimensional array of <code>Cell</code> instances
     * representing all visible components contained within the
     * <code>Grid</code>.
     *
     * @return the array of <code>Cell</code>s
     */
    private Cell[] createCells() {
        Component[] children = grid.getVisibleComponents();
       
        if (children.length == 0) {
            // Abort if Grid is empty.
            return null;
        }

        Cell[] cells = new Cell[children.length];

        for (int i = 0; i < children.length; ++i) {
            LayoutData layoutData = (LayoutData) children[i].getRenderProperty(Grid.PROPERTY_LAYOUT_DATA);
            if (layoutData instanceof GridLayoutData) {
                GridLayoutData gcLayoutData = (GridLayoutData) layoutData;
                int xSpan = horizontalOrientation ? gcLayoutData.getColumnSpan() : gcLayoutData.getRowSpan();
                int ySpan = horizontalOrientation ? gcLayoutData.getRowSpan() : gcLayoutData.getColumnSpan();
                cells[i] = new Cell(children[i], i, xSpan, ySpan);
            } else {
                cells[i] = new Cell(children[i], i, 1, 1);
            }
        }
        return cells;
    }
   
    /**
     * Retrieves the array of cells representing a particular coordinate on
     * the y-axis of the rendered representation.
     *
     * @param y the y-axis index
     * @param expand a flag indicating whether the <code>CellArray</code>
     *        should be expanded in the event the given y-axis does not
     *        exist.
     * @see #cellArrays
     */
    private Cell[] getCellArray(int y, boolean expand) {
        while (expand && y >= cellArrays.size()) {
            cellArrays.add(new Cell[gridXSize]);
        }
        return (Cell[]) cellArrays.get(y);
    }
   
    /**
     * Returns the <code>Component</code> that should be rendered at the
     * specified position.
     *
     * @param column the column index
     * @param row the row index
     * @return the <code>Component</code> (may be null)
     */
    public Component getContent(int column, int row) {
        if (horizontalOrientation) {
            Cell cell = getCellArray(row, false)[column];
            return cell == null ? null : cell.component;
        } else {
            Cell cell = getCellArray(column, false)[row];
            return cell == null ? null : cell.component;
        }
    }

    /**
     * Returns the index of the <code>Component</code> that should be rendered
     * at the specified position within its parent <code>Grid</code>
     * container.
     *
     * @param column the column index
     * @param row the row index
     * @return the index of the <code>Component</code> within its
     *         container.
     */
    public int getComponentIndex(int column, int row) {
        if (horizontalOrientation) {
            Cell cell = getCellArray(row, false)[column];
            return cell == null ? -1 : cell.index;
        } else {
            Cell cell = getCellArray(column, false)[row];
            return cell == null ? -1 : cell.index;
        }
    }
   
    /**
     * Returns the number of columns that should be rendered.
     *
     * @return the number of rendered columns
     */
    public int getColumnCount() {
        return horizontalOrientation ? gridXSize : gridYSize;
    }
   
    /**
     * Returns the number of rows that should be rendered.
     *
     * @return the number of rendered rows
     */
    public int getRowCount() {
        return horizontalOrientation ? gridYSize : gridXSize;
    }
   
    /**
     * Returns the width of the specified column index
     *
     * @param column the column index
     * @return the width
     */
    public Extent getColumnWidth(int column) {
        return (Extent) (horizontalOrientation ? xExtents : yExtents).get(column);
    }
   
    /**
     * Returns the height of the specified row index
     *
     * @param row the row index
     * @return the height
     */
    public Extent getRowHeight(int row) {
        return (Extent) (horizontalOrientation ? yExtents : xExtents).get(row);
    }
   
    /**
     * Returns the column span of the cell at the specified rendered index.
     *
     * @param column the column index
     * @param row the row index
     * @return the column span (-1 will be returned in the event that no
     *         cell exists at the specified index)
     */
    public int getColumnSpan(int column, int row) {
        if (horizontalOrientation) {
            Cell cell = getCellArray(row, false)[column];
            return cell == null ? -1 : cell.xSpan;
        } else {
            Cell cell = getCellArray(column, false)[row];
            return cell == null ? -1 : cell.ySpan;
        }
    }
   
    /**
     * Returns the row span of the cell at the specified rendered index.
     *
     * @param column the column index
     * @param row the row index
     * @return the row span (-1 will be returned in the event that no
     *         cell exists at the specified index)
     */
    public int getRowSpan(int column, int row) {
        if (horizontalOrientation) {
            Cell cell = getCellArray(row, false)[column];
            return cell == null ? -1 : cell.ySpan;
        } else {
            Cell cell = getCellArray(column, false)[row];
            return cell == null ? -1 : cell.xSpan;
        }
    }

    /**
     * Remove duplicates from the x-axis where all cells simply
     * "span over" a given x-axis coordinate.
     */
    private void reduceX() {
        // Determine duplicate cell sets on x-axis.
        BitSet xRemoves = new BitSet();
        int x = 1;
        int length = getCellArray(0, false).length;
        while (x  < length) {
            int y = 0;
            boolean identical = true;
            while (y < cellArrays.size()) {
                if (getCellArray(y, false)[x] != getCellArray(y, false)[x - 1]) {
                    identical = false;
                    break;
                }
                ++y;
            }
            if (identical) {
                xRemoves.set(x, true);
            }
            ++x;
        }

        // If no reductions are necessary on the x-axis, do nothing.
        if (xRemoves.nextSetBit(0) == -1) {
            return;
        }
       
        for (int removedX = gridXSize - 1; removedX >= 0; --removedX) {
            if (!xRemoves.get(removedX)) {
                continue;
            }

            for (int y = 0; y < gridYSize; ++y) {
                if (y == 0 || getCellArray(y, false)[removedX - 1] != getCellArray(y - 1, false)[removedX - 1]) {
                    // Reduce x-span, taking care not to reduce it multiple times if cell has a y-span.
                    Cell[] cellArray = getCellArray(y, false);
                    if (cellArray[removedX - 1] != null) {
                        --getCellArray(y, false)[removedX - 1].xSpan;
                    }
                }
                for (x = removedX; x < gridXSize - 1; ++x) {
                    getCellArray(y, false)[x] = getCellArray(y, false)[x + 1];
                }
            }
           
            // Calculate Extent-size of merged indices.
            Extent retainedExtent = (Extent) xExtents.get(removedX - 1);
            Extent removedExtent = (Extent) xExtents.get(removedX);
            xExtents.remove(removedX);
            if (removedExtent != null) {
                xExtents.set(removedX - 1, Extent.add(removedExtent, retainedExtent));
            }
           
            --gridXSize;
        }
    }
   
    /**
     * Remove duplicates from the y-axis where all cells simply
     * "span over" a given y-axis coordinate.
     */
    private void reduceY() {
        // Determine duplicate cell sets on y-axis.
        BitSet yRemoves = new BitSet();
        int y = 1;
       
        int size = cellArrays.size();
        Cell[] previousCellArray;
        Cell[] currentCellArray = getCellArray(0, false);
       
        while (y < size) {
            previousCellArray = currentCellArray;
            currentCellArray = getCellArray(y, false);
           
            int x = 0;
            boolean identical = true;

            while (x < currentCellArray.length) {
                if (currentCellArray[x] != previousCellArray[x]) {
                    identical = false;
                    break;
                }
                ++x;
            }
            if (identical) {
                yRemoves.set(y, true);
            }
           
            ++y;
        }
       
        // If no reductions are necessary on the y-axis, do nothing.
        if (yRemoves.nextSetBit(0) == -1) {
            return;
        }
           
        for (int removedY = gridYSize -1; removedY >= 0; --removedY) {
            if (!yRemoves.get(removedY)) {
                continue;
            }
           
            // Shorten the y-spans of the cell array that will be retained to
            // reflect the fact that a cell array is being removed.
            Cell[] retainedCellArray = getCellArray(removedY - 1, false);
            for (int x = 0; x < gridXSize; ++x) {
                if (x == 0 || retainedCellArray[x] != retainedCellArray[x - 1]) {
                    // Reduce y-span, taking care not to reduce it multiple times if cell has a x-span.
                    if (retainedCellArray[x] != null) {
                        --retainedCellArray[x].ySpan;
                    }
                }
            }
           
            // Remove the duplicate cell array.
            cellArrays.remove(removedY);
           
            // Calculate Extent-size of merged indices.
            Extent retainedExtent = (Extent) yExtents.get(removedY - 1);
            Extent removedExtent = (Extent) yExtents.get(removedY);
            yExtents.remove(removedY);
            if (removedExtent != null) {
                yExtents.set(removedY - 1, Extent.add(removedExtent, retainedExtent));
            }

            // Decrement the grid size to reflect cell array removal.
            --gridYSize;
        }
    }
   
    /**
     * Iterates over cells to create the cell matrix, adjusting column and row spans as of cells to ensure
     * that no overlap occurs between column and row spans.
     * Additionally determines actual y-size of grid.  
     *
     * @param cells the grid cells
     */
    private void renderCellMatrix(Cell[] cells) {
        gridXSize = ((Integer) grid.getRenderProperty(Grid.PROPERTY_SIZE, DEFAULT_SIZE)).intValue();
       
        int x = 0, y = 0;
        Cell[] yCells = getCellArray(y, true);
        for (int componentIndex = 0; componentIndex < cells.length; ++componentIndex) {

            // Set x-span to fill remaining size in the even SPAN_FILL has been specified or if the cell would
            // otherwise extend past the specified size.
            if (cells[componentIndex].xSpan == GridLayoutData.SPAN_FILL || cells[componentIndex].xSpan > gridXSize - x) {
                cells[componentIndex].xSpan = gridXSize - x;
            }

            // Set x-span of any cell INCORRECTLY set to negative value to 1 (note that SPAN_FILL has already been handled).
            if (cells[componentIndex].xSpan < 1) {
                cells[componentIndex].xSpan = 1;
            }
            // Set y-span of any cell INCORRECTLY set to negative value (or more likely SPAN_FILL) to 1.
            if (cells[componentIndex].ySpan < 1) {
                cells[componentIndex].ySpan = 1;
            }
           
            if (cells[componentIndex].xSpan != 1 || cells[componentIndex].ySpan != 1) {
                // Scan to ensure no y-spans are blocking this x-span.
                // If a y-span is blocking, shorten the x-span to not
                // interfere.
                for (int xIndex = 1; xIndex < cells[componentIndex].xSpan; ++xIndex) {
                    if (yCells[x + xIndex] != null) {
                        // Blocking component found.
                        cells[componentIndex].xSpan = xIndex;
                        break;
                    }
                }
                for (int yIndex = 0; yIndex < cells[componentIndex].ySpan; ++yIndex) {
                    Cell[] yIndexCells = getCellArray(y + yIndex, true);
                    for (int xIndex = 0; xIndex < cells[componentIndex].xSpan; ++xIndex) {
                        yIndexCells[x + xIndex] = cells[componentIndex];
                    }
                }
            }
            yCells[x] = cells[componentIndex];

            if (componentIndex < cells.length - 1) {
                // Move rendering cursor.
                boolean nextRenderPointFound = false;
                while (!nextRenderPointFound) {
                    if (x < gridXSize - 1) {
                        ++x;
                    } else {
                        // Move cursor to next line.
                        x = 0;
                        ++y;
                        yCells = getCellArray(y, true);
                       
                    }
                    nextRenderPointFound = yCells[x] == null;
                }
            }
        }

        // Store actual 'y' dimension.
        gridYSize = cellArrays.size();
    }
   
    /**
     * Special case: Trim excess null cells from Grid x-size if the
     * Grid y-size is 1.
     */
    private void trimX() {
        if (gridYSize == 1) {
            Cell[] cellArray = getCellArray(0, false);
            for (int i = 0; i < gridXSize; ++i) {
                if (cellArray[i] == null) {
                    gridXSize = i;
                }
            }
        }
    }
}
TOP

Related Classes of nextapp.echo2.webcontainer.syncpeer.GridProcessor

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.