/** * Copyright (c) 2011-2018 by Andrew Mustun. All rights reserved. * * This file is part of the QCAD project. * * QCAD is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * QCAD is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with QCAD. */ include("../Circle.js"); include("scripts/ShapeAlgorithms.js"); include("scripts/Apollonius.js"); /** * \class Circle2TR * \brief Circle from radius and two tangential entities. * \ingroup ecma_draw_circle */ function Circle2TR(guiAction) { Circle.call(this, guiAction); this.entity1 = undefined; this.shape1 = undefined; this.entity2 = undefined; this.entity2Id = undefined; this.shape2 = undefined; this.radius = undefined; this.center = undefined; this.pos = undefined; this.candidates = []; this.setUiOptions("Circle2TR.ui"); } Circle2TR.prototype = new Circle(); Circle2TR.State = { ChoosingShape1 : 0, ChoosingShape2 : 1, ChoosingSolution : 2 }; Circle2TR.prototype.beginEvent = function() { Circle.prototype.beginEvent.call(this); this.setState(Circle2TR.State.ChoosingShape1); }; Circle2TR.prototype.setState = function(state) { Circle.prototype.setState.call(this, state); this.getDocumentInterface().setClickMode(RAction.PickEntity); this.setCrosshairCursor(); var appWin = RMainWindowQt.getMainWindow(); switch (this.state) { case Circle2TR.State.ChoosingShape1: this.entity1 = undefined; this.shape1 = undefined; this.entity2 = undefined; this.entity2Id = undefined; this.shape2 = undefined; this.center = undefined; this.pos = undefined; this.candidates = []; var trFirstEntity = qsTr("First line, arc or circle"); this.setCommandPrompt(trFirstEntity); this.setLeftMouseTip(trFirstEntity); this.setRightMouseTip(EAction.trCancel); break; case Circle2TR.State.ChoosingShape2: this.entity2 = undefined; this.entity2Id = undefined; this.shape2 = undefined; this.center = undefined; this.pos = undefined; this.candidates = []; var trSecondEntity = qsTr("Second line, arc or circle"); this.setCommandPrompt(trSecondEntity); this.setLeftMouseTip(trSecondEntity); this.setRightMouseTip(EAction.trBack); break; case Circle2TR.State.ChoosingSolution: this.pos = undefined; var trSolution = qsTr("Choose solution"); this.setCommandPrompt(trSolution); this.setLeftMouseTip(trSolution); this.setRightMouseTip(EAction.trBack); break; } this.simulateMouseMoveEvent(); }; Circle2TR.prototype.escapeEvent = function() { switch (this.state) { case Circle2TR.State.ChoosingShape1: EAction.prototype.escapeEvent.call(this); break; case Circle2TR.State.ChoosingShape2: this.setState(Circle2TR.State.ChoosingShape1); break; case Circle2TR.State.ChoosingSolution: this.setState(Circle2TR.State.ChoosingShape2); break; } }; Circle2TR.prototype.pickEntity = function(event, preview) { this.error = ""; var di = this.getDocumentInterface(); var doc = this.getDocument(); var entityId = this.getEntityId(event, preview); var entity = doc.queryEntity(entityId); var pos = event.getModelPosition(); var shape = undefined; if (this.state!==Circle2TR.State.ChoosingSolution) { if (!this.isEntitySnappable(entity)) { // entity not on a snappable layer: return; } if (isNull(entity)) { return; } else { shape = entity.getClosestSimpleShape(pos); if (!isLineBasedShape(shape) && !isArcShape(shape) && !isCircleShape(shape)) { if (!preview) { EAction.warnNotLineArcCircle(); return; } shape = undefined; } } } switch (this.state) { case Circle2TR.State.ChoosingShape1: this.entity1 = entity; this.shape1 = shape; if (preview) { this.updatePreview(); } else { this.setState(Circle2TR.State.ChoosingShape2); } break; case Circle2TR.State.ChoosingShape2: // optimization breaks when choosing two segments of the same polyline: //if (entityId!==this.entity2Id) { // force re-calculation: this.candidates = []; //} this.entity2 = entity; this.entity2Id = entityId; this.shape2 = shape; if (preview) { this.updatePreview(); } else { var op = this.getOperation(false); if (!isNull(op)) { // di.applyOperation(op); // this.setState(Circle2TR.State.ChoosingShape1); // only one solution: if (this.candidates.length===1) { di.applyOperation(op); this.setState(Circle2TR.State.ChoosingShape1); } // multiple solutions: else { this.setState(Circle2TR.State.ChoosingSolution); } } else { this.setState(Circle2TR.State.ChoosingShape1); this.error = qsTr("No solution"); } } break; case Circle2TR.State.ChoosingSolution: this.pos = event.getModelPosition(); if (preview) { this.updatePreview(); } else { var op = this.getOperation(false); if (!isNull(op)) { di.applyOperation(op); this.setState(Circle2TR.State.ChoosingShape1); } } break; } if (!preview && this.error.length!==0) { EAction.handleUserWarning(this.error); } }; Circle2TR.prototype.getOperation = function(preview) { var shapes = this.getCircles2TR(preview); if (isNull(shapes)) { return undefined; } var doc = this.getDocument(); var op = new RAddObjectsOperation(); op.setText(this.getToolTitle()); for (var i=0; i