Package com.mxgraph.swing.handler

Source Code of com.mxgraph.swing.handler.mxSelectionCellsHandler

/**
* Copyright (c) 2008, Gaudenz Alder
*
* Known issue: Drag image size depends on the initial position and may sometimes
* not align with the grid when dragging. This is because the rounding of the width
* and height at the initial position may be different than that at the current
* position as the left and bottom side of the shape must align to the grid lines.
*/
package com.mxgraph.swing.handler;

import java.awt.Graphics;
import java.awt.Rectangle;
import java.awt.Stroke;
import java.awt.event.MouseEvent;
import java.awt.event.MouseListener;
import java.awt.event.MouseMotionListener;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.Map;

import javax.swing.SwingUtilities;

import com.mxgraph.swing.mxGraphComponent;
import com.mxgraph.util.mxEvent;
import com.mxgraph.util.mxEventObject;
import com.mxgraph.util.mxEventSource.mxIEventListener;
import com.mxgraph.view.mxCellState;
import com.mxgraph.view.mxGraph;

public class mxSelectionCellsHandler implements MouseListener,
    MouseMotionListener
{

  /**
   *
   */
  private static final long serialVersionUID = -882368002120921842L;

  /**
   * Defines the default value for maxHandlers. Default is 100.
   */
  public static int DEFAULT_MAX_HANDLERS = 100;

  /**
   * Reference to the enclosing graph component.
   */
  protected mxGraphComponent graphComponent;

  /**
   * Specifies if this handler is enabled.
   */
  protected boolean enabled = true;

  /**
   * Specifies if this handler is visible.
   */
  protected boolean visible = true;

  /**
   * Reference to the enclosing graph component.
   */
  protected Rectangle bounds = null;

  /**
   * Defines the maximum number of handlers to paint individually.
   * Default is DEFAULT_MAX_HANDLES.
   */
  protected int maxHandlers = DEFAULT_MAX_HANDLERS;

  /**
   * Maps from cells to handlers in the order of the selection cells.
   */
  protected transient LinkedHashMap<Object, mxCellHandler> handlers = new LinkedHashMap<Object, mxCellHandler>();

  /**
   *
   */
  protected transient mxIEventListener refreshHandler = new mxIEventListener()
  {
    public void invoke(Object source, mxEventObject evt)
    {
      if (isEnabled())
      {
        refresh();
      }
    }
  };

  /**
   *
   */
  protected transient PropertyChangeListener labelMoveHandler = new PropertyChangeListener()
  {

    /*
     * (non-Javadoc)
     * @see java.beans.PropertyChangeListener#propertyChange(java.beans.PropertyChangeEvent)
     */
    public void propertyChange(PropertyChangeEvent evt)
    {
      if (evt.getPropertyName().equals("vertexLabelsMovable")
          || evt.getPropertyName().equals("edgeLabelsMovable"))
      {
        refresh();
      }
    }

  };

  /**
   *
   * @param graphComponent
   */
  public mxSelectionCellsHandler(final mxGraphComponent graphComponent)
  {
    this.graphComponent = graphComponent;

    // Listens to all mouse events on the rendering control
    graphComponent.getGraphControl().addMouseListener(this);
    graphComponent.getGraphControl().addMouseMotionListener(this);

    // Installs the graph listeners and keeps them in sync
    addGraphListeners(graphComponent.getGraph());

    graphComponent.addPropertyChangeListener(new PropertyChangeListener()
    {
      public void propertyChange(PropertyChangeEvent evt)
      {
        if (evt.getPropertyName().equals("graph"))
        {
          removeGraphListeners((mxGraph) evt.getOldValue());
          addGraphListeners((mxGraph) evt.getNewValue());
        }
      }
    });

    // Installs the paint handler
    graphComponent.addListener(mxEvent.PAINT, new mxIEventListener()
    {
      public void invoke(Object sender, mxEventObject evt)
      {
        Graphics g = (Graphics) evt.getProperty("g");
        paintHandles(g);
      }
    });
  }

  /**
   * Installs the listeners to update the handles after any changes.
   */
  protected void addGraphListeners(mxGraph graph)
  {
    // LATER: Install change listener for graph model, selection model, view
    if (graph != null)
    {
      graph.getSelectionModel().addListener(mxEvent.CHANGE,
          refreshHandler);
      graph.getModel().addListener(mxEvent.CHANGE, refreshHandler);
      graph.getView().addListener(mxEvent.SCALE, refreshHandler);
      graph.getView().addListener(mxEvent.TRANSLATE, refreshHandler);
      graph.getView().addListener(mxEvent.SCALE_AND_TRANSLATE,
          refreshHandler);
      graph.getView().addListener(mxEvent.DOWN, refreshHandler);
      graph.getView().addListener(mxEvent.UP, refreshHandler);

      // Refreshes the handles if moveVertexLabels or moveEdgeLabels changes
      graph.addPropertyChangeListener(labelMoveHandler);
    }
  }

  /**
   * Removes all installed listeners.
   */
  protected void removeGraphListeners(mxGraph graph)
  {
    if (graph != null)
    {
      graph.getSelectionModel().removeListener(refreshHandler,
          mxEvent.CHANGE);
      graph.getModel().removeListener(refreshHandler, mxEvent.CHANGE);
      graph.getView().removeListener(refreshHandler, mxEvent.SCALE);
      graph.getView().removeListener(refreshHandler, mxEvent.TRANSLATE);
      graph.getView().removeListener(refreshHandler,
          mxEvent.SCALE_AND_TRANSLATE);
      graph.getView().removeListener(refreshHandler, mxEvent.DOWN);
      graph.getView().removeListener(refreshHandler, mxEvent.UP);

      // Refreshes the handles if moveVertexLabels or moveEdgeLabels changes
      graph.removePropertyChangeListener(labelMoveHandler);
    }
  }

  /**
   *
   */
  public mxGraphComponent getGraphComponent()
  {
    return graphComponent;
  }

  /**
   *
   */
  public boolean isEnabled()
  {
    return enabled;
  }

  /**
   *
   */
  public void setEnabled(boolean value)
  {
    enabled = value;
  }

  /**
   *
   */
  public boolean isVisible()
  {
    return visible;
  }

  /**
   *
   */
  public void setVisible(boolean value)
  {
    visible = value;
  }

  /**
   *
   */
  public int getMaxHandlers()
  {
    return maxHandlers;
  }

  /**
   *
   */
  public void setMaxHandlers(int value)
  {
    maxHandlers = value;
  }

  /**
   *
   */
  public mxCellHandler getHandler(Object cell)
  {
    return handlers.get(cell);
  }

  /**
   * Dispatches the mousepressed event to the subhandles. This is
   * called from the connection handler as subhandles have precedence
   * over the connection handler.
   */
  public void mousePressed(MouseEvent e)
  {
    if (graphComponent.isEnabled()
        && !graphComponent.isForceMarqueeEvent(e) && isEnabled())
    {
      Iterator<mxCellHandler> it = handlers.values().iterator();

      while (it.hasNext() && !e.isConsumed())
      {
        it.next().mousePressed(e);
      }
    }
  }

  /**
   *
   */
  public void mouseMoved(MouseEvent e)
  {
    if (graphComponent.isEnabled() && isEnabled())
    {
      Iterator<mxCellHandler> it = handlers.values().iterator();

      while (it.hasNext() && !e.isConsumed())
      {
        it.next().mouseMoved(e);
      }
    }
  }

  /**
   *
   */
  public void mouseDragged(MouseEvent e)
  {
    if (graphComponent.isEnabled() && isEnabled())
    {
      Iterator<mxCellHandler> it = handlers.values().iterator();

      while (it.hasNext() && !e.isConsumed())
      {
        it.next().mouseDragged(e);
      }
    }
  }

  /**
   *
   */
  public void mouseReleased(MouseEvent e)
  {
    if (graphComponent.isEnabled() && isEnabled())
    {
      Iterator<mxCellHandler> it = handlers.values().iterator();

      while (it.hasNext() && !e.isConsumed())
      {
        it.next().mouseReleased(e);
      }
    }

    reset();
  }

  /**
   * Redirects the tooltip handling of the JComponent to the graph
   * component, which in turn may use getHandleToolTipText in this class to
   * find a tooltip associated with a handle.
   */
  public String getToolTipText(MouseEvent e)
  {
    MouseEvent tmp = SwingUtilities.convertMouseEvent(e.getComponent(), e,
        graphComponent.getGraphControl());
    Iterator<mxCellHandler> it = handlers.values().iterator();
    String tip = null;

    while (it.hasNext() && tip == null)
    {
      tip = it.next().getToolTipText(tmp);
    }

    return tip;
  }

  /**
   *
   */
  public void reset()
  {
    Iterator<mxCellHandler> it = handlers.values().iterator();

    while (it.hasNext())
    {
      it.next().reset();
    }
  }

  /**
   *
   */
  public void refresh()
  {
    mxGraph graph = graphComponent.getGraph();

    // Creates a new map for the handlers and tries to
    // to reuse existing handlers from the old map
    LinkedHashMap<Object, mxCellHandler> oldHandlers = handlers;
    handlers = new LinkedHashMap<Object, mxCellHandler>();

    // Creates handles for all selection cells
    Object[] tmp = graph.getSelectionCells();
    boolean handlesVisible = tmp.length <= getMaxHandlers();
    Rectangle handleBounds = null;

    for (int i = 0; i < tmp.length; i++)
    {
      mxCellState state = graph.getView().getState(tmp[i]);

      if (state != null && state.getCell() != graph.getView().getCurrentRoot())
      {
        mxCellHandler handler = oldHandlers.remove(tmp[i]);

        if (handler != null)
        {
          handler.refresh(state);
        }
        else
        {
          handler = graphComponent.createHandler(state);
        }

        if (handler != null)
        {
          handler.setHandlesVisible(handlesVisible);
          handlers.put(tmp[i], handler);
          Rectangle bounds = handler.getBounds();
          Stroke stroke = handler.getSelectionStroke();

          if (stroke != null)
          {
            bounds = stroke.createStrokedShape(bounds).getBounds();
          }

          if (handleBounds == null)
          {
            handleBounds = bounds;
          }
          else
          {
            handleBounds.add(bounds);
          }
        }
      }
    }
   
    for (mxCellHandler handler: oldHandlers.values())
    {
      handler.destroy();
    }

    Rectangle dirty = bounds;

    if (handleBounds != null)
    {
      if (dirty != null)
      {
        dirty.add(handleBounds);
      }
      else
      {
        dirty = handleBounds;
      }
    }

    if (dirty != null)
    {
      graphComponent.getGraphControl().repaint(dirty);
    }

    // Stores current bounds for later use
    bounds = handleBounds;
  }

  /**
   *
   */
  public void paintHandles(Graphics g)
  {
    Iterator<mxCellHandler> it = handlers.values().iterator();

    while (it.hasNext())
    {
      it.next().paint(g);
    }
  }

  /*
   * (non-Javadoc)
   * @see java.awt.event.MouseListener#mouseClicked(java.awt.event.MouseEvent)
   */
  public void mouseClicked(MouseEvent arg0)
  {
    // empty
  }

  /*
   * (non-Javadoc)
   * @see java.awt.event.MouseListener#mouseEntered(java.awt.event.MouseEvent)
   */
  public void mouseEntered(MouseEvent arg0)
  {
    // empty
  }

  /*
   * (non-Javadoc)
   * @see java.awt.event.MouseListener#mouseExited(java.awt.event.MouseEvent)
   */
  public void mouseExited(MouseEvent arg0)
  {
    // empty
  }

}
TOP

Related Classes of com.mxgraph.swing.handler.mxSelectionCellsHandler

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.