Package com.mxgraph.canvas

Source Code of com.mxgraph.canvas.mxSvgCanvas

/**
* Copyright (c) 2007, Gaudenz Alder
*/
package com.mxgraph.canvas;

import java.awt.Font;
import java.io.BufferedInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
import java.util.Hashtable;
import java.util.List;
import java.util.Map;

import org.w3c.dom.Document;
import org.w3c.dom.Element;

import com.mxgraph.util.mxBase64;
import com.mxgraph.util.mxConstants;
import com.mxgraph.util.mxPoint;
import com.mxgraph.util.mxRectangle;
import com.mxgraph.util.mxUtils;
import com.mxgraph.view.mxCellState;

/**
* An implementation of a canvas that uses SVG for painting. This canvas
* ignores the STYLE_LABEL_BACKGROUNDCOLOR and
* STYLE_LABEL_BORDERCOLOR styles due to limitations of SVG.
*/
public class mxSvgCanvas extends mxBasicCanvas
{

  /**
   * Holds the HTML document that represents the canvas.
   */
  protected Document document;

  /**
   * Used internally for looking up elements. Workaround for getElementById
   * not working.
   */
  private Map<String, Element> gradients = new Hashtable<String, Element>();

  /**
   * Used internally for looking up images.
   */
  private Map<String, Element> images = new Hashtable<String, Element>();

  /**
   *
   */
  protected Element defs = null;

  /**
   * Specifies if images should be embedded as base64 encoded strings.
   * Default is false.
   */
  protected boolean embedded = false;

  /**
   * Constructs a new SVG canvas for the specified dimension and scale.
   */
  public mxSvgCanvas()
  {
    this(null);
  }

  /**
   * Constructs a new SVG canvas for the specified bounds, scale and
   * background color.
   */
  public mxSvgCanvas(Document document)
  {
    setDocument(document);
  }

  /**
   *
   */
  public void appendSvgElement(Element node)
  {
    if (document != null)
    {
      document.getDocumentElement().appendChild(node);
    }
  }

  /**
   *
   */
  protected Element getDefsElement()
  {
    if (defs == null)
    {
      defs = document.createElement("defs");

      Element svgNode = document.getDocumentElement();

      if (svgNode.hasChildNodes())
      {
        svgNode.insertBefore(defs, svgNode.getFirstChild());
      }
      else
      {
        svgNode.appendChild(defs);
      }
    }

    return defs;
  }

  /**
   *
   */
  public Element getGradientElement(String start, String end, String direction)
  {
    String id = getGradientId(start, end, direction);
    Element gradient = gradients.get(id);

    if (gradient == null)
    {
      gradient = createGradientElement(start, end, direction);
      gradient.setAttribute("id", "g" + (gradients.size() + 1));
      getDefsElement().appendChild(gradient);
      gradients.put(id, gradient);
    }

    return gradient;
  }

  /**
   *
   */
  public Element getGlassGradientElement()
  {
    String id = "mx-glass-gradient";

    Element glassGradient = gradients.get(id);

    if (glassGradient == null)
    {
      glassGradient = document.createElement("linearGradient");
      glassGradient.setAttribute("x1", "0%");
      glassGradient.setAttribute("y1", "0%");
      glassGradient.setAttribute("x2", "0%");
      glassGradient.setAttribute("y2", "100%");

      Element stop1 = document.createElement("stop");
      stop1.setAttribute("offset", "0%");
      stop1.setAttribute("style", "stop-color:#ffffff;stop-opacity:0.9");
      glassGradient.appendChild(stop1);

      Element stop2 = document.createElement("stop");
      stop2.setAttribute("offset", "100%");
      stop2.setAttribute("style", "stop-color:#ffffff;stop-opacity:0.1");
      glassGradient.appendChild(stop2);

      glassGradient.setAttribute("id", "g" + (gradients.size() + 1));
      getDefsElement().appendChild(glassGradient);
      gradients.put(id, glassGradient);
    }

    return glassGradient;
  }

  /**
   *
   */
  protected Element createGradientElement(String start, String end,
      String direction)
  {
    Element gradient = document.createElement("linearGradient");
    gradient.setAttribute("x1", "0%");
    gradient.setAttribute("y1", "0%");
    gradient.setAttribute("x2", "0%");
    gradient.setAttribute("y2", "0%");

    if (direction == null || direction.equals(mxConstants.DIRECTION_SOUTH))
    {
      gradient.setAttribute("y2", "100%");
    }
    else if (direction.equals(mxConstants.DIRECTION_EAST))
    {
      gradient.setAttribute("x2", "100%");
    }
    else if (direction.equals(mxConstants.DIRECTION_NORTH))
    {
      gradient.setAttribute("y1", "100%");
    }
    else if (direction.equals(mxConstants.DIRECTION_WEST))
    {
      gradient.setAttribute("x1", "100%");
    }

    Element stop = document.createElement("stop");
    stop.setAttribute("offset", "0%");
    stop.setAttribute("style", "stop-color:" + start);
    gradient.appendChild(stop);

    stop = document.createElement("stop");
    stop.setAttribute("offset", "100%");
    stop.setAttribute("style", "stop-color:" + end);
    gradient.appendChild(stop);

    return gradient;
  }

