/* 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 "conductorproperties.h" #include "qetapp.h" /** Constructeur par defaut */ SingleLineProperties::SingleLineProperties() : hasGround(true), hasNeutral(true), is_pen(false), phases(1) { } /// Destructeur SingleLineProperties::~SingleLineProperties() { } /** Definit le nombre de phases (0, 1, 2, ou 3) @param n Nombre de phases */ void SingleLineProperties::setPhasesCount(int n) { phases = qBound(0, n, 3); } /// @return le nombre de phases (0, 1, 2, ou 3) unsigned short int SingleLineProperties::phasesCount() { return(phases); } /** @return true if the singleline conductor should be drawn using the PEN (Protective Earth Neutral) representation and if it features the ground and the neutral. */ bool SingleLineProperties::isPen() const { return(hasNeutral && hasGround && is_pen); } /** Dessine les symboles propres a un conducteur unifilaire @param painter QPainter a utiliser pour dessiner les symboles @param direction direction du segment sur lequel les symboles apparaitront @param rect rectangle englobant le dessin ; utilise pour specifier a la fois la position et la taille du dessin */ void SingleLineProperties::draw(QPainter *painter, QET::ConductorSegmentType direction, const QRectF &rect) { // s'il n'y a rien a dessiner, on retourne immediatement if (!hasNeutral && !hasGround && !phases) return; // prepare le QPainter painter -> save(); QPen pen(painter -> pen()); pen.setCapStyle(Qt::FlatCap); pen.setJoinStyle(Qt::MiterJoin); pen.setStyle(Qt::SolidLine); painter -> setPen(pen); painter -> setRenderHint(QPainter::Antialiasing, true); uint symbols_count = (hasNeutral ? 1 : 0) + (hasGround ? 1 : 0) - (isPen() ? 1 : 0) + phases; qreal interleave_base = (direction == QET::Horizontal ? rect.width() : rect.height()); qreal interleave = interleave_base / (symbols_count + 1);; qreal symbol_width = interleave_base / 12; for (uint i = 1 ; i <= symbols_count ; ++ i) { // dessine le tronc du symbole QPointF symbol_p1, symbol_p2; if (direction == QET::Horizontal) { symbol_p1 = QPointF(rect.x() + (i * interleave) + symbol_width, rect.y() + rect.height() * 0.75); symbol_p2 = QPointF(rect.x() + (i * interleave) - symbol_width, rect.y() + rect.height() * 0.25); } else { symbol_p2 = QPointF(rect.x() + rect.width() * 0.75, rect.y() + (i * interleave) - symbol_width); symbol_p1 = QPointF(rect.x() + rect.width() * 0.25, rect.y() + (i * interleave) + symbol_width); } painter -> drawLine(QLineF(symbol_p1, symbol_p2)); // dessine le reste des symboles terre et neutre if (isPen()) { if (i == 1) { drawPen(painter, direction, symbol_p2, symbol_width); } } else { if (hasGround && i == 1) { drawGround(painter, direction, symbol_p2, symbol_width * 2.0); } else if (hasNeutral && ((i == 1 && !hasGround) || (i == 2 && hasGround))) { drawNeutral(painter, direction, symbol_p2, symbol_width * 1.5); } } } painter -> restore(); } /** Dessine le segment correspondant au symbole de la terre sur un conducteur unifilaire @param painter QPainter a utiliser pour dessiner le segment @param direction direction du segment sur lequel le symbole apparaitra @param center centre du segment @param size taille du segment */ void SingleLineProperties::drawGround(QPainter *painter, QET::ConductorSegmentType direction, QPointF center, qreal size) { painter -> save(); // prepare le QPainter painter -> setRenderHint(QPainter::Antialiasing, false); QPen pen2(painter -> pen()); pen2.setCapStyle(Qt::SquareCap); painter -> setPen(pen2); // dessine le segment representant la terre qreal half_size = size / 2.0; QPointF offset_point( (direction == QET::Horizontal) ? half_size : 0.0, (direction == QET::Horizontal) ? 0.0 : half_size ); painter -> drawLine( QLineF( center + offset_point, center - offset_point ) ); painter -> restore(); } /** Dessine le cercle correspondant au symbole du neutre sur un conducteur unifilaire @param painter QPainter a utiliser pour dessiner le segment @param direction direction du segment sur lequel le symbole apparaitra @param center centre du cercle @param size diametre du cercle */ void SingleLineProperties::drawNeutral(QPainter *painter, QET::ConductorSegmentType direction, QPointF center, qreal size) { Q_UNUSED(direction); painter -> save(); // prepare le QPainter if (painter -> brush() == Qt::NoBrush) painter -> setBrush(Qt::black); painter -> setPen(Qt::NoPen); // desine le cercle representant le neutre painter -> drawEllipse( QRectF( center - QPointF(size / 2.0, size / 2.0), QSizeF(size, size) ) ); painter -> restore(); } /** Draw the PEN (Protective Earth Neutral) symbol using \a painter at position \a center, using a size hint of \a size. @param direction Indicate the direction of the underlying conductor segment */ void SingleLineProperties::drawPen(QPainter *painter, QET::ConductorSegmentType direction, QPointF center, qreal size) { painter -> save(); //painter -> setBrush(Qt::white); // desine le cercle representant le neutre //painter -> drawEllipse( // QRectF( // center - QPointF(size * 1.5 / 2.0, size * 1.5 / 2.0), // QSizeF(size * 1.5, size * 1.5) // ) //); drawNeutral(painter, direction, center, size * 1.5); int offset = (size * 1.5 / 2.0); QPointF pos = center + (direction == QET::Horizontal ? QPointF(0.0, -offset - 0.5) : QPointF(offset + 0.5, 0.0)); drawGround(painter, direction, pos, 2.0 * size); painter -> restore(); } /** Exporte les parametres du conducteur unifilaire sous formes d'attributs XML ajoutes a l'element e. @param e Element XML auquel seront ajoutes des attributs */ void SingleLineProperties::toXml(QDomElement &e) const { e.setAttribute("ground", hasGround ? "true" : "false"); e.setAttribute("neutral", hasNeutral ? "true" : "false"); e.setAttribute("phase", phases); if (isPen()) e.setAttribute("pen", "true"); } /** Importe les parametres du conducteur unifilaire a partir des attributs XML de l'element e @param e Element XML dont les attributs seront lus */ void SingleLineProperties::fromXml(QDomElement &e) { hasGround = e.attribute("ground") == "true"; hasNeutral = e.attribute("neutral") == "true"; setPhasesCount(e.attribute("phase").toInt()); is_pen = (hasGround && hasNeutral && e.attribute("pen", "false") == "true"); } /** Constructeur : par defaut, les proprietes font un conducteur multifilaire noir dont le texte est "_" */ ConductorProperties::ConductorProperties() : type(Multi), color(Qt::black), text("_"), text_size(9), verti_rotate_text(270), horiz_rotate_text(0), m_show_text(true), m_one_text_per_folio(false), style(Qt::SolidLine) {} /** Destructeur */ ConductorProperties::~ConductorProperties() { } /** * @brief ConductorProperties::toXml * Export conductor propertie, in the XML element 'e' * @param e the xml element */ void ConductorProperties::toXml(QDomElement &e) const { e.setAttribute("type", typeToString(type)); if (color != QColor(Qt::black)) e.setAttribute("color", color.name()); if (type == Single) singleLineProperties.toXml(e); e.setAttribute("num", text); e.setAttribute("function", m_function); e.setAttribute("tension-protocol", m_tension_protocol); e.setAttribute("numsize", text_size); e.setAttribute("displaytext", m_show_text); e.setAttribute("onetextperfolio", m_one_text_per_folio); e.setAttribute("vertirotatetext", verti_rotate_text); e.setAttribute("horizrotatetext", horiz_rotate_text); QString conductor_style = writeStyle(); if (!conductor_style.isEmpty()) e.setAttribute("style", conductor_style); } /** * @brief ConductorProperties::fromXml * Import conductor propertie, from the attribute of the xml element 'e' * @param e the xml document */ void ConductorProperties::fromXml(QDomElement &e) { // get conductor color QColor xml_color= QColor(e.attribute("color")); color = (xml_color.isValid()? xml_color : QColor(Qt::black)); // read style of conductor readStyle(e.attribute("style")); if (e.attribute("type") == typeToString(Single)) { // get specific properties for single conductor singleLineProperties.fromXml(e); type = Single; } else type = Multi; text = e.attribute("num"); m_function = e.attribute("function"); m_tension_protocol = e.attribute("tension-protocol"); text_size = e.attribute("numsize", QString::number(9)).toInt(); m_show_text = e.attribute("displaytext", QString::number(1)).toInt(); m_one_text_per_folio = e.attribute("onetextperfolio", QString::number(0)).toInt(); verti_rotate_text = e.attribute("vertirotatetext").toDouble(); horiz_rotate_text = e.attribute("horizrotatetext").toDouble(); //Keep retrocompatible with version older than 0,4 //If the propertie @type is simple (removed since QET 0,4), we set text no visible. //@TODO remove this code for qet 0.6 or later if (e.attribute("type") == "simple") m_show_text = false; } /** @param settings Parametres a ecrire @param prefix prefixe a ajouter devant les noms des parametres */ void ConductorProperties::toSettings(QSettings &settings, const QString &prefix) const { settings.setValue(prefix + "color", color.name()); settings.setValue(prefix + "style", writeStyle()); settings.setValue(prefix + "type", typeToString(type)); settings.setValue(prefix + "text", text); settings.setValue(prefix + "function", m_function); settings.setValue(prefix + "tension-protocol", m_tension_protocol); settings.setValue(prefix + "textsize", QString::number(text_size)); settings.setValue(prefix + "displaytext", m_show_text); settings.setValue(prefix + "onetextperfolio", m_one_text_per_folio); settings.setValue(prefix + "vertirotatetext", QString::number(verti_rotate_text)); settings.setValue(prefix + "horizrotatetext", QString::number(horiz_rotate_text)); singleLineProperties.toSettings(settings, prefix); } /** @param settings Parametres a lire @param prefix prefixe a ajouter devant les noms des parametres */ void ConductorProperties::fromSettings(QSettings &settings, const QString &prefix) { QColor settings_color = QColor(settings.value(prefix + "color").toString()); color = (settings_color.isValid()? settings_color : QColor(Qt::black)); QString setting_type = settings.value(prefix + "type", typeToString(Multi)).toString(); type = (setting_type == typeToString(Single)? Single : Multi); singleLineProperties.fromSettings(settings, prefix); text = settings.value(prefix + "text", "_").toString(); m_function = settings.value(prefix + "function", "").toString(); m_tension_protocol = settings.value(prefix + "tension-protocol", "").toString(); text_size = settings.value(prefix + "textsize", "7").toInt(); m_show_text = settings.value(prefix + "displaytext", true).toBool(); m_one_text_per_folio = settings.value(prefix + "onetextperfolio", false).toBool(); verti_rotate_text = settings.value((prefix + "vertirotatetext"), "270").toDouble(); horiz_rotate_text = settings.value((prefix + "horizrotatetext"), "0").toDouble(); readStyle(settings.value(prefix + "style").toString()); } /** @param t type du conducteur */ QString ConductorProperties::typeToString(ConductorType t) { switch(t) { case Single: return("single"); case Multi: return("multi"); default: return(QString()); } } /** * @brief ConductorProperties::defaultProperties * @return the default properties stored in the setting file */ ConductorProperties ConductorProperties::defaultProperties() { QSettings &settings = QETApp::settings(); ConductorProperties def; def.fromSettings(settings, "diagrameditor/defaultconductor"); return(def); } /** @param other l'autre ensemble de proprietes avec lequel il faut effectuer la comparaison @return true si les deux ensembles de proprietes sont identiques, false sinon */ bool ConductorProperties::operator==(const ConductorProperties &other) const{ return( other.type == type &&\ other.color == color &&\ other.style == style &&\ other.text == text &&\ other.m_function == m_function &&\ other.m_tension_protocol == m_tension_protocol &&\ other.m_show_text == m_show_text &&\ other.text_size == text_size &&\ other.verti_rotate_text == verti_rotate_text &&\ other.horiz_rotate_text == horiz_rotate_text &&\ other.singleLineProperties == singleLineProperties &&\ other.m_one_text_per_folio == m_one_text_per_folio ); } /** @param other l'autre ensemble de proprietes avec lequel il faut effectuer la comparaison @return true si les deux ensembles de proprietes sont differents, false sinon */ bool ConductorProperties::operator!=(const ConductorProperties &other) const{ return(!(*this == other)); } /** Applique les styles passes en parametre dans cet objet @param style_string Chaine decrivant le style du conducteur */ void ConductorProperties::readStyle(const QString &style_string) { style = Qt::SolidLine; // style par defaut if (style_string.isEmpty()) return; // recupere la liste des couples style / valeur QStringList styles = style_string.split(";", QString::SkipEmptyParts); QRegExp rx("^\\s*([a-z-]+)\\s*:\\s*([a-z-]+)\\s*$"); foreach (QString style_str, styles) { if (rx.exactMatch(style_str)) { QString style_name = rx.cap(1); QString style_value = rx.cap(2); if (style_name == "line-style") { if (style_value == "dashed") style = Qt::DashLine; else if (style_value == "dashdotted") style = Qt::DashDotLine; else if (style_value == "normal") style = Qt::SolidLine; } } } } /** Exporte le style du conducteur sous forme d'une chaine de caracteres @return une chaine de caracteres decrivant le style du conducteur */ QString ConductorProperties::writeStyle() const { if (style == Qt::DashLine) { return("line-style: dashed;"); } else if (style == Qt::DashDotLine) { return("line-style: dashdotted"); } else { return(QString()); } } /** @param other l'autre ensemble de proprietes avec lequel il faut effectuer la comparaison @return true si les deux ensembles de proprietes sont identiques, false sinon */ int SingleLineProperties::operator==(const SingleLineProperties &other) const { return( other.hasGround == hasGround &&\ other.hasNeutral == hasNeutral &&\ other.is_pen == is_pen &&\ other.phases == phases ); } /** @param other l'autre ensemble de proprietes avec lequel il faut effectuer la comparaison @return true si les deux ensembles de proprietes sont differents, false sinon */ int SingleLineProperties::operator!=(const SingleLineProperties &other) const { return(!(other == (*this))); } /** @param settings Parametres a ecrire @param prefix prefix a ajouter devant les noms des parametres */ void SingleLineProperties::toSettings(QSettings &settings, const QString &prefix) const { settings.setValue(prefix + "hasGround", hasGround); settings.setValue(prefix + "hasNeutral", hasNeutral); settings.setValue(prefix + "phases", phases); settings.setValue(prefix + "pen", is_pen); } /** @param settings Parametres a lire @param prefix prefix a ajouter devant les noms des parametres */ void SingleLineProperties::fromSettings(QSettings &settings, const QString &prefix) { hasGround = settings.value(prefix + "hasGround", true).toBool(); hasNeutral = settings.value(prefix + "hasNeutral", true).toBool(); phases = settings.value(prefix + "phases", 1).toInt(); is_pen = settings.value(prefix + "pen", false).toBool(); }