File indexing completed on 2024-09-08 11:04:05

0001 /***************************************************************************
0002  *   Copyright (C) 2003-2005 by David Saxton                               *
0003  *   david@bluehaze.org                                                    *
0004  *                                                                         *
0005  *   This program is free software; you can redistribute it and/or modify  *
0006  *   it under the terms of the GNU General Public License as published by  *
0007  *   the Free Software Foundation; either version 2 of the License, or     *
0008  *   (at your option) any later version.                                   *
0009  ***************************************************************************/
0010 
0011 #include "fpnode.h"
0012 #include "connector.h"
0013 #include "flowconnector.h"
0014 #include "flowpart.h"
0015 #include "icndocument.h"
0016 
0017 #include <QPainter>
0018 
0019 #include <ktechlab_debug.h>
0020 
0021 FPNode::FPNode(ICNDocument *icnDocument, Node::node_type type, int dir, const QPoint &pos, QString *id)
0022     : Node(icnDocument, type, dir, pos, id)
0023 {
0024     if (icnDocument)
0025         icnDocument->registerItem(this);
0026     m_outputConnector = nullptr;
0027 }
0028 
0029 FPNode::~FPNode()
0030 {
0031 }
0032 
0033 FlowPart *FPNode::outputFlowPart() const
0034 {
0035     // for InputFlowNode this member is overridden
0036 
0037     if (!m_outputConnector)
0038         return nullptr;
0039     if (m_outputConnector->endNode() == nullptr)
0040         return nullptr;
0041 
0042     return (dynamic_cast<FPNode *>(m_outputConnector->endNode()))->outputFlowPart();
0043 }
0044 
0045 FlowPartList FPNode::inputFlowParts() const
0046 {
0047     // for InputFlowNode it's overridden
0048 
0049     FlowPartList list;
0050     FlowPart *flowPart = dynamic_cast<FlowPart *>(parentItem());
0051 
0052     if (flowPart) {
0053         list.append(flowPart);
0054         return list;
0055     }
0056 
0057     const FlowConnectorList::const_iterator end = m_inFlowConnList.end();
0058     for (FlowConnectorList::const_iterator it = m_inFlowConnList.begin(); it != end; ++it) {
0059         if (*it) {
0060             Node *startNode = (*it)->startNode();
0061             FlowPart *flowPart = startNode ? dynamic_cast<FlowPart *>(startNode->parentItem()) : nullptr;
0062             if (flowPart)
0063                 list.append(flowPart);
0064         }
0065     }
0066 
0067     return list;
0068 }
0069 
0070 inline QPolygon arrowPoints(int dir)
0071 {
0072     QPolygon pa(3);
0073     switch (dir) {
0074     case 0:
0075         pa[0] = QPoint(3, 0);
0076         pa[1] = QPoint(0, 2);
0077         pa[2] = QPoint(0, -2);
0078         break;
0079     case 180:
0080         pa[0] = QPoint(-3, 0);
0081         pa[1] = QPoint(0, 2);
0082         pa[2] = QPoint(0, -2);
0083         break;
0084     case 90:
0085         pa[0] = QPoint(2, 0);
0086         pa[1] = QPoint(-2, 0);
0087         pa[2] = QPoint(0, 3);
0088         break;
0089     case 270:
0090         pa[0] = QPoint(2, 0);
0091         pa[1] = QPoint(-2, 0);
0092         pa[2] = QPoint(0, -3);
0093         break;
0094     };
0095     return pa;
0096 }
0097 
0098 void FPNode::addOutputConnector(Connector *const connector)
0099 {
0100     // for Junction and output flownodes
0101     if (!handleNewConnector(connector))
0102         return;
0103 
0104     if (m_outputConnector)
0105         qCCritical(KTL_LOG) << "BUG: adding an output connector when we already have one";
0106 
0107     // FIXME dynamic_cast connector
0108     m_outputConnector = dynamic_cast<FlowConnector *>(connector);
0109 }
0110 
0111 void FPNode::addInputConnector(Connector *const connector)
0112 {
0113     // for Junction and Input flownodes
0114     if (!handleNewConnector(connector))
0115         return;
0116 
0117     // FIXME dynamic_cast connector
0118     m_inFlowConnList.append(dynamic_cast<FlowConnector *>(connector));
0119 }
0120 
0121 bool FPNode::handleNewConnector(Connector *connector)
0122 {
0123     if (!connector)
0124         return false;
0125 
0126     // FIXME dynamic_cast connector
0127     if (m_inFlowConnList.contains(dynamic_cast<FlowConnector *>(connector)) || (static_cast<Connector *>(m_outputConnector) == connector)) {
0128         qCWarning(KTL_LOG) << " Already have connector = " << connector;
0129         return false;
0130     }
0131 
0132     connect(this, &FPNode::removed, connector, &Connector::removeConnector);
0133     connect(connector, &Connector::removed, this, &FPNode::checkForRemoval);
0134     connect(connector, &Connector::selected, this, &FPNode::setNodeSelected);
0135 
0136     if (!isChildNode())
0137         p_icnDocument->slotRequestAssignNG();
0138 
0139     return true;
0140 }
0141 
0142 Connector *FPNode::createInputConnector(Node *startNode)
0143 {
0144     if ((!acceptInput()) || !startNode)
0145         return nullptr;
0146 
0147     // FIXME dynamic_cast used
0148     Connector *connector = new FlowConnector(dynamic_cast<FPNode *>(startNode), dynamic_cast<FPNode *>(this), p_icnDocument);
0149     addInputConnector(connector);
0150 
0151     return connector;
0152 }
0153 
0154 int FPNode::numCon(bool includeParentItem, bool includeHiddenConnectors) const
0155 {
0156     unsigned count = 0;
0157 
0158     FlowConnectorList connectors = m_inFlowConnList;
0159     if (m_outputConnector)
0160         connectors.append(m_outputConnector);
0161 
0162     FlowConnectorList::const_iterator end = connectors.end();
0163     for (FlowConnectorList::const_iterator it = connectors.begin(); it != end; ++it) {
0164         if (*it && (includeHiddenConnectors || (*it)->canvas()))
0165             count++;
0166     }
0167 
0168     if (isChildNode() && includeParentItem)
0169         count++;
0170 
0171     return count;
0172 }
0173 
0174 void FPNode::removeConnector(Connector *connector)
0175 {
0176     if (!connector)
0177         return;
0178 
0179     FlowConnectorList::iterator it;
0180 
0181     // FIXME dynamic_cast connector
0182     it = m_inFlowConnList.find(dynamic_cast<FlowConnector *>(connector));
0183     if (it != m_inFlowConnList.end()) {
0184         (*it)->removeConnectorNoArg();
0185         (*it) = nullptr;
0186     }
0187 
0188     if (static_cast<Connector *>(m_outputConnector) == connector) {
0189         connector->removeConnectorNoArg();
0190         m_outputConnector = nullptr;
0191     }
0192 }
0193 
0194 void FPNode::checkForRemoval(Connector *connector)
0195 {
0196     removeConnector(connector);
0197     setNodeSelected(false);
0198 
0199     removeNullConnectors();
0200 
0201     if (!p_parentItem) {
0202         int conCount = m_inFlowConnList.count();
0203         if (m_outputConnector)
0204             conCount++;
0205         if (conCount < 2)
0206             removeNode();
0207     }
0208     // for JunctionFlowNode this method is overridden!
0209 }
0210 
0211 void FPNode::removeNullConnectors()
0212 {
0213     m_inFlowConnList.remove(static_cast<FlowConnector *>(nullptr));
0214 }
0215 
0216 QPoint FPNode::findConnectorDivergePoint(bool *found)
0217 {
0218     bool temp;
0219     if (!found)
0220         found = &temp;
0221     *found = false;
0222 
0223     if (numCon(false, false) != 2)
0224         return QPoint(0, 0);
0225 
0226     QPointList p1;
0227     QPointList p2;
0228 
0229     int inSize = m_inFlowConnList.count();
0230 
0231     FlowConnectorList connectors = m_inFlowConnList;
0232     if (m_outputConnector)
0233         connectors.append(m_outputConnector);
0234 
0235     const FlowConnectorList::const_iterator end = connectors.end();
0236     bool gotP1 = false;
0237     bool gotP2 = false;
0238     int at = -1;
0239     for (FlowConnectorList::const_iterator it = connectors.begin(); it != end && !gotP2; ++it) {
0240         at++;
0241         if (!(*it) || !(*it)->canvas())
0242             continue;
0243 
0244         if (gotP1) {
0245             p2 = (*it)->connectorPoints(at < inSize);
0246             gotP2 = true;
0247         } else {
0248             p1 = (*it)->connectorPoints(at < inSize);
0249             gotP1 = true;
0250         }
0251     }
0252 
0253     if (!gotP1 || !gotP2)
0254         return QPoint(0, 0);
0255 
0256     unsigned maxLength = p1.size() > p2.size() ? p1.size() : p2.size();
0257 
0258     for (unsigned i = 1; i < maxLength; ++i) {
0259         if (p1[i] != p2[i]) {
0260             *found = true;
0261             return p1[i - 1];
0262         }
0263     }
0264     return QPoint(0, 0);
0265 }
0266 
0267 void FPNode::setVisible(bool yes)
0268 {
0269     if (isVisible() == yes)
0270         return;
0271 
0272     KtlQCanvasPolygon::setVisible(yes);
0273 
0274     const FlowConnectorList::iterator inputEnd = m_inFlowConnList.end();
0275     for (FlowConnectorList::iterator it = m_inFlowConnList.begin(); it != inputEnd; ++it) {
0276         Connector *connector = *it;
0277         if (connector) {
0278             if (isVisible())
0279                 connector->setVisible(true);
0280             else {
0281                 Node *node = connector->startNode();
0282                 connector->setVisible(node && node->isVisible());
0283             }
0284         }
0285     }
0286 
0287     Connector *connector = m_outputConnector;
0288     if (connector) {
0289         if (isVisible())
0290             connector->setVisible(true);
0291         else {
0292             Node *node = connector->endNode();
0293             connector->setVisible(node && node->isVisible());
0294         }
0295     }
0296 }
0297 
0298 bool FPNode::isConnected(Node *node, NodeList *checkedNodes)
0299 {
0300     if (this == node)
0301         return true;
0302 
0303     bool firstNode = !checkedNodes;
0304     if (firstNode)
0305         checkedNodes = new NodeList();
0306 
0307     else if (checkedNodes->contains(this))
0308         return false;
0309 
0310     checkedNodes->append(this);
0311 
0312     const FlowConnectorList::const_iterator inputEnd = m_inFlowConnList.end();
0313     for (FlowConnectorList::const_iterator it = m_inFlowConnList.begin(); it != inputEnd; ++it) {
0314         Connector *connector = *it;
0315         if (connector) {
0316             Node *startNode = connector->startNode();
0317             if (startNode && startNode->isConnected(node, checkedNodes)) {
0318                 if (firstNode) {
0319                     delete checkedNodes;
0320                 }
0321                 return true;
0322             }
0323         }
0324     }
0325 
0326     Connector *connector = m_outputConnector;
0327     if (connector) {
0328         Node *endNode = connector->endNode();
0329         if (endNode && endNode->isConnected(node, checkedNodes)) {
0330             if (firstNode) {
0331                 delete checkedNodes;
0332             }
0333             return true;
0334         }
0335     }
0336 
0337     if (firstNode) {
0338         delete checkedNodes;
0339     }
0340 
0341     return false;
0342 }
0343 
0344 ConnectorList FPNode::inputConnectorList() const
0345 {
0346     return static_cast<ConnectorList>(static_cast<FlowConnectorList>(m_inFlowConnList));
0347 }
0348 
0349 ConnectorList FPNode::outputConnectorList() const
0350 {
0351     ConnectorList out;
0352     if (m_outputConnector)
0353         out.append(static_cast<Connector *>(m_outputConnector)); // un upcast between downcasts :o
0354     return out;
0355 }
0356 
0357 ConnectorList FPNode::getAllConnectors() const
0358 {
0359     ConnectorList all = static_cast<ConnectorList>(static_cast<FlowConnectorList>(m_inFlowConnList));
0360     if (m_outputConnector)
0361         all.append(static_cast<Connector *>(m_outputConnector));
0362     return all;
0363 }
0364 
0365 Connector *FPNode::getAConnector() const
0366 {
0367     if (!m_inFlowConnList.isEmpty())
0368         return *m_inFlowConnList.begin();
0369 
0370     if (m_outputConnector)
0371         return m_outputConnector;
0372 
0373     return nullptr;
0374 }
0375 
0376 #include "moc_fpnode.cpp"