  /**
   *
   */
  public String getGradientId(String start, String end, String direction)
  {
    // Removes illegal characters from gradient ID
    if (start.startsWith("#"))
    {
      start = start.substring(1);
    }

    if (end.startsWith("#"))
    {
      end = end.substring(1);
    }

    // Workaround for gradient IDs not working in Safari 5 / Chrome 6
    // if they contain uppercase characters
    start = start.toLowerCase();
    end = end.toLowerCase();

    String dir = null;

    if (direction == null || direction.equals(mxConstants.DIRECTION_SOUTH))
    {
      dir = "south";
    }
    else if (direction.equals(mxConstants.DIRECTION_EAST))
    {
      dir = "east";
    }
    else
    {
      String tmp = start;
      start = end;
      end = tmp;

      if (direction.equals(mxConstants.DIRECTION_NORTH))
      {
        dir = "south";
      }
      else if (direction.equals(mxConstants.DIRECTION_WEST))
      {
        dir = "east";
      }
    }

    return "mx-gradient-" + start + "-" + end + "-" + dir;
  }

  /**
   * Returns true if the given string ends with .png, .jpg or .gif.
   */
  protected boolean isImageResource(String src)
  {
    return src != null
        && (src.toLowerCase().endsWith(".png")
            || src.toLowerCase().endsWith(".jpg") || src
            .toLowerCase().endsWith(".gif"));
  }

  /**
   *
   */
  protected InputStream getResource(String src)
  {
    InputStream stream = null;

    try
    {
      stream = new BufferedInputStream(new URL(src).openStream());
    }
    catch (Exception e1)
    {
      stream = getClass().getResourceAsStream(src);
    }

    return stream;
  }

  /**
   * @throws IOException
   *
   */
  protected String createDataUrl(String src) throws IOException
  {
    String result = null;
    InputStream inputStream = isImageResource(src) ? getResource(src) : null;

    if (inputStream != null)
    {
      ByteArrayOutputStream outputStream = new ByteArrayOutputStream(1024);
      byte[] bytes = new byte[512];

      // Read bytes from the input stream in bytes.length-sized chunks and write
      // them into the output stream
      int readBytes;
      while ((readBytes = inputStream.read(bytes)) > 0)
      {
        outputStream.write(bytes, 0, readBytes);
      }

      // Convert the contents of the output stream into a Data URL
      String format = "png";
      int dot = src.lastIndexOf('.');

      if (dot > 0 && dot < src.length())
      {
        format = src.substring(dot + 1);
      }

      result = "data:image/"
          + format
          + ";base64,"
          + mxBase64
              .encodeToString(outputStream.toByteArray(), false);
    }

    return result;
  }

  /**
   *
   */
  protected Element getEmbeddedImageElement(String src)
  {
    Element img = images.get(src);

    if (img == null)
    {
      img = document.createElement("svg");
      img.setAttribute("width", "100%");
      img.setAttribute("height", "100%");

      Element inner = document.createElement("image");
      inner.setAttribute("width", "100%");
      inner.setAttribute("height", "100%");

      // Store before transforming to DataURL
      images.put(src, img);

      if (!src.startsWith("data:image/"))
      {
        try
        {
          String tmp = createDataUrl(src);
         
          if (tmp != null)
          {
            src = tmp;
          }
        }
        catch (IOException e)
        {
          // ignore
        }
      }

      inner.setAttributeNS(mxConstants.NS_XLINK, "xlink:href", src);
      img.appendChild(inner);
      img.setAttribute("id", "i" + (images.size()));
      getDefsElement().appendChild(img);
    }

    return img;
  }

  /**
   *
   */
  protected Element createImageElement(double x, double y, double w,
      double h, String src, boolean aspect, boolean flipH, boolean flipV,
      boolean embedded)
  {
    Element elem = null;

    if (embedded)
    {
      elem = document.createElement("use");

      Element img = getEmbeddedImageElement(src);
      elem.setAttributeNS(mxConstants.NS_XLINK, "xlink:href",
          "#" + img.getAttribute("id"));
    }
    else
    {
      elem = document.createElement("image");

      elem.setAttributeNS(mxConstants.NS_XLINK, "xlink:href", src);
    }

    elem.setAttribute("x", String.valueOf(x));
    elem.setAttribute("y", String.valueOf(y));
    elem.setAttribute("width", String.valueOf(w));
    elem.setAttribute("height", String.valueOf(h));

    // FIXME: SVG element must be used for reference to image with
    // aspect but for images with no aspect this does not work.
    if (aspect)
    {
      elem.setAttribute("preserveAspectRatio", "xMidYMid");
    }
    else
    {
      elem.setAttribute("preserveAspectRatio", "none");
    }

    double sx = 1;
    double sy = 1;
    double dx = 0;
    double dy = 0;

    if (flipH)
    {
      sx *= -1;
      dx = -w - 2 * x;
    }

    if (flipV)
    {
      sy *= -1;
      dy = -h - 2 * y;
    }

    String transform = "";

    if (sx != 1 || sy != 1)
    {
      transform += "scale(" + sx + " " + sy + ") ";
    }

    if (dx != 0 || dy != 0)
    {
      transform += "translate(" + dx + " " + dy + ") ";
    }

    if (transform.length() > 0)
    {
      elem.setAttribute("transform", transform);
    }

    return elem;
  }

  /**
   *
   */
  public void setDocument(Document document)
  {
    this.document = document;
  }

  /**
   * Returns a reference to the document that represents the canvas.
   *
   * @return Returns the document.
   */
  public Document getDocument()
  {
    return document;
  }

  /**
   *
   */
  public void setEmbedded(boolean value)
  {
    embedded = value;
  }

  /**
   *
   */
  public boolean isEmbedded()
  {
    return embedded;
  }

