/* ****************************************************************************** * Copyright (c) 2006-2012 XMind Ltd. and others. * * This file is a part of XMind 3. XMind releases 3 and * above are dual-licensed under the Eclipse Public License (EPL), * which is available at http://www.eclipse.org/legal/epl-v10.html * and the GNU Lesser General Public License (LGPL), * which is available at http://www.gnu.org/licenses/lgpl.html * See http://www.xmind.net/license.html for details. * * Contributors: * XMind Ltd. - initial API and implementation *******************************************************************************/ package org.xmind.ui.internal.decorations; import java.util.List; import org.eclipse.draw2d.IFigure; import org.eclipse.draw2d.PositionConstants; import org.xmind.gef.draw2d.geometry.PrecisionLine; import org.xmind.gef.draw2d.geometry.PrecisionLine.LineType; import org.xmind.gef.draw2d.geometry.PrecisionPoint; import org.xmind.gef.draw2d.graphics.Path; import org.xmind.ui.decorations.AbstractBranchConnection; public class CurveBranchConnection extends AbstractBranchConnection { private static final double CPRatio = 1.0 / 3; private PrecisionPoint s1 = new PrecisionPoint(); private PrecisionPoint s2 = new PrecisionPoint(); private PrecisionPoint t1 = new PrecisionPoint(); private PrecisionPoint t2 = new PrecisionPoint(); private PrecisionPoint c1 = new PrecisionPoint(); private PrecisionPoint c2 = new PrecisionPoint(); private PrecisionPoint c3 = new PrecisionPoint(); private PrecisionPoint c4 = new PrecisionPoint(); private PrecisionPoint c5 = new PrecisionPoint(); private PrecisionPoint c6 = new PrecisionPoint(); private PrecisionPoint control = null; public CurveBranchConnection() { super(); } public CurveBranchConnection(String id) { super(id); } protected void route(IFigure figure, Path shape) { PrecisionPoint sourcePos = getSourcePosition(figure); PrecisionPoint targetPos = getTargetPosition(figure); if (isTapered()) { shape.moveTo(s1); shape.quadTo(c1, t1); shape.lineTo(t2); shape.quadTo(c2, s2); shape.cubicTo(c6, c5, s1); shape.close(); } else { shape.moveTo(sourcePos); shape.quadTo(control, targetPos); } } protected void calculateControlPoints(IFigure figure, PrecisionPoint sourcePos, PrecisionPoint targetPos) { control = new PrecisionPoint(); calcControlPoint(sourcePos, targetPos, isTargetHorizontal(), control); if (isTapered()) { calcTaperedPositions(sourcePos, control, 0, s1, s2); calcTaperedPositions(control, sourcePos, 0, 2, c2, c1); calcTaperedPositions(control, targetPos, 1, 1, t1, t2); calcTaperedPositions(control, targetPos, 0, 2, c3, c4); PrecisionLine l1 = new PrecisionLine(s1, c1, LineType.Ray); PrecisionLine l2 = new PrecisionLine(t1, c3, LineType.Ray); List<PrecisionPoint> ps = l1.getLinesIntersections(l2); if (!ps.isEmpty() && ps.size() == 1) { c1.setLocation(ps.get(0)); } l1.setOrigin(s2); l1.setTerminus(c2); l2.setOrigin(t2); l2.setTerminus(c4); ps = l1.getLinesIntersections(l2); if (!ps.isEmpty() && ps.size() == 1) { c2.setLocation(ps.get(0)); } calculateSourceControlPoints(c1, s1, s2, c5); calculateSourceControlPoints(c2, s2, s1, c6); } } protected boolean isPositionValid() { return super.isPositionValid() && control != null; } public void invalidate() { super.invalidate(); control = null; } protected boolean isTargetHorizontal() { return (getTargetOrientation() & PositionConstants.EAST_WEST) != 0; } protected double getControlPointRatio() { return CPRatio; } protected PrecisionPoint calcControlPoint(PrecisionPoint source, PrecisionPoint target, boolean targetHorizontal, PrecisionPoint result) { return result .setLocation(// targetHorizontal ? target.x * getControlPointRatio() + source.x * (1 - getControlPointRatio()) : target.x, // targetHorizontal ? target.y : target.y * getControlPointRatio() + source.y * (1 - getControlPointRatio())); } }