/******************************************************************************* * Copyright (c) 2006-2012 * Software Technology Group, Dresden University of Technology * DevBoost GmbH, Berlin, Amtsgericht Charlottenburg, HRB 140026 * * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * Software Technology Group - TU Dresden, Germany; * DevBoost GmbH - Berlin, Germany * - initial API and implementation ******************************************************************************/ /* * @(#)DefaultSelectAreaTracker.java 4.0 2008-05-17 * * Copyright (c) 1996-2008 by the original authors of JHotDraw * and all its contributors. * All rights reserved. * * The copyright of this software is owned by the authors and * contributors of the JHotDraw project ("the copyright holders"). * You may not use, copy or modify this software, except in * accordance with the license agreement you entered into with * the copyright holders. For details see accompanying license terms. */ package org.jhotdraw.draw; import java.awt.event.*; import java.awt.*; import java.awt.geom.Point2D; import java.util.*; /** * <code>DefaultSelectAreaTracker</code> implements interactions with the background * area of a <code>Drawing</code>. * <p> * The <code>DefaultSelectAreaTracker</code> handles one of the three states of the * <code>SelectionTool</code>. It comes into action, when the user presses * the mouse button over the background of a <code>Drawing</code>. * <p> * Design pattern:<br> * Name: Chain of Responsibility.<br> * Role: Handler.<br> * Partners: {@link SelectionTool} as Handler, {@link DragTracker} as Handler, * {@link HandleTracker} as Handler. * <p> * Design pattern:<br> * Name: State.<br> * Role: State.<br> * Partners: {@link SelectionTool} as Context, {@link DragTracker} as * State, {@link HandleTracker} as State. * * @see SelectionTool * * @author Werner Randelshofer * @version 4.0 2008-05-17 Displays handles with index -1, if the mouse * hovers over a figure. * <br>3.0 2006-02-15 Updated to handle multiple views. * <br>1.0 2003-12-01 Derived from JHotDraw 5.4b1. */ public class DefaultSelectAreaTracker extends AbstractTool implements SelectAreaTracker { /** * The bounds of the rubberband. */ private Rectangle rubberband = new Rectangle(); /** * Rubberband color. When this is null, the tracker does not * draw the rubberband. */ private Color rubberbandColor = Color.BLACK; /** * Rubberband stroke. */ private Stroke rubberbandStroke = new BasicStroke(); /** * The hover handles, are the handles of the figure over which the * mouse pointer is currently hovering. */ private LinkedList<Handle> hoverHandles = new LinkedList<Handle>(); /** * The hover Figure is the figure, over which the mouse is currently * hovering. */ private Figure hoverFigure = null; /** Creates a new instance. */ public DefaultSelectAreaTracker() { } @Override public void mousePressed(MouseEvent evt) { super.mousePressed(evt); clearRubberBand(); } @Override public void mouseReleased(MouseEvent evt) { selectGroup(evt.isShiftDown()); clearRubberBand(); } public void mouseDragged(MouseEvent evt) { Rectangle invalidatedArea = (Rectangle) rubberband.clone(); rubberband.setBounds( Math.min(anchor.x, evt.getX()), Math.min(anchor.y, evt.getY()), Math.abs(anchor.x - evt.getX()), Math.abs(anchor.y - evt.getY())); if (invalidatedArea.isEmpty()) { invalidatedArea = (Rectangle) rubberband.clone(); } else { invalidatedArea = invalidatedArea.union(rubberband); } fireAreaInvalidated(invalidatedArea); } @Override public void mouseMoved(MouseEvent evt) { clearRubberBand(); Point point = evt.getPoint(); DrawingView view = editor.findView((Container) evt.getSource()); updateCursor(view, point); if (view == null || editor.getActiveView() != view) { clearHoverHandles(); } else { // Search first, if one of the selected figures contains // the current mouse location, and is selectable. // Only then search for other // figures. This search sequence is consistent with the // search sequence of the SelectionTool. Figure figure = null; Point2D.Double p = view.viewToDrawing(point); for (Figure f : view.getSelectedFigures()) { if (f.contains(p)) { figure = f; } } if (figure == null) { figure = view.findFigure(point); while (figure != null && !figure.isSelectable()) { figure = view.getDrawing().findFigureBehind(p, figure); } } updateHoverHandles(view, figure); } } @Override public void mouseExited(MouseEvent evt) { DrawingView view = editor.findView((Container) evt.getSource()); updateHoverHandles(view, null); } private void clearRubberBand() { if (!rubberband.isEmpty()) { fireAreaInvalidated(rubberband); rubberband.width = -1; } } @Override public void draw(Graphics2D g) { g.setStroke(rubberbandStroke); g.setColor(rubberbandColor); g.drawRect(rubberband.x, rubberband.y, rubberband.width - 1, rubberband.height - 1); if (hoverHandles.size() > 0 && !getView().isFigureSelected(hoverFigure)) { for (Handle h : hoverHandles) { h.draw(g); } } } private void selectGroup(boolean toggle) { Collection<Figure> figures = getView().findFiguresWithin(rubberband); for (Figure f : figures) { if (f.isSelectable()) { getView().addToSelection(f); } } } protected void clearHoverHandles() { updateHoverHandles(null, null); } protected void updateHoverHandles(DrawingView view, Figure f) { if (f != hoverFigure) { Rectangle r = null; if (hoverFigure != null) { for (Handle h : hoverHandles) { if (r == null) { r = h.getDrawingArea(); } else { r.add(h.getDrawingArea()); } h.setView(null); h.dispose(); } hoverHandles.clear(); } hoverFigure = f; if (hoverFigure != null && f.isSelectable()) { hoverHandles.addAll(hoverFigure.createHandles(-1)); for (Handle h : hoverHandles) { h.setView(view); if (r == null) { r = h.getDrawingArea(); } else { r.add(h.getDrawingArea()); } } } if (r != null) { r.grow(1, 1); fireAreaInvalidated(r); } } } @Override public void activate(DrawingEditor editor) { super.activate(editor); clearHoverHandles(); } @Override public void deactivate(DrawingEditor editor) { super.deactivate(editor); clearHoverHandles(); } }