  /*
   * (non-Javadoc)
   * @see com.mxgraph.canvas.mxICanvas#drawCell()
   */
  public Object drawCell(mxCellState state)
  {
    Map<String, Object> style = state.getStyle();
    Element elem = null;

    if (state.getAbsolutePointCount() > 1)
    {
      List<mxPoint> pts = state.getAbsolutePoints();

      // Transpose all points by cloning into a new array
      pts = mxUtils.translatePoints(pts, translate.x, translate.y);

      // Draws the line
      elem = drawLine(pts, style);

      // Applies opacity
      float opacity = mxUtils.getFloat(style, mxConstants.STYLE_OPACITY,
          100);

      if (opacity != 100)
      {
        String value = String.valueOf(opacity / 100);
        elem.setAttribute("fill-opacity", value);
        elem.setAttribute("stroke-opacity", value);
      }
    }
    else
    {
      int x = (int) state.getX() + translate.x;
      int y = (int) state.getY() + translate.y;
      int w = (int) state.getWidth();
      int h = (int) state.getHeight();

      if (!mxUtils.getString(style, mxConstants.STYLE_SHAPE, "").equals(
          mxConstants.SHAPE_SWIMLANE))
      {
        elem = drawShape(x, y, w, h, style);
      }
      else
      {
        int start = (int) Math.round(mxUtils.getInt(style,
            mxConstants.STYLE_STARTSIZE,
            mxConstants.DEFAULT_STARTSIZE)
            * scale);

        // Removes some styles to draw the content area
        Map<String, Object> cloned = new Hashtable<String, Object>(
            style);
        cloned.remove(mxConstants.STYLE_FILLCOLOR);
        cloned.remove(mxConstants.STYLE_ROUNDED);

        if (mxUtils.isTrue(style, mxConstants.STYLE_HORIZONTAL, true))
        {
          elem = drawShape(x, y, w, start, style);
          drawShape(x, y + start, w, h - start, cloned);
        }
        else
        {
          elem = drawShape(x, y, start, h, style);
          drawShape(x + start, y, w - start, h, cloned);
        }
      }
    }

    return elem;
  }

  /*
   * (non-Javadoc)
   * @see com.mxgraph.canvas.mxICanvas#drawLabel()
   */
  public Object drawLabel(String label, mxCellState state, boolean html)
  {
    mxRectangle bounds = state.getLabelBounds();

    if (drawLabels && bounds != null)
    {
      int x = (int) bounds.getX() + translate.x;
      int y = (int) bounds.getY() + translate.y;
      int w = (int) bounds.getWidth();
      int h = (int) bounds.getHeight();
      Map<String, Object> style = state.getStyle();

      return drawText(label, x, y, w, h, style);
    }

    return null;
  }

