4 #include <QDoubleSpinBox>
9 #include <SimoxUtility/algorithm/string.h>
28 if (!widgetConnections.empty())
31 for (
const auto& conn : widgetConnections)
35 widgetConnections.clear();
38 this->widget = widget;
41 widgetConnections.append(
43 &QTreeWidget::itemExpanded,
44 [
this](QTreeWidgetItem* item)
46 if (item->parent() || this->layers ==
nullptr)
51 std::string
id = item->text(0).toStdString();
52 auto* layer = this->layers->findLayer(this->layerID);
56 if (element !=
nullptr)
58 this->updateLayerElementProperties(item, element->
data);
63 widget->setColumnCount(2);
64 widget->setHeaderLabels({
"Property",
"Value"});
65 widget->setHeaderHidden(
false);
74 if (layerID == this->layerID)
84 this->maxElementCountDefault =
max;
90 if (this->enabled !=
value)
92 this->enabled =
value;
108 this->layers = layers;
110 showMoreItem =
nullptr;
111 maxElementCount = maxElementCountDefault;
121 if (enabled && layers)
123 auto* layer = layers->
findLayer(layerID);
124 if (layer !=
nullptr)
126 updateLayerElements(layer->elements);
132 LayerInfoTree::clearLayerElements()
137 showMoreItem =
nullptr;
142 LayerInfoTree::updateLayerElements(
const std::vector<viz::CoinLayerElement>& elements)
148 int index = widget->indexOfTopLevelItem(showMoreItem);
150 <<
"Invariant: If showMoreItem != nullptr, then the widget contains it.";
151 delete widget->takeTopLevelItem(
index);
152 showMoreItem =
nullptr;
156 int currentIndex = 0;
159 std::string
const& name = element.data->id;
161 if (maxElementCount >= 0 && currentIndex >= maxElementCount)
168 if (currentIndex >= widget->topLevelItemCount())
171 QTreeWidgetItem* item = insertLayerElementItem(currentIndex, name, element.data);
172 updateLayerElementProperties(item, element.data);
178 QTreeWidgetItem* currentItem = widget->topLevelItem(currentIndex);
182 while (currentItem !=
nullptr && name.c_str() > currentItem->text(0))
185 delete widget->takeTopLevelItem(currentIndex);
186 currentItem = widget->topLevelItem(currentIndex);
189 if (currentItem !=
nullptr && name.c_str() < currentItem->text(0))
192 currentItem = insertLayerElementItem(currentIndex, name, element.data);
193 updateLayerElementProperties(currentItem, element.data);
198 if (currentItem !=
nullptr && name.c_str() == currentItem->text(0))
201 updateLayerElementItem(currentItem, element.data);
202 if (currentItem->isExpanded())
204 updateLayerElementProperties(currentItem, element.data);
213 LayerInfoTree::insertLayerElementItem(
int index,
214 const std::string& name,
215 const viz::data::ElementPtr& element)
217 QTreeWidgetItem* item =
218 new QTreeWidgetItem(QStringList{name.c_str(), getTypeName(element).c_str()});
221 widget->insertTopLevelItem(
index, item);
226 LayerInfoTree::updateLayerElementItem(QTreeWidgetItem* item,
227 const viz::data::ElementPtr& element)
230 item->setText(1, getTypeName(element).c_str());
234 LayerInfoTree::updateLayerElementProperties(QTreeWidgetItem* item,
235 const viz::data::ElementPtr& element)
237 bool recursive =
true;
238 updateJsonChildren(item, serializeElement(element), recursive);
242 LayerInfoTree::getTypeName(
const viz::data::ElementPtr& element)
const
244 const std::string typeName = simox::meta::get_type_name(*element);
247 :
"<unknown type '" + typeName +
"'>";
251 LayerInfoTree::serializeElement(
const viz::data::ElementPtr& element)
const
255 return nlohmann::json(element);
257 catch (
const viz::error::NoSerializerForType&)
259 return nlohmann::json::object();
264 LayerInfoTree::updateJsonChildren(QTreeWidgetItem* parent,
265 const nlohmann::json& json,
270 nlohmann::json jmeta = nlohmann::json::object();
277 if (iItem >= parent->childCount())
279 QTreeWidgetItem* item = addJsonChild(parent, key);
280 updateJsonItem(item, key, j, jmeta, recursive);
284 QTreeWidgetItem* currentItem = parent->child(iItem);
285 updateJsonItem(currentItem, key, j, jmeta, recursive);
289 while (parent->childCount() >
int(json.size()))
291 parent->removeChild(parent->child(parent->childCount() - 1));
294 else if (json.is_object())
296 namespace meta = viz::json::meta;
297 const nlohmann::json& jmeta =
301 for (
auto it : json.items())
304 if (isMetaKey(it.key()))
310 if (jmeta.count(it.key()) > 0 && jmeta.at(it.key()) ==
meta::HIDE)
315 if (iItem >= parent->childCount())
317 QTreeWidgetItem* item = addJsonChild(parent, it.key());
318 updateJsonItem(item, it.key(), it.value(), jmeta, recursive);
323 QTreeWidgetItem* currentItem = parent->child(iItem);
325 while (it.key().c_str() > currentItem->text(0))
328 parent->removeChild(currentItem);
329 currentItem = parent->child(iItem);
332 if (it.key().c_str() < currentItem->text(0))
335 currentItem = addJsonChild(parent, it.key());
336 updateJsonItem(currentItem, it.key(), it.value(), jmeta, recursive);
341 if (it.key().c_str() == currentItem->text(0))
344 updateJsonItem(currentItem, it.key(), it.value(), jmeta, recursive);
353 LayerInfoTree::addJsonChild(QTreeWidgetItem* parent,
const std::string& key)
355 QTreeWidgetItem* child =
new QTreeWidgetItem(QStringList{key.c_str(),
""});
358 parent->addChild(child);
364 LayerInfoTree::updateJsonItem(QTreeWidgetItem* item,
365 const std::string& key,
366 const nlohmann::json&
value,
367 const nlohmann::json& parentMeta,
370 const int columnKey = 0;
373 item->setText(columnKey, key.c_str());
375 updateJsonItemValue(item, key,
value, parentMeta);
380 updateJsonChildren(item,
value, recursive);
385 LayerInfoTree::updateJsonItemValue(QTreeWidgetItem* item,
386 const std::string& key,
387 const nlohmann::json&
value,
388 const nlohmann::json& parentMeta)
390 const int columnValue = 1;
392 if (parentMeta.count(key) > 0)
394 const nlohmann::json& metaValue = parentMeta.at(key);
395 if (metaValue.is_string())
397 const std::string metaString = metaValue.get<std::string>();
400 item->setText(columnValue, getValuePreview(
value).toString());
402 item->setData(0, Qt::CheckStateRole, QVariant());
409 if (QWidget* w = widget->itemWidget(item, columnValue))
411 updateValueWidget(
value, w);
413 else if (QWidget* w = getValueWidget(item,
value); w)
415 widget->setItemWidget(item, columnValue, w);
420 item->setText(columnValue, getValuePreview(
value).
toString());
425 LayerInfoTree::getValuePreview(
const nlohmann::json& json)
const
427 std::stringstream ss;
431 case nlohmann::json::value_t::null:
432 return QVariant(
"null");
434 case nlohmann::json::value_t::object:
436 std::vector<std::string> keys;
437 for (
auto it : json.items())
439 if (!isMetaKey(it.key()))
441 keys.push_back(it.key());
444 ss <<
"object [" << simox::alg::join(keys,
", ") <<
"]";
445 return QVariant(ss.str().c_str());
448 case nlohmann::json::value_t::array:
449 ss <<
"array of size " << json.size();
450 return QVariant(ss.str().c_str());
452 case nlohmann::json::value_t::string:
453 return QVariant(json.get<std::string>().c_str());
454 case nlohmann::json::value_t::boolean:
455 return QVariant(json.get<
bool>());
456 case nlohmann::json::value_t::number_integer:
457 return QVariant(json.get<
int>());
458 case nlohmann::json::value_t::number_unsigned:
459 return QVariant(json.get<
unsigned int>());
460 case nlohmann::json::value_t::number_float:
461 return QVariant(json.get<
float>());
464 return QVariant(
"n/a");
469 LayerInfoTree::getValueWidget(QTreeWidgetItem* item,
const nlohmann::json& json)
const
471 auto setChecked = [item]() { item->setCheckState(0, Qt::CheckState::Checked); };
475 case nlohmann::json::value_t::string:
478 QLineEdit* w =
new QLineEdit(json.get<std::string>().c_str());
479 connect(w, &QLineEdit::editingFinished,
this, setChecked);
482 case nlohmann::json::value_t::boolean:
484 QCheckBox* w =
new QCheckBox(
"Enabled");
485 w->setChecked(json.get<
bool>());
486 connect(w, &QCheckBox::stateChanged,
this, setChecked);
489 case nlohmann::json::value_t::number_integer:
490 case nlohmann::json::value_t::number_unsigned:
492 QSpinBox* w =
new QSpinBox();
493 w->setMinimum(std::numeric_limits<int>::lowest());
495 w->setValue(json.get<
int>());
496 connect(w, qOverload<int>(&QSpinBox::valueChanged),
this, setChecked);
499 case nlohmann::json::value_t::number_float:
501 QDoubleSpinBox* w =
new QDoubleSpinBox();
502 w->setMinimum(std::numeric_limits<double>::lowest());
504 w->setValue(json.get<
double>());
505 connect(w, qOverload<double>(&QDoubleSpinBox::valueChanged),
this, setChecked);
508 case nlohmann::json::value_t::null:
509 case nlohmann::json::value_t::object:
510 case nlohmann::json::value_t::array:
519 LayerInfoTree::updateValueWidget(
const nlohmann::json& json, QWidget* widget)
const
523 case nlohmann::json::value_t::string:
525 QLineEdit* w =
dynamic_cast<QLineEdit*
>(widget);
527 w->setText(json.get<std::string>().c_str());
530 case nlohmann::json::value_t::boolean:
532 QCheckBox* w =
dynamic_cast<QCheckBox*
>(widget);
534 w->setChecked(json.get<
bool>());
537 case nlohmann::json::value_t::number_integer:
538 case nlohmann::json::value_t::number_unsigned:
540 QSpinBox* w =
dynamic_cast<QSpinBox*
>(widget);
542 w->setValue(json.get<
int>());
545 case nlohmann::json::value_t::number_float:
547 QDoubleSpinBox* w =
dynamic_cast<QDoubleSpinBox*
>(widget);
549 w->setValue(json.get<
double>());
552 case nlohmann::json::value_t::null:
553 case nlohmann::json::value_t::object:
554 case nlohmann::json::value_t::array:
561 LayerInfoTree::isMetaKey(
const std::string& key)
563 return key.substr(0, 2) ==
"__" && key.substr(key.size() - 2) ==
"__";
567 LayerInfoTree::addShowMoreItem()
570 <<
"There should not be a showMoreItem when this function is called.";
572 auto* layer = this->layers->
findLayer(this->layerID);
573 if (layer ==
nullptr)
575 ARMARX_WARNING <<
"No layer with ID '" << this->layerID.first <<
"/"
576 << this->layerID.second <<
"' found";
580 std::stringstream ss;
581 ss <<
"Showing " << widget->topLevelItemCount() <<
" of " << layer->elements.size()
585 QTreeWidgetItem* item =
new QTreeWidgetItem(QStringList{
"", ss.str().c_str()});
586 widget->addTopLevelItem(item);
588 QPushButton* button =
new QPushButton(
"Show more");
589 widget->setItemWidget(item, 0, button);
592 &QPushButton::pressed,
595 maxElementCount += maxElementCountDefault;
596 this->updateLayerElements(layer->elements);
599 this->showMoreItem = item;