/******************************************************************************* * 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 ******************************************************************************/ /* * @(#)QuadTreeCompositeFigure.java 1.0 July 17, 2007 * * Copyright (c) 2007 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 org.jhotdraw.geom.Dimension2DDouble; import org.jhotdraw.geom.QuadTree; import org.jhotdraw.util.ReversedList; import java.awt.*; import java.awt.geom.*; import javax.swing.*; import javax.swing.event.*; import org.jhotdraw.util.*; import java.util.*; /** * QuadTreeCompositeFigure. * * * @author Werner Randelshofer * @version 1.0 July 17, 2007 Created. */ public abstract class QuadTreeCompositeFigure extends AbstractCompositeFigure { private ArrayList<Figure> children = new ArrayList<Figure>(); private QuadTree<Figure> quadTree = new QuadTree<Figure>(); private boolean needsSorting = false; private FigureHandler figureHandler; private Dimension2DDouble canvasSize; /** Creates a new instance. */ public QuadTreeCompositeFigure() { figureHandler = createFigureHandler(); } protected FigureHandler createFigureHandler() { return new FigureHandler(); } public int indexOf(Figure figure) { return children.indexOf(figure); } public void basicAdd(int index, Figure figure) { children.add(index, figure); quadTree.add(figure, figure.getDrawingArea()); figure.addFigureListener(figureHandler); needsSorting = true; } public Figure basicRemoveChild(int index) { Figure figure = children.get(index); children.remove(index); quadTree.remove(figure); figure.removeFigureListener(figureHandler); needsSorting = true; return figure; } public void draw(Graphics2D g) { Rectangle2D clipBounds = g.getClipBounds(); if (clipBounds != null) { Collection<Figure> c = quadTree.findIntersects(clipBounds); Collection<Figure> toDraw = sort(c); draw(g, toDraw); } else { draw(g, children); } } /** * Implementation note: Sorting can not be done for orphaned children. */ public java.util.List<Figure> sort(Collection<Figure> c) { ensureSorted(); ArrayList<Figure> sorted = new ArrayList<Figure>(c.size()); for (Figure f : children) { if (c.contains(f)) { sorted.add(f); } } return sorted; } public void draw(Graphics2D g, Collection<Figure> c) { for (Figure f : c) { f.draw(g); } } public java.util.List<Figure> getFigures(Rectangle2D.Double bounds) { return new LinkedList<Figure>(quadTree.findInside(bounds)); } @Override public java.util.List<Figure> getChildren() { return Collections.unmodifiableList(children); } @Override public Figure findFigureInside(Point2D.Double p) { Collection<Figure> c = quadTree.findContains(p); for (Figure f : getFiguresFrontToBack()) { if (c.contains(f) && f.contains(p)){ return f.findFigureInside(p); } } return null; } /** * Returns an iterator to iterate in * Z-order front to back over the children. */ public java.util.List<Figure> getFiguresFrontToBack() { ensureSorted(); return new ReversedList<Figure>(children); } public Figure findFigure(Point2D.Double p) { Collection<Figure> c = quadTree.findContains(p); switch (c.size()) { case 0 : return null; case 1: { Figure f = c.iterator().next(); return (f.contains(p)) ? f : null; } default : { for (Figure f : getFiguresFrontToBack()) { if (c.contains(f) && f.contains(p)) return f; } return null; } } } public Figure findFigureExcept(Point2D.Double p, Figure ignore) { Collection<Figure> c = quadTree.findContains(p); switch (c.size()) { case 0 : { return null; } case 1: { Figure f = c.iterator().next(); return (f == ignore || ! f.contains(p)) ? null : f; } default : { for (Figure f : getFiguresFrontToBack()) { if (f != ignore && f.contains(p)) return f; } return null; } } } public Figure findFigureExcept(Point2D.Double p, Collection ignore) { Collection<Figure> c = quadTree.findContains(p); switch (c.size()) { case 0 : { return null; } case 1: { Figure f = c.iterator().next(); return (! ignore.contains(f) || ! f.contains(p)) ? null : f; } default : { for (Figure f : getFiguresFrontToBack()) { if (! ignore.contains(f) && f.contains(p)) return f; } return null; } } } public Figure findFigureBehind(Point2D.Double p, Figure figure) { boolean isBehind = false; for (Figure f : getFiguresFrontToBack()) { if (isBehind) { if (f.isVisible() && f.contains(p)) { return f; } } else { isBehind = figure == f; } } return null; } public Figure findFigureBehind(Point2D.Double p, Collection<Figure> figures) { int inFrontOf = figures.size(); for (Figure f : getFiguresFrontToBack()) { if (inFrontOf == 0) { if (f.isVisible() && f.contains(p)) { return f; } } else { if (figures.contains(f)) { inFrontOf--; } } } return null; } public java.util.List<Figure> findFigures(Rectangle2D.Double r) { LinkedList<Figure> c = new LinkedList<Figure>(quadTree.findIntersects(r)); switch (c.size()) { case 0 : // fall through case 1: return c; default : return sort(c); } } public java.util.List<Figure> findFiguresWithin(Rectangle2D.Double bounds) { LinkedList<Figure> contained = new LinkedList<Figure>(); for (Figure f : children) { Rectangle2D r = f.getBounds(); if (AttributeKeys.TRANSFORM.get(f) != null) { r = AttributeKeys.TRANSFORM.get(f).createTransformedShape(r).getBounds2D(); } if (f.isVisible() && bounds.contains(r)) { contained.add(f); } } return contained; } public void bringToFront(Figure figure) { if (children.remove(figure)) { children.add(figure); needsSorting = true; fireAreaInvalidated(figure.getDrawingArea()); } } public void sendToBack(Figure figure) { if (children.remove(figure)) { children.add(0, figure); needsSorting = true; fireAreaInvalidated(figure.getDrawingArea()); } } public boolean contains(Figure f) { return children.contains(f); } /** * Ensures that the children are sorted in z-order sequence. */ private void ensureSorted() { if (needsSorting) { Collections.sort(children, FigureLayerComparator.INSTANCE); needsSorting = false; } } public void setCanvasSize(Dimension2DDouble newValue) { Dimension2DDouble oldValue = canvasSize; canvasSize = newValue; firePropertyChange("canvasSize", oldValue, newValue); } public Dimension2DDouble getCanvasSize() { return canvasSize; } /** * Handles all figure events fired by Figures contained in the Drawing. */ protected class FigureHandler extends FigureAdapter implements UndoableEditListener { /** * We propagate all edit events from our children to * undoable edit listeners, which have registered with us. */ public void undoableEditHappened(UndoableEditEvent e) { fireUndoableEditHappened(e.getEdit()); } @Override public void areaInvalidated(FigureEvent e) { fireAreaInvalidated(e.getInvalidatedArea()); } @Override public void figureChanged(FigureEvent e) { quadTree.remove(e.getFigure()); quadTree.add(e.getFigure(), e.getFigure().getDrawingArea()); needsSorting = true; fireAreaInvalidated(e.getInvalidatedArea()); } @Override public void figureRequestRemove(FigureEvent e) { remove(e.getFigure()); } } }