/*
Copyright 2006-2015 The QElectroTech Team
This file is part of QElectroTech.
QElectroTech 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 2 of the License, or
(at your option) any later version.
QElectroTech 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 QElectroTech. If not, see .
*/
#include "elementsmover.h"
#include "conductor.h"
#include "conductortextitem.h"
#include "diagram.h"
#include "diagramcommands.h"
#include "element.h"
#include "independenttextitem.h"
#include "diagramimageitem.h"
#include "elementtextitem.h"
#include "conductorautonumerotation.h"
/**
* @brief ElementsMover::ElementsMover Constructor
*/
ElementsMover::ElementsMover() :
movement_running_(false),
current_movement_(),
diagram_(0),
movement_driver_(0),
moved_content_()
{
}
/**
* @brief ElementsMover::~ElementsMover Destructor
*/
ElementsMover::~ElementsMover() {
}
/**
* @brief ElementsMover::isReady
* @return True if this element mover is ready to be used.
* A element mover is ready when the previous managed movement is finish.
*/
bool ElementsMover::isReady() const {
return(!movement_running_);
}
/**
* @brief ElementsMover::beginMovement
* Start a new movement
* @param diagram diagram where the movement is applied
* @param driver_item item moved by mouse and don't be moved by Element mover
* @return the numbers of items to be moved or -1 if movement can't be init.
*/
int ElementsMover::beginMovement(Diagram *diagram, QGraphicsItem *driver_item) {
// They must be no movement in progress
if (movement_running_) return(-1);
// Be sure we have diagram to work
if (!diagram) return(-1);
diagram_ = diagram;
// Take count of driver item
movement_driver_ = driver_item;
// At the beginning of movement, move is NULL
current_movement_ = QPointF(0.0, 0.0);
moved_content_ = diagram -> selectedContent();
moved_content_.removeNonMovableItems();
if (!moved_content_.count()) return(-1);
/* At this point, we've got all info to manage movement.
* There is now a move in progress */
movement_running_ = true;
return(moved_content_.count());
}
/**
* @brief ElementsMover::continueMovement
* Add a move to the current movement.
* @param movement movement to applied
*/
void ElementsMover::continueMovement(const QPointF &movement) {
if (!movement_running_ || movement.isNull()) return;
current_movement_ += movement;
//Move every movable item, except conductor
typedef DiagramContent dc;
foreach (QGraphicsItem *qgi, moved_content_.items(dc::Elements | dc::TextFields | dc::Images | dc::Shapes)) {
if (qgi == movement_driver_) continue;
qgi -> setPos(qgi->pos() + movement);
}
// Move some conductors
foreach(Conductor *conductor, moved_content_.conductorsToMove) {
conductor -> setPos(conductor -> pos() + movement);
}
// Recalcul the path of other conductors
foreach(Conductor *conductor, moved_content_.conductorsToUpdate) {
conductor -> updatePath();
}
}
/**
* @brief ElementsMover::endMovement
* Ended the current movement by creating an undo added to the undostack of the diagram.
* If there is only one element moved, we try to auto-connect new conductor from this element
* and other possible element.
*/
void ElementsMover::endMovement()
{
// A movement must be inited
if (!movement_running_) return;
QUndoCommand *undo_object = nullptr;
//Create undo move if there is a movement
if (!current_movement_.isNull())
undo_object = new MoveElementsCommand(diagram_, moved_content_, current_movement_);
//There is only one element moved, and project authorize auto conductor,
//we try auto connection of conductor;
typedef DiagramContent dc;
if (moved_content_.items(dc::TextFields | dc::Images | dc::Shapes).size() == 0 &&
moved_content_.items(dc::Elements).size() == 1 &&
diagram_ -> project() -> autoConductor())
{
Element *elmt = moved_content_.elements.toList().first();
while (!elmt -> AlignedFreeTerminals().isEmpty())
{
QPair pair = elmt -> AlignedFreeTerminals().takeFirst();
Conductor *conductor = new Conductor(pair.first, pair.second);
conductor -> setProperties(diagram_ -> defaultConductorProperties);
//Create an undo object for each new auto conductor, with undo_object for parent if exist
//Else the first undo for this auto conductor become the undo_object
if (undo_object)
new AddItemCommand(conductor, diagram_, QPointF(), undo_object);
else
undo_object = new AddItemCommand(conductor, diagram_, QPointF());
//Autonum the new conductor, the undo command associated for this, have for parent undo_object
ConductorAutoNumerotation can (conductor, diagram_, undo_object);
can.numerate();
};
}
//Add undo_object if exist
if (undo_object)
diagram_ -> undoStack().push(undo_object);
// There is no movement in progress now
movement_running_ = false;
}