/**************************************************************************** ** ** This file is part of the LibreCAD project, a 2D CAD program ** ** Copyright (C) 2010 R. van Twisk (librecad@rvt.dds.nl) ** Copyright (C) 2001-2003 RibbonSoft. All rights reserved. ** ** ** This file may be distributed and/or modified under the terms of the ** GNU General Public License version 2 as published by the Free Software ** Foundation and appearing in the file gpl-2.0.txt included in the ** packaging of this file. ** ** This program 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 this program; if not, write to the Free Software ** Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA ** ** This copyright notice MUST APPEAR in all copies of the script! ** **********************************************************************/ #include "rs_actiondrawpolyline.h" #include #include "rs_dialogfactory.h" #include "rs_graphicview.h" #include "rs_commands.h" #include "rs_commandevent.h" RS_ActionDrawPolyline::RS_ActionDrawPolyline(RS_EntityContainer& container, RS_GraphicView& graphicView) :RS_PreviewActionInterface("Draw polylines", container, graphicView) { Reversed=1; reset(); } RS_ActionDrawPolyline::~RS_ActionDrawPolyline() {} QAction* RS_ActionDrawPolyline::createGUIAction(RS2::ActionType /*type*/, QObject* /*parent*/) { // (tr("Polyline") QAction* action = new QAction(tr("&Polyline"), NULL); action->setIcon(QIcon(":/extui/polyline.png")); action->setStatusTip(tr("Draw polylines")); return action; } void RS_ActionDrawPolyline::reset() { polyline = NULL; data = RS_PolylineData(RS_Vector(false), RS_Vector(false), false); start = RS_Vector(false); history.clear(); bHistory.clear(); } void RS_ActionDrawPolyline::init(int status) { RS_PreviewActionInterface::init(status); reset(); } void RS_ActionDrawPolyline::trigger() { RS_PreviewActionInterface::trigger(); if (polyline==NULL) { return; } // add the entity //RS_Polyline* polyline = new RS_Polyline(container, data); //polyline->setLayerToActive(); //polyline->setPenToActive(); //container->addEntity(polyline); // upd. undo list: if (document!=NULL) { document->startUndoCycle(); document->addUndoable(polyline); document->endUndoCycle(); } // upd view deleteSnapper(); graphicView->moveRelativeZero(RS_Vector(0.0,0.0)); graphicView->drawEntity(polyline); graphicView->moveRelativeZero(polyline->getEndpoint()); drawSnapper(); RS_DEBUG->print("RS_ActionDrawLinePolyline::trigger(): polyline added: %d", polyline->getId()); polyline = NULL; } void RS_ActionDrawPolyline::mouseMoveEvent(QMouseEvent* e) { RS_DEBUG->print("RS_ActionDrawLinePolyline::mouseMoveEvent begin"); RS_Vector mouse = snapPoint(e); double bulge=solveBulge(mouse); if (getStatus()==SetNextPoint && point.valid) { deletePreview(); // clearPreview(); //RS_Polyline* p = polyline->clone(); //p->reparent(preview); //preview->addEntity(p); if (fabs(bulge)addEntity(new RS_Line(preview, RS_LineData(point, mouse))); } else preview->addEntity(new RS_Arc(preview,arc_data)); drawPreview(); } RS_DEBUG->print("RS_ActionDrawLinePolyline::mouseMoveEvent end"); } void RS_ActionDrawPolyline::mouseReleaseEvent(QMouseEvent* e) { if (e->button()==Qt::LeftButton) { RS_CoordinateEvent ce(snapPoint(e)); coordinateEvent(&ce); } else if (e->button()==Qt::RightButton) { if (getStatus()==SetNextPoint) { trigger(); } deletePreview(); // clearPreview(); deleteSnapper(); init(getStatus()-1); } } double RS_ActionDrawPolyline::solveBulge(RS_Vector mouse) { double b(0.); bool suc; RS_Arc arc(NULL, RS_ArcData()); RS_Line line(NULL,RS_LineData()); double direction,direction2,delta; RS_AtomicEntity* lastentity; calculatedSegment=false; switch (Mode){ // case Line: // b=0.0; // break; case Tangential: if (polyline!=NULL){ lastentity = (RS_AtomicEntity*)polyline->lastEntity(); direction = RS_Math::correctAngle( lastentity->getDirection2()+M_PI); line.setStartpoint(point); line.setEndpoint(mouse); direction2=RS_Math::correctAngle(line.getDirection2()+M_PI); delta=direction2-direction; if( fabs(remainder(delta,M_PI))>RS_TOLERANCE_ANGLE ) { b=tan(delta/2); suc = arc.createFrom2PBulge(point,mouse,b); if (suc) arc_data = arc.getData(); else b=0; } break; // if(deltaM_PI-RS_TOLERANCE_ANGLE)) // b=0; // else{ // b=tan((direction2-direction)/2); // suc = arc.createFrom2PBulge(point,mouse,b); // if (suc) // arc_data = arc.getData(); // else // b=0; // } } // else // b=0; // break; case TanRad: if (polyline!=NULL){ lastentity = (RS_AtomicEntity*)polyline->lastEntity(); direction = RS_Math::correctAngle( lastentity->getDirection2()+M_PI); suc = arc.createFrom2PDirectionRadius(point, mouse, direction,Radius); if (suc){ arc_data = arc.getData(); b=arc.getBulge(); calculatedEndpoint = arc.getEndpoint(); calculatedSegment=true; } // else // b=0; } // else // b=0; break; /* case TanAng: b=tan(Reversed*Angle*M_PI/720.0); break; case TanRadAng: b=tan(Reversed*Angle*M_PI/720.0); break;*/ case Ang: b=tan(Reversed*Angle*M_PI/720.0); suc = arc.createFrom2PBulge(point,mouse,b); if (suc) arc_data = arc.getData(); else b=0; break; default: break; /* case RadAngEndp: b=tan(Reversed*Angle*M_PI/720.0); break; case RadAngCenp: b=tan(Reversed*Angle*M_PI/720.0);*/ } return b; } void RS_ActionDrawPolyline::coordinateEvent(RS_CoordinateEvent* e) { if (e==NULL) { return; } RS_Vector mouse = e->getCoordinate(); double bulge=solveBulge(mouse); if (calculatedSegment) mouse=calculatedEndpoint; switch (getStatus()) { case SetStartpoint: // data.startpoint = mouse; //printf ("SetStartpoint\n"); point = mouse; history.clear(); history.append(mouse); bHistory.clear(); bHistory.append(0.0); start = point; setStatus(SetNextPoint); graphicView->moveRelativeZero(mouse); updateMouseButtonHints(); break; case SetNextPoint: graphicView->moveRelativeZero(mouse); point = mouse; history.append(mouse); bHistory.append(bulge); if (polyline==NULL) { //printf("polyline==NULL\n"); polyline = new RS_Polyline(container, data); polyline->addVertex(start, 0.0); } if (polyline!=NULL) { polyline->setNextBulge(bulge); polyline->addVertex(mouse, 0.0); polyline->setEndpoint(mouse); if (polyline->count()==1) { polyline->setLayerToActive(); polyline->setPenToActive(); container->addEntity(polyline); } deletePreview(); // clearPreview(); deleteSnapper(); graphicView->drawEntity(polyline); drawSnapper(); } //trigger(); //data.startpoint = data.endpoint; updateMouseButtonHints(); //graphicView->moveRelativeZero(mouse); break; default: break; } } void RS_ActionDrawPolyline::commandEvent(RS_CommandEvent* e) { QString c = e->getCommand().toLower(); switch (getStatus()) { case SetStartpoint: if (checkCommand("help", c)) { RS_DIALOGFACTORY->commandMessage(msgAvailableCommands() + getAvailableCommands().join(", ")); return; } break; case SetNextPoint: if (checkCommand("close", c)) { close(); updateMouseButtonHints(); return; } if (checkCommand("undo", c)) { undo(); updateMouseButtonHints(); return; } break; default: break; } } QStringList RS_ActionDrawPolyline::getAvailableCommands() { QStringList cmd; switch (getStatus()) { case SetStartpoint: break; case SetNextPoint: if (history.size()>=2) { cmd += command("undo"); } if (history.size()>=3) { cmd += command("close"); } break; default: break; } return cmd; } void RS_ActionDrawPolyline::updateMouseButtonHints() { switch (getStatus()) { case SetStartpoint: RS_DIALOGFACTORY->updateMouseWidget(tr("Specify first point"), tr("Cancel")); break; case SetNextPoint: { QString msg = ""; if (history.size()>=3) { msg += RS_COMMANDS->command("close"); msg += "/"; } if (history.size()>=2) { msg += RS_COMMANDS->command("undo"); } if (history.size()>=2) { RS_DIALOGFACTORY->updateMouseWidget( tr("Specify next point or [%1]").arg(msg), tr("Back")); } else { RS_DIALOGFACTORY->updateMouseWidget( tr("Specify next point"), tr("Back")); } } break; default: RS_DIALOGFACTORY->updateMouseWidget("", ""); break; } } void RS_ActionDrawPolyline::showOptions() { RS_ActionInterface::showOptions(); RS_DIALOGFACTORY->requestOptions(this, true); } void RS_ActionDrawPolyline::hideOptions() { RS_ActionInterface::hideOptions(); RS_DIALOGFACTORY->requestOptions(this, false); } void RS_ActionDrawPolyline::updateMouseCursor() { graphicView->setMouseCursor(RS2::CadCursor); } void RS_ActionDrawPolyline::updateToolBar() { if (RS_DIALOGFACTORY!=NULL) { if (isFinished()) { RS_DIALOGFACTORY->resetToolBar(); } } } void RS_ActionDrawPolyline::close() { if (history.size()>2 && start.valid) { //data.endpoint = start; //trigger(); if (polyline!=NULL) { if (Mode==TanRad) Mode=Line; RS_CoordinateEvent e(polyline->getStartpoint()); coordinateEvent(&e); } polyline->setClosed(true); trigger(); setStatus(SetStartpoint); graphicView->moveRelativeZero(start); } else { RS_DIALOGFACTORY->commandMessage( tr("Cannot close sequence of lines: " "Not enough entities defined yet.")); } } void RS_ActionDrawPolyline::undo() { if (history.size()>1) { history.removeLast(); bHistory.removeLast(); deletePreview(); point = history.last(); if(history.size()==1){ graphicView->moveRelativeZero(history.at(0)); //remove polyline from container, //container calls delete over polyline container->removeEntity(polyline); polyline = NULL; graphicView->drawEntity(polyline); } if (polyline!=NULL) { polyline->removeLastVertex(); graphicView->moveRelativeZero(polyline->getEndpoint()); graphicView->drawEntity(polyline); } } else { RS_DIALOGFACTORY->commandMessage( tr("Cannot undo: " "Not enough entities defined yet.")); } } // EOF