  /**
   * Draws the shape specified with the STYLE_SHAPE key in the given style.
   *
   * @param x X-coordinate of the shape.
   * @param y Y-coordinate of the shape.
   * @param w Width of the shape.
   * @param h Height of the shape.
   * @param style Style of the the shape.
   */
  public Element drawShape(int x, int y, int w, int h,
      Map<String, Object> style)
  {
    String fillColor = mxUtils.getString(style,
        mxConstants.STYLE_FILLCOLOR, "none");
    String gradientColor = mxUtils.getString(style,
        mxConstants.STYLE_GRADIENTCOLOR, "none");
    String strokeColor = mxUtils.getString(style,
        mxConstants.STYLE_STROKECOLOR, "none");
    float strokeWidth = (float) (mxUtils.getFloat(style,
        mxConstants.STYLE_STROKEWIDTH, 1) * scale);
    float opacity = mxUtils.getFloat(style, mxConstants.STYLE_OPACITY, 100);

    // Draws the shape
    String shape = mxUtils.getString(style, mxConstants.STYLE_SHAPE, "");
    Element elem = null;
    Element background = null;

    if (shape.equals(mxConstants.SHAPE_IMAGE))
    {
      String img = getImageForStyle(style);

      if (img != null)
      {
        // Vertical and horizontal image flipping
        boolean flipH = mxUtils.isTrue(style,
            mxConstants.STYLE_IMAGE_FLIPH, false);
        boolean flipV = mxUtils.isTrue(style,
            mxConstants.STYLE_IMAGE_FLIPV, false);

        elem = createImageElement(x, y, w, h, img,
            PRESERVE_IMAGE_ASPECT, flipH, flipV, isEmbedded());
      }
    }
    else if (shape.equals(mxConstants.SHAPE_LINE))
    {
      String direction = mxUtils.getString(style,
          mxConstants.STYLE_DIRECTION, mxConstants.DIRECTION_EAST);
      String d = null;

      if (direction.equals(mxConstants.DIRECTION_EAST)
          || direction.equals(mxConstants.DIRECTION_WEST))
      {
        int mid = (y + h / 2);
        d = "M " + x + " " + mid + " L " + (x + w) + " " + mid;
      }
      else
      {
        int mid = (x + w / 2);
        d = "M " + mid + " " + y + " L " + mid + " " + (y + h);
      }

      elem = document.createElement("path");
      elem.setAttribute("d", d + " Z");
    }
    else if (shape.equals(mxConstants.SHAPE_ELLIPSE))
    {
      elem = document.createElement("ellipse");

      elem.setAttribute("cx", String.valueOf(x + w / 2));
      elem.setAttribute("cy", String.valueOf(y + h / 2));
      elem.setAttribute("rx", String.valueOf(w / 2));
      elem.setAttribute("ry", String.valueOf(h / 2));
    }
    else if (shape.equals(mxConstants.SHAPE_DOUBLE_ELLIPSE))
    {
      elem = document.createElement("g");
      background = document.createElement("ellipse");
      background.setAttribute("cx", String.valueOf(x + w / 2));
      background.setAttribute("cy", String.valueOf(y + h / 2));
      background.setAttribute("rx", String.valueOf(w / 2));
      background.setAttribute("ry", String.valueOf(h / 2));
      elem.appendChild(background);

      int inset = (int) ((3 + strokeWidth) * scale);

      Element foreground = document.createElement("ellipse");
      foreground.setAttribute("fill", "none");
      foreground.setAttribute("stroke", strokeColor);
      foreground
          .setAttribute("stroke-width", String.valueOf(strokeWidth));

      foreground.setAttribute("cx", String.valueOf(x + w / 2));
      foreground.setAttribute("cy", String.valueOf(y + h / 2));
      foreground.setAttribute("rx", String.valueOf(w / 2 - inset));
      foreground.setAttribute("ry", String.valueOf(h / 2 - inset));
      elem.appendChild(foreground);
    }
    else if (shape.equals(mxConstants.SHAPE_RHOMBUS))
    {
      elem = document.createElement("path");

      String d = "M " + (x + w / 2) + " " + y + " L " + (x + w) + " "
          + (y + h / 2) + " L " + (x + w / 2) + " " + (y + h) + " L "
          + x + " " + (y + h / 2);

      elem.setAttribute("d", d + " Z");
    }
    else if (shape.equals(mxConstants.SHAPE_TRIANGLE))
    {
      elem = document.createElement("path");
      String direction = mxUtils.getString(style,
          mxConstants.STYLE_DIRECTION, "");
      String d = null;

      if (direction.equals(mxConstants.DIRECTION_NORTH))
      {
        d = "M " + x + " " + (y + h) + " L " + (x + w / 2) + " " + y
            + " L " + (x + w) + " " + (y + h);
      }
      else if (direction.equals(mxConstants.DIRECTION_SOUTH))
      {
        d = "M " + x + " " + y + " L " + (x + w / 2) + " " + (y + h)
            + " L " + (x + w) + " " + y;
      }
      else if (direction.equals(mxConstants.DIRECTION_WEST))
      {
        d = "M " + (x + w) + " " + y + " L " + x + " " + (y + h / 2)
            + " L " + (x + w) + " " + (y + h);
      }
      else
      // east
      {
        d = "M " + x + " " + y + " L " + (x + w) + " " + (y + h / 2)
            + " L " + x + " " + (y + h);
      }

      elem.setAttribute("d", d + " Z");
    }
    else if (shape.equals(mxConstants.SHAPE_HEXAGON))
    {
      elem = document.createElement("path");
      String direction = mxUtils.getString(style,
          mxConstants.STYLE_DIRECTION, "");
      String d = null;

      if (direction.equals(mxConstants.DIRECTION_NORTH)
          || direction.equals(mxConstants.DIRECTION_SOUTH))
      {
        d = "M " + (x + 0.5 * w) + " " + y + " L " + (x + w) + " "
            + (y + 0.25 * h) + " L " + (x + w) + " "
            + (y + 0.75 * h) + " L " + (x + 0.5 * w) + " "
            + (y + h) + " L " + x + " " + (y + 0.75 * h) + " L "
            + x + " " + (y + 0.25 * h);
      }
      else
      {
        d = "M " + (x + 0.25 * w) + " " + y + " L " + (x + 0.75 * w)
            + " " + y + " L " + (x + w) + " " + (y + 0.5 * h)
            + " L " + (x + 0.75 * w) + " " + (y + h) + " L "
            + (x + 0.25 * w) + " " + (y + h) + " L " + x + " "
            + (y + 0.5 * h);
      }

      elem.setAttribute("d", d + " Z");
    }
    else if (shape.equals(mxConstants.SHAPE_CLOUD))
    {
      elem = document.createElement("path");

      String d = "M " + (x + 0.25 * w) + " " + (y + 0.25 * h) + " C "
          + (x + 0.05 * w) + " " + (y + 0.25 * h) + " " + x + " "
          + (y + 0.5 * h) + " " + (x + 0.16 * w) + " "
          + (y + 0.55 * h) + " C " + x + " " + (y + 0.66 * h) + " "
          + (x + 0.18 * w) + " " + (y + 0.9 * h) + " "
          + (x + 0.31 * w) + " " + (y + 0.8 * h) + " C "
          + (x + 0.4 * w) + " " + (y + h) + " " + (x + 0.7 * w) + " "
          + (y + h) + " " + (x + 0.8 * w) + " " + (y + 0.8 * h)
          + " C " + (x + w) + " " + (y + 0.8 * h) + " " + (x + w)
          + " " + (y + 0.6 * h) + " " + (x + 0.875 * w) + " "
          + (y + 0.5 * h) + " C " + (x + w) + " " + (y + 0.3 * h)
          + " " + (x + 0.8 * w) + " " + (y + 0.1 * h) + " "
          + (x + 0.625 * w) + " " + (y + 0.2 * h) + " C "
          + (x + 0.5 * w) + " " + (y + 0.05 * h) + " "
          + (x + 0.3 * w) + " " + (y + 0.05 * h) + " "
          + (x + 0.25 * w) + " " + (y + 0.25 * h);

      elem.setAttribute("d", d + " Z");
    }
    else if (shape.equals(mxConstants.SHAPE_ACTOR))
    {
      elem = document.createElement("path");
      double width3 = w / 3;

      String d = " M " + x + " " + (y + h) + " C " + x + " "
          + (y + 3 * h / 5) + " " + x + " " + (y + 2 * h / 5) + " "
          + (x + w / 2) + " " + (y + 2 * h / 5) + " C "
          + (x + w / 2 - width3) + " " + (y + 2 * h / 5) + " "
          + (x + w / 2 - width3) + " " + y + " " + (x + w / 2) + " "
          + y + " C " + (x + w / 2 + width3) + " " + y + " "
          + (x + w / 2 + width3) + " " + (y + 2 * h / 5) + " "
          + (x + w / 2) + " " + (y + 2 * h / 5) + " C " + (x + w)
          + " " + (y + 2 * h / 5) + " " + (x + w) + " "
          + (y + 3 * h / 5) + " " + (x + w) + " " + (y + h);

      elem.setAttribute("d", d + " Z");
    }
    else if (shape.equals(mxConstants.SHAPE_CYLINDER))
    {
      elem = document.createElement("g");
      background = document.createElement("path");

      double dy = Math.min(40, Math.floor(h / 5));
      String d = " M " + x + " " + (y + dy) + " C " + x + " "
          + (y - dy / 3) + " " + (x + w) + " " + (y - dy / 3) + " "
          + (x + w) + " " + (y + dy) + " L " + (x + w) + " "
          + (y + h - dy) + " C " + (x + w) + " " + (y + h + dy / 3)
          + " " + x + " " + (y + h + dy / 3) + " " + x + " "
          + (y + h - dy);
      background.setAttribute("d", d + " Z");
      elem.appendChild(background);

      Element foreground = document.createElement("path");
      d = "M " + x + " " + (y + dy) + " C " + x + " " + (y + 2 * dy)
          + " " + (x + w) + " " + (y + 2 * dy) + " " + (x + w) + " "
          + (y + dy);

      foreground.setAttribute("d", d);
      foreground.setAttribute("fill", "none");
      foreground.setAttribute("stroke", strokeColor);
      foreground
          .setAttribute("stroke-width", String.valueOf(strokeWidth));

      elem.appendChild(foreground);
    }
    else
    {
      background = document.createElement("rect");
      elem = background;

      elem.setAttribute("x", String.valueOf(x));
      elem.setAttribute("y", String.valueOf(y));
      elem.setAttribute("width", String.valueOf(w));
      elem.setAttribute("height", String.valueOf(h));

      if (mxUtils.isTrue(style, mxConstants.STYLE_ROUNDED, false))
      {
        String r = String.valueOf(Math.min(w
            * mxConstants.RECTANGLE_ROUNDING_FACTOR, h
            * mxConstants.RECTANGLE_ROUNDING_FACTOR));

        elem.setAttribute("rx", r);
        elem.setAttribute("ry", r);
      }

      // Paints the label image
      if (shape.equals(mxConstants.SHAPE_LABEL))
      {
        String img = getImageForStyle(style);

        if (img != null)
        {
          String imgAlign = mxUtils.getString(style,
              mxConstants.STYLE_IMAGE_ALIGN,
              mxConstants.ALIGN_LEFT);
          String imgValign = mxUtils.getString(style,
              mxConstants.STYLE_IMAGE_VERTICAL_ALIGN,
              mxConstants.ALIGN_MIDDLE);
          int imgWidth = (int) (mxUtils.getInt(style,
              mxConstants.STYLE_IMAGE_WIDTH,
              mxConstants.DEFAULT_IMAGESIZE) * scale);
          int imgHeight = (int) (mxUtils.getInt(style,
              mxConstants.STYLE_IMAGE_HEIGHT,
              mxConstants.DEFAULT_IMAGESIZE) * scale);
          int spacing = (int) (mxUtils.getInt(style,
              mxConstants.STYLE_SPACING, 2) * scale);

          mxRectangle imageBounds = new mxRectangle(x, y, w, h);

          if (imgAlign.equals(mxConstants.ALIGN_CENTER))
          {
            imageBounds.setX(imageBounds.getX()
                + (imageBounds.getWidth() - imgWidth) / 2);
          }
          else if (imgAlign.equals(mxConstants.ALIGN_RIGHT))
          {
            imageBounds.setX(imageBounds.getX()
                + imageBounds.getWidth() - imgWidth - spacing
                - 2);
          }
          else
          // LEFT
          {
            imageBounds.setX(imageBounds.getX() + spacing + 4);
          }

          if (imgValign.equals(mxConstants.ALIGN_TOP))
          {
            imageBounds.setY(imageBounds.getY() + spacing);
          }
          else if (imgValign.equals(mxConstants.ALIGN_BOTTOM))
          {
            imageBounds
                .setY(imageBounds.getY()
                    + imageBounds.getHeight() - imgHeight
                    - spacing);
          }
          else
          // MIDDLE
          {
            imageBounds.setY(imageBounds.getY()
                + (imageBounds.getHeight() - imgHeight) / 2);
          }

          imageBounds.setWidth(imgWidth);
          imageBounds.setHeight(imgHeight);

          elem = document.createElement("g");
          elem.appendChild(background);

          Element imageElement = createImageElement(
              imageBounds.getX(), imageBounds.getY(),
              imageBounds.getWidth(), imageBounds.getHeight(),
              img, false, false, false, isEmbedded());

          if (opacity != 100)
          {
            String value = String.valueOf(opacity / 100);
            imageElement.setAttribute("opacity", value);
          }

          elem.appendChild(imageElement);
        }

        // Paints the glass effect
        if (mxUtils.isTrue(style, mxConstants.STYLE_GLASS, false))
        {
          double size = 0.4;

          // TODO: Mask with rectangle or rounded rectangle of label
          // Creates glass overlay
          Element glassOverlay = document.createElement("path");

          // LATER: Not sure what the behaviour is for mutiple SVG elements in page.
          // Probably its possible that this points to an element in another SVG
          // node which when removed will result in an undefined background.
          glassOverlay.setAttribute("fill", "url(#"
              + getGlassGradientElement().getAttribute("id")
              + ")");

          String d = "m " + (x - strokeWidth) + ","
              + (y - strokeWidth) + " L " + (x - strokeWidth)
              + "," + (y + h * size) + " Q " + (x + w * 0.5)
              + "," + (y + h * 0.7) + " " + (x + w + strokeWidth)
              + "," + (y + h * size) + " L "
              + (x + w + strokeWidth) + "," + (y - strokeWidth)
              + " z";
          glassOverlay.setAttribute("stroke-width",
              String.valueOf(strokeWidth / 2));
          glassOverlay.setAttribute("d", d);
          elem.appendChild(glassOverlay);
        }
      }
    }

    double rotation = mxUtils.getDouble(style, mxConstants.STYLE_ROTATION);
    int cx = x + w / 2;
    int cy = y + h / 2;

    Element bg = background;

    if (bg == null)
    {
      bg = elem;
    }

    if (!bg.getNodeName().equalsIgnoreCase("use")
        && !bg.getNodeName().equalsIgnoreCase("image"))
    {
      if (!fillColor.equalsIgnoreCase("none")
          && !gradientColor.equalsIgnoreCase("none"))
      {
        String direction = mxUtils.getString(style,
            mxConstants.STYLE_GRADIENT_DIRECTION);
        Element gradient = getGradientElement(fillColor, gradientColor,
            direction);

        if (gradient != null)
        {
          bg.setAttribute("fill",
              "url(#" + gradient.getAttribute("id") + ")");
        }
      }
      else
      {
        bg.setAttribute("fill", fillColor);
      }

      bg.setAttribute("stroke", strokeColor);
      bg.setAttribute("stroke-width", String.valueOf(strokeWidth));

      // Adds the shadow element
      Element shadowElement = null;

      if (mxUtils.isTrue(style, mxConstants.STYLE_SHADOW, false)
          && !fillColor.equals("none"))
      {
        shadowElement = (Element) bg.cloneNode(true);

        shadowElement.setAttribute("transform",
            mxConstants.SVG_SHADOWTRANSFORM);
        shadowElement.setAttribute("fill", mxConstants.W3C_SHADOWCOLOR);
        shadowElement.setAttribute("stroke",
            mxConstants.W3C_SHADOWCOLOR);
        shadowElement.setAttribute("stroke-width",
            String.valueOf(strokeWidth));

        if (rotation != 0)
        {
          shadowElement.setAttribute("transform", "rotate("
              + rotation + "," + cx + "," + cy + ") "
              + mxConstants.SVG_SHADOWTRANSFORM);
        }

        if (opacity != 100)
        {
          String value = String.valueOf(opacity / 100);
          shadowElement.setAttribute("fill-opacity", value);
          shadowElement.setAttribute("stroke-opacity", value);
        }

        appendSvgElement(shadowElement);
      }
    }

    if (rotation != 0)
    {
      elem.setAttribute("transform", elem.getAttribute("transform")
          + " rotate(" + rotation + "," + cx + "," + cy + ")");

    }

    if (opacity != 100)
    {
      String value = String.valueOf(opacity / 100);
      elem.setAttribute("fill-opacity", value);
      elem.setAttribute("stroke-opacity", value);
    }

    if (mxUtils.isTrue(style, mxConstants.STYLE_DASHED))
    {
      String pattern = mxUtils.getString(style, mxConstants.STYLE_DASH_PATTERN, "3, 3");
      elem.setAttribute("stroke-dasharray", pattern);
    }

    appendSvgElement(elem);

    return elem;
  }

