Package org.pentaho.reporting.engine.classic.core.modules.parser.simple.readhandlers

Source Code of org.pentaho.reporting.engine.classic.core.modules.parser.simple.readhandlers.LineReadHandler

/*
* This program is free software; you can redistribute it and/or modify it under the
* terms of the GNU Lesser General Public License, version 2.1 as published by the Free Software
* Foundation.
*
* You should have received a copy of the GNU Lesser General Public License along with this
* program; if not, you can obtain a copy at http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html
* or from the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*
* 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 Lesser General Public License for more details.
*
* Copyright (c) 2001 - 2009 Object Refinery Ltd, Pentaho Corporation and Contributors..  All rights reserved.
*/

package org.pentaho.reporting.engine.classic.core.modules.parser.simple.readhandlers;

import java.awt.Color;
import java.awt.Shape;
import java.awt.Stroke;
import java.awt.geom.Line2D;
import java.awt.geom.Rectangle2D;
import java.util.ArrayList;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.pentaho.reporting.engine.classic.core.Element;
import org.pentaho.reporting.engine.classic.core.elementfactory.ContentElementFactory;
import org.pentaho.reporting.engine.classic.core.elementfactory.HorizontalLineElementFactory;
import org.pentaho.reporting.engine.classic.core.elementfactory.VerticalLineElementFactory;
import org.pentaho.reporting.engine.classic.core.modules.parser.base.PropertyAttributes;
import org.pentaho.reporting.engine.classic.core.modules.parser.base.ReportElementReadHandler;
import org.pentaho.reporting.engine.classic.core.modules.parser.base.ReportParserUtil;
import org.pentaho.reporting.engine.classic.core.modules.parser.base.common.AbstractPropertyXmlReadHandler;
import org.pentaho.reporting.engine.classic.core.modules.parser.base.common.StyleExpressionHandler;
import org.pentaho.reporting.engine.classic.core.util.ShapeTransform;
import org.pentaho.reporting.libraries.xmlns.common.ParserUtil;
import org.pentaho.reporting.libraries.xmlns.parser.ParseException;
import org.pentaho.reporting.libraries.xmlns.parser.XmlReadHandler;
import org.xml.sax.SAXException;

