Logo Search packages:      
Sourcecode: jaxe version File versions  Download package

JaxeTableView.java

/*
 Jaxe - Editeur XML en Java

 Copyright (C) 2002 Observatoire de Paris-Meudon

 Ce programme est un logiciel libre ; vous pouvez le redistribuer et/ou le modifier conformément aux dispositions de la Licence Publique Générale GNU, telle que publiée par la Free Software Foundation ; version 2 de la licence, ou encore (à votre choix) toute version ultérieure.
  
 Ce programme est distribué dans l'espoir qu'il sera utile, mais SANS AUCUNE GARANTIE ; sans même la garantie implicite de COMMERCIALISATION ou D'ADAPTATION A UN OBJET PARTICULIER. Pour plus de détail, voir la Licence Publique Générale GNU .

 Vous devez avoir reçu un exemplaire de la Licence Publique Générale GNU en même temps que ce programme ; si ce n'est pas le cas, écrivez à la Free Software Foundation Inc., 675 Mass Ave, Cambridge, MA 02139, Etats-Unis.
 */

package jaxe;

import org.apache.log4j.Logger;

import java.awt.Color;
import java.awt.Graphics;
import java.awt.Rectangle;
import java.awt.Shape;

import javax.swing.SizeRequirements;
import javax.swing.text.AttributeSet;
import javax.swing.text.BoxView;
import javax.swing.text.Element;
import javax.swing.text.ParagraphView;
import javax.swing.text.TableView;
import javax.swing.text.View;
import javax.swing.text.ViewFactory;
import javax.swing.text.html.HTML;

// attention, ici Element = javax.swing.text.Element

/**
 * Vue pour les tables de JETableTexte
 */
