28 #include <QMessageBox>
38 #include "../io/GroupXmlReader.h"
39 #include "../io/GroupXmlWriter.h"
40 #include "../io/XmlWriter.h"
45 using namespace statechartio;
46 using namespace statechartmodel;
49 iceCommunicator(
ic), variantInfo(variantInfo)
60 QList<statechartmodel::StatePtr> statesToAnnotate;
62 for (
int i = 0; i < groups.count(); i++)
67 for (
int j = 0; j < group->getRootNode()->getChildren().count(); j++)
69 loadNode(group->getRootNode()->getChildren().at(j),
70 group->getBoostGroupPath(),
81 QList<statechartmodel::StatePtr> statesToAnnotate;
83 for (
int j = 0; j < group->getRootNode()->getChildren().count(); j++)
85 loadNode(group->getRootNode()->getChildren().at(j),
86 group->getBoostGroupPath(),
96 QMap<QString, statechartmodel::StatePtr> referenceMap;
97 QMapIterator<QString, StateTreeNodePtr> i(uuidToNodeMap);
102 referenceMap.insert(i.key(), i.value()->getState());
105 for (
int i = 0; i < statesToAnnotate.count(); i++)
107 statesToAnnotate.at(i)->addReferences(referenceMap);
114 QMap<QString, statechartmodel::StatePtr> referenceMap;
115 QMapIterator<QString, StateTreeNodePtr> i(uuidToNodeMap);
120 referenceMap.insert(i.key(), i.value()->getState());
123 for (
auto i : uuidToNodeMap)
125 i->getState()->addReferences(referenceMap);
131 std::filesystem::path parentPath,
132 QList<statechartmodel::StatePtr>& allTheStates)
134 std::filesystem::path nodePath = node->getBoostPath(parentPath);
141 QString::fromUtf8(nodePath.string().c_str()));
142 std::shared_ptr<XmlReader> reader(
new XmlReader(iceCommunicator, variantInfo));
143 reader->parseXml(xml);
147 uuidToNodeMap.insert(state->getUUID(), node);
148 allTheStates.append(state);
157 for (
int i = 0; i < node->getChildren().count(); i++)
159 loadNode(node->getChildren().at(i), nodePath, allTheStates);
173 group->getPackageName();
175 groups.append(group);
183 uuidToNodeMap.clear();
189 groups.append(group);
197 uuidToNodeMap.insert(node->getState()->getUUID(), node);
206 uuidToNodeMap.remove(node->getState()->getUUID());
215 progressDialog->setMinimum(0);
216 progressDialog->setMaximum(groups.count());
217 progressDialog->setLabelText(
"Saving statechart groups");
219 auto startTime = IceUtil::Time::now();
220 #pragma omp parallel for schedule(static, 1)
221 for (
int i = 0; i < groups.count(); i++)
228 std::filesystem::create_directories(group->getBoostDefinitionFilePath().parent_path());
231 for (
int j = 0; j < group->getRootNode()->getChildren().count(); j++)
233 saveNode(group->getRootNode()->getChildren().at(j), group->getBoostGroupPath());
235 if (omp_get_thread_num() == 0)
237 if (progressDialog && (IceUtil::Time::now() - startTime).toMilliSeconds() > 1000)
239 progressDialog->setValue(i + 1);
249 std::filesystem::path nodePath = node->getBoostPath(parentPath);
260 if (node->getState())
263 writer.
serialize(node->getState(), uuidToNodeMap);
266 nodePath.string(), writer.
getXmlString(
true).toUtf8().data()))
268 ARMARX_INFO_S <<
"Writing new state definition to " << nodePath.string();
276 catch (std::exception& e)
278 QMetaObject::invokeMethod(
281 Q_ARG(QString,
"Saving state failed!"),
283 QString::fromStdString(std::string(e.what())).split(
"Backtrace:").first()));
289 std::filesystem::create_directory(nodePath);
291 for (
int i = 0; i < node->getChildren().count(); i++)
293 saveNode(node->getChildren().at(i), nodePath);
301 QString doxString =
"/**\n";
302 doxString +=
"@defgroup " + group->getPackageName() +
"-" + group->getName() +
" " +
303 group->getName() +
"";
304 doxString +=
"\n@ingroup Statecharts\n";
305 doxString += group->getDescription();
306 doxString +=
"@defgroup " + group->getPackageName() +
"-" + group->getName() +
"-Substates " +
307 group->getName() +
" Substates\n";
308 doxString +=
"@ingroup " + group->getPackageName() +
"-" + group->getName();
317 std::unique_lock lock(findersMutex);
318 auto it = finders.find(packageName);
319 if (it == finders.end())
321 finders.insert(std::make_pair(packageName,
CMakePackageFinder(packageName.toStdString())));
322 return finders.at(packageName);
335 bool loaded = variantInfo->isPackageLoaded(packageName);
336 if (!finder.packageFound())
341 <<
" - generated code might be incomplete";
348 ARMARX_INFO_S <<
"Loading variant info for " << packageName <<
" and its dependencies";
350 packageName, finder.getConfigDir(),
false, variantInfo);
353 [](
const std::map<VariantTypeId, std::string>& map,
const std::string&
value)
355 for (
const auto&
v : map)
368 if (lib->getPackageName() != packageName)
380 variantInfo->loadLibraryOfVariant(var->getBaseTypeName());
382 lib->setUnloadOnDestruct(
false);
388 if (iceCommunicator->getValueFactoryManager()->find(var->getBaseTypeName()))
390 ARMARX_INFO <<
"Loaded variant " << var->getBaseTypeName();
395 <<
" could not be loaded. Adding Variant TypeName only.";
401 for (
auto& dep : finder.getDependencies())
413 progressDialog->setMinimum(0);
414 progressDialog->setMaximum(groups.count());
415 progressDialog->setLabelText(
"Generating code for statechart groups");
417 auto startTime = IceUtil::Time::now();
418 #pragma omp parallel for schedule(static, 1)
419 for (
int i = 0; i < groups.size(); ++i)
424 if (omp_get_thread_num() == 0)
426 if (progressDialog && (IceUtil::Time::now() - startTime).toMilliSeconds() > 1000)
428 progressDialog->setValue(i + 1);
434 if (progressDialog && (IceUtil::Time::now() - startTime).toMilliSeconds() > 1000)
436 QMetaObject::invokeMethod(progressDialog,
"setValue", Q_ARG(
int, (i + 1)));
442 std::set<std::string>
445 std::set<std::string> variantTypes;
446 std::set<std::string> alreadyLinkedTypes;
449 std::set<std::string>* currentTypeSet = &variantTypes;
451 if (stateNode->checkCppExists())
454 currentTypeSet = &alreadyLinkedTypes;
456 auto state = stateNode->getState();
461 for (
auto param : state->getInputAndLocalParameters())
463 QString type = param->type;
464 currentTypeSet->insert(type.toStdString());
466 for (
auto param : state->getOutputParameters())
468 QString type = param->type;
469 currentTypeSet->insert(type.toStdString());
474 std::set<std::string> resultingTypes;
475 std::set_difference(variantTypes.begin(),
477 alreadyLinkedTypes.begin(),
478 alreadyLinkedTypes.end(),
479 std::inserter(resultingTypes, resultingTypes.end()));
481 std::set<std::string> innerVariantTypes;
482 for (
auto& type : resultingTypes)
485 while (containerType->subType)
487 containerType = containerType->subType;
489 if (!variantInfo->isBasic(containerType->typeId))
491 innerVariantTypes.insert(containerType->typeId);
495 return innerVariantTypes;
507 if (!finder.packageFound())
509 QMetaObject::invokeMethod(
this,
511 Q_ARG(QString,
"State Generation Error"),
513 "Could not find the package '" +
514 QString::fromStdString(finder.getName()) +
515 "' - Did you run cmake for that package already? "
516 "Skipping state generation of stategroup " +
527 std::map<QString, CMakePackageFinder>::const_iterator it;
529 std::unique_lock lock(findersMutex);
530 it = finders.find(group->getPackageName());
533 if (it == finders.end())
537 if (!checkPackage(finder))
542 it = finders.insert(std::make_pair(group->getPackageName(), finder)).first;
548 if (!checkPackage(packageFinder))
555 <<
"' is empty! Skipping generating files.";
560 std::filesystem::path doxyFile = packageFinder.
getBuildDir();
562 if (!doxyFile.string().empty())
564 doxyFile /=
"doxygen";
571 doxyFile /= group->getName().toStdString() +
".dox";
574 std::set<std::string> proxies;
575 proxies.insert(
"ArmarXCoreInterfaces.systemObserver");
576 proxies.insert(
"ArmarXCoreInterfaces.conditionHandler");
577 for (QString p : group->getProxies())
579 proxies.insert(p.toUtf8().data());
581 Ice::StringSeq proxyVec(proxies.begin(), proxies.end());
582 if (group->contextGenerationEnabled())
584 std::set<std::string> innerVariantTypes = getVariantOfStatesWithNoCpp(group);
587 group->getPackageName().toUtf8().data(),
588 group->getName().toUtf8().data(),
605 std::filesystem::path buildDir,
607 std::vector<std::string> proxies)
614 if (node->getState() && node->getCppExists())
617 writer.
serialize(node->getState(), uuidToNodeMap);
618 std::string xml = writer.
getXmlString(
false).toUtf8().data();
620 node->getState()->getStateName().toUtf8().data(),
623 node->getGroup()->getPackageName().toUtf8().data(),
624 node->getGroup()->getName().toUtf8().data(),
626 node->getGroup()->contextGenerationEnabled(),
638 catch (std::exception& e)
640 QMessageBox::warning(
642 "Saving state failed!",
643 QString::fromStdString(std::string(e.what())).split(
"Backtrace:").first());
659 bool identical =
false;
661 if (std::filesystem::exists(std::filesystem::path(targetPath)))
664 identical = contents == currentContents;
669 QString qtargetPath = QString::fromUtf8(targetPath.data());
676 ARMARX_LOG_S <<
"Skipping identical file " << targetPath;
687 QList<StatechartGroupPtr>
690 return QList<StatechartGroupPtr>(groups);
696 auto it = std::find_if(groups.constBegin(),
700 if (it != groups.constEnd())
724 QMap<QString, StateTreeNodePtr>::const_iterator it = uuidToNodeMap.find(uuid);
726 if (it != uuidToNodeMap.end())
740 node->checkCppExists();
741 node->display = state->getStateName();
748 QList<StateInstancePtr>
751 QList<StateInstancePtr> instances;
752 QMapIterator<QString, StateTreeNodePtr> i(uuidToNodeMap);
757 QMapIterator<QString, StateInstancePtr> itSubstate = i.value()->getState()->getSubstates();
759 while (itSubstate.hasNext())
764 if (state && node->getState() == state)
766 instances.append(itSubstate.value());
785 if (g->getBoostGroupPath() == groupPath)
803 StateTreeModel::showWarning(QString title, QString
message)
805 QMessageBox::warning(0, title,
message);