  /**
   * Draws the given lines as segments between all points of the given list
   * of mxPoints.
   *
   * @param pts List of points that define the line.
   * @param style Style to be used for painting the line.
   */
  public Element drawLine(List<mxPoint> pts, Map<String, Object> style)
  {
    Element group = document.createElement("g");
    Element path = document.createElement("path");

    boolean rounded = mxUtils.isTrue(style, mxConstants.STYLE_ROUNDED,
        false);
    String strokeColor = mxUtils.getString(style,
        mxConstants.STYLE_STROKECOLOR);
    float tmpStroke = (mxUtils.getFloat(style,
        mxConstants.STYLE_STROKEWIDTH, 1));
    float strokeWidth = (float) (tmpStroke * scale);

    if (strokeColor != null && strokeWidth > 0)
    {
      // Draws the start marker
      Object marker = style.get(mxConstants.STYLE_STARTARROW);

      mxPoint pt = pts.get(1);
      mxPoint p0 = pts.get(0);
      mxPoint offset = null;

      if (marker != null)
      {
        float size = (mxUtils.getFloat(style,
            mxConstants.STYLE_STARTSIZE,
            mxConstants.DEFAULT_MARKERSIZE));
        offset = drawMarker(group, marker, pt, p0, size, tmpStroke,
            strokeColor);
      }
      else
      {
        double dx = pt.getX() - p0.getX();
        double dy = pt.getY() - p0.getY();

        double dist = Math.max(1, Math.sqrt(dx * dx + dy * dy));
        double nx = dx * strokeWidth / dist;
        double ny = dy * strokeWidth / dist;

        offset = new mxPoint(nx / 2, ny / 2);
      }

      // Applies offset to the point
      if (offset != null)
      {
        p0 = (mxPoint) p0.clone();
        p0.setX(p0.getX() + offset.getX());
        p0.setY(p0.getY() + offset.getY());

        offset = null;
      }

      // Draws the end marker
      marker = style.get(mxConstants.STYLE_ENDARROW);

      pt = pts.get(pts.size() - 2);
      mxPoint pe = pts.get(pts.size() - 1);

      if (marker != null)
      {
        float size = (mxUtils.getFloat(style,
            mxConstants.STYLE_ENDSIZE,
            mxConstants.DEFAULT_MARKERSIZE));
        offset = drawMarker(group, marker, pt, pe, size, tmpStroke,
            strokeColor);
      }
      else
      {
        double dx = pt.getX() - p0.getX();
        double dy = pt.getY() - p0.getY();

        double dist = Math.max(1, Math.sqrt(dx * dx + dy * dy));
        double nx = dx * strokeWidth / dist;
        double ny = dy * strokeWidth / dist;

        offset = new mxPoint(nx / 2, ny / 2);
      }

      // Applies offset to the point
      if (offset != null)
      {
        pe = (mxPoint) pe.clone();
        pe.setX(pe.getX() + offset.getX());
        pe.setY(pe.getY() + offset.getY());

        offset = null;
      }

      // Draws the line segments
      double arcSize = mxConstants.LINE_ARCSIZE * scale;
      pt = p0;
      String d = "M " + pt.getX() + " " + pt.getY();

      for (int i = 1; i < pts.size() - 1; i++)
      {
        mxPoint tmp = pts.get(i);
        double dx = pt.getX() - tmp.getX();
        double dy = pt.getY() - tmp.getY();

        if ((rounded && i < pts.size() - 1) && (dx != 0 || dy != 0))
        {
          // Draws a line from the last point to the current
          // point with a spacing of size off the current point
          // into direction of the last point
          double dist = Math.sqrt(dx * dx + dy * dy);
          double nx1 = dx * Math.min(arcSize, dist / 2) / dist;
          double ny1 = dy * Math.min(arcSize, dist / 2) / dist;

          double x1 = tmp.getX() + nx1;
          double y1 = tmp.getY() + ny1;
          d += " L " + x1 + " " + y1;

          // Draws a curve from the last point to the current
          // point with a spacing of size off the current point
          // into direction of the next point
          mxPoint next = pts.get(i + 1);
          dx = next.getX() - tmp.getX();
          dy = next.getY() - tmp.getY();

          dist = Math.max(1, Math.sqrt(dx * dx + dy * dy));
          double nx2 = dx * Math.min(arcSize, dist / 2) / dist;
          double ny2 = dy * Math.min(arcSize, dist / 2) / dist;

          double x2 = tmp.getX() + nx2;
          double y2 = tmp.getY() + ny2;

          d += " Q " + tmp.getX() + " " + tmp.getY() + " " + x2 + " "
              + y2;
          tmp = new mxPoint(x2, y2);
        }
        else
        {
          d += " L " + tmp.getX() + " " + tmp.getY();
        }

        pt = tmp;
      }

      d += " L " + pe.getX() + " " + pe.getY();

      path.setAttribute("d", d);
      path.setAttribute("stroke", strokeColor);
      path.setAttribute("fill", "none");
      path.setAttribute("stroke-width", String.valueOf(strokeWidth));

      if (mxUtils.isTrue(style, mxConstants.STYLE_DASHED))
      {
        String pattern = mxUtils.getString(style, mxConstants.STYLE_DASH_PATTERN, "3, 3");
        path.setAttribute("stroke-dasharray", pattern);
      }

      group.appendChild(path);
      appendSvgElement(group);
    }

    return group;
  }