public class LineReadHandler extends AbstractPropertyXmlReadHandler
    implements ReportElementReadHandler
{
  private static final Log logger = LogFactory.getLog(LineReadHandler.class);
  private Element element;
  private static final String NAME_ATT = "name";
  private static final String COLOR_ATT = "color";
  private ArrayList styleExpressionHandlers;

  public LineReadHandler()
  {
    styleExpressionHandlers = new ArrayList();
  }

  /**
   * Starts parsing.
   *
   * @param atts the attributes.
   * @throws org.xml.sax.SAXException if there is a parsing error.
   */
  protected void startParsing(final PropertyAttributes atts)
      throws SAXException
  {
    final float x1 = ReportParserUtil.parseRelativeFloat
        (atts.getValue(getUri(), "x1"), "Element x1 not specified", getLocator());
    final float y1 = ReportParserUtil.parseRelativeFloat
        (atts.getValue(getUri(), "y1"), "Element y1 not specified", getLocator());
    final float x2 = ReportParserUtil.parseRelativeFloat
        (atts.getValue(getUri(), "x2"), "Element x2 not specified", getLocator());
    final float y2 = ReportParserUtil.parseRelativeFloat
        (atts.getValue(getUri(), "y2"), "Element y2 not specified", getLocator());

    if (x1 == x2 && y1 == y2)
    {
      LineReadHandler.logger.warn("creating a horizontal line with 'x1 == x2 && y1 == y2' is deprecated. " +
          "Use relative coordinates instead.");
      final Stroke stroke = readStroke(atts);
      final String name = atts.getValue(getUri(), LineReadHandler.NAME_ATT);
      final Color c = ReportParserUtil.parseColor(atts.getValue(getUri(), LineReadHandler.COLOR_ATT));

      final HorizontalLineElementFactory elementFactory = new HorizontalLineElementFactory();
      elementFactory.setName(name);
      elementFactory.setColor(c);
      elementFactory.setStroke(stroke);
      elementFactory.setX(new Float(0));
      elementFactory.setY(new Float(y2));
      elementFactory.setMinimumWidth(new Float(-100));
      elementFactory.setMinimumHeight(new Float(0));
      elementFactory.setShouldDraw(Boolean.TRUE);
      element = elementFactory.createElement();
      return;
    }


    final boolean relativeSizes = (x1 < 0) || (y1 < 0) || (x2 < 0) || (y2 < 0);


    if (relativeSizes == false)
    {
      createSimpleLine(atts, x1, y1, x2, y2);
    }
    else
    {
      createRelativeLine(atts, x1, y1, x2, y2);
    }
  }

  private void createRelativeLine(final PropertyAttributes atts,
                                  final float x1,
                                  final float y1,
                                  final float x2,
                                  final float y2)
      throws SAXException
  {
    final Stroke stroke = readStroke(atts);
    final String name = atts.getValue(getUri(), LineReadHandler.NAME_ATT);
    final Color c = ReportParserUtil.parseColor
        (atts.getValue(getUri(), LineReadHandler.COLOR_ATT), null);

    final String widthValue = atts.getValue(getUri(), "width");
    final float width;
    if (widthValue != null)
    {
      width = ReportParserUtil.parseRelativeFloat
          (widthValue, "Width is invalid", getLocator());
    }
    else
    {
      width = computeDimension(name, x1, x2);
    }

    final String heightValue = atts.getValue(getUri(), "height");
    final float height;
    if (heightValue != null)
    {
      height = ReportParserUtil.parseRelativeFloat
          (heightValue, "Height is invalid", getLocator());
    }
    else
    {
      height = computeDimension(name, y1, y2);
    }

    // create the bounds as specified by the user
    final Rectangle2D bounds =
        new Rectangle2D.Float(computePosition(name, x1, x2),
            computePosition(name, y1, y2), width, height);

    if (x1 == x2)
    {
      // assume that we have a vertical line
      final VerticalLineElementFactory elementFactory = new VerticalLineElementFactory();
      elementFactory.setName(name);
      elementFactory.setColor(c);
      elementFactory.setStroke(stroke);
      elementFactory.setX(new Float(bounds.getX()));
      elementFactory.setY(new Float(bounds.getY()));
      elementFactory.setMinimumWidth(new Float(bounds.getWidth()));
      elementFactory.setMinimumHeight(new Float(bounds.getHeight()));
      elementFactory.setScale(Boolean.TRUE);
      elementFactory.setKeepAspectRatio(Boolean.FALSE);
      elementFactory.setShouldDraw(Boolean.TRUE);
      element = elementFactory.createElement();
    }
    else if (y1 == y2)
    {
      // assume that we have a horizontal line
      final HorizontalLineElementFactory elementFactory = new HorizontalLineElementFactory();
      elementFactory.setName(name);
      elementFactory.setColor(c);
      elementFactory.setStroke(stroke);
      elementFactory.setX(new Float(bounds.getX()));
      elementFactory.setY(new Float(bounds.getY()));
      elementFactory.setMinimumWidth(new Float(bounds.getWidth()));
      elementFactory.setMinimumHeight(new Float(bounds.getHeight()));
      elementFactory.setScale(Boolean.TRUE);
      elementFactory.setKeepAspectRatio(Boolean.FALSE);
      elementFactory.setShouldDraw(Boolean.TRUE);
      element = elementFactory.createElement();
    }
    else
    {
      // here comes the magic - we transform the line into the absolute space;
      // this should preserve the general appearance. Heck, and if not, then
      // it is part of the users responsibility to resolve that. Magic does not
      // solve all problems, you know.
      final Line2D line = new Line2D.Float
          (Math.abs(x1), Math.abs(y1), Math.abs(x2), Math.abs(y2));
      final Rectangle2D shapeBounds = line.getBounds2D();
      final Shape transformedShape =
          ShapeTransform.translateShape(line, -shapeBounds.getX(), -shapeBounds.getY());
      // and use that shape with the user's bounds to create the element.
      final ContentElementFactory elementFactory = new ContentElementFactory();
      elementFactory.setName(name);
      elementFactory.setColor(c);
      elementFactory.setStroke(stroke);
      elementFactory.setX(new Float(bounds.getX()));
      elementFactory.setY(new Float(bounds.getY()));
      elementFactory.setMinimumWidth(new Float(bounds.getWidth()));
      elementFactory.setMinimumHeight(new Float(bounds.getHeight()));
      elementFactory.setContent(transformedShape);
      elementFactory.setScale(Boolean.TRUE);
      elementFactory.setKeepAspectRatio(Boolean.FALSE);
      elementFactory.setShouldDraw(Boolean.TRUE);
      element = elementFactory.createElement();
    }

  }

  private float computePosition(final String name, final float x1, final float x2)
  {
    final boolean x1Relative;
    final boolean x2Relative;

    if (x1 == 0)
    {
      x1Relative = x2 < 0;
      x2Relative = x1Relative;
    }
    else if (x2 == 0)
    {
      x1Relative = x1 < 0;
      x2Relative = x1Relative;
    }
    else
    {
      x1Relative = x1 < 0;
      x2Relative = x2 < 0;
    }

    if (x1Relative && x2Relative)
    {
      // relative sizes are given as negative numbers.
      return Math.max(x2, x1);
    }
    else if (x1Relative == false && x2Relative == false)
    {
      // absolute sizes are given as positive numbers.
      return Math.min(x2, x1);
    }
    else
    {
      LineReadHandler.logger.warn("Mixing relative and absolute positions in '" + name + "'. " +
          "The definition is ambigous. (" + x1 + ", " + x2 + ')');

      // return the absolute element as computed position.
      if (x1Relative)
      {
        return x2;
      }
      else
      {
        return x1;
      }
    }
  }

  private float computeDimension(final String name, final float x1, final float x2)
  {
    final boolean x1Relative;
    final boolean x2Relative;

    if (x1 == 0)
    {
      x1Relative = x2 < 0;
      x2Relative = x1Relative;
    }
    else if (x2 == 0)
    {
      x1Relative = x1 < 0;
      x2Relative = x1Relative;
    }
    else
    {
      x1Relative = x1 < 0;
      x2Relative = x2 < 0;
    }

    if (x1Relative && x2Relative)
    {
      // relative sizes are given as negative numbers.
      return Math.min(x2, x1) - Math.max(x2, x1);
    }
    else if (x1Relative == false && x2Relative == false)
    {
      // absolute sizes are given as positive numbers.
      return Math.max(x2, x1) - Math.min(x2, x1);
    }
    else
    {
      // use the relative element as width ..
      LineReadHandler.logger.warn("Mixing relative and absolute sizes in '" + name + "'. " +
          "The definition is ambigous. (" + x1 + ", " + x2 + ')');
      if (x1Relative)
      {
        return x1;
      }
      else
      {
        return x2;
      }
    }
  }

  private void createSimpleLine(final PropertyAttributes atts,
                                final float x1,
                                final float y1,
                                final float x2,
                                final float y2)
      throws SAXException
  {
    final Stroke stroke = readStroke(atts);
    final String name = atts.getValue(getUri(), LineReadHandler.NAME_ATT);
    final Color c = ReportParserUtil.parseColor(atts.getValue(getUri(), LineReadHandler.COLOR_ATT), null);
    final String widthValue = atts.getValue(getUri(), "width");
    final float width;
    if (widthValue != null)
    {
      width = ReportParserUtil.parseRelativeFloat
          (widthValue, "Width is invalid", getLocator());
    }
    else
    {
      width = Math.max(x2, x1) - Math.min(x2, x1);
    }

    final String heightValue = atts.getValue(getUri(), "height");
    final float height;
    if (heightValue != null)
    {
      height = ReportParserUtil.parseRelativeFloat
          (heightValue, "Height is invalid", getLocator());
    }
    else
    {
      height = Math.max(y2, y1) - Math.min(y2, y1);
    }

    // create the bounds as specified by the user
    final Rectangle2D bounds =
        new Rectangle2D.Float(Math.min(x1, x2), Math.min(y1, y2), width, height);


    if (x1 == x2)
    {
      // assume that we have a vertical line
      final VerticalLineElementFactory elementFactory = new VerticalLineElementFactory();
      elementFactory.setName(name);
      elementFactory.setColor(c);
      elementFactory.setStroke(stroke);
      elementFactory.setX(new Float(bounds.getX()));
      elementFactory.setY(new Float(bounds.getY()));
      elementFactory.setMinimumWidth(new Float(bounds.getWidth()));
      elementFactory.setMinimumHeight(new Float(bounds.getHeight()));
      elementFactory.setScale(Boolean.TRUE);
      elementFactory.setKeepAspectRatio(Boolean.FALSE);
      elementFactory.setShouldDraw(Boolean.TRUE);
      element = elementFactory.createElement();
    }
    else if (y1 == y2)
    {
      // assume that we have a horizontal line
      final HorizontalLineElementFactory elementFactory = new HorizontalLineElementFactory();
      elementFactory.setName(name);
      elementFactory.setColor(c);
      elementFactory.setStroke(stroke);
      elementFactory.setX(new Float(bounds.getX()));
      elementFactory.setY(new Float(bounds.getY()));
      elementFactory.setMinimumWidth(new Float(bounds.getWidth()));
      elementFactory.setMinimumHeight(new Float(bounds.getHeight()));
      elementFactory.setScale(Boolean.TRUE);
      elementFactory.setKeepAspectRatio(Boolean.FALSE);
      elementFactory.setShouldDraw(Boolean.TRUE);
      element = elementFactory.createElement();
    }
    else
    {
      // here comes the magic - we transform the line into the absolute space;
      // this should preserve the general appearance. Heck, and if not, then
      // it is part of the users reponsibility to resolve that. Magic does not
      // solve all problems, you know.
      final Line2D line = new Line2D.Float
          (Math.abs(x1), Math.abs(y1), Math.abs(x2), Math.abs(y2));
      final Rectangle2D shapeBounds = line.getBounds2D();
      final Shape transformedShape =
          ShapeTransform.translateShape(line, -shapeBounds.getX(), -shapeBounds.getY());
      // and use that shape with the user's bounds to create the element.
      final ContentElementFactory elementFactory = new ContentElementFactory();
      elementFactory.setName(name);
      elementFactory.setColor(c);
      elementFactory.setStroke(stroke);
      elementFactory.setX(new Float(shapeBounds.getX()));
      elementFactory.setY(new Float(shapeBounds.getY()));
      elementFactory.setMinimumWidth(new Float(shapeBounds.getWidth()));
      elementFactory.setMinimumHeight(new Float(shapeBounds.getHeight()));
      elementFactory.setContent(transformedShape);
      elementFactory.setScale(Boolean.TRUE);
      elementFactory.setKeepAspectRatio(Boolean.FALSE);
      elementFactory.setShouldDraw(Boolean.TRUE);
      element = elementFactory.createElement();
    }
  }


  private Stroke readStroke(final PropertyAttributes atts)
      throws ParseException
  {
    final String strokeStyle = atts.getValue(getUri(), "stroke-style");
    final String weightAttr = atts.getValue(getUri(), "weight");
    float weight = 1;
    if (weightAttr != null)
    {
      weight = ParserUtil.parseFloat
          (weightAttr, "Weight is given, but no number.", getLocator());
    }

    // "dashed | solid | dotted | dot-dot-dash | dot-dash"
    return ReportParserUtil.parseStroke(strokeStyle, weight);
  }

  protected XmlReadHandler getHandlerForChild(final String uri,
                                              final String tagName,
                                              final PropertyAttributes attrs)
      throws SAXException
  {
    if (isSameNamespace(uri) == false)
    {
      return null;
    }

    if ("style-expression".equals(tagName))
    {
      final StyleExpressionHandler handler = new StyleExpressionHandler();
      styleExpressionHandlers.add(handler);
      return handler;
    }
    return null;
  }

  /**
   * Done parsing.
   *
   * @throws org.xml.sax.SAXException if there is a parsing error.
   */
  protected void doneParsing() throws SAXException
  {
    for (int i = 0; i < styleExpressionHandlers.size(); i++)
    {
      final StyleExpressionHandler handler =
          (StyleExpressionHandler) styleExpressionHandlers.get(i);
      if (handler.getKey() != null)
      {
        element.setStyleExpression(handler.getKey(), handler.getExpression());
      }
    }
  }

  /**
   * Returns the object for this element or null, if this element does not create an object.
   *
   * @return the object.
   */
  public Object getObject()
  {
    return element;
  }

}
TOP

Related Classes of org.pentaho.reporting.engine.classic.core.modules.parser.simple.readhandlers.LineReadHandler

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.