QAlgorithm
 All Classes Files Functions Typedefs Properties Macros
QAlgorithm.cpp
Go to the documentation of this file.
1 // QAlgorithm: a class for Qt/C++ implementing generic algorithm logic.
2 // Copyright (C) 2018 Filippo Santarelli
3 //
4 // This program is free software: you can redistribute it and/or modify
5 // it under the terms of the GNU Lesser General Public License as published by
6 // the Free Software Foundation, either version 3 of the License, or
7 // (at your option) any later version.
8 //
9 // This program is distributed in the hope that it will be useful,
10 // but WITHOUT ANY WARRANTY; without even the implied warranty of
11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 // GNU Lesser General Public License for more details.
13 //
14 // You should have received a copy of the GNU Lesser General Public License
15 // along with this program. If not, see <http://www.gnu.org/licenses/>.
16 //
17 // Contact me at: filippo2.santarelli@gmail.com
18 //
19 
20 #include "QAlgorithm.h"
21 
22 quint32 QAlgorithm::print_counter = 1;
23 
25 {
26  return finished;
27 }
28 
30 {
31  return started;
32 }
33 
35 {
36  started = true;
37  Q_EMIT justStarted();
38 }
39 
41 {
42  finished = true;
43  Q_EMIT justFinished();
44 }
45 
47 {
48  return !getAncestors().values().contains(false);
49 }
50 
52 {
53  return ancestors;
54 }
55 
57 {
58  return descendants;
59 }
60 
62 {
63  foreach(auto& shr_ancestor, getAncestors().keys())
64  {
65  if (shr_ancestor == ancestor)
66  {
67  return shr_ancestor;
68  }
69  }
70  return QAShrAlgorithm();
71 }
72 
74 {
75  return findAncestor(ancestor.data());
76 }
77 
79 {
80  foreach(auto& shr_descendant, getDescendants().keys())
81  {
82  if (shr_descendant == descendant)
83  {
84  return shr_descendant;
85  }
86  }
87  return QAShrAlgorithm();
88 }
89 
91 {
92  return findDescendant(descendant.data());
93 }
94 
96 {
97  // Check among the descendants
98  foreach(auto descendant, getDescendants().keys())
99  {
100  auto shr_this = descendant->findAncestor(this);
101  if(!shr_this.isNull()) return shr_this;
102  }
103  // Otherwise check among the ancestors
104  foreach(auto ancestor, getAncestors().keys())
105  {
106  auto shr_this = ancestor->findDescendant(this);
107  if(!shr_this.isNull()) return shr_this;
108  }
109  // If nothing was found return null shared pointer
110  return QAShrAlgorithm();
111 }
112 
113 QAlgorithm::QAlgorithm(QObject* parent) : QObject(parent), QRunnable()
114 {
115  qRegisterMetaType<QAPropertyMap>();
116  qRegisterMetaType<QAPropagationRules>();
117  qRegisterMetaType<QAShrAlgorithm>();
118 }
119 
121 {
122  // Scan the object properties
123  for(const QString& PropName: parameters.keys())
124  {
125  bool set = false;
126  for(int k = 0; k < metaObject()->propertyCount(); ++k)
127  {
128  if(metaObject()->property(k).name() == QA_PAR + PropName ||
129  metaObject()->property(k).name() == QA_IN + PropName)
130  {
131  // This property is a parameter or an input
132  // Write the desired value in the property
133  if (!setProperty(metaObject()->property(k).name(), parameters.value(PropName)))
134  {
135  qWarning() << "Cannot set parameter/input" << PropName;
136  }
137  set = true;
138  }
139  }
140  if(!set) qWarning() << "Trying to set" << PropName << "but it is not among object's properties";
141  }
142 }
143 
145 {
146  // Prevent the QThreadPool to delete a parent instance
147  setAutoDelete(false);
148  // Make internal connections
149  connect(this, &QAlgorithm::justFinished, this, &QAlgorithm::propagateExecution, Qt::AutoConnection);
150  connect(&watcher, &QFutureWatcher<void>::finished, this, &QAlgorithm::setFinished, Qt::AutoConnection);
151 }
152 
154 {
155  auto shr_this = findSharedThis();
156  if(!shr_this.isNull())
157  {
158  // Notify ancestors
159  foreach(auto ancestor, getAncestors().keys()) ancestor->descendants[shr_this] = true;
160  // Notify descendants, transfer output to and execute them
161  foreach(auto descendant, getDescendants().keys())
162  {
163  descendant->ancestors[shr_this] = true;
164  descendant->getInput(shr_this);
165  if(!descendant->getKeepInput())
166  {
167  QAlgorithm::closeConnection(shr_this, descendant);
168  // Set each input property to null
169  // useful if input has been received with implicit sharing
170  for(int k = 0; k < metaObject()->propertyCount(); ++k)
171  {
172  if(QString(metaObject()->property(k).name()).startsWith(QA_IN))
173  {
174  setProperty(metaObject()->property(k).name(), QVariant());
175  }
176  }
177  }
178  if(!descendant->isStarted())
179  {
180  if (getParallelExecution()) descendant->parallelExecution();
181  else descendant->serialExecution();
182  }
183  }
184  }
185 }
186 
188 {
189  // Create an alias to this for better readability
190  auto child = this;
191  // Scan parent's properties and grab all the possible outputs and parameters
192  for(int k = 0; k < parent->metaObject()->propertyCount(); ++k)
193  {
194  QString parentPropName = parent->metaObject()->property(k).name();
195  // Get the parent's property base name
196  // Both output and parameter properties are checked
197  QString parentPropBaseName;
198  if(parentPropName.startsWith(QA_OUT))
199  {
200  // Output property
201  parentPropBaseName = parentPropName.mid(strlen(QA_OUT));
202  }
203  else if(parentPropName.startsWith(QA_PAR))
204  {
205  // Parameter
206  parentPropBaseName = parentPropName.mid(strlen(QA_PAR));
207  }
208  else continue;
209  // Get the child's property base name, based on PropagationRules values
210  QString childPropBaseName;
211  if(!getPropagationRules().contains(parentPropBaseName))
212  {
213  childPropBaseName = parentPropBaseName;
214  }
215  else
216  {
217  // The values associated with the given key
218  auto values = getPropagationRules().values(parentPropBaseName);
219  if(values.size() > 1)
220  {
221  // If there are more than one value, then use the first one that contain the parent object name
222  childPropBaseName = values.filter(parent->objectName()).first();
223  }
224  else
225  {
226  childPropBaseName = values.first();
227  }
228  }
229  // Parameters are sent only if they are explicitly mentioned in the PropagationRules
230  if(parentPropName.startsWith(QA_PAR) && !getPropagationRules().contains(parentPropBaseName))
231  {
232  continue;
233  }
234  // Check if the property is in child's input or parameter properties
235  for(int i = 0; i < child->metaObject()->propertyCount(); ++i)
236  {
237  QString childPropName = child->metaObject()->property(i).name();
238  // If the current property corresponds to an input with the same base name as before,
239  // then assign it; the same will be done also for parameters.
240  if(childPropName == QA_IN+childPropBaseName || childPropName == QA_PAR+childPropBaseName)
241  {
242  QVariant parentProp = parent->property(parentPropName.toStdString().c_str());
243  if(parentProp.isValid())
244  {
245  if(!child->setProperty(childPropName.toStdString().c_str(), parentProp))
246  {
247  qWarning() << "getInput():" << childPropName << "failed to set for" << child->printName();
248  return false;
249  }
250  }
251  else
252  {
253  qWarning() << "getInput():" << parentPropName << "failed to read for" << parent->printName();
254  return false;
255  }
256  }
257  }
258  }
259  return true;
260 }
261 
263 {
264  // Check if every ancestor has finished
265  if(allInputsReady())
266  {
267  // Perform the core part of the algorithm is a separate thread
268  setStarted();
269  result = QtConcurrent::run(this, &QAlgorithm::run);
270  watcher.setFuture(result);
271  }
272  else
273  {
274  // Run those not started
275  for(const auto& ancestor: getAncestors().keys(false))
276  {
277  // Only start processes not already started
278  if(!ancestor->isStarted()) ancestor->parallelExecution();
279  }
280  }
281 }
282 
284 {
285  // Check if every ancestor has finished
286  if(!allInputsReady())
287  {
288  // Run those not started
289  for(const auto& ancestor: getAncestors().keys(false))
290  {
291  // Only start processes not already started
292  if(!ancestor->isStarted()) ancestor->serialExecution();
293  }
294  }
295  // Set the ParallelExecution policy to false
296  setParallelExecution(false);
297  // Perform the core part of the algorithm in the same thread
298  setStarted();
299  run();
300  setFinished();
301 }
302 
303 void QAlgorithm::abort(QString message) const
304 {
305  Q_EMIT raise(message);
306 }
307 
308 void QAlgorithm::printGraph(const QString &path) const
309 {
310  QFile dotFile;
311  QString outFileName;
312  if (path.isEmpty())
313  {
314  dotFile.setFileName(QDir::home().absoluteFilePath("QAlgorithmTree.gv"));
315  outFileName = QDir::home().absoluteFilePath("QAlgorithmTree.svg");
316  }
317  else
318  {
319  dotFile.setFileName(path);
320  QStringList svgpath = path.split(".");
321  svgpath.removeLast();
322  outFileName = svgpath.join(".")+".svg";
323  }
324  if (!dotFile.open(QFile::WriteOnly)) Q_EMIT raise("Cannot write graph to given file");
325  else
326  {
327  QTextStream dot(&dotFile);
328  QLocale nospace;
329  nospace.setNumberOptions(QLocale::NumberOption::OmitGroupSeparator);
330  dot << "digraph g{\n";
331  auto flatMap = flattenTree();
332  // Save node labels
333  foreach(auto alg, flatMap.keys())
334  {
335  QString id = nospace.toString(quint64(alg.data()));
336  QString idspace = QLocale().toString(quint64(alg.data()));
337  QString name = alg->metaObject()->className();
338  QString nickname = alg->objectName();
339  QString dotstring = "var"+id.replace(" ", "")+"[label=\""+name+"\\nID "+idspace;
340  if(!nickname.isEmpty()) dotstring += "\\nNick: "+nickname;
341  dotstring += "\"];\n";
342  dot << dotstring;
343  }
344  // Save connections between nodes
345  foreach(auto parent, flatMap.keys())
346  {
347  QString parentName = "var" + nospace.toString(quint64(parent.data()));
348  foreach(auto child, flatMap[parent])
349  {
350  QString childName = "var" + nospace.toString(quint64(child.data()));
351  dot << parentName << " -> " << childName << "\n";
352  }
353  }
354  dot << "}\n";
355  dotFile.close();
356  // Convert dot file to svg
357  QStringList cmdArgs =
358  {dotFile.fileName(), "-Tsvg", "-o", outFileName}
359  ;
360  int r = QProcess::execute("/usr/local/bin/circo", cmdArgs);
361  if (r == -1)
362  {
363  qWarning("%s", "The dot process crashed");
364  }
365  else if (r == -2)
366  {
367  qWarning("%s", "Cannot start the dot process");
368  }
369  else
370  {
371  r = QProcess::execute("/bin/rm",
372  {dotFile.fileName()}
373  );
374  if (r == -1)
375  {
376  qWarning("%s", "The rm process crashed");
377  }
378  else if (r == -2)
379  {
380  qWarning("%s", "Cannot start the rm process");
381  }
382  }
383  }
384 }
385 
386 std::pair<QString, QVariant> QAlgorithm::makePropagationRules(std::initializer_list<std::pair<QString,QString>> pairs)
387 {
388  return std::pair<QString, QVariant>({"PropagationRules", QVariant::fromValue(QAPropagationRules(pairs))});
389 }
390 
391 QString QAlgorithm::printName() const
392 {
393  QString msg;
394  msg += QString(metaObject()->className()) + " ";
395  msg += QLocale().toString(qlonglong(this)) + " ";
396  if (!objectName().isEmpty()) msg += objectName();
397  return msg;
398 }
399 
401 {
402  // Search for a shared pointer attached to this instance
403  auto keysList = tree.keys();
404  // Check if the function has already been applied to this
405  auto shr_this_it = std::find_if(keysList.begin(), keysList.end(), [this](auto& ptr)
406  {return ptr == this;}
407  );
408  if (shr_this_it == keysList.end())
409  {
410  // Otherwise link each descendant to this in the map
411  auto shr_this = findSharedThis();
412  if(shr_this.isNull()) qWarning() << "This instance has no properly set connection, flattenTree will not work";
413  else
414  {
415  // Insert shr_this as new key in the map
416  tree[shr_this] = QSet<QAShrAlgorithm>();
417  // Append each descendant to it
418  for (auto descendant: getDescendants().keys()) tree[shr_this] << descendant;
419  // Recursive step: apply the function to each relative not yet scanned
420  for (auto relative: getDescendants().keys()+getAncestors().keys())
421  {
422  if (!tree.contains(relative)) tree = relative->flattenTree(tree);
423  }
424  }
425  }
426  else qWarning() << "flattenTree: possible loop";
427  return tree;
428 }
429 
431 {
432  ancestor->descendants[descendant] = descendant->isFinished();
433  descendant->ancestors[ancestor] = ancestor->isFinished();
434  connect(ancestor.data(), &QAlgorithm::raise, descendant.data(), &QAlgorithm::abort, Qt::QueuedConnection);
435  connect(descendant.data(), &QAlgorithm::raise, ancestor.data(), &QAlgorithm::abort, Qt::QueuedConnection);
436 }
437 
439 {
440  ancestor->descendants.remove(descendant);
441  descendant->ancestors.remove(ancestor);
442  disconnect(ancestor.data(), &QAlgorithm::raise, descendant.data(), &QAlgorithm::abort);
443  disconnect(descendant.data(), &QAlgorithm::raise, ancestor.data(), &QAlgorithm::abort);
444 }
445 
447 {
448  return ancestor->getDescendants().contains(descendant) && descendant->getAncestors().contains(ancestor);
449 }
450 
452 {
453  if (QAlgorithm::checkConnection(p2, p1))
454  {
455  return (p2->getDescendants().count() == 1) && (p1->getAncestors().count() == 1);
456  }
457  else if (QAlgorithm::checkConnection(p1, p2))
458  {
459  return (p1->getDescendants().count() == 1) && (p2->getAncestors().count() == 1);
460  }
461  else return false;
462 }
463 
465 {
466  QAlgorithm::setConnection(ancestor, descendant);
467  return descendant;
468 }
469 
471 {
472  ancestor >> descendant;
473  return ancestor;
474 }
475 
476 QDebug operator<<(QDebug debug, const QAlgorithm& c)
477 {
478  QDebugStateSaver saver(debug);
479  const QMetaObject* obj = c.metaObject();
480 
481  // Write the algorithm class name
482  debug << endl << "------------------------------" << c.printName() << "subclass of QAlgorithm" << endl;
483 
484  // Write the input properties of the algorithm
485  debug << "Algorithm with input:" << endl;
486  for(int k = 0; k < obj->propertyCount(); k++)
487  {
488  QMetaProperty prop = obj->property(k);
489  // Check whether the current property's name starts with "algin_"
490  QString propName = prop.name();
491  if(propName.startsWith(QA_IN))
492  {
493  propName.remove(QA_IN);
494  debug << propName.rightJustified(30,' ',true) << "\t" << prop.read(&c) << endl;
495  }
496  }
497 
498  // Write the parameters of the algorithm
499  debug << "Algorithm with parameters:" << endl;
500  for(int k = 0; k < obj->propertyCount(); k++)
501  {
502  QMetaProperty prop = obj->property(k);
503  // Check whether the current property's name starts with "algin_"
504  QString propName = prop.name();
505  if(propName.startsWith(QA_PAR))
506  {
507  propName.remove(QA_PAR);
508  debug << propName.rightJustified(30,' ',true) << "\t" << prop.read(&c) << endl;
509  }
510  }
511 
512  // Write the output properties of the algorithm
513  debug << "Algorithm with output:" << endl;
514  for(int k = 0; k < obj->propertyCount(); k++)
515  {
516  QMetaProperty prop = obj->property(k);
517  // Check whether the current property's name starts with "algout_"
518  QString propName = prop.name();
519  if(propName.startsWith(QA_OUT))
520  {
521  propName = propName.remove(QA_OUT);
522  debug << propName.rightJustified(30,' ',true) << "\t" << prop.read(&c) << endl;
523  }
524  }
525 
526  debug << "------------------------------" << endl;
527 
528  return debug;
529 }
530 
532 {
533  auto _printTree = [](decltype(tree) map)
534  {
535  foreach(auto key, map.keys())
536  {
537  qInfo() << "key" << key->printName();
538  foreach(auto value, map[key])
539  {
540  qInfo() << "\tvalue" << value->printName();
541  }
542  }
543  }
544  ;
545  if (tree.isEmpty()) _printTree(flattenTree());
546  else _printTree(tree);
547 }
548 
550 {
551  auto flatMap = leaf->flattenTree();
552  QMap<QAShrAlgorithm, QList<QAShrAlgorithm>> replacements;
553  foreach (auto ptr, flatMap.uniqueKeys())
554  {
555  foreach(auto child, flatMap[ptr])
556  {
557  if (QAlgorithm::isRemovableConnection(ptr, child))
558  {
559  // Insert a new replacement (assuming uniqueness of flat map keys)
560  replacements[ptr] << child;
561  }
562  }
563  }
564  // From the set of removable connection, link the pairs that form a removable connection all together
565  bool some_changes = true;
566  while(some_changes)
567  {
568  some_changes = false;
569  foreach(auto p1, replacements.keys())
570  {
571  // Take the last element in the list of removable connections
572  auto p2 = replacements[p1].last();
573  // If it has its own removable connections, add them to p1's
574  if (replacements.contains(p2))
575  {
576  replacements[p1] += replacements.take(p2);
577  some_changes = true;
578  break; // recompute keys after any change
579  }
580  }
581  }
582  // Create the envelopes for each replacement
583  foreach(QList<QAShrAlgorithm> nodes, replacements.values())
584  {
585  // Tell each node, except the last, to serially execute its children
586  nodes.removeLast();
587  foreach(auto node, nodes)
588  {
589  node->setParallelExecution(false);
590  }
591  }
592 }
593 
594 QDataStream& operator<<(QDataStream& stream, const QAlgorithm& c)
595 {
596  QAPropertyMap properties;
597  for(int k = 0; k < c.metaObject()->propertyCount(); k++)
598  {
599  QMetaProperty prop = c.metaObject()->property(k);
600  QString propName = prop.name();
601  QVariant propValue = prop.read(&c);
602  if(propValue.isValid() &&
603  (propName.startsWith(QA_IN) || propName.startsWith(QA_OUT) || propName.startsWith(QA_PAR)))
604  properties.insert(propName, propValue);
605  }
606  return (stream << properties);
607 }
608 
609 QDataStream& operator>>(QDataStream& stream, QAlgorithm& c)
610 {
611  QAPropertyMap properties;
612  stream >> properties;
613  for(int k = 0; k < c.metaObject()->propertyCount(); k++)
614  {
615  QMetaProperty prop = c.metaObject()->property(k);
616  QString propName = prop.name();
617  if(properties.contains(propName))
618  {
619  if (!prop.write(&c, properties.values(propName)))
620  {
621  qWarning() << c.printName() << "Unable to write property value, report to the QAlgorithm developer";
622  }
623  }
624  }
625  return stream;
626 }
Q_SLOT void serialExecution()
Start computing the algorithm tree on the calling thread.
Definition: QAlgorithm.cpp:283
Q_SIGNAL void raise(QString message) const
Signal emitted whenever an error occurs.
bool finished
Whether the algorithm finished to run and outputs are ready.
Definition: QAlgorithm.h:105
Q_SIGNAL void justStarted()
Signal emitted on algorithm's start.
QAShrAlgorithm findDescendant(const QAlgorithm *descendant) const
Find an descendant.
Definition: QAlgorithm.cpp:78
QFuture< void > result
Definition: QAlgorithm.h:197
Q_SLOT void parallelExecution()
Start computing the algorithm tree on different threads.
Definition: QAlgorithm.cpp:262
Q_SLOT void abort(QString message="Unknown Error") const
Emit the given error signal.
Definition: QAlgorithm.cpp:303
bool started
Whether the algorithm started to run.
Definition: QAlgorithm.h:106
#define QA_OUT
Prefix for output properties.
Definition: qa_macros.h:34
static std::pair< QString, QVariant > makePropagationRules(std::initializer_list< std::pair< QString, QString >> lst)
Convenience method for writing PropagationRules.
Definition: QAlgorithm.cpp:386
QAShrAlgorithm findAncestor(const QAlgorithm *ancestor) const
Find an ancestor.
Definition: QAlgorithm.cpp:61
static void closeConnection(QAShrAlgorithm ancestor, QAShrAlgorithm descendant)
Disconnect two algorithms.
Definition: QAlgorithm.cpp:438
virtual void setParameters(const QAPropertyMap &parameters)
Set parameters for the algorithm.
Definition: QAlgorithm.cpp:120
#define QA_PAR
Prefix for parameters.
Definition: qa_macros.h:39
QACompletionMap getAncestors() const
Get the value of ancestors.
Definition: QAlgorithm.cpp:51
QAShrAlgorithm operator>>(QAShrAlgorithm ancestor, QAShrAlgorithm descendant)
Creates connections like setConnection().
Definition: QAlgorithm.cpp:464
static bool checkConnection(const QAShrAlgorithm ancestor, const QAShrAlgorithm descendant)
Check if two algorithms are connected.
Definition: QAlgorithm.cpp:446
virtual bool getInput(QAShrAlgorithm parent)
Load inputs from parent's outputs.
Definition: QAlgorithm.cpp:187
QACompletionMap descendants
Map with descendants and their completion.
Definition: QAlgorithm.h:108
virtual void setup()
Set of instructions to set up the algorithm.
Definition: QAlgorithm.cpp:144
QAFlatRepresentation flattenTree(QAFlatRepresentation tree=QAFlatRepresentation()) const
Creates a flat representation of the algorithm tree.
Definition: QAlgorithm.cpp:400
QFutureWatcher< void > watcher
Definition: QAlgorithm.h:198
QMap< QAShrAlgorithm, QSet< QAShrAlgorithm > > QAFlatRepresentation
Definition: QAlgorithm.h:37
bool isStarted() const
Get the value of started.
Definition: QAlgorithm.cpp:29
QAlgorithm(QObject *parent=Q_NULLPTR)
Constructor.
Definition: QAlgorithm.cpp:113
virtual void run()=0
Core part of the algorithm, to be reimplemented in subclasses.
QAShrAlgorithm operator<<(QAShrAlgorithm descendant, QAShrAlgorithm ancestor)
Creates connections like setConnection().
Definition: QAlgorithm.cpp:470
bool isFinished() const
Get the value of finished.
Definition: QAlgorithm.cpp:24
static void improveTree(QAlgorithm *leaf)
Replace all removable connections, thus improving the tree performance.
Definition: QAlgorithm.cpp:549
static quint32 print_counter
Definition: QAlgorithm.h:195
QSharedPointer< QAlgorithm > QAShrAlgorithm
Definition: QAlgorithm.h:31
Abstract class that implements a generic algorithm.
Definition: QAlgorithm.h:96
void setFinished()
Set the algorithm as comleted and signals it.
Definition: QAlgorithm.cpp:40
static void setConnection(QAShrAlgorithm ancestor, QAShrAlgorithm descendant)
Connect two algorithms.
Definition: QAlgorithm.cpp:430
Q_SIGNAL void justFinished()
Signal emitted on algorithm's end.
QMap< QString, QVariant > QAPropertyMap
Definition: QAlgorithm.h:34
#define QA_IN
Prefix for input properties.
Definition: qa_macros.h:29
void printGraph(const QString &path=QString()) const
Create a GraphViz diagram of the algorithm tree.
Definition: QAlgorithm.cpp:308
QString printName() const
Returns name, memory address and class name of the algorithm.
Definition: QAlgorithm.cpp:391
bool allInputsReady() const
Checks if the algorithm is ready to run.
Definition: QAlgorithm.cpp:46
QMap< QAShrAlgorithm, bool > QACompletionMap
Definition: QAlgorithm.h:36
QACompletionMap getDescendants() const
Get the value of descendants.
Definition: QAlgorithm.cpp:56
QMultiMap< QString, QString > QAPropagationRules
Definition: QAlgorithm.h:35
void printTree(const QAFlatRepresentation &tree=QAFlatRepresentation()) const
Outputs a text representation of the algorithm tree.
Definition: QAlgorithm.cpp:531
void setStarted()
Set the algorithm as started and signals it.
Definition: QAlgorithm.cpp:34
QAShrAlgorithm findSharedThis() const
Find a shared pointer to this instance.
Definition: QAlgorithm.cpp:95
static bool isRemovableConnection(const QAShrAlgorithm p1, const QAShrAlgorithm p2)
Check if two algorithms are connected and the connection is removable.
Definition: QAlgorithm.cpp:451
QACompletionMap ancestors
Map with ancestors and their completion.
Definition: QAlgorithm.h:107
Q_SLOT void propagateExecution()
Execute descendants.
Definition: QAlgorithm.cpp:153