  /**
   * Draws the specified marker as a child path in the given parent.
   */
  public mxPoint drawMarker(Element parent, Object type, mxPoint p0,
      mxPoint pe, float size, float strokeWidth, String color)
  {
    mxPoint offset = null;

    // Computes the norm and the inverse norm
    double dx = pe.getX() - p0.getX();
    double dy = pe.getY() - p0.getY();

    double dist = Math.max(1, Math.sqrt(dx * dx + dy * dy));
    double absSize = size * scale;
    double nx = dx * absSize / dist;
    double ny = dy * absSize / dist;

    pe = (mxPoint) pe.clone();
    pe.setX(pe.getX() - nx * strokeWidth / (2 * size));
    pe.setY(pe.getY() - ny * strokeWidth / (2 * size));

    nx *= 0.5 + strokeWidth / 2;
    ny *= 0.5 + strokeWidth / 2;

    Element path = document.createElement("path");
    path.setAttribute("stroke-width", String.valueOf(strokeWidth * scale));
    path.setAttribute("stroke", color);
    path.setAttribute("fill", color);

    String d = null;

    if (type.equals(mxConstants.ARROW_CLASSIC)
        || type.equals(mxConstants.ARROW_BLOCK))
    {
      d = "M "
          + pe.getX()
          + " "
          + pe.getY()
          + " L "
          + (pe.getX() - nx - ny / 2)
          + " "
          + (pe.getY() - ny + nx / 2)
          + ((!type.equals(mxConstants.ARROW_CLASSIC)) ? "" : " L "
              + (pe.getX() - nx * 3 / 4) + " "
              + (pe.getY() - ny * 3 / 4)) + " L "
          + (pe.getX() + ny / 2 - nx) + " "
          + (pe.getY() - ny - nx / 2) + " z";
    }
    else if (type.equals(mxConstants.ARROW_OPEN))
    {
      nx *= 1.2;
      ny *= 1.2;

      d = "M " + (pe.getX() - nx - ny / 2) + " "
          + (pe.getY() - ny + nx / 2) + " L " + (pe.getX() - nx / 6)
          + " " + (pe.getY() - ny / 6) + " L "
          + (pe.getX() + ny / 2 - nx) + " "
          + (pe.getY() - ny - nx / 2) + " M " + pe.getX() + " "
          + pe.getY();
      path.setAttribute("fill", "none");
    }
    else if (type.equals(mxConstants.ARROW_OVAL))
    {
      nx *= 1.2;
      ny *= 1.2;
      absSize *= 1.2;

      d = "M " + (pe.getX() - ny / 2) + " " + (pe.getY() + nx / 2)
          + " a " + (absSize / 2) + " " + (absSize / 2) + " 0  1,1 "
          + (nx / 8) + " " + (ny / 8) + " z";
    }
    else if (type.equals(mxConstants.ARROW_DIAMOND))
    {
      d = "M " + (pe.getX() + nx / 2) + " " + (pe.getY() + ny / 2)
          + " L " + (pe.getX() - ny / 2) + " " + (pe.getY() + nx / 2)
          + " L " + (pe.getX() - nx / 2) + " " + (pe.getY() - ny / 2)
          + " L " + (pe.getX() + ny / 2) + " " + (pe.getY() - nx / 2)
          + " z";
    }

    if (d != null)
    {
      path.setAttribute("d", d);
      parent.appendChild(path);
    }

    return offset;
  }

