mirror of
https://github.com/qelectrotech/qelectrotech-source-mirror.git
synced 2025-09-13 20:23:04 +02:00
git-svn-id: svn+ssh://svn.tuxfamily.org/svnroot/qet/qet/trunk@4091 bfdf4180-ca20-0410-9c96-a3a8aa849046
404 lines
12 KiB
C++
404 lines
12 KiB
C++
/*
|
|
Copyright 2006-2013 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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#include "diagramtextitem.h"
|
|
#include "diagramcommands.h"
|
|
#include "qetapp.h"
|
|
#include "richtext/richtexteditor_p.h"
|
|
#include "diagram.h"
|
|
|
|
/**
|
|
* @brief DiagramTextItem::DiagramTextItem
|
|
* @param parent : parent item
|
|
*/
|
|
DiagramTextItem::DiagramTextItem(QGraphicsItem *parent) :
|
|
QGraphicsTextItem(parent),
|
|
m_mouse_hover(false),
|
|
previous_text_(),
|
|
rotation_angle_(0.0),
|
|
m_first_move (true)
|
|
{ build(); }
|
|
|
|
/**
|
|
* @brief DiagramTextItem::DiagramTextItem
|
|
* @param text : text to display
|
|
* @param parent : parent item
|
|
*/
|
|
DiagramTextItem::DiagramTextItem(const QString &text, QGraphicsItem *parent) :
|
|
QGraphicsTextItem(text, parent),
|
|
m_mouse_hover(false),
|
|
previous_text_(text),
|
|
rotation_angle_(0.0)
|
|
{ build(); }
|
|
|
|
/**
|
|
* @brief DiagramTextItem::build
|
|
* Build this item with default value
|
|
*/
|
|
void DiagramTextItem::build()
|
|
{
|
|
//set Zvalue at 10 to be upper than the DiagramImageItem
|
|
setZValue(10);
|
|
setAcceptHoverEvents(true);
|
|
setDefaultTextColor(Qt::black);
|
|
setFont(QETApp::diagramTextsFont());
|
|
setFlags(QGraphicsItem::ItemIsSelectable|QGraphicsItem::ItemIsMovable | QGraphicsItem::ItemSendsGeometryChanges);
|
|
setNoEditable(false);
|
|
setToolTip(tr("Maintenir ctrl pour un déplacement libre"));
|
|
}
|
|
|
|
/**
|
|
* @brief DiagramTextItem::diagram
|
|
* @return The diagram of this item or 0 if this text isn't in a diagram
|
|
*/
|
|
Diagram *DiagramTextItem::diagram() const {
|
|
return(qobject_cast<Diagram *>(scene()));
|
|
}
|
|
|
|
/**
|
|
* @brief DiagramTextItem::toXml
|
|
* This method do nothing and return an empty DomElement
|
|
* This is used to be inherited by child class
|
|
* @return
|
|
*/
|
|
QDomElement DiagramTextItem::toXml(QDomDocument &) const {
|
|
return QDomElement();
|
|
}
|
|
|
|
/**
|
|
@return l'angle de rotation actuel de ce texte
|
|
*/
|
|
qreal DiagramTextItem::rotationAngle() const {
|
|
return(rotation_angle_);
|
|
}
|
|
|
|
/**
|
|
Permet de tourner le texte a un angle donne de maniere absolue.
|
|
Un angle de 0 degres correspond a un texte horizontal non retourne.
|
|
@param rotation Nouvel angle de rotation de ce texte
|
|
@see applyRotation
|
|
*/
|
|
void DiagramTextItem::setRotationAngle(const qreal &rotation) {
|
|
qreal applied_rotation = QET::correctAngle(rotation);
|
|
applyRotation(applied_rotation - rotation_angle_);
|
|
rotation_angle_ = applied_rotation;
|
|
}
|
|
|
|
/**
|
|
Permet de tourner le texte de maniere relative.
|
|
L'angle added_rotation est ajoute a l'orientation actuelle du texte.
|
|
@param added_rotation Angle a ajouter a la rotation actuelle
|
|
@see applyRotation
|
|
*/
|
|
void DiagramTextItem::rotateBy(const qreal &added_rotation) {
|
|
qreal applied_added_rotation = QET::correctAngle(added_rotation);
|
|
rotation_angle_ = QET::correctAngle(rotation_angle_ + applied_added_rotation);
|
|
applyRotation(applied_added_rotation);
|
|
}
|
|
|
|
/**
|
|
Traduit en coordonnees de la scene un mouvement / vecteur initialement
|
|
exprime en coordonnees locales.
|
|
@param movement Vecteur exprime en coordonnees locales
|
|
@return le meme vecteur, exprime en coordonnees de la scene
|
|
*/
|
|
QPointF DiagramTextItem::mapMovementToScene(const QPointF &movement) const {
|
|
// on definit deux points en coordonnees locales
|
|
QPointF local_origin(0.0, 0.0);
|
|
QPointF local_movement_point(movement);
|
|
|
|
// on les mappe sur la scene
|
|
QPointF scene_origin(mapToScene(local_origin));
|
|
QPointF scene_movement_point(mapToScene(local_movement_point));
|
|
|
|
// on calcule le vecteur represente par ces deux points
|
|
return(scene_movement_point - scene_origin);
|
|
}
|
|
|
|
/**
|
|
Traduit en coordonnees locales un mouvement / vecteur initialement
|
|
exprime en coordonnees de la scene.
|
|
@param movement Vecteur exprime en coordonnees de la scene
|
|
@return le meme vecteur, exprime en coordonnees locales
|
|
*/
|
|
QPointF DiagramTextItem::mapMovementFromScene(const QPointF &movement) const {
|
|
// on definit deux points sur la scene
|
|
QPointF scene_origin(0.0, 0.0);
|
|
QPointF scene_movement_point(movement);
|
|
|
|
// on les mappe sur ce QGraphicsItem
|
|
QPointF local_origin(mapFromScene(scene_origin));
|
|
QPointF local_movement_point(mapFromScene(scene_movement_point));
|
|
|
|
// on calcule le vecteur represente par ces deux points
|
|
return(local_movement_point - local_origin);
|
|
}
|
|
|
|
/**
|
|
Traduit en coordonnees de l'item parent un mouvement / vecteur initialement
|
|
exprime en coordonnees locales.
|
|
@param movement Vecteur exprime en coordonnees locales
|
|
@return le meme vecteur, exprime en coordonnees du parent
|
|
*/
|
|
QPointF DiagramTextItem::mapMovementToParent(const QPointF &movement) const {
|
|
// on definit deux points en coordonnees locales
|
|
QPointF local_origin(0.0, 0.0);
|
|
QPointF local_movement_point(movement);
|
|
|
|
// on les mappe sur la scene
|
|
QPointF parent_origin(mapToParent(local_origin));
|
|
QPointF parent_movement_point(mapToParent(local_movement_point));
|
|
|
|
// on calcule le vecteur represente par ces deux points
|
|
return(parent_movement_point - parent_origin);
|
|
}
|
|
|
|
/**
|
|
Traduit en coordonnees locales un mouvement / vecteur initialement
|
|
exprime en coordonnees du parent.
|
|
@param movement Vecteur exprime en coordonnees du parent
|
|
@return le meme vecteur, exprime en coordonnees locales
|
|
*/
|
|
QPointF DiagramTextItem::mapMovementFromParent(const QPointF &movement) const {
|
|
// on definit deux points sur le parent
|
|
QPointF parent_origin(0.0, 0.0);
|
|
QPointF parent_movement_point(movement);
|
|
|
|
// on les mappe sur ce QGraphicsItem
|
|
QPointF local_origin(mapFromParent(parent_origin));
|
|
QPointF local_movement_point(mapFromParent(parent_movement_point));
|
|
|
|
// on calcule le vecteur represente par ces deux points
|
|
return(local_movement_point - local_origin);
|
|
}
|
|
|
|
void DiagramTextItem::setFontSize(int &s) {
|
|
setFont(QETApp::diagramTextsFont(s));
|
|
}
|
|
|
|
/**
|
|
* @brief DiagramTextItem::paint
|
|
* Draw this text field. This method draw the text by calling QGraphicsTextItem::paint.
|
|
* If text is hovered, this method draw the bounding rect in grey
|
|
* @param painter : painter to use
|
|
* @param option : style option
|
|
* @param widget : widget where must to draw
|
|
*/
|
|
void DiagramTextItem::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
|
|
{
|
|
painter -> setRenderHint(QPainter::Antialiasing, false);
|
|
QGraphicsTextItem::paint(painter, option, widget);
|
|
|
|
if (m_mouse_hover && !isSelected())
|
|
{
|
|
painter -> save();
|
|
|
|
//Disable renderhints
|
|
painter -> setRenderHint(QPainter::Antialiasing, false);
|
|
painter -> setRenderHint(QPainter::TextAntialiasing, false);
|
|
painter -> setRenderHint(QPainter::SmoothPixmapTransform, false);
|
|
|
|
//Draw the selected rect in grey
|
|
QPen t;
|
|
t.setColor(Qt::gray);
|
|
t.setStyle(Qt::DashDotLine);
|
|
t.setCosmetic(true);
|
|
painter -> setPen(t);
|
|
painter -> drawRoundRect(boundingRect(), 10, 10);
|
|
|
|
painter -> restore();
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
Gere la prise de focus du champ de texte
|
|
@param e Objet decrivant la prise de focus
|
|
*/
|
|
void DiagramTextItem::focusInEvent(QFocusEvent *e) {
|
|
QGraphicsTextItem::focusInEvent(e);
|
|
|
|
// empeche le deplacement du texte pendant son edition
|
|
setFlag(QGraphicsItem::ItemIsMovable, false);
|
|
|
|
// memorise le texte avant que l'utilisateur n'y touche
|
|
previous_text_ = toHtml();
|
|
// cela permettra de determiner si l'utilisateur a modifie le texte a la fin de l'edition
|
|
}
|
|
|
|
/**
|
|
Gere la perte de focus du champ de texte
|
|
@param e Objet decrivant la perte de focus
|
|
*/
|
|
void DiagramTextItem::focusOutEvent(QFocusEvent *e) {
|
|
QGraphicsTextItem::focusOutEvent(e);
|
|
|
|
// signale la modification du texte si besoin
|
|
if (toPlainText() != previous_text_) {
|
|
emit(diagramTextChanged(this, previous_text_, toHtml()));
|
|
previous_text_ = toHtml();
|
|
}
|
|
|
|
// deselectionne le texte
|
|
QTextCursor cursor = textCursor();
|
|
cursor.clearSelection();
|
|
setTextCursor(cursor);
|
|
|
|
// hack a la con pour etre re-entrant
|
|
setTextInteractionFlags(Qt::NoTextInteraction);
|
|
|
|
// autorise de nouveau le deplacement du texte
|
|
setFlag(QGraphicsItem::ItemIsMovable, true);
|
|
setFlag(QGraphicsTextItem::ItemIsFocusable, false);
|
|
}
|
|
|
|
/**
|
|
Gere les double-clics sur ce champ de texte.
|
|
@param event un QGraphicsSceneMouseEvent decrivant le double-clic
|
|
*/
|
|
void DiagramTextItem::mouseDoubleClickEvent(QGraphicsSceneMouseEvent *event) {
|
|
if (!(textInteractionFlags() & Qt::TextEditable) && !no_editable) {
|
|
// rend le champ de texte editable
|
|
setTextInteractionFlags(Qt::TextEditorInteraction);
|
|
|
|
// edite le champ de texte
|
|
setFocus(Qt::MouseFocusReason);
|
|
} else {
|
|
QGraphicsTextItem::mouseDoubleClickEvent(event);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief DiagramTextItem::mousePressEvent
|
|
* @param event
|
|
*/
|
|
void DiagramTextItem::mousePressEvent (QGraphicsSceneMouseEvent *event) {
|
|
m_first_move = true;
|
|
if (event -> modifiers() & Qt::ControlModifier) {
|
|
setSelected(!isSelected());
|
|
}
|
|
//Save the pos of item at the beggining of the movement
|
|
m_mouse_to_origin_movement = pos() - event->scenePos();
|
|
QGraphicsTextItem::mousePressEvent(event);
|
|
}
|
|
|
|
/**
|
|
* @brief DiagramTextItem::mouseMoveEvent
|
|
* @param event
|
|
*/
|
|
void DiagramTextItem::mouseMoveEvent(QGraphicsSceneMouseEvent *event) {
|
|
if (textInteractionFlags() & Qt::TextEditable) QGraphicsTextItem::mouseMoveEvent(event);
|
|
|
|
else if ((flags() & QGraphicsItem::ItemIsMovable) && (event -> buttons() & Qt::LeftButton)) {
|
|
Diagram *diagram_ = diagram();
|
|
|
|
//This is first move, we signal it to parent diagram
|
|
if(diagram_ && m_first_move)
|
|
diagram_ -> beginMoveElements(this);
|
|
|
|
QPointF old_pos = pos();
|
|
|
|
//Set the actual pos
|
|
QPointF new_pos = event->scenePos() + m_mouse_to_origin_movement;
|
|
event->modifiers() == Qt::ControlModifier ? setPos(new_pos) : setPos(Diagram::snapToGrid(new_pos));
|
|
|
|
|
|
//Update the actual movement for other selected item
|
|
if (diagram_)
|
|
diagram_ -> continueMoveElements(pos() - old_pos);
|
|
}
|
|
|
|
else event -> ignore();
|
|
|
|
m_first_move = false;
|
|
}
|
|
|
|
/**
|
|
* @brief DiagramTextItem::mouseReleaseEvent
|
|
* @param event
|
|
*/
|
|
void DiagramTextItem::mouseReleaseEvent (QGraphicsSceneMouseEvent *event) {
|
|
//Signal to diagram movement is finish
|
|
if (diagram())
|
|
diagram() -> endMoveElements();
|
|
|
|
if (!(event -> modifiers() & Qt::ControlModifier))
|
|
QGraphicsTextItem::mouseReleaseEvent(event);
|
|
}
|
|
|
|
/**
|
|
Effectue la rotation du texte en elle-meme
|
|
Pour les DiagramTextItem, la rotation s'effectue autour du point (0, 0).
|
|
Cette methode peut toutefois etre redefinie dans des classes filles
|
|
@param angle Angle de la rotation a effectuer
|
|
*/
|
|
void DiagramTextItem::applyRotation(const qreal &angle) {
|
|
setRotation(QET::correctAngle(rotation()+angle));
|
|
}
|
|
|
|
/**
|
|
* @brief Edit the text with HtmlEditor
|
|
*/
|
|
void DiagramTextItem::edit()
|
|
{
|
|
QWidget *parent = nullptr;
|
|
if (scene() && scene()->views().size())
|
|
parent = scene()->views().first();
|
|
|
|
qdesigner_internal::RichTextEditorDialog editor(parent);
|
|
connect(&editor, &qdesigner_internal::RichTextEditorDialog::applyEditText, [this](QString text) {this->setHtml(text);});
|
|
editor.setText(toHtml());
|
|
editor.exec();
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
When mouse over element
|
|
change m_mouse_hover to true (used in paint() function )
|
|
@param e QGraphicsSceneHoverEvent
|
|
*/
|
|
void DiagramTextItem::hoverEnterEvent(QGraphicsSceneHoverEvent *e) {
|
|
Q_UNUSED(e);
|
|
|
|
m_mouse_hover = true;
|
|
QString str_ToolTip = toPlainText();
|
|
setToolTip( str_ToolTip );
|
|
update();
|
|
}
|
|
|
|
/**
|
|
When mouse over element leave the position
|
|
change m_mouse_hover to false (used in paint() function )
|
|
@param e QGraphicsSceneHoverEvent
|
|
*/
|
|
void DiagramTextItem::hoverLeaveEvent(QGraphicsSceneHoverEvent *e) {
|
|
Q_UNUSED(e);
|
|
//qDebug() << "Leave mouse over";
|
|
m_mouse_hover = false;
|
|
update();
|
|
}
|
|
|
|
/**
|
|
Do nothing default function .
|
|
@param e QGraphicsSceneHoverEvent
|
|
*/
|
|
void DiagramTextItem::hoverMoveEvent(QGraphicsSceneHoverEvent *e) {
|
|
Q_UNUSED(e);
|
|
QGraphicsTextItem::hoverMoveEvent(e);
|
|
}
|