579 lines
15 KiB
C++
Raw Normal View History

/*
2025-01-04 13:37:40 +01:00
Copyright 2006-2025 The QElectroTech Team
2020-10-17 20:25:30 +02:00
This file is part of QElectroTech.
2020-10-03 15:48:40 +02:00
2020-10-17 20:25:30 +02:00
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.
2020-10-03 15:48:40 +02:00
2020-10-17 20:25:30 +02:00
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.
2020-10-03 15:48:40 +02:00
2020-10-17 20:25:30 +02:00
You should have received a copy of the GNU General Public License
along with QElectroTech. If not, see <http://www.gnu.org/licenses/>.
*/
#include "partrectangle.h"
2020-12-09 15:28:43 +01:00
2020-12-10 18:44:03 +01:00
#include "../../QPropertyUndoCommand/qpropertyundocommand.h"
#include "../../QetGraphicsItemModeler/qetgraphicshandleritem.h"
#include "../../QetGraphicsItemModeler/qetgraphicshandlerutility.h"
#include "../elementscene.h"
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::PartRectangle
Constructor
@param editor the QETElementEditor of this item
@param parent parent item
2020-08-16 11:19:36 +02:00
*/
PartRectangle::PartRectangle(QETElementEditor *editor, QGraphicsItem *parent) :
2020-10-17 20:25:30 +02:00
CustomElementGraphicPart(editor, parent)
{
2024-04-24 14:14:40 +02:00
m_rot=0;
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::~PartRectangle
2020-08-16 11:19:36 +02:00
*/
2020-09-07 22:03:40 +02:00
PartRectangle::~PartRectangle()
{
2020-10-17 20:25:30 +02:00
removeHandler();
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::paint
Draw this Rectangle
@param painter
@param options
@param widget
2020-08-16 11:19:36 +02:00
*/
void PartRectangle::paint(QPainter *painter, const QStyleOptionGraphicsItem *options, QWidget *widget)
{
2020-10-17 20:25:30 +02:00
Q_UNUSED(widget);
applyStylesToQPainter(*painter);
QPen t = painter -> pen();
t.setCosmetic(options && options -> levelOfDetailFromTransform(painter->worldTransform()) < 1.0);
if (isSelected())
t.setColor(Qt::red);
2020-10-17 20:25:30 +02:00
t.setJoinStyle(Qt::MiterJoin);
2020-10-03 15:48:40 +02:00
2020-10-17 20:25:30 +02:00
//Force the pen to width 0 if one of dimension is null
if (!rect().width() || !rect().height())
t.setWidth(0);
2020-10-03 15:48:40 +02:00
2020-10-17 20:25:30 +02:00
painter->setPen(t);
painter->drawRoundedRect(m_rect, m_xRadius, m_yRadius);
2020-10-17 20:25:30 +02:00
if (m_hovered)
drawShadowShape(painter);
2020-10-17 20:25:30 +02:00
if (isSelected())
drawCross(m_rect.center(), painter);
}
/**
@brief PartRectangle::toXml
2020-10-17 20:25:30 +02:00
Export this rectangle in xml
@param xml_document : Xml document to use for create the xml element.
@return an xml element that describe this ellipse
2020-08-16 11:19:36 +02:00
*/
const QDomElement PartRectangle::toXml(QDomDocument &xml_document) const
{
QDomElement xml_element = xml_document.createElement("rect");
QPointF top_left(sceneTopLeft());
qreal x = (qRound(top_left.x() * 100.0) / 100.0);
qreal y = (qRound(top_left.y() * 100.0) / 100.0);
qreal w = (qRound(m_rect.width() * 100.0) / 100.0);
qreal h = (qRound(m_rect.height() * 100.0) / 100.0);
qreal rx = (qRound(m_xRadius * 100.0) / 100.0);
qreal ry = (qRound(m_yRadius * 100.0) / 100.0);
xml_element.setAttribute("x", QString::number(x));
xml_element.setAttribute("y", QString::number(y));
xml_element.setAttribute("width", QString::number(w));
xml_element.setAttribute("height", QString::number(h));
xml_element.setAttribute("rx", QString::number(rx));
xml_element.setAttribute("ry", QString::number(ry));
stylesToXml(xml_element);
return(xml_element);
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::fromXml
Import the properties of this rectangle from a xml element.
@param qde : Xml document to use.
2020-08-16 11:19:36 +02:00
*/
void PartRectangle::fromXml(const QDomElement &qde)
{
2020-10-17 20:25:30 +02:00
stylesFromXml(qde);
setPos(mapFromScene(qde.attribute("x", "0").toDouble(),
qde.attribute("y", "0").toDouble()));
QRectF rect(QPointF(0,0), QSizeF(qde.attribute("width", "0").toDouble(),
qde.attribute("height", "0").toDouble()));
2020-10-17 20:25:30 +02:00
setRect(rect.normalized());
setXRadius(qde.attribute("rx", "0").toDouble());
setYRadius(qde.attribute("ry", "0").toDouble());
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::rect
@return : Returns the item's rectangle.
2020-08-16 11:19:36 +02:00
*/
2020-09-07 22:03:40 +02:00
QRectF PartRectangle::rect() const
{
2020-10-17 20:25:30 +02:00
return m_rect;
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::setRect
Sets the item's rectangle to be the given rectangle.
@param rect
2020-08-16 11:19:36 +02:00
*/
void PartRectangle::setRect(const QRectF &rect)
{
2020-10-17 20:25:30 +02:00
if (rect == m_rect) return;
prepareGeometryChange();
m_rect = rect;
adjustHandlerPos();
2020-10-17 20:25:30 +02:00
emit rectChanged();
}
void PartRectangle::setXRadius(qreal X)
{
2020-10-17 20:25:30 +02:00
m_xRadius = X;
update();
adjustHandlerPos();
2020-10-17 20:25:30 +02:00
emit XRadiusChanged();
}
void PartRectangle::setYRadius(qreal Y)
{
2020-10-17 20:25:30 +02:00
m_yRadius = Y;
update();
adjustHandlerPos();
2020-10-17 20:25:30 +02:00
emit YRadiusChanged();
}
void PartRectangle::setRotation(qreal angle) {
qreal diffAngle = qRound((angle - rotation()) * 100.0) / 100.0;
m_rot = QET::correctAngle(angle, true);
auto p1 = mapToScene(m_rect.x(),m_rect.y());
qreal width = m_rect.height();
qreal height = m_rect.width();
qreal x; qreal y;
if (diffAngle > 0) {
x = (p1.y() + m_rect.height()) * (-1);
y = p1.x();
} else {
x = p1.y();
y = (p1.x() + m_rect.width()) * (-1);
}
p1 = mapFromScene(x, y);
m_rect = QRectF(p1.x(), p1.y(), width, height);
std::swap (m_xRadius, m_yRadius);
prepareGeometryChange();
adjustHandlerPos();
emit rectChanged();
}
qreal PartRectangle::rotation() const {
return qRound(m_rot * 100.0) / 100.0;
}
void PartRectangle::flip() {
auto height = m_rect.height();
auto p1 = mapToScene(m_rect.x(),m_rect.y());
qreal x = p1.x();
qreal y = ((-1.0) * p1.y()) - height;
p1 = mapFromScene(x, y);
m_rect.setX(p1.x());
m_rect.setY(p1.y());
m_rect.setHeight(height);
prepareGeometryChange();
adjustHandlerPos();
emit rectChanged();
}
void PartRectangle::mirror() {
auto width = m_rect.width();
auto p1 = mapToScene(m_rect.x(),m_rect.y());
qreal x = ((-1.0) * p1.x()) - width;
qreal y = p1.y();
p1 = mapFromScene(x, y);
m_rect.setX(p1.x());
m_rect.setY(p1.y());
m_rect.setWidth(width);
prepareGeometryChange();
adjustHandlerPos();
emit rectChanged();
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::sceneGeometricRect
@return the minimum, margin-less rectangle this part can fit into, in scene
coordinates. It is different from boundingRect() because it is not supposed
to imply any margin, and it is different from shape because it is a regular
rectangle, not a complex shape.
2020-08-16 11:19:36 +02:00
*/
2020-09-07 22:03:40 +02:00
QRectF PartRectangle::sceneGeometricRect() const
{
2020-10-17 20:25:30 +02:00
return(mapToScene(rect()).boundingRect());
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::sceneTopLeft
@return the top left of rectangle, in scene coordinate
2020-08-16 11:19:36 +02:00
*/
2020-09-07 22:03:40 +02:00
QPointF PartRectangle::sceneTopLeft() const
{
2024-04-24 14:14:40 +02:00
return(mapToScene(rect().topLeft()));
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::shape
@return the shape of this item
2020-08-16 11:19:36 +02:00
*/
QPainterPath PartRectangle::shape() const
{
2020-10-17 20:25:30 +02:00
QPainterPath shape;
shape.addRoundedRect(m_rect, m_xRadius, m_yRadius);
2020-10-17 20:25:30 +02:00
QPainterPathStroker pps;
pps.setWidth(m_hovered? penWeight()+SHADOWS_HEIGHT : penWeight());
shape = pps.createStroke(shape);
2020-10-17 20:25:30 +02:00
return shape;
}
QPainterPath PartRectangle::shadowShape() const
{
2020-10-17 20:25:30 +02:00
QPainterPath shape;
shape.addRoundedRect(m_rect, m_xRadius, m_yRadius);
2020-10-17 20:25:30 +02:00
QPainterPathStroker pps;
pps.setWidth(penWeight());
2020-10-17 20:25:30 +02:00
return (pps.createStroke(shape));
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::boundingRect
@return Bounding rectangle this part can fit into
2020-08-16 11:19:36 +02:00
*/
QRectF PartRectangle::boundingRect() const
{
2020-10-17 20:25:30 +02:00
qreal adjust = (SHADOWS_HEIGHT + penWeight()) / 2;
//We add 0.5 because CustomElementGraphicPart::drawShadowShape
//draw a shape bigger of 0.5 when pen weight is to 0.
if (penWeight() == 0) adjust += 0.5;
2020-10-17 20:25:30 +02:00
QRectF r = m_rect.normalized();
r.adjust(-adjust, -adjust, adjust, adjust);
2020-10-17 20:25:30 +02:00
return(r);
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::isUseless
@return true if this part is irrelevant and does not deserve to be Retained / registered.
An rectangle is relevant when he's not null.
2020-08-16 11:19:36 +02:00
*/
2020-09-07 22:03:40 +02:00
bool PartRectangle::isUseless() const
{
2020-10-17 20:25:30 +02:00
return(rect().isNull());
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::startUserTransformation
Start the user-induced transformation, provided this primitive is contained
within the initial_selection_rect bounding rectangle.
@param initial_selection_rect
2020-08-16 11:19:36 +02:00
*/
void PartRectangle::startUserTransformation(const QRectF &initial_selection_rect)
{
2020-10-17 20:25:30 +02:00
Q_UNUSED(initial_selection_rect)
// we keep track of our own rectangle at the moment in scene coordinates too
saved_points_.clear();
saved_points_ << mapToScene(rect().topLeft()) << mapToScene(rect().bottomRight());
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::handleUserTransformation
Handle the user-induced transformation from \a initial_selection_rect to \a new_selection_rect
@param initial_selection_rect
@param new_selection_rect
2020-08-16 11:19:36 +02:00
*/
void PartRectangle::handleUserTransformation(const QRectF &initial_selection_rect, const QRectF &new_selection_rect)
{
2020-10-17 20:25:30 +02:00
QList<QPointF> mapped_points = mapPoints(initial_selection_rect, new_selection_rect, saved_points_);
setRect(QRectF(mapFromScene(mapped_points.at(0)), mapFromScene(mapped_points.at(1))));
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::mouseReleaseEvent
Handle mouse release event
@param event
2020-08-16 11:19:36 +02:00
*/
void PartRectangle::mouseReleaseEvent(QGraphicsSceneMouseEvent *event)
{
2020-10-17 20:25:30 +02:00
if (event->button() == Qt::LeftButton && event->buttonDownPos(Qt::LeftButton) == event->pos())
switchResizeMode();
2020-10-03 15:48:40 +02:00
2020-10-17 20:25:30 +02:00
CustomElementGraphicPart::mouseReleaseEvent(event);
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::itemChange
@param change
@param value
@return
2020-08-16 11:19:36 +02:00
*/
QVariant PartRectangle::itemChange(QGraphicsItem::GraphicsItemChange change, const QVariant &value)
{
if (change == ItemPositionHasChanged)
2020-10-17 20:25:30 +02:00
{
adjustHandlerPos();
2020-10-17 20:25:30 +02:00
}
else if (change == ItemSceneChange)
{
setSelected(false); //This item is removed from scene, then we deselect this, and so, the handlers is also removed.
}
return QGraphicsItem::itemChange(change, value);
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::sceneEventFilter
@param watched
@param event
@return
2020-08-16 11:19:36 +02:00
*/
bool PartRectangle::sceneEventFilter(QGraphicsItem *watched, QEvent *event)
{
2020-10-17 20:25:30 +02:00
//Watched must be an handler
if(watched->type() == QetGraphicsHandlerItem::Type)
{
QetGraphicsHandlerItem *qghi = qgraphicsitem_cast<QetGraphicsHandlerItem *>(watched);
if(m_handler_vector.contains(qghi)) //Handler must be in m_vector_index, then we can start resize
{
m_vector_index = m_handler_vector.indexOf(qghi);
if (m_vector_index != -1)
{
if(event->type() == QEvent::GraphicsSceneMousePress) //Click
{
handlerMousePressEvent(qghi, static_cast<QGraphicsSceneMouseEvent *>(event));
return true;
}
else if(event->type() == QEvent::GraphicsSceneMouseMove) //Move
{
handlerMouseMoveEvent(qghi, static_cast<QGraphicsSceneMouseEvent *>(event));
return true;
}
else if (event->type() == QEvent::GraphicsSceneMouseRelease) //Release
{
handlerMouseReleaseEvent(qghi, static_cast<QGraphicsSceneMouseEvent *>(event));
return true;
}
}
}
}
return false;
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::switchResizeMode
2020-08-16 11:19:36 +02:00
*/
void PartRectangle::switchResizeMode()
{
2020-10-17 20:25:30 +02:00
if (m_resize_mode == 1)
{
m_resize_mode = 2;
for (QetGraphicsHandlerItem *qghi : m_handler_vector)
2020-10-17 20:25:30 +02:00
qghi->setColor(Qt::darkGreen);
}
else if (m_resize_mode == 2)
{
m_resize_mode = 3;
qDeleteAll(m_handler_vector);
m_handler_vector.clear();
addHandler();
for (QetGraphicsHandlerItem *qghi : m_handler_vector) {
2020-10-17 20:25:30 +02:00
qghi->setColor(Qt::magenta);
}
}
else if (m_resize_mode == 3)
{
m_resize_mode = 1;
qDeleteAll(m_handler_vector);
m_handler_vector.clear();
addHandler();
for (QetGraphicsHandlerItem *qghi : m_handler_vector) {
2020-10-17 20:25:30 +02:00
qghi->setColor(Qt::blue);
}
}
}
/**
@brief PartRectangle::adjustHandlerPos
2020-08-16 11:19:36 +02:00
*/
void PartRectangle::adjustHandlerPos()
{
2020-10-17 20:25:30 +02:00
if (m_handler_vector.isEmpty()) {
return;
}
QVector <QPointF> points_vector;
if(m_resize_mode != 3) {
points_vector = QetGraphicsHandlerUtility::pointsForRect(m_rect);
}
else {
points_vector = QetGraphicsHandlerUtility::pointForRadiusRect(m_rect, m_xRadius, m_yRadius);
}
if (m_handler_vector.size() == points_vector.size())
{
points_vector = mapToScene(points_vector);
for (int i = 0 ; i < points_vector.size() ; ++i)
m_handler_vector.at(i)->setPos(points_vector.at(i));
}
else
{
qDeleteAll(m_handler_vector);
m_handler_vector.clear();
addHandler();
}
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::handlerMousePressEvent
@param qghi
@param event
2020-08-16 11:19:36 +02:00
*/
void PartRectangle::handlerMousePressEvent(QetGraphicsHandlerItem *qghi, QGraphicsSceneMouseEvent *event)
{
2020-10-17 20:25:30 +02:00
Q_UNUSED(qghi)
Q_UNUSED(event)
m_old_rect = m_rect;
m_old_xRadius = m_xRadius;
m_old_yRadius = m_yRadius;
if(m_xRadius == 0 && m_yRadius == 0) {
m_modifie_radius_equaly = true;
}
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::handlerMouseMoveEvent
@param qghi
@param event
2020-08-16 11:19:36 +02:00
*/
void PartRectangle::handlerMouseMoveEvent(QetGraphicsHandlerItem *qghi, QGraphicsSceneMouseEvent *event)
{
2020-10-17 20:25:30 +02:00
Q_UNUSED(qghi)
QPointF new_pos = event->scenePos();
if (event->modifiers() != Qt::ControlModifier)
new_pos = elementScene()->snapToGrid(event->scenePos());
new_pos = mapFromScene(new_pos);
if (m_resize_mode == 1)
setRect(QetGraphicsHandlerUtility::rectForPosAtIndex(m_rect, new_pos, m_vector_index));
else if (m_resize_mode == 2)
setRect(QetGraphicsHandlerUtility::mirrorRectForPosAtIndex(m_rect, new_pos, m_vector_index));
else
{
qreal radius = QetGraphicsHandlerUtility::radiusForPosAtIndex(m_rect, new_pos, m_vector_index);
if(m_modifie_radius_equaly) {
setXRadius(radius);
setYRadius(radius);
}
else if(m_vector_index == 0) {
setXRadius(radius);
}
else {
setYRadius(radius);
}
}
adjustHandlerPos();
}
void PartRectangle::handlerMouseReleaseEvent(QetGraphicsHandlerItem *qghi, QGraphicsSceneMouseEvent *event)
{
2020-10-17 20:25:30 +02:00
Q_UNUSED(qghi)
Q_UNUSED(event)
m_modifie_radius_equaly = false;
QUndoCommand *undo = new QUndoCommand("Modifier un rectangle");
if (m_old_rect != m_rect) {
QPropertyUndoCommand *u = new QPropertyUndoCommand(this, "rect", QVariant(m_old_rect.normalized()), QVariant(m_rect.normalized()), undo);
u->setAnimated(true, false);
}
if (m_old_xRadius != m_xRadius) {
QPropertyUndoCommand *u = new QPropertyUndoCommand(this, "xRadius", QVariant(m_old_xRadius), QVariant(m_xRadius), undo);
u->setAnimated();
}
if (m_old_yRadius != m_yRadius) {
QPropertyUndoCommand *u = new QPropertyUndoCommand(this, "yRadius", QVariant(m_old_yRadius), QVariant(m_yRadius), undo);
u->setAnimated();
}
elementScene()->undoStack().push(undo);
m_vector_index = -1;
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::addHandler
Add handlers for this item
2020-08-16 11:19:36 +02:00
*/
void PartRectangle::addHandler()
{
2020-10-17 20:25:30 +02:00
if (m_handler_vector.isEmpty() && scene())
{
if (m_resize_mode != 3) {
m_handler_vector = QetGraphicsHandlerItem::handlerForPoint(mapToScene(QetGraphicsHandlerUtility::pointsForRect(m_rect)));
}
else {
m_handler_vector = QetGraphicsHandlerItem::handlerForPoint(mapToScene(QetGraphicsHandlerUtility::pointForRadiusRect(m_rect, m_xRadius, m_yRadius)));
}
for (QetGraphicsHandlerItem *handler : m_handler_vector)
2020-10-17 20:25:30 +02:00
{
QColor color;
if(m_resize_mode == 1) {color = Qt::blue;}
2020-10-17 20:25:30 +02:00
else if (m_resize_mode == 2) {color = Qt::darkGreen;}
else {color = Qt::magenta;}
2020-10-17 20:25:30 +02:00
handler->setColor(color);
scene()->addItem(handler);
handler->installSceneEventFilter(this);
handler->setZValue(this->zValue()+1);
}
}
}
/**
2020-10-17 20:25:30 +02:00
@brief PartRectangle::removeHandler
Remove the handlers of this item
2020-08-16 11:19:36 +02:00
*/
void PartRectangle::removeHandler()
{
2020-10-17 20:25:30 +02:00
if (!m_handler_vector.isEmpty())
{
qDeleteAll(m_handler_vector);
m_handler_vector.clear();
}
}