00037 public class JaxeTableView extends TableView implements ViewFactory {
    /**
     * Logger for this class
     */
00041     private static final Logger LOG = Logger.getLogger(JaxeTableView.class);

    String TRtag = "tr";

    String TDtag = "td";

    String THtag = "th";

    String CaptionTag = "caption";

    public JaxeTableView(final Element elem) {
        super(elem);
        setInsets((short) 2, (short) 2, (short) 2, (short) 2);
    }

    @Override
    public void paint(final Graphics g, final Shape allocation) {
        // paint interior
        final int n = getViewCount();
        for (int i = 0; i < n; i++) {
            final View v = getView(i);
            v.paint(g, getChildAllocation(i, allocation));
        }

        final Rectangle alloc = (allocation instanceof Rectangle) ? (Rectangle) allocation
                : allocation.getBounds();

        g.setColor(Color.gray);
        g.draw3DRect(alloc.x, alloc.y, alloc.width - 1, alloc.height - 1, true);
        g.setColor(Color.black);
    }

    @Override
    public ViewFactory getViewFactory() {
        return this;
    }

    @Override
    protected View getViewAtPoint(final int x, final int y, final Rectangle alloc) {
        final int n = getViewCount();
        View v = null;
        final Rectangle allocation = new Rectangle();
        for (int i = 0; i < n; i++) {
            allocation.setBounds(alloc);
            childAllocation(i, allocation);
            v = getView(i);
            if (v instanceof JaxeTableRow) {
                v = ((JaxeTableRow) v).findViewAtPoint(x, y, allocation);
                if (v != null) {
                    alloc.setBounds(allocation);
                    return v;
                }
            }
        }
        return super.getViewAtPoint(x, y, alloc);
    }

    /**
     * The table itself acts as a factory for the various views that actually
     * represent pieces of the table. All other factory activity is delegated to
     * the factory returned by the parent of the table.
     */
00103     public View create(final Element elem) {
        final String kind = elem.getName();
        if (kind != null) {
            if (kind.equals(TRtag)) {
                return new JaxeTableRow(elem);
            } else if ((kind.equals(TDtag)) || (kind.equals(THtag))) {
                return new JaxeTableCell(elem);
            } else if (kind.equals(CaptionTag)) { return new ParagraphView(elem); }
        }

        final View p = getParent();
        if (p != null) {
            final ViewFactory f = p.getViewFactory();
            if (f != null) { return f.create(elem); }
        }
        return null;
    }

    private int getColsOccupied(final View v) {
        final AttributeSet a = v.getElement().getAttributes();
        final String s = (String) a.getAttribute(HTML.Attribute.COLSPAN);
        if (s != null) {
            try {
                return Math.max(Integer.parseInt(s), 1);
            } catch (final NumberFormatException nfe) {
            }
        }
        return 1;
    }

    private int getRowsOccupied(final View v) {
        final AttributeSet a = v.getElement().getAttributes();
        final String s = (String) a.getAttribute(HTML.Attribute.ROWSPAN);
        if (s != null) {
            try {
                return Math.max(Integer.parseInt(s), 1);
            } catch (final NumberFormatException nfe) {
            }
        }
        return 1;
    }

    /**
     * Calculates the height of all tablecells
     * @return height of the table
     */
00149     float calculateAllCells() {

        final int n = getViewCount();
        int maxcols = 0;
        int maxrows = 0;

        //How many colums are in the table?
        for (int i = 0; i < n; i++) {
            final View v = getView(i);
            if (v instanceof JaxeTableRow) {
                final JaxeTableRow jtr = (JaxeTableRow) v;

                int cols = 0;

                final int m = jtr.getViewCount();
                for (int j = 0; j < m; j++) {

                    cols += getColsOccupied(jtr.getView(j));
                }

                maxcols = Math.max(cols, maxcols);

            }
        }

        // How may rows are in the table?
        maxrows = n;
        
        final float[][] sizes = new float[maxrows][maxcols];
        final float[] maxrowsize = new float[maxrows];

        //Now read the sizes of all cells which have no rowspan
        for (int i = 0; i < maxrows; i++) {
            final View v = getView(i);
            maxrowsize[i] = 0;
            if (v instanceof JaxeTableRow) {
                final JaxeTableRow jtr = (JaxeTableRow) v;

                for (int j = 0, s = jtr.getViewCount(); j < s; j++) {
                    final JaxeTableCell tcell = (JaxeTableCell) jtr.getView(j);
                    final int rows = getRowsOccupied(tcell);
                    if (rows == 1) {
                        sizes[i][j] = tcell.getRealSpan(Y_AXIS);
                        maxrowsize[i] = Math.max(maxrowsize[i], sizes[i][j]);
                    } else {
                        sizes[i][j] = -1;
                    }

                }
            }
        }

        //The rowsize ist the highest cellsize
        for (int row = 0; row < maxrows; row++) {
            float rowpref = 0;
            for (int col = 0; col < maxcols; col++) {
                rowpref = Math.max(rowpref, sizes[row][col]);
            }
            final JaxeTableRow jtr = (JaxeTableRow) getView(row);
            jtr.setPreferredSize((int) rowpref);
            for (int cellc = 0,s = jtr.getViewCount(); cellc < s; cellc++) {
                final JaxeTableCell cell = (JaxeTableCell) jtr.getView(cellc);
                final int rows = getRowsOccupied(cell);
                if (rows == 1) {
                    cell.setPreferredHeight(rowpref);
                }
            }
        }

        boolean retry = false;
        do {
            retry = false;
        //Now multirowcells get their sizes
        for (int i = 0; i < maxrows && !retry; i++) {
            final View v = getView(i);
            if ((v instanceof JaxeTableRow) && (!retry)){
                final JaxeTableRow jtr = (JaxeTableRow) v;

                for (int j = 0, s = jtr.getViewCount(); j < s && !retry; j++) {
                    final JaxeTableCell tcell = (JaxeTableCell) jtr.getView(j);
                    final int rows = getRowsOccupied(tcell);

                    if ((rows > 1) && (!retry)){
                        float maxspan = 0;

                        for (int anz = 0; anz < rows; anz++) {
                            if (i+anz < maxrowsize.length) {
                                maxspan += maxrowsize[i + anz];
                            }
                        }
                        
                        if (maxspan > tcell.getRealSpan(Y_AXIS)) {
                            tcell.setPreferredHeight(maxspan);
                        } else if (maxspan < tcell.getRealSpan(Y_AXIS)) {
                            tcell.setPreferredHeight(tcell.getRealSpan(Y_AXIS));
                            final float diff = (tcell.getRealSpan(Y_AXIS) - maxspan)
                                    / rows;
                            for (int anz = 0; anz < rows; anz++) {
                                
                                if (i+anz < maxrowsize.length) {
                                    final JaxeTableRow nextJtr = (JaxeTableRow) getView(i + anz);
                                    nextJtr.setPreferredSize((int) (nextJtr
                                            .getPreferredSpan(Y_AXIS) + diff));
                                    maxrowsize[i + anz] = maxrowsize[i + anz] + diff;
                                }
                                
                            }
                            retry = true; // Repeat resize because 1 cell was altered
                        }
                    }

                }
            }
        }
        } while (retry);

        //Normalize all cellsizes and calculate the tableheight
        float tableheight = 0;
        for (int row = 0; row < maxrows; row++) {
            final JaxeTableRow jtr = (JaxeTableRow) getView(row);
            for (int cellc = 0; cellc < jtr.getViewCount(); cellc++) {
                final JaxeTableCell cell = (JaxeTableCell) jtr.getView(cellc);
                if (getRowsOccupied(cell) == 1) {
                    cell.setPreferredHeight((int) maxrowsize[row]);
                }
            }
            tableheight += maxrowsize[row];
        }

        return tableheight;
    }

    @Override
    protected SizeRequirements calculateMajorAxisRequirements(final int axis,
            SizeRequirements r) {

        if (axis == Y_AXIS) {
            super.calculateMajorAxisRequirements(axis, r);
            r = new SizeRequirements();
            
            r.preferred = r.maximum = r.minimum = (int) calculateAllCells();
        } else {
            r = super.calculateMajorAxisRequirements(axis, r);
            r.maximum = r.preferred;
        }

        return (r);
    }

    @Override
    public void setParent(final View parent) {
        // a useless replace is causing a setParent(null). As a result, child
        // views
        // forget about their parents and are unable to change it back when
        // setParent(v) is called afterwards.
        if (parent != null) super.setParent(parent);
    }

    class JaxeTableRow extends TableRow {

        public JaxeTableRow(final Element elem) {
            super(elem);
       }

        /**
         * This is called by a child to indicate its preferred span has changed.
         * This is implemented to execute the superclass behavior and well as
         * try to determine if a row with a multi-row cell hangs across this
         * row. If a multi-row cell covers this row it also needs to propagate a
         * preferenceChanged so that it will recalculate the multi-row cell.
         * 
         * @param child
         *            the child view
         * @param width
         *            true if the width preference should change
         * @param height
         *            true if the height preference should change
         */

        View findViewAtPoint(final int x, final int y, final Rectangle alloc) {
            final int n = getViewCount();
            for (int i = 0; i < n; i++) {
                final Rectangle s = getChildAllocation(i, alloc).getBounds();
                
                final JaxeTableCell cell  = (JaxeTableCell)getView(i);
                s.height = (int) cell.getPreferredHeight();
                
                if (s.contains(x, y)) {
                    childAllocation(i, alloc);
                    return getView(i);
                }
            }
            return null;
        }

        @Override
        protected void layoutMinorAxis(final int targetSpan, final int axis, final int[] offsets,
                final int[] spans) {
            //switch
            calculateAllCells();
            int col = 0;
            final int ncells = getViewCount();
            int max = 0;

            for (int cell = 0; cell < ncells; cell++, col++) {
                final View cv = getView(cell);
                spans[cell] = (int) cv.getPreferredSpan(Y_AXIS);

                if (spans[cell] > max) {
                    max = spans[cell];
                }

            }
            super.layoutMinorAxis(targetSpan, axis, offsets, spans);
        }

        @Override
        public void paint(final Graphics g, final Shape allocation) {
            final Rectangle alloc = (allocation instanceof Rectangle) ? (Rectangle) allocation
                    : allocation.getBounds();

            /** used in paint. */
            final Rectangle tempRect = new Rectangle();

            final int n = getViewCount();
            final int x = alloc.x + getLeftInset();
            final int y = alloc.y + getTopInset();
            for (int i = 0; i < n; i++) {
                tempRect.x = x + getOffset(X_AXIS, i);
                tempRect.y = y + getOffset(Y_AXIS, i);
                tempRect.width = getSpan(X_AXIS, i);
                tempRect.height = getSpan(Y_AXIS, i);
                paintChild(g, tempRect, i);
            }
        }

        @Override
        public float getPreferredSpan(final int axis) {
            final float f = super.getPreferredSpan(axis);
            if (axis == Y_AXIS) { return pref; }
            return f;
        }

        @Override
        public float getMinimumSpan(final int axis) {
            final float f = super.getMinimumSpan(axis);
            if (axis == Y_AXIS) { return pref; }
            return f;
        }

        @Override
        public float getMaximumSpan(final int axis) {
            final float f = super.getMaximumSpan(axis);

            if (axis == Y_AXIS) { return pref; }

            return f;
        }

        int pref = 0;

        public void setPreferredSize(final int size) {
            pref = size;
        }

        @Override
        protected SizeRequirements calculateMinorAxisRequirements(final int axis,
                SizeRequirements r) {
            if (axis == Y_AXIS) {
                super.calculateMinorAxisRequirements(axis, r);
                r = new SizeRequirements();
                r.alignment = 0;
                r.preferred = pref;
                r.minimum = pref;
                r.maximum = pref;
            } else {
                r = super.calculateMinorAxisRequirements(axis, r);
            }
            return r;
        }
    }

    class JaxeTableCell extends BoxView {

        public JaxeTableCell(final Element elem) {
            super(elem, Y_AXIS);
            setInsets((short) 3, (short) 3, (short) 3, (short) 3);
        }

        @Override
        protected SizeRequirements calculateMinorAxisRequirements(final int axis, final SizeRequirements r) {
            final SizeRequirements r2 = super.calculateMinorAxisRequirements(axis, r);
            if (r2.maximum == 2147483647)
                r2.maximum = 32767; // beaucoup mieux que 2147483647, va comprendre...
            return(r2);
        }

        @Override
        public void paint(final Graphics g, final Shape allocation) {
            final Rectangle alloc = (allocation instanceof Rectangle) ? (Rectangle) allocation
                    : allocation.getBounds();

            alloc.height = (int) getPreferredHeight();
            final Rectangle tempRect = new Rectangle();
            final int n = getViewCount();
            final int x = alloc.x + getLeftInset();
            final int y = alloc.y + getTopInset();

            for (int i = 0; i < n; i++) {
                tempRect.x = x + getOffset(X_AXIS, i);
                tempRect.y = y + getOffset(Y_AXIS, i);
                tempRect.width = getSpan(X_AXIS, i);
                tempRect.height = (int) getPreferredHeight();
                paintChild(g, tempRect, i);
            }
            g.setColor(Color.black);
            g.drawRect(alloc.x, alloc.y, alloc.width - 1, alloc.height - 1);
        }

        float _height = -1;

        public void setPreferredHeight(final float height) {
            _height = height;
        }

        public float getPreferredHeight() {
            return _height;
        }

        @Override
        public float getMinimumSpan(final int axis) {
            if (axis == Y_AXIS) {
                return _height;
            }
            return super.getMinimumSpan(axis);
        }

        @Override
        public float getPreferredSpan(final int axis) {
            if (axis == Y_AXIS) {
                return _height;
            }
            return super.getPreferredSpan(axis);
        }

        @Override
        public float getMaximumSpan(final int axis) {
            if (axis == Y_AXIS) {
                return _height;
            }
            return super.getMaximumSpan(axis);
        }

        public float getRealSpan(final int axis) {
            return super.getPreferredSpan(axis);
        }

    }
}

Generated by  Doxygen 1.6.0   Back to index