mirror of
https://github.com/scratchfoundation/scratch-vm.git
synced 2025-01-11 23:30:09 -05:00
Update fields of blocks to reference new variable names for renamed global so that the VM always has up-to-date and accurate information about the blocks.
This commit is contained in:
parent
99c1ce9286
commit
37b276935e
1 changed files with 20 additions and 1 deletions
|
@ -535,6 +535,7 @@ class Target extends EventEmitter {
|
||||||
* then the local variable is renamed to distinguish itself from the pre-existing variable.
|
* then the local variable is renamed to distinguish itself from the pre-existing variable.
|
||||||
* All blocks that reference the local variable will be updated to use the new name.
|
* All blocks that reference the local variable will be updated to use the new name.
|
||||||
*/
|
*/
|
||||||
|
// TODO (#1360) This function is too long, add some helpers for the different chunks and cases...
|
||||||
fixUpVariableReferences () {
|
fixUpVariableReferences () {
|
||||||
if (!this.runtime) return; // There's no runtime context to conflict with
|
if (!this.runtime) return; // There's no runtime context to conflict with
|
||||||
if (this.isStage) return; // Stage can't have variable conflicts with itself (and also can't be uploaded)
|
if (this.isStage) return; // Stage can't have variable conflicts with itself (and also can't be uploaded)
|
||||||
|
@ -562,6 +563,7 @@ class Target extends EventEmitter {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
const conflictIdsToReplace = Object.create(null);
|
const conflictIdsToReplace = Object.create(null);
|
||||||
|
const conflictNamesToReplace = Object.create(null);
|
||||||
|
|
||||||
// Cache the list of all variable names by type so that we don't need to
|
// Cache the list of all variable names by type so that we don't need to
|
||||||
// re-calculate this in every iteration of the following loop.
|
// re-calculate this in every iteration of the following loop.
|
||||||
|
@ -615,6 +617,9 @@ class Target extends EventEmitter {
|
||||||
const allNames = allVarNames(varType);
|
const allNames = allVarNames(varType);
|
||||||
const freshName = StringUtil.unusedName(varName, allNames);
|
const freshName = StringUtil.unusedName(varName, allNames);
|
||||||
stage.createVariable(varId, freshName, varType);
|
stage.createVariable(varId, freshName, varType);
|
||||||
|
if (!conflictNamesToReplace[varId]) {
|
||||||
|
conflictNamesToReplace[varId] = freshName;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -626,7 +631,7 @@ class Target extends EventEmitter {
|
||||||
const type = this.variables[varId].type;
|
const type = this.variables[varId].type;
|
||||||
renameConflictingLocalVar(varId, name, type);
|
renameConflictingLocalVar(varId, name, type);
|
||||||
}
|
}
|
||||||
// Finally, handle global var conflicts (e.g. a sprite is uploaded, and has
|
// Handle global var conflicts with existing global vars (e.g. a sprite is uploaded, and has
|
||||||
// blocks referencing some variable that the sprite does not own, and this
|
// blocks referencing some variable that the sprite does not own, and this
|
||||||
// variable conflicts with a global var)
|
// variable conflicts with a global var)
|
||||||
// In this case, we want to merge the new variable referenes with the
|
// In this case, we want to merge the new variable referenes with the
|
||||||
|
@ -636,6 +641,20 @@ class Target extends EventEmitter {
|
||||||
const referencesToUpdate = allReferences[conflictId];
|
const referencesToUpdate = allReferences[conflictId];
|
||||||
this.mergeVariables(conflictId, existingId, referencesToUpdate);
|
this.mergeVariables(conflictId, existingId, referencesToUpdate);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Handle global var conflicts existing local vars (e.g a sprite is uploaded,
|
||||||
|
// and has blocks referencing some variable that the sprite does not own, and this
|
||||||
|
// variable conflcits with another sprite's local var).
|
||||||
|
// In this case, we want to go through the variable references and update
|
||||||
|
// the name of the variable in that reference.
|
||||||
|
for (const conflictId in conflictNamesToReplace) {
|
||||||
|
const newName = conflictNamesToReplace[conflictId];
|
||||||
|
const referencesToUpdate = allReferences[conflictId];
|
||||||
|
referencesToUpdate.map(ref => {
|
||||||
|
ref.referencingField.value = newName;
|
||||||
|
return ref;
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue