Skip to content

Commit

Permalink
Fix WbField::isParameter for Unconnected Fields (#6735)
Browse files Browse the repository at this point in the history
* fix WbField::isParameter

* fix dependencies on old behavior

* remove WbTemplateManager::regenerateNodeFromField/ParameterChange

* move WbField::isParameter to cpp file

* fix WbField::isParameter definition

* update changelog and fix PR reference comment

* formatting
  • Loading branch information
CoolSpy3 authored Dec 22, 2024
1 parent 2428a8d commit fdd04d5
Show file tree
Hide file tree
Showing 7 changed files with 19 additions and 25 deletions.
1 change: 1 addition & 0 deletions docs/reference/changelog-r2024.md
Original file line number Diff line number Diff line change
Expand Up @@ -33,3 +33,4 @@ Released on December **th, 2023.
- Fixed bug where the rotation axis of the omni wheel rollers were incorrect ([6710](https://github.com/cyberbotics/webots/pull/6710)).
- Fixed a bug where Webots would crash if a geometry was inserted into a `Shape` node used a bounding box ([#6691](https://github.com/cyberbotics/webots/pull/6691)).
- Removed the old `wb_supervisor_field_import_sf_node` and `wb_supervisor_field_import_mf_node` functions from the list of editor autocomplete suggestions ([#6701](https://github.com/cyberbotics/webots/pull/6701)).
- Fixed a bug preventing nodes from being inserted into unconnected proto fields ([#6735](https://github.com/cyberbotics/webots/pull/6735)).
27 changes: 8 additions & 19 deletions src/webots/nodes/utils/WbTemplateManager.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -131,10 +131,9 @@ bool WbTemplateManager::nodeNeedsToSubscribe(WbNode *node) {
void WbTemplateManager::recursiveFieldSubscribeToRegenerateNode(WbNode *node, bool subscribedNode, bool subscribedDescendant) {
if (subscribedNode || subscribedDescendant) {
if (node->isProtoInstance())
connect(node, &WbNode::parameterChanged, this, &WbTemplateManager::regenerateNodeFromParameterChange,
Qt::UniqueConnection);
connect(node, &WbNode::parameterChanged, this, &WbTemplateManager::regenerateNodeFromField, Qt::UniqueConnection);
else
connect(node, &WbNode::fieldChanged, this, &WbTemplateManager::regenerateNodeFromFieldChange, Qt::UniqueConnection);
connect(node, &WbNode::fieldChanged, this, &WbTemplateManager::regenerateNodeFromField, Qt::UniqueConnection);
}

// if PROTO node:
Expand Down Expand Up @@ -177,33 +176,23 @@ void WbTemplateManager::recursiveFieldSubscribeToRegenerateNode(WbNode *node, bo
}
}

void WbTemplateManager::regenerateNodeFromFieldChange(WbField *field) {
// retrieve the right node
WbNode *templateNode = dynamic_cast<WbNode *>(sender());
assert(templateNode);
if (templateNode)
regenerateNodeFromField(templateNode, field, false);
}

void WbTemplateManager::regenerateNodeFromParameterChange(WbField *field) {
// intermediate function to determine which node should be updated
// Note: The security is probably overkill there, but its also safer for the first versions of the template mechanism
void WbTemplateManager::regenerateNodeFromField(WbField *field) {
// retrieve the right node
WbNode *templateNode = dynamic_cast<WbNode *>(sender());
assert(templateNode);
if (templateNode)
regenerateNodeFromField(templateNode, field, true);
}
if (!templateNode)
return;

// intermediate function to determine which node should be updated
// Note: The security is probably overkill there, but its also safer for the first versions of the template mechanism
void WbTemplateManager::regenerateNodeFromField(WbNode *templateNode, WbField *field, bool isParameter) {
// 1. retrieve upper template node where the modification appeared in a template regenerator field
WbNode *upperTemplateNode = WbVrmlNodeUtilities::findUpperTemplateNeedingRegenerationFromField(field, templateNode);

if (!upperTemplateNode)
return;

// 2. check it's not a parameter managed by ODE
if (!isParameter && dynamic_cast<const WbSolid *>(templateNode) &&
if (!field->isParameter() && dynamic_cast<const WbSolid *>(templateNode) &&
((field->name() == "translation" && field->type() == WB_SF_VEC3F) ||
(field->name() == "rotation" && field->type() == WB_SF_ROTATION) ||
(field->name() == "position" && field->type() == WB_SF_FLOAT)))
Expand Down
4 changes: 1 addition & 3 deletions src/webots/nodes/utils/WbTemplateManager.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -52,8 +52,7 @@ class WbTemplateManager : public QObject {

private slots:
void unsubscribe(QObject *node);
void regenerateNodeFromFieldChange(WbField *field);
void regenerateNodeFromParameterChange(WbField *field);
void regenerateNodeFromField(WbField *field);
void regenerateNode(WbNode *node, bool restarted = false);
void nodeNeedRegeneration();

Expand All @@ -67,7 +66,6 @@ private slots:

bool nodeNeedsToSubscribe(WbNode *node);
void recursiveFieldSubscribeToRegenerateNode(WbNode *node, bool subscribedNode, bool subscribedDescendant);
void regenerateNodeFromField(WbNode *templateNode, WbField *field, bool isParameter);

QList<WbNode *> mTemplates;
QSet<const WbNode *> mNodesSubscribedForRegeneration;
Expand Down
2 changes: 1 addition & 1 deletion src/webots/scene_tree/WbAddNodeDialog.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -450,7 +450,7 @@ void WbAddNodeDialog::buildTree() {
static const QString INVALID_FOR_INSERTION_IN_BOUNDING_OBJECT("N");

const WbField *const actualField =
(mField->isParameter() && !mField->alias().isEmpty()) ? mField->internalFields().at(0) : mField;
(!mField->internalFields().isEmpty() && !mField->alias().isEmpty()) ? mField->internalFields().at(0) : mField;
bool boInfo = actualField->name() == "boundingObject";
if (!boInfo)
boInfo = nodeUse & WbNode::BOUNDING_OBJECT_USE;
Expand Down
2 changes: 1 addition & 1 deletion src/webots/scene_tree/WbExtendedStringEditor.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -453,7 +453,7 @@ void WbExtendedStringEditor::edit(bool copyOriginalValue) {

if (copyOriginalValue) {
const WbField *effectiveField = field();
if (effectiveField->isParameter())
if (!effectiveField->internalFields().isEmpty())
effectiveField = effectiveField->internalFields().at(0);

mStringType = fieldNameToStringType(effectiveField->name(), effectiveField->parentNode());
Expand Down
6 changes: 6 additions & 0 deletions src/webots/vrml/WbField.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -104,6 +104,12 @@ bool WbField::isDeprecated() const {
return mModel->isDeprecated();
}

// Because of unconnected fields, the only way to definitively check if a field is a parameter is to check its parent node
// If that is not possible, fallback to the old behavior (See #6604 and #6735)
bool WbField::isParameter() const {
return parentNode() ? parentNode()->isProtoInstance() : !mInternalFields.isEmpty();
}

void WbField::readValue(WbTokenizer *tokenizer, const QString &worldPath) {
if (mWasRead)
tokenizer->reportError(tr("Duplicate field value: '%1'").arg(name()));
Expand Down
2 changes: 1 addition & 1 deletion src/webots/vrml/WbField.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -69,7 +69,7 @@ class WbField : public QObject {
void redirectTo(WbField *parameter, bool skipCopy = false);
WbField *parameter() const { return mParameter; }
const QList<WbField *> &internalFields() const { return mInternalFields; }
bool isParameter() const { return mInternalFields.size() != 0; }
bool isParameter() const;

void clearInternalFields() { mInternalFields.clear(); }

Expand Down

0 comments on commit fdd04d5

Please sign in to comment.