WidgetController.cpp
Go to the documentation of this file.
1 /*
2  * This file is part of ArmarX.
3  *
4  * ArmarX is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License version 2 as
6  * published by the Free Software Foundation.
7  *
8  * ArmarX is distributed in the hope that it will be useful, but
9  * WITHOUT ANY WARRANTY; without even the implied warranty of
10  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11  * GNU General Public License for more details.
12  *
13  * You should have received a copy of the GNU General Public License
14  * along with this program. If not, see <http:// www.gnu.org/licenses/>.
15  *
16  * \package Navigation::gui-plugins::WidgetController
17  * \author Rainer Kartmann ( rainer dot kartmann at kit dot edu )
18  * \date 2021
19  * \copyright http:// www.gnu.org/licenses/gpl-2.0.txt
20  * GNU General Public License
21  */
22 
23 #include "WidgetController.h"
24 
25 #include <map>
26 #include <memory>
27 #include <set>
28 #include <sstream>
29 #include <utility>
30 #include <vector>
31 
32 #include <QDialog>
33 #include <QHBoxLayout>
34 #include <QLabel>
35 #include <QLineEdit>
36 #include <QMenu>
37 #include <QMessageBox>
38 #include <QMouseEvent>
39 #include <QObject>
40 #include <QPushButton>
41 #include <QSignalBlocker>
42 #include <qboxlayout.h>
43 #include <qhashfunctions.h>
44 #include <qicon.h>
45 #include <qlist.h>
46 #include <qmessagebox.h>
47 #include <qobject.h>
48 #include <qobjectdefs.h>
49 #include <qpair.h>
50 #include <qpointer.h>
51 #include <qsettings.h>
52 #include <qwidget.h>
53 
57 
59 
62 #include <RobotAPI/interface/armem/mns/MemoryNameSystemInterface.h>
72 #include <RobotAPI/libraries/armem_locations/aron/Location.aron.generated.h>
76 
77 #include <armarx/navigation/components/navigation_memory/ComponentInterface.h>
80 #include <armarx/navigation/core/aron/Graph.aron.generated.h>
86 #include <armarx/navigation/gui-plugins/LocationGraphEditor/ui_LocationGraphEditorWidget.h>
88 
95 #include "widgets/utils.h"
96 #include <SemanticObjectRelations/Shapes/Shape.h>
97 
98 
99 static const QString SETTING_LAST_SCENE = "lastScene";
100 
102 {
103 
104  QString
106  {
107  return "Navigation.LocationGraphEditor";
108  }
109 
110  QIcon
112  {
113  return QIcon{"://icons/location_graph_editor.svg"};
114  }
115 
116  WidgetController::WidgetController() : settings{"KIT", "WidgetController"}
117  {
118  widget.setupUi(getWidget());
119 
120  loadAutomaticSettings();
121 
122  widget.loadGraphButton->setEnabled(false);
123 
124  view.vertexData = new VertexDataWidget();
125  view.vertexData->setEnabled(false); // Enable on first selection of vertex.
126  widget.locationDataGroupBox->layout()->addWidget(view.vertexData);
127  if (QBoxLayout* layout = dynamic_cast<QBoxLayout*>(widget.locationDataGroupBox->layout()))
128  {
129  layout->addStretch();
130  }
131 
132  view.vertexTable = new VertexTableWidget();
133  widget.locationsTableGroupBox->layout()->addWidget(view.vertexTable);
134 
135  view.edgeTable = new EdgeTableWidget();
136  widget.edgesTableGroupBox->layout()->addWidget(view.edgeTable);
137 
138  view.robotVisu = new RobotVisuWidget(*this);
139  widget.robotVisuGroupBox->layout()->addWidget(view.robotVisu);
140 
141  view.objectPoses = new ObjectPoseClientWidget(*this);
142  widget.objectPoseClientGroupBox->layout()->addWidget(view.objectPoses);
143 
144 
145  connect(this, &This::connected, this, &This::queryGraphs);
146  connect(this, &This::connected, this, &This::graphChanged);
147 
148  connect(widget.createGraphButton, &QPushButton::pressed, this, &This::createGraphDialog);
149 
150  connect(widget.queryGraphsButton, &QPushButton::pressed, this, &This::queryGraphs);
151 
152  connect(this, &This::locationMemoryChanged, this, &This::memoryChanged);
153  connect(this, &This::graphMemoryChanged, this, &This::memoryChanged);
154 
155  connect(this, &This::graphMemoryChanged, this, &This::updateGraphList);
156 
157  connect(widget.loadGraphButton, &QPushButton::pressed, this, &This::loadGraph);
158  connect(widget.commitGraphButton, &QPushButton::pressed, this, &This::commit);
159 
160  // View updates
161  connect(this, &This::graphChanged, this, &This::updateGraphView);
162  connect(
163  view.vertexData, &VertexDataWidget::vertexDataChanged, this, &This::updateGraphView);
164  connect(view.robotVisu, &RobotVisuWidget::connected, this, &This::updateGraphView);
165  connect(view.robotVisu,
167  this,
168  [this]() { view.vertexData->setRobotConnection(&view.robotVisu->connection()); });
169  connect(view.robotVisu, &RobotVisuWidget::settingsChanged, this, &This::updateGraphView);
170 
171  connect(view.objectPoses,
173  this,
174  [this]()
175  { view.vertexData->setObjectPoseConnection(&view.objectPoses->connection()); });
176 
177  // Selection
178  connect(view.vertexTable,
179  &VertexTableWidget::currentItemChanged,
180  [this](QTableWidgetItem* current, QTableWidgetItem* previous)
181  {
182  (void)previous;
183  this->selectVertex(current);
184  });
185 
186  connect(view.vertexTable,
187  &VertexTableWidget::itemSelectionChanged,
188  this,
189  &This::updateVertexHighlighting);
190  connect(view.edgeTable,
191  &EdgeTableWidget::itemSelectionChanged,
192  this,
193  &This::updateEdgeHighlighting);
194 
195  // Graph modification
196  connect(view.vertexTable,
198  this,
199  &This::createVertexDialog);
200  connect(view.vertexTable, &VertexTableWidget::newEdgesRequested, this, &This::addEdges);
201  connect(view.vertexTable,
203  this,
204  &This::removeEdgesOfVertex);
205 
206  connect(view.edgeTable, &EdgeTableWidget::edgeRemovalRequested, this, &This::removeEdges);
207 
208  // export
209  connect(widget.exportButton, &QPushButton::clicked, this, &This::exportLocationGraph);
210  }
211 
212  void
213  WidgetController::exportLocationGraph()
214  {
215  QString labelText;
216 
217  auto navigationMemoryPrx =
218  memory::NavigationMemoryInterfacePrx::uncheckedCast(remote.locationReader.readingPrx);
219  if (navigationMemoryPrx)
220  {
221  const ::armarx::PackagePath packagePath(
222  widget.packageNameEdit->text().toStdString(),
223  widget.packageDirectoryEdit->text().toStdString());
224 
225  navigationMemoryPrx->storeLocationGraph(packagePath.serialize());
226  labelText = "Exported locations and graph.";
227  }
228  else
229  {
230  labelText = "Export failed: The connected navigation memory does not implement the "
231  "NavigationMemoryInterface.";
232  }
233 
234  widget.exportLabel->setText(labelText);
235  }
236 
238  {
240  }
241 
242  QPointer<QDialog>
244  {
245  if (not configDialog)
246  {
247  configDialog = new SimpleConfigDialog(parent);
248  configDialog->addProxyFinder<armem::mns::MemoryNameSystemInterfacePrx>(
249  "MemoryNameSystem", "Memory Name System", remote.memoryNameSystemName);
250  }
251  return qobject_cast<SimpleConfigDialog*>(configDialog);
252  }
253 
254  void
256  {
257  remote.memoryNameSystemName = configDialog->getProxyName("MemoryNameSystem");
258  }
259 
260  void
261  WidgetController::loadSettings(QSettings* settings)
262  {
263  remote.memoryNameSystemName =
264  settings
265  ->value("memoryNameSystemName", QString::fromStdString(remote.memoryNameSystemName))
266  .toString()
267  .toStdString();
268  }
269 
270  void
271  WidgetController::saveSettings(QSettings* settings)
272  {
273  settings->setValue("memoryNameSystemName",
274  QString::fromStdString(remote.memoryNameSystemName));
275  }
276 
277  void
279  {
280  lastSelectedSceneName =
281  settings.value(SETTING_LAST_SCENE, lastSelectedSceneName).toString();
282  }
283 
284  void
286  {
287  settings.setValue(SETTING_LAST_SCENE, lastSelectedSceneName);
288  }
289 
290  void
292  {
293  usingProxy(remote.memoryNameSystemName);
294  }
295 
296  void
298  {
299  remote.connect(*this);
300  {
301  std::stringstream ss;
302  ss << "Navigation Graphs (Entities from Core Segment "
304  widget.graphGroupBox->setTitle(QString::fromStdString(ss.str()));
305  widget.graphGroupBox->setEnabled(true);
306  }
307 
308  emit connected();
309  }
310 
311  void
312  WidgetController::Remote::connect(Component& parent)
313  {
314  auto mnsProxy =
315  parent.getProxy<armem::mns::MemoryNameSystemInterfacePrx>(memoryNameSystemName);
316  memoryNameSystem = armem::client::MemoryNameSystem(mnsProxy, &parent);
317 
318  locationReader = memoryNameSystem.useReader(navigation::location::coreSegmentID);
319  locationWriter = memoryNameSystem.useWriter(navigation::location::coreSegmentID);
320 
321  graphReader = memoryNameSystem.useReader(navigation::graph::coreSegmentID);
322  graphWriter = memoryNameSystem.useWriter(navigation::graph::coreSegmentID);
323 
324  arviz = std::make_unique<viz::Client>(viz::Client::createForGuiPlugin(parent));
325  }
326 
327  void
328  WidgetController::queryMemory()
329  {
330  armem::client::QueryResult locResult = queryLocations();
331  armem::client::QueryResult graphResult = queryGraphs();
332 
333  if (not(locResult.success and graphResult.success))
334  {
335  QStringList status;
336  std::stringstream ss;
337  if (not locResult.success)
338  {
339  status.append(QString::fromStdString(locResult.errorMessage));
340  }
341  if (not graphResult.success)
342  {
343  status.append(QString::fromStdString(graphResult.errorMessage));
344  }
345  widget.statusLabel->setText(status.join("\n"));
346  }
347  }
348 
349  armem::client::QueryResult
350  WidgetController::queryLocations()
351  {
352  armem::client::QueryResult result =
353  remote.locationReader.getLatestSnapshotsIn(navigation::location::coreSegmentID);
354  if (result.success)
355  {
356  model.locationsMemory = std::move(result.memory);
357  emit locationMemoryChanged();
358  }
359  return result;
360  }
361 
362  armem::client::QueryResult
363  WidgetController::queryGraphs()
364  {
365  armem::client::QueryResult result =
366  remote.graphReader.getLatestSnapshotsIn(navigation::graph::coreSegmentID);
367  if (result.success)
368  {
369  model.graphMemory = std::move(result.memory);
370  emit graphMemoryChanged();
371  }
372  return result;
373  }
374 
375  void
376  WidgetController::updateGraphList()
377  {
378  QString previousText = widget.graphsComboBox->currentText();
379  widget.graphsComboBox->clear();
380 
381  int i = 0;
382  int previousIndex = -1; // To keep selection.
383  model.graphMemory.forEachEntity(
384  [&](const armem::wm::Entity& entity)
385  {
386  bool hasChanged =
387  (entity.id() == model.graphEntityID ? model.graph.hasChanged() : false);
388  QString text = getGraphDisplayName(entity.id(), hasChanged);
389  QString id = QString::fromStdString(entity.id().str());
390 
391  widget.graphsComboBox->addItem(text, id);
392  if (previousIndex < 0 and text == previousText)
393  {
394  previousIndex = i;
395  }
396  ++i;
397  });
398  if (previousIndex >= 0)
399  {
400  widget.graphsComboBox->setCurrentIndex(previousIndex);
401  }
402  widget.loadGraphButton->setEnabled(widget.graphsComboBox->count() > 0);
403  }
404 
405  void
406  WidgetController::loadGraph()
407  {
408  if (model.graph.hasChanged())
409  {
410  if (not loadGraphDialog())
411  {
412  return;
413  }
414  }
415 
417  widget.graphsComboBox->currentData().toString().toStdString());
418 
419  // Refresh local memory.
420  queryMemory();
421 
422  const armem::wm::EntityInstance* instance = model.graphMemory.findLatestInstance(entityID);
423  if (instance)
424  {
425  widget.statusLabel->setText(QString::fromStdString(
426  "Loaded snapshot " + instance->id().getEntitySnapshotID().str()));
427  }
428  else
429  {
430  std::stringstream ss;
431  ss << "No latest instance of entity " << entityID << " in memory.";
432  widget.statusLabel->setText(QString::fromStdString(ss.str()));
433  return;
434  }
435 
437  {
439  ARMARX_CHECK_NOT_NULL(instance->data());
440  dto.fromAron(instance->data());
441  fromAron(dto, nav);
442  }
443  // Resolve locations and remove vertices which could not be resolved.
444  {
445  resolveLocations(nav, model.locationsMemory);
446  std::vector<semrel::ShapeID> remove;
447  for (navigation::core::Graph::Vertex vertex : nav.vertices())
448  {
449  if (not vertex.attrib().hasPose())
450  {
451  remove.push_back(vertex.objectID());
452  }
453  }
454  if (not remove.empty())
455  {
456  for (semrel::ShapeID vertexID : remove)
457  {
458  nav.removeVertex(nav.vertex(vertexID));
459  }
460  std::stringstream ss;
461  ss << "Dropped " << remove.size() << " locations which could not be resolved.";
462  widget.statusLabel->setText(QString::fromStdString(ss.str()));
463  }
464  }
465 
466  ARMARX_VERBOSE << "Loading graph " << nav.str();
467  setGraph(entityID, nav);
468  }
469 
470  bool
471  WidgetController::loadGraphDialog()
472  {
473  ARMARX_CHECK(model.graph.hasChanged());
474 
475  QMessageBox msgBox;
476  msgBox.setText("The current graph and/or locations have uncommitted changes. ");
477  msgBox.setInformativeText("Do you want to discard them?");
478  QStringList detailLines;
479  if (model.graph.attrib().edgesChanged)
480  {
481  detailLines.append("Graph edges have changed.");
482  }
483  for (auto vertex : model.graph.vertices())
484  {
485  if (vertex.attrib().changed)
486  {
487  detailLines.append("Location " + QString::fromStdString(vertex.attrib().getName()) +
488  " has changed.");
489  }
490  }
491  msgBox.setDetailedText(detailLines.join("\n"));
492  msgBox.setStandardButtons(QMessageBox::Discard | QMessageBox::Cancel);
493  msgBox.setDefaultButton(QMessageBox::Cancel);
494 
495  switch (msgBox.exec())
496  {
497  case QMessageBox::Discard:
498  return true; // Ok go.
499  case QMessageBox::Cancel:
500  return false; // Abort loading.
501  default:
502  return false; // Something went wrong.
503  }
504  }
505 
506  void
507  WidgetController::commit()
508  {
509  armem::CommitResult locResults = commitLocations();
510  armem::EntityUpdateResult graphResult = commitGraph();
511 
512  {
513  std::stringstream ss;
514  if (locResults.allSuccess())
515  {
516  ss << "Committed " << locResults.results.size() << " location snapshot";
517  if (locResults.results.size() != 1)
518  {
519  ss << "s"; // plural
520  }
521  if (graphResult.success)
522  {
523  ss << " and 1 graph snapshot " << graphResult.snapshotID;
524  }
525  else
526  {
527  ss << " but failed to commit graph: \n" << graphResult.errorMessage;
528  }
529  }
530  else
531  {
532  int numLocs = static_cast<int>(locResults.results.size());
533  int numSuccess = 0;
534  for (const auto& r : locResults.results)
535  {
536  numSuccess += int(r.success);
537  }
538  int numFailed = numLocs - numSuccess;
539 
540  if (graphResult.success)
541  {
542  ss << "Committed 1 graph snapshot " << graphResult.snapshotID;
543  ss << " and " << numSuccess << " locations, but failed to commit " << numFailed
544  << " locations: \n"
545  << locResults.allErrorMessages();
546  }
547  else
548  {
549  ss << "Failed to commit graph and " << numFailed << " of " << numLocs
550  << " locations: \n";
551  ss << graphResult.errorMessage << "\n";
552  ss << locResults.allErrorMessages();
553  }
554  }
555 
556  widget.statusLabel->setText(QString::fromStdString(ss.str()));
557  }
558 
559  // `changed` flags may have changed
560  emit graphChanged();
561  }
562 
564  WidgetController::commitLocations()
565  {
566  armem::Commit commit;
567 
568  for (auto vertex : model.graph.vertices())
569  {
570  if (vertex.attrib().changed)
571  {
572  armem::EntityUpdate& update = commit.add();
573  fromAron(vertex.attrib().aron.locationID, update.entityID);
574  update.referencedTime = armem::Time::Now();
575 
576  navigation::location::arondto::Location dto;
577  toAron(dto.framedPose, vertex.attrib().getPose());
578  update.instancesData = {dto.toAron()};
579  }
580  }
581 
582  armem::CommitResult result = remote.locationWriter.commit(commit);
583  auto it = result.results.begin();
584  for (auto vertex : model.graph.vertices())
585  {
586  if (vertex.attrib().changed)
587  {
588  ARMARX_CHECK(it != result.results.end());
589  if (it->success)
590  {
591  // Only clear dirty flag when update was successful.
592  vertex.attrib().changed = false;
593  }
594  ++it;
595  }
596  }
597  return result;
598  }
599 
601  WidgetController::commitGraph()
602  {
604  {
605  navigation::core::Graph nav = fromGuiGraph(model.graph);
606  toAron(dto, nav);
607  }
608 
610  update.entityID = model.graphEntityID;
611  update.referencedTime = armem::Time::Now();
612  update.instancesData = {dto.toAron()};
613 
614  armem::EntityUpdateResult result = remote.graphWriter.commit(update);
615  if (result.success)
616  {
617  // Clear dirty flag
618  model.graph.attrib().edgesChanged = false;
619  }
620  return result;
621  }
622 
623  template <class GraphT>
624  semrel::ShapeID
625  findNextFreeVertexID(const GraphT& graph, semrel::ShapeID vertexID)
626  {
627  while (graph.hasVertex(vertexID))
628  {
629  ++vertexID;
630  }
631  return vertexID;
632  }
633 
634  void
635  WidgetController::setGraph(const armem::MemoryID& entityID, const core::Graph& nav)
636  {
637  model.graphEntityID = entityID;
638 
639  // Build the gui graph (model).
640  {
641  QSignalBlocker blocker(this);
642 
643  clearGraph();
644  model.graph.attrib().edgesChanged = false;
645 
646  std::set<armem::MemoryID> coveredLocationIDs;
647  for (auto vertex : nav.vertices())
648  {
649  ARMARX_CHECK(vertex.attrib().hasPose());
650  addVertex(vertex.objectID(), {vertex.attrib()});
651 
652  armem::MemoryID id =
653  aron::fromAron<armem::MemoryID>(vertex.attrib().aron.locationID).getEntityID();
654  if (coveredLocationIDs.count(id) > 0)
655  {
656  ARMARX_WARNING << "Two vertices have the same location!";
657  }
658  else
659  {
660  coveredLocationIDs.insert(id);
661  }
662  }
663  // Add locations which have not been part of graph.
664  // ToDo: This should be an explicit step in the GUI.
665  {
666  semrel::ShapeID vertexID{0};
667  model.locationsMemory.forEachInstance(
668  [&](const armem::wm::EntityInstance& instance)
669  {
670  if (coveredLocationIDs.count(instance.id().getEntityID()) == 0)
671  {
672  vertexID = findNextFreeVertexID(model.graph, vertexID);
673  addVertex(vertexID, instance);
674  }
675  });
676  }
677 
678 
679  for (auto edge : nav.edges())
680  {
681  addEdge(model.graph.vertex(edge.sourceObjectID()),
682  model.graph.vertex(edge.targetObjectID()),
683  {edge.attrib()});
684  }
685  }
686 
687  // Trigger a view update.
688  emit graphChanged();
689  }
690 
691  void
692  WidgetController::setEmptyGraph(const armem::MemoryID& entityID)
693  {
694  model.graphEntityID = entityID;
695 
696  {
697  QSignalBlocker blocker(this);
698  clearGraph();
699 
700  semrel::ShapeID id{0};
701  queryLocations();
702  model.locationsMemory.forEachInstance(
703  [this, &id](const armem::wm::EntityInstance& instance)
704  {
705  addVertex(id, instance);
706  ++id;
707  });
708  }
709 
710  // Mark graph as changed.
711  model.graph.attrib().edgesChanged = true;
712 
713  emit graphChanged();
714  }
715 
716  GuiGraph::Vertex
717  WidgetController::addVertex(semrel::ShapeID vertexID, const VertexData& defaultAttribs)
718  {
719  ARMARX_CHECK(not model.graph.hasVertex(vertexID));
720 
721  GuiGraph::Vertex vertex = model.graph.addVertex(vertexID, defaultAttribs);
722  vertex.attrib().tableWidgetItem = view.vertexTable->addVertex();
723 
724  emit graphChanged();
725 
726  return vertex;
727  }
728 
729  GuiGraph::Vertex
730  WidgetController::addVertex(semrel::ShapeID vertexID,
731  const armem::wm::EntityInstance& locationInstance)
732  {
733  navigation::location::arondto::Location location;
734  location.fromAron(locationInstance.data());
735 
736  VertexData attrib;
737  toAron(attrib.aron.locationID, locationInstance.id().getEntityID());
738  FramedPose pose;
739  fromAron(location.framedPose, pose);
740  attrib.setPose(pose);
741  return addVertex(vertexID, attrib);
742  }
743 
744  GuiGraph::Edge
745  WidgetController::addEdge(GuiGraph::ConstVertex source,
746  GuiGraph::ConstVertex target,
747  const EdgeData& defaultAttribs)
748  {
749  ARMARX_CHECK(not model.graph.hasEdge(source.objectID(), target.objectID()))
750  << "Edge must not exist before being added: '" << source.attrib().getName() << "' -> '"
751  << target.attrib().getName() << "'";
752 
753  GuiGraph::Edge edge = model.graph.addEdge(source, target, defaultAttribs);
754  edge.attrib().tableWidgetItem = view.edgeTable->addEdge(edge);
755 
756  emit graphChanged();
757 
758  return edge;
759  }
760 
761  void
762  WidgetController::updateGraphView()
763  {
764  for (auto vertex : model.graph.vertices())
765  {
766  updateVertexView(vertex);
767  }
768  for (auto edge : model.graph.edges())
769  {
770  updateEdgeView(edge);
771  }
772 
773  int index =
774  widget.graphsComboBox->findData(QString::fromStdString(model.graphEntityID.str()));
775  if (index >= 0)
776  {
777  widget.graphsComboBox->setItemText(
778  index, getGraphDisplayName(model.graphEntityID, model.graph.hasChanged()));
779  }
780 
781  updateArViz();
782  }
783 
784  void
785  WidgetController::updateVertexView(GuiGraph::Vertex vertex)
786  {
787  view.vertexTable->updateVertex(vertex);
788 
789  if (/* DISABLES CODE */ (false)) // Disable this for the moment.
790  {
791  // Highlight all edges between highlighted vertices
792  std::set<semrel::ShapeID> highlightedVertices;
793  for (auto v : model.graph.vertices())
794  {
795  if (v.attrib().highlighted)
796  {
797  highlightedVertices.insert(v.objectID());
798  }
799  }
800 
801  for (auto edge : model.graph.edges())
802  {
803  bool verticesHighlighted = highlightedVertices.count(edge.sourceObjectID()) and
804  highlightedVertices.count(edge.targetObjectID());
805 
806  // Already highlighted but vertices not highlighted => to false, update
807  // Not highlighted but vertices highlighted => to true, update
808  if (edge.attrib().highlighted != verticesHighlighted)
809  {
810  edge.attrib().highlighted = verticesHighlighted;
811  }
812  }
813  }
814  }
815 
816  void
817  WidgetController::updateEdgeView(GuiGraph::Edge edge)
818  {
819  view.edgeTable->updateEdge(edge);
820  }
821 
822  void
823  WidgetController::updateArViz()
824  {
825  if (remote.arviz)
826  {
827  objpose::ObjectPoseMap objectMap;
828  std::vector<ObjectInfo> objectInfo;
829  graph::visu::ObjectParserInfo info{objectMap, objectInfo};
830 
831  // if ObjectPoseClient is connected, load objects to resolve graph locations
832  if (view.objectPoses->isConnected())
833  {
834  objectMap = view.objectPoses->connection().getObjectPoseMap();
835  objectInfo = view.objectPoses->connection().getObjectInfo();
836  }
837 
838  std::vector<viz::Layer> layers;
839  {
840  viz::Layer& vertices = layers.emplace_back(remote.arviz->layer("Locations"));
841  applyVisu(vertices, model.visu.vertex, model.graph.vertices(), info);
842  }
843  {
844  viz::Layer& edges = layers.emplace_back(remote.arviz->layer("Graph Edges"));
845  applyVisu(edges, model.visu.edge, model.graph.edges(), info);
846  }
847  {
848  viz::Layer& robot = layers.emplace_back(remote.arviz->layer("Robot"));
849  if (view.vertexData->vertex().has_value() and view.robotVisu->isEnabled())
850  {
851  const auto res = core::resolveLocation(
852  objectMap, objectInfo, view.vertexData->vertex()->attrib().getPose());
853  if (res.pose.has_value())
854  {
855  robot.add(
856  view.robotVisu->connection().vizRobot("robot").pose(res.pose.value()));
857  }
858  }
859  }
860  remote.arviz->commit(layers);
861  }
862  }
863 
864  template <class T>
865  static void
866  updateElementHighlighting(QList<QTableWidgetItem*> selectedItems,
867  std::map<QTableWidgetItem*, T>&& itemToElementMap)
868  {
869  for (QTableWidgetItem* selected : selectedItems)
870  {
871  if (auto it = itemToElementMap.find(selected); it != itemToElementMap.end())
872  {
873  it->second.attrib().highlighted = true;
874  itemToElementMap.erase(it);
875  }
876  }
877  for (auto& [_, unselected] : itemToElementMap)
878  {
879  unselected.attrib().highlighted = false;
880  }
881  }
882 
883  void
884  WidgetController::updateVertexHighlighting()
885  {
886  updateElementHighlighting(view.vertexTable->selectedVertexItems(),
887  model.graph.getTableItemToVertexMap());
888  emit graphChanged();
889  }
890 
891  void
892  WidgetController::updateEdgeHighlighting()
893  {
894  updateElementHighlighting(view.edgeTable->selectedEdgeItems(),
895  model.graph.getTableItemToEdgeMap());
896  emit graphChanged();
897  }
898 
899  void
900  WidgetController::selectVertex(QTableWidgetItem* vertexItem)
901  {
902  if (vertexItem == nullptr)
903  {
904  view.vertexData->clearVertex();
905  }
906  if (auto vertex = model.graph.getVertexFromTableItem(vertexItem))
907  {
908  selectVertex(vertex.value());
909  }
910  }
911 
912  void
913  WidgetController::selectVertex(GuiGraph::Vertex vertex)
914  {
915  view.vertexData->setVertex(vertex);
916  }
917 
918  void
919  WidgetController::clearGraph()
920  {
921  {
922  QSignalBlocker blocker(this);
923  clearEdges();
924  clearVertices();
925  }
926 
927  // Clear data structure
928  ARMARX_CHECK_EQUAL(model.graph.numEdges(), 0);
929  ARMARX_CHECK_EQUAL(model.graph.numVertices(), 0);
930 
931  emit graphChanged();
932  }
933 
934  void
935  WidgetController::clearEdges()
936  {
937  // Remove in reverse order to be more array-friendly.
938  std::vector<GuiGraph::Edge> edges{model.graph.edges().begin(), model.graph.edges().end()};
939  {
940  QSignalBlocker blocker(this);
941  for (auto it = edges.rbegin(); it != edges.rend(); ++it)
942  {
943  GuiGraph::Edge edge = *it;
944  removeEdge(edge);
945  }
946  }
947 
948  ARMARX_CHECK_EQUAL(view.edgeTable->rowCount(), 0);
949  ARMARX_CHECK_EQUAL(model.graph.numEdges(), 0);
950 
951  emit graphChanged();
952  }
953 
954  void
955  WidgetController::clearVertices()
956  {
957  ARMARX_CHECK_EQUAL(model.graph.numEdges(), 0)
958  << "The graph may not have any edges when clearing the vertices.";
959 
960  // Remove in reverse order to be more array-friendly.
961  std::vector<GuiGraph::Vertex> vertices{model.graph.vertices().begin(),
962  model.graph.vertices().end()};
963  {
964  QSignalBlocker blocker(this);
965  for (auto it = vertices.rbegin(); it != vertices.rend(); ++it)
966  {
967  GuiGraph::Vertex vertex = *it;
968  removeVertex(vertex);
969  }
970  }
971 
972  ARMARX_CHECK_EQUAL(view.vertexTable->rowCount(), 0);
973  ARMARX_CHECK_EQUAL(model.graph.numVertices(), 0);
974 
975  emit graphChanged();
976  }
977 
978  void
979  WidgetController::removeEdge(GuiGraph::Edge& edge)
980  {
981  ARMARX_CHECK(model.graph.hasEdge(edge.sourceDescriptor(), edge.targetDescriptor()))
982  << "Cannot remove edge that does not exist.";
983 
984  // Remove view elements
985  {
986  QSignalBlocker blocker(view.edgeTable);
987  view.edgeTable->removeEdge(edge);
988  }
989 
990  // Remove from model
991  model.graph.removeEdge(edge);
992 
993  ARMARX_CHECK(not model.graph.hasEdge(edge.sourceDescriptor(), edge.targetDescriptor()))
994  << edge.sourceDescriptor() << " -> " << edge.targetDescriptor();
995 
996  emit graphChanged();
997  }
998 
999  void
1000  WidgetController::removeVertex(GuiGraph::Vertex& vertex)
1001  {
1002  ARMARX_CHECK_EQUAL(vertex.inDegree(), 0)
1003  << "A vertex must not have any edges before being removed. "
1004  << vertex.attrib().getName();
1005  ARMARX_CHECK_EQUAL(vertex.outDegree(), 0)
1006  << "A vertex must not have any edges before being removed. "
1007  << vertex.attrib().getName();
1008 
1009  // Remove view elements
1010  {
1011  QSignalBlocker blocker(view.vertexTable);
1012  view.vertexTable->removeVertex(vertex);
1013  }
1014  if (view.vertexData->vertex().has_value() and view.vertexData->vertex().value() == vertex)
1015  {
1016  view.vertexData->clearVertex();
1017  }
1018 
1019  // Remove from model
1020  model.graph.removeVertex(vertex);
1021 
1022  emit graphChanged();
1023  }
1024 
1025  void
1026  WidgetController::createVertexDialog()
1027  {
1028  NewEntityIdDialog dialog(navigation::location::coreSegmentID, nullptr);
1029  switch (dialog.exec())
1030  {
1031  case QDialog::Accepted:
1032  // Ok go.
1033  break;
1034  case QDialog::Rejected:
1035  // Abort creation.
1036  return;
1037  }
1038 
1039  // Find free vertex ID
1040  const semrel::ShapeID vertexID = findNextFreeVertexID(model.graph, semrel::ShapeID{0});
1041 
1042  // Initialize attributes
1043  VertexData attribs;
1044  toAron(attribs.aron.locationID, dialog.entityID());
1045  attribs.setPose(FramedPose(core::Pose::Identity().matrix(), armarx::GlobalFrame, ""));
1046  attribs.changed = true;
1047 
1048  addVertex(vertexID, attribs);
1049  }
1050 
1051  void
1052  WidgetController::addEdges(QList<QPair<QTableWidgetItem*, QTableWidgetItem*>> vertexItems)
1053  {
1054  const auto itemToVertexMap = model.graph.getTableItemToVertexMap();
1055  {
1056  QSignalBlocker blocker(this);
1057 
1058  for (const auto& [sourceItem, targetItem] : vertexItems)
1059  {
1060  GuiGraph::Vertex source = itemToVertexMap.at(sourceItem);
1061  GuiGraph::Vertex target = itemToVertexMap.at(targetItem);
1062  if (not model.graph.hasEdge(source, target))
1063  {
1064  addEdge(source, target, {});
1065  }
1066  }
1067  }
1068 
1069  model.graph.attrib().edgesChanged = true;
1070  emit graphChanged();
1071  }
1072 
1073  void
1074  WidgetController::removeEdges(QList<QTableWidgetItem*> edgeItems)
1075  {
1076  const auto itemToEdgeMap = model.graph.getTableItemToEdgeMap();
1077  {
1078  QSignalBlocker blocker(this);
1079 
1080  for (const auto& edgeItem : edgeItems)
1081  {
1082  GuiGraph::Edge edge = itemToEdgeMap.at(edgeItem);
1083  removeEdge(edge);
1084  }
1085  }
1086 
1087  model.graph.attrib().edgesChanged = true;
1088  emit graphChanged();
1089  }
1090 
1091  void
1092  WidgetController::removeEdgesOfVertex(QList<QTableWidgetItem*> vertexItems,
1093  utils::EdgeDirection direction)
1094  {
1095  auto removeEdges = [this](auto&& range)
1096  {
1097  for (auto edge : range)
1098  {
1099  removeEdge(edge);
1100  }
1101  };
1102  {
1103  QSignalBlocker blocker(this);
1104 
1105  const auto itemToVertexMap = model.graph.getTableItemToVertexMap();
1106  for (const auto& vertexItem : vertexItems)
1107  {
1108  GuiGraph::Vertex vertex = itemToVertexMap.at(vertexItem);
1109  switch (direction)
1110  {
1112  removeEdges(vertex.inEdges());
1113  break;
1114 
1116  removeEdges(vertex.outEdges());
1117  break;
1118 
1120  removeEdges(vertex.inEdges());
1121  removeEdges(vertex.outEdges());
1122  break;
1123  }
1124  }
1125  }
1126 
1127  model.graph.attrib().edgesChanged = true;
1128  emit graphChanged();
1129  }
1130 
1131  void
1132  WidgetController::createGraphDialog()
1133  {
1134  NewEntityIdDialog dialog(navigation::graph::coreSegmentID, nullptr);
1135  switch (dialog.exec())
1136  {
1137  case QDialog::Accepted:
1138  break;
1139  case QDialog::Rejected:
1140  return;
1141  }
1142 
1143  const armem::MemoryID entityID = dialog.entityID();
1144 
1145  const bool hasChanged = true;
1146  QString text = getGraphDisplayName(entityID, hasChanged);
1147  QString data = QString::fromStdString(entityID.str());
1148  widget.graphsComboBox->addItem(text, data);
1149  widget.graphsComboBox->setCurrentIndex(widget.graphsComboBox->count() - 1);
1150 
1151  setEmptyGraph(entityID);
1152  }
1153 
1154  QString
1155  WidgetController::getGraphDisplayName(const armem::MemoryID& entityID, bool changed) const
1156  {
1157  QString name =
1158  QString::fromStdString(entityID.providerSegmentName + "/" + entityID.entityName);
1159  if (changed)
1160  {
1161  name += "*";
1162  }
1163  return name;
1164  }
1165 
1166 } // namespace armarx::navigation::qt_plugins::location_graph_editor
armarx::navigation::qt_plugins::location_graph_editor::utils::EdgeDirection::From
@ From
Client.h
armarx::armem::detail::SuccessHeader::success
bool success
Definition: SuccessHeader.h:19
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::configured
void configured() override
This function must be implemented by the user, if he supplies a config dialog.
Definition: WidgetController.cpp:255
armarx::navigation::qt_plugins::location_graph_editor::RobotVisuWidget::settingsChanged
void settingsChanged()
armarx::navigation::graph::coreSegmentID
const armem::MemoryID coreSegmentID
Definition: constants.h:30
ARMARX_VERBOSE
#define ARMARX_VERBOSE
Definition: Logging.h:187
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::locationMemoryChanged
void locationMemoryChanged()
armarx::navigation::core::resolveLocations
void resolveLocations(Graph &graph, const MemoryContainerT &locationContainer)
Definition: Graph.h:141
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::saveAutomaticSettings
virtual void saveAutomaticSettings()
Definition: WidgetController.cpp:285
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::graphMemoryChanged
void graphMemoryChanged()
armarx::armem::server::wm::EntityInstance
armem::wm::EntityInstance EntityInstance
Definition: forward_declarations.h:65
VertexTableWidget.h
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::saveSettings
void saveSettings(QSettings *settings) override
Definition: WidgetController.cpp:271
armarx::armem::CommitResult::allErrorMessages
std::vector< std::string > allErrorMessages() const
Definition: Commit.cpp:73
armarx::navigation::core::resolveLocation
void resolveLocation(Graph::Vertex &vertex, const aron::data::DictPtr &locationData)
Definition: Graph.cpp:250
EdgeTableWidget.h
armarx::armem::MemoryID::providerSegmentName
std::string providerSegmentName
Definition: MemoryID.h:52
armarx::armem::Commit
A bundle of updates to be sent to the memory.
Definition: Commit.h:84
framed.h
index
uint8_t index
Definition: EtherCATFrame.h:59
basic_types.h
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::loadSettings
void loadSettings(QSettings *settings) override
Implement to load the settings that are part of the GUI configuration.
Definition: WidgetController.cpp:261
armarx::VariantType::FramedPose
const VariantTypeId FramedPose
Definition: FramedPose.h:36
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::~WidgetController
virtual ~WidgetController() override
Definition: WidgetController.cpp:237
armarx::core::time::DateTime::Now
static DateTime Now()
Definition: DateTime.cpp:51
armarx::armem::EntityUpdateResult
Result of an EntityUpdate.
Definition: Commit.h:69
MemoryID.h
ARMARX_CHECK_NOT_NULL
#define ARMARX_CHECK_NOT_NULL(ptr)
This macro evaluates whether ptr is not null and if it turns out to be false it will throw an Express...
Definition: ExpressionException.h:206
forward_declarations.h
armarx::armem::MemoryID::str
std::string str(bool escapeDelimiters=true) const
Get a string representation of this memory ID.
Definition: MemoryID.cpp:102
armarx::GlobalFrame
const std::string GlobalFrame
Definition: FramedPose.h:65
boost::target
Vertex target(const detail::edge_base< Directed, Vertex > &e, const PCG &)
Definition: point_cloud_graph.h:668
Layer.h
armarx::armem::client::QueryResult
Result of a QueryInput.
Definition: Query.h:50
armarx::navigation::qt_plugins::location_graph_editor::fromGuiGraph
navigation::core::Graph fromGuiGraph(const GuiGraph &gui)
Definition: GuiGraph.cpp:127
armarx::navigation::qt_plugins::location_graph_editor::VertexTableWidget::newVertexRequested
void newVertexRequested()
Query.h
armarx::viz::Layer::add
void add(ElementT const &element)
Definition: Layer.h:31
aron_conversions.h
SimpleConfigDialog.h
ObjectPoseClientWidget.h
armarx::navigation::qt_plugins::location_graph_editor::VertexDataWidget::vertexDataChanged
void vertexDataChanged()
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::GetWidgetIcon
static QIcon GetWidgetIcon()
Definition: WidgetController.cpp:111
ARMARX_CHECK
#define ARMARX_CHECK(expression)
Shortcut for ARMARX_CHECK_EXPRESSION.
Definition: ExpressionException.h:82
constants.h
armarx::navigation::qt_plugins::location_graph_editor::utils::EdgeDirection
EdgeDirection
Definition: utils.h:43
RobotVisuWidget.h
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::GetWidgetName
static QString GetWidgetName()
Definition: WidgetController.cpp:105
armarx::status
status
Definition: FiniteStateMachine.h:244
GfxTL::Identity
void Identity(MatrixXX< N, N, T > *a)
Definition: MatrixXX.h:570
armarx::armem::detail::SuccessHeader::errorMessage
std::string errorMessage
Definition: SuccessHeader.h:20
armarx::navigation::qt_plugins::location_graph_editor::VertexTableWidget::newEdgesRequested
void newEdgesRequested(QList< QPair< QTableWidgetItem *, QTableWidgetItem * >> edges)
armarx::armem::MemoryID
A memory ID.
Definition: MemoryID.h:47
LocationUtils.h
armarx::viz::Client::createForGuiPlugin
static Client createForGuiPlugin(armarx::Component &component, std::string const &topicName="ArVizTopic", std::string const &storageName="ArVizStorage")
Definition: Client.cpp:62
FramedPose.h
data
uint8_t data[1]
Definition: EtherCATFrame.h:68
Visu.h
armarx::navigation::core::Graph
Definition: Graph.h:89
VertexDataWidget.h
armarx::navigation::qt_plugins::location_graph_editor::EdgeTableWidget::edgeRemovalRequested
void edgeRemovalRequested(QList< QTableWidgetItem * > edgeItems)
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::getConfigDialog
QPointer< QDialog > getConfigDialog(QWidget *parent=nullptr) override
getConfigDialog returns a pointer to the a configuration widget of this controller.
Definition: WidgetController.cpp:243
armarx::armem::EntityUpdate
An update of an entity for a specific point in time.
Definition: Commit.h:25
armarx::navigation::qt_plugins::location_graph_editor::findNextFreeVertexID
semrel::ShapeID findNextFreeVertexID(const GraphT &graph, semrel::ShapeID vertexID)
Definition: WidgetController.cpp:625
armarx::armem::CommitResult
Result of a Commit.
Definition: Commit.h:105
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::onConnectComponent
void onConnectComponent() override
Pure virtual hook for the subclass.
Definition: WidgetController.cpp:297
armarx::Graph
boost::subgraph< CloudGraph > Graph
Definition: Common.h:58
armarx::navigation::qt_plugins::location_graph_editor
Definition: GuiGraph.cpp:34
ObjectInfo.h
Visu.h
boost::source
Vertex source(const detail::edge_base< Directed, Vertex > &e, const PCG &)
Definition: point_cloud_graph.h:661
armarx::armem::MemoryID::entityName
std::string entityName
Definition: MemoryID.h:53
armarx::armem::server::ltm::util::mongodb::detail::update
bool update(mongocxx::collection &coll, const nlohmann::json &query, const nlohmann::json &update)
Definition: mongodb.cpp:68
armarx::armem::EntityUpdateResult::snapshotID
MemoryID snapshotID
Definition: Commit.h:73
aron_conversions.h
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::loadAutomaticSettings
virtual void loadAutomaticSettings()
Definition: WidgetController.cpp:278
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::WidgetController
WidgetController()
Definition: WidgetController.cpp:116
memory_definitions.h
armarx::Component
Baseclass for all ArmarX ManagedIceObjects requiring properties.
Definition: Component.h:91
armarx::armem::MemoryID::fromString
static MemoryID fromString(const std::string &string)
Alias for constructor from string.
Definition: MemoryID.cpp:188
ExpressionException.h
Commit.h
utils.h
NewEntityIdDialog.h
armarx::armem::MemoryID::getEntityID
MemoryID getEntityID() const
Definition: MemoryID.cpp:310
forward_declarations.h
Graph.h
armarx::ctrlutil::v
double v(double t, double v0, double a0, double j)
Definition: CtrlUtil.h:39
armarx::navigation::qt_plugins::location_graph_editor::VertexTableWidget::edgeRemovalRequested
void edgeRemovalRequested(QList< QTableWidgetItem * > vertexItems, utils::EdgeDirection direction)
armarx::armem::Commit::add
EntityUpdate & add()
Definition: Commit.cpp:80
armarx::armem::CommitResult::results
std::vector< EntityUpdateResult > results
Definition: Commit.h:107
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::graphChanged
void graphChanged()
GuiGraph.h
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::onInitComponent
void onInitComponent() override
Pure virtual hook for the subclass.
Definition: WidgetController.cpp:291
armarx::navigation::qt_plugins::location_graph_editor::utils::EdgeDirection::Bidirectional
@ Bidirectional
armarx::fromAron
void fromAron(const arondto::PackagePath &dto, PackageFileLocation &bo)
armarx::armem::client::MemoryNameSystem
The memory name system (MNS) client.
Definition: MemoryNameSystem.h:68
armarx::armem::EntityUpdateResult::errorMessage
std::string errorMessage
Definition: Commit.h:76
armarx::navigation::location::coreSegmentID
const armem::MemoryID coreSegmentID
Definition: constants.h:30
armarx::navigation::qt_plugins::location_graph_editor::VertexDataWidget
Definition: VertexDataWidget.h:56
MemoryNameSystem.h
armarx::navigation::qt_plugins::location_graph_editor::WidgetController::connected
void connected()
Logging.h
armarx::toAron
void toAron(arondto::PackagePath &dto, const PackageFileLocation &bo)
ARMARX_CHECK_EQUAL
#define ARMARX_CHECK_EQUAL(lhs, rhs)
This macro evaluates whether lhs is equal (==) rhs and if it turns out to be false it will throw an E...
Definition: ExpressionException.h:130
ARMARX_WARNING
#define ARMARX_WARNING
Definition: Logging.h:193
armarx::navigation::qt_plugins::location_graph_editor::utils::EdgeDirection::To
@ To
armarx::ManagedIceObject::getProxy
Ice::ObjectPrx getProxy(long timeoutMs=0, bool waitForScheduler=true) const
Returns the proxy of this object (optionally it waits for the proxy)
Definition: ManagedIceObject.cpp:407
armarx::viz::Layer
Definition: Layer.h:12
armarx::armem::EntityUpdateResult::success
bool success
Definition: Commit.h:71
armarx::navigation::qt_plugins::location_graph_editor::ObjectPoseClientWidget::connected
void connected()
armarx::ManagedIceObject::usingProxy
bool usingProxy(const std::string &name, const std::string &endpoints="")
Registers a proxy for retrieval after initialization and adds it to the dependency list.
Definition: ManagedIceObject.cpp:154
armarx::navigation::qt_plugins::location_graph_editor::RobotVisuWidget::connected
void connected()
armarx::armem::CommitResult::allSuccess
bool allSuccess() const
Definition: Commit.cpp:64
armarx::navigation::qt_plugins::location_graph_editor::applyVisu
void applyVisu(viz::Layer &layer, const std::optional< VisuT > &visu, RangeT &&range, graph::visu::ObjectParserInfo info)
Definition: Visu.h:105
WidgetController.h
constants.h
PackagePath.h
armarx::SimpleConfigDialog
A config-dialog containing one (or multiple) proxy finders.
Definition: SimpleConfigDialog.h:84
armarx::objpose::ObjectPoseMap
std::map< ObjectID, ObjectPose > ObjectPoseMap
Definition: forward_declarations.h:21