  /**
   * Draws the specified text either using drawHtmlString or using drawString.
   *
   * @param text Text to be painted.
   * @param x X-coordinate of the text.
   * @param y Y-coordinate of the text.
   * @param w Width of the text.
   * @param h Height of the text.
   * @param style Style to be used for painting the text.
   */
  public Object drawText(String text, int x, int y, int w, int h,
      Map<String, Object> style)
  {
    Element elem = null;
    String fontColor = mxUtils.getString(style,
        mxConstants.STYLE_FONTCOLOR, "black");
    String fontFamily = mxUtils.getString(style,
        mxConstants.STYLE_FONTFAMILY, mxConstants.DEFAULT_FONTFAMILIES);
    int fontSize = (int) (mxUtils.getInt(style, mxConstants.STYLE_FONTSIZE,
        mxConstants.DEFAULT_FONTSIZE) * scale);

    if (text != null && text.length() > 0)
    {
      float strokeWidth = (float) (mxUtils.getFloat(style,
          mxConstants.STYLE_STROKEWIDTH, 1) * scale);

      // Applies the opacity
      float opacity = mxUtils.getFloat(style,
          mxConstants.STYLE_TEXT_OPACITY, 100);

      // Draws the label background and border
      String bg = mxUtils.getString(style,
          mxConstants.STYLE_LABEL_BACKGROUNDCOLOR);
      String border = mxUtils.getString(style,
          mxConstants.STYLE_LABEL_BORDERCOLOR);

      String transform = null;

      if (!mxUtils.isTrue(style, mxConstants.STYLE_HORIZONTAL, true))
      {
        double cx = x + w / 2;
        double cy = y + h / 2;
        transform = "rotate(270 " + cx + " " + cy + ")";
      }

      if (bg != null || border != null)
      {
        Element background = document.createElement("rect");

        background.setAttribute("x", String.valueOf(x));
        background.setAttribute("y", String.valueOf(y));
        background.setAttribute("width", String.valueOf(w));
        background.setAttribute("height", String.valueOf(h));

        if (bg != null)
        {
          background.setAttribute("fill", bg);
        }
        else
        {
          background.setAttribute("fill", "none");
        }

        if (border != null)
        {
          background.setAttribute("stroke", border);
        }
        else
        {
          background.setAttribute("stroke", "none");
        }

        background.setAttribute("stroke-width",
            String.valueOf(strokeWidth));

        if (opacity != 100)
        {
          String value = String.valueOf(opacity / 100);
          background.setAttribute("fill-opacity", value);
          background.setAttribute("stroke-opacity", value);
        }

        if (transform != null)
        {
          background.setAttribute("transform", transform);
        }

        appendSvgElement(background);
      }

      elem = document.createElement("text");

      int fontStyle = mxUtils.getInt(style, mxConstants.STYLE_FONTSTYLE);
      String weight = ((fontStyle & mxConstants.FONT_BOLD) == mxConstants.FONT_BOLD) ? "bold"
          : "normal";
      elem.setAttribute("font-weight", weight);
      String uline = ((fontStyle & mxConstants.FONT_UNDERLINE) == mxConstants.FONT_UNDERLINE) ? "underline"
          : "none";
      elem.setAttribute("font-decoration", uline);

      if ((fontStyle & mxConstants.FONT_ITALIC) == mxConstants.FONT_ITALIC)
      {
        elem.setAttribute("font-style", "italic");
      }

      elem.setAttribute("font-size", String.valueOf(fontSize));
      elem.setAttribute("font-family", fontFamily);
      elem.setAttribute("fill", fontColor);

      if (opacity != 100)
      {
        String value = String.valueOf(opacity / 100);
        elem.setAttribute("fill-opacity", value);
        elem.setAttribute("stroke-opacity", value);
      }

      int swingFontStyle = ((fontStyle & mxConstants.FONT_BOLD) == mxConstants.FONT_BOLD) ? Font.BOLD
          : Font.PLAIN;
      swingFontStyle += ((fontStyle & mxConstants.FONT_ITALIC) == mxConstants.FONT_ITALIC) ? Font.ITALIC
          : Font.PLAIN;

      String[] lines = text.split("\n");
      y += fontSize
          + (h - lines.length * (fontSize + mxConstants.LINESPACING))
          / 2 - 2;

      String align = mxUtils.getString(style, mxConstants.STYLE_ALIGN,
          mxConstants.ALIGN_CENTER);
      String anchor = "start";

      if (align.equals(mxConstants.ALIGN_RIGHT))
      {
        anchor = "end";
        x += w - mxConstants.LABEL_INSET * scale;
      }
      else if (align.equals(mxConstants.ALIGN_CENTER))
      {
        anchor = "middle";
        x += w / 2;
      }
      else
      {
        x += mxConstants.LABEL_INSET * scale;
      }

      elem.setAttribute("text-anchor", anchor);

      for (int i = 0; i < lines.length; i++)
      {
        Element tspan = document.createElement("tspan");

        tspan.setAttribute("x", String.valueOf(x));
        tspan.setAttribute("y", String.valueOf(y));

        tspan.appendChild(document.createTextNode(lines[i]));
        elem.appendChild(tspan);

        y += fontSize + mxConstants.LINESPACING;
      }

      if (transform != null)
      {
        elem.setAttribute("transform", transform);
      }

      appendSvgElement(elem);
    }

    return elem;
  }

}
TOP

Related Classes of com.mxgraph.canvas.mxSvgCanvas

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.